silverbullet/common/markdown_parser/parse_tree.ts

68 lines
1.6 KiB
TypeScript
Raw Permalink Normal View History

2024-02-29 22:23:05 +08:00
import type { ParseTree } from "../../plug-api/lib/tree.ts";
import type { SyntaxNode } from "@lezer/common";
import type { Language } from "@codemirror/language";
2022-04-12 02:34:09 +08:00
export function lezerToParseTree(
text: string,
n: SyntaxNode,
offset = 0,
2022-04-12 02:34:09 +08:00
): ParseTree {
let children: ParseTree[] = [];
let nodeText: string | undefined;
let child = n.firstChild;
while (child) {
children.push(lezerToParseTree(text, child));
child = child.nextSibling;
}
if (children.length === 0) {
children = [
{
from: n.from + offset,
to: n.to + offset,
text: text.substring(n.from, n.to),
},
];
} else {
const newChildren: ParseTree[] = [];
2022-04-12 02:34:09 +08:00
let index = n.from;
for (const child of children) {
const s = text.substring(index, child.from);
2022-04-12 02:34:09 +08:00
if (s) {
newChildren.push({
from: index + offset,
to: child.from! + offset,
text: s,
});
}
newChildren.push(child);
index = child.to!;
}
const s = text.substring(index, n.to);
2022-04-12 02:34:09 +08:00
if (s) {
newChildren.push({ from: index + offset, to: n.to + offset, text: s });
}
children = newChildren;
}
const result: ParseTree = {
2022-04-12 02:34:09 +08:00
type: n.name,
from: n.from + offset,
to: n.to + offset,
};
if (children.length > 0) {
result.children = children;
}
if (nodeText) {
result.text = nodeText;
}
return result;
}
export function parse(language: Language, text: string): ParseTree {
2023-01-02 03:28:25 +08:00
// Remove \r for Windows before parsing
text = text.replaceAll("\r", "");
const tree = lezerToParseTree(text, language.parser.parse(text).topNode);
2022-04-12 02:34:09 +08:00
return tree;
}