755 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
			
		
		
	
	
			755 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
| // CodeMirror, copyright (c) by Marijn Haverbeke and others
 | ||
| // Distributed under an MIT license: http://codemirror.net/LICENSE
 | ||
| 
 | ||
| (function() {
 | ||
|   var mode = CodeMirror.getMode({tabSize: 4}, "markdown");
 | ||
|   function MT(name) { test.mode(name, mode, Array.prototype.slice.call(arguments, 1)); }
 | ||
|   var modeHighlightFormatting = CodeMirror.getMode({tabSize: 4}, {name: "markdown", highlightFormatting: true});
 | ||
|   function FT(name) { test.mode(name, modeHighlightFormatting, Array.prototype.slice.call(arguments, 1)); }
 | ||
| 
 | ||
|   FT("formatting_emAsterisk",
 | ||
|      "[em&formatting&formatting-em *][em foo][em&formatting&formatting-em *]");
 | ||
| 
 | ||
|   FT("formatting_emUnderscore",
 | ||
|      "[em&formatting&formatting-em _][em foo][em&formatting&formatting-em _]");
 | ||
| 
 | ||
|   FT("formatting_strongAsterisk",
 | ||
|      "[strong&formatting&formatting-strong **][strong foo][strong&formatting&formatting-strong **]");
 | ||
| 
 | ||
|   FT("formatting_strongUnderscore",
 | ||
|      "[strong&formatting&formatting-strong __][strong foo][strong&formatting&formatting-strong __]");
 | ||
| 
 | ||
|   FT("formatting_codeBackticks",
 | ||
|      "[comment&formatting&formatting-code `][comment foo][comment&formatting&formatting-code `]");
 | ||
| 
 | ||
|   FT("formatting_doubleBackticks",
 | ||
|      "[comment&formatting&formatting-code ``][comment foo ` bar][comment&formatting&formatting-code ``]");
 | ||
| 
 | ||
|   FT("formatting_atxHeader",
 | ||
|      "[header&header-1&formatting&formatting-header&formatting-header-1 #][header&header-1  foo # bar ][header&header-1&formatting&formatting-header&formatting-header-1 #]");
 | ||
| 
 | ||
|   FT("formatting_setextHeader",
 | ||
|      "foo",
 | ||
|      "[header&header-1&formatting&formatting-header&formatting-header-1 =]");
 | ||
| 
 | ||
|   FT("formatting_blockquote",
 | ||
|      "[quote"e-1&formatting&formatting-quote&formatting-quote-1 > ][quote"e-1 foo]");
 | ||
| 
 | ||
|   FT("formatting_list",
 | ||
|      "[variable-2&formatting&formatting-list&formatting-list-ul - ][variable-2 foo]");
 | ||
|   FT("formatting_list",
 | ||
|      "[variable-2&formatting&formatting-list&formatting-list-ol 1. ][variable-2 foo]");
 | ||
| 
 | ||
|   FT("formatting_link",
 | ||
|      "[link&formatting&formatting-link [][link foo][link&formatting&formatting-link ]]][string&formatting&formatting-link-string (][string http://example.com/][string&formatting&formatting-link-string )]");
 | ||
| 
 | ||
|   FT("formatting_linkReference",
 | ||
|      "[link&formatting&formatting-link [][link foo][link&formatting&formatting-link ]]][string&formatting&formatting-link-string [][string bar][string&formatting&formatting-link-string ]]]",
 | ||
|      "[link&formatting&formatting-link [][link bar][link&formatting&formatting-link ]]:] [string http://example.com/]");
 | ||
| 
 | ||
|   FT("formatting_linkWeb",
 | ||
|      "[link&formatting&formatting-link <][link http://example.com/][link&formatting&formatting-link >]");
 | ||
| 
 | ||
|   FT("formatting_linkEmail",
 | ||
|      "[link&formatting&formatting-link <][link user@example.com][link&formatting&formatting-link >]");
 | ||
| 
 | ||
|   FT("formatting_escape",
 | ||
|      "[formatting-escape \\*]");
 | ||
| 
 | ||
|   MT("plainText",
 | ||
|      "foo");
 | ||
| 
 | ||
|   // Don't style single trailing space
 | ||
|   MT("trailingSpace1",
 | ||
|      "foo ");
 | ||
| 
 | ||
|   // Two or more trailing spaces should be styled with line break character
 | ||
|   MT("trailingSpace2",
 | ||
|      "foo[trailing-space-a  ][trailing-space-new-line  ]");
 | ||
| 
 | ||
|   MT("trailingSpace3",
 | ||
|      "foo[trailing-space-a  ][trailing-space-b  ][trailing-space-new-line  ]");
 | ||
| 
 | ||
|   MT("trailingSpace4",
 | ||
|      "foo[trailing-space-a  ][trailing-space-b  ][trailing-space-a  ][trailing-space-new-line  ]");
 | ||
| 
 | ||
|   // Code blocks using 4 spaces (regardless of CodeMirror.tabSize value)
 | ||
|   MT("codeBlocksUsing4Spaces",
 | ||
|      "    [comment foo]");
 | ||
| 
 | ||
|   // Code blocks using 4 spaces with internal indentation
 | ||
|   MT("codeBlocksUsing4SpacesIndentation",
 | ||
|      "    [comment bar]",
 | ||
|      "        [comment hello]",
 | ||
|      "            [comment world]",
 | ||
|      "    [comment foo]",
 | ||
|      "bar");
 | ||
| 
 | ||
|   // Code blocks using 4 spaces with internal indentation
 | ||
|   MT("codeBlocksUsing4SpacesIndentation",
 | ||
|      " foo",
 | ||
|      "    [comment bar]",
 | ||
|      "        [comment hello]",
 | ||
|      "    [comment world]");
 | ||
| 
 | ||
|   // Code blocks should end even after extra indented lines
 | ||
|   MT("codeBlocksWithTrailingIndentedLine",
 | ||
|      "    [comment foo]",
 | ||
|      "        [comment bar]",
 | ||
|      "    [comment baz]",
 | ||
|      "    ",
 | ||
|      "hello");
 | ||
| 
 | ||
|   // Code blocks using 1 tab (regardless of CodeMirror.indentWithTabs value)
 | ||
|   MT("codeBlocksUsing1Tab",
 | ||
|      "\t[comment foo]");
 | ||
| 
 | ||
|   // Inline code using backticks
 | ||
|   MT("inlineCodeUsingBackticks",
 | ||
|      "foo [comment `bar`]");
 | ||
| 
 | ||
|   // Block code using single backtick (shouldn't work)
 | ||
|   MT("blockCodeSingleBacktick",
 | ||
|      "[comment `]",
 | ||
|      "foo",
 | ||
|      "[comment `]");
 | ||
| 
 | ||
|   // Unclosed backticks
 | ||
|   // Instead of simply marking as CODE, it would be nice to have an
 | ||
|   // incomplete flag for CODE, that is styled slightly different.
 | ||
|   MT("unclosedBackticks",
 | ||
|      "foo [comment `bar]");
 | ||
| 
 | ||
|   // Per documentation: "To include a literal backtick character within a
 | ||
|   // code span, you can use multiple backticks as the opening and closing
 | ||
|   // delimiters"
 | ||
|   MT("doubleBackticks",
 | ||
|      "[comment ``foo ` bar``]");
 | ||
| 
 | ||
|   // Tests based on Dingus
 | ||
|   // http://daringfireball.net/projects/markdown/dingus
 | ||
|   //
 | ||
|   // Multiple backticks within an inline code block
 | ||
|   MT("consecutiveBackticks",
 | ||
|      "[comment `foo```bar`]");
 | ||
| 
 | ||
|   // Multiple backticks within an inline code block with a second code block
 | ||
|   MT("consecutiveBackticks",
 | ||
|      "[comment `foo```bar`] hello [comment `world`]");
 | ||
| 
 | ||
|   // Unclosed with several different groups of backticks
 | ||
|   MT("unclosedBackticks",
 | ||
|      "[comment ``foo ``` bar` hello]");
 | ||
| 
 | ||
|   // Closed with several different groups of backticks
 | ||
|   MT("closedBackticks",
 | ||
|      "[comment ``foo ``` bar` hello``] world");
 | ||
| 
 | ||
|   // atx headers
 | ||
|   // http://daringfireball.net/projects/markdown/syntax#header
 | ||
| 
 | ||
|   MT("atxH1",
 | ||
|      "[header&header-1 # foo]");
 | ||
| 
 | ||
|   MT("atxH2",
 | ||
|      "[header&header-2 ## foo]");
 | ||
| 
 | ||
|   MT("atxH3",
 | ||
|      "[header&header-3 ### foo]");
 | ||
| 
 | ||
|   MT("atxH4",
 | ||
|      "[header&header-4 #### foo]");
 | ||
| 
 | ||
|   MT("atxH5",
 | ||
|      "[header&header-5 ##### foo]");
 | ||
| 
 | ||
|   MT("atxH6",
 | ||
|      "[header&header-6 ###### foo]");
 | ||
| 
 | ||
|   // H6 - 7x '#' should still be H6, per Dingus
 | ||
|   // http://daringfireball.net/projects/markdown/dingus
 | ||
|   MT("atxH6NotH7",
 | ||
|      "[header&header-6 ####### foo]");
 | ||
| 
 | ||
|   // Inline styles should be parsed inside headers
 | ||
|   MT("atxH1inline",
 | ||
|      "[header&header-1 # foo ][header&header-1&em *bar*]");
 | ||
| 
 | ||
|   // Setext headers - H1, H2
 | ||
|   // Per documentation, "Any number of underlining =’s or -’s will work."
 | ||
|   // http://daringfireball.net/projects/markdown/syntax#header
 | ||
|   // Ideally, the text would be marked as `header` as well, but this is
 | ||
|   // not really feasible at the moment. So, instead, we're testing against
 | ||
|   // what works today, to avoid any regressions.
 | ||
|   //
 | ||
|   // Check if single underlining = works
 | ||
|   MT("setextH1",
 | ||
|      "foo",
 | ||
|      "[header&header-1 =]");
 | ||
| 
 | ||
|   // Check if 3+ ='s work
 | ||
|   MT("setextH1",
 | ||
|      "foo",
 | ||
|      "[header&header-1 ===]");
 | ||
| 
 | ||
|   // Check if single underlining - works
 | ||
|   MT("setextH2",
 | ||
|      "foo",
 | ||
|      "[header&header-2 -]");
 | ||
| 
 | ||
|   // Check if 3+ -'s work
 | ||
|   MT("setextH2",
 | ||
|      "foo",
 | ||
|      "[header&header-2 ---]");
 | ||
| 
 | ||
|   // Single-line blockquote with trailing space
 | ||
|   MT("blockquoteSpace",
 | ||
|      "[quote"e-1 > foo]");
 | ||
| 
 | ||
|   // Single-line blockquote
 | ||
|   MT("blockquoteNoSpace",
 | ||
|      "[quote"e-1 >foo]");
 | ||
| 
 | ||
|   // No blank line before blockquote
 | ||
|   MT("blockquoteNoBlankLine",
 | ||
|      "foo",
 | ||
|      "[quote"e-1 > bar]");
 | ||
| 
 | ||
|   // Nested blockquote
 | ||
|   MT("blockquoteSpace",
 | ||
|      "[quote"e-1 > foo]",
 | ||
|      "[quote"e-1 >][quote"e-2 > foo]",
 | ||
|      "[quote"e-1 >][quote"e-2 >][quote"e-3 > foo]");
 | ||
| 
 | ||
|   // Single-line blockquote followed by normal paragraph
 | ||
|   MT("blockquoteThenParagraph",
 | ||
|      "[quote"e-1 >foo]",
 | ||
|      "",
 | ||
|      "bar");
 | ||
