123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158 |
- /**
- * @typedef {import('micromark-util-types').Construct} Construct
- * @typedef {import('micromark-util-types').Exiter} Exiter
- * @typedef {import('micromark-util-types').State} State
- * @typedef {import('micromark-util-types').TokenizeContext} TokenizeContext
- * @typedef {import('micromark-util-types').Tokenizer} Tokenizer
- */
- import {factorySpace} from 'micromark-factory-space'
- import {markdownSpace} from 'micromark-util-character'
- import {codes, constants, types} from 'micromark-util-symbol'
- import {ok as assert} from 'devlop'
- /** @type {Construct} */
- export const blockQuote = {
- name: 'blockQuote',
- tokenize: tokenizeBlockQuoteStart,
- continuation: {tokenize: tokenizeBlockQuoteContinuation},
- exit
- }
- /**
- * @this {TokenizeContext}
- * @type {Tokenizer}
- */
- function tokenizeBlockQuoteStart(effects, ok, nok) {
- const self = this
- return start
- /**
- * Start of block quote.
- *
- * ```markdown
- * > | > a
- * ^
- * ```
- *
- * @type {State}
- */
- function start(code) {
- if (code === codes.greaterThan) {
- const state = self.containerState
- assert(state, 'expected `containerState` to be defined in container')
- if (!state.open) {
- effects.enter(types.blockQuote, {_container: true})
- state.open = true
- }
- effects.enter(types.blockQuotePrefix)
- effects.enter(types.blockQuoteMarker)
- effects.consume(code)
- effects.exit(types.blockQuoteMarker)
- return after
- }
- return nok(code)
- }
- /**
- * After `>`, before optional whitespace.
- *
- * ```markdown
- * > | > a
- * ^
- * ```
- *
- * @type {State}
- */
- function after(code) {
- if (markdownSpace(code)) {
- effects.enter(types.blockQuotePrefixWhitespace)
- effects.consume(code)
- effects.exit(types.blockQuotePrefixWhitespace)
- effects.exit(types.blockQuotePrefix)
- return ok
- }
- effects.exit(types.blockQuotePrefix)
- return ok(code)
- }
- }
- /**
- * Start of block quote continuation.
- *
- * ```markdown
- * | > a
- * > | > b
- * ^
- * ```
- *
- * @this {TokenizeContext}
- * @type {Tokenizer}
- */
- function tokenizeBlockQuoteContinuation(effects, ok, nok) {
- const self = this
- return contStart
- /**
- * Start of block quote continuation.
- *
- * Also used to parse the first block quote opening.
- *
- * ```markdown
- * | > a
- * > | > b
- * ^
- * ```
- *
- * @type {State}
- */
- function contStart(code) {
- if (markdownSpace(code)) {
- // Always populated by defaults.
- assert(
- self.parser.constructs.disable.null,
- 'expected `disable.null` to be populated'
- )
- return factorySpace(
- effects,
- contBefore,
- types.linePrefix,
- self.parser.constructs.disable.null.includes('codeIndented')
- ? undefined
- : constants.tabSize
- )(code)
- }
- return contBefore(code)
- }
- /**
- * At `>`, after optional whitespace.
- *
- * Also used to parse the first block quote opening.
- *
- * ```markdown
- * | > a
- * > | > b
- * ^
- * ```
- *
- * @type {State}
- */
- function contBefore(code) {
- return effects.attempt(blockQuote, ok, nok)(code)
- }
- }
- /** @type {Exiter} */
- function exit(effects) {
- effects.exit(types.blockQuote)
- }
|