gradio/js/code/interactive/frontmatter.ts

64 lines
1.6 KiB
TypeScript
Raw Normal View History

import type {
Element,
MarkdownExtension,
BlockContext,
Line
} from "@lezer/markdown";
import { parseMixed } from "@lezer/common";
import { yaml } from "@codemirror/legacy-modes/mode/yaml";
import { foldInside, foldNodeProp, StreamLanguage } from "@codemirror/language";
import { styleTags, tags } from "@lezer/highlight";
const frontMatterFence = /^---\s*$/m;
export const frontmatter: MarkdownExtension = {
defineNodes: [{ name: "Frontmatter", block: true }, "FrontmatterMark"],
props: [
styleTags({
Frontmatter: [tags.documentMeta, tags.monospace],
FrontmatterMark: tags.processingInstruction
}),
foldNodeProp.add({
Frontmatter: foldInside,
FrontmatterMark: () => null
})
],
wrap: parseMixed((node) => {
const { parser } = StreamLanguage.define(yaml);
if (node.type.name === "Frontmatter") {
return {
parser,
overlay: [{ from: node.from + 4, to: node.to - 4 }]
};
} else {
return null;
}
}),
parseBlock: [
{
name: "Frontmatter",
before: "HorizontalRule",
parse: (cx: BlockContext, line: Line): boolean => {
let end: number | undefined = undefined;
const children = new Array<Element>();
if (cx.lineStart === 0 && frontMatterFence.test(line.text)) {
children.push(cx.elt("FrontmatterMark", 0, 4));
while (cx.nextLine()) {
if (frontMatterFence.test(line.text)) {
end = cx.lineStart + 4;
break;
}
}
if (end !== undefined) {
children.push(cx.elt("FrontmatterMark", end - 4, end));
cx.addElement(cx.elt("Frontmatter", 0, end, children));
}
return true;
} else {
return false;
}
}
}
]
};