drnic-haml 2.3.0
Sign up to get free protection for your applications and to get access to all the features.
- data/.yardopts +5 -0
- data/CONTRIBUTING +4 -0
- data/MIT-LICENSE +20 -0
- data/README.md +347 -0
- data/REVISION +1 -0
- data/Rakefile +371 -0
- data/VERSION +1 -0
- data/VERSION_NAME +1 -0
- data/bin/css2sass +7 -0
- data/bin/haml +9 -0
- data/bin/html2haml +7 -0
- data/bin/sass +8 -0
- data/extra/haml-mode.el +663 -0
- data/extra/sass-mode.el +205 -0
- data/extra/update_watch.rb +13 -0
- data/init.rb +8 -0
- data/lib/haml.rb +40 -0
- data/lib/haml/buffer.rb +307 -0
- data/lib/haml/engine.rb +301 -0
- data/lib/haml/error.rb +22 -0
- data/lib/haml/exec.rb +470 -0
- data/lib/haml/filters.rb +341 -0
- data/lib/haml/helpers.rb +560 -0
- data/lib/haml/helpers/action_view_extensions.rb +40 -0
- data/lib/haml/helpers/action_view_mods.rb +176 -0
- data/lib/haml/herb.rb +96 -0
- data/lib/haml/html.rb +308 -0
- data/lib/haml/precompiler.rb +997 -0
- data/lib/haml/shared.rb +78 -0
- data/lib/haml/template.rb +51 -0
- data/lib/haml/template/patch.rb +58 -0
- data/lib/haml/template/plugin.rb +71 -0
- data/lib/haml/util.rb +244 -0
- data/lib/haml/version.rb +64 -0
- data/lib/sass.rb +24 -0
- data/lib/sass/css.rb +423 -0
- data/lib/sass/engine.rb +491 -0
- data/lib/sass/environment.rb +79 -0
- data/lib/sass/error.rb +162 -0
- data/lib/sass/files.rb +133 -0
- data/lib/sass/plugin.rb +170 -0
- data/lib/sass/plugin/merb.rb +57 -0
- data/lib/sass/plugin/rails.rb +23 -0
- data/lib/sass/repl.rb +58 -0
- data/lib/sass/script.rb +55 -0
- data/lib/sass/script/bool.rb +17 -0
- data/lib/sass/script/color.rb +183 -0
- data/lib/sass/script/funcall.rb +50 -0
- data/lib/sass/script/functions.rb +199 -0
- data/lib/sass/script/lexer.rb +191 -0
- data/lib/sass/script/literal.rb +177 -0
- data/lib/sass/script/node.rb +14 -0
- data/lib/sass/script/number.rb +381 -0
- data/lib/sass/script/operation.rb +45 -0
- data/lib/sass/script/parser.rb +222 -0
- data/lib/sass/script/string.rb +12 -0
- data/lib/sass/script/unary_operation.rb +34 -0
- data/lib/sass/script/variable.rb +31 -0
- data/lib/sass/tree/comment_node.rb +84 -0
- data/lib/sass/tree/debug_node.rb +30 -0
- data/lib/sass/tree/directive_node.rb +70 -0
- data/lib/sass/tree/for_node.rb +48 -0
- data/lib/sass/tree/if_node.rb +54 -0
- data/lib/sass/tree/import_node.rb +69 -0
- data/lib/sass/tree/mixin_def_node.rb +29 -0
- data/lib/sass/tree/mixin_node.rb +48 -0
- data/lib/sass/tree/node.rb +252 -0
- data/lib/sass/tree/prop_node.rb +106 -0
- data/lib/sass/tree/root_node.rb +56 -0
- data/lib/sass/tree/rule_node.rb +220 -0
- data/lib/sass/tree/variable_node.rb +34 -0
- data/lib/sass/tree/while_node.rb +31 -0
- data/rails/init.rb +1 -0
- data/test/benchmark.rb +99 -0
- data/test/haml/engine_test.rb +1129 -0
- data/test/haml/helper_test.rb +282 -0
- data/test/haml/html2haml_test.rb +258 -0
- data/test/haml/markaby/standard.mab +52 -0
- data/test/haml/mocks/article.rb +6 -0
- data/test/haml/results/content_for_layout.xhtml +12 -0
- data/test/haml/results/eval_suppressed.xhtml +9 -0
- data/test/haml/results/filters.xhtml +62 -0
- data/test/haml/results/helpers.xhtml +93 -0
- data/test/haml/results/helpful.xhtml +10 -0
- data/test/haml/results/just_stuff.xhtml +68 -0
- data/test/haml/results/list.xhtml +12 -0
- data/test/haml/results/nuke_inner_whitespace.xhtml +40 -0
- data/test/haml/results/nuke_outer_whitespace.xhtml +148 -0
- data/test/haml/results/original_engine.xhtml +20 -0
- data/test/haml/results/partial_layout.xhtml +5 -0
- data/test/haml/results/partials.xhtml +21 -0
- data/test/haml/results/render_layout.xhtml +3 -0
- data/test/haml/results/silent_script.xhtml +74 -0
- data/test/haml/results/standard.xhtml +162 -0
- data/test/haml/results/tag_parsing.xhtml +23 -0
- data/test/haml/results/very_basic.xhtml +5 -0
- data/test/haml/results/whitespace_handling.xhtml +89 -0
- data/test/haml/rhtml/_av_partial_1.rhtml +12 -0
- data/test/haml/rhtml/_av_partial_2.rhtml +8 -0
- data/test/haml/rhtml/action_view.rhtml +62 -0
- data/test/haml/rhtml/standard.rhtml +54 -0
- data/test/haml/spec_test.rb +44 -0
- data/test/haml/template_test.rb +217 -0
- data/test/haml/templates/_av_partial_1.haml +9 -0
- data/test/haml/templates/_av_partial_1_ugly.haml +9 -0
- data/test/haml/templates/_av_partial_2.haml +5 -0
- data/test/haml/templates/_av_partial_2_ugly.haml +5 -0
- data/test/haml/templates/_layout.erb +3 -0
- data/test/haml/templates/_layout_for_partial.haml +3 -0
- data/test/haml/templates/_partial.haml +8 -0
- data/test/haml/templates/_text_area.haml +3 -0
- data/test/haml/templates/action_view.haml +47 -0
- data/test/haml/templates/action_view_ugly.haml +47 -0
- data/test/haml/templates/breakage.haml +8 -0
- data/test/haml/templates/content_for_layout.haml +8 -0
- data/test/haml/templates/eval_suppressed.haml +11 -0
- data/test/haml/templates/filters.haml +66 -0
- data/test/haml/templates/helpers.haml +95 -0
- data/test/haml/templates/helpful.haml +11 -0
- data/test/haml/templates/just_stuff.haml +83 -0
- data/test/haml/templates/list.haml +12 -0
- data/test/haml/templates/nuke_inner_whitespace.haml +32 -0
- data/test/haml/templates/nuke_outer_whitespace.haml +144 -0
- data/test/haml/templates/original_engine.haml +17 -0
- data/test/haml/templates/partial_layout.haml +3 -0
- data/test/haml/templates/partialize.haml +1 -0
- data/test/haml/templates/partials.haml +12 -0
- data/test/haml/templates/render_layout.haml +2 -0
- data/test/haml/templates/silent_script.haml +40 -0
- data/test/haml/templates/standard.haml +42 -0
- data/test/haml/templates/standard_ugly.haml +42 -0
- data/test/haml/templates/tag_parsing.haml +21 -0
- data/test/haml/templates/very_basic.haml +4 -0
- data/test/haml/templates/whitespace_handling.haml +87 -0
- data/test/haml/util_test.rb +92 -0
- data/test/linked_rails.rb +12 -0
- data/test/sass/css2sass_test.rb +294 -0
- data/test/sass/engine_test.rb +956 -0
- data/test/sass/functions_test.rb +126 -0
- data/test/sass/more_results/more1.css +9 -0
- data/test/sass/more_results/more1_with_line_comments.css +26 -0
- data/test/sass/more_results/more_import.css +29 -0
- data/test/sass/more_templates/_more_partial.sass +2 -0
- data/test/sass/more_templates/more1.sass +23 -0
- data/test/sass/more_templates/more_import.sass +11 -0
- data/test/sass/plugin_test.rb +229 -0
- data/test/sass/results/alt.css +4 -0
- data/test/sass/results/basic.css +9 -0
- data/test/sass/results/compact.css +5 -0
- data/test/sass/results/complex.css +87 -0
- data/test/sass/results/compressed.css +1 -0
- data/test/sass/results/expanded.css +19 -0
- data/test/sass/results/import.css +29 -0
- data/test/sass/results/line_numbers.css +49 -0
- data/test/sass/results/mixins.css +95 -0
- data/test/sass/results/multiline.css +24 -0
- data/test/sass/results/nested.css +22 -0
- data/test/sass/results/parent_ref.css +13 -0
- data/test/sass/results/script.css +16 -0
- data/test/sass/results/subdir/nested_subdir/nested_subdir.css +1 -0
- data/test/sass/results/subdir/subdir.css +3 -0
- data/test/sass/results/units.css +11 -0
- data/test/sass/script_test.rb +261 -0
- data/test/sass/templates/_partial.sass +2 -0
- data/test/sass/templates/alt.sass +16 -0
- data/test/sass/templates/basic.sass +23 -0
- data/test/sass/templates/bork1.sass +2 -0
- data/test/sass/templates/bork2.sass +2 -0
- data/test/sass/templates/bork3.sass +2 -0
- data/test/sass/templates/compact.sass +17 -0
- data/test/sass/templates/complex.sass +307 -0
- data/test/sass/templates/compressed.sass +15 -0
- data/test/sass/templates/expanded.sass +17 -0
- data/test/sass/templates/import.sass +11 -0
- data/test/sass/templates/importee.sass +19 -0
- data/test/sass/templates/line_numbers.sass +13 -0
- data/test/sass/templates/mixins.sass +76 -0
- data/test/sass/templates/multiline.sass +20 -0
- data/test/sass/templates/nested.sass +25 -0
- data/test/sass/templates/nested_bork1.sass +2 -0
- data/test/sass/templates/nested_bork2.sass +2 -0
- data/test/sass/templates/nested_bork3.sass +2 -0
- data/test/sass/templates/parent_ref.sass +25 -0
- data/test/sass/templates/script.sass +101 -0
- data/test/sass/templates/subdir/nested_subdir/_nested_partial.sass +2 -0
- data/test/sass/templates/subdir/nested_subdir/nested_subdir.sass +3 -0
- data/test/sass/templates/subdir/subdir.sass +6 -0
- data/test/sass/templates/units.sass +11 -0
- data/test/test_helper.rb +44 -0
- metadata +298 -0
data/lib/haml/shared.rb
ADDED
@@ -0,0 +1,78 @@
|
|
1
|
+
require 'strscan'
|
2
|
+
|
3
|
+
module Haml
|
4
|
+
# This module contains functionality that's shared between Haml and Sass.
|
5
|
+
module Shared
|
6
|
+
extend self
|
7
|
+
|
8
|
+
# Scans through a string looking for the interoplation-opening `#{`
|
9
|
+
# and, when it's found, yields the scanner to the calling code
|
10
|
+
# so it can handle it properly.
|
11
|
+
#
|
12
|
+
# The scanner will have any backslashes immediately in front of the `#{`
|
13
|
+
# as the second capture group (`scan[2]`),
|
14
|
+
# and the text prior to that as the first (`scan[1]`).
|
15
|
+
#
|
16
|
+
# @yieldparam scan [StringScanner] The scanner scanning through the string
|
17
|
+
# @return [String] The text remaining in the scanner after all `#{`s have been processed
|
18
|
+
def handle_interpolation(str)
|
19
|
+
scan = StringScanner.new(str)
|
20
|
+
yield scan while scan.scan(/(.*?)(\\*)\#\{/)
|
21
|
+
scan.rest
|
22
|
+
end
|
23
|
+
|
24
|
+
# Moves a scanner through a balanced pair of characters.
|
25
|
+
# For example:
|
26
|
+
#
|
27
|
+
# Foo (Bar (Baz bang) bop) (Bang (bop bip))
|
28
|
+
# ^ ^
|
29
|
+
# from to
|
30
|
+
#
|
31
|
+
# @param scanner [StringScanner] The string scanner to move
|
32
|
+
# @param start [Character] The character opening the balanced pair.
|
33
|
+
# A `Fixnum` in 1.8, a `String` in 1.9
|
34
|
+
# @param finish [Character] The character closing the balanced pair.
|
35
|
+
# A `Fixnum` in 1.8, a `String` in 1.9
|
36
|
+
# @param count [Fixnum] The number of opening characters matched
|
37
|
+
# before calling this method
|
38
|
+
# @return [(String, String)] The string matched within the balanced pair
|
39
|
+
# and the rest of the string.
|
40
|
+
# `["Foo (Bar (Baz bang) bop)", " (Bang (bop bip))"]` in the example above.
|
41
|
+
def balance(scanner, start, finish, count = 0)
|
42
|
+
str = ''
|
43
|
+
scanner = StringScanner.new(scanner) unless scanner.is_a? StringScanner
|
44
|
+
regexp = Regexp.new("(.*?)[\\#{start.chr}\\#{finish.chr}]", Regexp::MULTILINE)
|
45
|
+
while scanner.scan(regexp)
|
46
|
+
str << scanner.matched
|
47
|
+
count += 1 if scanner.matched[-1] == start
|
48
|
+
count -= 1 if scanner.matched[-1] == finish
|
49
|
+
return [str.strip, scanner.rest] if count == 0
|
50
|
+
end
|
51
|
+
end
|
52
|
+
|
53
|
+
# Formats a string for use in error messages about indentation.
|
54
|
+
#
|
55
|
+
# @param indentation [String] The string used for indentation
|
56
|
+
# @param was [Boolean] Whether or not to add `"was"` or `"were"`
|
57
|
+
# (depending on how many characters were in `indentation`)
|
58
|
+
# @return [String] The name of the indentation (e.g. `"12 spaces"`, `"1 tab"`)
|
59
|
+
def human_indentation(indentation, was = false)
|
60
|
+
if !indentation.include?(?\t)
|
61
|
+
noun = 'space'
|
62
|
+
elsif !indentation.include?(?\s)
|
63
|
+
noun = 'tab'
|
64
|
+
else
|
65
|
+
return indentation.inspect + (was ? ' was' : '')
|
66
|
+
end
|
67
|
+
|
68
|
+
singular = indentation.length == 1
|
69
|
+
if was
|
70
|
+
was = singular ? ' was' : ' were'
|
71
|
+
else
|
72
|
+
was = ''
|
73
|
+
end
|
74
|
+
|
75
|
+
"#{indentation.length} #{noun}#{'s' unless singular}#{was}"
|
76
|
+
end
|
77
|
+
end
|
78
|
+
end
|
@@ -0,0 +1,51 @@
|
|
1
|
+
require 'haml/engine'
|
2
|
+
|
3
|
+
module Haml
|
4
|
+
# The class that keeps track of the global options for Haml within Rails.
|
5
|
+
module Template
|
6
|
+
extend self
|
7
|
+
|
8
|
+
@options = {}
|
9
|
+
# The options hash for Haml when used within Rails.
|
10
|
+
# See {file:HAML_REFERENCE.md#haml_options the Haml options documentation}.
|
11
|
+
#
|
12
|
+
# @return [Hash<Symbol, Object>]
|
13
|
+
attr_accessor :options
|
14
|
+
end
|
15
|
+
end
|
16
|
+
|
17
|
+
if defined?(RAILS_ENV) && RAILS_ENV == "production"
|
18
|
+
Haml::Template.options[:ugly] = true
|
19
|
+
end
|
20
|
+
|
21
|
+
# Decide how we want to load Haml into Rails.
|
22
|
+
# Patching was necessary for versions <= 2.0.1,
|
23
|
+
# but we can make it a normal handler for higher versions.
|
24
|
+
if defined?(ActionView::TemplateHandler)
|
25
|
+
require 'haml/template/plugin'
|
26
|
+
else
|
27
|
+
require 'haml/template/patch'
|
28
|
+
end
|
29
|
+
|
30
|
+
if defined?(RAILS_ROOT)
|
31
|
+
# Update init.rb to the current version
|
32
|
+
# if it's out of date.
|
33
|
+
#
|
34
|
+
# We can probably remove this as of v1.9,
|
35
|
+
# because the new init file is sufficiently flexible
|
36
|
+
# to not need updating.
|
37
|
+
rails_init_file = File.join(RAILS_ROOT, 'vendor', 'plugins', 'haml', 'init.rb')
|
38
|
+
haml_init_file = Haml::Util.scope('init.rb')
|
39
|
+
begin
|
40
|
+
if File.exists?(rails_init_file)
|
41
|
+
require 'fileutils'
|
42
|
+
FileUtils.cp(haml_init_file, rails_init_file) unless FileUtils.cmp(rails_init_file, haml_init_file)
|
43
|
+
end
|
44
|
+
rescue SystemCallError
|
45
|
+
warn <<END
|
46
|
+
HAML WARNING:
|
47
|
+
#{rails_init_file} is out of date and couldn't be automatically updated.
|
48
|
+
Please run `haml --rails #{File.expand_path(RAILS_ROOT)}' to update it.
|
49
|
+
END
|
50
|
+
end
|
51
|
+
end
|
@@ -0,0 +1,58 @@
|
|
1
|
+
# This file makes Haml work with Rails
|
2
|
+
# by monkeypatching the core template-compilation methods.
|
3
|
+
# This is necessary for versions <= 2.0.1 because the template handler API
|
4
|
+
# wasn't sufficiently powerful to deal with caching and so forth.
|
5
|
+
|
6
|
+
# This module refers to the ActionView module that's part of Ruby on Rails.
|
7
|
+
# Haml can be used as an alternate templating engine for it,
|
8
|
+
# and includes several modifications to make it more Haml-friendly.
|
9
|
+
# The documentation can be found
|
10
|
+
# here[http://rubyonrails.org/api/classes/ActionView/Base.html].
|
11
|
+
module ActionView
|
12
|
+
class Base
|
13
|
+
def delegate_template_exists_with_haml(template_path)
|
14
|
+
template_exists?(template_path, :haml) && [:haml]
|
15
|
+
end
|
16
|
+
alias_method :delegate_template_exists_without_haml, :delegate_template_exists?
|
17
|
+
alias_method :delegate_template_exists?, :delegate_template_exists_with_haml
|
18
|
+
|
19
|
+
def compile_template_with_haml(extension, template, file_name, local_assigns)
|
20
|
+
return compile_haml(template, file_name, local_assigns) if extension.to_s == "haml"
|
21
|
+
compile_template_without_haml(extension, template, file_name, local_assigns)
|
22
|
+
end
|
23
|
+
alias_method :compile_template_without_haml, :compile_template
|
24
|
+
alias_method :compile_template, :compile_template_with_haml
|
25
|
+
|
26
|
+
def compile_haml(template, file_name, local_assigns)
|
27
|
+
render_symbol = assign_method_name(:haml, template, file_name)
|
28
|
+
locals = local_assigns.keys
|
29
|
+
|
30
|
+
@@template_args[render_symbol] ||= {}
|
31
|
+
locals_keys = @@template_args[render_symbol].keys | locals
|
32
|
+
@@template_args[render_symbol] = Haml::Util.to_hash(locals_keys.map {|k| [k, true]})
|
33
|
+
|
34
|
+
options = Haml::Template.options.dup
|
35
|
+
options[:filename] = file_name || 'compiled-template'
|
36
|
+
|
37
|
+
begin
|
38
|
+
Haml::Engine.new(template, options).def_method(CompiledTemplates, render_symbol, *locals_keys)
|
39
|
+
rescue Exception => e
|
40
|
+
if logger
|
41
|
+
logger.debug "ERROR: compiling #{render_symbol} RAISED #{e}"
|
42
|
+
logger.debug "Backtrace: #{e.backtrace.join("\n")}"
|
43
|
+
end
|
44
|
+
|
45
|
+
base_path = if defined?(extract_base_path_from)
|
46
|
+
# Rails 2.0.x
|
47
|
+
extract_base_path_from(file_name) || view_paths.first
|
48
|
+
else
|
49
|
+
# Rails <=1.2.6
|
50
|
+
@base_path
|
51
|
+
end
|
52
|
+
raise ActionView::TemplateError.new(base_path, file_name || template, @assigns, template, e)
|
53
|
+
end
|
54
|
+
|
55
|
+
@@compile_time[render_symbol] = Time.now
|
56
|
+
end
|
57
|
+
end
|
58
|
+
end
|
@@ -0,0 +1,71 @@
|
|
1
|
+
# This file makes Haml work with Rails
|
2
|
+
# using the > 2.0.1 template handler API.
|
3
|
+
|
4
|
+
module Haml
|
5
|
+
class Plugin < ActionView::TemplateHandler
|
6
|
+
include ActionView::TemplateHandlers::Compilable if defined?(ActionView::TemplateHandlers::Compilable)
|
7
|
+
|
8
|
+
def compile(template)
|
9
|
+
options = Haml::Template.options.dup
|
10
|
+
|
11
|
+
# template is a template object in Rails >=2.1.0,
|
12
|
+
# a source string previously
|
13
|
+
if template.respond_to? :source
|
14
|
+
# Template has a generic identifier in Rails >=3.0.0
|
15
|
+
options[:filename] = template.respond_to?(:identifier) ? template.identifier : template.filename
|
16
|
+
source = template.source
|
17
|
+
else
|
18
|
+
source = template
|
19
|
+
end
|
20
|
+
|
21
|
+
Haml::Engine.new(source, options).send(:precompiled_with_ambles, [])
|
22
|
+
end
|
23
|
+
|
24
|
+
def cache_fragment(block, name = {}, options = nil)
|
25
|
+
@view.fragment_for(block, name, options) do
|
26
|
+
eval("_hamlout.buffer", block.binding)
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
30
|
+
end
|
31
|
+
|
32
|
+
if defined? ActionView::Template and ActionView::Template.respond_to? :register_template_handler
|
33
|
+
ActionView::Template
|
34
|
+
else
|
35
|
+
ActionView::Base
|
36
|
+
end.register_template_handler(:haml, Haml::Plugin)
|
37
|
+
|
38
|
+
# In Rails 2.0.2, ActionView::TemplateError took arguments
|
39
|
+
# that we can't fill in from the Haml::Plugin context.
|
40
|
+
# Thus, we've got to monkeypatch ActionView::Base to catch the error.
|
41
|
+
if ActionView::TemplateError.instance_method(:initialize).arity == 5
|
42
|
+
class ActionView::Base
|
43
|
+
def compile_template(handler, template, file_name, local_assigns)
|
44
|
+
render_symbol = assign_method_name(handler, template, file_name)
|
45
|
+
|
46
|
+
# Move begin up two lines so it captures compilation exceptions.
|
47
|
+
begin
|
48
|
+
render_source = create_template_source(handler, template, render_symbol, local_assigns.keys)
|
49
|
+
line_offset = @@template_args[render_symbol].size + handler.line_offset
|
50
|
+
|
51
|
+
file_name = 'compiled-template' if file_name.blank?
|
52
|
+
CompiledTemplates.module_eval(render_source, file_name, -line_offset)
|
53
|
+
rescue Exception => e # errors from template code
|
54
|
+
if logger
|
55
|
+
logger.debug "ERROR: compiling #{render_symbol} RAISED #{e}"
|
56
|
+
logger.debug "Function body: #{render_source}"
|
57
|
+
logger.debug "Backtrace: #{e.backtrace.join("\n")}"
|
58
|
+
end
|
59
|
+
|
60
|
+
# There's no way to tell Haml about the filename,
|
61
|
+
# so we've got to insert it ourselves.
|
62
|
+
e.backtrace[0].gsub!('(haml)', file_name) if e.is_a?(Haml::Error)
|
63
|
+
|
64
|
+
raise ActionView::TemplateError.new(extract_base_path_from(file_name) || view_paths.first, file_name || template, @assigns, template, e)
|
65
|
+
end
|
66
|
+
|
67
|
+
@@compile_time[render_symbol] = Time.now
|
68
|
+
# logger.debug "Compiled template #{file_name || template}\n ==> #{render_symbol}" if logger
|
69
|
+
end
|
70
|
+
end
|
71
|
+
end
|
data/lib/haml/util.rb
ADDED
@@ -0,0 +1,244 @@
|
|
1
|
+
require 'erb'
|
2
|
+
require 'set'
|
3
|
+
require 'enumerator'
|
4
|
+
|
5
|
+
module Haml
|
6
|
+
# A module containing various useful functions.
|
7
|
+
module Util
|
8
|
+
extend self
|
9
|
+
|
10
|
+
# An array of ints representing the Ruby version number.
|
11
|
+
RUBY_VERSION = ::RUBY_VERSION.split(".").map {|s| s.to_i}
|
12
|
+
|
13
|
+
# Returns the path of a file relative to the Haml root directory.
|
14
|
+
#
|
15
|
+
# @param file [String] The filename relative to the Haml root
|
16
|
+
# @return [String] The filename relative to the the working directory
|
17
|
+
def scope(file)
|
18
|
+
File.join(File.dirname(File.dirname(File.dirname(File.expand_path(__FILE__)))), file)
|
19
|
+
end
|
20
|
+
|
21
|
+
# Converts an array of `[key, value]` pairs to a hash.
|
22
|
+
# For example:
|
23
|
+
#
|
24
|
+
# to_hash([[:foo, "bar"], [:baz, "bang"]])
|
25
|
+
# #=> {:foo => "bar", :baz => "bang"}
|
26
|
+
#
|
27
|
+
# @param arr [Array<(Object, Object)>] An array of pairs
|
28
|
+
# @return [Hash] A hash
|
29
|
+
def to_hash(arr)
|
30
|
+
arr.compact.inject({}) {|h, (k, v)| h[k] = v; h}
|
31
|
+
end
|
32
|
+
|
33
|
+
# Maps the keys in a hash according to a block.
|
34
|
+
# For example:
|
35
|
+
#
|
36
|
+
# map_keys({:foo => "bar", :baz => "bang"}) {|k| k.to_s}
|
37
|
+
# #=> {"foo" => "bar", "baz" => "bang"}
|
38
|
+
#
|
39
|
+
# @param hash [Hash] The hash to map
|
40
|
+
# @yield [key] A block in which the keys are transformed
|
41
|
+
# @yieldparam key [Object] The key that should be mapped
|
42
|
+
# @yieldreturn [Object] The new value for the key
|
43
|
+
# @return [Hash] The mapped hash
|
44
|
+
# @see #map_vals
|
45
|
+
# @see #map_hash
|
46
|
+
def map_keys(hash)
|
47
|
+
to_hash(hash.map {|k, v| [yield(k), v]})
|
48
|
+
end
|
49
|
+
|
50
|
+
# Maps the values in a hash according to a block.
|
51
|
+
# For example:
|
52
|
+
#
|
53
|
+
# map_values({:foo => "bar", :baz => "bang"}) {|v| v.to_sym}
|
54
|
+
# #=> {:foo => :bar, :baz => :bang}
|
55
|
+
#
|
56
|
+
# @param hash [Hash] The hash to map
|
57
|
+
# @yield [value] A block in which the values are transformed
|
58
|
+
# @yieldparam value [Object] The value that should be mapped
|
59
|
+
# @yieldreturn [Object] The new value for the value
|
60
|
+
# @return [Hash] The mapped hash
|
61
|
+
# @see #map_keys
|
62
|
+
# @see #map_hash
|
63
|
+
def map_vals(hash)
|
64
|
+
to_hash(hash.map {|k, v| [k, yield(v)]})
|
65
|
+
end
|
66
|
+
|
67
|
+
# Maps the key-value pairs of a hash according to a block.
|
68
|
+
# For example:
|
69
|
+
#
|
70
|
+
# map_hash({:foo => "bar", :baz => "bang"}) {|k, v| [k.to_s, v.to_sym]}
|
71
|
+
# #=> {"foo" => :bar, "baz" => :bang}
|
72
|
+
#
|
73
|
+
# @param hash [Hash] The hash to map
|
74
|
+
# @yield [key, value] A block in which the key-value pairs are transformed
|
75
|
+
# @yieldparam [key] The hash key
|
76
|
+
# @yieldparam [value] The hash value
|
77
|
+
# @yieldreturn [(Object, Object)] The new value for the `[key, value]` pair
|
78
|
+
# @return [Hash] The mapped hash
|
79
|
+
# @see #map_keys
|
80
|
+
# @see #map_vals
|
81
|
+
def map_hash(hash, &block)
|
82
|
+
to_hash(hash.map(&block))
|
83
|
+
end
|
84
|
+
|
85
|
+
# Computes the powerset of the given array.
|
86
|
+
# This is the set of all subsets of the array.
|
87
|
+
# For example:
|
88
|
+
#
|
89
|
+
# powerset([1, 2, 3]) #=>
|
90
|
+
# Set[Set[], Set[1], Set[2], Set[3], Set[1, 2], Set[2, 3], Set[1, 3], Set[1, 2, 3]]
|
91
|
+
#
|
92
|
+
# @param arr [Enumerable]
|
93
|
+
# @return [Set<Set>] The subsets of `arr`
|
94
|
+
def powerset(arr)
|
95
|
+
arr.inject([Set.new].to_set) do |powerset, el|
|
96
|
+
new_powerset = Set.new
|
97
|
+
powerset.each do |subset|
|
98
|
+
new_powerset << subset
|
99
|
+
new_powerset << subset + [el]
|
100
|
+
end
|
101
|
+
new_powerset
|
102
|
+
end
|
103
|
+
end
|
104
|
+
|
105
|
+
# Concatenates all strings that are adjacent in an array,
|
106
|
+
# while leaving other elements as they are.
|
107
|
+
# For example:
|
108
|
+
#
|
109
|
+
# merge_adjacent_strings([1, "foo", "bar", 2, "baz"])
|
110
|
+
# #=> [1, "foobar", 2, "baz"]
|
111
|
+
#
|
112
|
+
# @param enum [Enumerable]
|
113
|
+
# @return [Array] The enumerable with strings merged
|
114
|
+
def merge_adjacent_strings(enum)
|
115
|
+
e = enum.inject([]) do |a, e|
|
116
|
+
if e.is_a?(String) && a.last.is_a?(String)
|
117
|
+
a.last << e
|
118
|
+
else
|
119
|
+
a << e
|
120
|
+
end
|
121
|
+
a
|
122
|
+
end
|
123
|
+
end
|
124
|
+
|
125
|
+
# Whether or not this is running under Ruby 1.8 or lower.
|
126
|
+
#
|
127
|
+
# @return [Boolean]
|
128
|
+
def ruby1_8?
|
129
|
+
Haml::Util::RUBY_VERSION[0] == 1 && Haml::Util::RUBY_VERSION[1] < 9
|
130
|
+
end
|
131
|
+
|
132
|
+
def check_encoding(str)
|
133
|
+
return if ruby1_8?
|
134
|
+
return if str.valid_encoding?
|
135
|
+
encoding = str.encoding
|
136
|
+
newlines = Regexp.new("\r\n|\r|\n".encode(encoding).force_encoding("binary"))
|
137
|
+
str.force_encoding("binary").split(newlines).each_with_index do |line, i|
|
138
|
+
begin
|
139
|
+
line.encode(encoding)
|
140
|
+
rescue Encoding::UndefinedConversionError => e
|
141
|
+
yield <<MSG.rstrip, i + 1
|
142
|
+
Invalid #{encoding.name} character #{e.error_char.dump}
|
143
|
+
MSG
|
144
|
+
end
|
145
|
+
end
|
146
|
+
end
|
147
|
+
|
148
|
+
# Checks to see if a class has a given method.
|
149
|
+
# For example:
|
150
|
+
#
|
151
|
+
# Haml::Util.has?(:public_instance_method, String, :gsub) #=> true
|
152
|
+
#
|
153
|
+
# Method collections like `Class#instance_methods`
|
154
|
+
# return strings in Ruby 1.8 and symbols in Ruby 1.9 and on,
|
155
|
+
# so this handles checking for them in a compatible way.
|
156
|
+
#
|
157
|
+
# @param attr [#to_s] The (singular) name of the method-collection method
|
158
|
+
# (e.g. `:instance_methods`, `:private_methods`)
|
159
|
+
# @param klass [Module] The class to check the methods of which to check
|
160
|
+
# @param method [String, Symbol] The name of the method do check for
|
161
|
+
# @return [Boolean] Whether or not the given collection has the given method
|
162
|
+
def has?(attr, klass, method)
|
163
|
+
klass.send("#{attr}s").include?(ruby1_8? ? method.to_s : method.to_sym)
|
164
|
+
end
|
165
|
+
|
166
|
+
# A version of `Enumerable#enum_with_index` that works in Ruby 1.8 and 1.9.
|
167
|
+
#
|
168
|
+
# @param enum [Enumerable] The enumerable to get the enumerator for
|
169
|
+
# @return [Enumerator] The with-index enumerator
|
170
|
+
def enum_with_index(enum)
|
171
|
+
ruby1_8? ? enum.enum_with_index : enum.each_with_index
|
172
|
+
end
|
173
|
+
|
174
|
+
# The context in which the ERB for \{#def\_static\_method} will be run.
|
175
|
+
class StaticConditionalContext
|
176
|
+
# @param set [#include?] The set of variables that are defined for this context.
|
177
|
+
def initialize(set)
|
178
|
+
@set = set
|
179
|
+
end
|
180
|
+
|
181
|
+
# Checks whether or not a variable is defined for this context.
|
182
|
+
#
|
183
|
+
# @param name [Symbol] The name of the variable
|
184
|
+
# @return [Boolean]
|
185
|
+
def method_missing(name, *args, &block)
|
186
|
+
super unless args.empty? && block.nil?
|
187
|
+
@set.include?(name)
|
188
|
+
end
|
189
|
+
end
|
190
|
+
|
191
|
+
# This is used for methods in {Haml::Buffer} that need to be very fast,
|
192
|
+
# and take a lot of boolean parameters
|
193
|
+
# that are known at compile-time.
|
194
|
+
# Instead of passing the parameters in normally,
|
195
|
+
# a separate method is defined for every possible combination of those parameters;
|
196
|
+
# these are then called using \{#static\_method\_name}.
|
197
|
+
#
|
198
|
+
# To define a static method, an ERB template for the method is provided.
|
199
|
+
# All conditionals based on the static parameters
|
200
|
+
# are done as embedded Ruby within this template.
|
201
|
+
# For example:
|
202
|
+
#
|
203
|
+
# def_static_method(Foo, :my_static_method, [:foo, :bar], :baz, :bang, <<RUBY)
|
204
|
+
# <% if baz && bang %>
|
205
|
+
# return foo + bar
|
206
|
+
# <% elsif baz || bang %>
|
207
|
+
# return foo - bar
|
208
|
+
# <% else %>
|
209
|
+
# return 17
|
210
|
+
# <% end %>
|
211
|
+
# RUBY
|
212
|
+
#
|
213
|
+
# \{#static\_method\_name} can be used to call static methods.
|
214
|
+
#
|
215
|
+
# @overload def_static_method(klass, name, args, *vars, erb)
|
216
|
+
# @param klass [Module] The class on which to define the static method
|
217
|
+
# @param name [#to_s] The (base) name of the static method
|
218
|
+
# @param args [Array<Symbol>] The names of the arguments to the defined methods
|
219
|
+
# (**not** to the ERB template)
|
220
|
+
# @param vars [Array<Symbol>] The names of the static boolean variables
|
221
|
+
# to be made available to the ERB template
|
222
|
+
# @param erb [String] The template for the method code
|
223
|
+
def def_static_method(klass, name, args, *vars)
|
224
|
+
erb = vars.pop
|
225
|
+
powerset(vars).each do |set|
|
226
|
+
context = StaticConditionalContext.new(set).instance_eval {binding}
|
227
|
+
klass.class_eval(<<METHOD)
|
228
|
+
def #{static_method_name(name, *vars.map {|v| set.include?(v)})}(#{args.join(', ')})
|
229
|
+
#{ERB.new(erb).result(context)}
|
230
|
+
end
|
231
|
+
METHOD
|
232
|
+
end
|
233
|
+
end
|
234
|
+
|
235
|
+
# Computes the name for a method defined via \{#def\_static\_method}.
|
236
|
+
#
|
237
|
+
# @param name [String] The base name of the static method
|
238
|
+
# @param vars [Array<Boolean>] The static variable assignment
|
239
|
+
# @return [String] The real name of the static method
|
240
|
+
def static_method_name(name, *vars)
|
241
|
+
"#{name}_#{vars.map {|v| !!v}.join('_')}"
|
242
|
+
end
|
243
|
+
end
|
244
|
+
end
|