1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
|
# frozen_string_literal: true
# Redmine - project management software
# Copyright (C) 2006-2023 Jean-Philippe Lang
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
module Redmine
module Themes
# Return an array of installed themes
def self.themes
@@installed_themes ||= scan_themes
end
# Rescan themes directory
def self.rescan
@@installed_themes = scan_themes
end
# Return theme for given id, or nil if it's not found
def self.theme(id, options={})
return nil if id.blank?
found = themes.find {|t| t.id == id}
if found.nil? && options[:rescan] != false
rescan
found = theme(id, :rescan => false)
end
found
end
# Class used to represent a theme
class Theme
attr_reader :path, :name, :dir
def initialize(path)
@path = path
@dir = File.basename(path)
@name = @dir.humanize
@stylesheets = nil
@javascripts = nil
end
# Directory name used as the theme id
def id; dir end
def ==(theme)
theme.is_a?(Theme) && theme.dir == dir
end
def <=>(theme)
return nil unless theme.is_a?(Theme)
name <=> theme.name
end
def stylesheets
@stylesheets ||= assets("stylesheets", "css")
end
def images
@images ||= assets("images")
end
def javascripts
@javascripts ||= assets("javascripts", "js")
end
def favicons
@favicons ||= assets("favicon")
end
def favicon
favicons.first
end
def favicon?
favicon.present?
end
def stylesheet_path(source)
"#{asset_prefix}#{source}"
end
def image_path(source)
"#{asset_prefix}#{source}"
end
def javascript_path(source)
"#{asset_prefix}#{source}"
end
def favicon_path
"#{asset_prefix}#{favicon}"
end
def asset_prefix
"themes/#{dir}/"
end
def asset_paths
base_dir = Pathname.new(path)
paths = base_dir.children.filter_map do |child|
child if child.directory? &&
child.basename.to_s != "src" &&
!child.basename.to_s.start_with?('.')
end
Redmine::AssetPath.new(base_dir, paths, asset_prefix)
end
private
def assets(dir, ext=nil)
if ext
Dir.glob("#{path}/#{dir}/*.#{ext}").collect {|f| File.basename(f, ".#{ext}")}
else
Dir.glob("#{path}/#{dir}/*").collect {|f| File.basename(f)}
end
end
end
module Helper
def current_theme
unless instance_variable_defined?(:@current_theme)
@current_theme = Redmine::Themes.theme(Setting.ui_theme)
end
@current_theme
end
# Returns the header tags for the current theme
def heads_for_theme
if current_theme && current_theme.javascripts.include?('theme')
javascript_include_tag current_theme.javascript_path('theme')
end
end
end
def self.scan_themes
dirs = Dir.glob("#{Rails.public_path}/themes/*").select do |f|
# A theme should at least override application.css
File.directory?(f) && File.exist?("#{f}/stylesheets/application.css")
end
dirs.collect {|dir| Theme.new(dir)}.sort
end
private_class_method :scan_themes
end
end
|