Browse Source

Blockquote parsing improved

pull/14/head
Alex Kocharin 10 years ago
parent
commit
e3a7ddf4a8
  1. 3
      lib/helpers.js
  2. 3
      lib/lexer_block.js
  3. 116
      lib/lexer_block/blockquote.js
  4. 1
      lib/lexer_block/code.js
  5. 1
      lib/lexer_block/fences.js
  6. 1
      lib/lexer_block/lheading.js
  7. 4
      lib/lexer_block/list.js
  8. 7
      lib/state.js

3
lib/helpers.js

@ -70,7 +70,8 @@ function getLines(state, begin, end, indent, keepLastLF) {
first = state.bMarks[line] + Math.min(state.tShift[line], indent); first = state.bMarks[line] + Math.min(state.tShift[line], indent);
if (line + 1 < end || keepLastLF) { if (line + 1 < end || keepLastLF) {
last = state.bMarks[line + 1]; // TODO: boundary check?
last = state.eMarks[line] + 1;
} else { } else {
last = state.eMarks[line]; last = state.eMarks[line];
} }

3
lib/lexer_block.js

@ -124,6 +124,7 @@ LexerBlock.prototype.tokenize = function (state, startLine, endLine, stopOnTwoNe
if (line >= endLine) { break; } if (line >= endLine) { break; }
if (state.tShift[line] < state.blkIndent) { break; } if (state.tShift[line] < state.blkIndent) { break; }
if (state.bqMarks[line] < state.bqLevel) { break; }
state.tight = !hasEmptyLines; state.tight = !hasEmptyLines;
@ -149,7 +150,7 @@ LexerBlock.prototype.tokenize = function (state, startLine, endLine, stopOnTwoNe
if (line < endLine && isEmpty(state, line)) { if (line < endLine && isEmpty(state, line)) {
hasEmptyLines = true; hasEmptyLines = true;
line++; state.line = line = line + 1;
// two empty lines should stop the parser // two empty lines should stop the parser
if (line < endLine && stopOnTwoNewlines && isEmpty(state, line)) { break; } if (line < endLine && stopOnTwoNewlines && isEmpty(state, line)) { break; }

116
lib/lexer_block/blockquote.js

@ -3,12 +3,12 @@
'use strict'; 'use strict';
var getLines = require('../helpers').getLines; var skipSpaces = require('../helpers').skipSpaces;
var isEmpty = require('../helpers').isEmpty;
module.exports = function blockquote(state, startLine, endLine, silent) { module.exports = function blockquote(state, startLine, endLine, silent) {
var nextLine, subState, insideLines, lineMax, var nextLine, lastLineEmpty, oldTShift, oldBMarks, i,
rules_named = state.lexerBlock.rules_named,
pos = state.bMarks[startLine] + state.tShift[startLine], pos = state.bMarks[startLine] + state.tShift[startLine],
max = state.eMarks[startLine]; max = state.eMarks[startLine];
@ -21,37 +21,97 @@ module.exports = function blockquote(state, startLine, endLine, silent) {
// so no point trying to find the end of it in silent mode // so no point trying to find the end of it in silent mode
if (silent) { return true; } if (silent) { return true; }
lineMax = state.lineMax; // skip one optional space after '>'
insideLines = 1; if (state.src.charCodeAt(pos) === 0x20) { pos++; }
state.tokens.push({ type: 'blockquote_open' });
nextLine = startLine + 1; state.bqMarks[startLine]++;
for (;;) { state.bqLevel++;
if (nextLine < lineMax) {
pos = state.bMarks[nextLine] + state.tShift[nextLine]; oldBMarks = [ state.bMarks[startLine] ];
max = state.eMarks[nextLine]; state.bMarks[startLine] = pos;
if (pos < max && state.src.charCodeAt(pos++) === 0x3E/* > */) { // check if we have an empty blockquote
if (nextLine >= endLine) { break; } pos = pos < max ? skipSpaces(state, pos) : pos;
nextLine++; lastLineEmpty = pos >= max;
insideLines++;
continue; oldTShift = [ state.tShift[startLine] ];
} state.tShift[startLine] = pos - state.bMarks[startLine];
// Search the end of the block
//
// Block ends with either:
// 1. an empty line outside:
// ```
// > test
//
// ```
// 2. an empty line inside:
// ```
// >
// test
// ```
// 3. another tag
// ```
// > test
// - - -
// ```
for (nextLine = startLine + 1; nextLine < endLine; nextLine++) {
pos = state.bMarks[nextLine] + state.tShift[nextLine];
max = state.eMarks[nextLine];
if (pos >= max) {
// Case 1: line is not inside the blockquote, and this line is empty.
break;
} }
if (insideLines === 0) { break; } if (state.src.charCodeAt(pos++) === 0x3E/* > */) {
state.bqMarks[nextLine]++;
// This line is inside the blockquote.
// skip one optional space after '>'
if (state.src.charCodeAt(pos) === 0x20) { pos++; }
oldBMarks.push(state.bMarks[nextLine]);
state.bMarks[nextLine] = pos;
while (nextLine < lineMax) { pos = pos < max ? skipSpaces(state, pos) : pos;
if (isEmpty(state, nextLine)) { break; } lastLineEmpty = pos >= max;
nextLine++;
oldTShift.push(state.tShift[nextLine]);
state.tShift[nextLine] = pos - state.bMarks[nextLine];
continue;
} }
subState = state.clone(getLines(state, startLine, nextLine, 0, true)
.replace(/^ {0,3}> ?/mg, '')); // Case 2: line is not inside the blockquote, and the last line was empty.
state.lexerBlock.tokenize(subState, 0, insideLines); if (lastLineEmpty) { break; }
nextLine = startLine = subState.line + startLine;
insideLines = 0; // Case 3: another tag found.
if (rules_named.fences(state, nextLine, endLine, true)) { break; }
if (rules_named.hr(state, nextLine, endLine, true)) { break; }
if (rules_named.list(state, nextLine, endLine, true)) { break; }
if (rules_named.heading(state, nextLine, endLine, true)) { break; }
// setex header can't interrupt paragraph
// if (rules_named.lheading(state, nextLine, endLine, true)) { break; }
if (rules_named.blockquote(state, nextLine, endLine, true)) { break; }
if (rules_named.table(state, nextLine, endLine, true)) { break; }
//if (rules_named.tag(state, nextLine, endLine, true)) { break; }
//if (rules_named.def(state, nextLine, endLine, true)) { break; }
oldBMarks.push(state.bMarks[nextLine]);
oldTShift.push(state.tShift[nextLine]);
} }
state.tokens.push({ type: 'blockquote_open' });
state.lexerBlock.tokenize(state, startLine, nextLine);
state.tokens.push({ type: 'blockquote_close' }); state.tokens.push({ type: 'blockquote_close' });
state.line = nextLine; // Restore original tShift; this might not be necessary since the parser
// has already been here, but just to make sure we can do that.
for (i = 0; i < oldTShift.length; i++) {
state.bMarks[i + startLine] = oldBMarks[i];
state.tShift[i + startLine] = oldTShift[i];
}
state.bqLevel--;
return true; return true;
}; };

1
lib/lexer_block/code.js

@ -15,6 +15,7 @@ module.exports = function code(state, startLine, endLine, silent) {
last = nextLine = startLine + 1; last = nextLine = startLine + 1;
while (nextLine < endLine) { while (nextLine < endLine) {
if (state.bqMarks[nextLine] < state.bqLevel) { break; }
if (isEmpty(state, nextLine)) { if (isEmpty(state, nextLine)) {
nextLine++; nextLine++;
if (state.options.pedantic) { if (state.options.pedantic) {

1
lib/lexer_block/fences.js

@ -61,6 +61,7 @@ module.exports = function fences(state, startLine, endLine, silent) {
// test // test
break; break;
} }
if (pos < max && state.bqMarks[nextLine] < state.bqLevel) { break; }
if (state.src.charCodeAt(pos) !== marker) { continue; } if (state.src.charCodeAt(pos) !== marker) { continue; }

1
lib/lexer_block/lheading.js

@ -14,6 +14,7 @@ module.exports = function lheading(state, startLine, endLine, silent) {
if (next >= endLine) { return false; } if (next >= endLine) { return false; }
if (state.tShift[next] < state.blkIndent) { return false; } if (state.tShift[next] < state.blkIndent) { return false; }
if (state.bqMarks[next] < state.bqLevel) { return false; }
// Scan next line // Scan next line
if (state.tShift[next] - state.blkIndent > 3) { return false; } if (state.tShift[next] - state.blkIndent > 3) { return false; }

4
lib/lexer_block/list.js

@ -133,6 +133,8 @@ module.exports = function list(state, startLine, endLine, silent) {
prevEmptyEnd = false; prevEmptyEnd = false;
while (nextLine < endLine) { while (nextLine < endLine) {
if (state.bqMarks[nextLine] < state.bqLevel) { break; }
contentStart = skipSpaces(state, posAfterMarker); contentStart = skipSpaces(state, posAfterMarker);
max = state.eMarks[nextLine]; max = state.eMarks[nextLine];
@ -215,8 +217,6 @@ module.exports = function list(state, startLine, endLine, silent) {
} }
if (markerCharCode !== state.src.charCodeAt(posAfterMarker - 1)) { break; } if (markerCharCode !== state.src.charCodeAt(posAfterMarker - 1)) { break; }
} }
// Finilize list // Finilize list

7
lib/state.js

@ -86,6 +86,13 @@ function State(src, lexerBlock, lexerInline, renderer, tokens, options) {
this.lineMax = this.bMarks.length - 1; // don't count last fake line this.lineMax = this.bMarks.length - 1; // don't count last fake line
this.tight = false; // loose/tight mode for lists this.tight = false; // loose/tight mode for lists
// Stuff for blockquotes
this.bqLevel = 0;
this.bqMarks = [];
for (start = 0; start < this.bMarks.length; start++) {
this.bqMarks.push(0);
}
// renderer // renderer
this.result = ''; this.result = '';
} }

Loading…
Cancel
Save