| 
 | ||
|   // Multi-line blockquote (lazy mode)
 | ||
|   MT("multiBlockquoteLazy",
 | ||
|      "[quote"e-1 >foo]",
 | ||
|      "[quote"e-1 bar]");
 | ||
| 
 | ||
|   // Multi-line blockquote followed by normal paragraph (lazy mode)
 | ||
|   MT("multiBlockquoteLazyThenParagraph",
 | ||
|      "[quote"e-1 >foo]",
 | ||
|      "[quote"e-1 bar]",
 | ||
|      "",
 | ||
|      "hello");
 | ||
| 
 | ||
|   // Multi-line blockquote (non-lazy mode)
 | ||
|   MT("multiBlockquote",
 | ||
|      "[quote"e-1 >foo]",
 | ||
|      "[quote"e-1 >bar]");
 | ||
| 
 | ||
|   // Multi-line blockquote followed by normal paragraph (non-lazy mode)
 | ||
|   MT("multiBlockquoteThenParagraph",
 | ||
|      "[quote"e-1 >foo]",
 | ||
|      "[quote"e-1 >bar]",
 | ||
|      "",
 | ||
|      "hello");
 | ||
| 
 | ||
|   // Check list types
 | ||
| 
 | ||
|   MT("listAsterisk",
 | ||
|      "foo",
 | ||
|      "bar",
 | ||
|      "",
 | ||
|      "[variable-2 * foo]",
 | ||
|      "[variable-2 * bar]");
 | ||
