97 lines
2.7 KiB
JavaScript
97 lines
2.7 KiB
JavaScript
|
/**
|
|||
|
* @typedef {import('../types.js').Handle} Handle
|
|||
|
* @typedef {import('../types.js').Info} Info
|
|||
|
* @typedef {import('../types.js').PhrasingParents} PhrasingParents
|
|||
|
* @typedef {import('../types.js').State} State
|
|||
|
*/
|
|||
|
|
|||
|
/**
|
|||
|
* Serialize the children of a parent that contains phrasing children.
|
|||
|
*
|
|||
|
* These children will be joined flush together.
|
|||
|
*
|
|||
|
* @param {PhrasingParents} parent
|
|||
|
* Parent of flow nodes.
|
|||
|
* @param {State} state
|
|||
|
* Info passed around about the current state.
|
|||
|
* @param {Info} info
|
|||
|
* Info on where we are in the document we are generating.
|
|||
|
* @returns {string}
|
|||
|
* Serialized children, joined together.
|
|||
|
*/
|
|||
|
export function containerPhrasing(parent, state, info) {
|
|||
|
const indexStack = state.indexStack
|
|||
|
const children = parent.children || []
|
|||
|
/** @type {Array<string>} */
|
|||
|
const results = []
|
|||
|
let index = -1
|
|||
|
let before = info.before
|
|||
|
|
|||
|
indexStack.push(-1)
|
|||
|
let tracker = state.createTracker(info)
|
|||
|
|
|||
|
while (++index < children.length) {
|
|||
|
const child = children[index]
|
|||
|
/** @type {string} */
|
|||
|
let after
|
|||
|
|
|||
|
indexStack[indexStack.length - 1] = index
|
|||
|
|
|||
|
if (index + 1 < children.length) {
|
|||
|
/** @type {Handle} */
|
|||
|
// @ts-expect-error: hush, it’s actually a `zwitch`.
|
|||
|
let handle = state.handle.handlers[children[index + 1].type]
|
|||
|
/** @type {Handle} */
|
|||
|
// @ts-expect-error: hush, it’s actually a `zwitch`.
|
|||
|
if (handle && handle.peek) handle = handle.peek
|
|||
|
after = handle
|
|||
|
? handle(children[index + 1], parent, state, {
|
|||
|
before: '',
|
|||
|
after: '',
|
|||
|
...tracker.current()
|
|||
|
}).charAt(0)
|
|||
|
: ''
|
|||
|
} else {
|
|||
|
after = info.after
|
|||
|
}
|
|||
|
|
|||
|
// In some cases, html (text) can be found in phrasing right after an eol.
|
|||
|
// When we’d serialize that, in most cases that would be seen as html
|
|||
|
// (flow).
|
|||
|
// As we can’t escape or so to prevent it from happening, we take a somewhat
|
|||
|
// reasonable approach: replace that eol with a space.
|
|||
|
// See: <https://github.com/syntax-tree/mdast-util-to-markdown/issues/15>
|
|||
|
if (
|
|||
|
results.length > 0 &&
|
|||
|
(before === '\r' || before === '\n') &&
|
|||
|
child.type === 'html'
|
|||
|
) {
|
|||
|
results[results.length - 1] = results[results.length - 1].replace(
|
|||
|
/(\r?\n|\r)$/,
|
|||
|
' '
|
|||
|
)
|
|||
|
before = ' '
|
|||
|
|
|||
|
// To do: does this work to reset tracker?
|
|||
|
tracker = state.createTracker(info)
|
|||
|
tracker.move(results.join(''))
|
|||
|
}
|
|||
|
|
|||
|
results.push(
|
|||
|
tracker.move(
|
|||
|
state.handle(child, parent, state, {
|
|||
|
...tracker.current(),
|
|||
|
before,
|
|||
|
after
|
|||
|
})
|
|||
|
)
|
|||
|
)
|
|||
|
|
|||
|
before = results[results.length - 1].slice(-1)
|
|||
|
}
|
|||
|
|
|||
|
indexStack.pop()
|
|||
|
|
|||
|
return results.join('')
|
|||
|
}
|