This commit is contained in:
2024-03-22 03:47:51 +05:30
parent 8bcf3d211e
commit 89819f6fe2
28440 changed files with 3211033 additions and 2 deletions

View File

@@ -0,0 +1,294 @@
/**
* @typedef {import('micromark-util-types').Construct} Construct
* @typedef {import('micromark-util-types').State} State
* @typedef {import('micromark-util-types').Token} Token
* @typedef {import('micromark-util-types').TokenizeContext} TokenizeContext
* @typedef {import('micromark-util-types').Tokenizer} Tokenizer
*/
import {factorySpace} from 'micromark-factory-space'
import {markdownLineEnding} from 'micromark-util-character'
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 === 'linePrefix'
? tail[2].sliceSerialize(tail[1], true).length
: 0
let sizeOpen = 0
/** @type {Token} */
let previous
return start
/** @type {State} */
function start(code) {
effects.enter('directiveContainer')
effects.enter('directiveContainerFence')
effects.enter('directiveContainerSequence')
return sequenceOpen(code)
}
/** @type {State} */
function sequenceOpen(code) {
if (code === 58) {
effects.consume(code)
sizeOpen++
return sequenceOpen
}
if (sizeOpen < 3) {
return nok(code)
}
effects.exit('directiveContainerSequence')
return factoryName.call(
self,
effects,
afterName,
nok,
'directiveContainerName'
)(code)
}
/** @type {State} */
function afterName(code) {
return code === 91
? effects.attempt(label, afterLabel, afterLabel)(code)
: afterLabel(code)
}
/** @type {State} */
function afterLabel(code) {
return code === 123
? effects.attempt(attributes, afterAttributes, afterAttributes)(code)
: afterAttributes(code)
}
/** @type {State} */
function afterAttributes(code) {
return factorySpace(effects, openAfter, 'whitespace')(code)
}
/** @type {State} */
function openAfter(code) {
effects.exit('directiveContainerFence')
if (code === null) {
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 === null) {
effects.exit('directiveContainer')
return ok(code)
}
effects.enter('directiveContainerContent')
return lineStart(code)
}
/** @type {State} */
function lineStart(code) {
if (code === null) {
return after(code)
}
return effects.attempt(
{
tokenize: tokenizeClosingFence,
partial: true
},
after,
initialSize
? factorySpace(effects, chunkStart, 'linePrefix', initialSize + 1)
: chunkStart
)(code)
}
/** @type {State} */
function chunkStart(code) {
if (code === null) {
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 === null) {
const t = effects.exit('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('chunkDocument')
self.parser.lazy[t.start.line] = false
return lineStart
}
/** @type {State} */
function lineAfter(code) {
const t = effects.exit('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, 'linePrefix', 4)
/** @type {State} */
function closingPrefixAfter(code) {
effects.enter('directiveContainerFence')
effects.enter('directiveContainerSequence')
return closingSequence(code)
}
/** @type {State} */
function closingSequence(code) {
if (code === 58) {
effects.consume(code)
size++
return closingSequence
}
if (size < sizeOpen) return nok(code)
effects.exit('directiveContainerSequence')
return factorySpace(effects, closingSequenceEnd, 'whitespace')(code)
}
/** @type {State} */
function closingSequenceEnd(code) {
if (code === null || 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) {
effects.enter('lineEnding')
effects.consume(code)
effects.exit('lineEnding')
return lineStart
}
/** @type {State} */
function lineStart(code) {
return self.parser.lazy[self.now().line] ? nok(code) : ok(code)
}
}