188 lines
4.9 KiB
JavaScript
188 lines
4.9 KiB
JavaScript
/**
|
||
* @typedef {import('./types.js').Enter} Enter
|
||
* @typedef {import('./types.js').Info} Info
|
||
* @typedef {import('./types.js').Join} Join
|
||
* @typedef {import('./types.js').FlowContent} FlowContent
|
||
* @typedef {import('./types.js').Node} Node
|
||
* @typedef {import('./types.js').Options} Options
|
||
* @typedef {import('./types.js').Parent} Parent
|
||
* @typedef {import('./types.js').PhrasingContent} PhrasingContent
|
||
* @typedef {import('./types.js').SafeConfig} SafeConfig
|
||
* @typedef {import('./types.js').State} State
|
||
* @typedef {import('./types.js').TrackFields} TrackFields
|
||
*/
|
||
|
||
import {zwitch} from 'zwitch'
|
||
import {configure} from './configure.js'
|
||
import {handle as handlers} from './handle/index.js'
|
||
import {join} from './join.js'
|
||
import {unsafe} from './unsafe.js'
|
||
import {association} from './util/association.js'
|
||
import {containerPhrasing} from './util/container-phrasing.js'
|
||
import {containerFlow} from './util/container-flow.js'
|
||
import {indentLines} from './util/indent-lines.js'
|
||
import {safe} from './util/safe.js'
|
||
import {track} from './util/track.js'
|
||
|
||
/**
|
||
* Turn an mdast syntax tree into markdown.
|
||
*
|
||
* @param {Node} tree
|
||
* Tree to serialize.
|
||
* @param {Options} [options]
|
||
* Configuration (optional).
|
||
* @returns {string}
|
||
* Serialized markdown representing `tree`.
|
||
*/
|
||
export function toMarkdown(tree, options = {}) {
|
||
/** @type {State} */
|
||
const state = {
|
||
enter,
|
||
indentLines,
|
||
associationId: association,
|
||
containerPhrasing: containerPhrasingBound,
|
||
containerFlow: containerFlowBound,
|
||
createTracker: track,
|
||
safe: safeBound,
|
||
stack: [],
|
||
unsafe: [],
|
||
join: [],
|
||
// @ts-expect-error: we’ll fill it next.
|
||
handlers: {},
|
||
options: {},
|
||
indexStack: [],
|
||
// @ts-expect-error: we’ll add `handle` later.
|
||
handle: undefined
|
||
}
|
||
|
||
configure(state, {unsafe, join, handlers})
|
||
configure(state, options)
|
||
|
||
if (state.options.tightDefinitions) {
|
||
configure(state, {join: [joinDefinition]})
|
||
}
|
||
|
||
state.handle = zwitch('type', {
|
||
invalid,
|
||
unknown,
|
||
handlers: state.handlers
|
||
})
|
||
|
||
let result = state.handle(tree, undefined, state, {
|
||
before: '\n',
|
||
after: '\n',
|
||
now: {line: 1, column: 1},
|
||
lineShift: 0
|
||
})
|
||
|
||
if (
|
||
result &&
|
||
result.charCodeAt(result.length - 1) !== 10 &&
|
||
result.charCodeAt(result.length - 1) !== 13
|
||
) {
|
||
result += '\n'
|
||
}
|
||
|
||
return result
|
||
|
||
/** @type {Enter} */
|
||
function enter(name) {
|
||
state.stack.push(name)
|
||
return exit
|
||
|
||
function exit() {
|
||
state.stack.pop()
|
||
}
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @param {unknown} value
|
||
* @returns {never}
|
||
*/
|
||
function invalid(value) {
|
||
throw new Error('Cannot handle value `' + value + '`, expected node')
|
||
}
|
||
|
||
/**
|
||
* @param {unknown} node
|
||
* @returns {never}
|
||
*/
|
||
function unknown(node) {
|
||
// @ts-expect-error: fine.
|
||
throw new Error('Cannot handle unknown node `' + node.type + '`')
|
||
}
|
||
|
||
/** @type {Join} */
|
||
function joinDefinition(left, right) {
|
||
// No blank line between adjacent definitions.
|
||
if (left.type === 'definition' && left.type === right.type) {
|
||
return 0
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Serialize the children of a parent that contains phrasing children.
|
||
*
|
||
* These children will be joined flush together.
|
||
*
|
||
* @this {State}
|
||
* Info passed around about the current state.
|
||
* @param {Parent & {children: Array<PhrasingContent>}} parent
|
||
* Parent of flow nodes.
|
||
* @param {Info} info
|
||
* Info on where we are in the document we are generating.
|
||
* @returns {string}
|
||
* Serialized children, joined together.
|
||
*/
|
||
function containerPhrasingBound(parent, info) {
|
||
return containerPhrasing(parent, this, info)
|
||
}
|
||
|
||
/**
|
||
* Serialize the children of a parent that contains flow children.
|
||
*
|
||
* These children will typically be joined by blank lines.
|
||
* What they are joined by exactly is defined by `Join` functions.
|
||
*
|
||
* @this {State}
|
||
* Info passed around about the current state.
|
||
* @param {Parent & {children: Array<FlowContent>}} parent
|
||
* Parent of flow nodes.
|
||
* @param {TrackFields} info
|
||
* Info on where we are in the document we are generating.
|
||
* @returns {string}
|
||
* Serialized children, joined by (blank) lines.
|
||
*/
|
||
function containerFlowBound(parent, info) {
|
||
return containerFlow(parent, this, info)
|
||
}
|
||
|
||
/**
|
||
* Make a string safe for embedding in markdown constructs.
|
||
*
|
||
* In markdown, almost all punctuation characters can, in certain cases,
|
||
* result in something.
|
||
* Whether they do is highly subjective to where they happen and in what
|
||
* they happen.
|
||
*
|
||
* To solve this, `mdast-util-to-markdown` tracks:
|
||
*
|
||
* * Characters before and after something;
|
||
* * What “constructs” we are in.
|
||
*
|
||
* This information is then used by this function to escape or encode
|
||
* special characters.
|
||
*
|
||
* @this {State}
|
||
* Info passed around about the current state.
|
||
* @param {string | null | undefined} value
|
||
* Raw value to make safe.
|
||
* @param {SafeConfig} config
|
||
* Configuration.
|
||
* @returns {string}
|
||
* Serialized markdown safe for embedding.
|
||
*/
|
||
function safeBound(value, config) {
|
||
return safe(this, value, config)
|
||
}
|