discourse/app/models/site_customization.rb

171 lines
4.9 KiB
Ruby
Raw Normal View History

require_dependency 'sass/discourse_sass_compiler'
2014-11-20 16:32:16 -05:00
require_dependency 'sass/discourse_stylesheets'
require_dependency 'distributed_cache'
2013-02-05 14:16:51 -05:00
class SiteCustomization < ActiveRecord::Base
ENABLED_KEY = '7e202ef2-56d7-47d5-98d8-a9c8d15e57dd'
@cache = DistributedCache.new('site_customization')
2013-02-05 14:16:51 -05:00
2013-02-07 10:45:24 -05:00
before_create do
2013-02-05 14:16:51 -05:00
self.enabled ||= false
self.key ||= SecureRandom.uuid
true
end
def compile_stylesheet(scss)
DiscourseSassCompiler.compile(scss, 'custom')
rescue => e
puts e.backtrace.join("\n") unless Sass::SyntaxError === e
raise e
end
2013-02-07 10:45:24 -05:00
before_save do
['stylesheet', 'mobile_stylesheet'].each do |stylesheet_attr|
if self.send("#{stylesheet_attr}_changed?")
begin
self.send("#{stylesheet_attr}_baked=", compile_stylesheet(self.send(stylesheet_attr)))
rescue Sass::SyntaxError => e
self.send("#{stylesheet_attr}_baked=", DiscourseSassCompiler.error_as_css(e, "custom stylesheet"))
end
2013-02-05 14:16:51 -05:00
end
end
end
after_save do
remove_from_cache!
2014-11-10 15:51:55 -05:00
if stylesheet_changed? || mobile_stylesheet_changed?
MessageBus.publish "/file-change/#{key}", SecureRandom.hex
2014-12-22 21:03:35 -05:00
MessageBus.publish "/file-change/#{SiteCustomization::ENABLED_KEY}", SecureRandom.hex
2013-02-05 14:16:51 -05:00
end
MessageBus.publish "/header-change/#{key}", header if header_changed?
2014-11-10 15:51:55 -05:00
MessageBus.publish "/footer-change/#{key}", footer if footer_changed?
2014-11-20 16:32:16 -05:00
DiscourseStylesheets.cache.clear
2013-02-05 14:16:51 -05:00
end
after_destroy do
remove_from_cache!
2013-02-05 14:16:51 -05:00
end
def self.enabled_key
ENABLED_KEY.dup << RailsMultisite::ConnectionManagement.current_db
end
def self.enabled_stylesheet_contents(target=:desktop)
@cache["enabled_stylesheet_#{target}"] ||= where(enabled: true)
.order(:name)
.pluck(target == :desktop ? :stylesheet_baked : :mobile_stylesheet_baked)
.compact
.join("\n")
end
2013-02-05 14:16:51 -05:00
def self.stylesheet_contents(key, target)
if key == ENABLED_KEY
enabled_stylesheet_contents(target)
2013-02-07 02:25:18 -05:00
else
where(key: key)
.pluck(target == :mobile ? :mobile_stylesheet_baked : :stylesheet_baked)
.first
2013-02-07 02:25:18 -05:00
end
2013-02-05 14:16:51 -05:00
end
def self.custom_stylesheet(preview_style=nil, target=:desktop)
preview_style ||= ENABLED_KEY
if preview_style == ENABLED_KEY
stylesheet_link_tag(ENABLED_KEY, target, enabled_stylesheet_contents(target))
2014-11-10 15:51:55 -05:00
else
lookup_field(preview_style, target, :stylesheet_link_tag)
2014-11-10 15:51:55 -05:00
end
end
%i{header footer head_tag body_tag}.each do |name|
define_singleton_method("custom_#{name}") do |preview_style=nil, target=:desktop|
preview_style ||= ENABLED_KEY
lookup_field(preview_style, target, name)
end
end
def self.lookup_field(key, target, field)
2013-02-05 14:16:51 -05:00
return if key.blank?
2013-02-07 10:45:24 -05:00
cache_key = key + target.to_s + field.to_s;
2013-02-07 10:45:24 -05:00
lookup = @cache[cache_key]
return lookup.html_safe if lookup
2013-02-05 14:16:51 -05:00
styles = if key == ENABLED_KEY
order(:name).where(enabled:true).to_a
else
[find_by(key: key)].compact
end
2013-02-05 14:16:51 -05:00
val = if styles.present?
styles.map do |style|
lookup = target == :mobile ? "mobile_#{field}" : field
style.send(lookup)
end.compact.join("\n")
end
(@cache[cache_key] = val || "").html_safe
2013-02-05 14:16:51 -05:00
end
def self.remove_from_cache!(key, broadcast = true)
MessageBus.publish('/site_customization', key: key) if broadcast
clear_cache!
end
def self.clear_cache!
@cache.clear
2013-02-05 14:16:51 -05:00
end
def remove_from_cache!
self.class.remove_from_cache!(self.class.enabled_key)
self.class.remove_from_cache!(key)
2013-02-05 14:16:51 -05:00
end
def mobile_stylesheet_link_tag
stylesheet_link_tag(:mobile)
2013-02-05 14:16:51 -05:00
end
def stylesheet_link_tag(target=:desktop)
content = target == :mobile ? mobile_stylesheet : stylesheet
SiteCustomization.stylesheet_link_tag(key, target, content)
2013-02-05 14:16:51 -05:00
end
def self.stylesheet_link_tag(key, target, content)
return "" unless content.present?
hash = Digest::MD5.hexdigest(content)
link_css_tag "/site_customizations/#{key}.css?target=#{target}&v=#{hash}"
2014-07-29 00:55:48 -04:00
end
def self.link_css_tag(href)
href = (GlobalSetting.cdn_url || "") + "#{href}&__ws=#{Discourse.current_hostname}"
%Q{<link class="custom-css" rel="stylesheet" href="#{href}" type="text/css" media="all">}.html_safe
end
2013-02-05 14:16:51 -05:00
end
# == Schema Information
#
# Table name: site_customizations
#
# id :integer not null, primary key
# name :string(255) not null
# stylesheet :text
# header :text
# user_id :integer not null
# enabled :boolean not null
# key :string(255) not null
# created_at :datetime not null
# updated_at :datetime not null
# stylesheet_baked :text default(""), not null
# mobile_stylesheet :text
# mobile_header :text
# mobile_stylesheet_baked :text
# footer :text
# mobile_footer :text
#
# Indexes
#
# index_site_customizations_on_key (key)
#