| 
 | ||
|   MT("listPlus",
 | ||
|      "foo",
 | ||
|      "bar",
 | ||
|      "",
 | ||
|      "[variable-2 + foo]",
 | ||
|      "[variable-2 + bar]");
 | ||
| 
 | ||
|   MT("listDash",
 | ||
|      "foo",
 | ||
|      "bar",
 | ||
|      "",
 | ||
|      "[variable-2 - foo]",
 | ||
|      "[variable-2 - bar]");
 | ||
| 
 | ||
|   MT("listNumber",
 | ||
|      "foo",
 | ||
|      "bar",
 | ||
|      "",
 | ||
|      "[variable-2 1. foo]",
 | ||
|      "[variable-2 2. bar]");
 | ||
| 
 | ||
|   // Lists require a preceding blank line (per Dingus)
 | ||
|   MT("listBogus",
 | ||
|      "foo",
 | ||
|      "1. bar",
 | ||
|      "2. hello");
 | ||
| 
 | ||
|   // List after header
 | ||
|   MT("listAfterHeader",
 | ||
|      "[header&header-1 # foo]",
 | ||
|      "[variable-2 - bar]");
 | ||
| 
 | ||
|   // Formatting in lists (*)
 | ||
|   MT("listAsteriskFormatting",
 | ||
|      "[variable-2 * ][variable-2&em *foo*][variable-2  bar]",
 | ||
|      "[variable-2 * ][variable-2&strong **foo**][variable-2  bar]",
 | ||
|      "[variable-2 * ][variable-2&strong **][variable-2&em&strong *foo**][variable-2&em *][variable-2  bar]",
 | ||
|      "[variable-2 * ][variable-2&comment `foo`][variable-2  bar]");
 | ||
