Module: Hamlit::HamlHelpers
- Extended by:
- HamlHelpers
- Includes:
- XssMods
- Included in:
- HamlBuffer, HamlHelpers
- Defined in:
- lib/hamlit/rails_template.rb,
lib/hamlit/parser/haml_helpers.rb,
lib/hamlit/parser/haml_xss_mods.rb
Overview
This module contains various helpful methods to make it easier to do various tasks. Haml::Helpers is automatically included in the context that a Haml template is parsed in, so all these methods are at your disposal from within the template.
Defined Under Namespace
Modules: XssMods Classes: ErrorReturn
Constant Summary collapse
- HTML_ESCAPE =
Characters that need to be escaped to HTML entities from user input
{ '&' => '&', '<' => '<', '>' => '>', '"' => '"', "'" => ''' }
- HTML_ESCAPE_REGEX =
/[\"><&]/
- HTML_ESCAPE_ONCE_REGEX =
/[\"><]|&(?!(?:[a-zA-Z]+|#(?:\d+|[xX][0-9a-fA-F]+));)/
- @@action_view_defined =
false
Class Method Summary collapse
-
.action_view? ⇒ Boolean
Whether or not ActionView is loaded.
Instance Method Summary collapse
-
#block_is_haml?(block) ⇒ Boolean
Returns whether or not ‘block` is defined directly in a Haml template.
-
#capture_haml(*args) {|args| ... } ⇒ Object
Captures the result of a block of Haml code, gets rid of the excess indentation, and returns it as a string.
-
#escape_once(text) ⇒ String
Escapes HTML entities in ‘text`, but without escaping an ampersand that is already part of an escaped entity.
-
#find_and_preserve(input = nil, tags = haml_buffer.options[:preserve], &block) ⇒ Object
Uses #preserve to convert any newlines inside whitespace-sensitive tags into the HTML entities for endlines.
-
#haml_concat(text = "") ⇒ Object
Outputs text directly to the Haml buffer, with the proper indentation.
-
#haml_indent ⇒ String
The indentation string for the current line.
-
#haml_tag(name, *rest, &block) ⇒ Object
Creates an HTML tag with the given name and optionally text and attributes.
-
#haml_tag_if(condition, *tag) ⇒ Object
Conditionally wrap a block in an element.
-
#html_attrs(lang = 'en-US') ⇒ {#to_s => String}
Returns a hash containing default assignments for the ‘xmlns`, `lang`, and `xml:lang` attributes of the `html` HTML element.
-
#html_escape(text) ⇒ String
Returns a copy of ‘text` with ampersands, angle brackets and quotes escaped into HTML entities.
-
#init_haml_helpers ⇒ Object
Note: this does not need to be called when using Haml helpers normally in Rails.
-
#is_haml? ⇒ Boolean
Returns whether or not the current template is a Haml template.
-
#list_of(enum, opts = {}) {|item| ... } ⇒ Object
Takes an ‘Enumerable` object and a block and iterates over the enum, yielding each element to a Haml block and putting the result into `<li>` elements.
-
#non_haml { ... } ⇒ Object
Runs a block of code in a non-Haml context (i.e. #is_haml? will return false).
-
#precede(str) { ... } ⇒ Object
Prepends a string to the beginning of a Haml block, with no whitespace between.
-
#preserve(input = nil, &block) ⇒ Object
(also: #flatten)
Takes any string, finds all the newlines, and converts them to HTML entities so they’ll render correctly in whitespace-sensitive tags without screwing up the indentation.
-
#succeed(str) { ... } ⇒ Object
Appends a string to the end of a Haml block, with no whitespace between.
-
#surround(front, back = front) { ... } ⇒ Object
Surrounds a block of Haml code with strings, with no whitespace in between.
-
#tab_down(i = 1) ⇒ Object
Decrements the number of tabs the buffer automatically adds to the lines of the template.
-
#tab_up(i = 1) ⇒ Object
Increments the number of tabs the buffer automatically adds to the lines of the template.
-
#with_tabs(i) { ... } ⇒ Object
Sets the number of tabs the buffer automatically adds to the lines of the template, but only for the duration of the block.
Methods included from XssMods
#capture_haml_with_haml_xss, #escape_once_with_haml_xss, #find_and_preserve_with_haml_xss, #haml_concat_with_haml_xss, #haml_indent_with_haml_xss, #html_escape_with_haml_xss, included, #list_of_with_haml_xss, #precede_with_haml_xss, #preserve_with_haml_xss, #succeed_with_haml_xss, #surround_with_haml_xss
Class Method Details
.action_view? ⇒ Boolean
Returns Whether or not ActionView is loaded.
58 59 60 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 58 def self.action_view? @@action_view_defined end |
Instance Method Details
#block_is_haml?(block) ⇒ Boolean
Returns whether or not ‘block` is defined directly in a Haml template.
648 649 650 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 648 def block_is_haml?(block) eval('!!defined?(_hamlout)', block.binding) end |
#capture_haml(*args) {|args| ... } ⇒ Object
Captures the result of a block of Haml code, gets rid of the excess indentation, and returns it as a string. For example, after the following,
.foo
- foo = capture_haml(13) do |a|
%p= a
the local variable ‘foo` would be assigned to `“<p>13</p>n”`.
379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 379 def capture_haml(*args, &block) buffer = eval('if defined? _hamlout then _hamlout else nil end', block.binding) || haml_buffer with_haml_buffer(buffer) do position = haml_buffer.buffer.length haml_buffer.capture_position = position value = block.call(*args) captured = haml_buffer.buffer.slice!(position..-1) if captured == '' and value != haml_buffer.buffer captured = (value.is_a?(String) ? value : nil) end return nil if captured.nil? return (haml_buffer.[:ugly] ? captured : prettify(captured)) end ensure haml_buffer.capture_position = nil end |
#escape_once(text) ⇒ String
Escapes HTML entities in ‘text`, but without escaping an ampersand that is already part of an escaped entity.
628 629 630 631 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 628 def escape_once(text) text = text.to_s text.gsub(HTML_ESCAPE_ONCE_REGEX, HTML_ESCAPE) end |
#find_and_preserve(input, tags = haml_buffer.options[:preserve]) ⇒ Object #find_and_preserve(tags = haml_buffer.options[:preserve]) { ... } ⇒ Object
Uses #preserve to convert any newlines inside whitespace-sensitive tags into the HTML entities for endlines.
113 114 115 116 117 118 119 120 121 122 123 124 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 113 def find_and_preserve(input = nil, = haml_buffer.[:preserve], &block) return find_and_preserve(capture_haml(&block), input || ) if block = .each_with_object('') do |t, s| s << '|' unless s.empty? s << Regexp.escape(t) end re = /<(#{})([^>]*)>(.*?)(<\/\1>)/im input.to_s.gsub(re) do |s| s =~ re # Can't rely on $1, etc. existing since Rails' SafeBuffer#gsub is incompatible "<#{$1}#{$2}>#{preserve($3)}</#{$1}>" end end |
#haml_concat(text = "") ⇒ Object
Outputs text directly to the Haml buffer, with the proper indentation.
403 404 405 406 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 403 def haml_concat(text = "") haml_internal_concat text ErrorReturn.new("haml_concat") end |
#haml_indent ⇒ String
Returns The indentation string for the current line.
434 435 436 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 434 def haml_indent ' ' * haml_buffer.tabulation end |
#haml_tag(name, *rest, attributes = {}) { ... } ⇒ Object #haml_tag(name, text, *flags, attributes = {}) ⇒ Object
Creates an HTML tag with the given name and optionally text and attributes. Can take a block that will run between the opening and closing tags. If the block is a Haml block or outputs text using #haml_concat, the text will be properly indented.
‘name` can be a string using the standard Haml class/id shorthand (e.g. “span#foo.bar”, “#foo”). Just like standard Haml tags, these class and id values will be merged with manually-specified attributes.
‘flags` is a list of symbol flags like those that can be put at the end of a Haml tag (`:/`, `:<`, and `:>`). Currently, only `:/` and `:<` are supported.
‘haml_tag` outputs directly to the buffer; its return value should not be used. If you need to get the results as a string, use #capture_haml.
For example,
haml_tag :table do
haml_tag :tr do
haml_tag 'td.cell' do
haml_tag :strong, "strong!"
haml_concat "data"
end
haml_tag :td do
haml_concat "more_data"
end
end
end
outputs
<table>
<tr>
<td class='cell'>
<strong>
strong!
</strong>
data
</td>
<td>
more_data
</td>
</tr>
</table>
495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 495 def haml_tag(name, *rest, &block) ret = ErrorReturn.new("haml_tag") text = rest.shift.to_s unless [Symbol, Hash, NilClass].any? {|t| rest.first.is_a? t} flags = [] flags << rest.shift while rest.first.is_a? Symbol attrs = (rest.shift || {}) attrs.keys.each {|key| attrs[key.to_s] = attrs.delete(key)} unless attrs.empty? name, attrs = merge_name_and_attributes(name.to_s, attrs) attributes = ::Hamlit::HamlCompiler.build_attributes(haml_buffer.html?, haml_buffer.[:attr_wrapper], haml_buffer.[:escape_attrs], haml_buffer.[:hyphenate_data_attrs], attrs) if text.nil? && block.nil? && (haml_buffer.[:autoclose].include?(name) || flags.include?(:/)) haml_internal_concat_raw "<#{name}#{attributes}#{' /' if haml_buffer.[:format] == :xhtml}>" return ret end if flags.include?(:/) raise ::Hamlit::HamlError.new(::Hamlit::HamlError.(:self_closing_content)) if text raise ::Hamlit::HamlError.new(::Hamlit::HamlError.(:illegal_nesting_self_closing)) if block end tag = "<#{name}#{attributes}>" end_tag = "</#{name}>" if block.nil? text = text.to_s if text.include?("\n") haml_internal_concat_raw tag tab_up haml_internal_concat text tab_down haml_internal_concat_raw end_tag else haml_internal_concat_raw tag, false haml_internal_concat text, false, false haml_internal_concat_raw end_tag, true, false end return ret end if text raise ::Hamlit::HamlError.new(::Hamlit::HamlError.(:illegal_nesting_line, name)) end if flags.include?(:<) haml_internal_concat_raw tag, false haml_internal_concat "#{capture_haml(&block).strip}", false, false haml_internal_concat_raw end_tag, true, false return ret end haml_internal_concat_raw tag tab_up block.call tab_down haml_internal_concat_raw end_tag ret end |
#haml_tag_if(condition, *tag) ⇒ Object
Conditionally wrap a block in an element. If ‘condition` is `true` then this method renders the tag described by the arguments in `tag` (using #haml_tag) with the given block inside, otherwise it just renders the block.
For example,
- haml_tag_if important, '.important' do
%p
A (possibly) important paragraph.
will produce
<div class='important'>
<p>
A (possibly) important paragraph.
</p>
</div>
if ‘important` is truthy, and just
<p>
A (possibly) important paragraph.
</p>
otherwise.
Like #haml_tag, ‘haml_tag_if` outputs directly to the buffer and its return value should not be used. Use #capture_haml if you need to use its results as a string.
593 594 595 596 597 598 599 600 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 593 def haml_tag_if(condition, *tag) if condition haml_tag(*tag){ yield } else yield end ErrorReturn.new("haml_tag_if") end |
#html_attrs(lang = 'en-US') ⇒ {#to_s => String}
Returns a hash containing default assignments for the ‘xmlns`, `lang`, and `xml:lang` attributes of the `html` HTML element. For example,
%html{html_attrs}
becomes
<html xmlns='http://www.w3.org/1999/xhtml' xml:lang='en-US' lang='en-US'>
234 235 236 237 238 239 240 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 234 def html_attrs(lang = 'en-US') if haml_buffer.[:format] == :xhtml {:xmlns => "http://www.w3.org/1999/xhtml", 'xml:lang' => lang, :lang => lang} else {:lang => lang} end end |
#html_escape(text) ⇒ String
Returns a copy of ‘text` with ampersands, angle brackets and quotes escaped into HTML entities.
Note that if ActionView is loaded and XSS protection is enabled (as is the default for Rails 3.0+, and optional for version 2.3.5+), this won’t escape text declared as “safe”.
616 617 618 619 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 616 def html_escape(text) text = text.to_s text.gsub(HTML_ESCAPE_REGEX, HTML_ESCAPE) end |
#init_haml_helpers ⇒ Object
Note: this does not need to be called when using Haml helpers normally in Rails.
Initializes the current object as though it were in the same context as a normal ActionView instance using Haml. This is useful if you want to use the helpers in a context other than the normal setup with ActionView. For example:
context = Object.new
class << context
include Haml::Helpers
end
context.init_haml_helpers
context.haml_tag :p, "Stuff"
78 79 80 81 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 78 def init_haml_helpers @haml_buffer = ::Hamlit::HamlBuffer.new(haml_buffer, ::Hamlit::HamlOptions.new.for_buffer) nil end |
#is_haml? ⇒ Boolean
Returns whether or not the current template is a Haml template.
This function, unlike other Haml::Helpers functions, also works in other ‘ActionView` templates, where it will always return false.
640 641 642 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 640 def is_haml? !@haml_buffer.nil? && @haml_buffer.active? end |
#list_of(enum, opts = {}) {|item| ... } ⇒ Object
Takes an ‘Enumerable` object and a block and iterates over the enum, yielding each element to a Haml block and putting the result into `<li>` elements. This creates a list of the results of the block. For example:
= list_of([['hello'], ['yall']]) do |i|
= i[0]
Produces:
<li>hello</li>
<li>yall</li>
And:
= list_of({:title => 'All the stuff', :description => 'A book about all the stuff.'}) do |key, val|
%h3= key.humanize
%p= val
Produces:
<li>
<h3>Title</h3>
<p>All the stuff</p>
</li>
<li>
<h3>Description</h3>
<p>A book about all the stuff.</p>
</li>
While:
= list_of(["Home", "About", "Contact", "FAQ"], {class: "nav", role: "nav"}) do |item|
%a{ href="#" }= item
Produces:
<li class='nav' role='nav'>
<a href='#'>Home</a>
</li>
<li class='nav' role='nav'>
<a href='#'>About</a>
</li>
<li class='nav' role='nav'>
<a href='#'>Contact</a>
</li>
<li class='nav' role='nav'>
<a href='#'>FAQ</a>
</li>
`[[class", "nav"], [role", "nav"]]` could have been used instead of `{class: "nav", role: "nav"}` (or any enumerable collection where each pair of items responds to #to_s)
205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 205 def list_of(enum, opts={}, &block) opts_attributes = opts.each_with_object('') {|(k, v), s| s << " #{k}='#{v}'"} enum.each_with_object('') do |i, ret| result = capture_haml(i, &block) if result.count("\n") > 1 result.gsub!("\n", "\n ") result = "\n #{result.strip!}\n" else result.strip! end ret << "\n" unless ret.empty? ret << %Q!<li#{opts_attributes}>#{result}</li>! end end |
#non_haml { ... } ⇒ Object
Runs a block of code in a non-Haml context (i.e. #is_haml? will return false).
This is mainly useful for rendering sub-templates such as partials in a non-Haml language, particularly where helpers may behave differently when run from Haml.
Note that this is automatically applied to Rails partials.
92 93 94 95 96 97 98 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 92 def non_haml was_active = @haml_buffer.active? @haml_buffer.active = false yield ensure @haml_buffer.active = was_active end |
#precede(str) { ... } ⇒ Object
Prepends a string to the beginning of a Haml block, with no whitespace between. For example:
= precede '*' do
%span.small Not really
Produces:
*<span class='small'>Not really</span>
342 343 344 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 342 def precede(str, &block) "#{str}#{capture_haml(&block).chomp}\n" end |
#preserve(input) ⇒ Object #preserve { ... } ⇒ Object Also known as: flatten
Takes any string, finds all the newlines, and converts them to HTML entities so they’ll render correctly in whitespace-sensitive tags without screwing up the indentation.
138 139 140 141 142 143 144 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 138 def preserve(input = nil, &block) return preserve(capture_haml(&block)) if block s = input.to_s.chomp("\n") s.gsub!(/\n/, '
') s.delete!("\r") s end |
#succeed(str) { ... } ⇒ Object
Appends a string to the end of a Haml block, with no whitespace between. For example:
click
= succeed '.' do
%a{:href=>"thing"} here
Produces:
click
<a href='thing'>here</a>.
361 362 363 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 361 def succeed(str, &block) "#{capture_haml(&block).chomp}#{str}\n" end |
#surround(front, back = front) { ... } ⇒ Object
Surrounds a block of Haml code with strings, with no whitespace in between. For example:
= surround '(', ')' do
%a{:href => "food"} chicken
Produces:
(<a href='food'>chicken</a>)
and
= surround '*' do
%strong angry
Produces:
*<strong>angry</strong>*
323 324 325 326 327 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 323 def surround(front, back = front, &block) output = capture_haml(&block) "#{front}#{output.chomp}#{back}\n" end |
#tab_down(i = 1) ⇒ Object
Decrements the number of tabs the buffer automatically adds to the lines of the template.
269 270 271 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 269 def tab_down(i = 1) haml_buffer.tabulation -= i end |
#tab_up(i = 1) ⇒ Object
Increments the number of tabs the buffer automatically adds to the lines of the template. For example:
%h1 foo
- tab_up
%p bar
- tab_down
%strong baz
Produces:
<h1>foo</h1>
<p>bar</p>
<strong>baz</strong>
260 261 262 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 260 def tab_up(i = 1) haml_buffer.tabulation += i end |
#with_tabs(i) { ... } ⇒ Object
Sets the number of tabs the buffer automatically adds to the lines of the template, but only for the duration of the block. For example:
%h1 foo
- with_tabs(2) do
%p bar
%strong baz
Produces:
<h1>foo</h1>
<p>bar</p>
<strong>baz</strong>
292 293 294 295 296 297 298 |
# File 'lib/hamlit/parser/haml_helpers.rb', line 292 def with_tabs(i) old_tabs = haml_buffer.tabulation haml_buffer.tabulation = i yield ensure haml_buffer.tabulation = old_tabs end |