프로젝트

일반

사용자정보

통계
| 브랜치(Branch): | 개정판:

markus / MarkusAutoUpdate / src / NetSparkle / Libraries / MarkdownSharp.cs @ 38d69491

이력 | 보기 | 이력해설 | 다운로드 (69.7 KB)

1
/*
2
 * MarkdownSharp
3
 * -------------
4
 * a C# Markdown processor
5
 *
6
 * Markdown is a text-to-HTML conversion tool for web writers
7
 * Copyright (c) 2004 John Gruber
8
 * http://daringfireball.net/projects/markdown/
9
 *
10
 * Markdown.NET
11
 * Copyright (c) 2004-2009 Milan Negovan
12
 * http://www.aspnetresources.com
13
 * http://aspnetresources.com/blog/markdown_announced.aspx
14
 *
15
 * MarkdownSharp
16
 * Copyright (c) 2009-2011 Jeff Atwood
17
 * http://stackoverflow.com
18
 * http://www.codinghorror.com/blog/
19
 * http://code.google.com/p/markdownsharp/
20
 *
21
 * History: Milan ported the Markdown processor to C#. He granted license to me so I can open source it
22
 * and let the community contribute to and improve MarkdownSharp.
23
 *
24
 */
25

    
26
#region Copyright and license
27

    
28
/*
29

    
30
Copyright (c) 2009 - 2010 Jeff Atwood
31

    
32
http://www.opensource.org/licenses/mit-license.php
33
  
34
Permission is hereby granted, free of charge, to any person obtaining a copy
35
of this software and associated documentation files (the "Software"), to deal
36
in the Software without restriction, including without limitation the rights
37
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
38
copies of the Software, and to permit persons to whom the Software is
39
furnished to do so, subject to the following conditions:
40

    
41
The above copyright notice and this permission notice shall be included in
42
all copies or substantial portions of the Software.
43

    
44
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
45
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
46
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
47
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
48
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
49
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
50
THE SOFTWARE.
51

    
52
Copyright (c) 2003-2004 John Gruber
53
<http://daringfireball.net/>   
54
All rights reserved.
55

    
56
Redistribution and use in source and binary forms, with or without
57
modification, are permitted provided that the following conditions are
58
met:
59

    
60
* Redistributions of source code must retain the above copyright notice,
61
  this list of conditions and the following disclaimer.
62

    
63
* Redistributions in binary form must reproduce the above copyright
64
  notice, this list of conditions and the following disclaimer in the
65
  documentation and/or other materials provided with the distribution.
66

    
67
* Neither the name "Markdown" nor the names of its contributors may
68
  be used to endorse or promote products derived from this software
69
  without specific prior written permission.
70

    
71
This software is provided by the copyright holders and contributors "as
72
is" and any express or implied warranties, including, but not limited
73
to, the implied warranties of merchantability and fitness for a
74
particular purpose are disclaimed. In no event shall the copyright owner
75
or contributors be liable for any direct, indirect, incidental, special,
76
exemplary, or consequential damages (including, but not limited to,
77
procurement of substitute goods or services; loss of use, data, or
78
profits; or business interruption) however caused and on any theory of
79
liability, whether in contract, strict liability, or tort (including
80
negligence or otherwise) arising in any way out of the use of this
81
software, even if advised of the possibility of such damage.
82
*/
83

    
84
#endregion
85

    
86
using System;
87
using System.Collections.Generic;
88
using System.Configuration;
89
using System.Text;
90
using System.Text.RegularExpressions;
91

    
92
namespace MarkdownSharp
93
{
94

    
95
    /// <summary>
96
    /// 
97
    /// </summary>
98
    public class MarkdownOptions
99
    {
100
        /// <summary>
101
        /// when true, (most) bare plain URLs are auto-hyperlinked  
102
        /// WARNING: this is a significant deviation from the markdown spec
103
        /// </summary>
104
        public bool AutoHyperlink { get; set; }
105
        /// <summary>
106
        /// when true, RETURN becomes a literal newline  
107
        /// WARNING: this is a significant deviation from the markdown spec
108
        /// </summary>
109
        public bool AutoNewlines { get; set; }
110
        /// <summary>
111
        /// use ">" for HTML output, or " />" for XHTML output
112
        /// </summary>
113
        public string EmptyElementSuffix { get; set; }
114
        /// <summary>
115
        /// when true, problematic URL characters like [, ], (, and so forth will be encoded
116
        /// WARNING: this is a significant deviation from the markdown spec
117
        /// </summary>
118
        public bool EncodeProblemUrlCharacters { get; set; }
119
        /// <summary>
120
        /// when false, email addresses will never be auto-linked  
121
        /// WARNING: this is a significant deviation from the markdown spec
122
        /// </summary>
123
        public bool LinkEmails { get; set; }
124
        /// <summary>
125
        /// when true, bold and italic require non-word characters on either side  
126
        /// WARNING: this is a significant deviation from the markdown spec
127
        /// </summary>
128
        public bool StrictBoldItalic { get; set; }
129
    }
130

    
131

    
132
    /// <summary>
133
    /// Markdown is a text-to-HTML conversion tool for web writers.
134
    /// Markdown allows you to write using an easy-to-read, easy-to-write plain text format,
135
    /// then convert it to structurally valid XHTML (or HTML).
136
    /// </summary>
137
    public class Markdown
138
    {
139
        private const string _version = "1.13";
140

    
141
        #region Constructors and Options
142

    
143
        /// <summary>
144
        /// Create a new Markdown instance using default options
145
        /// </summary>
146
        public Markdown()
147
            : this(false)
148
        {
149
        }
150

    
151
        /// <summary>
152
        /// Create a new Markdown instance and optionally load options from a configuration
153
        /// file. There they should be stored in the appSettings section, available options are:
154
        ///
155
        ///     Markdown.StrictBoldItalic (true/false)
156
        ///     Markdown.EmptyElementSuffix (">" or " />" without the quotes)
157
        ///     Markdown.LinkEmails (true/false)
158
        ///     Markdown.AutoNewLines (true/false)
159
        ///     Markdown.AutoHyperlink (true/false)
160
        ///     Markdown.EncodeProblemUrlCharacters (true/false)
161
        ///     
162
        /// </summary>
163
        public Markdown(bool loadOptionsFromConfigFile)
164
        {
165
            if (!loadOptionsFromConfigFile) return;
166
//
167
//            var settings = ConfigurationManager.AppSettings;
168
//            foreach (string key in settings.Keys)
169
//            {
170
//                switch (key)
171
//                {
172
//                    case "Markdown.AutoHyperlink":
173
//                        _autoHyperlink = Convert.ToBoolean(settings[key]);
174
//                        break;
175
//                    case "Markdown.AutoNewlines":
176
//                        _autoNewlines = Convert.ToBoolean(settings[key]);
177
//                        break;
178
//                    case "Markdown.EmptyElementSuffix":
179
//                        _emptyElementSuffix = settings[key];
180
//                        break;
181
//                    case "Markdown.EncodeProblemUrlCharacters":
182
//                        _encodeProblemUrlCharacters = Convert.ToBoolean(settings[key]);
183
//                        break;
184
//                    case "Markdown.LinkEmails":
185
//                        _linkEmails = Convert.ToBoolean(settings[key]);
186
//                        break;
187
//                    case "Markdown.StrictBoldItalic":
188
//                        _strictBoldItalic = Convert.ToBoolean(settings[key]);
189
//                        break;
190
//                }
191
//            }
192
        }
193

    
194
        /// <summary>
195
        /// Create a new Markdown instance and set the options from the MarkdownOptions object.
196
        /// </summary>
197
        public Markdown(MarkdownOptions options)
198
        {
199
            _autoHyperlink = options.AutoHyperlink;
200
            _autoNewlines = options.AutoNewlines;
201
            _emptyElementSuffix = options.EmptyElementSuffix;
202
            _encodeProblemUrlCharacters = options.EncodeProblemUrlCharacters;
203
            _linkEmails = options.LinkEmails;
204
            _strictBoldItalic = options.StrictBoldItalic;
205
        }
206

    
207

    
208
        /// <summary>
209
        /// use ">" for HTML output, or " />" for XHTML output
210
        /// </summary>
211
        public string EmptyElementSuffix
212
        {
213
            get { return _emptyElementSuffix; }
214
            set { _emptyElementSuffix = value; }
215
        }
216
        private string _emptyElementSuffix = " />";
217

    
218
        /// <summary>
219
        /// when false, email addresses will never be auto-linked  
220
        /// WARNING: this is a significant deviation from the markdown spec
221
        /// </summary>
222
        public bool LinkEmails
223
        {
224
            get { return _linkEmails; }
225
            set { _linkEmails = value; }
226
        }
227
        private bool _linkEmails = true;
228

    
229
        /// <summary>
230
        /// when true, bold and italic require non-word characters on either side  
231
        /// WARNING: this is a significant deviation from the markdown spec
232
        /// </summary>
233
        public bool StrictBoldItalic
234
        {
235
            get { return _strictBoldItalic; }
236
            set { _strictBoldItalic = value; }
237
        }
238
        private bool _strictBoldItalic = false;
239

    
240
        /// <summary>
241
        /// when true, RETURN becomes a literal newline  
242
        /// WARNING: this is a significant deviation from the markdown spec
243
        /// </summary>
244
        public bool AutoNewLines
245
        {
246
            get { return _autoNewlines; }
247
            set { _autoNewlines = value; }
248
        }
249
        private bool _autoNewlines = false;
250

    
251
        /// <summary>
252
        /// when true, (most) bare plain URLs are auto-hyperlinked  
253
        /// WARNING: this is a significant deviation from the markdown spec
254
        /// </summary>
255
        public bool AutoHyperlink
256
        {
257
            get { return _autoHyperlink; }
258
            set { _autoHyperlink = value; }
259
        }
260
        private bool _autoHyperlink = false;
261

    
262
        /// <summary>
263
        /// when true, problematic URL characters like [, ], (, and so forth will be encoded
264
        /// WARNING: this is a significant deviation from the markdown spec
265
        /// </summary>
266
        public bool EncodeProblemUrlCharacters
267
        {
268
            get { return _encodeProblemUrlCharacters; }
269
            set { _encodeProblemUrlCharacters = value; }
270
        }
271
        private bool _encodeProblemUrlCharacters = false;
272

    
273
        #endregion
274

    
275
        private enum TokenType { Text, Tag }
276

    
277
        private struct Token
278
        {
279
            public Token(TokenType type, string value)
280
            {
281
                this.Type = type;
282
                this.Value = value;
283
            }
284
            public TokenType Type;
285
            public string Value;
286
        }
287

    
288
        /// <summary>
289
        /// maximum nested depth of [] and () supported by the transform; implementation detail
290
        /// </summary>
291
        private const int _nestDepth = 6;
292

    
293
        /// <summary>
294
        /// Tabs are automatically converted to spaces as part of the transform  
295
        /// this constant determines how "wide" those tabs become in spaces  
296
        /// </summary>
297
        private const int _tabWidth = 4;
298

    
299
        private const string _markerUL = @"[*+-]";
300
        private const string _markerOL = @"\d+[.]";
301

    
302
        private static readonly Dictionary<string, string> _escapeTable;
303
        private static readonly Dictionary<string, string> _invertedEscapeTable;
304
        private static readonly Dictionary<string, string> _backslashEscapeTable;
305

    
306
        private readonly Dictionary<string, string> _urls = new Dictionary<string, string>();
307
        private readonly Dictionary<string, string> _titles = new Dictionary<string, string>();
308
        private readonly Dictionary<string, string> _htmlBlocks = new Dictionary<string, string>();
309

    
310
        private int _listLevel;
311
        private static string AutoLinkPreventionMarker = "\x1AP"; // temporarily replaces "://" where auto-linking shouldn't happen;
312

    
313
        /// <summary>
314
        /// In the static constuctor we'll initialize what stays the same across all transforms.
315
        /// </summary>
316
        static Markdown()
317
        {
318
            // Table of hash values for escaped characters:
319
            _escapeTable = new Dictionary<string, string>();
320
            _invertedEscapeTable = new Dictionary<string, string>();
321
            // Table of hash value for backslash escaped characters:
322
            _backslashEscapeTable = new Dictionary<string, string>();
323

    
324
            string backslashPattern = "";
325

    
326
            foreach (char c in @"\`*_{}[]()>#+-.!/")
327
            {
328
                string key = c.ToString();
329
                string hash = GetHashKey(key, isHtmlBlock: false);
330
                _escapeTable.Add(key, hash);
331
                _invertedEscapeTable.Add(hash, key);
332
                _backslashEscapeTable.Add(@"\" + key, hash);
333
                backslashPattern += Regex.Escape(@"\" + key) + "|";
334
            }
335

    
336
            _backslashEscapes = new Regex(backslashPattern.Substring(0, backslashPattern.Length - 1), RegexOptions.Compiled);
337
        }
338

    
339
        /// <summary>
340
        /// current version of MarkdownSharp;  
341
        /// see http://code.google.com/p/markdownsharp/ for the latest code or to contribute
342
        /// </summary>
343
        public string Version
344
        {
345
            get { return _version; }
346
        }
347

    
348
        /// <summary>
349
        /// Transforms the provided Markdown-formatted text to HTML;  
350
        /// see http://en.wikipedia.org/wiki/Markdown
351
        /// </summary>
352
        /// <remarks>
353
        /// The order in which other subs are called here is
354
        /// essential. Link and image substitutions need to happen before
355
        /// EscapeSpecialChars(), so that any *'s or _'s in the a
356
        /// and img tags get encoded.
357
        /// </remarks>
358
        public string Transform(string text)
359
        {
360
            if (String.IsNullOrEmpty(text)) return "";
361

    
362
            Setup();
363

    
364
            text = Normalize(text);
365

    
366
            text = HashHTMLBlocks(text);
367
            text = StripLinkDefinitions(text);
368
            text = RunBlockGamut(text);
369
            text = Unescape(text);
370

    
371
            Cleanup();
372

    
373
            return text + "\n";
374
        }
375

    
376

    
377
        /// <summary>
378
        /// Perform transformations that form block-level tags like paragraphs, headers, and list items.
379
        /// </summary>
380
        private string RunBlockGamut(string text, bool unhash = true)
381
        {
382
            text = DoHeaders(text);
383
            text = DoHorizontalRules(text);
384
            text = DoLists(text);
385
            text = DoCodeBlocks(text);
386
            text = DoBlockQuotes(text);
387

    
388
            // We already ran HashHTMLBlocks() before, in Markdown(), but that
389
            // was to escape raw HTML in the original Markdown source. This time,
390
            // we're escaping the markup we've just created, so that we don't wrap
391
            // <p> tags around block-level tags.
392
            text = HashHTMLBlocks(text);
393

    
394
            text = FormParagraphs(text, unhash: unhash);
395

    
396
            return text;
397
        }
398

    
399

    
400
        /// <summary>
401
        /// Perform transformations that occur *within* block-level tags like paragraphs, headers, and list items.
402
        /// </summary>
403
        private string RunSpanGamut(string text)
404
        {
405
            text = DoCodeSpans(text);
406
            text = EscapeSpecialCharsWithinTagAttributes(text);
407
            text = EscapeBackslashes(text);
408

    
409
            // Images must come first, because ![foo][f] looks like an anchor.
410
            text = DoImages(text);
411
            text = DoAnchors(text);
412

    
413
            // Must come after DoAnchors(), because you can use < and >
414
            // delimiters in inline links like [this](<url>).
415
            text = DoAutoLinks(text);
416

    
417
            text = text.Replace(AutoLinkPreventionMarker, "://");
418

    
419
            text = EncodeAmpsAndAngles(text);
420
            text = DoItalicsAndBold(text);
421
            text = DoHardBreaks(text);
422

    
423
            return text;
424
        }
425

    
426
        private static Regex _newlinesLeadingTrailing = new Regex(@"^\n+|\n+\z", RegexOptions.Compiled);
427
        private static Regex _newlinesMultiple = new Regex(@"\n{2,}", RegexOptions.Compiled);
428
        private static Regex _leadingWhitespace = new Regex(@"^[ ]*", RegexOptions.Compiled);
429

    
430
        private static Regex _htmlBlockHash = new Regex("\x1AH\\d+H", RegexOptions.Compiled);
431

    
432
        /// <summary>
433
        /// splits on two or more newlines, to form "paragraphs";    
434
        /// each paragraph is then unhashed (if it is a hash and unhashing isn't turned off) or wrapped in HTML p tag
435
        /// </summary>
436
        private string FormParagraphs(string text, bool unhash = true)
437
        {
438
            // split on two or more newlines
439
            string[] grafs = _newlinesMultiple.Split(_newlinesLeadingTrailing.Replace(text, ""));
440

    
441
            for (int i = 0; i < grafs.Length; i++)
442
            {
443
                if (grafs[i].StartsWith("\x1AH"))
444
                {
445
                    // unhashify HTML blocks
446
                    if (unhash)
447
                    {
448
                        int sanityCheck = 50; // just for safety, guard against an infinite loop
449
                        bool keepGoing = true; // as long as replacements where made, keep going
450
                        while (keepGoing && sanityCheck > 0)
451
                        {
452
                            keepGoing = false;
453
                            grafs[i] = _htmlBlockHash.Replace(grafs[i], match =>
454
                            {
455
                                keepGoing = true;
456
                                return _htmlBlocks[match.Value];
457
                            });
458
                            sanityCheck--;
459
                        }
460
                        /* if (keepGoing)
461
                        {
462
                            // Logging of an infinite loop goes here.
463
                            // If such a thing should happen, please open a new issue on http://code.google.com/p/markdownsharp/
464
                            // with the input that caused it.
465
                        }*/
466
                    }
467
                }
468
                else
469
                {
470
                    // do span level processing inside the block, then wrap result in <p> tags
471
                    grafs[i] = _leadingWhitespace.Replace(RunSpanGamut(grafs[i]), "<p>") + "</p>";
472
                }
473
            }
474

    
475
            return string.Join("\n\n", grafs);
476
        }
477

    
478

    
479
        private void Setup()
480
        {
481
            // Clear the global hashes. If we don't clear these, you get conflicts
482
            // from other articles when generating a page which contains more than
483
            // one article (e.g. an index page that shows the N most recent
484
            // articles):
485
            _urls.Clear();
486
            _titles.Clear();
487
            _htmlBlocks.Clear();
488
            _listLevel = 0;
489
        }
490

    
491
        private void Cleanup()
492
        {
493
            Setup();
494
        }
495

    
496
        private static string _nestedBracketsPattern;
497

    
498
        /// <summary>
499
        /// Reusable pattern to match balanced [brackets]. See Friedl's
500
        /// "Mastering Regular Expressions", 2nd Ed., pp. 328-331.
501
        /// </summary>
502
        private static string GetNestedBracketsPattern()
503
        {
504
            // in other words [this] and [this[also]] and [this[also[too]]]
505
            // up to _nestDepth
506
            if (_nestedBracketsPattern == null)
507
                _nestedBracketsPattern =
508
                    RepeatString(@"
509
                    (?>              # Atomic matching
510
                       [^\[\]]+      # Anything other than brackets
511
                     |
512
                       \[
513
                           ", _nestDepth) + RepeatString(
514
                    @" \]
515
                    )*"
516
                    , _nestDepth);
517
            return _nestedBracketsPattern;
518
        }
519

    
520
        private static string _nestedParensPattern;
521

    
522
        /// <summary>
523
        /// Reusable pattern to match balanced (parens). See Friedl's
524
        /// "Mastering Regular Expressions", 2nd Ed., pp. 328-331.
525
        /// </summary>
526
        private static string GetNestedParensPattern()
527
        {
528
            // in other words (this) and (this(also)) and (this(also(too)))
529
            // up to _nestDepth
530
            if (_nestedParensPattern == null)
531
                _nestedParensPattern =
532
                    RepeatString(@"
533
                    (?>              # Atomic matching
534
                       [^()\s]+      # Anything other than parens or whitespace
535
                     |
536
                       \(
537
                           ", _nestDepth) + RepeatString(
538
                    @" \)
539
                    )*"
540
                    , _nestDepth);
541
            return _nestedParensPattern;
542
        }
543

    
544
        private static Regex _linkDef = new Regex(string.Format(@"
545
                        ^[ ]{{0,{0}}}\[(.+)\]:  # id = $1
546
                          [ ]*
547
                          \n?                   # maybe *one* newline
548
                          [ ]*
549
                        <?(\S+?)>?              # url = $2
550
                          [ ]*
551
                          \n?                   # maybe one newline
552
                          [ ]*
553
                        (?:
554
                            (?<=\s)             # lookbehind for whitespace
555
                            [""(]
556
                            (.+?)               # title = $3
557
                            ["")]
558
                            [ ]*
559
                        )?                      # title is optional
560
                        (?:\n+|\Z)", _tabWidth - 1), RegexOptions.Multiline | RegexOptions.IgnorePatternWhitespace | RegexOptions.Compiled);
561

    
562
        /// <summary>
563
        /// Strips link definitions from text, stores the URLs and titles in hash references.
564
        /// </summary>
565
        /// <remarks>
566
        /// ^[id]: url "optional title"
567
        /// </remarks>
568
        private string StripLinkDefinitions(string text)
569
        {
570
            return _linkDef.Replace(text, new MatchEvaluator(LinkEvaluator));
571
        }
572

    
573
        private string LinkEvaluator(Match match)
574
        {
575
            string linkID = match.Groups[1].Value.ToLowerInvariant();
576
            _urls[linkID] = EncodeAmpsAndAngles(match.Groups[2].Value);
577

    
578
            if (match.Groups[3] != null && match.Groups[3].Length > 0)
579
                _titles[linkID] = match.Groups[3].Value.Replace("\"", "&quot;");
580

    
581
            return "";
582
        }
583

    
584
        // compiling this monster regex results in worse performance. trust me.
585
        private static Regex _blocksHtml = new Regex(GetBlockPattern(), RegexOptions.Multiline | RegexOptions.IgnorePatternWhitespace);
586

    
587

    
588
        /// <summary>
589
        /// derived pretty much verbatim from PHP Markdown
590
        /// </summary>
591
        private static string GetBlockPattern()
592
        {
593

    
594
            // Hashify HTML blocks:
595
            // We only want to do this for block-level HTML tags, such as headers,
596
            // lists, and tables. That's because we still want to wrap <p>s around
597
            // "paragraphs" that are wrapped in non-block-level tags, such as anchors,
598
            // phrase emphasis, and spans. The list of tags we're looking for is
599
            // hard-coded:
600
            //
601
            // *  List "a" is made of tags which can be both inline or block-level.
602
            //    These will be treated block-level when the start tag is alone on
603
            //    its line, otherwise they're not matched here and will be taken as
604
            //    inline later.
605
            // *  List "b" is made of tags which are always block-level;
606
            //
607
            string blockTagsA = "ins|del";
608
            string blockTagsB = "p|div|h[1-6]|blockquote|pre|table|dl|ol|ul|address|script|noscript|form|fieldset|iframe|math";
609

    
610
            // Regular expression for the content of a block tag.
611
            string attr = @"
612
            (?>                    # optional tag attributes
613
              \s                  # starts with whitespace
614
              (?>
615
                [^>""/]+              # text outside quotes
616
              |
617
                /+(?!>)                # slash not followed by >
618
              |
619
                ""[^""]*""            # text inside double quotes (tolerate >)
620
              |
621
                '[^']*'                  # text inside single quotes (tolerate >)
622
              )*
623
            )?  
624
            ";
625

    
626
            string content = RepeatString(@"
627
                (?>
628
                  [^<]+              # content without tag
629
                |
630
                  <\2              # nested opening tag
631
                    " + attr + @"       # attributes
632
                  (?>
633
                      />
634
                  |
635
                      >", _nestDepth) +   // end of opening tag
636
                      ".*?" +             // last level nested tag content
637
            RepeatString(@"
638
                      </\2\s*>          # closing nested tag
639
                  )
640
                  |        
641
                  <(?!/\2\s*>           # other tags with a different name
642
                  )
643
                )*", _nestDepth);
644

    
645
            string content2 = content.Replace(@"\2", @"\3");
646

    
647
            // First, look for nested blocks, e.g.:
648
            //   <div>
649
            //     <div>
650
            //     tags for inner block must be indented.
651
            //     </div>
652
            //   </div>
653
            //
654
            // The outermost tags must start at the left margin for this to match, and
655
            // the inner nested divs must be indented.
656
            // We need to do this before the next, more liberal match, because the next
657
            // match will start at the first `<div>` and stop at the first `</div>`.
658
            string pattern = @"
659
            (?>
660
                  (?>
661
                    (?<=\n)     # Starting at the beginning of a line
662
                    |           # or
663
                    \A\n?       # the beginning of the doc
664
                  )
665
                  (             # save in $1
666

    
667
                    # Match from `\n<tag>` to `</tag>\n`, handling nested tags
668
                    # in between.
669
                      
670
                        <($block_tags_b_re)   # start tag = $2
671
                        $attr>                # attributes followed by > and \n
672
                        $content              # content, support nesting
673
                        </\2>                 # the matching end tag
674
                        [ ]*                  # trailing spaces
675
                        (?=\n+|\Z)            # followed by a newline or end of document
676

    
677
                  | # Special version for tags of group a.
678

    
679
                        <($block_tags_a_re)   # start tag = $3
680
                        $attr>[ ]*\n          # attributes followed by >
681
                        $content2             # content, support nesting
682
                        </\3>                 # the matching end tag
683
                        [ ]*                  # trailing spaces
684
                        (?=\n+|\Z)            # followed by a newline or end of document
685
                      
686
                  | # Special case just for <hr />. It was easier to make a special
687
                    # case than to make the other regex more complicated.
688
                  
689
                        [ ]{0,$less_than_tab}
690
                        <hr
691
                        $attr                 # attributes
692
                        /?>                   # the matching end tag
693
                        [ ]*
694
                        (?=\n{2,}|\Z)         # followed by a blank line or end of document
695
                  
696
                  | # Special case for standalone HTML comments:
697
                  
698
                      (?<=\n\n|\A)            # preceded by a blank line or start of document
699
                      [ ]{0,$less_than_tab}
700
                      (?s:
701
                        <!--(?:|(?:[^>-]|-[^>])(?:[^-]|-[^-])*)-->
702
                      )
703
                      [ ]*
704
                      (?=\n{2,}|\Z)            # followed by a blank line or end of document
705
                  
706
                  | # PHP and ASP-style processor instructions (<? and <%)
707
                  
708
                      [ ]{0,$less_than_tab}
709
                      (?s:
710
                        <([?%])                # $4
711
                        .*?
712
                        \4>
713
                      )
714
                      [ ]*
715
                      (?=\n{2,}|\Z)            # followed by a blank line or end of document
716
                      
717
                  )
718
            )";
719

    
720
            pattern = pattern.Replace("$less_than_tab", (_tabWidth - 1).ToString());
721
            pattern = pattern.Replace("$block_tags_b_re", blockTagsB);
722
            pattern = pattern.Replace("$block_tags_a_re", blockTagsA);
723
            pattern = pattern.Replace("$attr", attr);
724
            pattern = pattern.Replace("$content2", content2);
725
            pattern = pattern.Replace("$content", content);
726

    
727
            return pattern;
728
        }
729

    
730
        /// <summary>
731
        /// replaces any block-level HTML blocks with hash entries
732
        /// </summary>
733
        private string HashHTMLBlocks(string text)
734
        {
735
            return _blocksHtml.Replace(text, new MatchEvaluator(HtmlEvaluator));
736
        }
737

    
738
        private string HtmlEvaluator(Match match)
739
        {
740
            string text = match.Groups[1].Value;
741
            string key = GetHashKey(text, isHtmlBlock: true);
742
            _htmlBlocks[key] = text;
743

    
744
            return string.Concat("\n\n", key, "\n\n");
745
        }
746

    
747
        private static string GetHashKey(string s, bool isHtmlBlock)
748
        {
749
            var delim = isHtmlBlock ? 'H' : 'E';
750
            return "\x1A" + delim + Math.Abs(s.GetHashCode()).ToString() + delim;
751
        }
752

    
753
        private static Regex _htmlTokens = new Regex(@"
754
            (<!--(?:|(?:[^>-]|-[^>])(?:[^-]|-[^-])*)-->)|        # match <!-- foo -->
755
            (<\?.*?\?>)|                 # match <?foo?> " +
756
            RepeatString(@"
757
            (<[A-Za-z\/!$](?:[^<>]|", _nestDepth) + RepeatString(@")*>)", _nestDepth) +
758
                                       " # match <tag> and </tag>",
759
            RegexOptions.Multiline | RegexOptions.Singleline | RegexOptions.ExplicitCapture | RegexOptions.IgnorePatternWhitespace | RegexOptions.Compiled);
760

    
761
        /// <summary>
762
        /// returns an array of HTML tokens comprising the input string. Each token is
763
        /// either a tag (possibly with nested, tags contained therein, such
764
        /// as &lt;a href="&lt;MTFoo&gt;"&gt;, or a run of text between tags. Each element of the
765
        /// array is a two-element array; the first is either 'tag' or 'text'; the second is
766
        /// the actual value.
767
        /// </summary>
768
        private List<Token> TokenizeHTML(string text)
769
        {
770
            int pos = 0;
771
            int tagStart = 0;
772
            var tokens = new List<Token>();
773

    
774
            // this regex is derived from the _tokenize() subroutine in Brad Choate's MTRegex plugin.
775
            // http://www.bradchoate.com/past/mtregex.php
776
            foreach (Match m in _htmlTokens.Matches(text))
777
            {
778
                tagStart = m.Index;
779

    
780
                if (pos < tagStart)
781
                    tokens.Add(new Token(TokenType.Text, text.Substring(pos, tagStart - pos)));
782

    
783
                tokens.Add(new Token(TokenType.Tag, m.Value));
784
                pos = tagStart + m.Length;
785
            }
786

    
787
            if (pos < text.Length)
788
                tokens.Add(new Token(TokenType.Text, text.Substring(pos, text.Length - pos)));
789

    
790
            return tokens;
791
        }
792

    
793

    
794
        private static Regex _anchorRef = new Regex(string.Format(@"
795
            (                               # wrap whole match in $1
796
                \[
797
                    ({0})                   # link text = $2
798
                \]
799

    
800
                [ ]?                        # one optional space
801
                (?:\n[ ]*)?                 # one optional newline followed by spaces
802

    
803
                \[
804
                    (.*?)                   # id = $3
805
                \]
806
            )", GetNestedBracketsPattern()), RegexOptions.Singleline | RegexOptions.IgnorePatternWhitespace | RegexOptions.Compiled);
807

    
808
        private static Regex _anchorInline = new Regex(string.Format(@"
809
                (                           # wrap whole match in $1
810
                    \[
811
                        ({0})               # link text = $2
812
                    \]
813
                    \(                      # literal paren
814
                        [ ]*
815
                        ({1})               # href = $3
816
                        [ ]*
817
                        (                   # $4
818
                        (['""])           # quote char = $5
819
                        (.*?)               # title = $6
820
                        \5                  # matching quote
821
                        [ ]*                # ignore any spaces between closing quote and )
822
                        )?                  # title is optional
823
                    \)
824
                )", GetNestedBracketsPattern(), GetNestedParensPattern()),
825
                  RegexOptions.Singleline | RegexOptions.IgnorePatternWhitespace | RegexOptions.Compiled);
826

    
827
        private static Regex _anchorRefShortcut = new Regex(@"
828
            (                               # wrap whole match in $1
829
              \[
830
                 ([^\[\]]+)                 # link text = $2; can't contain [ or ]
831
              \]
832
            )", RegexOptions.Singleline | RegexOptions.IgnorePatternWhitespace | RegexOptions.Compiled);
833

    
834
        /// <summary>
835
        /// Turn Markdown link shortcuts into HTML anchor tags
836
        /// </summary>
837
        /// <remarks>
838
        /// [link text](url "title")
839
        /// [link text][id]
840
        /// [id]
841
        /// </remarks>
842
        private string DoAnchors(string text)
843
        {
844
            // First, handle reference-style links: [link text] [id]
845
            text = _anchorRef.Replace(text, new MatchEvaluator(AnchorRefEvaluator));
846

    
847
            // Next, inline-style links: [link text](url "optional title") or [link text](url "optional title")
848
            text = _anchorInline.Replace(text, new MatchEvaluator(AnchorInlineEvaluator));
849

    
850
            //  Last, handle reference-style shortcuts: [link text]
851
            //  These must come last in case you've also got [link test][1]
852
            //  or [link test](/foo)
853
            text = _anchorRefShortcut.Replace(text, new MatchEvaluator(AnchorRefShortcutEvaluator));
854
            return text;
855
        }
856

    
857
        private string SaveFromAutoLinking(string s)
858
        {
859
            return s.Replace("://", AutoLinkPreventionMarker);
860
        }
861

    
862
        private string AnchorRefEvaluator(Match match)
863
        {
864
            string wholeMatch = match.Groups[1].Value;
865
            string linkText = SaveFromAutoLinking(match.Groups[2].Value);
866
            string linkID = match.Groups[3].Value.ToLowerInvariant();
867

    
868
            string result;
869

    
870
            // for shortcut links like [this][].
871
            if (linkID == "")
872
                linkID = linkText.ToLowerInvariant();
873

    
874
            if (_urls.ContainsKey(linkID))
875
            {
876
                string url = _urls[linkID];
877

    
878
                url = EncodeProblemUrlChars(url);
879
                url = EscapeBoldItalic(url);
880
                result = "<a href=\"" + url + "\"";
881

    
882
                if (_titles.ContainsKey(linkID))
883
                {
884
                    string title = AttributeEncode(_titles[linkID]);
885
                    title = AttributeEncode(EscapeBoldItalic(title));
886
                    result += " title=\"" + title + "\"";
887
                }
888

    
889
                result += ">" + linkText + "</a>";
890
            }
891
            else
892
                result = wholeMatch;
893

    
894
            return result;
895
        }
896

    
897
        private string AnchorRefShortcutEvaluator(Match match)
898
        {
899
            string wholeMatch = match.Groups[1].Value;
900
            string linkText = SaveFromAutoLinking(match.Groups[2].Value);
901
            string linkID = Regex.Replace(linkText.ToLowerInvariant(), @"[ ]*\n[ ]*", " ");  // lower case and remove newlines / extra spaces
902

    
903
            string result;
904

    
905
            if (_urls.ContainsKey(linkID))
906
            {
907
                string url = _urls[linkID];
908

    
909
                url = EncodeProblemUrlChars(url);
910
                url = EscapeBoldItalic(url);
911
                result = "<a href=\"" + url + "\"";
912

    
913
                if (_titles.ContainsKey(linkID))
914
                {
915
                    string title = AttributeEncode(_titles[linkID]);
916
                    title = EscapeBoldItalic(title);
917
                    result += " title=\"" + title + "\"";
918
                }
919

    
920
                result += ">" + linkText + "</a>";
921
            }
922
            else
923
                result = wholeMatch;
924

    
925
            return result;
926
        }
927

    
928

    
929
        private string AnchorInlineEvaluator(Match match)
930
        {
931
            string linkText = SaveFromAutoLinking(match.Groups[2].Value);
932
            string url = match.Groups[3].Value;
933
            string title = match.Groups[6].Value;
934
            string result;
935

    
936
            url = EncodeProblemUrlChars(url);
937
            url = EscapeBoldItalic(url);
938
            if (url.StartsWith("<") && url.EndsWith(">"))
939
                url = url.Substring(1, url.Length - 2); // remove <>'s surrounding URL, if present            
940

    
941
            result = string.Format("<a href=\"{0}\"", url);
942

    
943
            if (!String.IsNullOrEmpty(title))
944
            {
945
                title = AttributeEncode(title);
946
                title = EscapeBoldItalic(title);
947
                result += string.Format(" title=\"{0}\"", title);
948
            }
949

    
950
            result += string.Format(">{0}</a>", linkText);
951
            return result;
952
        }
953

    
954
        private static Regex _imagesRef = new Regex(@"
955
                    (               # wrap whole match in $1
956
                    !\[
957
                        (.*?)       # alt text = $2
958
                    \]
959

    
960
                    [ ]?            # one optional space
961
                    (?:\n[ ]*)?     # one optional newline followed by spaces
962

    
963
                    \[
964
                        (.*?)       # id = $3
965
                    \]
966

    
967
                    )", RegexOptions.IgnorePatternWhitespace | RegexOptions.Singleline | RegexOptions.Compiled);
968

    
969
        private static Regex _imagesInline = new Regex(String.Format(@"
970
              (                     # wrap whole match in $1
971
                !\[
972
                    (.*?)           # alt text = $2
973
                \]
974
                \s?                 # one optional whitespace character
975
                \(                  # literal paren
976
                    [ ]*
977
                    ({0})           # href = $3
978
                    [ ]*
979
                    (               # $4
980
                    (['""])       # quote char = $5
981
                    (.*?)           # title = $6
982
                    \5              # matching quote
983
                    [ ]*
984
                    )?              # title is optional
985
                \)
986
              )", GetNestedParensPattern()),
987
                  RegexOptions.IgnorePatternWhitespace | RegexOptions.Singleline | RegexOptions.Compiled);
988

    
989
        /// <summary>
990
        /// Turn Markdown image shortcuts into HTML img tags.
991
        /// </summary>
992
        /// <remarks>
993
        /// ![alt text][id]
994
        /// ![alt text](url "optional title")
995
        /// </remarks>
996
        private string DoImages(string text)
997
        {
998
            // First, handle reference-style labeled images: ![alt text][id]
999
            text = _imagesRef.Replace(text, new MatchEvaluator(ImageReferenceEvaluator));
1000

    
1001
            // Next, handle inline images:  ![alt text](url "optional title")
1002
            // Don't forget: encode * and _
1003
            text = _imagesInline.Replace(text, new MatchEvaluator(ImageInlineEvaluator));
1004

    
1005
            return text;
1006
        }
1007

    
1008
        // This prevents the creation of horribly broken HTML when some syntax ambiguities
1009
        // collide. It likely still doesn't do what the user meant, but at least we're not
1010
        // outputting garbage.
1011
        private string EscapeImageAltText(string s)
1012
        {
1013
            s = EscapeBoldItalic(s);
1014
            s = Regex.Replace(s, @"[\[\]()]", m => _escapeTable[m.ToString()]);
1015
            return s;
1016
        }
1017

    
1018
        private string ImageReferenceEvaluator(Match match)
1019
        {
1020
            string wholeMatch = match.Groups[1].Value;
1021
            string altText = match.Groups[2].Value;
1022
            string linkID = match.Groups[3].Value.ToLowerInvariant();
1023

    
1024
            // for shortcut links like ![this][].
1025
            if (linkID == "")
1026
                linkID = altText.ToLowerInvariant();
1027

    
1028
            if (_urls.ContainsKey(linkID))
1029
            {
1030
                string url = _urls[linkID];
1031
                string title = null;
1032

    
1033
                if (_titles.ContainsKey(linkID))
1034
                    title = _titles[linkID];
1035

    
1036
                return ImageTag(url, altText, title);
1037
            }
1038
            else
1039
            {
1040
                // If there's no such link ID, leave intact:
1041
                return wholeMatch;
1042
            }
1043
        }
1044

    
1045
        private string ImageInlineEvaluator(Match match)
1046
        {
1047
            string alt = match.Groups[2].Value;
1048
            string url = match.Groups[3].Value;
1049
            string title = match.Groups[6].Value;
1050

    
1051
            if (url.StartsWith("<") && url.EndsWith(">"))
1052
                url = url.Substring(1, url.Length - 2);    // Remove <>'s surrounding URL, if present
1053

    
1054
            return ImageTag(url, alt, title);
1055
        }
1056

    
1057
        private string ImageTag(string url, string altText, string title)
1058
        {
1059
            altText = EscapeImageAltText(AttributeEncode(altText));
1060
            url = EncodeProblemUrlChars(url);
1061
            url = EscapeBoldItalic(url);
1062
            var result = string.Format("<img src=\"{0}\" alt=\"{1}\"", url, altText);
1063
            if (!String.IsNullOrEmpty(title))
1064
            {
1065
                title = AttributeEncode(EscapeBoldItalic(title));
1066
                result += string.Format(" title=\"{0}\"", title);
1067
            }
1068
            result += _emptyElementSuffix;
1069
            return result;
1070
        }
1071

    
1072
        private static Regex _headerSetext = new Regex(@"
1073
                ^(.+?)
1074
                [ ]*
1075
                \n
1076
                (=+|-+)     # $1 = string of ='s or -'s
1077
                [ ]*
1078
                \n+",
1079
            RegexOptions.Multiline | RegexOptions.IgnorePatternWhitespace | RegexOptions.Compiled);
1080

    
1081
        private static Regex _headerAtx = new Regex(@"
1082
                ^(\#{1,6})  # $1 = string of #'s
1083
                [ ]*
1084
                (.+?)       # $2 = Header text
1085
                [ ]*
1086
                \#*         # optional closing #'s (not counted)
1087
                \n+",
1088
            RegexOptions.Multiline | RegexOptions.IgnorePatternWhitespace | RegexOptions.Compiled);
1089

    
1090
        /// <summary>
1091
        /// Turn Markdown headers into HTML header tags
1092
        /// </summary>
1093
        /// <remarks>
1094
        /// Header 1  
1095
        /// ========  
1096
        ///
1097
        /// Header 2  
1098
        /// --------  
1099
        ///
1100
        /// # Header 1  
1101
        /// ## Header 2  
1102
        /// ## Header 2 with closing hashes ##  
1103
        /// ...  
1104
        /// ###### Header 6  
1105
        /// </remarks>
1106
        private string DoHeaders(string text)
1107
        {
1108
            text = _headerSetext.Replace(text, new MatchEvaluator(SetextHeaderEvaluator));
1109
            text = _headerAtx.Replace(text, new MatchEvaluator(AtxHeaderEvaluator));
1110
            return text;
1111
        }
1112

    
1113
        private string SetextHeaderEvaluator(Match match)
1114
        {
1115
            string header = match.Groups[1].Value;
1116
            int level = match.Groups[2].Value.StartsWith("=") ? 1 : 2;
1117
            return string.Format("<h{1}>{0}</h{1}>\n\n", RunSpanGamut(header), level);
1118
        }
1119

    
1120
        private string AtxHeaderEvaluator(Match match)
1121
        {
1122
            string header = match.Groups[2].Value;
1123
            int level = match.Groups[1].Value.Length;
1124
            return string.Format("<h{1}>{0}</h{1}>\n\n", RunSpanGamut(header), level);
1125
        }
1126

    
1127

    
1128
        private static Regex _horizontalRules = new Regex(@"
1129
            ^[ ]{0,3}         # Leading space
1130
                ([-*_])       # $1: First marker
1131
                (?>           # Repeated marker group
1132
                    [ ]{0,2}  # Zero, one, or two spaces.
1133
                    \1        # Marker character
1134
                ){2,}         # Group repeated at least twice
1135
                [ ]*          # Trailing spaces
1136
                $             # End of line.
1137
            ", RegexOptions.Multiline | RegexOptions.IgnorePatternWhitespace | RegexOptions.Compiled);
1138

    
1139
        /// <summary>
1140
        /// Turn Markdown horizontal rules into HTML hr tags
1141
        /// </summary>
1142
        /// <remarks>
1143
        /// ***  
1144
        /// * * *  
1145
        /// ---
1146
        /// - - -
1147
        /// </remarks>
1148
        private string DoHorizontalRules(string text)
1149
        {
1150
            return _horizontalRules.Replace(text, "<hr" + _emptyElementSuffix + "\n");
1151
        }
1152

    
1153
        private static string _wholeList = string.Format(@"
1154
            (                               # $1 = whole list
1155
              (                             # $2
1156
                [ ]{{0,{1}}}
1157
                ({0})                       # $3 = first list item marker
1158
                [ ]+
1159
              )
1160
              (?s:.+?)
1161
              (                             # $4
1162
                  \z
1163
                |
1164
                  \n{{2,}}
1165
                  (?=\S)
1166
                  (?!                       # Negative lookahead for another list item marker
1167
                    [ ]*
1168
                    {0}[ ]+
1169
                  )
1170
              )
1171
            )", string.Format("(?:{0}|{1})", _markerUL, _markerOL), _tabWidth - 1);
1172

    
1173
        private static Regex _listNested = new Regex(@"^" + _wholeList,
1174
            RegexOptions.Multiline | RegexOptions.IgnorePatternWhitespace | RegexOptions.Compiled);
1175

    
1176
        private static Regex _listTopLevel = new Regex(@"(?:(?<=\n\n)|\A\n?)" + _wholeList,
1177
            RegexOptions.Multiline | RegexOptions.IgnorePatternWhitespace | RegexOptions.Compiled);
1178

    
1179
        /// <summary>
1180
        /// Turn Markdown lists into HTML ul and ol and li tags
1181
        /// </summary>
1182
        private string DoLists(string text)
1183
        {
1184
            // We use a different prefix before nested lists than top-level lists.
1185
            // See extended comment in _ProcessListItems().
1186
            if (_listLevel > 0)
1187
                text = _listNested.Replace(text, new MatchEvaluator(ListEvaluator));
1188
            else
1189
                text = _listTopLevel.Replace(text, new MatchEvaluator(ListEvaluator));
1190

    
1191
            return text;
1192
        }
1193

    
1194
        private string ListEvaluator(Match match)
1195
        {
1196
            string list = match.Groups[1].Value;
1197
            string listType = Regex.IsMatch(match.Groups[3].Value, _markerUL) ? "ul" : "ol";
1198
            string result;
1199

    
1200
            result = ProcessListItems(list, listType == "ul" ? _markerUL : _markerOL);
1201

    
1202
            result = string.Format("<{0}>\n{1}</{0}>\n", listType, result);
1203
            return result;
1204
        }
1205

    
1206
        /// <summary>
1207
        /// Process the contents of a single ordered or unordered list, splitting it
1208
        /// into individual list items.
1209
        /// </summary>
1210
        private string ProcessListItems(string list, string marker)
1211
        {
1212
            // The listLevel global keeps track of when we're inside a list.
1213
            // Each time we enter a list, we increment it; when we leave a list,
1214
            // we decrement. If it's zero, we're not in a list anymore.
1215

    
1216
            // We do this because when we're not inside a list, we want to treat
1217
            // something like this:
1218

    
1219
            //    I recommend upgrading to version
1220
            //    8. Oops, now this line is treated
1221
            //    as a sub-list.
1222

    
1223
            // As a single paragraph, despite the fact that the second line starts
1224
            // with a digit-period-space sequence.
1225

    
1226
            // Whereas when we're inside a list (or sub-list), that line will be
1227
            // treated as the start of a sub-list. What a kludge, huh? This is
1228
            // an aspect of Markdown's syntax that's hard to parse perfectly
1229
            // without resorting to mind-reading. Perhaps the solution is to
1230
            // change the syntax rules such that sub-lists must start with a
1231
            // starting cardinal number; e.g. "1." or "a.".
1232

    
1233
            _listLevel++;
1234

    
1235
            // Trim trailing blank lines:
1236
            list = Regex.Replace(list, @"\n{2,}\z", "\n");
1237

    
1238
            string pattern = string.Format(
1239
              @"(^[ ]*)                    # leading whitespace = $1
1240
                ({0}) [ ]+                 # list marker = $2
1241
                ((?s:.+?)                  # list item text = $3
1242
                (\n+))      
1243
                (?= (\z | \1 ({0}) [ ]+))", marker);
1244

    
1245
            bool lastItemHadADoubleNewline = false;
1246

    
1247
            // has to be a closure, so subsequent invocations can share the bool
1248
            MatchEvaluator ListItemEvaluator = (Match match) =>
1249
            {
1250
                string item = match.Groups[3].Value;
1251

    
1252
                bool endsWithDoubleNewline = item.EndsWith("\n\n");
1253
                bool containsDoubleNewline = endsWithDoubleNewline || item.Contains("\n\n");
1254

    
1255
                if (containsDoubleNewline || lastItemHadADoubleNewline)
1256
                    // we could correct any bad indentation here..
1257
                    item = RunBlockGamut(Outdent(item) + "\n", unhash: false);
1258
                else
1259
                {
1260
                    // recursion for sub-lists
1261
                    item = DoLists(Outdent(item));
1262
                    item = item.TrimEnd('\n');
1263
                    item = RunSpanGamut(item);
1264
                }
1265
                lastItemHadADoubleNewline = endsWithDoubleNewline;
1266
                return string.Format("<li>{0}</li>\n", item);
1267
            };
1268

    
1269
            list = Regex.Replace(list, pattern, new MatchEvaluator(ListItemEvaluator),
1270
                                  RegexOptions.IgnorePatternWhitespace | RegexOptions.Multiline);
1271
            _listLevel--;
1272
            return list;
1273
        }
1274

    
1275
        private static Regex _codeBlock = new Regex(string.Format(@"
1276
                    (?:\n\n|\A\n?)
1277
                    (                        # $1 = the code block -- one or more lines, starting with a space
1278
                    (?:
1279
                        (?:[ ]{{{0}}})       # Lines must start with a tab-width of spaces
1280
                        .*\n+
1281
                    )+
1282
                    )
1283
                    ((?=^[ ]{{0,{0}}}[^ \t\n])|\Z) # Lookahead for non-space at line-start, or end of doc",
1284
                    _tabWidth), RegexOptions.Multiline | RegexOptions.IgnorePatternWhitespace | RegexOptions.Compiled);
1285

    
1286
        /// <summary>
1287
        /// /// Turn Markdown 4-space indented code into HTML pre code blocks
1288
        /// </summary>
1289
        private string DoCodeBlocks(string text)
1290
        {
1291
            text = _codeBlock.Replace(text, new MatchEvaluator(CodeBlockEvaluator));
1292
            return text;
1293
        }
1294

    
1295
        private string CodeBlockEvaluator(Match match)
1296
        {
1297
            string codeBlock = match.Groups[1].Value;
1298

    
1299
            codeBlock = EncodeCode(Outdent(codeBlock));
1300
            codeBlock = _newlinesLeadingTrailing.Replace(codeBlock, "");
1301

    
1302
            return string.Concat("\n\n<pre><code>", codeBlock, "\n</code></pre>\n\n");
1303
        }
1304

    
1305
        private static Regex _codeSpan = new Regex(@"
1306
                    (?<!\\)   # Character before opening ` can't be a backslash
1307
                    (`+)      # $1 = Opening run of `
1308
                    (.+?)     # $2 = The code block
1309
                    (?<!`)
1310
                    \1
1311
                    (?!`)", RegexOptions.IgnorePatternWhitespace | RegexOptions.Singleline | RegexOptions.Compiled);
1312

    
1313
        /// <summary>
1314
        /// Turn Markdown `code spans` into HTML code tags
1315
        /// </summary>
1316
        private string DoCodeSpans(string text)
1317
        {
1318
            //    * You can use multiple backticks as the delimiters if you want to
1319
            //        include literal backticks in the code span. So, this input:
1320
            //
1321
            //        Just type ``foo `bar` baz`` at the prompt.
1322
            //
1323
            //        Will translate to:
1324
            //
1325
            //          <p>Just type <code>foo `bar` baz</code> at the prompt.</p>
1326
            //
1327
            //        There's no arbitrary limit to the number of backticks you
1328
            //        can use as delimters. If you need three consecutive backticks
1329
            //        in your code, use four for delimiters, etc.
1330
            //
1331
            //    * You can use spaces to get literal backticks at the edges:
1332
            //
1333
            //          ... type `` `bar` `` ...
1334
            //
1335
            //        Turns to:
1336
            //
1337
            //          ... type <code>`bar`</code> ...         
1338
            //
1339

    
1340
            return _codeSpan.Replace(text, new MatchEvaluator(CodeSpanEvaluator));
1341
        }
1342

    
1343
        private string CodeSpanEvaluator(Match match)
1344
        {
1345
            string span = match.Groups[2].Value;
1346
            span = Regex.Replace(span, @"^[ ]*", ""); // leading whitespace
1347
            span = Regex.Replace(span, @"[ ]*$", ""); // trailing whitespace
1348
            span = EncodeCode(span);
1349
            span = SaveFromAutoLinking(span); // to prevent auto-linking. Not necessary in code *blocks*, but in code spans.
1350

    
1351
            return string.Concat("<code>", span, "</code>");
1352
        }
1353

    
1354

    
1355
        private static Regex _bold = new Regex(@"(\*\*|__) (?=\S) (.+?[*_]*) (?<=\S) \1",
1356
            RegexOptions.IgnorePatternWhitespace | RegexOptions.Singleline | RegexOptions.Compiled);
1357
        private static Regex _strictBold = new Regex(@"([\W_]|^) (\*\*|__) (?=\S) ([^\r]*?\S[\*_]*) \2 ([\W_]|$)",
1358
            RegexOptions.IgnorePatternWhitespace | RegexOptions.Singleline | RegexOptions.Compiled);
1359

    
1360
        private static Regex _italic = new Regex(@"(\*|_) (?=\S) (.+?) (?<=\S) \1",
1361
            RegexOptions.IgnorePatternWhitespace | RegexOptions.Singleline | RegexOptions.Compiled);
1362
        private static Regex _strictItalic = new Regex(@"([\W_]|^) (\*|_) (?=\S) ([^\r\*_]*?\S) \2 ([\W_]|$)",
1363
            RegexOptions.IgnorePatternWhitespace | RegexOptions.Singleline | RegexOptions.Compiled);
1364

    
1365
        /// <summary>
1366
        /// Turn Markdown *italics* and **bold** into HTML strong and em tags
1367
        /// </summary>
1368
        private string DoItalicsAndBold(string text)
1369
        {
1370

    
1371
            // <strong> must go first, then <em>
1372
            if (_strictBoldItalic)
1373
            {
1374
                text = _strictBold.Replace(text, "$1<strong>$3</strong>$4");
1375
                text = _strictItalic.Replace(text, "$1<em>$3</em>$4");
1376
            }
1377
            else
1378
            {
1379
                text = _bold.Replace(text, "<strong>$2</strong>");
1380
                text = _italic.Replace(text, "<em>$2</em>");
1381
            }
1382
            return text;
1383
        }
1384

    
1385
        /// <summary>
1386
        /// Turn markdown line breaks (two space at end of line) into HTML break tags
1387
        /// </summary>
1388
        private string DoHardBreaks(string text)
1389
        {
1390
            if (_autoNewlines)
1391
                text = Regex.Replace(text, @"\n", string.Format("<br{0}\n", _emptyElementSuffix));
1392
            else
1393
                text = Regex.Replace(text, @" {2,}\n", string.Format("<br{0}\n", _emptyElementSuffix));
1394
            return text;
1395
        }
1396

    
1397
        private static Regex _blockquote = new Regex(@"
1398
            (                           # Wrap whole match in $1
1399
                (
1400
                ^[ ]*>[ ]?              # '>' at the start of a line
1401
                    .+\n                # rest of the first line
1402
                (.+\n)*                 # subsequent consecutive lines
1403
                \n*                     # blanks
1404
                )+
1405
            )", RegexOptions.IgnorePatternWhitespace | RegexOptions.Multiline | RegexOptions.Compiled);
1406

    
1407
        /// <summary>
1408
        /// Turn Markdown > quoted blocks into HTML blockquote blocks
1409
        /// </summary>
1410
        private string DoBlockQuotes(string text)
1411
        {
1412
            return _blockquote.Replace(text, new MatchEvaluator(BlockQuoteEvaluator));
1413
        }
1414

    
1415
        private string BlockQuoteEvaluator(Match match)
1416
        {
1417
            string bq = match.Groups[1].Value;
1418

    
1419
            bq = Regex.Replace(bq, @"^[ ]*>[ ]?", "", RegexOptions.Multiline);       // trim one level of quoting
1420
            bq = Regex.Replace(bq, @"^[ ]+$", "", RegexOptions.Multiline);           // trim whitespace-only lines
1421
            bq = RunBlockGamut(bq);                                                  // recurse
1422

    
1423
            bq = Regex.Replace(bq, @"^", "  ", RegexOptions.Multiline);
1424

    
1425
            // These leading spaces screw with <pre> content, so we need to fix that:
1426
            bq = Regex.Replace(bq, @"(\s*<pre>.+?</pre>)", new MatchEvaluator(BlockQuoteEvaluator2), RegexOptions.IgnorePatternWhitespace | RegexOptions.Singleline);
1427

    
1428
            bq = string.Format("<blockquote>\n{0}\n</blockquote>", bq);
1429
            string key = GetHashKey(bq, isHtmlBlock: true);
1430
            _htmlBlocks[key] = bq;
1431

    
1432
            return "\n\n" + key + "\n\n";
1433
        }
1434

    
1435
        private string BlockQuoteEvaluator2(Match match)
1436
        {
1437
            return Regex.Replace(match.Groups[1].Value, @"^  ", "", RegexOptions.Multiline);
1438
        }
1439

    
1440
        private static Regex _autolinkBare = new Regex(@"(<|="")?\b(https?|ftp)(://[-A-Z0-9+&@#/%?=~_|\[\]\(\)!:,\.;]*[-A-Z0-9+&@#/%=~_|\[\])])(?=$|\W)",
1441
            RegexOptions.IgnoreCase | RegexOptions.Compiled);
1442

    
1443
        private static string handleTrailingParens(Match match)
1444
        {
1445
            // The first group is essentially a negative lookbehind -- if there's a < or a =", we don't touch this.
1446
            // We're not using a *real* lookbehind, because of links with in links, like <a href="http://web.archive.org/web/20121130000728/http://www.google.com/">
1447
            // With a real lookbehind, the full link would never be matched, and thus the http://www.google.com *would* be matched.
1448
            // With the simulated lookbehind, the full link *is* matched (just not handled, because of this early return), causing
1449
            // the google link to not be matched again.
1450
            if (match.Groups[1].Success)
1451
                return match.Value;
1452

    
1453
            var protocol = match.Groups[2].Value;
1454
            var link = match.Groups[3].Value;
1455
            if (!link.EndsWith(")"))
1456
                return "<" + protocol + link + ">";
1457
            var level = 0;
1458
            foreach (Match c in Regex.Matches(link, "[()]"))
1459
            {
1460
                if (c.Value == "(")
1461
                {
1462
                    if (level <= 0)
1463
                        level = 1;
1464
                    else
1465
                        level++;
1466
                }
1467
                else
1468
                {
1469
                    level--;
1470
                }
1471
            }
1472
            var tail = "";
1473
            if (level < 0)
1474
            {
1475
                link = Regex.Replace(link, @"\){1," + (-level) + "}$", m => { tail = m.Value; return ""; });
1476
            }
1477
            return "<" + protocol + link + ">" + tail;
1478
        }
1479

    
1480
        /// <summary>
1481
        /// Turn angle-delimited URLs into HTML anchor tags
1482
        /// </summary>
1483
        /// <remarks>
1484
        /// &lt;http://www.example.com&gt;
1485
        /// </remarks>
1486
        private string DoAutoLinks(string text)
1487
        {
1488

    
1489
            if (_autoHyperlink)
1490
            {
1491
                // fixup arbitrary URLs by adding Markdown < > so they get linked as well
1492
                // note that at this point, all other URL in the text are already hyperlinked as <a href=""></a>
1493
                // *except* for the <http://www.foo.com> case
1494
                text = _autolinkBare.Replace(text, handleTrailingParens);
1495
            }
1496

    
1497
            // Hyperlinks: <http://foo.com>
1498
            text = Regex.Replace(text, "<((https?|ftp):[^'\">\\s]+)>", new MatchEvaluator(HyperlinkEvaluator));
1499

    
1500
            if (_linkEmails)
1501
            {
1502
                // Email addresses: <address@domain.foo>
1503
                string pattern =
1504
                    @"<
1505
                      (?:mailto:)?
1506
                      (
1507
                        [-.\w]+
1508
                        \@
1509
                        [-a-z0-9]+(\.[-a-z0-9]+)*\.[a-z]+
1510
                      )
1511
                      >";
1512
                text = Regex.Replace(text, pattern, new MatchEvaluator(EmailEvaluator), RegexOptions.IgnoreCase | RegexOptions.IgnorePatternWhitespace);
1513
            }
1514

    
1515
            return text;
1516
        }
1517

    
1518
        private string HyperlinkEvaluator(Match match)
1519
        {
1520
            string link = match.Groups[1].Value;
1521
            return string.Format("<a href=\"{0}\">{0}</a>", link);
1522
        }
1523

    
1524
        private string EmailEvaluator(Match match)
1525
        {
1526
            string email = Unescape(match.Groups[1].Value);
1527

    
1528
            //
1529
            //    Input: an email address, e.g. "foo@example.com"
1530
            //
1531
            //    Output: the email address as a mailto link, with each character
1532
            //            of the address encoded as either a decimal or hex entity, in
1533
            //            the hopes of foiling most address harvesting spam bots. E.g.:
1534
            //
1535
            //      <a href="&#x6D;&#97;&#105;&#108;&#x74;&#111;:&#102;&#111;&#111;&#64;&#101;
1536
            //        x&#x61;&#109;&#x70;&#108;&#x65;&#x2E;&#99;&#111;&#109;">&#102;&#111;&#111;
1537
            //        &#64;&#101;x&#x61;&#109;&#x70;&#108;&#x65;&#x2E;&#99;&#111;&#109;</a>
1538
            //
1539
            //    Based by a filter by Matthew Wickline, posted to the BBEdit-Talk
1540
            //    mailing list: <http://tinyurl.com/yu7ue>
1541
            //
1542
            email = "mailto:" + email;
1543

    
1544
            // leave ':' alone (to spot mailto: later)
1545
            email = EncodeEmailAddress(email);
1546

    
1547
            email = string.Format("<a href=\"{0}\">{0}</a>", email);
1548

    
1549
            // strip the mailto: from the visible part
1550
            email = Regex.Replace(email, "\">.+?:", "\">");
1551
            return email;
1552
        }
1553

    
1554

    
1555
        private static Regex _outDent = new Regex(@"^[ ]{1," + _tabWidth + @"}", RegexOptions.Multiline | RegexOptions.Compiled);
1556

    
1557
        /// <summary>
1558
        /// Remove one level of line-leading spaces
1559
        /// </summary>
1560
        private string Outdent(string block)
1561
        {
1562
            return _outDent.Replace(block, "");
1563
        }
1564

    
1565

    
1566
        #region Encoding and Normalization
1567

    
1568

    
1569
        /// <summary>
1570
        /// encodes email address randomly  
1571
        /// roughly 10% raw, 45% hex, 45% dec
1572
        /// note that @ is always encoded and : never is
1573
        /// </summary>
1574
        private string EncodeEmailAddress(string addr)
1575
        {
1576
            var sb = new StringBuilder(addr.Length * 5);
1577
            var rand = new Random();
1578
            int r;
1579
            foreach (char c in addr)
1580
            {
1581
                r = rand.Next(1, 100);
1582
                if ((r > 90 || c == ':') && c != '@')
1583
                    sb.Append(c);                         // m
1584
                else if (r < 45)
1585
                    sb.AppendFormat("&#x{0:x};", (int)c); // &#x6D
1586
                else
1587
                    sb.AppendFormat("&#{0};", (int)c);    // &#109
1588
            }
1589
            return sb.ToString();
1590
        }
1591

    
1592
        private static Regex _codeEncoder = new Regex(@"&|<|>|\\|\*|_|\{|\}|\[|\]", RegexOptions.Compiled);
1593

    
1594
        /// <summary>
1595
        /// Encode/escape certain Markdown characters inside code blocks and spans where they are literals
1596
        /// </summary>
1597
        private string EncodeCode(string code)
1598
        {
1599
            return _codeEncoder.Replace(code, EncodeCodeEvaluator);
1600
        }
1601
        private string EncodeCodeEvaluator(Match match)
1602
        {
1603
            switch (match.Value)
1604
            {
1605
                // Encode all ampersands; HTML entities are not
1606
                // entities within a Markdown code span.
1607
                case "&":
1608
                    return "&amp;";
1609
                // Do the angle bracket song and dance
1610
                case "<":
1611
                    return "&lt;";
1612
                case ">":
1613
                    return "&gt;";
1614
                // escape characters that are magic in Markdown
1615
                default:
1616
                    return _escapeTable[match.Value];
1617
            }
1618
        }
1619

    
1620

    
1621
        private static Regex _amps = new Regex(@"&(?!((#[0-9]+)|(#[xX][a-fA-F0-9]+)|([a-zA-Z][a-zA-Z0-9]*));)", RegexOptions.ExplicitCapture | RegexOptions.Compiled);
1622
        private static Regex _angles = new Regex(@"<(?![A-Za-z/?\$!])", RegexOptions.ExplicitCapture | RegexOptions.Compiled);
1623

    
1624
        /// <summary>
1625
        /// Encode any ampersands (that aren't part of an HTML entity) and left or right angle brackets
1626
        /// </summary>
1627
        private string EncodeAmpsAndAngles(string s)
1628
        {
1629
            s = _amps.Replace(s, "&amp;");
1630
            s = _angles.Replace(s, "&lt;");
1631
            return s;
1632
        }
1633

    
1634
        private static Regex _backslashEscapes;
1635

    
1636
        /// <summary>
1637
        /// Encodes any escaped characters such as \`, \*, \[ etc
1638
        /// </summary>
1639
        private string EscapeBackslashes(string s)
1640
        {
1641
            return _backslashEscapes.Replace(s, new MatchEvaluator(EscapeBackslashesEvaluator));
1642
        }
1643
        private string EscapeBackslashesEvaluator(Match match)
1644
        {
1645
            return _backslashEscapeTable[match.Value];
1646
        }
1647

    
1648
        private static Regex _unescapes = new Regex("\x1A" + "E\\d+E", RegexOptions.Compiled);
1649

    
1650
        /// <summary>
1651
        /// swap back in all the special characters we've hidden
1652
        /// </summary>
1653
        private string Unescape(string s)
1654
        {
1655
            return _unescapes.Replace(s, new MatchEvaluator(UnescapeEvaluator));
1656
        }
1657
        private string UnescapeEvaluator(Match match)
1658
        {
1659
            return _invertedEscapeTable[match.Value];
1660
        }
1661

    
1662

    
1663
        /// <summary>
1664
        /// escapes Bold [ * ] and Italic [ _ ] characters
1665
        /// </summary>
1666
        private string EscapeBoldItalic(string s)
1667
        {
1668
            s = s.Replace("*", _escapeTable["*"]);
1669
            s = s.Replace("_", _escapeTable["_"]);
1670
            return s;
1671
        }
1672

    
1673
        private static string AttributeEncode(string s)
1674
        {
1675
            return s.Replace(">", "&gt;").Replace("<", "&lt;").Replace("\"", "&quot;");
1676
        }
1677

    
1678
        private static char[] _problemUrlChars = @"""'*()[]$:".ToCharArray();
1679

    
1680
        /// <summary>
1681
        /// hex-encodes some unusual "problem" chars in URLs to avoid URL detection problems
1682
        /// </summary>
1683
        private string EncodeProblemUrlChars(string url)
1684
        {
1685
            if (!_encodeProblemUrlCharacters) return url;
1686

    
1687
            var sb = new StringBuilder(url.Length);
1688
            bool encode;
1689
            char c;
1690

    
1691
            for (int i = 0; i < url.Length; i++)
1692
            {
1693
                c = url[i];
1694
                encode = Array.IndexOf(_problemUrlChars, c) != -1;
1695
                if (encode && c == ':' && i < url.Length - 1)
1696
                    encode = !(url[i + 1] == '/') && !(url[i + 1] >= '0' && url[i + 1] <= '9');
1697

    
1698
                if (encode)
1699
                    sb.Append("%" + String.Format("{0:x}", (byte)c));
1700
                else
1701
                    sb.Append(c);
1702
            }
1703

    
1704
            return sb.ToString();
1705
        }
1706

    
1707

    
1708
        /// <summary>
1709
        /// Within tags -- meaning between &lt; and &gt; -- encode [\ ` * _] so they
1710
        /// don't conflict with their use in Markdown for code, italics and strong.
1711
        /// We're replacing each such character with its corresponding hash
1712
        /// value; this is likely overkill, but it should prevent us from colliding
1713
        /// with the escape values by accident.
1714
        /// </summary>
1715
        private string EscapeSpecialCharsWithinTagAttributes(string text)
1716
        {
1717
            var tokens = TokenizeHTML(text);
1718

    
1719
            // now, rebuild text from the tokens
1720
            var sb = new StringBuilder(text.Length);
1721

    
1722
            foreach (var token in tokens)
1723
            {
1724
                string value = token.Value;
1725

    
1726
                if (token.Type == TokenType.Tag)
1727
                {
1728
                    value = value.Replace(@"\", _escapeTable[@"\"]);
1729

    
1730
                    if (_autoHyperlink && value.StartsWith("<!")) // escape slashes in comments to prevent autolinking there -- http://meta.stackoverflow.com/questions/95987/html-comment-containing-url-breaks-if-followed-by-another-html-comment
1731
                        value = value.Replace("/", _escapeTable["/"]);
1732

    
1733
                    value = Regex.Replace(value, "(?<=.)</?code>(?=.)", _escapeTable[@"`"]);
1734
                    value = EscapeBoldItalic(value);
1735
                }
1736

    
1737
                sb.Append(value);
1738
            }
1739

    
1740
            return sb.ToString();
1741
        }
1742

    
1743
        /// <summary>
1744
        /// convert all tabs to _tabWidth spaces;
1745
        /// standardizes line endings from DOS (CR LF) or Mac (CR) to UNIX (LF);
1746
        /// makes sure text ends with a couple of newlines;
1747
        /// removes any blank lines (only spaces) in the text
1748
        /// </summary>
1749
        private string Normalize(string text)
1750
        {
1751
            var output = new StringBuilder(text.Length);
1752
            var line = new StringBuilder();
1753
            bool valid = false;
1754

    
1755
            for (int i = 0; i < text.Length; i++)
1756
            {
1757
                switch (text[i])
1758
                {
1759
                    case '\n':
1760
                        if (valid) output.Append(line);
1761
                        output.Append('\n');
1762
                        line.Length = 0; valid = false;
1763
                        break;
1764
                    case '\r':
1765
                        if ((i < text.Length - 1) && (text[i + 1] != '\n'))
1766
                        {
1767
                            if (valid) output.Append(line);
1768
                            output.Append('\n');
1769
                            line.Length = 0; valid = false;
1770
                        }
1771
                        break;
1772
                    case '\t':
1773
                        int width = (_tabWidth - line.Length % _tabWidth);
1774
                        for (int k = 0; k < width; k++)
1775
                            line.Append(' ');
1776
                        break;
1777
                    case '\x1A':
1778
                        break;
1779
                    default:
1780
                        if (!valid && text[i] != ' ') valid = true;
1781
                        line.Append(text[i]);
1782
                        break;
1783
                }
1784
            }
1785

    
1786
            if (valid) output.Append(line);
1787
            output.Append('\n');
1788

    
1789
            // add two newlines to the end before return
1790
            return output.Append("\n\n").ToString();
1791
        }
1792

    
1793
        #endregion
1794

    
1795
        /// <summary>
1796
        /// this is to emulate what's evailable in PHP
1797
        /// </summary>
1798
        private static string RepeatString(string text, int count)
1799
        {
1800
            var sb = new StringBuilder(text.Length * count);
1801
            for (int i = 0; i < count; i++)
1802
                sb.Append(text);
1803
            return sb.ToString();
1804
        }
1805

    
1806
    }
1807
}
클립보드 이미지 추가 (최대 크기: 500 MB)