252 lines
6.1 KiB
JavaScript
252 lines
6.1 KiB
JavaScript
/**
|
|
* @typedef {import('micromark-util-types').Construct} Construct
|
|
* @typedef {import('micromark-util-types').Tokenizer} Tokenizer
|
|
* @typedef {import('micromark-util-types').State} State
|
|
* @typedef {import('micromark-util-types').Token} Token
|
|
*/
|
|
|
|
import assert from 'assert'
|
|
import {factorySpace} from 'micromark-factory-space'
|
|
import {markdownLineEnding} from 'micromark-util-character'
|
|
import {codes} from 'micromark-util-symbol/codes.js'
|
|
import {constants} from 'micromark-util-symbol/constants.js'
|
|
import {types} from 'micromark-util-symbol/types.js'
|
|
import {factoryAttributes} from './factory-attributes.js'
|
|
import {factoryLabel} from './factory-label.js'
|
|
import {factoryName} from './factory-name.js'
|
|
|
|
/** @type {Construct} */
|
|
export const directiveContainer = {
|
|
tokenize: tokenizeDirectiveContainer,
|
|
concrete: true
|
|
}
|
|
|
|
const label = {tokenize: tokenizeLabel, partial: true}
|
|
const attributes = {tokenize: tokenizeAttributes, partial: true}
|
|
|
|
/** @type {Tokenizer} */
|
|
function tokenizeDirectiveContainer(effects, ok, nok) {
|
|
const self = this
|
|
const tail = self.events[self.events.length - 1]
|
|
const initialSize =
|
|
tail && tail[1].type === types.linePrefix
|
|
? tail[2].sliceSerialize(tail[1], true).length
|
|
: 0
|
|
let sizeOpen = 0
|
|
/** @type {Token} */
|
|
let previous
|
|
|
|
return start
|
|
|
|
/** @type {State} */
|
|
function start(code) {
|
|
assert(code === codes.colon, 'expected `:`')
|
|
effects.enter('directiveContainer')
|
|
effects.enter('directiveContainerFence')
|
|
effects.enter('directiveContainerSequence')
|
|
return sequenceOpen(code)
|
|
}
|
|
|
|
/** @type {State} */
|
|
function sequenceOpen(code) {
|
|
if (code === codes.colon) {
|
|
effects.consume(code)
|
|
sizeOpen++
|
|
return sequenceOpen
|
|
}
|
|
|
|
if (sizeOpen < constants.codeFencedSequenceSizeMin) {
|
|
return nok(code)
|
|
}
|
|
|
|
effects.exit('directiveContainerSequence')
|
|
return factoryName.call(
|
|
self,
|
|
effects,
|
|
afterName,
|
|
nok,
|
|
'directiveContainerName'
|
|
)(code)
|
|
}
|
|
|
|
/** @type {State} */
|
|
function afterName(code) {
|
|
return code === codes.leftSquareBracket
|
|
? effects.attempt(label, afterLabel, afterLabel)(code)
|
|
: afterLabel(code)
|
|
}
|
|
|
|
/** @type {State} */
|
|
function afterLabel(code) {
|
|
return code === codes.leftCurlyBrace
|
|
? effects.attempt(attributes, afterAttributes, afterAttributes)(code)
|
|
: afterAttributes(code)
|
|
}
|
|
|
|
/** @type {State} */
|
|
function afterAttributes(code) {
|
|
return factorySpace(effects, openAfter, types.whitespace)(code)
|
|
}
|
|
|
|
/** @type {State} */
|
|
function openAfter(code) {
|
|
effects.exit('directiveContainerFence')
|
|
|
|
if (code === codes.eof) {
|
|
effects.exit('directiveContainer')
|
|
return ok(code)
|
|
}
|
|
|
|
if (markdownLineEnding(code)) {
|
|
effects.enter(types.lineEnding)
|
|
effects.consume(code)
|
|
effects.exit(types.lineEnding)
|
|
return self.interrupt ? ok : contentStart
|
|
}
|
|
|
|
return nok(code)
|
|
}
|
|
|
|
/** @type {State} */
|
|
function contentStart(code) {
|
|
if (code === codes.eof) {
|
|
effects.exit('directiveContainer')
|
|
return ok(code)
|
|
}
|
|
|
|
effects.enter('directiveContainerContent')
|
|
return lineStart(code)
|
|
}
|
|
|
|
/** @type {State} */
|
|
function lineStart(code) {
|
|
if (code === codes.eof) {
|
|
return after(code)
|
|
}
|
|
|
|
return effects.attempt(
|
|
{tokenize: tokenizeClosingFence, partial: true},
|
|
after,
|
|
initialSize
|
|
? factorySpace(effects, chunkStart, types.linePrefix, initialSize + 1)
|
|
: chunkStart
|
|
)(code)
|
|
}
|
|
|
|
/** @type {State} */
|
|
function chunkStart(code) {
|
|
if (code === codes.eof) {
|
|
return after(code)
|
|
}
|
|
|
|
const token = effects.enter('chunkDocument', {
|
|
contentType: 'document',
|
|
previous
|
|
})
|
|
if (previous) previous.next = token
|
|
previous = token
|
|
return contentContinue(code)
|
|
}
|
|
|
|
/** @type {State} */
|
|
function contentContinue(code) {
|
|
if (code === codes.eof) {
|
|
effects.exit('chunkDocument')
|
|
return after(code)
|
|
}
|
|
|
|
if (markdownLineEnding(code)) {
|
|
effects.consume(code)
|
|
effects.exit('chunkDocument')
|
|
return lineStart
|
|
}
|
|
|
|
effects.consume(code)
|
|
return contentContinue
|
|
}
|
|
|
|
/** @type {State} */
|
|
function after(code) {
|
|
effects.exit('directiveContainerContent')
|
|
effects.exit('directiveContainer')
|
|
return ok(code)
|
|
}
|
|
|
|
/** @type {Tokenizer} */
|
|
function tokenizeClosingFence(effects, ok, nok) {
|
|
let size = 0
|
|
|
|
return factorySpace(
|
|
effects,
|
|
closingPrefixAfter,
|
|
types.linePrefix,
|
|
constants.tabSize
|
|
)
|
|
|
|
/** @type {State} */
|
|
function closingPrefixAfter(code) {
|
|
effects.enter('directiveContainerFence')
|
|
effects.enter('directiveContainerSequence')
|
|
return closingSequence(code)
|
|
}
|
|
|
|
/** @type {State} */
|
|
function closingSequence(code) {
|
|
if (code === codes.colon) {
|
|
effects.consume(code)
|
|
size++
|
|
return closingSequence
|
|
}
|
|
|
|
if (size < sizeOpen) return nok(code)
|
|
effects.exit('directiveContainerSequence')
|
|
return factorySpace(effects, closingSequenceEnd, types.whitespace)(code)
|
|
}
|
|
|
|
/** @type {State} */
|
|
function closingSequenceEnd(code) {
|
|
if (code === codes.eof || markdownLineEnding(code)) {
|
|
effects.exit('directiveContainerFence')
|
|
return ok(code)
|
|
}
|
|
|
|
return nok(code)
|
|
}
|
|
}
|
|
}
|
|
|
|
/** @type {Tokenizer} */
|
|
function tokenizeLabel(effects, ok, nok) {
|
|
// Always a `[`
|
|
return factoryLabel(
|
|
effects,
|
|
ok,
|
|
nok,
|
|
'directiveContainerLabel',
|
|
'directiveContainerLabelMarker',
|
|
'directiveContainerLabelString',
|
|
true
|
|
)
|
|
}
|
|
|
|
/** @type {Tokenizer} */
|
|
function tokenizeAttributes(effects, ok, nok) {
|
|
// Always a `{`
|
|
return factoryAttributes(
|
|
effects,
|
|
ok,
|
|
nok,
|
|
'directiveContainerAttributes',
|
|
'directiveContainerAttributesMarker',
|
|
'directiveContainerAttribute',
|
|
'directiveContainerAttributeId',
|
|
'directiveContainerAttributeClass',
|
|
'directiveContainerAttributeName',
|
|
'directiveContainerAttributeInitializerMarker',
|
|
'directiveContainerAttributeValueLiteral',
|
|
'directiveContainerAttributeValue',
|
|
'directiveContainerAttributeValueMarker',
|
|
'directiveContainerAttributeValueData',
|
|
true
|
|
)
|
|
}
|