micromark
Version:
small commonmark compliant markdown parser with positional info and concrete tokens
175 lines (147 loc) • 4.14 kB
JavaScript
import markdownLineEnding from '../character/markdown-line-ending.mjs'
import spaceFactory from './factory-space.mjs'
import prefixSize from '../util/prefix-size.mjs'
import markdownLineEndingOrSpace from '../character/markdown-line-ending-or-space.mjs'
var codeFenced = {
name: 'codeFenced',
tokenize: tokenizeCodeFenced,
concrete: true
}
function tokenizeCodeFenced(effects, ok, nok) {
var self = this
var closingFenceConstruct = {
tokenize: tokenizeClosingFence,
partial: true
}
var initialPrefix = prefixSize(this.events, 'linePrefix')
var sizeOpen = 0
var marker
return start
function start(code) {
effects.enter('codeFenced')
effects.enter('codeFencedFence')
effects.enter('codeFencedFenceSequence')
marker = code
return sequenceOpen(code)
}
function sequenceOpen(code) {
if (code === marker) {
effects.consume(code)
sizeOpen++
return sequenceOpen
}
effects.exit('codeFencedFenceSequence')
return sizeOpen < 3
? nok(code)
: spaceFactory(effects, infoOpen, 'whitespace')(code)
}
function infoOpen(code) {
if (code === null || markdownLineEnding(code)) {
return openAfter(code)
}
effects.enter('codeFencedFenceInfo')
effects.enter('chunkString', {
contentType: 'string'
})
return info(code)
}
function info(code) {
if (code === null || markdownLineEndingOrSpace(code)) {
effects.exit('chunkString')
effects.exit('codeFencedFenceInfo')
return spaceFactory(effects, infoAfter, 'whitespace')(code)
}
if (code === 96 && code === marker) return nok(code)
effects.consume(code)
return info
}
function infoAfter(code) {
if (code === null || markdownLineEnding(code)) {
return openAfter(code)
}
effects.enter('codeFencedFenceMeta')
effects.enter('chunkString', {
contentType: 'string'
})
return meta(code)
}
function meta(code) {
if (code === null || markdownLineEnding(code)) {
effects.exit('chunkString')
effects.exit('codeFencedFenceMeta')
return openAfter(code)
}
if (code === 96 && code === marker) return nok(code)
effects.consume(code)
return meta
}
function openAfter(code) {
effects.exit('codeFencedFence')
return self.interrupt ? ok(code) : content(code)
}
function content(code) {
if (code === null) {
return after(code)
}
if (markdownLineEnding(code)) {
effects.enter('lineEnding')
effects.consume(code)
effects.exit('lineEnding')
return effects.attempt(
closingFenceConstruct,
after,
initialPrefix
? spaceFactory(effects, content, 'linePrefix', initialPrefix + 1)
: content
)
}
effects.enter('codeFlowValue')
return contentContinue(code)
}
function contentContinue(code) {
if (code === null || markdownLineEnding(code)) {
effects.exit('codeFlowValue')
return content(code)
}
effects.consume(code)
return contentContinue
}
function after(code) {
effects.exit('codeFenced')
return ok(code)
}
function tokenizeClosingFence(effects, ok, nok) {
var size = 0
return spaceFactory(
effects,
closingSequenceStart,
'linePrefix',
this.parser.constructs.disable.null.indexOf('codeIndented') > -1
? undefined
: 4
)
function closingSequenceStart(code) {
effects.enter('codeFencedFence')
effects.enter('codeFencedFenceSequence')
return closingSequence(code)
}
function closingSequence(code) {
if (code === marker) {
effects.consume(code)
size++
return closingSequence
}
if (size < sizeOpen) return nok(code)
effects.exit('codeFencedFenceSequence')
return spaceFactory(effects, closingSequenceEnd, 'whitespace')(code)
}
function closingSequenceEnd(code) {
if (code === null || markdownLineEnding(code)) {
effects.exit('codeFencedFence')
return ok(code)
}
return nok(code)
}
}
}
export default codeFenced