| 
 | ||
|   // Formatting in lists (+)
 | ||
|   MT("listPlusFormatting",
 | ||
|      "[variable-2 + ][variable-2&em *foo*][variable-2  bar]",
 | ||
|      "[variable-2 + ][variable-2&strong **foo**][variable-2  bar]",
 | ||
|      "[variable-2 + ][variable-2&strong **][variable-2&em&strong *foo**][variable-2&em *][variable-2  bar]",
 | ||
|      "[variable-2 + ][variable-2&comment `foo`][variable-2  bar]");
 | ||
| 
 | ||
|   // Formatting in lists (-)
 | ||
|   MT("listDashFormatting",
 | ||
|      "[variable-2 - ][variable-2&em *foo*][variable-2  bar]",
 | ||
|      "[variable-2 - ][variable-2&strong **foo**][variable-2  bar]",
 | ||
|      "[variable-2 - ][variable-2&strong **][variable-2&em&strong *foo**][variable-2&em *][variable-2  bar]",
 | ||
|      "[variable-2 - ][variable-2&comment `foo`][variable-2  bar]");
 | ||
| 
 | ||
|   // Formatting in lists (1.)
 | ||
|   MT("listNumberFormatting",
 | ||
|      "[variable-2 1. ][variable-2&em *foo*][variable-2  bar]",
 | ||
|      "[variable-2 2. ][variable-2&strong **foo**][variable-2  bar]",
 | ||
|      "[variable-2 3. ][variable-2&strong **][variable-2&em&strong *foo**][variable-2&em *][variable-2  bar]",
 | ||
|      "[variable-2 4. ][variable-2&comment `foo`][variable-2  bar]");
 | ||
| 
 | ||
|   // Paragraph lists
 | ||
|   MT("listParagraph",
 | ||
|      "[variable-2 * foo]",
 | ||
|      "",
 | ||
|      "[variable-2 * bar]");
 | ||
| 
 | ||
|   // Multi-paragraph lists
 | ||
|   //
 | ||
|   // 4 spaces
 | ||
|   MT("listMultiParagraph",
 | ||
|      "[variable-2 * foo]",
 | ||
|      "",
 | ||
|      "[variable-2 * bar]",
 | ||
|      "",
 | ||
|      "    [variable-2 hello]");
 | ||
| 
 | ||
|   // 4 spaces, extra blank lines (should still be list, per Dingus)
 | ||
|   MT("listMultiParagraphExtra",
 | ||
|      "[variable-2 * foo]",
 | ||
|      "",
 | ||
|      "[variable-2 * bar]",
 | ||
|      "",
 | ||
|      "",
 | ||
|      "    [variable-2 hello]");
 | ||
| 
 | ||
|   // 4 spaces, plus 1 space (should still be list, per Dingus)
 | ||
