silverbullet/packages/common/tree.ts

151 lines
3.4 KiB
TypeScript
Raw Normal View History

2022-04-12 02:34:09 +08:00
export type ParseTree = {
2022-04-04 17:51:41 +08:00
type?: string; // undefined === text node
2022-04-04 21:25:07 +08:00
from?: number;
to?: number;
2022-04-04 17:51:41 +08:00
text?: string;
2022-04-12 02:34:09 +08:00
children?: ParseTree[];
// Only present after running addParentPointers
parent?: ParseTree;
2022-04-04 00:12:16 +08:00
};
2022-04-12 02:34:09 +08:00
export function addParentPointers(tree: ParseTree) {
if (!tree.children) {
return;
}
for (let child of tree.children) {
if (child.parent) {
// Already added parent pointers before
return;
}
2022-04-12 02:34:09 +08:00
child.parent = tree;
addParentPointers(child);
}
}
export function removeParentPointers(tree: ParseTree) {
delete tree.parent;
if (!tree.children) {
return;
}
for (let child of tree.children) {
removeParentPointers(child);
}
}
export function findParentMatching(
tree: ParseTree,
matchFn: (tree: ParseTree) => boolean
): ParseTree | null {
let node = tree.parent;
while (node) {
if (matchFn(node)) {
return node;
}
node = node.parent;
}
return null;
}
export function collectNodesOfType(
tree: ParseTree,
nodeType: string
): ParseTree[] {
return collectNodesMatching(tree, (n) => n.type === nodeType);
}
export function collectNodesMatching(
tree: ParseTree,
matchFn: (tree: ParseTree) => boolean
): ParseTree[] {
if (matchFn(tree)) {
return [tree];
}
let results: ParseTree[] = [];
if (tree.children) {
for (let child of tree.children) {
results = [...results, ...collectNodesMatching(child, matchFn)];
}
2022-04-04 17:51:41 +08:00
}
2022-04-12 02:34:09 +08:00
return results;
}
2022-04-04 00:12:16 +08:00
2022-04-12 02:34:09 +08:00
// return value: returning undefined = not matched, continue, null = delete, new node = replace
export function replaceNodesMatching(
tree: ParseTree,
substituteFn: (tree: ParseTree) => ParseTree | null | undefined
) {
if (tree.children) {
let children = tree.children.slice();
2022-04-04 17:51:41 +08:00
for (let child of children) {
2022-04-12 02:34:09 +08:00
let subst = substituteFn(child);
if (subst !== undefined) {
let pos = tree.children.indexOf(child);
if (subst) {
tree.children.splice(pos, 1, subst);
} else {
// null = delete
tree.children.splice(pos, 1);
}
} else {
replaceNodesMatching(child, substituteFn);
2022-04-04 17:51:41 +08:00
}
2022-04-04 00:12:16 +08:00
}
2022-04-04 17:51:41 +08:00
}
2022-04-12 02:34:09 +08:00
}
export function findNodeMatching(
tree: ParseTree,
matchFn: (tree: ParseTree) => boolean
): ParseTree | null {
return collectNodesMatching(tree, matchFn)[0];
}
export function findNodeOfType(
tree: ParseTree,
nodeType: string
): ParseTree | null {
return collectNodesMatching(tree, (n) => n.type === nodeType)[0];
}
2022-04-04 00:12:16 +08:00
export function traverseTree(
tree: ParseTree,
// Return value = should stop traversal?
matchFn: (tree: ParseTree) => boolean
): void {
// Do a collect, but ignore the result
collectNodesMatching(tree, matchFn);
}
2022-04-12 02:34:09 +08:00
// Finds non-text node at position
export function nodeAtPos(tree: ParseTree, pos: number): ParseTree | null {
if (pos < tree.from! || pos > tree.to!) {
return null;
2022-04-04 17:51:41 +08:00
}
2022-04-12 02:34:09 +08:00
if (!tree.children) {
return tree;
2022-04-04 17:51:41 +08:00
}
2022-04-12 02:34:09 +08:00
for (let child of tree.children) {
let n = nodeAtPos(child, pos);
if (n && n.text !== undefined) {
// Got a text node, let's return its parent
return tree;
} else if (n) {
// Got it
return n;
}
}
return null;
2022-04-04 00:12:16 +08:00
}
2022-04-12 02:34:09 +08:00
// Turn ParseTree back into text
export function renderToText(tree: ParseTree): string {
let pieces: string[] = [];
if (tree.text !== undefined) {
return tree.text;
}
for (let child of tree.children!) {
pieces.push(renderToText(child));
}
return pieces.join("");
2022-04-04 00:12:16 +08:00
}