|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
import {factorySpace} from 'micromark-factory-space' |
|
import { |
|
markdownLineEnding, |
|
markdownLineEndingOrSpace, |
|
markdownSpace |
|
} from 'micromark-util-character' |
|
import {splice} from 'micromark-util-chunked' |
|
|
|
export const headingAtx = { |
|
name: 'headingAtx', |
|
tokenize: tokenizeHeadingAtx, |
|
resolve: resolveHeadingAtx |
|
} |
|
|
|
|
|
function resolveHeadingAtx(events, context) { |
|
let contentEnd = events.length - 2 |
|
let contentStart = 3 |
|
|
|
let content |
|
|
|
let text |
|
|
|
|
|
if (events[contentStart][1].type === 'whitespace') { |
|
contentStart += 2 |
|
} |
|
|
|
|
|
if ( |
|
contentEnd - 2 > contentStart && |
|
events[contentEnd][1].type === 'whitespace' |
|
) { |
|
contentEnd -= 2 |
|
} |
|
if ( |
|
events[contentEnd][1].type === 'atxHeadingSequence' && |
|
(contentStart === contentEnd - 1 || |
|
(contentEnd - 4 > contentStart && |
|
events[contentEnd - 2][1].type === 'whitespace')) |
|
) { |
|
contentEnd -= contentStart + 1 === contentEnd ? 2 : 4 |
|
} |
|
if (contentEnd > contentStart) { |
|
content = { |
|
type: 'atxHeadingText', |
|
start: events[contentStart][1].start, |
|
end: events[contentEnd][1].end |
|
} |
|
text = { |
|
type: 'chunkText', |
|
start: events[contentStart][1].start, |
|
end: events[contentEnd][1].end, |
|
contentType: 'text' |
|
} |
|
splice(events, contentStart, contentEnd - contentStart + 1, [ |
|
['enter', content, context], |
|
['enter', text, context], |
|
['exit', text, context], |
|
['exit', content, context] |
|
]) |
|
} |
|
return events |
|
} |
|
|
|
|
|
|
|
|
|
|
|
function tokenizeHeadingAtx(effects, ok, nok) { |
|
let size = 0 |
|
return start |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function start(code) { |
|
|
|
effects.enter('atxHeading') |
|
return before(code) |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function before(code) { |
|
effects.enter('atxHeadingSequence') |
|
return sequenceOpen(code) |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function sequenceOpen(code) { |
|
if (code === 35 && size++ < 6) { |
|
effects.consume(code) |
|
return sequenceOpen |
|
} |
|
|
|
|
|
if (code === null || markdownLineEndingOrSpace(code)) { |
|
effects.exit('atxHeadingSequence') |
|
return atBreak(code) |
|
} |
|
return nok(code) |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function atBreak(code) { |
|
if (code === 35) { |
|
effects.enter('atxHeadingSequence') |
|
return sequenceFurther(code) |
|
} |
|
if (code === null || markdownLineEnding(code)) { |
|
effects.exit('atxHeading') |
|
|
|
|
|
|
|
return ok(code) |
|
} |
|
if (markdownSpace(code)) { |
|
return factorySpace(effects, atBreak, 'whitespace')(code) |
|
} |
|
|
|
|
|
|
|
effects.enter('atxHeadingText') |
|
return data(code) |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function sequenceFurther(code) { |
|
if (code === 35) { |
|
effects.consume(code) |
|
return sequenceFurther |
|
} |
|
effects.exit('atxHeadingSequence') |
|
return atBreak(code) |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function data(code) { |
|
if (code === null || code === 35 || markdownLineEndingOrSpace(code)) { |
|
effects.exit('atxHeadingText') |
|
return atBreak(code) |
|
} |
|
effects.consume(code) |
|
return data |
|
} |
|
} |
|
|