kjelsrud.dev/node_modules/estree-util-attach-comments/lib/index.js

192 lines
4.8 KiB
JavaScript
Raw Normal View History

2023-07-19 21:31:30 +02:00
/**
* @typedef {import('estree').BaseNode} EstreeNode
* @typedef {import('estree').Comment} EstreeComment
*
* @typedef State
* Info passed around.
* @property {Array<EstreeComment>} comments
* Comments.
* @property {number} index
* Index of comment.
*
* @typedef Fields
* @property {boolean} leading
* @property {boolean} trailing
*/
const own = {}.hasOwnProperty
/**
* Attach semistandard estree comment nodes to the tree.
*
* This mutates the given `tree`.
* It takes `comments`, walks the tree, and adds comments as close as possible
* to where they originated.
*
* Comment nodes are given two boolean fields: `leading` (`true` for
* `/* a *\/ b`) and `trailing` (`true` for `a /* b *\/`).
* Both fields are `false` for dangling comments: `[/* a *\/]`.
* This is what `recast` uses too, and is somewhat similar to Babel, which is
* not estree but instead uses `leadingComments`, `trailingComments`, and
* `innerComments` arrays on nodes.
*
* The algorithm checks any node: even recent (or future) proposals or
* nonstandard syntax such as JSX, because it ducktypes to find nodes instead
* of having a list of visitor keys.
*
* The algorithm supports `loc` fields (line/column), `range` fields (offsets),
* and direct `start` / `end` fields.
*
* @template {EstreeNode} Tree
* Node type.
* @param {Tree} tree
* Tree to attach to.
* @param {Array<EstreeComment> | null | undefined} [comments]
* List of comments.
* @returns {Tree}
* Given tree.
*/
// To do: next major: dont return given `tree`.
export function attachComments(tree, comments) {
const list = (comments || []).concat().sort(compare)
if (list.length > 0) walk(tree, {comments: list, index: 0})
return tree
}
/**
* Attach semistandard estree comment nodes to the tree.
*
* @param {EstreeNode} node
* Node.
* @param {State} state
* Info passed around.
* @returns {void}
* Nothing.
*/
function walk(node, state) {
// Done, we can quit.
if (state.index === state.comments.length) {
return
}
/** @type {Array<EstreeNode>} */
const children = []
/** @type {Array<EstreeComment>} */
const comments = []
/** @type {string} */
let key
// Find all children of `node`
for (key in node) {
if (own.call(node, key)) {
/** @type {EstreeNode | Array<EstreeNode>} */
// @ts-expect-error: indexable.
const value = node[key]
// Ignore comments.
if (value && typeof value === 'object' && key !== 'comments') {
if (Array.isArray(value)) {
let index = -1
while (++index < value.length) {
if (value[index] && typeof value[index].type === 'string') {
children.push(value[index])
}
}
} else if (typeof value.type === 'string') {
children.push(value)
}
}
}
}
// Sort the children.
children.sort(compare)
// Initial comments.
comments.push(...slice(state, node, false, {leading: true, trailing: false}))
let index = -1
while (++index < children.length) {
walk(children[index], state)
}
// Dangling or trailing comments.
comments.push(
...slice(state, node, true, {
leading: false,
trailing: children.length > 0
})
)
if (comments.length > 0) {
// @ts-expect-error, yes, because theyre nonstandard.
node.comments = comments
}
}
/**
* @param {State} state
* Info passed around.
* @param {EstreeNode} node
* Node.
* @param {boolean} compareEnd
* Whether to compare on the end (default is on start).
* @param {Fields} fields
* Fields.
* @returns {Array<EstreeComment>}
* Slice from `state.comments`.
*/
function slice(state, node, compareEnd, fields) {
/** @type {Array<EstreeComment>} */
const result = []
while (
state.comments[state.index] &&
compare(state.comments[state.index], node, compareEnd) < 1
) {
result.push(Object.assign({}, state.comments[state.index++], fields))
}
return result
}
/**
* Sort two nodes (or comments).
*
* @param {EstreeNode | EstreeComment} left
* A node.
* @param {EstreeNode | EstreeComment} right
* The other node.
* @param {boolean | undefined} [compareEnd=false]
* Compare on `end` of `right`, default is to compare on `start`.
* @returns {number}
* Sorting.
*/
function compare(left, right, compareEnd) {
const field = compareEnd ? 'end' : 'start'
// Offsets.
if (left.range && right.range) {
return left.range[0] - right.range[compareEnd ? 1 : 0]
}
// Points.
if (left.loc && left.loc.start && right.loc && right.loc[field]) {
return (
left.loc.start.line - right.loc[field].line ||
left.loc.start.column - right.loc[field].column
)
}
// Just `start` (and `end`) on nodes.
// Default in most parsers.
if ('start' in left && field in right) {
// @ts-expect-error Added by Acorn
return left.start - right[field]
}
return Number.NaN
}