##// END OF EJS Templates
Fixed pre tags containing "<pre*" (#4125)....
Jean-Philippe Lang -
r2916:1d8b4ee77857
parent child
Show More
@@ -1,1182 +1,1182
1 1 # vim:ts=4:sw=4:
2 2 # = RedCloth - Textile and Markdown Hybrid for Ruby
3 3 #
4 4 # Homepage:: http://whytheluckystiff.net/ruby/redcloth/
5 5 # Author:: why the lucky stiff (http://whytheluckystiff.net/)
6 6 # Copyright:: (cc) 2004 why the lucky stiff (and his puppet organizations.)
7 7 # License:: BSD
8 8 #
9 9 # (see http://hobix.com/textile/ for a Textile Reference.)
10 10 #
11 11 # Based on (and also inspired by) both:
12 12 #
13 13 # PyTextile: http://diveintomark.org/projects/textile/textile.py.txt
14 14 # Textism for PHP: http://www.textism.com/tools/textile/
15 15 #
16 16 #
17 17
18 18 # = RedCloth
19 19 #
20 20 # RedCloth is a Ruby library for converting Textile and/or Markdown
21 21 # into HTML. You can use either format, intermingled or separately.
22 22 # You can also extend RedCloth to honor your own custom text stylings.
23 23 #
24 24 # RedCloth users are encouraged to use Textile if they are generating
25 25 # HTML and to use Markdown if others will be viewing the plain text.
26 26 #
27 27 # == What is Textile?
28 28 #
29 29 # Textile is a simple formatting style for text
30 30 # documents, loosely based on some HTML conventions.
31 31 #
32 32 # == Sample Textile Text
33 33 #
34 34 # h2. This is a title
35 35 #
36 36 # h3. This is a subhead
37 37 #
38 38 # This is a bit of paragraph.
39 39 #
40 40 # bq. This is a blockquote.
41 41 #
42 42 # = Writing Textile
43 43 #
44 44 # A Textile document consists of paragraphs. Paragraphs
45 45 # can be specially formatted by adding a small instruction
46 46 # to the beginning of the paragraph.
47 47 #
48 48 # h[n]. Header of size [n].
49 49 # bq. Blockquote.
50 50 # # Numeric list.
51 51 # * Bulleted list.
52 52 #
53 53 # == Quick Phrase Modifiers
54 54 #
55 55 # Quick phrase modifiers are also included, to allow formatting
56 56 # of small portions of text within a paragraph.
57 57 #
58 58 # \_emphasis\_
59 59 # \_\_italicized\_\_
60 60 # \*strong\*
61 61 # \*\*bold\*\*
62 62 # ??citation??
63 63 # -deleted text-
64 64 # +inserted text+
65 65 # ^superscript^
66 66 # ~subscript~
67 67 # @code@
68 68 # %(classname)span%
69 69 #
70 70 # ==notextile== (leave text alone)
71 71 #
72 72 # == Links
73 73 #
74 74 # To make a hypertext link, put the link text in "quotation
75 75 # marks" followed immediately by a colon and the URL of the link.
76 76 #
77 77 # Optional: text in (parentheses) following the link text,
78 78 # but before the closing quotation mark, will become a Title
79 79 # attribute for the link, visible as a tool tip when a cursor is above it.
80 80 #
81 81 # Example:
82 82 #
83 83 # "This is a link (This is a title) ":http://www.textism.com
84 84 #
85 85 # Will become:
86 86 #
87 87 # <a href="http://www.textism.com" title="This is a title">This is a link</a>
88 88 #
89 89 # == Images
90 90 #
91 91 # To insert an image, put the URL for the image inside exclamation marks.
92 92 #
93 93 # Optional: text that immediately follows the URL in (parentheses) will
94 94 # be used as the Alt text for the image. Images on the web should always
95 95 # have descriptive Alt text for the benefit of readers using non-graphical
96 96 # browsers.
97 97 #
98 98 # Optional: place a colon followed by a URL immediately after the
99 99 # closing ! to make the image into a link.
100 100 #
101 101 # Example:
102 102 #
103 103 # !http://www.textism.com/common/textist.gif(Textist)!
104 104 #
105 105 # Will become:
106 106 #
107 107 # <img src="http://www.textism.com/common/textist.gif" alt="Textist" />
108 108 #
109 109 # With a link:
110 110 #
111 111 # !/common/textist.gif(Textist)!:http://textism.com
112 112 #
113 113 # Will become:
114 114 #
115 115 # <a href="http://textism.com"><img src="/common/textist.gif" alt="Textist" /></a>
116 116 #
117 117 # == Defining Acronyms
118 118 #
119 119 # HTML allows authors to define acronyms via the tag. The definition appears as a
120 120 # tool tip when a cursor hovers over the acronym. A crucial aid to clear writing,
121 121 # this should be used at least once for each acronym in documents where they appear.
122 122 #
123 123 # To quickly define an acronym in Textile, place the full text in (parentheses)
124 124 # immediately following the acronym.
125 125 #
126 126 # Example:
127 127 #
128 128 # ACLU(American Civil Liberties Union)
129 129 #
130 130 # Will become:
131 131 #
132 132 # <acronym title="American Civil Liberties Union">ACLU</acronym>
133 133 #
134 134 # == Adding Tables
135 135 #
136 136 # In Textile, simple tables can be added by seperating each column by
137 137 # a pipe.
138 138 #
139 139 # |a|simple|table|row|
140 140 # |And|Another|table|row|
141 141 #
142 142 # Attributes are defined by style definitions in parentheses.
143 143 #
144 144 # table(border:1px solid black).
145 145 # (background:#ddd;color:red). |{}| | | |
146 146 #
147 147 # == Using RedCloth
148 148 #
149 149 # RedCloth is simply an extension of the String class, which can handle
150 150 # Textile formatting. Use it like a String and output HTML with its
151 151 # RedCloth#to_html method.
152 152 #
153 153 # doc = RedCloth.new "
154 154 #
155 155 # h2. Test document
156 156 #
157 157 # Just a simple test."
158 158 #
159 159 # puts doc.to_html
160 160 #
161 161 # By default, RedCloth uses both Textile and Markdown formatting, with
162 162 # Textile formatting taking precedence. If you want to turn off Markdown
163 163 # formatting, to boost speed and limit the processor:
164 164 #
165 165 # class RedCloth::Textile.new( str )
166 166
167 167 class RedCloth3 < String
168 168
169 169 VERSION = '3.0.4'
170 170 DEFAULT_RULES = [:textile, :markdown]
171 171
172 172 #
173 173 # Two accessor for setting security restrictions.
174 174 #
175 175 # This is a nice thing if you're using RedCloth for
176 176 # formatting in public places (e.g. Wikis) where you
177 177 # don't want users to abuse HTML for bad things.
178 178 #
179 179 # If +:filter_html+ is set, HTML which wasn't
180 180 # created by the Textile processor will be escaped.
181 181 #
182 182 # If +:filter_styles+ is set, it will also disable
183 183 # the style markup specifier. ('{color: red}')
184 184 #
185 185 attr_accessor :filter_html, :filter_styles
186 186
187 187 #
188 188 # Accessor for toggling hard breaks.
189 189 #
190 190 # If +:hard_breaks+ is set, single newlines will
191 191 # be converted to HTML break tags. This is the
192 192 # default behavior for traditional RedCloth.
193 193 #
194 194 attr_accessor :hard_breaks
195 195
196 196 # Accessor for toggling lite mode.
197 197 #
198 198 # In lite mode, block-level rules are ignored. This means
199 199 # that tables, paragraphs, lists, and such aren't available.
200 200 # Only the inline markup for bold, italics, entities and so on.
201 201 #
202 202 # r = RedCloth.new( "And then? She *fell*!", [:lite_mode] )
203 203 # r.to_html
204 204 # #=> "And then? She <strong>fell</strong>!"
205 205 #
206 206 attr_accessor :lite_mode
207 207
208 208 #
209 209 # Accessor for toggling span caps.
210 210 #
211 211 # Textile places `span' tags around capitalized
212 212 # words by default, but this wreaks havoc on Wikis.
213 213 # If +:no_span_caps+ is set, this will be
214 214 # suppressed.
215 215 #
216 216 attr_accessor :no_span_caps
217 217
218 218 #
219 219 # Establishes the markup predence. Available rules include:
220 220 #
221 221 # == Textile Rules
222 222 #
223 223 # The following textile rules can be set individually. Or add the complete
224 224 # set of rules with the single :textile rule, which supplies the rule set in
225 225 # the following precedence:
226 226 #
227 227 # refs_textile:: Textile references (i.e. [hobix]http://hobix.com/)
228 228 # block_textile_table:: Textile table block structures
229 229 # block_textile_lists:: Textile list structures
230 230 # block_textile_prefix:: Textile blocks with prefixes (i.e. bq., h2., etc.)
231 231 # inline_textile_image:: Textile inline images
232 232 # inline_textile_link:: Textile inline links
233 233 # inline_textile_span:: Textile inline spans
234 234 # glyphs_textile:: Textile entities (such as em-dashes and smart quotes)
235 235 #
236 236 # == Markdown
237 237 #
238 238 # refs_markdown:: Markdown references (for example: [hobix]: http://hobix.com/)
239 239 # block_markdown_setext:: Markdown setext headers
240 240 # block_markdown_atx:: Markdown atx headers
241 241 # block_markdown_rule:: Markdown horizontal rules
242 242 # block_markdown_bq:: Markdown blockquotes
243 243 # block_markdown_lists:: Markdown lists
244 244 # inline_markdown_link:: Markdown links
245 245 attr_accessor :rules
246 246
247 247 # Returns a new RedCloth object, based on _string_ and
248 248 # enforcing all the included _restrictions_.
249 249 #
250 250 # r = RedCloth.new( "h1. A <b>bold</b> man", [:filter_html] )
251 251 # r.to_html
252 252 # #=>"<h1>A &lt;b&gt;bold&lt;/b&gt; man</h1>"
253 253 #
254 254 def initialize( string, restrictions = [] )
255 255 restrictions.each { |r| method( "#{ r }=" ).call( true ) }
256 256 super( string )
257 257 end
258 258
259 259 #
260 260 # Generates HTML from the Textile contents.
261 261 #
262 262 # r = RedCloth.new( "And then? She *fell*!" )
263 263 # r.to_html( true )
264 264 # #=>"And then? She <strong>fell</strong>!"
265 265 #
266 266 def to_html( *rules )
267 267 rules = DEFAULT_RULES if rules.empty?
268 268 # make our working copy
269 269 text = self.dup
270 270
271 271 @urlrefs = {}
272 272 @shelf = []
273 273 textile_rules = [:refs_textile, :block_textile_table, :block_textile_lists,
274 274 :block_textile_prefix, :inline_textile_image, :inline_textile_link,
275 275 :inline_textile_code, :inline_textile_span, :glyphs_textile]
276 276 markdown_rules = [:refs_markdown, :block_markdown_setext, :block_markdown_atx, :block_markdown_rule,
277 277 :block_markdown_bq, :block_markdown_lists,
278 278 :inline_markdown_reflink, :inline_markdown_link]
279 279 @rules = rules.collect do |rule|
280 280 case rule
281 281 when :markdown
282 282 markdown_rules
283 283 when :textile
284 284 textile_rules
285 285 else
286 286 rule
287 287 end
288 288 end.flatten
289 289
290 290 # standard clean up
291 291 incoming_entities text
292 292 clean_white_space text
293 293
294 294 # start processor
295 295 @pre_list = []
296 296 rip_offtags text
297 297 no_textile text
298 298 escape_html_tags text
299 299 hard_break text
300 300 unless @lite_mode
301 301 refs text
302 302 # need to do this before text is split by #blocks
303 303 block_textile_quotes text
304 304 blocks text
305 305 end
306 306 inline text
307 307 smooth_offtags text
308 308
309 309 retrieve text
310 310
311 311 text.gsub!( /<\/?notextile>/, '' )
312 312 text.gsub!( /x%x%/, '&#38;' )
313 313 clean_html text if filter_html
314 314 text.strip!
315 315 text
316 316
317 317 end
318 318
319 319 #######
320 320 private
321 321 #######
322 322 #
323 323 # Mapping of 8-bit ASCII codes to HTML numerical entity equivalents.
324 324 # (from PyTextile)
325 325 #
326 326 TEXTILE_TAGS =
327 327
328 328 [[128, 8364], [129, 0], [130, 8218], [131, 402], [132, 8222], [133, 8230],
329 329 [134, 8224], [135, 8225], [136, 710], [137, 8240], [138, 352], [139, 8249],
330 330 [140, 338], [141, 0], [142, 0], [143, 0], [144, 0], [145, 8216], [146, 8217],
331 331 [147, 8220], [148, 8221], [149, 8226], [150, 8211], [151, 8212], [152, 732],
332 332 [153, 8482], [154, 353], [155, 8250], [156, 339], [157, 0], [158, 0], [159, 376]].
333 333
334 334 collect! do |a, b|
335 335 [a.chr, ( b.zero? and "" or "&#{ b };" )]
336 336 end
337 337
338 338 #
339 339 # Regular expressions to convert to HTML.
340 340 #
341 341 A_HLGN = /(?:(?:<>|<|>|\=|[()]+)+)/
342 342 A_VLGN = /[\-^~]/
343 343 C_CLAS = '(?:\([^)]+\))'
344 344 C_LNGE = '(?:\[[^\[\]]+\])'
345 345 C_STYL = '(?:\{[^}]+\})'
346 346 S_CSPN = '(?:\\\\\d+)'
347 347 S_RSPN = '(?:/\d+)'
348 348 A = "(?:#{A_HLGN}?#{A_VLGN}?|#{A_VLGN}?#{A_HLGN}?)"
349 349 S = "(?:#{S_CSPN}?#{S_RSPN}|#{S_RSPN}?#{S_CSPN}?)"
350 350 C = "(?:#{C_CLAS}?#{C_STYL}?#{C_LNGE}?|#{C_STYL}?#{C_LNGE}?#{C_CLAS}?|#{C_LNGE}?#{C_STYL}?#{C_CLAS}?)"
351 351 # PUNCT = Regexp::quote( '!"#$%&\'()*+,-./:;<=>?@[\\]^_`{|}~' )
352 352 PUNCT = Regexp::quote( '!"#$%&\'*+,-./:;=?@\\^_`|~' )
353 353 PUNCT_NOQ = Regexp::quote( '!"#$&\',./:;=?@\\`|' )
354 354 PUNCT_Q = Regexp::quote( '*-_+^~%' )
355 355 HYPERLINK = '(\S+?)([^\w\s/;=\?]*?)(?=\s|<|$)'
356 356
357 357 # Text markup tags, don't conflict with block tags
358 358 SIMPLE_HTML_TAGS = [
359 359 'tt', 'b', 'i', 'big', 'small', 'em', 'strong', 'dfn', 'code',
360 360 'samp', 'kbd', 'var', 'cite', 'abbr', 'acronym', 'a', 'img', 'br',
361 361 'br', 'map', 'q', 'sub', 'sup', 'span', 'bdo'
362 362 ]
363 363
364 364 QTAGS = [
365 365 ['**', 'b', :limit],
366 366 ['*', 'strong', :limit],
367 367 ['??', 'cite', :limit],
368 368 ['-', 'del', :limit],
369 369 ['__', 'i', :limit],
370 370 ['_', 'em', :limit],
371 371 ['%', 'span', :limit],
372 372 ['+', 'ins', :limit],
373 373 ['^', 'sup', :limit],
374 374 ['~', 'sub', :limit]
375 375 ]
376 376 QTAGS.collect! do |rc, ht, rtype|
377 377 rcq = Regexp::quote rc
378 378 re =
379 379 case rtype
380 380 when :limit
381 381 /(^|[>\s\(])
382 382 (#{rcq})
383 383 (#{C})
384 384 (?::(\S+?))?
385 385 (\w|[^\s\-].*?[^\s\-])
386 386 #{rcq}
387 387 (?=[[:punct:]]|\s|\)|$)/x
388 388 else
389 389 /(#{rcq})
390 390 (#{C})
391 391 (?::(\S+))?
392 392 (\w|[^\s\-].*?[^\s\-])
393 393 #{rcq}/xm
394 394 end
395 395 [rc, ht, re, rtype]
396 396 end
397 397
398 398 # Elements to handle
399 399 GLYPHS = [
400 400 # [ /([^\s\[{(>])?\'([dmst]\b|ll\b|ve\b|\s|:|$)/, '\1&#8217;\2' ], # single closing
401 401 # [ /([^\s\[{(>#{PUNCT_Q}][#{PUNCT_Q}]*)\'/, '\1&#8217;' ], # single closing
402 402 # [ /\'(?=[#{PUNCT_Q}]*(s\b|[\s#{PUNCT_NOQ}]))/, '&#8217;' ], # single closing
403 403 # [ /\'/, '&#8216;' ], # single opening
404 404 # [ /</, '&lt;' ], # less-than
405 405 # [ />/, '&gt;' ], # greater-than
406 406 # [ /([^\s\[{(])?"(\s|:|$)/, '\1&#8221;\2' ], # double closing
407 407 # [ /([^\s\[{(>#{PUNCT_Q}][#{PUNCT_Q}]*)"/, '\1&#8221;' ], # double closing
408 408 # [ /"(?=[#{PUNCT_Q}]*[\s#{PUNCT_NOQ}])/, '&#8221;' ], # double closing
409 409 # [ /"/, '&#8220;' ], # double opening
410 410 # [ /\b( )?\.{3}/, '\1&#8230;' ], # ellipsis
411 411 # [ /\b([A-Z][A-Z0-9]{2,})\b(?:[(]([^)]*)[)])/, '<acronym title="\2">\1</acronym>' ], # 3+ uppercase acronym
412 412 # [ /(^|[^"][>\s])([A-Z][A-Z0-9 ]+[A-Z0-9])([^<A-Za-z0-9]|$)/, '\1<span class="caps">\2</span>\3', :no_span_caps ], # 3+ uppercase caps
413 413 # [ /(\.\s)?\s?--\s?/, '\1&#8212;' ], # em dash
414 414 # [ /\s->\s/, ' &rarr; ' ], # right arrow
415 415 # [ /\s-\s/, ' &#8211; ' ], # en dash
416 416 # [ /(\d+) ?x ?(\d+)/, '\1&#215;\2' ], # dimension sign
417 417 # [ /\b ?[(\[]TM[\])]/i, '&#8482;' ], # trademark
418 418 # [ /\b ?[(\[]R[\])]/i, '&#174;' ], # registered
419 419 # [ /\b ?[(\[]C[\])]/i, '&#169;' ] # copyright
420 420 ]
421 421
422 422 H_ALGN_VALS = {
423 423 '<' => 'left',
424 424 '=' => 'center',
425 425 '>' => 'right',
426 426 '<>' => 'justify'
427 427 }
428 428
429 429 V_ALGN_VALS = {
430 430 '^' => 'top',
431 431 '-' => 'middle',
432 432 '~' => 'bottom'
433 433 }
434 434
435 435 #
436 436 # Flexible HTML escaping
437 437 #
438 438 def htmlesc( str, mode=:Quotes )
439 439 if str
440 440 str.gsub!( '&', '&amp;' )
441 441 str.gsub!( '"', '&quot;' ) if mode != :NoQuotes
442 442 str.gsub!( "'", '&#039;' ) if mode == :Quotes
443 443 str.gsub!( '<', '&lt;')
444 444 str.gsub!( '>', '&gt;')
445 445 end
446 446 str
447 447 end
448 448
449 449 # Search and replace for Textile glyphs (quotes, dashes, other symbols)
450 450 def pgl( text )
451 451 #GLYPHS.each do |re, resub, tog|
452 452 # next if tog and method( tog ).call
453 453 # text.gsub! re, resub
454 454 #end
455 455 text.gsub!(/\b([A-Z][A-Z0-9]{2,})\b(?:[(]([^)]*)[)])/) do |m|
456 456 "<acronym title=\"#{htmlesc $2}\">#{$1}</acronym>"
457 457 end
458 458 end
459 459
460 460 # Parses Textile attribute lists and builds an HTML attribute string
461 461 def pba( text_in, element = "" )
462 462
463 463 return '' unless text_in
464 464
465 465 style = []
466 466 text = text_in.dup
467 467 if element == 'td'
468 468 colspan = $1 if text =~ /\\(\d+)/
469 469 rowspan = $1 if text =~ /\/(\d+)/
470 470 style << "vertical-align:#{ v_align( $& ) };" if text =~ A_VLGN
471 471 end
472 472
473 473 style << "#{ htmlesc $1 };" if text.sub!( /\{([^}]*)\}/, '' ) && !filter_styles
474 474
475 475 lang = $1 if
476 476 text.sub!( /\[([^)]+?)\]/, '' )
477 477
478 478 cls = $1 if
479 479 text.sub!( /\(([^()]+?)\)/, '' )
480 480
481 481 style << "padding-left:#{ $1.length }em;" if
482 482 text.sub!( /([(]+)/, '' )
483 483
484 484 style << "padding-right:#{ $1.length }em;" if text.sub!( /([)]+)/, '' )
485 485
486 486 style << "text-align:#{ h_align( $& ) };" if text =~ A_HLGN
487 487
488 488 cls, id = $1, $2 if cls =~ /^(.*?)#(.*)$/
489 489
490 490 atts = ''
491 491 atts << " style=\"#{ style.join }\"" unless style.empty?
492 492 atts << " class=\"#{ cls }\"" unless cls.to_s.empty?
493 493 atts << " lang=\"#{ lang }\"" if lang
494 494 atts << " id=\"#{ id }\"" if id
495 495 atts << " colspan=\"#{ colspan }\"" if colspan
496 496 atts << " rowspan=\"#{ rowspan }\"" if rowspan
497 497
498 498 atts
499 499 end
500 500
501 501 TABLE_RE = /^(?:table(_?#{S}#{A}#{C})\. ?\n)?^(#{A}#{C}\.? ?\|.*?\|)(\n\n|\Z)/m
502 502
503 503 # Parses a Textile table block, building HTML from the result.
504 504 def block_textile_table( text )
505 505 text.gsub!( TABLE_RE ) do |matches|
506 506
507 507 tatts, fullrow = $~[1..2]
508 508 tatts = pba( tatts, 'table' )
509 509 tatts = shelve( tatts ) if tatts
510 510 rows = []
511 511
512 512 fullrow.each_line do |row|
513 513 ratts, row = pba( $1, 'tr' ), $2 if row =~ /^(#{A}#{C}\. )(.*)/m
514 514 cells = []
515 515 row.split( /(\|)(?![^\[\|]*\]\])/ )[1..-2].each do |cell|
516 516 next if cell == '|'
517 517 ctyp = 'd'
518 518 ctyp = 'h' if cell =~ /^_/
519 519
520 520 catts = ''
521 521 catts, cell = pba( $1, 'td' ), $2 if cell =~ /^(_?#{S}#{A}#{C}\. ?)(.*)/
522 522
523 523 catts = shelve( catts ) if catts
524 524 cells << "\t\t\t<t#{ ctyp }#{ catts }>#{ cell }</t#{ ctyp }>"
525 525 end
526 526 ratts = shelve( ratts ) if ratts
527 527 rows << "\t\t<tr#{ ratts }>\n#{ cells.join( "\n" ) }\n\t\t</tr>"
528 528 end
529 529 "\t<table#{ tatts }>\n#{ rows.join( "\n" ) }\n\t</table>\n\n"
530 530 end
531 531 end
532 532
533 533 LISTS_RE = /^([#*]+?#{C} .*?)$(?![^#*])/m
534 534 LISTS_CONTENT_RE = /^([#*]+)(#{A}#{C}) (.*)$/m
535 535
536 536 # Parses Textile lists and generates HTML
537 537 def block_textile_lists( text )
538 538 text.gsub!( LISTS_RE ) do |match|
539 539 lines = match.split( /\n/ )
540 540 last_line = -1
541 541 depth = []
542 542 lines.each_with_index do |line, line_id|
543 543 if line =~ LISTS_CONTENT_RE
544 544 tl,atts,content = $~[1..3]
545 545 if depth.last
546 546 if depth.last.length > tl.length
547 547 (depth.length - 1).downto(0) do |i|
548 548 break if depth[i].length == tl.length
549 549 lines[line_id - 1] << "</li>\n\t</#{ lT( depth[i] ) }l>\n\t"
550 550 depth.pop
551 551 end
552 552 end
553 553 if depth.last and depth.last.length == tl.length
554 554 lines[line_id - 1] << '</li>'
555 555 end
556 556 end
557 557 unless depth.last == tl
558 558 depth << tl
559 559 atts = pba( atts )
560 560 atts = shelve( atts ) if atts
561 561 lines[line_id] = "\t<#{ lT(tl) }l#{ atts }>\n\t<li>#{ content }"
562 562 else
563 563 lines[line_id] = "\t\t<li>#{ content }"
564 564 end
565 565 last_line = line_id
566 566
567 567 else
568 568 last_line = line_id
569 569 end
570 570 if line_id - last_line > 1 or line_id == lines.length - 1
571 571 depth.delete_if do |v|
572 572 lines[last_line] << "</li>\n\t</#{ lT( v ) }l>"
573 573 end
574 574 end
575 575 end
576 576 lines.join( "\n" )
577 577 end
578 578 end
579 579
580 580 QUOTES_RE = /(^>+([^\n]*?)(\n|$))+/m
581 581 QUOTES_CONTENT_RE = /^([> ]+)(.*)$/m
582 582
583 583 def block_textile_quotes( text )
584 584 text.gsub!( QUOTES_RE ) do |match|
585 585 lines = match.split( /\n/ )
586 586 quotes = ''
587 587 indent = 0
588 588 lines.each do |line|
589 589 line =~ QUOTES_CONTENT_RE
590 590 bq,content = $1, $2
591 591 l = bq.count('>')
592 592 if l != indent
593 593 quotes << ("\n\n" + (l>indent ? '<blockquote>' * (l-indent) : '</blockquote>' * (indent-l)) + "\n\n")
594 594 indent = l
595 595 end
596 596 quotes << (content + "\n")
597 597 end
598 598 quotes << ("\n" + '</blockquote>' * indent + "\n\n")
599 599 quotes
600 600 end
601 601 end
602 602
603 603 CODE_RE = /(\W)
604 604 @
605 605 (?:\|(\w+?)\|)?
606 606 (.+?)
607 607 @
608 608 (?=\W)/x
609 609
610 610 def inline_textile_code( text )
611 611 text.gsub!( CODE_RE ) do |m|
612 612 before,lang,code,after = $~[1..4]
613 613 lang = " lang=\"#{ lang }\"" if lang
614 614 rip_offtags( "#{ before }<code#{ lang }>#{ code }</code>#{ after }" )
615 615 end
616 616 end
617 617
618 618 def lT( text )
619 619 text =~ /\#$/ ? 'o' : 'u'
620 620 end
621 621
622 622 def hard_break( text )
623 623 text.gsub!( /(.)\n(?!\Z| *([#*=]+(\s|$)|[{|]))/, "\\1<br />" ) if hard_breaks
624 624 end
625 625
626 626 BLOCKS_GROUP_RE = /\n{2,}(?! )/m
627 627
628 628 def blocks( text, deep_code = false )
629 629 text.replace( text.split( BLOCKS_GROUP_RE ).collect do |blk|
630 630 plain = blk !~ /\A[#*> ]/
631 631
632 632 # skip blocks that are complex HTML
633 633 if blk =~ /^<\/?(\w+).*>/ and not SIMPLE_HTML_TAGS.include? $1
634 634 blk
635 635 else
636 636 # search for indentation levels
637 637 blk.strip!
638 638 if blk.empty?
639 639 blk
640 640 else
641 641 code_blk = nil
642 642 blk.gsub!( /((?:\n(?:\n^ +[^\n]*)+)+)/m ) do |iblk|
643 643 flush_left iblk
644 644 blocks iblk, plain
645 645 iblk.gsub( /^(\S)/, "\t\\1" )
646 646 if plain
647 647 code_blk = iblk; ""
648 648 else
649 649 iblk
650 650 end
651 651 end
652 652
653 653 block_applied = 0
654 654 @rules.each do |rule_name|
655 655 block_applied += 1 if ( rule_name.to_s.match /^block_/ and method( rule_name ).call( blk ) )
656 656 end
657 657 if block_applied.zero?
658 658 if deep_code
659 659 blk = "\t<pre><code>#{ blk }</code></pre>"
660 660 else
661 661 blk = "\t<p>#{ blk }</p>"
662 662 end
663 663 end
664 664 # hard_break blk
665 665 blk + "\n#{ code_blk }"
666 666 end
667 667 end
668 668
669 669 end.join( "\n\n" ) )
670 670 end
671 671
672 672 def textile_bq( tag, atts, cite, content )
673 673 cite, cite_title = check_refs( cite )
674 674 cite = " cite=\"#{ cite }\"" if cite
675 675 atts = shelve( atts ) if atts
676 676 "\t<blockquote#{ cite }>\n\t\t<p#{ atts }>#{ content }</p>\n\t</blockquote>"
677 677 end
678 678
679 679 def textile_p( tag, atts, cite, content )
680 680 atts = shelve( atts ) if atts
681 681 "\t<#{ tag }#{ atts }>#{ content }</#{ tag }>"
682 682 end
683 683
684 684 alias textile_h1 textile_p
685 685 alias textile_h2 textile_p
686 686 alias textile_h3 textile_p
687 687 alias textile_h4 textile_p
688 688 alias textile_h5 textile_p
689 689 alias textile_h6 textile_p
690 690
691 691 def textile_fn_( tag, num, atts, cite, content )
692 692 atts << " id=\"fn#{ num }\" class=\"footnote\""
693 693 content = "<sup>#{ num }</sup> #{ content }"
694 694 atts = shelve( atts ) if atts
695 695 "\t<p#{ atts }>#{ content }</p>"
696 696 end
697 697
698 698 BLOCK_RE = /^(([a-z]+)(\d*))(#{A}#{C})\.(?::(\S+))? (.*)$/m
699 699
700 700 def block_textile_prefix( text )
701 701 if text =~ BLOCK_RE
702 702 tag,tagpre,num,atts,cite,content = $~[1..6]
703 703 atts = pba( atts )
704 704
705 705 # pass to prefix handler
706 706 if respond_to? "textile_#{ tag }", true
707 707 text.gsub!( $&, method( "textile_#{ tag }" ).call( tag, atts, cite, content ) )
708 708 elsif respond_to? "textile_#{ tagpre }_", true
709 709 text.gsub!( $&, method( "textile_#{ tagpre }_" ).call( tagpre, num, atts, cite, content ) )
710 710 end
711 711 end
712 712 end
713 713
714 714 SETEXT_RE = /\A(.+?)\n([=-])[=-]* *$/m
715 715 def block_markdown_setext( text )
716 716 if text =~ SETEXT_RE
717 717 tag = if $2 == "="; "h1"; else; "h2"; end
718 718 blk, cont = "<#{ tag }>#{ $1 }</#{ tag }>", $'
719 719 blocks cont
720 720 text.replace( blk + cont )
721 721 end
722 722 end
723 723
724 724 ATX_RE = /\A(\#{1,6}) # $1 = string of #'s
725 725 [ ]*
726 726 (.+?) # $2 = Header text
727 727 [ ]*
728 728 \#* # optional closing #'s (not counted)
729 729 $/x
730 730 def block_markdown_atx( text )
731 731 if text =~ ATX_RE
732 732 tag = "h#{ $1.length }"
733 733 blk, cont = "<#{ tag }>#{ $2 }</#{ tag }>\n\n", $'
734 734 blocks cont
735 735 text.replace( blk + cont )
736 736 end
737 737 end
738 738
739 739 MARKDOWN_BQ_RE = /\A(^ *> ?.+$(.+\n)*\n*)+/m
740 740
741 741 def block_markdown_bq( text )
742 742 text.gsub!( MARKDOWN_BQ_RE ) do |blk|
743 743 blk.gsub!( /^ *> ?/, '' )
744 744 flush_left blk
745 745 blocks blk
746 746 blk.gsub!( /^(\S)/, "\t\\1" )
747 747 "<blockquote>\n#{ blk }\n</blockquote>\n\n"
748 748 end
749 749 end
750 750
751 751 MARKDOWN_RULE_RE = /^(#{
752 752 ['*', '-', '_'].collect { |ch| ' ?(' + Regexp::quote( ch ) + ' ?){3,}' }.join( '|' )
753 753 })$/
754 754
755 755 def block_markdown_rule( text )
756 756 text.gsub!( MARKDOWN_RULE_RE ) do |blk|
757 757 "<hr />"
758 758 end
759 759 end
760 760
761 761 # XXX TODO XXX
762 762 def block_markdown_lists( text )
763 763 end
764 764
765 765 def inline_textile_span( text )
766 766 QTAGS.each do |qtag_rc, ht, qtag_re, rtype|
767 767 text.gsub!( qtag_re ) do |m|
768 768
769 769 case rtype
770 770 when :limit
771 771 sta,qtag,atts,cite,content = $~[1..5]
772 772 else
773 773 qtag,atts,cite,content = $~[1..4]
774 774 sta = ''
775 775 end
776 776 atts = pba( atts )
777 777 atts << " cite=\"#{ cite }\"" if cite
778 778 atts = shelve( atts ) if atts
779 779
780 780 "#{ sta }<#{ ht }#{ atts }>#{ content }</#{ ht }>"
781 781
782 782 end
783 783 end
784 784 end
785 785
786 786 LINK_RE = /
787 787 (
788 788 ([\s\[{(]|[#{PUNCT}])? # $pre
789 789 " # start
790 790 (#{C}) # $atts
791 791 ([^"\n]+?) # $text
792 792 \s?
793 793 (?:\(([^)]+?)\)(?="))? # $title
794 794 ":
795 795 ( # $url
796 796 (\/|[a-zA-Z]+:\/\/|www\.|mailto:) # $proto
797 797 [\w\/]\S+?
798 798 )
799 799 (\/)? # $slash
800 800 ([^\w\=\/;\(\)]*?) # $post
801 801 )
802 802 (?=<|\s|$)
803 803 /x
804 804 #"
805 805 def inline_textile_link( text )
806 806 text.gsub!( LINK_RE ) do |m|
807 807 all,pre,atts,text,title,url,proto,slash,post = $~[1..9]
808 808 if text.include?('<br />')
809 809 all
810 810 else
811 811 url, url_title = check_refs( url )
812 812 title ||= url_title
813 813
814 814 # Idea below : an URL with unbalanced parethesis and
815 815 # ending by ')' is put into external parenthesis
816 816 if ( url[-1]==?) and ((url.count("(") - url.count(")")) < 0 ) )
817 817 url=url[0..-2] # discard closing parenth from url
818 818 post = ")"+post # add closing parenth to post
819 819 end
820 820 atts = pba( atts )
821 821 atts = " href=\"#{ url }#{ slash }\"#{ atts }"
822 822 atts << " title=\"#{ htmlesc title }\"" if title
823 823 atts = shelve( atts ) if atts
824 824
825 825 external = (url =~ /^https?:\/\//) ? ' class="external"' : ''
826 826
827 827 "#{ pre }<a#{ atts }#{ external }>#{ text }</a>#{ post }"
828 828 end
829 829 end
830 830 end
831 831
832 832 MARKDOWN_REFLINK_RE = /
833 833 \[([^\[\]]+)\] # $text
834 834 [ ]? # opt. space
835 835 (?:\n[ ]*)? # one optional newline followed by spaces
836 836 \[(.*?)\] # $id
837 837 /x
838 838
839 839 def inline_markdown_reflink( text )
840 840 text.gsub!( MARKDOWN_REFLINK_RE ) do |m|
841 841 text, id = $~[1..2]
842 842
843 843 if id.empty?
844 844 url, title = check_refs( text )
845 845 else
846 846 url, title = check_refs( id )
847 847 end
848 848
849 849 atts = " href=\"#{ url }\""
850 850 atts << " title=\"#{ title }\"" if title
851 851 atts = shelve( atts )
852 852
853 853 "<a#{ atts }>#{ text }</a>"
854 854 end
855 855 end
856 856
857 857 MARKDOWN_LINK_RE = /
858 858 \[([^\[\]]+)\] # $text
859 859 \( # open paren
860 860 [ \t]* # opt space
861 861 <?(.+?)>? # $href
862 862 [ \t]* # opt space
863 863 (?: # whole title
864 864 (['"]) # $quote
865 865 (.*?) # $title
866 866 \3 # matching quote
867 867 )? # title is optional
868 868 \)
869 869 /x
870 870
871 871 def inline_markdown_link( text )
872 872 text.gsub!( MARKDOWN_LINK_RE ) do |m|
873 873 text, url, quote, title = $~[1..4]
874 874
875 875 atts = " href=\"#{ url }\""
876 876 atts << " title=\"#{ title }\"" if title
877 877 atts = shelve( atts )
878 878
879 879 "<a#{ atts }>#{ text }</a>"
880 880 end
881 881 end
882 882
883 883 TEXTILE_REFS_RE = /(^ *)\[([^\[\n]+?)\](#{HYPERLINK})(?=\s|$)/
884 884 MARKDOWN_REFS_RE = /(^ *)\[([^\n]+?)\]:\s+<?(#{HYPERLINK})>?(?:\s+"((?:[^"]|\\")+)")?(?=\s|$)/m
885 885
886 886 def refs( text )
887 887 @rules.each do |rule_name|
888 888 method( rule_name ).call( text ) if rule_name.to_s.match /^refs_/
889 889 end
890 890 end
891 891
892 892 def refs_textile( text )
893 893 text.gsub!( TEXTILE_REFS_RE ) do |m|
894 894 flag, url = $~[2..3]
895 895 @urlrefs[flag.downcase] = [url, nil]
896 896 nil
897 897 end
898 898 end
899 899
900 900 def refs_markdown( text )
901 901 text.gsub!( MARKDOWN_REFS_RE ) do |m|
902 902 flag, url = $~[2..3]
903 903 title = $~[6]
904 904 @urlrefs[flag.downcase] = [url, title]
905 905 nil
906 906 end
907 907 end
908 908
909 909 def check_refs( text )
910 910 ret = @urlrefs[text.downcase] if text
911 911 ret || [text, nil]
912 912 end
913 913
914 914 IMAGE_RE = /
915 915 (>|\s|^) # start of line?
916 916 \! # opening
917 917 (\<|\=|\>)? # optional alignment atts
918 918 (#{C}) # optional style,class atts
919 919 (?:\. )? # optional dot-space
920 920 ([^\s(!]+?) # presume this is the src
921 921 \s? # optional space
922 922 (?:\(((?:[^\(\)]|\([^\)]+\))+?)\))? # optional title
923 923 \! # closing
924 924 (?::#{ HYPERLINK })? # optional href
925 925 /x
926 926
927 927 def inline_textile_image( text )
928 928 text.gsub!( IMAGE_RE ) do |m|
929 929 stln,algn,atts,url,title,href,href_a1,href_a2 = $~[1..8]
930 930 htmlesc title
931 931 atts = pba( atts )
932 932 atts = " src=\"#{ url }\"#{ atts }"
933 933 atts << " title=\"#{ title }\"" if title
934 934 atts << " alt=\"#{ title }\""
935 935 # size = @getimagesize($url);
936 936 # if($size) $atts.= " $size[3]";
937 937
938 938 href, alt_title = check_refs( href ) if href
939 939 url, url_title = check_refs( url )
940 940
941 941 out = ''
942 942 out << "<a#{ shelve( " href=\"#{ href }\"" ) }>" if href
943 943 out << "<img#{ shelve( atts ) } />"
944 944 out << "</a>#{ href_a1 }#{ href_a2 }" if href
945 945
946 946 if algn
947 947 algn = h_align( algn )
948 948 if stln == "<p>"
949 949 out = "<p style=\"float:#{ algn }\">#{ out }"
950 950 else
951 951 out = "#{ stln }<div style=\"float:#{ algn }\">#{ out }</div>"
952 952 end
953 953 else
954 954 out = stln + out
955 955 end
956 956
957 957 out
958 958 end
959 959 end
960 960
961 961 def shelve( val )
962 962 @shelf << val
963 963 " :redsh##{ @shelf.length }:"
964 964 end
965 965
966 966 def retrieve( text )
967 967 @shelf.each_with_index do |r, i|
968 968 text.gsub!( " :redsh##{ i + 1 }:", r )
969 969 end
970 970 end
971 971
972 972 def incoming_entities( text )
973 973 ## turn any incoming ampersands into a dummy character for now.
974 974 ## This uses a negative lookahead for alphanumerics followed by a semicolon,
975 975 ## implying an incoming html entity, to be skipped
976 976
977 977 text.gsub!( /&(?![#a-z0-9]+;)/i, "x%x%" )
978 978 end
979 979
980 980 def no_textile( text )
981 981 text.gsub!( /(^|\s)==([^=]+.*?)==(\s|$)?/,
982 982 '\1<notextile>\2</notextile>\3' )
983 983 text.gsub!( /^ *==([^=]+.*?)==/m,
984 984 '\1<notextile>\2</notextile>\3' )
985 985 end
986 986
987 987 def clean_white_space( text )
988 988 # normalize line breaks
989 989 text.gsub!( /\r\n/, "\n" )
990 990 text.gsub!( /\r/, "\n" )
991 991 text.gsub!( /\t/, ' ' )
992 992 text.gsub!( /^ +$/, '' )
993 993 text.gsub!( /\n{3,}/, "\n\n" )
994 994 text.gsub!( /"$/, "\" " )
995 995
996 996 # if entire document is indented, flush
997 997 # to the left side
998 998 flush_left text
999 999 end
1000 1000
1001 1001 def flush_left( text )
1002 1002 indt = 0
1003 1003 if text =~ /^ /
1004 1004 while text !~ /^ {#{indt}}\S/
1005 1005 indt += 1
1006 1006 end unless text.empty?
1007 1007 if indt.nonzero?
1008 1008 text.gsub!( /^ {#{indt}}/, '' )
1009 1009 end
1010 1010 end
1011 1011 end
1012 1012
1013 1013 def footnote_ref( text )
1014 1014 text.gsub!( /\b\[([0-9]+?)\](\s)?/,
1015 1015 '<sup><a href="#fn\1">\1</a></sup>\2' )
1016 1016 end
1017 1017
1018 1018 OFFTAGS = /(code|pre|kbd|notextile)/
1019 OFFTAG_MATCH = /(?:(<\/#{ OFFTAGS }>)|(<#{ OFFTAGS }[^>]*>))(.*?)(?=<\/?#{ OFFTAGS }|\Z)/mi
1019 OFFTAG_MATCH = /(?:(<\/#{ OFFTAGS }>)|(<#{ OFFTAGS }[^>]*>))(.*?)(?=<\/?#{ OFFTAGS }\W|\Z)/mi
1020 1020 OFFTAG_OPEN = /<#{ OFFTAGS }/
1021 1021 OFFTAG_CLOSE = /<\/?#{ OFFTAGS }/
1022 1022 HASTAG_MATCH = /(<\/?\w[^\n]*?>)/m
1023 1023 ALLTAG_MATCH = /(<\/?\w[^\n]*?>)|.*?(?=<\/?\w[^\n]*?>|$)/m
1024 1024
1025 1025 def glyphs_textile( text, level = 0 )
1026 1026 if text !~ HASTAG_MATCH
1027 1027 pgl text
1028 1028 footnote_ref text
1029 1029 else
1030 1030 codepre = 0
1031 1031 text.gsub!( ALLTAG_MATCH ) do |line|
1032 1032 ## matches are off if we're between <code>, <pre> etc.
1033 1033 if $1
1034 1034 if line =~ OFFTAG_OPEN
1035 1035 codepre += 1
1036 1036 elsif line =~ OFFTAG_CLOSE
1037 1037 codepre -= 1
1038 1038 codepre = 0 if codepre < 0
1039 1039 end
1040 1040 elsif codepre.zero?
1041 1041 glyphs_textile( line, level + 1 )
1042 1042 else
1043 1043 htmlesc( line, :NoQuotes )
1044 1044 end
1045 1045 # p [level, codepre, line]
1046 1046
1047 1047 line
1048 1048 end
1049 1049 end
1050 1050 end
1051 1051
1052 1052 def rip_offtags( text )
1053 1053 if text =~ /<.*>/
1054 1054 ## strip and encode <pre> content
1055 1055 codepre, used_offtags = 0, {}
1056 1056 text.gsub!( OFFTAG_MATCH ) do |line|
1057 1057 if $3
1058 1058 offtag, aftertag = $4, $5
1059 1059 codepre += 1
1060 1060 used_offtags[offtag] = true
1061 1061 if codepre - used_offtags.length > 0
1062 1062 htmlesc( line, :NoQuotes )
1063 1063 @pre_list.last << line
1064 1064 line = ""
1065 1065 else
1066 1066 htmlesc( aftertag, :NoQuotes ) if aftertag
1067 1067 line = "<redpre##{ @pre_list.length }>"
1068 1068 $3.match(/<#{ OFFTAGS }([^>]*)>/)
1069 1069 tag = $1
1070 1070 $2.to_s.match(/(class\=\S+)/i)
1071 1071 tag << " #{$1}" if $1
1072 1072 @pre_list << "<#{ tag }>#{ aftertag }"
1073 1073 end
1074 1074 elsif $1 and codepre > 0
1075 1075 if codepre - used_offtags.length > 0
1076 1076 htmlesc( line, :NoQuotes )
1077 1077 @pre_list.last << line
1078 1078 line = ""
1079 1079 end
1080 1080 codepre -= 1 unless codepre.zero?
1081 1081 used_offtags = {} if codepre.zero?
1082 1082 end
1083 1083 line
1084 1084 end
1085 1085 end
1086 1086 text
1087 1087 end
1088 1088
1089 1089 def smooth_offtags( text )
1090 1090 unless @pre_list.empty?
1091 1091 ## replace <pre> content
1092 1092 text.gsub!( /<redpre#(\d+)>/ ) { @pre_list[$1.to_i] }
1093 1093 end
1094 1094 end
1095 1095
1096 1096 def inline( text )
1097 1097 [/^inline_/, /^glyphs_/].each do |meth_re|
1098 1098 @rules.each do |rule_name|
1099 1099 method( rule_name ).call( text ) if rule_name.to_s.match( meth_re )
1100 1100 end
1101 1101 end
1102 1102 end
1103 1103
1104 1104 def h_align( text )
1105 1105 H_ALGN_VALS[text]
1106 1106 end
1107 1107
1108 1108 def v_align( text )
1109 1109 V_ALGN_VALS[text]
1110 1110 end
1111 1111
1112 1112 def textile_popup_help( name, windowW, windowH )
1113 1113 ' <a target="_blank" href="http://hobix.com/textile/#' + helpvar + '" onclick="window.open(this.href, \'popupwindow\', \'width=' + windowW + ',height=' + windowH + ',scrollbars,resizable\'); return false;">' + name + '</a><br />'
1114 1114 end
1115 1115
1116 1116 # HTML cleansing stuff
1117 1117 BASIC_TAGS = {
1118 1118 'a' => ['href', 'title'],
1119 1119 'img' => ['src', 'alt', 'title'],
1120 1120 'br' => [],
1121 1121 'i' => nil,
1122 1122 'u' => nil,
1123 1123 'b' => nil,
1124 1124 'pre' => nil,
1125 1125 'kbd' => nil,
1126 1126 'code' => ['lang'],
1127 1127 'cite' => nil,
1128 1128 'strong' => nil,
1129 1129 'em' => nil,
1130 1130 'ins' => nil,
1131 1131 'sup' => nil,
1132 1132 'sub' => nil,
1133 1133 'del' => nil,
1134 1134 'table' => nil,
1135 1135 'tr' => nil,
1136 1136 'td' => ['colspan', 'rowspan'],
1137 1137 'th' => nil,
1138 1138 'ol' => nil,
1139 1139 'ul' => nil,
1140 1140 'li' => nil,
1141 1141 'p' => nil,
1142 1142 'h1' => nil,
1143 1143 'h2' => nil,
1144 1144 'h3' => nil,
1145 1145 'h4' => nil,
1146 1146 'h5' => nil,
1147 1147 'h6' => nil,
1148 1148 'blockquote' => ['cite']
1149 1149 }
1150 1150
1151 1151 def clean_html( text, tags = BASIC_TAGS )
1152 1152 text.gsub!( /<!\[CDATA\[/, '' )
1153 1153 text.gsub!( /<(\/*)(\w+)([^>]*)>/ ) do
1154 1154 raw = $~
1155 1155 tag = raw[2].downcase
1156 1156 if tags.has_key? tag
1157 1157 pcs = [tag]
1158 1158 tags[tag].each do |prop|
1159 1159 ['"', "'", ''].each do |q|
1160 1160 q2 = ( q != '' ? q : '\s' )
1161 1161 if raw[3] =~ /#{prop}\s*=\s*#{q}([^#{q2}]+)#{q}/i
1162 1162 attrv = $1
1163 1163 next if prop == 'src' and attrv =~ %r{^(?!http)\w+:}
1164 1164 pcs << "#{prop}=\"#{$1.gsub('"', '\\"')}\""
1165 1165 break
1166 1166 end
1167 1167 end
1168 1168 end if tags[tag]
1169 1169 "<#{raw[1]}#{pcs.join " "}>"
1170 1170 else
1171 1171 " "
1172 1172 end
1173 1173 end
1174 1174 end
1175 1175
1176 1176 ALLOWED_TAGS = %w(redpre pre code notextile)
1177 1177
1178 1178 def escape_html_tags(text)
1179 1179 text.gsub!(%r{<(\/?([!\w]+)[^<>\n]*)(>?)}) {|m| ALLOWED_TAGS.include?($2) ? "<#{$1}#{$3}" : "&lt;#{$1}#{'&gt;' unless $3.blank?}" }
1180 1180 end
1181 1181 end
1182 1182
@@ -1,506 +1,528
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 def test_pre_tags
262 raw = <<-RAW
263 Before
264
265 <pre>
266 <prepared-statement-cache-size>32</prepared-statement-cache-size>
267 </pre>
268
269 After
270 RAW
271
272 expected = <<-EXPECTED
273 <p>Before</p>
274 <pre>
275 &lt;prepared-statement-cache-size&gt;32&lt;/prepared-statement-cache-size&gt;
276 </pre>
277 <p>After</p>
278 EXPECTED
279
280 assert_equal expected.gsub(%r{[\r\n\t]}, ''), textilizable(raw).gsub(%r{[\r\n\t]}, '')
281 end
282
261 283 def test_syntax_highlight
262 284 raw = <<-RAW
263 285 <pre><code class="ruby">
264 286 # Some ruby code here
265 287 </pre></code>
266 288 RAW
267 289
268 290 expected = <<-EXPECTED
269 291 <pre><code class="ruby CodeRay"><span class="no">1</span> <span class="c"># Some ruby code here</span>
270 292 </pre></code>
271 293 EXPECTED
272 294
273 295 assert_equal expected.gsub(%r{[\r\n\t]}, ''), textilizable(raw).gsub(%r{[\r\n\t]}, '')
274 296 end
275 297
276 298 def test_wiki_links_in_tables
277 299 to_test = {"|[[Page|Link title]]|[[Other Page|Other title]]|\n|Cell 21|[[Last page]]|" =>
278 300 '<tr><td><a href="/projects/ecookbook/wiki/Page" class="wiki-page new">Link title</a></td>' +
279 301 '<td><a href="/projects/ecookbook/wiki/Other_Page" class="wiki-page new">Other title</a></td>' +
280 302 '</tr><tr><td>Cell 21</td><td><a href="/projects/ecookbook/wiki/Last_page" class="wiki-page new">Last page</a></td></tr>'
281 303 }
282 304 @project = Project.find(1)
283 305 to_test.each { |text, result| assert_equal "<table>#{result}</table>", textilizable(text).gsub(/[\t\n]/, '') }
284 306 end
285 307
286 308 def test_text_formatting
287 309 to_test = {'*_+bold, italic and underline+_*' => '<strong><em><ins>bold, italic and underline</ins></em></strong>',
288 310 '(_text within parentheses_)' => '(<em>text within parentheses</em>)',
289 311 'a *Humane Web* Text Generator' => 'a <strong>Humane Web</strong> Text Generator',
290 312 '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 313 '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 314 }
293 315 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
294 316 end
295 317
296 318 def test_wiki_horizontal_rule
297 319 assert_equal '<hr />', textilizable('---')
298 320 assert_equal '<p>Dashes: ---</p>', textilizable('Dashes: ---')
299 321 end
300 322
301 323 def test_acronym
302 324 assert_equal '<p>This is an acronym: <acronym title="American Civil Liberties Union">ACLU</acronym>.</p>',
303 325 textilizable('This is an acronym: ACLU(American Civil Liberties Union).')
304 326 end
305 327
306 328 def test_footnotes
307 329 raw = <<-RAW
308 330 This is some text[1].
309 331
310 332 fn1. This is the foot note
311 333 RAW
312 334
313 335 expected = <<-EXPECTED
314 336 <p>This is some text<sup><a href=\"#fn1\">1</a></sup>.</p>
315 337 <p id="fn1" class="footnote"><sup>1</sup> This is the foot note</p>
316 338 EXPECTED
317 339
318 340 assert_equal expected.gsub(%r{[\r\n\t]}, ''), textilizable(raw).gsub(%r{[\r\n\t]}, '')
319 341 end
320 342
321 343 def test_table_of_content
322 344 raw = <<-RAW
323 345 {{toc}}
324 346
325 347 h1. Title
326 348
327 349 Lorem ipsum dolor sit amet, consectetuer adipiscing elit. Maecenas sed libero.
328 350
329 351 h2. Subtitle with a [[Wiki]] link
330 352
331 353 Nullam commodo metus accumsan nulla. Curabitur lobortis dui id dolor.
332 354
333 355 h2. Subtitle with [[Wiki|another Wiki]] link
334 356
335 357 h2. Subtitle with %{color:red}red text%
336 358
337 359 h1. Another title
338 360
339 361 RAW
340 362
341 363 expected = '<ul class="toc">' +
342 364 '<li class="heading1"><a href="#Title">Title</a></li>' +
343 365 '<li class="heading2"><a href="#Subtitle-with-a-Wiki-link">Subtitle with a Wiki link</a></li>' +
344 366 '<li class="heading2"><a href="#Subtitle-with-another-Wiki-link">Subtitle with another Wiki link</a></li>' +
345 367 '<li class="heading2"><a href="#Subtitle-with-red-text">Subtitle with red text</a></li>' +
346 368 '<li class="heading1"><a href="#Another-title">Another title</a></li>' +
347 369 '</ul>'
348 370
349 371 assert textilizable(raw).gsub("\n", "").include?(expected)
350 372 end
351 373
352 374 def test_blockquote
353 375 # orig raw text
354 376 raw = <<-RAW
355 377 John said:
356 378 > Lorem ipsum dolor sit amet, consectetuer adipiscing elit. Maecenas sed libero.
357 379 > Nullam commodo metus accumsan nulla. Curabitur lobortis dui id dolor.
358 380 > * Donec odio lorem,
359 381 > * sagittis ac,
360 382 > * malesuada in,
361 383 > * adipiscing eu, dolor.
362 384 >
363 385 > >Nulla varius pulvinar diam. Proin id arcu id lorem scelerisque condimentum. Proin vehicula turpis vitae lacus.
364 386 > Proin a tellus. Nam vel neque.
365 387
366 388 He's right.
367 389 RAW
368 390
369 391 # expected html
370 392 expected = <<-EXPECTED
371 393 <p>John said:</p>
372 394 <blockquote>
373 395 Lorem ipsum dolor sit amet, consectetuer adipiscing elit. Maecenas sed libero.
374 396 Nullam commodo metus accumsan nulla. Curabitur lobortis dui id dolor.
375 397 <ul>
376 398 <li>Donec odio lorem,</li>
377 399 <li>sagittis ac,</li>
378 400 <li>malesuada in,</li>
379 401 <li>adipiscing eu, dolor.</li>
380 402 </ul>
381 403 <blockquote>
382 404 <p>Nulla varius pulvinar diam. Proin id arcu id lorem scelerisque condimentum. Proin vehicula turpis vitae lacus.</p>
383 405 </blockquote>
384 406 <p>Proin a tellus. Nam vel neque.</p>
385 407 </blockquote>
386 408 <p>He's right.</p>
387 409 EXPECTED
388 410
389 411 assert_equal expected.gsub(%r{\s+}, ''), textilizable(raw).gsub(%r{\s+}, '')
390 412 end
391 413
392 414 def test_table
393 415 raw = <<-RAW
394 416 This is a table with empty cells:
395 417
396 418 |cell11|cell12||
397 419 |cell21||cell23|
398 420 |cell31|cell32|cell33|
399 421 RAW
400 422
401 423 expected = <<-EXPECTED
402 424 <p>This is a table with empty cells:</p>
403 425
404 426 <table>
405 427 <tr><td>cell11</td><td>cell12</td><td></td></tr>
406 428 <tr><td>cell21</td><td></td><td>cell23</td></tr>
407 429 <tr><td>cell31</td><td>cell32</td><td>cell33</td></tr>
408 430 </table>
409 431 EXPECTED
410 432
411 433 assert_equal expected.gsub(%r{\s+}, ''), textilizable(raw).gsub(%r{\s+}, '')
412 434 end
413 435
414 436 def test_table_with_line_breaks
415 437 raw = <<-RAW
416 438 This is a table with line breaks:
417 439
418 440 |cell11
419 441 continued|cell12||
420 442 |-cell21-||cell23
421 443 cell23 line2
422 444 cell23 *line3*|
423 445 |cell31|cell32
424 446 cell32 line2|cell33|
425 447
426 448 RAW
427 449
428 450 expected = <<-EXPECTED
429 451 <p>This is a table with line breaks:</p>
430 452
431 453 <table>
432 454 <tr>
433 455 <td>cell11<br />continued</td>
434 456 <td>cell12</td>
435 457 <td></td>
436 458 </tr>
437 459 <tr>
438 460 <td><del>cell21</del></td>
439 461 <td></td>
440 462 <td>cell23<br/>cell23 line2<br/>cell23 <strong>line3</strong></td>
441 463 </tr>
442 464 <tr>
443 465 <td>cell31</td>
444 466 <td>cell32<br/>cell32 line2</td>
445 467 <td>cell33</td>
446 468 </tr>
447 469 </table>
448 470 EXPECTED
449 471
450 472 assert_equal expected.gsub(%r{\s+}, ''), textilizable(raw).gsub(%r{\s+}, '')
451 473 end
452 474
453 475 def test_default_formatter
454 476 Setting.text_formatting = 'unknown'
455 477 text = 'a *link*: http://www.example.net/'
456 478 assert_equal '<p>a *link*: <a href="http://www.example.net/">http://www.example.net/</a></p>', textilizable(text)
457 479 Setting.text_formatting = 'textile'
458 480 end
459 481
460 482 def test_due_date_distance_in_words
461 483 to_test = { Date.today => 'Due in 0 days',
462 484 Date.today + 1 => 'Due in 1 day',
463 485 Date.today + 100 => 'Due in about 3 months',
464 486 Date.today + 20000 => 'Due in over 54 years',
465 487 Date.today - 1 => '1 day late',
466 488 Date.today - 100 => 'about 3 months late',
467 489 Date.today - 20000 => 'over 54 years late',
468 490 }
469 491 to_test.each do |date, expected|
470 492 assert_equal expected, due_date_distance_in_words(date)
471 493 end
472 494 end
473 495
474 496 def test_avatar
475 497 # turn on avatars
476 498 Setting.gravatar_enabled = '1'
477 499 assert avatar(User.find_by_mail('jsmith@somenet.foo')).include?(Digest::MD5.hexdigest('jsmith@somenet.foo'))
478 500 assert avatar('jsmith <jsmith@somenet.foo>').include?(Digest::MD5.hexdigest('jsmith@somenet.foo'))
479 501 assert_nil avatar('jsmith')
480 502 assert_nil avatar(nil)
481 503
482 504 # turn off avatars
483 505 Setting.gravatar_enabled = '0'
484 506 assert_nil avatar(User.find_by_mail('jsmith@somenet.foo'))
485 507 end
486 508
487 509 def test_link_to_user
488 510 user = User.find(2)
489 511 t = link_to_user(user)
490 512 assert_equal "<a href=\"/users/2\">#{ user.name }</a>", t
491 513 end
492 514
493 515 def test_link_to_user_should_not_link_to_locked_user
494 516 user = User.find(5)
495 517 assert user.locked?
496 518 t = link_to_user(user)
497 519 assert_equal user.name, t
498 520 end
499 521
500 522 def test_link_to_user_should_not_link_to_anonymous
501 523 user = User.anonymous
502 524 assert user.anonymous?
503 525 t = link_to_user(user)
504 526 assert_equal ::I18n.t(:label_user_anonymous), t
505 527 end
506 528 end
General Comments 0
You need to be logged in to leave comments. Login now