|   MT("listMultiParagraphExtraSpace",
 | ||
|      "[variable-2 * foo]",
 | ||
|      "",
 | ||
|      "[variable-2 * bar]",
 | ||
|      "",
 | ||
|      "     [variable-2 hello]",
 | ||
|      "",
 | ||
|      "    [variable-2 world]");
 | ||
| 
 | ||
|   // 1 tab
 | ||
|   MT("listTab",
 | ||
|      "[variable-2 * foo]",
 | ||
|      "",
 | ||
|      "[variable-2 * bar]",
 | ||
|      "",
 | ||
|      "\t[variable-2 hello]");
 | ||
| 
 | ||
|   // No indent
 | ||
|   MT("listNoIndent",
 | ||
|      "[variable-2 * foo]",
 | ||
|      "",
 | ||
|      "[variable-2 * bar]",
 | ||
|      "",
 | ||
|      "hello");
 | ||
| 
 | ||
|   // Blockquote
 | ||
|   MT("blockquote",
 | ||
|      "[variable-2 * foo]",
 | ||
|      "",
 | ||
|      "[variable-2 * bar]",
 | ||
|      "",
 | ||
|      "    [variable-2"e"e-1 > hello]");
 | ||
| 
 | ||
|   // Code block
 | ||
|   MT("blockquoteCode",
 | ||
|      "[variable-2 * foo]",
 | ||
|      "",
 | ||
|      "[variable-2 * bar]",
 | ||
|      "",
 | ||
|      "        [comment > hello]",
 | ||
|      "",
 | ||
|      "    [variable-2 world]");
 | ||
| 
 | ||
|   // Code block followed by text
 | ||
|   MT("blockquoteCodeText",
 | ||
|      "[variable-2 * foo]",
 | ||
|      "",
 | ||
|      "    [variable-2 bar]",
 | ||
|      "",
 | ||
|      "        [comment hello]",
 | ||
|      "",
 | ||
|      "    [variable-2 world]");
 | ||
| 
 | ||
|   // Nested list
 | ||
| 
 | ||
|   MT("listAsteriskNested",
 | ||
|      "[variable-2 * foo]",
 | ||
|      "",
 | ||
|      "    [variable-3 * bar]");
 | ||
| 
 | ||
|   MT("listPlusNested",
 | ||
|      "[variable-2 + foo]",
 | ||
|      "",
 | ||
|      "    [variable-3 + bar]");
 | ||
| 
 | ||
|   MT("listDashNested",
 | ||
|      "[variable-2 - foo]",
 | ||
|      "",
 | ||
|      "    [variable-3 - bar]");
 | ||
| 
 | ||
|   MT("listNumberNested",
 | ||
|      "[variable-2 1. foo]",
 | ||
|      "",
 | ||
|      "    [variable-3 2. bar]");
 | ||
| 
 | ||
|   MT("listMixed",
 | ||
|      "[variable-2 * foo]",
 | ||
|      "",
 | ||
|      "    [variable-3 + bar]",
 | ||
|      "",
 | ||
|      "        [keyword - hello]",
 | ||
|      "",
 | ||
|      "            [variable-2 1. world]");
 | ||
| 
 | ||
|   MT("listBlockquote",
 | ||
|      "[variable-2 * foo]",
 | ||
|      "",
 | ||
|      "    [variable-3 + bar]",
 | ||
|      "",
 | ||
|      "        [quote"e-1&variable-3 > hello]");
 | ||
| 
 | ||
|   MT("listCode",
 | ||
|      "[variable-2 * foo]",
 | ||
|      "",
 | ||
|      "    [variable-3 + bar]",
 | ||
|      "",
 | ||
|      "            [comment hello]");
 | ||
| 
 | ||
|   // Code with internal indentation
 | ||
|   MT("listCodeIndentation",
 | ||
|      "[variable-2 * foo]",
 | ||
|      "",
 | ||
|      "        [comment bar]",
 | ||
|      "            [comment hello]",
 | ||
|      "                [comment world]",
 | ||
|      "        [comment foo]",
 | ||
|      "    [variable-2 bar]");
 | ||
| 
 | ||
|   // List nesting edge cases
 | ||
|   MT("listNested",
 | ||
|     "[variable-2 * foo]",
 | ||
|     "",
 | ||
|     "    [variable-3 * bar]",
 | ||
|     "",
 | ||
|     "       [variable-2 hello]"
 | ||
|   );
 | ||
