##// END OF EJS Templates
Fixes distance of date in words calculation....
Jean-Philippe Lang -
r2902:86a9d90f07d6
parent child
Show More
@@ -1,80 +1,80
1 1
2 2 require 'activerecord'
3 3
4 4 module ActiveRecord
5 5 class Base
6 6 include Redmine::I18n
7 7
8 8 # Translate attribute names for validation errors display
9 9 def self.human_attribute_name(attr)
10 10 l("field_#{attr.to_s.gsub(/_id$/, '')}")
11 11 end
12 12 end
13 13 end
14 14
15 15 module ActiveRecord
16 16 class Errors
17 17 def full_messages(options = {})
18 18 full_messages = []
19 19
20 20 @errors.each_key do |attr|
21 21 @errors[attr].each do |message|
22 22 next unless message
23 23
24 24 if attr == "base"
25 25 full_messages << message
26 26 elsif attr == "custom_values"
27 27 # Replace the generic "custom values is invalid"
28 28 # with the errors on custom values
29 29 @base.custom_values.each do |value|
30 30 value.errors.each do |attr, msg|
31 31 full_messages << value.custom_field.name + ' ' + msg
32 32 end
33 33 end
34 34 else
35 35 attr_name = @base.class.human_attribute_name(attr)
36 36 full_messages << attr_name + ' ' + message.to_s
37 37 end
38 38 end
39 39 end
40 40 full_messages
41 41 end
42 42 end
43 43 end
44 44
45 45 module ActionView
46 46 module Helpers
47 47 module DateHelper
48 48 # distance_of_time_in_words breaks when difference is greater than 30 years
49 49 def distance_of_date_in_words(from_date, to_date = 0, options = {})
50 50 from_date = from_date.to_date if from_date.respond_to?(:to_date)
51 51 to_date = to_date.to_date if to_date.respond_to?(:to_date)
52 52 distance_in_days = (to_date - from_date).abs
53 53
54 54 I18n.with_options :locale => options[:locale], :scope => :'datetime.distance_in_words' do |locale|
55 55 case distance_in_days
56 when 0..60 then locale.t :x_days, :count => distance_in_days
56 when 0..60 then locale.t :x_days, :count => distance_in_days.round
57 57 when 61..720 then locale.t :about_x_months, :count => (distance_in_days / 30).round
58 else locale.t :over_x_years, :count => (distance_in_days / 365).round
58 else locale.t :over_x_years, :count => (distance_in_days / 365).floor
59 59 end
60 60 end
61 61 end
62 62 end
63 63 end
64 64 end
65 65
66 66 ActionView::Base.field_error_proc = Proc.new{ |html_tag, instance| "#{html_tag}" }
67 67
68 68 # Adds :async_smtp and :async_sendmail delivery methods
69 69 # to perform email deliveries asynchronously
70 70 module AsynchronousMailer
71 71 %w(smtp sendmail).each do |type|
72 72 define_method("perform_delivery_async_#{type}") do |mail|
73 73 Thread.start do
74 74 send "perform_delivery_#{type}", mail
75 75 end
76 76 end
77 77 end
78 78 end
79 79
80 80 ActionMailer::Base.send :include, AsynchronousMailer
@@ -1,486 +1,486
1 1 # Redmine - project management software
2 2 # Copyright (C) 2006-2009 Jean-Philippe Lang
3 3 #
4 4 # This program is free software; you can redistribute it and/or
5 5 # modify it under the terms of the GNU General Public License
6 6 # as published by the Free Software Foundation; either version 2
7 7 # of the License, or (at your option) any later version.
8 8 #
9 9 # This program is distributed in the hope that it will be useful,
10 10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 12 # GNU General Public License for more details.
13 13 #
14 14 # You should have received a copy of the GNU General Public License
15 15 # along with this program; if not, write to the Free Software
16 16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17 17
18 18 require File.dirname(__FILE__) + '/../../test_helper'
19 19
20 20 class ApplicationHelperTest < HelperTestCase
21 21 include ApplicationHelper
22 22 include ActionView::Helpers::TextHelper
23 23 include ActionView::Helpers::DateHelper
24 24
25 25 fixtures :projects, :roles, :enabled_modules, :users,
26 26 :repositories, :changesets,
27 27 :trackers, :issue_statuses, :issues, :versions, :documents,
28 28 :wikis, :wiki_pages, :wiki_contents,
29 29 :boards, :messages,
30 30 :attachments
31 31
32 32 def setup
33 33 super
34 34 end
35 35
36 36 def test_auto_links
37 37 to_test = {
38 38 'http://foo.bar' => '<a class="external" href="http://foo.bar">http://foo.bar</a>',
39 39 'http://foo.bar/~user' => '<a class="external" href="http://foo.bar/~user">http://foo.bar/~user</a>',
40 40 'http://foo.bar.' => '<a class="external" href="http://foo.bar">http://foo.bar</a>.',
41 41 'https://foo.bar.' => '<a class="external" href="https://foo.bar">https://foo.bar</a>.',
42 42 'This is a link: http://foo.bar.' => 'This is a link: <a class="external" href="http://foo.bar">http://foo.bar</a>.',
43 43 'A link (eg. http://foo.bar).' => 'A link (eg. <a class="external" href="http://foo.bar">http://foo.bar</a>).',
44 44 'http://foo.bar/foo.bar#foo.bar.' => '<a class="external" href="http://foo.bar/foo.bar#foo.bar">http://foo.bar/foo.bar#foo.bar</a>.',
45 45 'http://www.foo.bar/Test_(foobar)' => '<a class="external" href="http://www.foo.bar/Test_(foobar)">http://www.foo.bar/Test_(foobar)</a>',
46 46 '(see inline link : http://www.foo.bar/Test_(foobar))' => '(see inline link : <a class="external" href="http://www.foo.bar/Test_(foobar)">http://www.foo.bar/Test_(foobar)</a>)',
47 47 '(see inline link : http://www.foo.bar/Test)' => '(see inline link : <a class="external" href="http://www.foo.bar/Test">http://www.foo.bar/Test</a>)',
48 48 '(see inline link : http://www.foo.bar/Test).' => '(see inline link : <a class="external" href="http://www.foo.bar/Test">http://www.foo.bar/Test</a>).',
49 49 '(see "inline link":http://www.foo.bar/Test_(foobar))' => '(see <a href="http://www.foo.bar/Test_(foobar)" class="external">inline link</a>)',
50 50 '(see "inline link":http://www.foo.bar/Test)' => '(see <a href="http://www.foo.bar/Test" class="external">inline link</a>)',
51 51 '(see "inline link":http://www.foo.bar/Test).' => '(see <a href="http://www.foo.bar/Test" class="external">inline link</a>).',
52 52 'www.foo.bar' => '<a class="external" href="http://www.foo.bar">www.foo.bar</a>',
53 53 'http://foo.bar/page?p=1&t=z&s=' => '<a class="external" href="http://foo.bar/page?p=1&#38;t=z&#38;s=">http://foo.bar/page?p=1&#38;t=z&#38;s=</a>',
54 54 'http://foo.bar/page#125' => '<a class="external" href="http://foo.bar/page#125">http://foo.bar/page#125</a>',
55 55 'http://foo@www.bar.com' => '<a class="external" href="http://foo@www.bar.com">http://foo@www.bar.com</a>',
56 56 'http://foo:bar@www.bar.com' => '<a class="external" href="http://foo:bar@www.bar.com">http://foo:bar@www.bar.com</a>',
57 57 'ftp://foo.bar' => '<a class="external" href="ftp://foo.bar">ftp://foo.bar</a>',
58 58 'ftps://foo.bar' => '<a class="external" href="ftps://foo.bar">ftps://foo.bar</a>',
59 59 'sftp://foo.bar' => '<a class="external" href="sftp://foo.bar">sftp://foo.bar</a>',
60 60 # two exclamation marks
61 61 'http://example.net/path!602815048C7B5C20!302.html' => '<a class="external" href="http://example.net/path!602815048C7B5C20!302.html">http://example.net/path!602815048C7B5C20!302.html</a>',
62 62 }
63 63 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
64 64 end
65 65
66 66 def test_auto_mailto
67 67 assert_equal '<p><a href="mailto:test@foo.bar" class="email">test@foo.bar</a></p>',
68 68 textilizable('test@foo.bar')
69 69 end
70 70
71 71 def test_inline_images
72 72 to_test = {
73 73 '!http://foo.bar/image.jpg!' => '<img src="http://foo.bar/image.jpg" alt="" />',
74 74 'floating !>http://foo.bar/image.jpg!' => 'floating <div style="float:right"><img src="http://foo.bar/image.jpg" alt="" /></div>',
75 75 'with class !(some-class)http://foo.bar/image.jpg!' => 'with class <img src="http://foo.bar/image.jpg" class="some-class" alt="" />',
76 76 # inline styles should be stripped
77 77 'with style !{width:100px;height100px}http://foo.bar/image.jpg!' => 'with style <img src="http://foo.bar/image.jpg" alt="" />',
78 78 'with title !http://foo.bar/image.jpg(This is a title)!' => 'with title <img src="http://foo.bar/image.jpg" title="This is a title" alt="This is a title" />',
79 79 'with title !http://foo.bar/image.jpg(This is a double-quoted "title")!' => 'with title <img src="http://foo.bar/image.jpg" title="This is a double-quoted &quot;title&quot;" alt="This is a double-quoted &quot;title&quot;" />',
80 80 }
81 81 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
82 82 end
83 83
84 84 def test_inline_images_inside_tags
85 85 raw = <<-RAW
86 86 h1. !foo.png! Heading
87 87
88 88 Centered image:
89 89
90 90 p=. !bar.gif!
91 91 RAW
92 92
93 93 assert textilizable(raw).include?('<img src="foo.png" alt="" />')
94 94 assert textilizable(raw).include?('<img src="bar.gif" alt="" />')
95 95 end
96 96
97 97 def test_acronyms
98 98 to_test = {
99 99 'this is an acronym: GPL(General Public License)' => 'this is an acronym: <acronym title="General Public License">GPL</acronym>',
100 100 'GPL(This is a double-quoted "title")' => '<acronym title="This is a double-quoted &quot;title&quot;">GPL</acronym>',
101 101 }
102 102 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
103 103
104 104 end
105 105
106 106 def test_attached_images
107 107 to_test = {
108 108 'Inline image: !logo.gif!' => 'Inline image: <img src="/attachments/download/3" title="This is a logo" alt="This is a logo" />',
109 109 'Inline image: !logo.GIF!' => 'Inline image: <img src="/attachments/download/3" title="This is a logo" alt="This is a logo" />',
110 110 'No match: !ogo.gif!' => 'No match: <img src="ogo.gif" alt="" />',
111 111 'No match: !ogo.GIF!' => 'No match: <img src="ogo.GIF" alt="" />',
112 112 # link image
113 113 '!logo.gif!:http://foo.bar/' => '<a href="http://foo.bar/"><img src="/attachments/download/3" title="This is a logo" alt="This is a logo" /></a>',
114 114 }
115 115 attachments = Attachment.find(:all)
116 116 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text, :attachments => attachments) }
117 117 end
118 118
119 119 def test_textile_external_links
120 120 to_test = {
121 121 'This is a "link":http://foo.bar' => 'This is a <a href="http://foo.bar" class="external">link</a>',
122 122 'This is an intern "link":/foo/bar' => 'This is an intern <a href="/foo/bar">link</a>',
123 123 '"link (Link title)":http://foo.bar' => '<a href="http://foo.bar" title="Link title" class="external">link</a>',
124 124 '"link (Link title with "double-quotes")":http://foo.bar' => '<a href="http://foo.bar" title="Link title with &quot;double-quotes&quot;" class="external">link</a>',
125 125 "This is not a \"Link\":\n\nAnother paragraph" => "This is not a \"Link\":</p>\n\n\n\t<p>Another paragraph",
126 126 # no multiline link text
127 127 "This is a double quote \"on the first line\nand another on a second line\":test" => "This is a double quote \"on the first line<br />and another on a second line\":test",
128 128 # mailto link
129 129 "\"system administrator\":mailto:sysadmin@example.com?subject=redmine%20permissions" => "<a href=\"mailto:sysadmin@example.com?subject=redmine%20permissions\">system administrator</a>",
130 130 # two exclamation marks
131 131 '"a link":http://example.net/path!602815048C7B5C20!302.html' => '<a href="http://example.net/path!602815048C7B5C20!302.html" class="external">a link</a>',
132 132 }
133 133 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
134 134 end
135 135
136 136 def test_redmine_links
137 137 issue_link = link_to('#3', {:controller => 'issues', :action => 'show', :id => 3},
138 138 :class => 'issue', :title => 'Error 281 when updating a recipe (New)')
139 139
140 140 changeset_link = link_to('r1', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :rev => 1},
141 141 :class => 'changeset', :title => 'My very first commit')
142 142 changeset_link2 = link_to('r2', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :rev => 2},
143 143 :class => 'changeset', :title => 'This commit fixes #1, #2 and references #1 & #3')
144 144
145 145 document_link = link_to('Test document', {:controller => 'documents', :action => 'show', :id => 1},
146 146 :class => 'document')
147 147
148 148 version_link = link_to('1.0', {:controller => 'versions', :action => 'show', :id => 2},
149 149 :class => 'version')
150 150
151 151 message_url = {:controller => 'messages', :action => 'show', :board_id => 1, :id => 4}
152 152
153 153 source_url = {:controller => 'repositories', :action => 'entry', :id => 'ecookbook', :path => ['some', 'file']}
154 154 source_url_with_ext = {:controller => 'repositories', :action => 'entry', :id => 'ecookbook', :path => ['some', 'file.ext']}
155 155
156 156 to_test = {
157 157 # tickets
158 158 '#3, #3 and #3.' => "#{issue_link}, #{issue_link} and #{issue_link}.",
159 159 # changesets
160 160 'r1' => changeset_link,
161 161 'r1.' => "#{changeset_link}.",
162 162 'r1, r2' => "#{changeset_link}, #{changeset_link2}",
163 163 'r1,r2' => "#{changeset_link},#{changeset_link2}",
164 164 # documents
165 165 'document#1' => document_link,
166 166 'document:"Test document"' => document_link,
167 167 # versions
168 168 'version#2' => version_link,
169 169 'version:1.0' => version_link,
170 170 'version:"1.0"' => version_link,
171 171 # source
172 172 'source:/some/file' => link_to('source:/some/file', source_url, :class => 'source'),
173 173 'source:/some/file.' => link_to('source:/some/file', source_url, :class => 'source') + ".",
174 174 'source:/some/file.ext.' => link_to('source:/some/file.ext', source_url_with_ext, :class => 'source') + ".",
175 175 'source:/some/file. ' => link_to('source:/some/file', source_url, :class => 'source') + ".",
176 176 'source:/some/file.ext. ' => link_to('source:/some/file.ext', source_url_with_ext, :class => 'source') + ".",
177 177 'source:/some/file, ' => link_to('source:/some/file', source_url, :class => 'source') + ",",
178 178 'source:/some/file@52' => link_to('source:/some/file@52', source_url.merge(:rev => 52), :class => 'source'),
179 179 'source:/some/file.ext@52' => link_to('source:/some/file.ext@52', source_url_with_ext.merge(:rev => 52), :class => 'source'),
180 180 'source:/some/file#L110' => link_to('source:/some/file#L110', source_url.merge(:anchor => 'L110'), :class => 'source'),
181 181 'source:/some/file.ext#L110' => link_to('source:/some/file.ext#L110', source_url_with_ext.merge(:anchor => 'L110'), :class => 'source'),
182 182 'source:/some/file@52#L110' => link_to('source:/some/file@52#L110', source_url.merge(:rev => 52, :anchor => 'L110'), :class => 'source'),
183 183 'export:/some/file' => link_to('export:/some/file', source_url.merge(:format => 'raw'), :class => 'source download'),
184 184 # message
185 185 'message#4' => link_to('Post 2', message_url, :class => 'message'),
186 186 'message#5' => link_to('RE: post 2', message_url.merge(:anchor => 'message-5'), :class => 'message'),
187 187 # escaping
188 188 '!#3.' => '#3.',
189 189 '!r1' => 'r1',
190 190 '!document#1' => 'document#1',
191 191 '!document:"Test document"' => 'document:"Test document"',
192 192 '!version#2' => 'version#2',
193 193 '!version:1.0' => 'version:1.0',
194 194 '!version:"1.0"' => 'version:"1.0"',
195 195 '!source:/some/file' => 'source:/some/file',
196 196 # invalid expressions
197 197 'source:' => 'source:',
198 198 # url hash
199 199 "http://foo.bar/FAQ#3" => '<a class="external" href="http://foo.bar/FAQ#3">http://foo.bar/FAQ#3</a>',
200 200 }
201 201 @project = Project.find(1)
202 202 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
203 203 end
204 204
205 205 def test_wiki_links
206 206 to_test = {
207 207 '[[CookBook documentation]]' => '<a href="/projects/ecookbook/wiki/CookBook_documentation" class="wiki-page">CookBook documentation</a>',
208 208 '[[Another page|Page]]' => '<a href="/projects/ecookbook/wiki/Another_page" class="wiki-page">Page</a>',
209 209 # link with anchor
210 210 '[[CookBook documentation#One-section]]' => '<a href="/projects/ecookbook/wiki/CookBook_documentation#One-section" class="wiki-page">CookBook documentation</a>',
211 211 '[[Another page#anchor|Page]]' => '<a href="/projects/ecookbook/wiki/Another_page#anchor" class="wiki-page">Page</a>',
212 212 # page that doesn't exist
213 213 '[[Unknown page]]' => '<a href="/projects/ecookbook/wiki/Unknown_page" class="wiki-page new">Unknown page</a>',
214 214 '[[Unknown page|404]]' => '<a href="/projects/ecookbook/wiki/Unknown_page" class="wiki-page new">404</a>',
215 215 # link to another project wiki
216 216 '[[onlinestore:]]' => '<a href="/projects/onlinestore/wiki/" class="wiki-page">onlinestore</a>',
217 217 '[[onlinestore:|Wiki]]' => '<a href="/projects/onlinestore/wiki/" class="wiki-page">Wiki</a>',
218 218 '[[onlinestore:Start page]]' => '<a href="/projects/onlinestore/wiki/Start_page" class="wiki-page">Start page</a>',
219 219 '[[onlinestore:Start page|Text]]' => '<a href="/projects/onlinestore/wiki/Start_page" class="wiki-page">Text</a>',
220 220 '[[onlinestore:Unknown page]]' => '<a href="/projects/onlinestore/wiki/Unknown_page" class="wiki-page new">Unknown page</a>',
221 221 # striked through link
222 222 '-[[Another page|Page]]-' => '<del><a href="/projects/ecookbook/wiki/Another_page" class="wiki-page">Page</a></del>',
223 223 '-[[Another page|Page]] link-' => '<del><a href="/projects/ecookbook/wiki/Another_page" class="wiki-page">Page</a> link</del>',
224 224 # escaping
225 225 '![[Another page|Page]]' => '[[Another page|Page]]',
226 226 # project does not exist
227 227 '[[unknowproject:Start]]' => '[[unknowproject:Start]]',
228 228 '[[unknowproject:Start|Page title]]' => '[[unknowproject:Start|Page title]]',
229 229 }
230 230 @project = Project.find(1)
231 231 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
232 232 end
233 233
234 234 def test_html_tags
235 235 to_test = {
236 236 "<div>content</div>" => "<p>&lt;div&gt;content&lt;/div&gt;</p>",
237 237 "<div class=\"bold\">content</div>" => "<p>&lt;div class=\"bold\"&gt;content&lt;/div&gt;</p>",
238 238 "<script>some script;</script>" => "<p>&lt;script&gt;some script;&lt;/script&gt;</p>",
239 239 # do not escape pre/code tags
240 240 "<pre>\nline 1\nline2</pre>" => "<pre>\nline 1\nline2</pre>",
241 241 "<pre><code>\nline 1\nline2</code></pre>" => "<pre><code>\nline 1\nline2</code></pre>",
242 242 "<pre><div>content</div></pre>" => "<pre>&lt;div&gt;content&lt;/div&gt;</pre>",
243 243 "HTML comment: <!-- no comments -->" => "<p>HTML comment: &lt;!-- no comments --&gt;</p>",
244 244 "<!-- opening comment" => "<p>&lt;!-- opening comment</p>",
245 245 # remove attributes except class
246 246 "<pre class='foo'>some text</pre>" => "<pre class='foo'>some text</pre>",
247 247 "<pre onmouseover='alert(1)'>some text</pre>" => "<pre>some text</pre>",
248 248 }
249 249 to_test.each { |text, result| assert_equal result, textilizable(text) }
250 250 end
251 251
252 252 def test_allowed_html_tags
253 253 to_test = {
254 254 "<pre>preformatted text</pre>" => "<pre>preformatted text</pre>",
255 255 "<notextile>no *textile* formatting</notextile>" => "no *textile* formatting",
256 256 "<notextile>this is <tag>a tag</tag></notextile>" => "this is &lt;tag&gt;a tag&lt;/tag&gt;"
257 257 }
258 258 to_test.each { |text, result| assert_equal result, textilizable(text) }
259 259 end
260 260
261 261 def test_syntax_highlight
262 262 raw = <<-RAW
263 263 <pre><code class="ruby">
264 264 # Some ruby code here
265 265 </pre></code>
266 266 RAW
267 267
268 268 expected = <<-EXPECTED
269 269 <pre><code class="ruby CodeRay"><span class="no">1</span> <span class="c"># Some ruby code here</span>
270 270 </pre></code>
271 271 EXPECTED
272 272
273 273 assert_equal expected.gsub(%r{[\r\n\t]}, ''), textilizable(raw).gsub(%r{[\r\n\t]}, '')
274 274 end
275 275
276 276 def test_wiki_links_in_tables
277 277 to_test = {"|[[Page|Link title]]|[[Other Page|Other title]]|\n|Cell 21|[[Last page]]|" =>
278 278 '<tr><td><a href="/projects/ecookbook/wiki/Page" class="wiki-page new">Link title</a></td>' +
279 279 '<td><a href="/projects/ecookbook/wiki/Other_Page" class="wiki-page new">Other title</a></td>' +
280 280 '</tr><tr><td>Cell 21</td><td><a href="/projects/ecookbook/wiki/Last_page" class="wiki-page new">Last page</a></td></tr>'
281 281 }
282 282 @project = Project.find(1)
283 283 to_test.each { |text, result| assert_equal "<table>#{result}</table>", textilizable(text).gsub(/[\t\n]/, '') }
284 284 end
285 285
286 286 def test_text_formatting
287 287 to_test = {'*_+bold, italic and underline+_*' => '<strong><em><ins>bold, italic and underline</ins></em></strong>',
288 288 '(_text within parentheses_)' => '(<em>text within parentheses</em>)',
289 289 'a *Humane Web* Text Generator' => 'a <strong>Humane Web</strong> Text Generator',
290 290 'a H *umane* W *eb* T *ext* G *enerator*' => 'a H <strong>umane</strong> W <strong>eb</strong> T <strong>ext</strong> G <strong>enerator</strong>',
291 291 'a *H* umane *W* eb *T* ext *G* enerator' => 'a <strong>H</strong> umane <strong>W</strong> eb <strong>T</strong> ext <strong>G</strong> enerator',
292 292 }
293 293 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
294 294 end
295 295
296 296 def test_wiki_horizontal_rule
297 297 assert_equal '<hr />', textilizable('---')
298 298 assert_equal '<p>Dashes: ---</p>', textilizable('Dashes: ---')
299 299 end
300 300
301 301 def test_acronym
302 302 assert_equal '<p>This is an acronym: <acronym title="American Civil Liberties Union">ACLU</acronym>.</p>',
303 303 textilizable('This is an acronym: ACLU(American Civil Liberties Union).')
304 304 end
305 305
306 306 def test_footnotes
307 307 raw = <<-RAW
308 308 This is some text[1].
309 309
310 310 fn1. This is the foot note
311 311 RAW
312 312
313 313 expected = <<-EXPECTED
314 314 <p>This is some text<sup><a href=\"#fn1\">1</a></sup>.</p>
315 315 <p id="fn1" class="footnote"><sup>1</sup> This is the foot note</p>
316 316 EXPECTED
317 317
318 318 assert_equal expected.gsub(%r{[\r\n\t]}, ''), textilizable(raw).gsub(%r{[\r\n\t]}, '')
319 319 end
320 320
321 321 def test_table_of_content
322 322 raw = <<-RAW
323 323 {{toc}}
324 324
325 325 h1. Title
326 326
327 327 Lorem ipsum dolor sit amet, consectetuer adipiscing elit. Maecenas sed libero.
328 328
329 329 h2. Subtitle with a [[Wiki]] link
330 330
331 331 Nullam commodo metus accumsan nulla. Curabitur lobortis dui id dolor.
332 332
333 333 h2. Subtitle with [[Wiki|another Wiki]] link
334 334
335 335 h2. Subtitle with %{color:red}red text%
336 336
337 337 h1. Another title
338 338
339 339 RAW
340 340
341 341 expected = '<ul class="toc">' +
342 342 '<li class="heading1"><a href="#Title">Title</a></li>' +
343 343 '<li class="heading2"><a href="#Subtitle-with-a-Wiki-link">Subtitle with a Wiki link</a></li>' +
344 344 '<li class="heading2"><a href="#Subtitle-with-another-Wiki-link">Subtitle with another Wiki link</a></li>' +
345 345 '<li class="heading2"><a href="#Subtitle-with-red-text">Subtitle with red text</a></li>' +
346 346 '<li class="heading1"><a href="#Another-title">Another title</a></li>' +
347 347 '</ul>'
348 348
349 349 assert textilizable(raw).gsub("\n", "").include?(expected)
350 350 end
351 351
352 352 def test_blockquote
353 353 # orig raw text
354 354 raw = <<-RAW
355 355 John said:
356 356 > Lorem ipsum dolor sit amet, consectetuer adipiscing elit. Maecenas sed libero.
357 357 > Nullam commodo metus accumsan nulla. Curabitur lobortis dui id dolor.
358 358 > * Donec odio lorem,
359 359 > * sagittis ac,
360 360 > * malesuada in,
361 361 > * adipiscing eu, dolor.
362 362 >
363 363 > >Nulla varius pulvinar diam. Proin id arcu id lorem scelerisque condimentum. Proin vehicula turpis vitae lacus.
364 364 > Proin a tellus. Nam vel neque.
365 365
366 366 He's right.
367 367 RAW
368 368
369 369 # expected html
370 370 expected = <<-EXPECTED
371 371 <p>John said:</p>
372 372 <blockquote>
373 373 Lorem ipsum dolor sit amet, consectetuer adipiscing elit. Maecenas sed libero.
374 374 Nullam commodo metus accumsan nulla. Curabitur lobortis dui id dolor.
375 375 <ul>
376 376 <li>Donec odio lorem,</li>
377 377 <li>sagittis ac,</li>
378 378 <li>malesuada in,</li>
379 379 <li>adipiscing eu, dolor.</li>
380 380 </ul>
381 381 <blockquote>
382 382 <p>Nulla varius pulvinar diam. Proin id arcu id lorem scelerisque condimentum. Proin vehicula turpis vitae lacus.</p>
383 383 </blockquote>
384 384 <p>Proin a tellus. Nam vel neque.</p>
385 385 </blockquote>
386 386 <p>He's right.</p>
387 387 EXPECTED
388 388
389 389 assert_equal expected.gsub(%r{\s+}, ''), textilizable(raw).gsub(%r{\s+}, '')
390 390 end
391 391
392 392 def test_table
393 393 raw = <<-RAW
394 394 This is a table with empty cells:
395 395
396 396 |cell11|cell12||
397 397 |cell21||cell23|
398 398 |cell31|cell32|cell33|
399 399 RAW
400 400
401 401 expected = <<-EXPECTED
402 402 <p>This is a table with empty cells:</p>
403 403
404 404 <table>
405 405 <tr><td>cell11</td><td>cell12</td><td></td></tr>
406 406 <tr><td>cell21</td><td></td><td>cell23</td></tr>
407 407 <tr><td>cell31</td><td>cell32</td><td>cell33</td></tr>
408 408 </table>
409 409 EXPECTED
410 410
411 411 assert_equal expected.gsub(%r{\s+}, ''), textilizable(raw).gsub(%r{\s+}, '')
412 412 end
413 413
414 414 def test_table_with_line_breaks
415 415 raw = <<-RAW
416 416 This is a table with line breaks:
417 417
418 418 |cell11
419 419 continued|cell12||
420 420 |-cell21-||cell23
421 421 cell23 line2
422 422 cell23 *line3*|
423 423 |cell31|cell32
424 424 cell32 line2|cell33|
425 425
426 426 RAW
427 427
428 428 expected = <<-EXPECTED
429 429 <p>This is a table with line breaks:</p>
430 430
431 431 <table>
432 432 <tr>
433 433 <td>cell11<br />continued</td>
434 434 <td>cell12</td>
435 435 <td></td>
436 436 </tr>
437 437 <tr>
438 438 <td><del>cell21</del></td>
439 439 <td></td>
440 440 <td>cell23<br/>cell23 line2<br/>cell23 <strong>line3</strong></td>
441 441 </tr>
442 442 <tr>
443 443 <td>cell31</td>
444 444 <td>cell32<br/>cell32 line2</td>
445 445 <td>cell33</td>
446 446 </tr>
447 447 </table>
448 448 EXPECTED
449 449
450 450 assert_equal expected.gsub(%r{\s+}, ''), textilizable(raw).gsub(%r{\s+}, '')
451 451 end
452 452
453 453 def test_default_formatter
454 454 Setting.text_formatting = 'unknown'
455 455 text = 'a *link*: http://www.example.net/'
456 456 assert_equal '<p>a *link*: <a href="http://www.example.net/">http://www.example.net/</a></p>', textilizable(text)
457 457 Setting.text_formatting = 'textile'
458 458 end
459 459
460 460 def test_due_date_distance_in_words
461 461 to_test = { Date.today => 'Due in 0 days',
462 462 Date.today + 1 => 'Due in 1 day',
463 463 Date.today + 100 => 'Due in about 3 months',
464 Date.today + 20000 => 'Due in over 55 years',
464 Date.today + 20000 => 'Due in over 54 years',
465 465 Date.today - 1 => '1 day late',
466 466 Date.today - 100 => 'about 3 months late',
467 Date.today - 20000 => 'over 55 years late',
467 Date.today - 20000 => 'over 54 years late',
468 468 }
469 469 to_test.each do |date, expected|
470 470 assert_equal expected, due_date_distance_in_words(date)
471 471 end
472 472 end
473 473
474 474 def test_avatar
475 475 # turn on avatars
476 476 Setting.gravatar_enabled = '1'
477 477 assert avatar(User.find_by_mail('jsmith@somenet.foo')).include?(Digest::MD5.hexdigest('jsmith@somenet.foo'))
478 478 assert avatar('jsmith <jsmith@somenet.foo>').include?(Digest::MD5.hexdigest('jsmith@somenet.foo'))
479 479 assert_nil avatar('jsmith')
480 480 assert_nil avatar(nil)
481 481
482 482 # turn off avatars
483 483 Setting.gravatar_enabled = '0'
484 484 assert_nil avatar(User.find_by_mail('jsmith@somenet.foo'))
485 485 end
486 486 end
General Comments 0
You need to be logged in to leave comments. Login now