2024-10-05 21:37:36 +08:00
|
|
|
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";
|
2024-10-05 21:37:36 +08:00
|
|
|
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";
|
2024-10-05 21:37:36 +08:00
|
|
|
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";
|
2024-10-05 21:37:36 +08:00
|
|
|
|
|
|
|
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;
|
2024-10-05 21:37:36 +08:00
|
|
|
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}`,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
},
|
|
|
|
),
|
2024-10-05 21:37:36 +08:00
|
|
|
}).range(node.to),
|
|
|
|
);
|
|
|
|
widgets.push(invisibleDecoration.range(node.from, node.to));
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
return Decoration.set(widgets, true);
|
|
|
|
});
|
|
|
|
}
|