|   MT("listNested",
 | ||
|     "[variable-2 * foo]",
 | ||
|     "",
 | ||
|     "    [variable-3 * bar]",
 | ||
|     "",
 | ||
|     "      [variable-3 * foo]"
 | ||
|   );
 | ||
| 
 | ||
|   // Code followed by text
 | ||
|   MT("listCodeText",
 | ||
|      "[variable-2 * foo]",
 | ||
|      "",
 | ||
|      "        [comment bar]",
 | ||
|      "",
 | ||
|      "hello");
 | ||
| 
 | ||
|   // Following tests directly from official Markdown documentation
 | ||
|   // http://daringfireball.net/projects/markdown/syntax#hr
 | ||
| 
 | ||
|   MT("hrSpace",
 | ||
|      "[hr * * *]");
 | ||
| 
 | ||
|   MT("hr",
 | ||
|      "[hr ***]");
 | ||
| 
 | ||
|   MT("hrLong",
 | ||
|      "[hr *****]");
 | ||
| 
 | ||
|   MT("hrSpaceDash",
 | ||
|      "[hr - - -]");
 | ||
| 
 | ||
|   MT("hrDashLong",
 | ||
|      "[hr ---------------------------------------]");
 | ||
| 
 | ||
|   // Inline link with title
 | ||
|   MT("linkTitle",
 | ||
|      "[link [[foo]]][string (http://example.com/ \"bar\")] hello");
 | ||
| 
 | ||
|   // Inline link without title
 | ||
|   MT("linkNoTitle",
 | ||
|      "[link [[foo]]][string (http://example.com/)] bar");
 | ||
| 
 | ||
|   // Inline link with image
 | ||
|   MT("linkImage",
 | ||
|      "[link [[][tag ![[foo]]][string (http://example.com/)][link ]]][string (http://example.com/)] bar");
 | ||
| 
 | ||
|   // Inline link with Em
 | ||
|   MT("linkEm",
 | ||
|      "[link [[][link&em *foo*][link ]]][string (http://example.com/)] bar");
 | ||
| 
 | ||
|   // Inline link with Strong
 | ||
|   MT("linkStrong",
 | ||
|      "[link [[][link&strong **foo**][link ]]][string (http://example.com/)] bar");
 | ||
| 
 | ||
|   // Inline link with EmStrong
 | ||
|   MT("linkEmStrong",
 | ||
|      "[link [[][link&strong **][link&em&strong *foo**][link&em *][link ]]][string (http://example.com/)] bar");
 | ||
| 
 | ||
|   // Image with title
 | ||
|   MT("imageTitle",
 | ||
|      "[tag ![[foo]]][string (http://example.com/ \"bar\")] hello");
 | ||
| 
 | ||
|   // Image without title
 | ||
|   MT("imageNoTitle",
 | ||
|      "[tag ![[foo]]][string (http://example.com/)] bar");
 | ||
| 
 | ||
|   // Image with asterisks
 | ||
|   MT("imageAsterisks",
 | ||
|      "[tag ![[*foo*]]][string (http://example.com/)] bar");
 | ||
| 
 | ||
|   // Not a link. Should be normal text due to square brackets being used
 | ||
|   // regularly in text, especially in quoted material, and no space is allowed
 | ||
|   // between square brackets and parentheses (per Dingus).
 | ||
|   MT("notALink",
 | ||
|      "[[foo]] (bar)");
 | ||
| 
 | ||
|   // Reference-style links
 | ||
|   MT("linkReference",
 | ||
|      "[link [[foo]]][string [[bar]]] hello");
 | ||
| 
 | ||
|   // Reference-style links with Em
 | ||
|   MT("linkReferenceEm",
 | ||
|      "[link [[][link&em *foo*][link ]]][string [[bar]]] hello");
 | ||
| 
 | ||
|   // Reference-style links with Strong
 | ||
|   MT("linkReferenceStrong",
 | ||
|      "[link [[][link&strong **foo**][link ]]][string [[bar]]] hello");
 | ||
| 
 | ||
|   // Reference-style links with EmStrong
 | ||
|   MT("linkReferenceEmStrong",
 | ||
|      "[link [[][link&strong **][link&em&strong *foo**][link&em *][link ]]][string [[bar]]] hello");
 | ||
| 
 | ||
|   // Reference-style links with optional space separator (per docuentation)
 | ||
|   // "You can optionally use a space to separate the sets of brackets"
 | ||
|   MT("linkReferenceSpace",
 | ||
|      "[link [[foo]]] [string [[bar]]] hello");
 | ||
| 
 | ||
