micromark-extension-directive/dev/lib/directive-container.js
2024-07-05 17:29:27 +02:00

304 lines
7 KiB
JavaScript

/**
* @import {Construct, State, Token, TokenizeContext, Tokenizer} from 'micromark-util-types'
*/
import {ok as assert} from 'devlop'
import {factorySpace} from 'micromark-factory-space'
import {markdownLineEnding} from 'micromark-util-character'
import {codes, constants, types} from 'micromark-util-symbol'
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}
const nonLazyLine = {tokenize: tokenizeNonLazyLine, partial: true}
/**
* @this {TokenizeContext}
* @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) {
return afterOpening(code)
}
if (markdownLineEnding(code)) {
if (self.interrupt) {
return ok(code)
}
return effects.attempt(nonLazyLine, contentStart, afterOpening)(code)
}
return nok(code)
}
/** @type {State} */
function afterOpening(code) {
effects.exit('directiveContainer')
return ok(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(types.chunkDocument, {
contentType: constants.contentTypeDocument,
previous
})
if (previous) previous.next = token
previous = token
return contentContinue(code)
}
/** @type {State} */
function contentContinue(code) {
if (code === codes.eof) {
const t = effects.exit(types.chunkDocument)
self.parser.lazy[t.start.line] = false
return after(code)
}
if (markdownLineEnding(code)) {
return effects.check(nonLazyLine, nonLazyLineAfter, lineAfter)(code)
}
effects.consume(code)
return contentContinue
}
/** @type {State} */
function nonLazyLineAfter(code) {
effects.consume(code)
const t = effects.exit(types.chunkDocument)
self.parser.lazy[t.start.line] = false
return lineStart
}
/** @type {State} */
function lineAfter(code) {
const t = effects.exit(types.chunkDocument)
self.parser.lazy[t.start.line] = false
return after(code)
}
/** @type {State} */
function after(code) {
effects.exit('directiveContainerContent')
effects.exit('directiveContainer')
return ok(code)
}
/**
* @this {TokenizeContext}
* @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)
}
}
}
/**
* @this {TokenizeContext}
* @type {Tokenizer}
*/
function tokenizeLabel(effects, ok, nok) {
// Always a `[`
return factoryLabel(
effects,
ok,
nok,
'directiveContainerLabel',
'directiveContainerLabelMarker',
'directiveContainerLabelString',
true
)
}
/**
* @this {TokenizeContext}
* @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
)
}
/**
* @this {TokenizeContext}
* @type {Tokenizer}
*/
function tokenizeNonLazyLine(effects, ok, nok) {
const self = this
return start
/** @type {State} */
function start(code) {
assert(markdownLineEnding(code), 'expected eol')
effects.enter(types.lineEnding)
effects.consume(code)
effects.exit(types.lineEnding)
return lineStart
}
/** @type {State} */
function lineStart(code) {
return self.parser.lazy[self.now().line] ? nok(code) : ok(code)
}
}