silverbullet/web/cm_plugins/directive.ts

128 lines
4.0 KiB
TypeScript
Raw Normal View History

2022-12-10 00:13:26 +08:00
import {
directiveEndRegex,
directiveStartRegex,
} from "../../plug-api/lib/query.ts";
import { Decoration, EditorState, syntaxTree } from "../deps.ts";
import type { Editor } from "../editor.tsx";
import { decoratorStateField, HtmlWidget, isCursorInRange } from "./util.ts";
2022-12-10 00:13:26 +08:00
// Does a few things: hides the directives when the cursor is not placed inside
// Adds a class to the start and end of the directive when the cursor is placed inside
export function directivePlugin(editor: Editor) {
2022-12-10 00:13:26 +08:00
return decoratorStateField((state: EditorState) => {
const widgets: any[] = [];
2022-12-10 00:13:26 +08:00
syntaxTree(state).iterate({
enter: ({ type, from, to, node }) => {
const parent = node.parent;
if (!parent) {
2022-12-10 00:13:26 +08:00
return;
}
const cursorInRange = isCursorInRange(state, [parent.from, parent.to]);
if (type.name === "DirectiveStart") {
if (cursorInRange) {
// Cursor outside this directive
2022-12-10 00:13:26 +08:00
widgets.push(
Decoration.line({ class: "sb-directive-start" }).range(from),
2022-12-10 00:13:26 +08:00
);
} else {
const text = state.sliceDoc(from, to);
const match = directiveStartRegex.exec(text);
if (!match) {
console.error("Something went wrong with this directive");
return;
}
const [fullMatch, directiveName] = match;
widgets.push(
Decoration.widget({
widget: new HtmlWidget(
`#${directiveName}`,
"sb-directive-placeholder",
(e) => {
e.stopPropagation();
editor.editorView?.dispatch({
selection: {
anchor: from + fullMatch.indexOf(directiveName),
},
});
},
),
}).range(from),
);
2022-12-10 00:13:26 +08:00
widgets.push(
Decoration.line({
class: "sb-directive-start sb-directive-start-outside",
}).range(
from,
),
2022-12-10 00:13:26 +08:00
);
}
return true;
}
if (type.name === "DirectiveEnd") {
// Cursor outside this directive
if (cursorInRange) {
2022-12-10 00:13:26 +08:00
widgets.push(
Decoration.line({ class: "sb-directive-end" }).range(from),
2022-12-10 00:13:26 +08:00
);
} else {
const text = state.sliceDoc(from, to);
const match = directiveEndRegex.exec(text);
if (!match) {
console.error("Something went wrong with this directive");
return;
}
const [fullMatch, directiveName] = match;
widgets.push(
Decoration.widget({
widget: new HtmlWidget(
`/${directiveName}`,
"sb-directive-placeholder",
(e) => {
e.stopPropagation();
editor.editorView?.dispatch({
selection: {
anchor: from + fullMatch.indexOf(directiveName),
},
});
},
),
}).range(from),
);
2022-12-10 00:13:26 +08:00
widgets.push(
Decoration.line({
class: "sb-directive-end sb-directive-end-outside",
}).range(
from,
),
2022-12-10 00:13:26 +08:00
);
}
return true;
}
if (type.name === "DirectiveBody") {
const lines = state.sliceDoc(from, to).split("\n");
let pos = from;
for (const line of lines) {
if (pos !== to) {
widgets.push(
Decoration.line({
class: "sb-directive-body",
}).range(pos),
);
}
pos += line.length + 1;
}
return true;
2022-12-10 00:13:26 +08:00
}
},
});
2022-12-10 00:13:26 +08:00
return Decoration.set(widgets, true);
});
}