diff --git a/src/Lexer.js b/src/Lexer.js index c1d8b03cb4..d34aa47bc0 100644 --- a/src/Lexer.js +++ b/src/Lexer.js @@ -163,10 +163,9 @@ module.exports = class Lexer { src = src.substring(token.raw.length); lastToken = tokens[tokens.length - 1]; // An indented code block cannot interrupt a paragraph. - if (lastToken && lastToken.type === 'paragraph') { + if (lastToken && (lastToken.type === 'paragraph' || lastToken.type === 'text')) { lastToken.raw += '\n' + token.raw; lastToken.text += '\n' + token.text; - this.inlineQueue.pop(); this.inlineQueue[this.inlineQueue.length - 1].src = lastToken.text; } else { tokens.push(token); @@ -217,9 +216,14 @@ module.exports = class Lexer { } // def - if (this.state.top && (token = this.tokenizer.def(src))) { + if (token = this.tokenizer.def(src)) { src = src.substring(token.raw.length); - if (!this.tokens.links[token.tag]) { + lastToken = tokens[tokens.length - 1]; + if (lastToken && (lastToken.type === 'paragraph' || lastToken.type === 'text')) { + lastToken.raw += '\n' + token.raw; + lastToken.text += '\n' + token.raw; + this.inlineQueue[this.inlineQueue.length - 1].src = lastToken.text; + } else if (!this.tokens.links[token.tag]) { this.tokens.links[token.tag] = { href: token.href, title: token.title diff --git a/src/Parser.js b/src/Parser.js index e804c143c4..ef5009aea8 100644 --- a/src/Parser.js +++ b/src/Parser.js @@ -149,7 +149,7 @@ module.exports = class Parser { if (item.task) { checkbox = this.renderer.checkbox(checked); if (loose) { - if (item.tokens.length > 0 && item.tokens[0].type === 'text') { + if (item.tokens.length > 0 && item.tokens[0].type === 'paragraph') { item.tokens[0].text = checkbox + ' ' + item.tokens[0].text; if (item.tokens[0].tokens && item.tokens[0].tokens.length > 0 && item.tokens[0].tokens[0].type === 'text') { item.tokens[0].tokens[0].text = checkbox + ' ' + item.tokens[0].tokens[0].text; diff --git a/src/Tokenizer.js b/src/Tokenizer.js index 922452e7be..85bdb33675 100644 --- a/src/Tokenizer.js +++ b/src/Tokenizer.js @@ -164,145 +164,149 @@ module.exports = class Tokenizer { } list(src) { - const cap = this.rules.block.list.exec(src); + let cap = this.rules.block.list.exec(src); if (cap) { - let raw = cap[0]; - const bull = cap[2]; + let raw, istask, ischecked, indent, i, blankLine, endsWithBlankLine, + line, lines, itemContents; + + let bull = cap[1].trim(); const isordered = bull.length > 1; const list = { type: 'list', - raw, + raw: '', ordered: isordered, start: isordered ? +bull.slice(0, -1) : '', loose: false, items: [] }; - // Get each top-level item. - const itemMatch = cap[0].match(this.rules.block.item); - - let next = false, - item, - space, - bcurr, - bnext, - addBack, - loose, - istask, - ischecked, - endMatch; - - let l = itemMatch.length; - bcurr = this.rules.block.listItemStart.exec(itemMatch[0]); - for (let i = 0; i < l; i++) { - item = itemMatch[i]; - raw = item; - - if (!this.options.pedantic) { - // Determine if current item contains the end of the list - endMatch = item.match(new RegExp('\\n\\s*\\n {0,' + (bcurr[0].length - 1) + '}\\S')); - if (endMatch) { - addBack = item.length - endMatch.index + itemMatch.slice(i + 1).join('\n').length; - list.raw = list.raw.substring(0, list.raw.length - addBack); - - item = item.substring(0, endMatch.index); - raw = item; - l = i + 1; - } + bull = isordered ? `\\d{1,9}\\${bull.slice(-1)}` : `\\${bull}`; + + if (this.options.pedantic) { + bull = isordered ? bull : '[*+-]'; + } + + // Get next list item + const itemRegex = new RegExp(`^( {0,3}${bull})((?: [^\\n]*| *)(?:\\n[^\\n]*)*(?:\\n|$))`); + + // Get each top-level item + while (src) { + if (this.rules.block.hr.test(src)) { // End list if we encounter an HR (possibly move into itemRegex?) + break; } - // Determine whether the next list item belongs here. - // Backpedal if it does not belong in this list. - if (i !== l - 1) { - bnext = this.rules.block.listItemStart.exec(itemMatch[i + 1]); - if ( - !this.options.pedantic - ? bnext[1].length >= bcurr[0].length || bnext[1].length > 3 - : bnext[1].length > bcurr[1].length - ) { - // nested list or continuation - itemMatch.splice(i, 2, itemMatch[i] + (!this.options.pedantic && bnext[1].length < bcurr[0].length && !itemMatch[i].match(/\n$/) ? '' : '\n') + itemMatch[i + 1]); - i--; - l--; - continue; - } else if ( - // different bullet style - !this.options.pedantic || this.options.smartLists - ? bnext[2][bnext[2].length - 1] !== bull[bull.length - 1] - : isordered === (bnext[2].length === 1) - ) { - addBack = itemMatch.slice(i + 1).join('\n').length; - list.raw = list.raw.substring(0, list.raw.length - addBack); - i = l - 1; - } - bcurr = bnext; + if (!(cap = itemRegex.exec(src))) { + break; } - // Remove the list item's bullet - // so it is seen as the next token. - space = item.length; - item = item.replace(/^ *([*+-]|\d+[.)]) ?/, ''); - - // Outdent whatever the - // list item contains. Hacky. - if (~item.indexOf('\n ')) { - space -= item.length; - item = !this.options.pedantic - ? item.replace(new RegExp('^ {1,' + space + '}', 'gm'), '') - : item.replace(/^ {1,4}/gm, ''); + lines = cap[2].split('\n'); + + if (this.options.pedantic) { + indent = 2; + itemContents = lines[0].trimLeft(); + } else { + indent = cap[2].search(/[^ ]/); // Find first non-space char + indent = cap[1].length + (indent > 4 ? 1 : indent); // intented code blocks after 4 spaces; indent is always 1 + itemContents = lines[0].slice(indent - cap[1].length); } - // trim item newlines at end - item = rtrim(item, '\n'); - if (i !== l - 1) { - raw = raw + '\n'; + blankLine = false; + raw = cap[0]; + + if (!lines[0] && /^ *$/.test(lines[1])) { // items begin with at most one blank line + raw = cap[1] + lines.slice(0, 2).join('\n') + '\n'; + list.loose = true; + lines = []; } - // Determine whether item is loose or not. - // Use: /(^|\n)(?! )[^\n]+\n\n(?!\s*$)/ - // for discount behavior. - loose = next || /\n\n(?!\s*$)/.test(raw); - if (i !== l - 1) { - next = raw.slice(-2) === '\n\n'; - if (!loose) loose = next; + const nextBulletRegex = new RegExp(`^ {0,${Math.min(3, indent - 1)}}(?:[*+-]|\\d{1,9}[.)])`); + + for (i = 1; i < lines.length; i++) { + line = lines[i]; + + if (this.options.pedantic) { // Re-align to follow commonmark nesting rules + line = line.replace(/^ {1,4}(?=( {4})*[^ ])/g, ' '); + } + + // End list item if found start of new bullet + if (nextBulletRegex.test(line)) { + raw = cap[1] + lines.slice(0, i).join('\n') + '\n'; + break; + } + + // Until we encounter a blank line, item contents do not need indentation + if (!blankLine) { + if (!line.trim()) { // Check if current line is empty + blankLine = true; + } + + // Dedent if possible + if (line.search(/[^ ]/) >= indent) { + itemContents += '\n' + line.slice(indent); + } else { + itemContents += '\n' + line; + } + continue; + } + + // Dedent this line + if (line.search(/[^ ]/) >= indent || !line.trim()) { + itemContents += '\n' + line.slice(indent); + continue; + } else { // Line was not properly indented; end of this item + raw = cap[1] + lines.slice(0, i).join('\n') + '\n'; + break; + } } - if (loose) { - list.loose = true; + if (!list.loose) { + // If the previous item ended with a blank line, the list is loose + if (endsWithBlankLine) { + list.loose = true; + } else if (/\n *\n *$/.test(raw)) { + endsWithBlankLine = true; + } } // Check for task list items if (this.options.gfm) { - istask = /^\[[ xX]\] /.test(item); - ischecked = undefined; + istask = /^\[[ xX]\] /.exec(itemContents); if (istask) { - ischecked = item[1] !== ' '; - item = item.replace(/^\[[ xX]\] +/, ''); + ischecked = istask[0] !== '[ ] '; + itemContents = itemContents.replace(/^\[[ xX]\] +/, ''); } } - this.lexer.state.top = false; - - const token = { + list.items.push({ type: 'list_item', - raw, - task: istask, + raw: raw, + task: !!istask, checked: ischecked, - loose: loose, - text: item, - tokens: this.lexer.blockTokens(item, []) - }; + loose: false, + text: itemContents + }); - // this.lexer.inline(token.text, ) - list.items.push(token); + list.raw += raw; + src = src.slice(raw.length); } - // l2 = token.items.length; - // for (j = 0; j < l2; j++) { - // this.inline(token.items[j].tokens); - // } - // break; + // Do not consume newlines at end of final item. Alternatively, make itemRegex *start* with any newlines to simplify/speed up endsWithBlankLine logic + list.items[list.items.length - 1].raw = raw.trimRight(); + list.items[list.items.length - 1].text = itemContents.trimRight(); + list.raw = list.raw.trimRight(); + + const l = list.items.length; + + // Item child tokens handled here at end because we needed to have the final item to trim it first + for (i = 0; i < l; i++) { + this.lexer.state.top = false; + list.items[i].tokens = this.lexer.blockTokens(list.items[i].text, []); + if (list.items[i].tokens.some(t => t.type === 'space')) { + list.loose = true; + list.items[i].loose = true; + } + } return list; } diff --git a/src/rules.js b/src/rules.js index 13961f2251..39fa8e56c4 100644 --- a/src/rules.js +++ b/src/rules.js @@ -10,11 +10,11 @@ const { const block = { newline: /^(?: *(?:\n|$))+/, code: /^( {4}[^\n]+(?:\n(?: *(?:\n|$))*)?)+/, - fences: /^ {0,3}(`{3,}(?=[^`\n]*\n)|~{3,})([^\n]*)\n(?:|([\s\S]*?)\n)(?: {0,3}\1[~`]* *(?:\n+|$)|$)/, + fences: /^ {0,3}(`{3,}(?=[^`\n]*\n)|~{3,})([^\n]*)\n(?:|([\s\S]*?)\n)(?: {0,3}\1[~`]* *(?=\n|$)|$)/, hr: /^ {0,3}((?:- *){3,}|(?:_ *){3,}|(?:\* *){3,})(?:\n+|$)/, heading: /^ {0,3}(#{1,6})(?=\s|$)(.*)(?:\n+|$)/, blockquote: /^( {0,3}> ?(paragraph|[^\n]*)(?:\n|$))+/, - list: /^( {0,3})(bull) [\s\S]+?(?:hr|def|\n{2,}(?! )(?! {0,3}bull )\n*|\s*$)/, + list: /^( {0,3}bull)( [^\n]+?)?(?:\n|$)/, html: '^ {0,3}(?:' // optional indentation + '<(script|pre|style|textarea)[\\s>][\\s\\S]*?(?:[^\\n]*\\n+|$)' // (1) + '|comment[^\\n]*(\\n+|$)' // (2) @@ -42,11 +42,6 @@ block.def = edit(block.def) .getRegex(); block.bullet = /(?:[*+-]|\d{1,9}[.)])/; -block.item = /^( *)(bull) ?[^\n]*(?:\n(?! *bull ?)[^\n]*)*/; -block.item = edit(block.item, 'gm') - .replace(/bull/g, block.bullet) - .getRegex(); - block.listItemStart = edit(/^( *)(bull) */) .replace('bull', block.bullet) .getRegex(); diff --git a/test/specs/commonmark/commonmark.0.30.json b/test/specs/commonmark/commonmark.0.30.json index 18a49682bd..62a0dfe076 100644 --- a/test/specs/commonmark/commonmark.0.30.json +++ b/test/specs/commonmark/commonmark.0.30.json @@ -2107,8 +2107,7 @@ "example": 262, "start_line": 4314, "end_line": 4326, - "section": "List items", - "shouldFail": true + "section": "List items" }, { "markdown": "1. foo\n\n ```\n bar\n ```\n\n baz\n\n > bam\n", @@ -2124,8 +2123,7 @@ "example": 264, "start_line": 4359, "end_line": 4377, - "section": "List items", - "shouldFail": true + "section": "List items" }, { "markdown": "123456789. ok\n", @@ -2197,8 +2195,7 @@ "example": 273, "start_line": 4493, "end_line": 4509, - "section": "List items", - "shouldFail": true + "section": "List items" }, { "markdown": "1. indented code\n\n paragraph\n\n more code\n", @@ -2206,8 +2203,7 @@ "example": 274, "start_line": 4515, "end_line": 4531, - "section": "List items", - "shouldFail": true + "section": "List items" }, { "markdown": " foo\n\nbar\n", @@ -2239,8 +2235,7 @@ "example": 278, "start_line": 4596, "end_line": 4617, - "section": "List items", - "shouldFail": true + "section": "List items" }, { "markdown": "- \n foo\n", @@ -2256,8 +2251,7 @@ "example": 280, "start_line": 4636, "end_line": 4645, - "section": "List items", - "shouldFail": true + "section": "List items" }, { "markdown": "- foo\n-\n- bar\n", @@ -2289,8 +2283,7 @@ "example": 284, "start_line": 4695, "end_line": 4701, - "section": "List items", - "shouldFail": true + "section": "List items" }, { "markdown": "foo\n*\n\nfoo\n1.\n", @@ -2466,8 +2459,7 @@ "example": 306, "start_line": 5388, "end_line": 5407, - "section": "Lists", - "shouldFail": true + "section": "Lists" }, { "markdown": "- foo\n - bar\n - baz\n\n\n bim\n", @@ -2475,8 +2467,7 @@ "example": 307, "start_line": 5409, "end_line": 5431, - "section": "Lists", - "shouldFail": true + "section": "Lists" }, { "markdown": "- foo\n- bar\n\n\n\n- baz\n- bim\n", @@ -2556,8 +2547,7 @@ "example": 317, "start_line": 5645, "end_line": 5663, - "section": "Lists", - "shouldFail": true + "section": "Lists" }, { "markdown": "- a\n- ```\n b\n\n\n ```\n- c\n", @@ -2565,8 +2555,7 @@ "example": 318, "start_line": 5668, "end_line": 5687, - "section": "Lists", - "shouldFail": true + "section": "Lists" }, { "markdown": "- a\n - b\n\n c\n- d\n", @@ -2574,8 +2563,7 @@ "example": 319, "start_line": 5694, "end_line": 5712, - "section": "Lists", - "shouldFail": true + "section": "Lists" }, { "markdown": "* a\n > b\n >\n* c\n", diff --git a/test/specs/gfm/commonmark.0.30.json b/test/specs/gfm/commonmark.0.30.json index 1b01311866..e32558a3a3 100644 --- a/test/specs/gfm/commonmark.0.30.json +++ b/test/specs/gfm/commonmark.0.30.json @@ -2107,8 +2107,7 @@ "example": 262, "start_line": 4314, "end_line": 4326, - "section": "List items", - "shouldFail": true + "section": "List items" }, { "markdown": "1. foo\n\n ```\n bar\n ```\n\n baz\n\n > bam\n", @@ -2124,8 +2123,7 @@ "example": 264, "start_line": 4359, "end_line": 4377, - "section": "List items", - "shouldFail": true + "section": "List items" }, { "markdown": "123456789. ok\n", @@ -2197,8 +2195,7 @@ "example": 273, "start_line": 4493, "end_line": 4509, - "section": "List items", - "shouldFail": true + "section": "List items" }, { "markdown": "1. indented code\n\n paragraph\n\n more code\n", @@ -2206,8 +2203,7 @@ "example": 274, "start_line": 4515, "end_line": 4531, - "section": "List items", - "shouldFail": true + "section": "List items" }, { "markdown": " foo\n\nbar\n", @@ -2239,8 +2235,7 @@ "example": 278, "start_line": 4596, "end_line": 4617, - "section": "List items", - "shouldFail": true + "section": "List items" }, { "markdown": "- \n foo\n", @@ -2256,8 +2251,7 @@ "example": 280, "start_line": 4636, "end_line": 4645, - "section": "List items", - "shouldFail": true + "section": "List items" }, { "markdown": "- foo\n-\n- bar\n", @@ -2289,8 +2283,7 @@ "example": 284, "start_line": 4695, "end_line": 4701, - "section": "List items", - "shouldFail": true + "section": "List items" }, { "markdown": "foo\n*\n\nfoo\n1.\n", @@ -2466,8 +2459,7 @@ "example": 306, "start_line": 5388, "end_line": 5407, - "section": "Lists", - "shouldFail": true + "section": "Lists" }, { "markdown": "- foo\n - bar\n - baz\n\n\n bim\n", @@ -2475,8 +2467,7 @@ "example": 307, "start_line": 5409, "end_line": 5431, - "section": "Lists", - "shouldFail": true + "section": "Lists" }, { "markdown": "- foo\n- bar\n\n\n\n- baz\n- bim\n", @@ -2556,8 +2547,7 @@ "example": 317, "start_line": 5645, "end_line": 5663, - "section": "Lists", - "shouldFail": true + "section": "Lists" }, { "markdown": "- a\n- ```\n b\n\n\n ```\n- c\n", @@ -2565,8 +2555,7 @@ "example": 318, "start_line": 5668, "end_line": 5687, - "section": "Lists", - "shouldFail": true + "section": "Lists" }, { "markdown": "- a\n - b\n\n c\n- d\n", @@ -2574,8 +2563,7 @@ "example": 319, "start_line": 5694, "end_line": 5712, - "section": "Lists", - "shouldFail": true + "section": "Lists" }, { "markdown": "* a\n > b\n >\n* c\n", diff --git a/test/specs/new/def_blocks.html b/test/specs/new/def_blocks.html index 5d8de49c04..23207da91e 100644 --- a/test/specs/new/def_blocks.html +++ b/test/specs/new/def_blocks.html @@ -12,13 +12,14 @@ - - - diff --git a/test/specs/new/list_paren_delimiter.html b/test/specs/new/list_paren_delimiter.html index 4999ee3efa..ba8ecf36dd 100644 --- a/test/specs/new/list_paren_delimiter.html +++ b/test/specs/new/list_paren_delimiter.html @@ -1,11 +1,13 @@
    -
  1. one
  2. -
  3. two
  4. -
  5. three
  6. +
  7. one
  8. +
  9. two
  10. +
  11. three
- + +
+
    -
  1. two
  2. -
  3. three
  4. -
  5. four
  6. -
\ No newline at end of file +
  • two
  • +
  • three
  • +
  • four
  • + diff --git a/test/specs/new/list_paren_delimiter.md b/test/specs/new/list_paren_delimiter.md index d7ff7f3aeb..2443a64a85 100644 --- a/test/specs/new/list_paren_delimiter.md +++ b/test/specs/new/list_paren_delimiter.md @@ -2,6 +2,7 @@ 2) two 3) three +*** 2) two 3) three diff --git a/test/specs/new/main.html b/test/specs/new/main.html deleted file mode 100644 index 8f78c4d30e..0000000000 --- a/test/specs/new/main.html +++ /dev/null @@ -1,4 +0,0 @@ -

    A heading

    Just a note, I've found that I can't test my markdown parser vs others. For example, both markdown.js and showdown code blocks in lists wrong. They're also completely inconsistent with regards to paragraphs in list items.

    A link. Not anymore.

    Paragraph.


    Another blockquote! I really need to get more creative with mockup text.. markdown.js breaks here again

    Another Heading

    Hello world. Here is a link. And an image alt.

    Code goes here.
    -Lots of it...
    diff --git a/test/specs/new/main.md b/test/specs/new/main.md deleted file mode 100644 index 58e17a6a76..0000000000 --- a/test/specs/new/main.md +++ /dev/null @@ -1,55 +0,0 @@ -[test]: http://google.com/ "Google" - -# A heading - -Just a note, I've found that I can't test my markdown parser vs others. -For example, both markdown.js and showdown code blocks in lists wrong. They're -also completely [inconsistent][test] with regards to paragraphs in list items. - -A link. Not anymore. - - - -* List Item 1 - -* List Item 2 - * New List Item 1 - Hi, this is a list item. - * New List Item 2 - Another item - Code goes here. - Lots of it... - * New List Item 3 - The last item - -* List Item 3 -The final item. - -* List Item 4 -The real final item. - -Paragraph. - -> * bq Item 1 -> * bq Item 2 -> * New bq Item 1 -> * New bq Item 2 -> Text here - -* * * - -> Another blockquote! -> I really need to get -> more creative with -> mockup text.. -> markdown.js breaks here again - -Another Heading -------------- - -Hello *world*. Here is a [link](//hello). -And an image ![alt](src). - - Code goes here. - Lots of it... diff --git a/test/unit/Lexer-spec.js b/test/unit/Lexer-spec.js index f94cd6e70d..af3f5781b1 100644 --- a/test/unit/Lexer-spec.js +++ b/test/unit/Lexer-spec.js @@ -295,7 +295,7 @@ a | b tokens: [ { type: 'list', - raw: '- item 1\n- item 2\n', + raw: '- item 1\n- item 2', ordered: false, start: '', loose: false, @@ -316,7 +316,7 @@ a | b }, { type: 'list_item', - raw: '- item 2\n', + raw: '- item 2', task: false, checked: undefined, loose: false, @@ -343,7 +343,7 @@ a | b tokens: jasmine.arrayContaining([ jasmine.objectContaining({ type: 'list', - raw: '1. item 1\n2. item 2\n', + raw: '1. item 1\n2. item 2', ordered: true, start: 1, items: [ @@ -351,7 +351,7 @@ a | b raw: '1. item 1\n' }), jasmine.objectContaining({ - raw: '2. item 2\n' + raw: '2. item 2' }) ] }) @@ -368,7 +368,7 @@ a | b tokens: jasmine.arrayContaining([ jasmine.objectContaining({ type: 'list', - raw: '1) item 1\n2) item 2\n', + raw: '1) item 1\n2) item 2', ordered: true, start: 1, items: [ @@ -376,7 +376,7 @@ a | b raw: '1) item 1\n' }), jasmine.objectContaining({ - raw: '2) item 2\n' + raw: '2) item 2' }) ] }) @@ -395,7 +395,7 @@ paragraph tokens: [ { type: 'list', - raw: '- item 1\n- item 2\n\n', + raw: '- item 1\n- item 2', ordered: false, start: '', loose: false, @@ -416,7 +416,7 @@ paragraph }, { type: 'list_item', - raw: '- item 2\n\n', + raw: '- item 2', task: false, checked: undefined, loose: false, @@ -430,6 +430,7 @@ paragraph } ] }, + { type: 'space', raw: '\n\n' }, { type: 'paragraph', raw: 'paragraph', @@ -453,7 +454,7 @@ paragraph tokens: jasmine.arrayContaining([ jasmine.objectContaining({ type: 'list', - raw: '2. item 1\n3. item 2\n', + raw: '2. item 1\n3. item 2', ordered: true, start: 2, items: [ @@ -461,7 +462,7 @@ paragraph raw: '2. item 1\n' }), jasmine.objectContaining({ - raw: '3. item 2\n' + raw: '3. item 2' }) ] }) @@ -479,14 +480,14 @@ paragraph tokens: jasmine.arrayContaining([ jasmine.objectContaining({ type: 'list', - raw: '- item 1\n\n- item 2\n', + raw: '- item 1\n\n- item 2', loose: true, items: [ jasmine.objectContaining({ raw: '- item 1\n\n' }), jasmine.objectContaining({ - raw: '- item 2\n' + raw: '- item 2' }) ] }) @@ -503,7 +504,7 @@ paragraph tokens: jasmine.arrayContaining([ jasmine.objectContaining({ type: 'list', - raw: '- [ ] item 1\n- [x] item 2\n', + raw: '- [ ] item 1\n- [x] item 2', items: [ jasmine.objectContaining({ raw: '- [ ] item 1\n', @@ -511,7 +512,7 @@ paragraph checked: false }), jasmine.objectContaining({ - raw: '- [x] item 2\n', + raw: '- [x] item 2', task: true, checked: true }) diff --git a/test/unit/marked-spec.js b/test/unit/marked-spec.js index 6b67a8fe61..79fa151520 100644 --- a/test/unit/marked-spec.js +++ b/test/unit/marked-spec.js @@ -1001,6 +1001,7 @@ br ['heading', '# heading'], ['text', 'heading'], ['code', '```code```'], + ['space', ''], ['table', '| a | b ||---|---|| 1 | 2 || 3 | 4 |'], ['text', 'a'], ['text', 'b'], @@ -1015,6 +1016,7 @@ br ['list_item', '- list'], ['text', 'list'], ['text', 'list'], + ['space', ''], ['html', '
    html
    '], ['paragraph', '[link](https://example.com)'], ['link', '[link](https://example.com)'],