|   // Should only allow a single space ("...use *a* space...")
 | ||
|   MT("linkReferenceDoubleSpace",
 | ||
|      "[[foo]]  [[bar]] hello");
 | ||
| 
 | ||
|   // Reference-style links with implicit link name
 | ||
|   MT("linkImplicit",
 | ||
|      "[link [[foo]]][string [[]]] hello");
 | ||
| 
 | ||
|   // @todo It would be nice if, at some point, the document was actually
 | ||
|   // checked to see if the referenced link exists
 | ||
| 
 | ||
|   // Link label, for reference-style links (taken from documentation)
 | ||
| 
 | ||
|   MT("labelNoTitle",
 | ||
|      "[link [[foo]]:] [string http://example.com/]");
 | ||
| 
 | ||
|   MT("labelIndented",
 | ||
|      "   [link [[foo]]:] [string http://example.com/]");
 | ||
| 
 | ||
|   MT("labelSpaceTitle",
 | ||
|      "[link [[foo bar]]:] [string http://example.com/ \"hello\"]");
 | ||
| 
 | ||
|   MT("labelDoubleTitle",
 | ||
|      "[link [[foo bar]]:] [string http://example.com/ \"hello\"] \"world\"");
 | ||
| 
 | ||
|   MT("labelTitleDoubleQuotes",
 | ||
|      "[link [[foo]]:] [string http://example.com/  \"bar\"]");
 | ||
| 
 | ||
|   MT("labelTitleSingleQuotes",
 | ||
|      "[link [[foo]]:] [string http://example.com/  'bar']");
 | ||
| 
 | ||
|   MT("labelTitleParenthese",
 | ||
|      "[link [[foo]]:] [string http://example.com/  (bar)]");
 | ||
| 
 | ||
|   MT("labelTitleInvalid",
 | ||
|      "[link [[foo]]:] [string http://example.com/] bar");
 | ||
| 
 | ||
|   MT("labelLinkAngleBrackets",
 | ||
|      "[link [[foo]]:] [string <http://example.com/>  \"bar\"]");
 | ||
| 
 | ||
|   MT("labelTitleNextDoubleQuotes",
 | ||
|      "[link [[foo]]:] [string http://example.com/]",
 | ||
|      "[string \"bar\"] hello");
 | ||
| 
 | ||
|   MT("labelTitleNextSingleQuotes",
 | ||
|      "[link [[foo]]:] [string http://example.com/]",
 | ||
|      "[string 'bar'] hello");
 | ||
| 
 | ||
|   MT("labelTitleNextParenthese",
 | ||
|      "[link [[foo]]:] [string http://example.com/]",
 | ||
|      "[string (bar)] hello");
 | ||
| 
 | ||
|   MT("labelTitleNextMixed",
 | ||
|      "[link [[foo]]:] [string http://example.com/]",
 | ||
|      "(bar\" hello");
 | ||
| 
 | ||
|   MT("linkWeb",
 | ||
|      "[link <http://example.com/>] foo");
 | ||
| 
 | ||
|   MT("linkWebDouble",
 | ||
|      "[link <http://example.com/>] foo [link <http://example.com/>]");
 | ||
| 
 | ||
|   MT("linkEmail",
 | ||
|      "[link <user@example.com>] foo");
 | ||
| 
 | ||
|   MT("linkEmailDouble",
 | ||
|      "[link <user@example.com>] foo [link <user@example.com>]");
 | ||
| 
 | ||
|   MT("emAsterisk",
 | ||
|      "[em *foo*] bar");
 | ||
| 
 | ||
|   MT("emUnderscore",
 | ||
|      "[em _foo_] bar");
 | ||
| 
 | ||
|   MT("emInWordAsterisk",
 | ||
|      "foo[em *bar*]hello");
 | ||
| 
 | ||
|   MT("emInWordUnderscore",
 | ||
|      "foo[em _bar_]hello");
 | ||
| 
 | ||
|   // Per documentation: "...surround an * or _ with spaces, it’ll be
 | ||
|   // treated as a literal asterisk or underscore."
 | ||
| 
 | ||
|   MT("emEscapedBySpaceIn",
 | ||
|      "foo [em _bar _ hello_] world");
 | ||
| 
 | ||
|   MT("emEscapedBySpaceOut",
 | ||
|      "foo _ bar[em _hello_]world");
 | ||
| 
 | ||
|   MT("emEscapedByNewline",
 | ||
|      "foo",
 | ||
|      "_ bar[em _hello_]world");
 | ||
| 
 | ||
