require 'polyglot' require 'deface/dsl/context' module Deface module DSL class Loader def self.load(filename, options = nil, &block) unless File.basename(filename) =~ /^[^\.]+(.html.(erb|haml|slim)){0,1}.deface$/ raise "Deface::DSL does not know how to read '#{filename}'. Override files should end with just .deface, .html.erb.deface, .html.haml.deface or .html.slim.deface" end unless file_in_dir_below_overrides?(filename) raise "Deface::DSL overrides must be in a sub-directory that matches the views virtual path. Move '#{filename}' into a sub-directory." end File.open(filename) do |file| context_name = File.basename(filename).gsub('.deface', '') file_contents = file.read build_context(context_name, filename, file_contents) end end def self.build_context(context_name, filename, file_contents) send build_context_method_name(context_name), context_name, filename, file_contents end def self.build_context_method_name(context_name) ext = File.extname(context_name).gsub(".", '') ext = "other" if ext.empty? "build_#{ext}_context" end def self.build_erb_context(context_name, filename, file_contents) build_context_and_extract_dsl_from('erb', context_name, filename, file_contents) end def self.build_haml_context(context_name, filename, file_contents) build_context_and_extract_dsl_from('haml', context_name, filename, file_contents) end def self.build_slim_context(context_name, filename, file_contents) build_context_and_extract_dsl_from('slim', context_name, filename, file_contents) end def self.build_context_and_extract_dsl_from(type, context_name, filename, file_contents) dsl_commands, the_rest = send "extract_dsl_commands_from_#{type}", (file_contents) context_name = context_name.gsub(".html.#{type}", '') context = Context.new(context_name) context.virtual_path(determine_virtual_path(filename)) context.instance_eval(dsl_commands) context.send type, the_rest context.create_override end def self.build_other_context(context_name, filename, file_contents) context = Context.new(context_name) context.virtual_path(determine_virtual_path(filename)) context.instance_eval(file_contents) context.create_override end def self.register Polyglot.register('deface', Deface::DSL::Loader) end def self.extract_dsl_commands_from_erb(html_file_contents) dsl_commands = '' while starts_with_html_comment?(html_file_contents) first_open_comment_index = html_file_contents.lstrip.index('<!--') first_close_comment_index = html_file_contents.index('-->') unless first_close_comment_index.nil? comment = html_file_contents[first_open_comment_index..first_close_comment_index+2] end comment.gsub('<!--', '').gsub('-->', '').strip.scan(/[^\s"']+|"[^"]*"|'[^']*'/).each do |part| dsl_commands =~ /('|")\z/ || part =~ /\A[^\d:='"%]/ ? dsl_commands << "\n" : dsl_commands << ' ' dsl_commands << part end html_file_contents = html_file_contents.gsub(comment, '') end [dsl_commands, html_file_contents] end def self.extract_dsl_commands_from_haml(file_contents) dsl_commands = '' while starts_with_haml_comment?(file_contents) first_open_comment_index = file_contents.lstrip.index('/') first_close_comment_index = file_contents.index("\n") unless first_close_comment_index.nil? comment = file_contents[first_open_comment_index..first_close_comment_index] end dsl_commands << comment.gsub('/', '').strip + "\n" file_contents = file_contents.gsub(comment, '') while file_contents.start_with?(' ') first_newline_index = file_contents.index("\n") comment = file_contents[0..first_newline_index] dsl_commands << comment.gsub('/', '').strip + "\n" file_contents = file_contents.gsub(comment, '') end end [dsl_commands, file_contents] end class << self alias_method :extract_dsl_commands_from_slim, :extract_dsl_commands_from_haml end private def self.starts_with_html_comment?(line) line.lstrip.index('<!--') == 0 end def self.starts_with_haml_comment?(line) line.lstrip.index('/') == 0 end def self.file_in_dir_below_overrides?(filename) File.fnmatch?("**/overrides/**/#{File.basename(filename)}", filename) end def self.determine_virtual_path(filename) result = '' pathname = Pathname.new(filename) pathname.ascend do |parent| if parent.basename.to_s == 'overrides' result = pathname.sub(parent.to_s + '/', '').dirname.to_s break end end result end end end end