68 lines
1.6 KiB
TypeScript
68 lines
1.6 KiB
TypeScript
import type { ParseTree } from "../../plug-api/lib/tree.ts";
|
|
import type { SyntaxNode } from "@lezer/common";
|
|
import type { Language } from "@codemirror/language";
|
|
|
|
export function lezerToParseTree(
|
|
text: string,
|
|
n: SyntaxNode,
|
|
offset = 0,
|
|
): 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[] = [];
|
|
let index = n.from;
|
|
for (const child of children) {
|
|
const s = text.substring(index, child.from);
|
|
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);
|
|
if (s) {
|
|
newChildren.push({ from: index + offset, to: n.to + offset, text: s });
|
|
}
|
|
children = newChildren;
|
|
}
|
|
|
|
const result: ParseTree = {
|
|
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 {
|
|
// Remove \r for Windows before parsing
|
|
text = text.replaceAll("\r", "");
|
|
const tree = lezerToParseTree(text, language.parser.parse(text).topNode);
|
|
return tree;
|
|
}
|