|   // Unclosed emphasis characters
 | ||
|   // Instead of simply marking as EM / STRONG, it would be nice to have an
 | ||
|   // incomplete flag for EM and STRONG, that is styled slightly different.
 | ||
|   MT("emIncompleteAsterisk",
 | ||
|      "foo [em *bar]");
 | ||
| 
 | ||
|   MT("emIncompleteUnderscore",
 | ||
|      "foo [em _bar]");
 | ||
| 
 | ||
|   MT("strongAsterisk",
 | ||
|      "[strong **foo**] bar");
 | ||
| 
 | ||
|   MT("strongUnderscore",
 | ||
|      "[strong __foo__] bar");
 | ||
| 
 | ||
|   MT("emStrongAsterisk",
 | ||
|      "[em *foo][em&strong **bar*][strong hello**] world");
 | ||
| 
 | ||
|   MT("emStrongUnderscore",
 | ||
|      "[em _foo][em&strong __bar_][strong hello__] world");
 | ||
| 
 | ||
|   // "...same character must be used to open and close an emphasis span.""
 | ||
|   MT("emStrongMixed",
 | ||
|      "[em _foo][em&strong **bar*hello__ world]");
 | ||
| 
 | ||
|   MT("emStrongMixed",
 | ||
|      "[em *foo][em&strong __bar_hello** world]");
 | ||
| 
 | ||
|   // These characters should be escaped:
 | ||
|   // \   backslash
 | ||
|   // `   backtick
 | ||
|   // *   asterisk
 | ||
|   // _   underscore
 | ||
|   // {}  curly braces
 | ||
|   // []  square brackets
 | ||
|   // ()  parentheses
 | ||
|   // #   hash mark
 | ||
|   // +   plus sign
 | ||
|   // -   minus sign (hyphen)
 | ||
|   // .   dot
 | ||
|   // !   exclamation mark
 | ||
| 
 | ||
|   MT("escapeBacktick",
 | ||
|      "foo \\`bar\\`");
 | ||
| 
 | ||
|   MT("doubleEscapeBacktick",
 | ||
|      "foo \\\\[comment `bar\\\\`]");
 | ||
| 
 | ||
|   MT("escapeAsterisk",
 | ||
|      "foo \\*bar\\*");
 | ||
| 
 | ||
|   MT("doubleEscapeAsterisk",
 | ||
|      "foo \\\\[em *bar\\\\*]");
 | ||
| 
 | ||
|   MT("escapeUnderscore",
 | ||
|      "foo \\_bar\\_");
 | ||
| 
 | ||
|   MT("doubleEscapeUnderscore",
 | ||
|      "foo \\\\[em _bar\\\\_]");
 | ||
| 
 | ||
|   MT("escapeHash",
 | ||
|      "\\# foo");
 | ||
| 
 | ||
|   MT("doubleEscapeHash",
 | ||
|      "\\\\# foo");
 | ||
| 
 | ||
|   MT("escapeNewline",
 | ||
|      "\\",
 | ||
|      "[em *foo*]");
 | ||
| 
 | ||
| 
 | ||
|   // Tests to make sure GFM-specific things aren't getting through
 | ||
| 
 | ||
|   MT("taskList",
 | ||
|      "[variable-2 * [ ]] bar]");
 | ||
| 
 | ||
|   MT("fencedCodeBlocks",
 | ||
|      "[comment ```]",
 | ||
|      "foo",
 | ||
|      "[comment ```]");
 | ||
| 
 | ||
|   // Tests that require XML mode
 | ||
| 
 | ||
|   MT("xmlMode",
 | ||
|      "[tag&bracket <][tag div][tag&bracket >]",
 | ||
|      "*foo*",
 | ||
|      "[tag&bracket <][tag http://github.com][tag&bracket />]",
 | ||
|      "[tag&bracket </][tag div][tag&bracket >]",
 | ||
|      "[link <http://github.com/>]");
 | ||
| 
 | ||
|   MT("xmlModeWithMarkdownInside",
 | ||
|      "[tag&bracket <][tag div] [attribute markdown]=[string 1][tag&bracket >]",
 | ||
|      "[em *foo*]",
 | ||
|      "[link <http://github.com/>]",
 | ||
|      "[tag </div>]",
 | ||
|      "[link <http://github.com/>]",
 | ||
|      "[tag&bracket <][tag div][tag&bracket >]",
 | ||
|      "[tag&bracket </][tag div][tag&bracket >]");
 | ||
| 
 | ||
| })();
 |