require 'rexml/document'
require 'digest/sha1'
require 'pathname'

module Dryml

  class Template

    DRYML_NAME = "[a-zA-Z\-][a-zA-Z0-9\-]*"
    DRYML_NAME_RX = /^#{DRYML_NAME}$/

    RUBY_NAME = "[a-zA-Z_][a-zA-Z0-9_]*"
    RUBY_NAME_RX = /^#{RUBY_NAME}$/

    CODE_ATTRIBUTE_CHAR = "&"

    NO_METADATA_TAGS = %w(doctype if else unless repeat do with name type-name)

    SPECIAL_ATTRIBUTES = %w(param merge merge-params merge-attrs
                            for-type
                            if unless repeat
                            part part-locals
                            restore)

    VALID_PARAMETER_TAG_ATTRIBUTES = %w(param replace)

    @build_cache = {}

    class << self
      attr_reader :build_cache

      def clear_build_cache
        @build_cache.clear()
      end
    end

    def initialize(src, environment, template_path)
      @src = src
      @environment = environment # a class or a module
      @template_path = template_path
      @template_path = @template_path.sub(%r(^#{Regexp.escape(Rails.root.to_s)}/), "") if Object.const_defined? :Rails

      @builder = Template.build_cache[@template_path] || DRYMLBuilder.new(self)
      @builder.set_environment(environment)

      @last_element = nil
    end

    attr_reader :tags, :template_path

    def compile(local_names=[], auto_taglibs=[])
      now = Time.now

      unless @template_path.ends_with?(EMPTY_PAGE)
        p = Pathname.new template_path
        p = Pathname.new(Rails.root) + p unless p.absolute? || !Object.const_defined?(:Rails)
        mtime = p.mtime rescue Time.now

        if !@builder.ready?(mtime)
          @builder.start
          parsed = true
          # parse the DRYML file creating a list of build instructions
          if is_taglib?
            process_src
          else
            create_render_page_method
          end

          # store build instructions in the cache
          Template.build_cache[@template_path] = @builder
        end
      end

      # compile the build instructions
      @builder.build(local_names, auto_taglibs, mtime)

      logger.try.info("  DRYML: Compiled #{template_path} in #{'%.2fs' % (Time.now - now)}") if parsed
    end


    def create_render_page_method
      erb_src = process_src

      @builder.add_build_instruction(:render_page, :src => erb_src, :line_num => 1)
    end


    def is_taglib?
      @environment.class == Module
    end


    def process_src
      @doc = Dryml::Parser::Document.new(@src, @template_path)
      result = children_to_erb(@doc.root)
      restore_erb_scriptlets(result)
    end


    def restore_erb_scriptlets(src)
      @doc.restore_erb_scriptlets(src)
    end


    def children_to_erb(nodes)
      nodes.map { |x| node_to_erb(x) }.join
    end


    def node_to_erb(node)
      case node

      # v important this comes before REXML::Text, as REXML::CData < REXML::Text
      when REXML::CData
        "<% safe_concat %(#{REXML::CData::START + node.to_s + REXML::CData::STOP}) %>"

      when REXML::Comment
        "<% safe_concat %(#{REXML::Comment::START + node.to_s + REXML::Comment::STOP}) %>"

      when REXML::Text
        text_with_scriplets_to_erb(node.to_s)

      when REXML::Element
        element_to_erb(node)
      end
    end


    def text_with_scriplets_to_erb(s)
     scriplet_rex = /(\[!\[DRYML-ERB\d+\s*\]!\])/m
     s.split(scriplet_rex).map do |t|
       case t
       when scriplet_rex
         t
       when /\S+/
         "<% safe_concat %(#{t}) %>"
       else
         t
       end
     end.join
    end


    def element_to_erb(el)
      dryml_exception("old-style parameter tag (<#{el.name}>)", el) if el.name.starts_with?(":")

      @last_element = el
      case el.dryml_name

      when "include"
        include_element(el)
        # return just the newlines to keep line-number matching - the
        # include has no presence in the erb source
        tag_newlines(el)

      when "set-theme"
        require_attribute(el, "name", /^#{DRYML_NAME}$/)
        @builder.add_build_instruction(:set_theme, :name => el.attributes['name'])

        # return nothing - set_theme has no presence in the erb source
        tag_newlines(el)

      when "def"
        def_element(el)

      when "extend"
        extend_element(el)

      when "set"
        set_element(el)

      when "set-scoped"
        set_scoped_element(el)

      when "param-content"
        param_content_element(el)

      else
        if el.dryml_name.not_in?(Dryml.static_tags) || el.attributes['param'] || el.attributes['restore']
          tag_call(el)
        else
          static_element_to_erb(el)
        end
      end
    end


    def include_element(el)
      require_toplevel(el)
      require_attribute(el, "as", /^#{DRYML_NAME}$/, true)
      options = {}
      %w(src module plugin as).each do |attr|
        options[attr.to_sym] = el.attributes[attr] if el.attributes[attr]
      end
      @builder.add_build_instruction(:include, options)
    end


    def import_module(mod, as=nil)
      @builder.import_module(mod, as)
    end


    def set_element(el)
      assigns = el.attributes.map do |name, value|
        next if name.in?(SPECIAL_ATTRIBUTES)
        dryml_exception("invalid name in <set> (remember to use '-' rather than '_')", el) unless name =~ /^#{DRYML_NAME}(\.#{DRYML_NAME})*$/
        "#{ruby_name name} = #{attribute_to_ruby(value)}; "
      end.join
      code = apply_control_attributes("begin; #{assigns}; end", el)
      "<% #{code}#{tag_newlines(el)} %>"
    end


    def set_scoped_element(el)
      variables = el.attributes.map do |name, value|
        dryml_exception("invalid name in <set-scoped> (remember to use '-' rather than '_')", el) unless name =~ DRYML_NAME_RX
        ":#{ruby_name name} => #{attribute_to_ruby(value)} "
      end
      "<% scope.new_scope(#{variables * ', '}) { #{tag_newlines(el)} %>#{children_to_erb(el)}<% } %>"
    end


    def declared_attributes(def_element)
      attrspec = def_element.attributes["attrs"]
      attr_names = attrspec ? attrspec.split(/\s*,\s*/).map{ |n| n.underscore.to_sym } : []
      invalids = attr_names & ([:with, :field, :this] + SPECIAL_ATTRIBUTES.*.to_sym)
      dryml_exception("invalid attrs in def: #{invalids * ', '}", def_element) unless invalids.empty?
      attr_names
    end


    def ruby_name(dryml_name)
      dryml_name.gsub('-', '_')
    end


    def with_containing_tag_name(el)
      old = @containing_tag_name
      @containing_tag_name = el.dryml_name
      yield
      @containing_tag_name = old
    end


    def define_polymorphic_dispatcher(el, name)
      # FIXME: The new erb context ends up being set-up twice
      src = %(
      def #{name}(attributes={}, parameters={})
        _tag_context(attributes) do
          attributes.delete :with
          attributes.delete :field
          call_polymorphic_tag('#{name}', attributes, parameters) { #{name}__base(attributes.except, parameters) }
        end
      end
      )
      @builder.add_build_instruction(:eval, :src => src, :line_num => element_line_num(el))
    end


    def extend_element(el)
      def_element(el, true)
    end


    def type_specific_suffix
      el = @def_element
      for_type = el.attributes['for']
      if for_type
        type_name = if defined?(HoboFields) && for_type =~ /^[a-z]/
                      # It's a symbolic type name - look up the Ruby type name
                      klass = HoboFields.to_class(for_type) or
                        dryml_exception("No such type in polymorphic tag definition: '#{for_type}'", el)
                      klass.name
                    else
                      for_type
                    end.underscore.gsub('/', '__')
        "__for_#{type_name}"
      end
    end


    def def_element(el, extend_tag=false)
      require_toplevel(el)
      require_attribute(el, "tag", DRYML_NAME_RX)
      require_attribute(el, "attrs", /^\s*#{DRYML_NAME}(\s*,\s*#{DRYML_NAME})*\s*$/, true)
      require_attribute(el, "alias-of", DRYML_NAME_RX, true)

      @def_element = el

      unsafe_name = el.attributes["tag"]
      name = Dryml.unreserve(unsafe_name)
      suffix = type_specific_suffix
      if suffix
        name        += suffix
        unsafe_name += suffix
      end

      if el.attributes['polymorphic']
        %w(for alias-of).each do |attr|
          dryml_exception("def cannot have both 'polymorphic' and '#{attr}' attributes") if el.attributes[attr]
        end

        define_polymorphic_dispatcher(el, ruby_name(name))
        name        += "__base"
        unsafe_name += "__base"
      end

      alias_of = el.attributes['alias-of']
      dryml_exception("def with alias-of must be empty", el) if alias_of and el.size > 0

      alias_of and @builder.add_build_instruction(:alias_method,
                                                  :new => ruby_name(name).to_sym,
                                                  :old => ruby_name(Dryml.unreserve(alias_of)).to_sym)

      res = if alias_of
              "<% #{tag_newlines(el)} %>"
            else
              src = tag_method(name, el, extend_tag) +
                "<% _register_tag_attrs(:#{ruby_name name}, #{declared_attributes(el).inspect.underscore}) %>"

              logger.debug(restore_erb_scriptlets(src)) if el.attributes["debug-source"]

              @builder.add_build_instruction(:def,
                                             :src => restore_erb_scriptlets(src),
                                             :line_num => element_line_num(el))
              # keep line numbers matching up
              "<% #{"\n" * src.count("\n")} %>"
            end
      @def_element = nil
      res
    end

    def self.descendents(el,&block)
      return if el.elements.empty?
      el.elements.each do |child|
        block.call(child)
        descendents(child,&block)
      end
    end

    # Using REXML::XPath is slow
    def self.descendent_select(el)
      result = []
      descendents(el) { |desc|
        result << desc if yield(desc)
      }
      result
    end

    def param_names_in_definition(el)
      self.class.descendent_select(el) { |el| el.attribute 'param' }.map do |e|
        name = get_param_name(e)
        dryml_exception("invalid param name: #{name.inspect}", e) unless
          is_code_attribute?(name) || name =~ RUBY_NAME_RX || name =~ /#\{/
        name.to_sym unless is_code_attribute?(name)
      end.compact
    end


    def tag_method(name, el, extend_tag=false)
      name = ruby_name name
      param_names = param_names_in_definition(el)

      if extend_tag
        @extend_key = 'a' + Digest::SHA1.hexdigest(el.to_s)[0..10]
        alias_statement = "; alias_method_chain_on_include :#{name}, :#{@extend_key}"
        name = "#{name}_with_#{@extend_key}"
      end

      src = "<% def #{name}(all_attributes={}, all_parameters={}); " +
        "parameters = Dryml::TagParameters.new(all_parameters, #{param_names.inspect.underscore}); " +
        "all_parameters = Dryml::TagParameters.new(all_parameters); " +
        tag_method_body(el) +
        "; end#{alias_statement} %>"
      @extend_key = nil
      src
    end


    def tag_method_body(el)
      attrs = declared_attributes(el)

      # A statement to assign values to local variables named after the tag's attrs
      # The trailing comma on `attributes` is supposed to be there!
      setup_locals = attrs.map{|a| "#{Dryml.unreserve(a).underscore}, "}.join + "attributes, = " +
        "_tag_locals(all_attributes, #{attrs.inspect})"

      start = "_tag_context(all_attributes) do #{setup_locals}"

      "#{start} " +
        # reproduce any line breaks in the start-tag so that line numbers are preserved
        tag_newlines(el) + "%>" +
        wrap_tag_method_body_with_metadata(children_to_erb(el)) +
        "<% output_buffer; end"
    end


    def wrap_source_with_metadata(content, kind, name, *args)
      if (!include_source_metadata) || name.in?(NO_METADATA_TAGS)
        content
      else
        metadata = [kind, name] + args + [@template_path]
        "<!--[DRYML|#{metadata * '|'}[-->" + content + "<!--]DRYML]-->"
      end
    end


    def wrap_tag_method_body_with_metadata(content)
      name   = @def_element.attributes['tag']
      for_   = @def_element.attributes['for']
      name += " for #{for_}" if for_
      wrap_source_with_metadata(content, "def", name, element_line_num(@def_element))
    end


    def wrap_tag_call_with_metadata(el, content)
      name = el.expanded_name
      param = el.attributes['param']

      if param == "&true"
        name += " param"
      elsif param
        name += " param='#{param}'"
      end

      wrap_source_with_metadata(content, "call", name, element_line_num(el))
    end


    def param_content_local_name(name)
      "_#{ruby_name name}__default_content"
    end


    def param_content_element(name_or_el)
      name = if name_or_el.is_a?(String)
               name_or_el
             else
               el = name_or_el
               el.attributes['for'] || @containing_tag_name
             end
      local_name = param_content_local_name(name)
      "<%= #{local_name}.call if #{local_name} %>"
    end


    def part_element(el, content)
      require_attribute(el, "part", DRYML_NAME_RX)

      if contains_param?(el)
        delegated_part_element(el, content)
      else
        simple_part_element(el, content)
      end
    end


    def simple_part_element(el, content)
      part_name  = el.attributes['part']
      dom_id = el.attributes['id'] || part_name
      part_name = ruby_name(part_name)
      part_locals = el.attributes["part-locals"]

      part_src = "<% def #{part_name}_part(#{part_locals._?.gsub('@', '')}) #{tag_newlines(el)}; new_context do %>" +
        content +
        "<% end; end %>"
      @builder.add_part(part_name, restore_erb_scriptlets(part_src), element_line_num(el))

      newlines = "\n" * part_src.count("\n")
      args = [attribute_to_ruby(dom_id), ":#{part_name}", part_locals].compact
      "<%=raw call_part(#{args * ', '}) #{newlines} %>"
    end


    def delegated_part_element(el, content)
      # TODO
    end


    def contains_param?(el)
      # TODO
      false
    end


    def part_delegate_tag_name(el)
      "#{@def_name}_#{el.attributes['part']}__part_delegate"
    end


    def current_def_name
      @def_element && @def_element.attributes['tag']
    end


    def get_param_name(el)
      param_name = el.attributes["param"]

      if param_name
        def_tag = find_ancestor(el) {|e| e.name == "def" || e.name == "extend" }
        dryml_exception("param is not allowed outside of tag definitions", el) if def_tag.nil?

        ruby_name(param_name == "&true" ? el.dryml_name : param_name)
      else
        nil
      end
    end


    def inside_def_for_type?
      @def_element && @def_element.attributes['for']
    end


    def call_name(el)
      dryml_exception("invalid tag name (remember to use '-' rather than '_')", el) unless el.dryml_name =~ /^#{DRYML_NAME}(\.#{DRYML_NAME})*$/

      name = Dryml.unreserve(ruby_name(el.dryml_name))
      if call_to_self_from_type_specific_def?(el)
        "#{name}__base"
      elsif old_tag_call?(el)
        name = name[4..-1] # remove 'old-' prefix
        name += type_specific_suffix if inside_def_for_type?
        "#{name}_without_#{@extend_key}"
      else
        name
      end
    end


    def old_tag_call?(el)
      @def_element && el.dryml_name == "old-#{current_def_name}"
    end


    def call_to_self_from_type_specific_def?(el)
      inside_def_for_type? && el.dryml_name == current_def_name &&!el.attributes['for-type']
    end


    def polymorphic_call_type(el)
      t = el.attributes['for-type']
      if t.nil?
        nil
      elsif t == "&true"
        'this_type'
      elsif t =~ /^[A-Z]/
        t
      elsif t =~ /^[a-z]/ && defined? HoboFields.to_class
        klass = HoboFields.to_class(t)
        klass.name
      elsif is_code_attribute?(t)
        t[1..-1]
      else
        dryml_exception("invalid for-type attribute", el)
      end
    end


    def tag_call(el)
      name = call_name(el)
      param_name = get_param_name(el)
      attributes = tag_attributes(el)
      newlines = tag_newlines(el)

      parameters = tag_newlines(el) + parameter_tags_hash(el)

      is_param_restore = el.attributes['restore']

      call = if param_name
               param_name = attribute_to_ruby(param_name, :symbolize => true)
               args = "#{attributes}, #{parameters}, all_parameters, #{param_name}"
               to_call = if is_param_restore
                           # The tag is available in a local variable
                           # holding a proc
                           param_restore_local_name(name)
                         elsif (call_type = polymorphic_call_type(el))
                           "find_polymorphic_tag(:#{ruby_name name}, #{call_type})"
                         else
                           ":#{ruby_name name}"
                         end
               "call_tag_parameter(#{to_call}, #{args})"
             else
               if is_param_restore
                 # The tag is a proc available in a local variable
                 "#{param_restore_local_name(name)}.call(#{attributes}, #{parameters})"
               elsif (call_type = polymorphic_call_type(el))
                 "send(find_polymorphic_tag(:#{ruby_name name}, #{call_type}), #{attributes}, #{parameters})"
               elsif attributes == "{}" && parameters == "{}"
                 if name =~ /^[A-Z]/
                   # it's a tag with a cap name - not a local
                   "#{ruby_name name}()"
                 else
                   # could be a tag or a local variable
                   "#{ruby_name name}.to_s"
                 end
               else
                 "#{ruby_name name}(#{attributes}, #{parameters})"
               end
             end

      call = apply_control_attributes(call, el)
      call = maybe_make_part_call(el, "<% concat(#{call}) %>")
      wrap_tag_call_with_metadata(el, call)
    end


    def merge_attribute(el)
      merge = el.attributes['merge']
      dryml_exception("merge cannot have a RHS", el) if merge && merge != "&true"
      merge
    end


    def parameter_tags_hash(el, containing_tag_name=nil)
      call_type = nil

      metadata_name = containing_tag_name || el.expanded_name

      param_items = el.map do |node|
        case node
        when REXML::Text
          text = node.to_s
          unless text.blank?
            dryml_exception("mixed content and parameter tags", el) if call_type == :named_params
            call_type = :default_param_only
          end
          text
        when REXML::Element
          e = node
          is_parameter_tag = e.parameter_tag?

          # Make sure there isn't a mix of parameter tags and normal content
          case call_type
          when nil
            call_type = is_parameter_tag ? :named_params : :default_param_only
          when :named_params
            dryml_exception("mixed parameter tags and non-parameter tags (did you forget a ':'?)", el) unless is_parameter_tag
          when :default_param_only
            dryml_exception("mixed parameter tags and non-parameter tags (did you forget a ':'?)", el) if is_parameter_tag
          end

          if is_parameter_tag
            parameter_tag_hash_item(e, metadata_name) + ", "
          end
        end
      end.join

      if call_type == :default_param_only || (call_type.nil? && param_items.length > 0) || (el.children.empty? && el.has_end_tag?)
        with_containing_tag_name(el) do
          param_items = " :default => #{default_param_proc(el, containing_tag_name)}, "
        end
      end

      param_items.concat without_parameters(el)

      merge_params = el.attributes['merge-params'] || merge_attribute(el)
      if merge_params
        extra_params = if merge_params == "&true"
                         "parameters"
                       elsif is_code_attribute?(merge_params)
                         merge_params[1..-1]
                       else
                         merge_param_names = merge_params.split(/\s*,\s*/).*.gsub("-", "_")
                         "all_parameters & #{merge_param_names.inspect}"
                       end
        "merge_parameter_hashes({#{param_items}}, (#{extra_params}) || {})"
      else
        "{#{param_items}}"
      end
    end


    def without_parameters(el)
      without_names = el.attributes.keys.map { |name| name =~ /^without-(.*)/ and $1 }.compact
      without_names.map { |name| ":#{ruby_name name}_replacement => proc {|__discard__| '' }, " }.join
    end


    def parameter_tag_hash_item(el, metadata_name)
      name = el.name.dup
      if name.sub!(/^before-/, "")
        before_parameter_tag_hash_item(name, el, metadata_name)
      elsif name.sub!(/^after-/, "")
        after_parameter_tag_hash_item(name, el, metadata_name)
      elsif name.sub!(/^prepend-/, "")
        prepend_parameter_tag_hash_item(name, el, metadata_name)
      elsif name.sub!(/^append-/, "")
        append_parameter_tag_hash_item(name, el, metadata_name)
      else
        hash_key = ruby_name name
        hash_key += "_replacement" if el.attribute("replace")
        if (param_name = get_param_name(el))
          ":#{hash_key} => merge_tag_parameter(#{param_proc(el, metadata_name)}, all_parameters[:#{param_name}])"
        else
          ":#{hash_key} => #{param_proc(el, metadata_name)}"
        end
      end
    end


    def before_parameter_tag_hash_item(name, el, metadata_name)
      param_name = get_param_name(el)
      dryml_exception("param declaration not allowed on 'before' parameters", el) if param_name
      content = children_to_erb(el) + "<% concat(#{param_restore_local_name(name)}.call({}, {})) %>"
      ":#{ruby_name name}_replacement => #{replace_parameter_proc(el, metadata_name, content)}"
    end


    def after_parameter_tag_hash_item(name, el, metadata_name)
      param_name = get_param_name(el)
      dryml_exception("param declaration not allowed on 'after' parameters", el) if param_name
      content = "<% concat(#{param_restore_local_name(name)}.call({}, {})) %>" + children_to_erb(el)
      ":#{ruby_name name}_replacement => #{replace_parameter_proc(el, metadata_name, content)}"
    end


    def append_parameter_tag_hash_item(name, el, metadata_name)
      ":#{ruby_name name} => proc { [{}, { :default => proc { |#{param_content_local_name(name)}| new_context { %>" +
        param_content_element(name) + children_to_erb(el) +
        "<% } } } ] }"
    end


    def prepend_parameter_tag_hash_item(name, el, metadata_name)
      ":#{ruby_name name} => proc { [{}, { :default => proc { |#{param_content_local_name(name)}| new_context { %>" +
        children_to_erb(el) + param_content_element(name) +
        "<% } } } ] }"
    end


    def default_param_proc(el, containing_param_name=nil)
      content = children_to_erb(el)
      content = wrap_source_with_metadata(content, "param", containing_param_name,
                                          element_line_num(el)) if containing_param_name
      "proc { |#{param_content_local_name(el.dryml_name)}| new_context { %>#{content}<% } #{tag_newlines(el)}}"
    end


    def param_restore_local_name(name)
      "_#{ruby_name name}_restore"
    end


    def wrap_replace_parameter(el, name)
      wrap_source_with_metadata(children_to_erb(el), "replace", name, element_line_num(el))
    end


    def param_proc(el, metadata_name_prefix)
      metadata_name = "#{metadata_name_prefix}><#{el.name}"

      nl = tag_newlines(el)

      if (repl = el.attribute("replace"))
        dryml_exception("replace attribute must not have a value", el) if repl.has_rhs?
        dryml_exception("replace parameters must not have attributes", el) if el.attributes.length > 1

        replace_parameter_proc(el, metadata_name)
      else
        attributes = el.attributes.dup
        # Providing one of 'with' or 'field' but not the other should cancel out the other
        attributes[:with] = "&nil"  if attributes.key?(:field) && !attributes.key?(:with)
        attributes[:field] = "&nil" if !attributes.key?(:field) && attributes.key?(:with)
        attribute_items = attributes.map do |name, value|
          if name.in?(VALID_PARAMETER_TAG_ATTRIBUTES)
            # just ignore
          elsif name.in?(SPECIAL_ATTRIBUTES)
            dryml_exception("attribute '#{name}' is not allowed on parameter tags (<#{el.name}:>)", el)
          else
            ":#{ruby_name name} => #{attribute_to_ruby(value, el)}"
          end
        end.compact

        nested_parameters_hash = parameter_tags_hash(el, metadata_name)
        "proc { [{#{attribute_items * ', '}}, #{nested_parameters_hash}] #{nl}}"
      end
    end


    def replace_parameter_proc(el, metadata_name, content=nil)
      content ||= wrap_replace_parameter(el, metadata_name)
      param_name = el.dryml_name.sub(/^(before|after|append|prepend)-/, "")
      "proc { |#{param_restore_local_name(param_name)}| new_context { %>#{content}<% } #{tag_newlines(el)}}"
    end


    def maybe_make_part_call(el, call)
      part_name = el.attributes['part']
      if part_name
        part_id = el.attributes['id'] || part_name

        "<% safe_concat(\"<div class='part-wrapper' id='#{attribute_to_ruby(part_id)[1..-2]}'>\") %>" +
          part_element(el, call) +
          "<% safe_concat(%(</div>)) %>"
      else
        call
      end
    end


    def field_shorthand_element?(el)
      el.expanded_name =~ /:./
    end


    def tag_attributes(el)
      attributes = el.attributes
      items = attributes.map do |n,v|
        dryml_exception("invalid attribute name '#{n}' (remember to use '-' rather than '_')", el) unless n =~ DRYML_NAME_RX

        next if n.in?(SPECIAL_ATTRIBUTES) || n =~ /^without-/
        next if el.attributes['part'] && n == 'id' # The id is rendered on the <div class="part-wrapper"> instead

        ":#{ruby_name n} => #{attribute_to_ruby(v)}"
      end.compact

      # if there's a ':' el.name is just the part after the ':'
      items << ":field => \"#{ruby_name el.name}\"" if field_shorthand_element?(el)

      hash = "{#{items.join(", ")}}"

      if merge_attribute(el)
        "merge_attrs(#{hash}, attributes)"
      elsif el.attributes['merge-attrs']
        merge_attrs = compile_merge_attrs(el)
        "merge_attrs(#{hash}, #{merge_attrs} || {})"
      else
        hash
      end
    end


    def compile_merge_attrs(el)
      merge_attrs = el.attributes['merge-attrs']
      if merge_attrs == "&true"
        "attributes"
      elsif is_code_attribute?(merge_attrs)
        "(#{merge_attrs[1..-1]})"
      else
        merge_attr_names = merge_attrs.split(/\s*,\s*/).*.gsub("-", "_").*.to_sym
        "(all_attributes & #{merge_attr_names.inspect})"
      end
    end


    def static_tag_to_method_call(el)
      part = el.attributes["part"]
      attrs = el.attributes.map do |n, v|
        next if n.in? SPECIAL_ATTRIBUTES
        val = restore_erb_scriptlets(v).gsub('"', '\"').gsub(/<%=(.*?)%>/, '#{\1}')
        %('#{n}' => "#{val}")
      end.compact

      # If there's a part but no id, the id defaults to the part name
      if part && !el.attributes["id"]
        attrs << ":id => '#{part}'"
      end

      # Convert the attributes hash to a call to merge_attrs if
      # there's a merge-attrs attribute
      attrs = if el.attributes['merge-attrs']
                merge_attrs = compile_merge_attrs(el)
                "merge_attrs({#{attrs * ', '}}, #{merge_attrs} || {})"
              else
                "{" + attrs.join(', ') + "}"
              end

      if el.children.empty?
        dryml_exception("part attribute on empty static tag", el) if part

        "<%= " + apply_control_attributes("element(:#{el.name}, #{attrs}, nil, true, #{!el.has_end_tag?} #{tag_newlines(el)})", el) + " %>"
      else
        if part
          body = part_element(el, children_to_erb(el))
        else
          body = children_to_erb(el)
        end

        output_tag = "element(:#{el.name}, #{attrs}, new_context { %>#{body}<% })"
        "<% concat(" + apply_control_attributes(output_tag, el) + ") %>"
      end
    end


    def static_element_to_erb(el)
      if promote_static_tag_to_method_call?(el)
        static_tag_to_method_call(el)
      else
        start_tag_src = el.start_tag_source.gsub(REXML::CData::START, "").gsub(REXML::CData::STOP, "")

        sts = "<% safe_concat(%(#{start_tag_src}))%>"
        if el.has_end_tag?
          sts + children_to_erb(el) + "<% safe_concat(%(</#{el.name}>)) %>"
        else
          sts
        end
      end
    end


    def promote_static_tag_to_method_call?(el)
      %w(part merge-attrs if unless repeat).any? {|x| el.attributes[x]}
    end


    def apply_control_attributes(expression, el)
      controls = %w(if unless repeat).map_hash { |x| el.attributes[x] }.compact

      dryml_exception("You can't have multiple control attributes on the same element", el) if
        controls.length > 1

      attr = controls.keys.first
      val = controls.values.first
      if val.nil?
        expression
      else
        control = if !el.attribute(attr).has_rhs?
                    "this"
                  elsif is_code_attribute?(val)
                    "#{val[1..-1]}"
                  else
                    val.gsub!('-', '_')
                    attr == "repeat" ? %("#{val}") : "this.#{val}"
                  end

        x = gensym
        case attr
        when "if"
          "(if !(#{control}).blank?; (#{x} = #{expression}; Dryml.last_if = true; #{x}) " +
            "else (Dryml.last_if = false; ''); end)"
        when "unless"
          "(if (#{control}).blank?; (#{x} = #{expression}; Dryml.last_if = true; #{x}) " +
            "else (Dryml.last_if = false; ''); end)"
        when "repeat"
          "repeat_attribute(#{control}) { #{expression} }"
        end
      end
    end


    def attribute_to_ruby(*args)
      options = args.extract_options!
      attr, el = args

      dryml_exception('erb scriptlet not allowed in this attribute (use #{ ... } instead)', el) if
        attr.is_a?(String) && attr.index("[![HOBO-ERB")

      if options[:symbolize] && attr =~ /^[a-zA-Z_][^a-zA-Z0-9_]*[\?!]?/
        ":#{attr}"
      else
        res = if attr.nil?
                "nil"
              elsif is_code_attribute?(attr)
                "(#{attr[1..-1]})"
              else
                if attr !~ /"/
                  if attr =~ /\#\{/
                    '"' + attr + '"'
                  else
                    '"' + attr + '"'
                  end
                elsif attr !~ /'/
                  "'#{attr}'"
                else
                  dryml_exception("invalid quote(s) in attribute value")
                end
              end
        options[:symbolize] ? (res + ".to_sym") : res
      end
    end

    def find_ancestor(el)
      e = el.parent
      until e.is_a? REXML::Document
        return e if yield(e)
        e = e.parent
      end
      return nil
    end

    def require_toplevel(el, message=nil)
      message ||= "can only be at the top level"
      dryml_exception("<#{el.dryml_name}> #{message}", el) if el.parent != @doc.root
    end

    def require_attribute(el, name, rx=nil, optional=false)
      val = el.attributes[name]
      if val
        dryml_exception("invalid #{name}=\"#{val}\" attribute on <#{el.dryml_name}>", el) unless rx && val =~ rx
      else
        dryml_exception("missing #{name} attribute on <#{el.dryml_name}>", el) unless optional
      end
    end

    def dryml_exception(message, el=nil)
      el ||= @last_element
      raise DrymlException.new(message, template_path, element_line_num(el))
    end

    def element_line_num(el)
      @doc.element_line_num(el)
    end

    def tag_newlines(el)
      src = el.start_tag_source
      "\n" * src.count("\n")
    end

    def is_code_attribute?(attr_value)
      attr_value =~ /^\&/ && attr_value !~ /^\&\S+;/
    end

    def logger
      ActionController::Base.logger rescue nil
    end

    def gensym(name="__tmp")
      @gensym_counter ||= 0
      @gensym_counter += 1
      "#{name}_#{@gensym_counter}"
    end

    def include_source_metadata
      # disabled for now -- we're still getting broken rendering with this feature on
      return false
      @include_source_metadata = Rails.env.development? && !ENV['DRYML_EDITOR'].blank? if @include_source_metadata.nil?
      @include_source_metadata
    end

  end

end