Merge pull request #1660 from ClearlyClaire/glitch-soc/features/themes-multiple-packs
Refactor glitch-soc's theme handling
This commit is contained in:
commit
f6acc039ec
4 changed files with 108 additions and 92 deletions
|
@ -10,6 +10,7 @@ class ApplicationController < ActionController::Base
|
||||||
include SessionTrackingConcern
|
include SessionTrackingConcern
|
||||||
include CacheConcern
|
include CacheConcern
|
||||||
include DomainControlHelper
|
include DomainControlHelper
|
||||||
|
include ThemingConcern
|
||||||
|
|
||||||
helper_method :current_account
|
helper_method :current_account
|
||||||
helper_method :current_session
|
helper_method :current_session
|
||||||
|
@ -73,75 +74,6 @@ class ApplicationController < ActionController::Base
|
||||||
new_user_session_path
|
new_user_session_path
|
||||||
end
|
end
|
||||||
|
|
||||||
def pack(data, pack_name, skin = 'default')
|
|
||||||
return nil unless pack?(data, pack_name)
|
|
||||||
pack_data = {
|
|
||||||
common: pack_name == 'common' ? nil : resolve_pack(data['name'] ? Themes.instance.flavour(current_flavour) : Themes.instance.core, 'common', skin),
|
|
||||||
flavour: data['name'],
|
|
||||||
pack: pack_name,
|
|
||||||
preload: nil,
|
|
||||||
skin: nil,
|
|
||||||
supported_locales: data['locales'],
|
|
||||||
}
|
|
||||||
if data['pack'][pack_name].is_a?(Hash)
|
|
||||||
pack_data[:common] = nil if data['pack'][pack_name]['use_common'] == false
|
|
||||||
pack_data[:pack] = nil unless data['pack'][pack_name]['filename']
|
|
||||||
if data['pack'][pack_name]['preload']
|
|
||||||
pack_data[:preload] = [data['pack'][pack_name]['preload']] if data['pack'][pack_name]['preload'].is_a?(String)
|
|
||||||
pack_data[:preload] = data['pack'][pack_name]['preload'] if data['pack'][pack_name]['preload'].is_a?(Array)
|
|
||||||
end
|
|
||||||
if skin != 'default' && data['skin'][skin]
|
|
||||||
pack_data[:skin] = skin if data['skin'][skin].include?(pack_name)
|
|
||||||
else # default skin
|
|
||||||
pack_data[:skin] = 'default' if data['pack'][pack_name]['stylesheet']
|
|
||||||
end
|
|
||||||
end
|
|
||||||
pack_data
|
|
||||||
end
|
|
||||||
|
|
||||||
def pack?(data, pack_name)
|
|
||||||
if data['pack'].is_a?(Hash) && data['pack'].key?(pack_name)
|
|
||||||
return true if data['pack'][pack_name].is_a?(String) || data['pack'][pack_name].is_a?(Hash)
|
|
||||||
end
|
|
||||||
false
|
|
||||||
end
|
|
||||||
|
|
||||||
def nil_pack(data, pack_name, skin = 'default')
|
|
||||||
{
|
|
||||||
common: pack_name == 'common' ? nil : resolve_pack(data['name'] ? Themes.instance.flavour(current_flavour) : Themes.instance.core, 'common', skin),
|
|
||||||
flavour: data['name'],
|
|
||||||
pack: nil,
|
|
||||||
preload: nil,
|
|
||||||
skin: nil,
|
|
||||||
supported_locales: data['locales'],
|
|
||||||
}
|
|
||||||
end
|
|
||||||
|
|
||||||
def resolve_pack(data, pack_name, skin = 'default')
|
|
||||||
result = pack(data, pack_name, skin)
|
|
||||||
unless result
|
|
||||||
if data['name'] && data.key?('fallback')
|
|
||||||
if data['fallback'].nil?
|
|
||||||
return nil_pack(data, pack_name, skin)
|
|
||||||
elsif data['fallback'].is_a?(String) && Themes.instance.flavour(data['fallback'])
|
|
||||||
return resolve_pack(Themes.instance.flavour(data['fallback']), pack_name)
|
|
||||||
elsif data['fallback'].is_a?(Array)
|
|
||||||
data['fallback'].each do |fallback|
|
|
||||||
return resolve_pack(Themes.instance.flavour(fallback), pack_name) if Themes.instance.flavour(fallback)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
return nil_pack(data, pack_name, skin)
|
|
||||||
end
|
|
||||||
return data.key?('name') && data['name'] != Setting.default_settings['flavour'] ? resolve_pack(Themes.instance.flavour(Setting.default_settings['flavour']), pack_name) : nil_pack(data, pack_name, skin)
|
|
||||||
end
|
|
||||||
result
|
|
||||||
end
|
|
||||||
|
|
||||||
def use_pack(pack_name)
|
|
||||||
@core = resolve_pack(Themes.instance.core, pack_name)
|
|
||||||
@theme = resolve_pack(Themes.instance.flavour(current_flavour), pack_name, current_skin)
|
|
||||||
end
|
|
||||||
|
|
||||||
protected
|
protected
|
||||||
|
|
||||||
def truthy_param?(key)
|
def truthy_param?(key)
|
||||||
|
|
80
app/controllers/concerns/theming_concern.rb
Normal file
80
app/controllers/concerns/theming_concern.rb
Normal file
|
@ -0,0 +1,80 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module ThemingConcern
|
||||||
|
extend ActiveSupport::Concern
|
||||||
|
|
||||||
|
def use_pack(pack_name)
|
||||||
|
@core = resolve_pack_with_common(Themes.instance.core, pack_name)
|
||||||
|
@theme = resolve_pack_with_common(Themes.instance.flavour(current_flavour), pack_name, current_skin)
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def valid_pack_data?(data, pack_name)
|
||||||
|
data['pack'].is_a?(Hash) && [String, Hash].any? { |c| data['pack'][pack_name].is_a?(c) }
|
||||||
|
end
|
||||||
|
|
||||||
|
def nil_pack(data)
|
||||||
|
{
|
||||||
|
use_common: true,
|
||||||
|
flavour: data['name'],
|
||||||
|
pack: nil,
|
||||||
|
preload: nil,
|
||||||
|
skin: nil,
|
||||||
|
supported_locales: data['locales'],
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
def pack(data, pack_name, skin)
|
||||||
|
pack_data = {
|
||||||
|
use_common: true,
|
||||||
|
flavour: data['name'],
|
||||||
|
pack: pack_name,
|
||||||
|
preload: nil,
|
||||||
|
skin: nil,
|
||||||
|
supported_locales: data['locales'],
|
||||||
|
}
|
||||||
|
|
||||||
|
return pack_data unless data['pack'][pack_name].is_a?(Hash)
|
||||||
|
|
||||||
|
pack_data[:use_common] = false if data['pack'][pack_name]['use_common'] == false
|
||||||
|
pack_data[:pack] = nil unless data['pack'][pack_name]['filename']
|
||||||
|
|
||||||
|
preloads = data['pack'][pack_name]['preload']
|
||||||
|
pack_data[:preload] = [preloads] if preloads.is_a?(String)
|
||||||
|
pack_data[:preload] = preloads if preloads.is_a?(Array)
|
||||||
|
|
||||||
|
if skin != 'default' && data['skin'][skin]
|
||||||
|
pack_data[:skin] = skin if data['skin'][skin].include?(pack_name)
|
||||||
|
elsif data['pack'][pack_name]['stylesheet']
|
||||||
|
pack_data[:skin] = 'default'
|
||||||
|
end
|
||||||
|
|
||||||
|
pack_data
|
||||||
|
end
|
||||||
|
|
||||||
|
def resolve_pack(data, pack_name, skin)
|
||||||
|
return pack(data, pack_name, skin) if valid_pack_data?(data, pack_name)
|
||||||
|
return if data['name'].blank?
|
||||||
|
|
||||||
|
fallbacks = []
|
||||||
|
if data.key?('fallback')
|
||||||
|
fallbacks = data['fallback'] if data['fallback'].is_a?(Array)
|
||||||
|
fallbacks = [data['fallback']] if data['fallback'].is_a?(String)
|
||||||
|
elsif data['name'] != Setting.default_settings['flavour']
|
||||||
|
fallbacks = [Setting.default_settings['flavour']]
|
||||||
|
end
|
||||||
|
|
||||||
|
fallbacks.each do |fallback|
|
||||||
|
return resolve_pack(Themes.instance.flavour(fallback), pack_name) if Themes.instance.flavour(fallback)
|
||||||
|
end
|
||||||
|
|
||||||
|
nil
|
||||||
|
end
|
||||||
|
|
||||||
|
def resolve_pack_with_common(data, pack_name, skin = 'default')
|
||||||
|
result = resolve_pack(data, pack_name, skin) || nil_pack(data)
|
||||||
|
result[:common] = resolve_pack(data, 'common', skin) if result.delete(:use_common)
|
||||||
|
result
|
||||||
|
end
|
||||||
|
end
|
|
@ -14,16 +14,20 @@ class Themes
|
||||||
result = Hash.new
|
result = Hash.new
|
||||||
Dir.glob(Rails.root.join('app', 'javascript', 'flavours', '*', 'theme.yml')) do |path|
|
Dir.glob(Rails.root.join('app', 'javascript', 'flavours', '*', 'theme.yml')) do |path|
|
||||||
data = YAML.load_file(path)
|
data = YAML.load_file(path)
|
||||||
|
next unless data['pack']
|
||||||
|
|
||||||
dir = File.dirname(path)
|
dir = File.dirname(path)
|
||||||
name = File.basename(dir)
|
name = File.basename(dir)
|
||||||
locales = []
|
locales = []
|
||||||
screenshots = []
|
screenshots = []
|
||||||
|
|
||||||
if data['locales']
|
if data['locales']
|
||||||
Dir.glob(File.join(dir, data['locales'], '*.{js,json}')) do |locale|
|
Dir.glob(File.join(dir, data['locales'], '*.{js,json}')) do |locale|
|
||||||
localeName = File.basename(locale, File.extname(locale))
|
localeName = File.basename(locale, File.extname(locale))
|
||||||
locales.push(localeName) unless localeName.match(/defaultMessages|whitelist|index/)
|
locales.push(localeName) unless localeName.match(/defaultMessages|whitelist|index/)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
if data['screenshot']
|
if data['screenshot']
|
||||||
if data['screenshot'].is_a? Array
|
if data['screenshot'].is_a? Array
|
||||||
screenshots = data['screenshot']
|
screenshots = data['screenshot']
|
||||||
|
@ -31,20 +35,20 @@ class Themes
|
||||||
screenshots.push(data['screenshot'])
|
screenshots.push(data['screenshot'])
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
if data['pack']
|
|
||||||
data['name'] = name
|
data['name'] = name
|
||||||
data['locales'] = locales
|
data['locales'] = locales
|
||||||
data['screenshot'] = screenshots
|
data['screenshot'] = screenshots
|
||||||
data['skin'] = { 'default' => [] }
|
data['skin'] = { 'default' => [] }
|
||||||
result[name] = data
|
result[name] = data
|
||||||
end
|
end
|
||||||
end
|
|
||||||
|
|
||||||
Dir.glob(Rails.root.join('app', 'javascript', 'skins', '*', '*')) do |path|
|
Dir.glob(Rails.root.join('app', 'javascript', 'skins', '*', '*')) do |path|
|
||||||
ext = File.extname(path)
|
ext = File.extname(path)
|
||||||
skin = File.basename(path)
|
skin = File.basename(path)
|
||||||
name = File.basename(File.dirname(path))
|
name = File.basename(File.dirname(path))
|
||||||
if result[name]
|
next unless result[name]
|
||||||
|
|
||||||
if File.directory?(path)
|
if File.directory?(path)
|
||||||
pack = []
|
pack = []
|
||||||
Dir.glob(File.join(path, '*.{css,scss}')) do |sheet|
|
Dir.glob(File.join(path, '*.{css,scss}')) do |sheet|
|
||||||
|
@ -54,15 +58,14 @@ class Themes
|
||||||
skin = File.basename(path, ext)
|
skin = File.basename(path, ext)
|
||||||
pack = ['common']
|
pack = ['common']
|
||||||
end
|
end
|
||||||
|
|
||||||
if skin != 'default'
|
if skin != 'default'
|
||||||
result[name]['skin'][skin] = pack
|
result[name]['skin'][skin] = pack
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
|
||||||
|
|
||||||
@core = core
|
@core = core
|
||||||
@conf = result
|
@conf = result
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def core
|
def core
|
||||||
|
|
|
@ -2,12 +2,13 @@
|
||||||
- if theme[:pack] != 'common' && theme[:common]
|
- if theme[:pack] != 'common' && theme[:common]
|
||||||
= render partial: 'layouts/theme', object: theme[:common]
|
= render partial: 'layouts/theme', object: theme[:common]
|
||||||
- if theme[:pack]
|
- if theme[:pack]
|
||||||
= javascript_pack_tag theme[:flavour] ? "flavours/#{theme[:flavour]}/#{theme[:pack]}" : "core/#{theme[:pack]}", crossorigin: 'anonymous'
|
- pack_path = theme[:flavour] ? "flavours/#{theme[:flavour]}/#{theme[:pack]}" : "core/#{theme[:pack]}"
|
||||||
|
= javascript_pack_tag pack_path, crossorigin: 'anonymous'
|
||||||
- if theme[:skin]
|
- if theme[:skin]
|
||||||
- if !theme[:flavour] || theme[:skin] == 'default'
|
- if !theme[:flavour] || theme[:skin] == 'default'
|
||||||
= stylesheet_pack_tag theme[:flavour] ? "flavours/#{theme[:flavour]}/#{theme[:pack]}" : "core/#{theme[:pack]}", media: 'all', crossorigin: 'anonymous'
|
= stylesheet_pack_tag pack_path, media: 'all', crossorigin: 'anonymous'
|
||||||
- else
|
- else
|
||||||
= stylesheet_pack_tag "skins/#{theme[:flavour]}/#{theme[:skin]}/#{theme[:pack]}", crossorigin: 'anonymous'
|
= stylesheet_pack_tag "skins/#{theme[:flavour]}/#{theme[:skin]}/#{theme[:pack]}", media: 'all', crossorigin: 'anonymous'
|
||||||
- if theme[:preload]
|
- if theme[:preload]
|
||||||
- theme[:preload].each do |link|
|
- theme[:preload].each do |link|
|
||||||
%link{ href: asset_pack_path("#{link}.js"), crossorigin: 'anonymous', rel: 'preload', as: 'script' }/
|
%link{ href: asset_pack_path("#{link}.js"), crossorigin: 'anonymous', rel: 'preload', as: 'script' }/
|
||||||
|
|
Loading…
Reference in a new issue