Skip to content

Commit

Permalink
Merge pull request #15820 from lucasmazza/lm-except-highlight-tweaks
Browse files Browse the repository at this point in the history
highlight/except support for regular expressions and blocks.
  • Loading branch information
rafaelfranca committed Jun 19, 2014
2 parents 9ac81b9 + 9f27e10 commit 27953bc
Show file tree
Hide file tree
Showing 3 changed files with 50 additions and 7 deletions.
9 changes: 9 additions & 0 deletions actionview/CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,12 @@
* The `highlight` helper now accepts a block to be used instead of the `highlighter`
option.

*Lucas Mazza*

* The `except` and `highlight` helpers now accept regular expressions.

*Jan Szumiec*

* Flatten the array parameter in `safe_join`, so it behaves consistently with
`Array#join`.

Expand Down
31 changes: 24 additions & 7 deletions actionview/lib/action_view/helpers/text_helper.rb
Original file line number Diff line number Diff line change
Expand Up @@ -103,11 +103,14 @@ def truncate(text, options = {}, &block)
# Highlights one or more +phrases+ everywhere in +text+ by inserting it into
# a <tt>:highlighter</tt> string. The highlighter can be specialized by passing <tt>:highlighter</tt>
# as a single-quoted string with <tt>\1</tt> where the phrase is to be inserted (defaults to
# '<mark>\1</mark>')
# '<mark>\1</mark>') or passing a block that receives each matched term.
#
# highlight('You searched for: rails', 'rails')
# # => You searched for: <mark>rails</mark>
#
# highlight('You searched for: rails', /for|rails/)
# # => You searched <mark>for</mark>: <mark>rails</mark>
#
# highlight('You searched for: ruby, rails, dhh', 'actionpack')
# # => You searched for: ruby, rails, dhh
#
Expand All @@ -116,15 +119,25 @@ def truncate(text, options = {}, &block)
#
# highlight('You searched for: rails', 'rails', highlighter: '<a href="search?q=\1">\1</a>')
# # => You searched for: <a href="search?q=rails">rails</a>
#
# highlight('You searched for: rails', 'rails') { |match| link_to(search_path(q: match, match)) }
# # => You searched for: <a href="search?q=rails">rails</a>
def highlight(text, phrases, options = {})
text = sanitize(text) if options.fetch(:sanitize, true)

if text.blank? || phrases.blank?
text
else
highlighter = options.fetch(:highlighter, '<mark>\1</mark>')
match = Array(phrases).map { |p| Regexp.escape(p) }.join('|')
text.gsub(/(#{match})(?![^<]*?>)/i, highlighter)
match = Array(phrases).map do |p|
Regexp === p ? p.to_s : Regexp.escape(p)
end.join('|')

if block_given?
text.gsub(/(#{match})(?![^<]*?>)/i) { |found| yield found }
else
highlighter = options.fetch(:highlighter, '<mark>\1</mark>')
text.gsub(/(#{match})(?![^<]*?>)/i, highlighter)
end
end.html_safe
end

Expand Down Expand Up @@ -155,9 +168,13 @@ def highlight(text, phrases, options = {})
def excerpt(text, phrase, options = {})
return unless text && phrase

separator = options[:separator] || ''
phrase = Regexp.escape(phrase)
regex = /#{phrase}/i
separator = options.fetch(:separator, nil) || ""
case phrase
when Regexp
regex = phrase
else
regex = /#{Regexp.escape(phrase)}/i
end

return unless matches = text.match(regex)
phrase = matches[0]
Expand Down
17 changes: 17 additions & 0 deletions actionview/test/template/text_helper_test.rb
Original file line number Diff line number Diff line change
Expand Up @@ -222,6 +222,11 @@ def test_highlight_with_regexp
)
end

def test_highlight_accepts_regexp
assert_equal("This day was challenging for judge <mark>Allen</mark> and his colleagues.",
highlight("This day was challenging for judge Allen and his colleagues.", /\ballen\b/i))
end

def test_highlight_with_multiple_phrases_in_one_pass
assert_equal %(<em>wow</em> <em>em</em>), highlight('wow em', %w(wow em), :highlighter => '<em>\1</em>')
end
Expand Down Expand Up @@ -260,13 +265,25 @@ def test_highlight_does_not_modify_the_options_hash
assert_equal options, passed_options
end

def test_highlight_with_block
assert_equal(
"<b>one</b> <b>two</b> <b>three</b>",
highlight("one two three", ["one", "two", "three"]) { |word| "<b>#{word}</b>" }
)
end

def test_excerpt
assert_equal("...is a beautiful morn...", excerpt("This is a beautiful morning", "beautiful", :radius => 5))
assert_equal("This is a...", excerpt("This is a beautiful morning", "this", :radius => 5))
assert_equal("...iful morning", excerpt("This is a beautiful morning", "morning", :radius => 5))
assert_nil excerpt("This is a beautiful morning", "day")
end

def test_excerpt_with_regex
assert_equal("...udge Allen and...", excerpt("This day was challenging for judge Allen and his colleagues.", /\ballen\b/i, :radius => 5))
assert_equal("...judge Allen and...", excerpt("This day was challenging for judge Allen and his colleagues.", /\ballen\b/i, :radius => 1, :separator => ' '))
end

def test_excerpt_should_not_be_html_safe
assert !excerpt('This is a beautiful! morning', 'beautiful', :radius => 5).html_safe?
end
Expand Down

0 comments on commit 27953bc

Please sign in to comment.