silverbullet/web/cm_plugins/lua_directive.ts

107 lines
3.6 KiB
TypeScript
Raw Normal View History

import type { EditorState, Range } from "@codemirror/state";
import { syntaxTree } from "@codemirror/language";
2024-10-08 01:17:12 +08:00
import { Decoration } from "@codemirror/view";
import {
decoratorStateField,
invisibleDecoration,
isCursorInRange,
shouldRenderWidgets,
} from "./util.ts";
import type { Client } from "../client.ts";
2024-10-07 15:08:36 +08:00
import { parse as parseLua } from "$common/space_lua/parse.ts";
2024-10-05 21:38:28 +08:00
import type {
LuaBlock,
LuaFunctionCallStatement,
} from "$common/space_lua/ast.ts";
import { evalExpression } from "$common/space_lua/eval.ts";
2024-10-20 21:06:23 +08:00
import {
LuaEnv,
LuaStackFrame,
luaValueToJS,
} from "$common/space_lua/runtime.ts";
2024-10-09 01:53:09 +08:00
import { LuaRuntimeError } from "$common/space_lua/runtime.ts";
import { encodePageRef } from "@silverbulletmd/silverbullet/lib/page_ref";
import { resolveASTReference } from "$common/space_lua.ts";
2024-10-13 21:14:22 +08:00
import { LuaWidget } from "./lua_widget.ts";
export function luaDirectivePlugin(client: Client) {
return decoratorStateField((state: EditorState) => {
const widgets: Range<Decoration>[] = [];
if (!shouldRenderWidgets(client)) {
console.info("Not rendering widgets");
return Decoration.set([]);
}
syntaxTree(state).iterate({
enter: (node) => {
if (node.name !== "LuaDirective") {
return;
}
if (isCursorInRange(state, [node.from, node.to])) {
return;
}
const text = state.sliceDoc(node.from + 2, node.to - 1);
2025-01-09 00:09:09 +08:00
const currentPageMeta = client.ui.viewState.currentPageMeta;
widgets.push(
Decoration.widget({
2024-10-13 21:14:22 +08:00
widget: new LuaWidget(
2024-10-08 01:17:12 +08:00
node.from,
client,
2025-01-09 00:09:09 +08:00
`lua:${text}:${currentPageMeta?.name}`,
2024-10-08 01:17:12 +08:00
text,
async (bodyText) => {
try {
const parsedLua = parseLua(`_(${bodyText})`) as LuaBlock;
const expr =
(parsedLua.statements[0] as LuaFunctionCallStatement).call
.args[0];
2025-01-09 00:09:09 +08:00
const tl = new LuaEnv();
tl.setLocal("pageMeta", currentPageMeta);
2025-01-09 18:45:15 +08:00
tl.setLocal("_GLOBAL", client.clientSystem.spaceLuaEnv.env);
2025-01-09 00:09:09 +08:00
const sf = new LuaStackFrame(tl, expr.ctx);
const threadLocalizedEnv = new LuaEnv(
client.clientSystem.spaceLuaEnv.env,
);
threadLocalizedEnv.setLocal("_CTX", tl);
const result = luaValueToJS(
2024-10-13 21:14:22 +08:00
await evalExpression(
expr,
2025-01-09 00:09:09 +08:00
threadLocalizedEnv,
2024-10-20 21:06:23 +08:00
sf,
2024-10-13 21:14:22 +08:00
),
2024-10-08 01:17:12 +08:00
);
2025-01-09 00:09:09 +08:00
// console.log("Result:", result);
return result;
2024-10-08 01:17:12 +08:00
} catch (e: any) {
2024-10-09 01:53:09 +08:00
if (e instanceof LuaRuntimeError) {
2024-10-20 21:06:23 +08:00
if (e.sf.astCtx) {
const source = resolveASTReference(e.sf.astCtx);
2024-10-09 01:53:09 +08:00
if (source) {
// We know the origin node of the error, let's reference it
return {
markdown: `**Lua error:** ${e.message} (Origin: [[${
encodePageRef(source)
}]])`,
};
}
}
}
2024-10-08 01:17:12 +08:00
return {
markdown: `**Lua error:** ${e.message}`,
};
}
},
),
}).range(node.to),
);
widgets.push(invisibleDecoration.range(node.from, node.to));
},
});
return Decoration.set(widgets, true);
});
}