1
0
silverbullet/webapp/syscalls/editor.ts

162 lines
4.6 KiB
TypeScript
Raw Normal View History

2022-04-03 16:12:16 +00:00
import {Editor} from "../editor";
import {syntaxTree} from "@codemirror/language";
import {Transaction} from "@codemirror/state";
import {SysCallMapping} from "../../plugos/system";
type SyntaxNode = {
name: string;
text: string;
from: number;
to: number;
};
function ensureAnchor(expr: any, start: boolean) {
var _a;
let { source } = expr;
let addStart = start && source[0] != "^",
addEnd = source[source.length - 1] != "$";
if (!addStart && !addEnd) return expr;
return new RegExp(
`${addStart ? "^" : ""}(?:${source})${addEnd ? "$" : ""}`,
(_a = expr.flags) !== null && _a !== void 0
? _a
: expr.ignoreCase
? "i"
: ""
);
}
2022-04-03 16:12:16 +00:00
export function editorSyscalls(editor: Editor): SysCallMapping {
return {
"editor.getCurrentPage": (): string => {
2022-04-03 16:12:16 +00:00
return editor.currentPage!;
},
"editor.getText": () => {
2022-04-03 16:12:16 +00:00
return editor.editorView?.state.sliceDoc();
},
"editor.getCursor": (): number => {
2022-04-03 16:12:16 +00:00
return editor.editorView!.state.selection.main.from;
},
"editor.save": async () => {
2022-04-03 16:12:16 +00:00
return editor.save(true);
},
"editor.navigate": async (ctx, name: string, pos: number) => {
2022-04-03 16:12:16 +00:00
await editor.navigate(name, pos);
},
"editor.reloadPage": async (ctx) => {
2022-04-03 16:12:16 +00:00
await editor.reloadPage();
},
"editor.openUrl": async (ctx, url: string) => {
2022-04-03 16:12:16 +00:00
window.open(url, "_blank")!.focus();
},
"editor.flashNotification": (ctx, message: string) => {
2022-04-03 16:12:16 +00:00
editor.flashNotification(message);
},
"editor.showRhs": (ctx, html: string) => {
2022-04-03 16:12:16 +00:00
editor.viewDispatch({
type: "show-rhs",
html: html,
});
},
"editor.insertAtPos": (ctx, text: string, pos: number) => {
2022-04-03 16:12:16 +00:00
editor.editorView!.dispatch({
changes: {
insert: text,
from: pos,
},
});
},
"editor.replaceRange": (ctx, from: number, to: number, text: string) => {
2022-04-03 16:12:16 +00:00
editor.editorView!.dispatch({
changes: {
insert: text,
from: from,
to: to,
},
});
},
"editor.moveCursor": (ctx, pos: number) => {
2022-04-03 16:12:16 +00:00
editor.editorView!.dispatch({
selection: {
anchor: pos,
},
});
},
"editor.insertAtCursor": (ctx, text: string) => {
2022-04-03 16:12:16 +00:00
let editorView = editor.editorView!;
let from = editorView.state.selection.main.from;
editorView.dispatch({
changes: {
insert: text,
from: from,
},
selection: {
anchor: from + text.length,
},
});
},
"editor.getSyntaxNodeUnderCursor": (): SyntaxNode | undefined => {
2022-04-03 16:12:16 +00:00
const editorState = editor.editorView!.state;
let selection = editorState.selection.main;
if (selection.empty) {
let node = syntaxTree(editorState).resolveInner(selection.from);
if (node) {
return {
name: node.name,
text: editorState.sliceDoc(node.from, node.to),
from: node.from,
to: node.to,
};
}
}
},
"editor.getLineUnderCursor": (): string => {
2022-04-03 16:12:16 +00:00
const editorState = editor.editorView!.state;
let selection = editorState.selection.main;
let line = editorState.doc.lineAt(selection.from);
return editorState.sliceDoc(line.from, line.to);
},
"editor.matchBefore": (
2022-04-03 16:12:16 +00:00
ctx,
regexp: string
): { from: number; to: number; text: string } | null => {
const editorState = editor.editorView!.state;
let selection = editorState.selection.main;
let from = selection.from;
if (selection.empty) {
let line = editorState.doc.lineAt(from);
let start = Math.max(line.from, from - 250);
let str = line.text.slice(start - line.from, from - line.from);
let found = str.search(ensureAnchor(new RegExp(regexp), false));
// console.log("Line", line, start, str, new RegExp(regexp), found);
return found < 0
? null
: { from: start + found, to: from, text: str.slice(found) };
}
return null;
},
"editor.getSyntaxNodeAtPos": (ctx, pos: number): SyntaxNode | undefined => {
2022-04-03 16:12:16 +00:00
const editorState = editor.editorView!.state;
let node = syntaxTree(editorState).resolveInner(pos);
if (node) {
return {
name: node.name,
text: editorState.sliceDoc(node.from, node.to),
from: node.from,
to: node.to,
};
}
2022-04-03 16:12:16 +00:00
},
"editor.dispatch": (ctx, change: Transaction) => {
2022-04-03 16:12:16 +00:00
editor.editorView!.dispatch(change);
},
"editor.prompt": (
ctx,
message: string,
defaultValue = ""
): string | null => {
2022-04-03 16:12:16 +00:00
return prompt(message, defaultValue);
},
};
}