vendor/plugins/haml/lib/haml/html.rb in radiant-0.9.1 vs vendor/plugins/haml/lib/haml/html.rb in radiant-1.0.0.rc1
- old
+ new
@@ -7,41 +7,79 @@
module Haml
class HTML
# A module containing utility methods that every Hpricot node
# should have.
module Node
+ # Whether this node has already been converted to Haml.
+ # Only used for text nodes and elements.
+ #
+ # @return [Boolean]
+ attr_accessor :converted_to_haml
+
# Returns the Haml representation of the given node.
#
# @param tabs [Fixnum] The indentation level of the resulting Haml.
# @option options (see Haml::HTML#initialize)
def to_haml(tabs, options)
- parse_text(self.to_s, tabs)
+ return "" if converted_to_haml || to_s.strip.empty?
+ text = uninterp(self.to_s)
+ node = next_node
+ while node.is_a?(::Hpricot::Elem) && node.name == "haml:loud"
+ node.converted_to_haml = true
+ text << '#{' <<
+ CGI.unescapeHTML(node.inner_text).gsub(/\n\s*/, ' ').strip << '}'
+
+ if node.next_node.is_a?(::Hpricot::Text)
+ node = node.next_node
+ text << uninterp(node.to_s)
+ node.converted_to_haml = true
+ end
+
+ node = node.next_node
+ end
+ return parse_text_with_interpolation(text, tabs)
end
private
+ def erb_to_interpolation(text, options)
+ return text unless options[:erb]
+ text = CGI.escapeHTML(uninterp(text))
+ %w[<haml:loud> </haml:loud>].each {|str| text.gsub!(CGI.escapeHTML(str), str)}
+ ::Hpricot::XML(text).children.inject("") do |str, elem|
+ if elem.is_a?(::Hpricot::Text)
+ str + CGI.unescapeHTML(elem.to_s)
+ else # <haml:loud> element
+ str + '#{' + CGI.unescapeHTML(elem.innerText.strip) + '}'
+ end
+ end
+ end
+
def tabulate(tabs)
' ' * tabs
end
+ def uninterp(text)
+ text.gsub('#{', '\#{') #'
+ end
+
def attr_hash
attributes.to_hash
end
def parse_text(text, tabs)
+ parse_text_with_interpolation(uninterp(text), tabs)
+ end
+
+ def parse_text_with_interpolation(text, tabs)
text.strip!
- text.gsub!('#{', '\#{') #'
- if text.empty?
- String.new
- else
- lines = text.split("\n")
+ return "" if text.empty?
- lines.map do |line|
- line.strip!
- "#{tabulate(tabs)}#{'\\' if Haml::Engine::SPECIAL_CHARACTERS.include?(line[0])}#{line}\n"
- end.join
- end
+ text.split("\n").map do |line|
+ line.strip!
+ "#{tabulate(tabs)}#{'\\' if Haml::Engine::SPECIAL_CHARACTERS.include?(line[0])}#{line}\n"
+ end.join
end
end
end
end
@@ -60,21 +98,41 @@
end
end
require 'hpricot'
+# @private
+HAML_TAGS = %w[haml:block haml:loud haml:silent]
+
+Hpricot::ElementContent.keys.each do |k|
+ HAML_TAGS.each do |el|
+ val = Hpricot::ElementContent[k]
+ val[el.hash] = true if val.is_a?(Hash)
+ end
+end
+
+HAML_TAGS.each do |t|
+ Hpricot::ElementContent[t] = {}
+ Hpricot::ElementContent.keys.each do |key|
+ Hpricot::ElementContent[t][key.hash] = true
+ end
+end
+
module Haml
# Converts HTML documents into Haml templates.
# Depends on [Hpricot](http://github.com/whymirror/hpricot) for HTML parsing.
+ # If ERB conversion is being used, also depends on
+ # [Erubis](http://www.kuwata-lab.com/erubis) to parse the ERB
+ # and [ruby_parser](http://parsetree.rubyforge.org/) to parse the Ruby code.
#
# Example usage:
#
# Haml::HTML.new("<a href='http://google.com'>Blat</a>").render
# #=> "%a{:href => 'http://google.com'} Blat"
class HTML
# @param template [String, Hpricot::Node] The HTML template to convert
- # @option options :rhtml [Boolean] (false) Whether or not to parse
+ # @option options :erb [Boolean] (false) Whether or not to parse
# ERB's `<%= %>` and `<% %>` into Haml's `=` and `-`
# @option options :xhtml [Boolean] (false) Whether or not to parse
# the HTML strictly as XHTML
def initialize(template, options = {})
@options = options
@@ -84,13 +142,15 @@
else
if template.is_a? IO
template = template.read
end
- if @options[:rhtml]
- match_to_html(template, /<%=(.*?)-?%>/m, 'loud')
- match_to_html(template, /<%-?(.*?)-?%>/m, 'silent')
+ template = Haml::Util.check_encoding(template) {|msg, line| raise Haml::Error.new(msg, line)}
+
+ if @options[:erb]
+ require 'haml/html/erb'
+ template = ERB.compile(template)
end
method = @options[:xhtml] ? Hpricot.method(:XML) : method(:Hpricot)
@template = method.call(template.gsub('&', '&'))
end
@@ -101,11 +161,10 @@
def render
@template.to_haml(0, @options)
end
alias_method :to_haml, :render
- # @private
TEXT_REGEXP = /^(\s*).*$/
# @see Hpricot
# @private
class ::Hpricot::Doc
@@ -127,24 +186,24 @@
# @see Hpricot
# @private
class ::Hpricot::CData
# @see Haml::HTML::Node#to_haml
def to_haml(tabs, options)
- "#{tabulate(tabs)}:cdata\n#{parse_text(self.content, tabs + 1)}"
+ content = parse_text_with_interpolation(
+ erb_to_interpolation(self.content, options), tabs + 1)
+ "#{tabulate(tabs)}:cdata\n#{content}"
end
end
# @see Hpricot
# @private
class ::Hpricot::DocType
# @see Haml::HTML::Node#to_haml
def to_haml(tabs, options)
attrs = public_id.nil? ? ["", "", ""] :
public_id.scan(/DTD\s+([^\s]+)\s*([^\s]*)\s*([^\s]*)\s*\/\//)[0]
- if attrs == nil
- raise Exception.new("Invalid doctype")
- end
+ raise Haml::SyntaxError.new("Invalid doctype") if attrs == nil
type, version, strictness = attrs.map { |a| a.downcase }
if type == "html"
version = ""
strictness = "strict" if strictness == ""
@@ -168,45 +227,140 @@
# @see Hpricot
# @private
class ::Hpricot::Comment
# @see Haml::HTML::Node#to_haml
def to_haml(tabs, options)
- "#{tabulate(tabs)}/\n#{parse_text(self.content, tabs + 1)}"
+ content = self.content
+ if content =~ /\A(\[[^\]]+\])>(.*)<!\[endif\]\z/m
+ condition = $1
+ content = $2
+ end
+
+ if content.include?("\n")
+ "#{tabulate(tabs)}/#{condition}\n#{parse_text(content, tabs + 1)}"
+ else
+ "#{tabulate(tabs)}/#{condition} #{content.strip}\n"
+ end
end
end
# @see Hpricot
# @private
class ::Hpricot::Elem
# @see Haml::HTML::Node#to_haml
def to_haml(tabs, options)
- output = "#{tabulate(tabs)}"
- if options[:rhtml] && name[0...5] == 'haml:'
- return output + send("haml_tag_#{name[5..-1]}", CGI.unescapeHTML(self.inner_text))
+ return "" if converted_to_haml
+ if name == "script" &&
+ (attr_hash['type'].nil? || attr_hash['type'] == "text/javascript") &&
+ (attr_hash.keys - ['type']).empty?
+ return to_haml_filter(:javascript, tabs, options)
+ elsif name == "style" &&
+ (attr_hash['type'].nil? || attr_hash['type'] == "text/css") &&
+ (attr_hash.keys - ['type']).empty?
+ return to_haml_filter(:css, tabs, options)
end
- output += "%#{name}" unless name == 'div' &&
- (static_id?(options) || static_classname?(options))
+ output = tabulate(tabs)
+ if options[:erb] && name[0...5] == 'haml:'
+ case name[5..-1]
+ when "loud"
+ lines = CGI.unescapeHTML(inner_text).split("\n").
+ map {|s| s.rstrip}.reject {|s| s.strip.empty?}
+ lines.first.gsub!(/^[ \t]*/, "= ")
+ if lines.size > 1 # Multiline script block
+ # Normalize the indentation so that the last line is the base
+ indent_str = lines.last[/^[ \t]*/]
+ indent_re = /^[ \t]{0,#{indent_str.count(" ") + 8 * indent_str.count("\t")}}/
+ lines.map! {|s| s.gsub!(indent_re, '')}
+
+ # Add an extra " " to make it indented relative to "= "
+ lines[1..-1].each {|s| s.gsub!(/^/, " ")}
+
+ # Add | at the end, properly aligned
+ length = lines.map {|s| s.size}.max + 1
+ lines.map! {|s| "%#{-length}s|" % s}
+
+ if next_sibling && next_sibling.is_a?(Hpricot::Elem) && next_sibling.name == "haml:loud" &&
+ next_sibling.inner_text.split("\n").reject {|s| s.strip.empty?}.size > 1
+ lines << "-#"
+ end
+ end
+ return lines.map {|s| output + s + "\n"}.join
+ when "silent"
+ return CGI.unescapeHTML(inner_text).split("\n").map do |line|
+ next "" if line.strip.empty?
+ "#{output}- #{line.strip}\n"
+ end.join
+ when "block"
+ return render_children("", tabs, options)
+ end
+ end
+
+ if self.next && self.next.text? && self.next.content =~ /\A[^\s]/
+ if self.previous.nil? || self.previous.text? &&
+ (self.previous.content =~ /[^\s]\Z/ ||
+ self.previous.content =~ /\A\s*\Z/ && self.previous.previous.nil?)
+ nuke_outer_whitespace = true
+ else
+ output << "= succeed #{self.next.content.slice!(/\A[^\s]+/).dump} do\n"
+ tabs += 1
+ output << tabulate(tabs)
+ end
+ end
+
+ output << "%#{name}" unless name == 'div' &&
+ (static_id?(options) ||
+ static_classname?(options) &&
+ attr_hash['class'].split(' ').any?(&method(:haml_css_attr?)))
+
if attr_hash
if static_id?(options)
- output += "##{attr_hash['id']}"
+ output << "##{attr_hash['id']}"
remove_attribute('id')
end
if static_classname?(options)
- attr_hash['class'].split(' ').each { |c| output += ".#{c}" }
+ leftover = attr_hash['class'].split(' ').reject do |c|
+ next unless haml_css_attr?(c)
+ output << ".#{c}"
+ end
remove_attribute('class')
+ set_attribute('class', leftover.join(' ')) unless leftover.empty?
end
- output += haml_attributes(options) if attr_hash.length > 0
+ output << haml_attributes(options) if attr_hash.length > 0
end
- (self.children || []).inject(output + "\n") do |output, child|
- output + child.to_haml(tabs + 1, options)
+ output << ">" if nuke_outer_whitespace
+ output << "/" if empty? && !etag
+
+ if children && children.size == 1
+ child = children.first
+ if child.is_a?(::Hpricot::Text)
+ if !child.to_s.include?("\n")
+ text = child.to_haml(tabs + 1, options)
+ return output + " " + text.lstrip.gsub(/^\\/, '') unless text.chomp.include?("\n")
+ return output + "\n" + text
+ elsif ["pre", "textarea"].include?(name) ||
+ (name == "code" && parent.is_a?(::Hpricot::Elem) && parent.name == "pre")
+ return output + "\n#{tabulate(tabs + 1)}:preserve\n" +
+ innerText.gsub(/^/, tabulate(tabs + 2))
+ end
+ elsif child.is_a?(::Hpricot::Elem) && child.name == "haml:loud"
+ return output + child.to_haml(tabs + 1, options).lstrip
+ end
end
+
+ render_children(output + "\n", tabs, options)
end
private
+
+ def render_children(so_far, tabs, options)
+ (self.children || []).inject(so_far) do |output, child|
+ output + child.to_haml(tabs + 1, options)
+ end
+ end
def dynamic_attributes
@dynamic_attributes ||= begin
Haml::Util.map_hash(attr_hash) do |name, value|
next if value.empty?
@@ -219,50 +373,56 @@
[name, full_match ? ruby_value : %("#{ruby_value}")]
end
end
end
- def haml_tag_loud(text)
- "= #{text.gsub(/\n\s*/, ' ').strip}\n"
- end
+ def to_haml_filter(filter, tabs, options)
+ content =
+ if children.first.is_a?(::Hpricot::CData)
+ children.first.content
+ else
+ CGI.unescapeHTML(self.innerText)
+ end
+
+ content = erb_to_interpolation(content, options)
+ content.gsub!(/\A\s*\n(\s*)/, '\1')
+ original_indent = content[/\A(\s*)/, 1]
+ if content.split("\n").all? {|l| l.strip.empty? || l =~ /^#{original_indent}/}
+ content.gsub!(/^#{original_indent}/, tabulate(tabs + 1))
+ end
- def haml_tag_silent(text)
- text.split("\n").map { |line| "- #{line.strip}\n" }.join
+ "#{tabulate(tabs)}:#{filter}\n#{content}"
end
def static_attribute?(name, options)
- attr_hash[name] and !dynamic_attribute?(name, options)
+ attr_hash[name] && !dynamic_attribute?(name, options)
end
def dynamic_attribute?(name, options)
- options[:rhtml] and dynamic_attributes.key?(name)
+ options[:erb] and dynamic_attributes.key?(name)
end
def static_id?(options)
- static_attribute?('id', options)
+ static_attribute?('id', options) && haml_css_attr?(attr_hash['id'])
end
def static_classname?(options)
static_attribute?('class', options)
end
+ def haml_css_attr?(attr)
+ attr =~ /^[-:\w]+$/
+ end
+
# Returns a string representation of an attributes hash
# that's prettier than that produced by Hash#inspect
def haml_attributes(options)
- attrs = attr_hash.map do |name, value|
+ attrs = attr_hash.sort.map do |name, value|
value = dynamic_attribute?(name, options) ? dynamic_attributes[name] : value.inspect
name = name.index(/\W/) ? name.inspect : ":#{name}"
"#{name} => #{value}"
end
- "{ #{attrs.join(', ')} }"
- end
- end
-
- private
-
- def match_to_html(string, regex, tag)
- string.gsub!(regex) do
- "<haml:#{tag}>#{CGI.escapeHTML($1)}</haml:#{tag}>"
+ "{#{attrs.join(', ')}}"
end
end
end
end