2023-07-14 22:56:20 +08:00
|
|
|
import { Client } from "../client.ts";
|
2023-06-15 01:27:18 +08:00
|
|
|
import {
|
|
|
|
foldAll,
|
|
|
|
foldCode,
|
2023-06-17 15:01:32 +08:00
|
|
|
toggleFold,
|
2023-06-15 01:27:18 +08:00
|
|
|
unfoldAll,
|
|
|
|
unfoldCode,
|
2024-03-16 22:29:24 +08:00
|
|
|
} from "@codemirror/language";
|
2024-05-26 05:12:48 +08:00
|
|
|
import { deleteLine, redo, undo } from "@codemirror/commands";
|
2024-03-16 22:29:24 +08:00
|
|
|
import { Transaction } from "@codemirror/state";
|
|
|
|
import { EditorView } from "@codemirror/view";
|
|
|
|
import { getCM as vimGetCm, Vim } from "@replit/codemirror-vim";
|
2024-02-09 04:00:45 +08:00
|
|
|
import { SysCallMapping } from "$lib/plugos/system.ts";
|
2024-03-16 22:29:24 +08:00
|
|
|
import type { FilterOption } from "$lib/web.ts";
|
2024-02-29 22:23:05 +08:00
|
|
|
import { UploadFile } from "../../plug-api/types.ts";
|
2024-02-09 04:00:45 +08:00
|
|
|
import { PageRef } from "$sb/lib/page_ref.ts";
|
2024-03-16 22:29:24 +08:00
|
|
|
import { openSearchPanel } from "@codemirror/search";
|
2024-02-09 04:00:45 +08:00
|
|
|
import { diffAndPrepareChanges } from "../cm_util.ts";
|
2022-03-20 16:56:28 +08:00
|
|
|
|
2024-01-21 02:16:07 +08:00
|
|
|
export function editorSyscalls(client: Client): SysCallMapping {
|
2022-09-30 22:59:57 +08:00
|
|
|
const syscalls: SysCallMapping = {
|
2022-04-04 00:42:12 +08:00
|
|
|
"editor.getCurrentPage": (): string => {
|
2024-01-24 21:44:39 +08:00
|
|
|
return client.currentPage;
|
2022-04-04 00:12:16 +08:00
|
|
|
},
|
2022-04-04 00:42:12 +08:00
|
|
|
"editor.getText": () => {
|
2024-01-21 02:16:07 +08:00
|
|
|
return client.editorView.state.sliceDoc();
|
2022-04-04 00:12:16 +08:00
|
|
|
},
|
2024-01-26 18:10:35 +08:00
|
|
|
"editor.setText": (_ctx, newText: string) => {
|
|
|
|
const currentText = client.editorView.state.sliceDoc();
|
|
|
|
const allChanges = diffAndPrepareChanges(currentText, newText);
|
|
|
|
client.editorView.dispatch({
|
|
|
|
changes: allChanges,
|
|
|
|
});
|
|
|
|
},
|
2022-04-04 00:42:12 +08:00
|
|
|
"editor.getCursor": (): number => {
|
2024-01-21 02:16:07 +08:00
|
|
|
return client.editorView.state.selection.main.from;
|
2022-04-04 00:12:16 +08:00
|
|
|
},
|
2022-06-23 23:08:42 +08:00
|
|
|
"editor.getSelection": (): { from: number; to: number } => {
|
2024-01-21 02:16:07 +08:00
|
|
|
return client.editorView.state.selection.main;
|
2022-06-23 23:08:42 +08:00
|
|
|
},
|
2022-10-16 01:02:56 +08:00
|
|
|
"editor.save": () => {
|
2024-01-21 02:16:07 +08:00
|
|
|
return client.save(true);
|
2022-04-04 00:12:16 +08:00
|
|
|
},
|
2022-08-01 21:06:32 +08:00
|
|
|
"editor.navigate": async (
|
2022-10-16 01:02:56 +08:00
|
|
|
_ctx,
|
2024-01-24 18:58:33 +08:00
|
|
|
pageRef: PageRef | string,
|
2022-10-10 20:50:21 +08:00
|
|
|
replaceState = false,
|
2022-10-29 15:23:12 +08:00
|
|
|
newWindow = false,
|
2022-08-01 21:06:32 +08:00
|
|
|
) => {
|
2024-01-24 18:58:33 +08:00
|
|
|
if (typeof pageRef === "string") {
|
|
|
|
pageRef = { page: pageRef };
|
|
|
|
}
|
|
|
|
await client.navigate(pageRef, replaceState, newWindow);
|
2022-04-04 00:12:16 +08:00
|
|
|
},
|
2022-10-11 00:19:08 +08:00
|
|
|
"editor.reloadPage": async () => {
|
2024-01-21 02:16:07 +08:00
|
|
|
await client.reloadPage();
|
2022-04-04 00:12:16 +08:00
|
|
|
},
|
2023-07-26 17:22:10 +08:00
|
|
|
"editor.reloadUI": () => {
|
|
|
|
location.reload();
|
|
|
|
},
|
2024-01-21 02:16:07 +08:00
|
|
|
"editor.reloadSettingsAndCommands": async () => {
|
|
|
|
await client.loadSettings();
|
2024-02-07 21:50:01 +08:00
|
|
|
await client.clientSystem.system.localSyscall(
|
|
|
|
"system.loadSpaceScripts",
|
|
|
|
[],
|
|
|
|
);
|
2024-03-09 19:26:58 +08:00
|
|
|
await client.clientSystem.system.localSyscall(
|
|
|
|
"system.loadSpaceStyles",
|
|
|
|
[],
|
|
|
|
);
|
2024-01-21 02:16:07 +08:00
|
|
|
},
|
2023-06-14 02:47:05 +08:00
|
|
|
"editor.openUrl": (_ctx, url: string, existingWindow = false) => {
|
|
|
|
if (!existingWindow) {
|
|
|
|
const win = window.open(url, "_blank");
|
|
|
|
if (win) {
|
|
|
|
win.focus();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
location.href = url;
|
2022-05-09 16:45:36 +08:00
|
|
|
}
|
2022-04-04 00:12:16 +08:00
|
|
|
},
|
2024-02-07 16:33:47 +08:00
|
|
|
"editor.goHistory": (_ctx, delta: number) => {
|
|
|
|
window.history.go(delta);
|
|
|
|
},
|
2023-01-08 22:29:34 +08:00
|
|
|
"editor.downloadFile": (_ctx, filename: string, dataUrl: string) => {
|
|
|
|
const link = document.createElement("a");
|
|
|
|
link.href = dataUrl;
|
|
|
|
link.download = filename;
|
|
|
|
link.click();
|
|
|
|
},
|
2023-11-22 22:33:25 +08:00
|
|
|
"editor.uploadFile": (
|
|
|
|
_ctx,
|
|
|
|
accept?: string,
|
2023-11-22 22:51:44 +08:00
|
|
|
capture?: string,
|
2023-11-22 22:33:25 +08:00
|
|
|
): Promise<UploadFile> => {
|
|
|
|
return new Promise<UploadFile>((resolve, reject) => {
|
|
|
|
const input = document.createElement("input");
|
|
|
|
input.type = "file";
|
|
|
|
if (accept) {
|
|
|
|
input.accept = accept;
|
|
|
|
}
|
|
|
|
if (capture) {
|
|
|
|
input.capture = capture;
|
|
|
|
}
|
|
|
|
|
|
|
|
input.onchange = () => {
|
|
|
|
const file = input.files?.item(0);
|
|
|
|
if (!file) {
|
|
|
|
reject(new Error("No file found"));
|
|
|
|
} else {
|
2023-11-22 22:51:44 +08:00
|
|
|
const reader = new FileReader();
|
2023-11-22 22:33:25 +08:00
|
|
|
reader.readAsArrayBuffer(file);
|
|
|
|
reader.onloadend = async (evt) => {
|
|
|
|
if (evt.target?.readyState == FileReader.DONE) {
|
|
|
|
resolve({
|
|
|
|
name: file.name,
|
2023-11-23 19:09:48 +08:00
|
|
|
contentType: file.type,
|
2023-11-22 22:51:44 +08:00
|
|
|
content: new Uint8Array(await file.arrayBuffer()),
|
2023-11-22 22:33:25 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
2023-11-22 22:51:44 +08:00
|
|
|
reader.onabort = (e) => {
|
|
|
|
reject(e);
|
|
|
|
};
|
|
|
|
reader.onerror = (e) => {
|
|
|
|
reject(e);
|
|
|
|
};
|
2023-11-22 22:33:25 +08:00
|
|
|
}
|
2023-11-22 22:51:44 +08:00
|
|
|
};
|
|
|
|
input.onabort = (e) => {
|
|
|
|
reject(e);
|
|
|
|
};
|
2023-11-22 22:33:25 +08:00
|
|
|
|
|
|
|
input.click();
|
|
|
|
});
|
|
|
|
},
|
2022-07-14 19:32:28 +08:00
|
|
|
"editor.flashNotification": (
|
2022-10-16 01:02:56 +08:00
|
|
|
_ctx,
|
2022-07-14 19:32:28 +08:00
|
|
|
message: string,
|
2022-10-10 20:50:21 +08:00
|
|
|
type: "error" | "info" = "info",
|
2022-07-14 19:32:28 +08:00
|
|
|
) => {
|
2024-01-21 02:16:07 +08:00
|
|
|
client.flashNotification(message, type);
|
2022-04-04 00:12:16 +08:00
|
|
|
},
|
2022-04-13 20:46:52 +08:00
|
|
|
"editor.filterBox": (
|
2022-10-16 01:02:56 +08:00
|
|
|
_ctx,
|
2022-04-13 20:46:52 +08:00
|
|
|
label: string,
|
|
|
|
options: FilterOption[],
|
2022-10-16 01:02:56 +08:00
|
|
|
helpText = "",
|
|
|
|
placeHolder = "",
|
2022-04-13 20:46:52 +08:00
|
|
|
): Promise<FilterOption | undefined> => {
|
2024-01-21 02:16:07 +08:00
|
|
|
return client.filterBox(label, options, helpText, placeHolder);
|
2022-04-13 20:46:52 +08:00
|
|
|
},
|
2022-09-30 22:59:57 +08:00
|
|
|
"editor.showPanel": (
|
2022-10-16 01:02:56 +08:00
|
|
|
_ctx,
|
2022-09-30 22:59:57 +08:00
|
|
|
id: string,
|
|
|
|
mode: number,
|
|
|
|
html: string,
|
2022-10-10 20:50:21 +08:00
|
|
|
script: string,
|
2022-09-30 22:59:57 +08:00
|
|
|
) => {
|
2024-01-21 02:16:07 +08:00
|
|
|
client.ui.viewDispatch({
|
2022-09-30 22:59:57 +08:00
|
|
|
type: "show-panel",
|
|
|
|
id: id as any,
|
|
|
|
config: { html, script, mode },
|
2022-04-04 00:12:16 +08:00
|
|
|
});
|
2023-10-03 20:16:33 +08:00
|
|
|
setTimeout(() => {
|
|
|
|
// Dummy dispatch to rerender the editor and toggle the panel
|
2024-01-21 02:16:07 +08:00
|
|
|
client.editorView.dispatch({});
|
2023-10-03 20:16:33 +08:00
|
|
|
});
|
2022-04-04 00:12:16 +08:00
|
|
|
},
|
2022-10-16 01:02:56 +08:00
|
|
|
"editor.hidePanel": (_ctx, id: string) => {
|
2024-01-21 02:16:07 +08:00
|
|
|
client.ui.viewDispatch({
|
2022-09-30 22:59:57 +08:00
|
|
|
type: "hide-panel",
|
|
|
|
id: id as any,
|
2022-04-04 21:25:07 +08:00
|
|
|
});
|
2023-10-03 20:16:33 +08:00
|
|
|
setTimeout(() => {
|
|
|
|
// Dummy dispatch to rerender the editor and toggle the panel
|
2024-01-21 02:16:07 +08:00
|
|
|
client.editorView.dispatch({});
|
2023-10-03 20:16:33 +08:00
|
|
|
});
|
2022-04-04 21:25:07 +08:00
|
|
|
},
|
2022-10-16 01:02:56 +08:00
|
|
|
"editor.insertAtPos": (_ctx, text: string, pos: number) => {
|
2024-01-21 02:16:07 +08:00
|
|
|
client.editorView.dispatch({
|
2022-04-04 00:12:16 +08:00
|
|
|
changes: {
|
|
|
|
insert: text,
|
|
|
|
from: pos,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
},
|
2022-10-16 01:02:56 +08:00
|
|
|
"editor.replaceRange": (_ctx, from: number, to: number, text: string) => {
|
2024-01-21 02:16:07 +08:00
|
|
|
client.editorView.dispatch({
|
2022-04-04 00:12:16 +08:00
|
|
|
changes: {
|
|
|
|
insert: text,
|
|
|
|
from: from,
|
|
|
|
to: to,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
},
|
2022-11-01 22:01:28 +08:00
|
|
|
"editor.moveCursor": (_ctx, pos: number, center = false) => {
|
2024-01-21 02:16:07 +08:00
|
|
|
client.editorView.dispatch({
|
2022-04-04 00:12:16 +08:00
|
|
|
selection: {
|
|
|
|
anchor: pos,
|
|
|
|
},
|
|
|
|
});
|
2022-11-01 22:01:28 +08:00
|
|
|
if (center) {
|
2024-01-21 02:16:07 +08:00
|
|
|
client.editorView.dispatch({
|
2022-11-01 22:01:28 +08:00
|
|
|
effects: [
|
|
|
|
EditorView.scrollIntoView(
|
|
|
|
pos,
|
|
|
|
{
|
|
|
|
y: "center",
|
|
|
|
},
|
|
|
|
),
|
|
|
|
],
|
|
|
|
});
|
|
|
|
}
|
2024-01-21 02:16:07 +08:00
|
|
|
client.editorView.focus();
|
2022-04-04 00:12:16 +08:00
|
|
|
},
|
2022-10-16 01:02:56 +08:00
|
|
|
"editor.setSelection": (_ctx, from: number, to: number) => {
|
2024-01-21 02:16:07 +08:00
|
|
|
client.editorView.dispatch({
|
2022-06-23 23:08:42 +08:00
|
|
|
selection: {
|
|
|
|
anchor: from,
|
|
|
|
head: to,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2022-10-16 01:02:56 +08:00
|
|
|
"editor.insertAtCursor": (_ctx, text: string) => {
|
2024-01-21 02:16:07 +08:00
|
|
|
const editorView = client.editorView;
|
2022-10-16 01:02:56 +08:00
|
|
|
const from = editorView.state.selection.main.from;
|
2022-04-04 00:12:16 +08:00
|
|
|
editorView.dispatch({
|
|
|
|
changes: {
|
|
|
|
insert: text,
|
|
|
|
from: from,
|
|
|
|
},
|
|
|
|
selection: {
|
|
|
|
anchor: from + text.length,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
},
|
2022-10-16 01:02:56 +08:00
|
|
|
"editor.dispatch": (_ctx, change: Transaction) => {
|
2024-01-21 02:16:07 +08:00
|
|
|
client.editorView.dispatch(change);
|
2022-04-04 00:12:16 +08:00
|
|
|
},
|
2022-04-04 00:42:12 +08:00
|
|
|
"editor.prompt": (
|
2022-10-16 01:02:56 +08:00
|
|
|
_ctx,
|
2022-04-04 00:42:12 +08:00
|
|
|
message: string,
|
2022-10-10 20:50:21 +08:00
|
|
|
defaultValue = "",
|
2022-12-21 23:08:51 +08:00
|
|
|
): Promise<string | undefined> => {
|
2024-01-21 02:16:07 +08:00
|
|
|
return client.prompt(message, defaultValue);
|
2022-04-04 00:12:16 +08:00
|
|
|
},
|
2024-01-15 23:43:12 +08:00
|
|
|
"editor.confirm": (_ctx, message: string): Promise<boolean> => {
|
2024-01-21 02:16:07 +08:00
|
|
|
return client.confirm(message);
|
2022-11-18 23:04:37 +08:00
|
|
|
},
|
2022-12-21 21:55:24 +08:00
|
|
|
"editor.getUiOption": (_ctx, key: string): any => {
|
2024-01-21 02:16:07 +08:00
|
|
|
return (client.ui.viewState.uiOptions as any)[key];
|
2022-12-21 21:55:24 +08:00
|
|
|
},
|
|
|
|
"editor.setUiOption": (_ctx, key: string, value: any) => {
|
2024-01-21 02:16:07 +08:00
|
|
|
client.ui.viewDispatch({
|
2022-12-21 21:55:24 +08:00
|
|
|
type: "set-ui-option",
|
|
|
|
key,
|
|
|
|
value,
|
2022-09-16 20:26:47 +08:00
|
|
|
});
|
|
|
|
},
|
2023-01-24 01:52:17 +08:00
|
|
|
"editor.vimEx": (_ctx, exCommand: string) => {
|
2024-01-21 02:16:07 +08:00
|
|
|
const cm = vimGetCm(client.editorView)!;
|
2023-01-24 01:52:17 +08:00
|
|
|
return Vim.handleEx(cm, exCommand);
|
|
|
|
},
|
2024-01-21 02:16:07 +08:00
|
|
|
"editor.openPageNavigator": (_ctx, mode: "page" | "template" = "page") => {
|
|
|
|
client.startPageNavigate(mode);
|
|
|
|
},
|
|
|
|
"editor.openCommandPalette": () => {
|
|
|
|
client.ui.viewDispatch({
|
|
|
|
type: "show-palette",
|
2024-02-23 06:25:02 +08:00
|
|
|
context: client.getContext(),
|
2024-01-21 02:16:07 +08:00
|
|
|
});
|
|
|
|
},
|
2024-05-26 05:12:48 +08:00
|
|
|
"editor.deleteLine": () => {
|
|
|
|
deleteLine(client.editorView);
|
|
|
|
},
|
2023-06-15 01:27:18 +08:00
|
|
|
// Folding
|
|
|
|
"editor.fold": () => {
|
2024-01-21 02:16:07 +08:00
|
|
|
foldCode(client.editorView);
|
2023-06-15 01:27:18 +08:00
|
|
|
},
|
|
|
|
"editor.unfold": () => {
|
2024-01-21 02:16:07 +08:00
|
|
|
unfoldCode(client.editorView);
|
2023-06-15 01:27:18 +08:00
|
|
|
},
|
2023-06-17 15:01:32 +08:00
|
|
|
"editor.toggleFold": () => {
|
2024-01-21 02:16:07 +08:00
|
|
|
toggleFold(client.editorView);
|
2023-06-17 15:01:32 +08:00
|
|
|
},
|
2023-06-15 01:27:18 +08:00
|
|
|
"editor.foldAll": () => {
|
2024-01-21 02:16:07 +08:00
|
|
|
foldAll(client.editorView);
|
2023-06-15 01:27:18 +08:00
|
|
|
},
|
|
|
|
"editor.unfoldAll": () => {
|
2024-01-21 02:16:07 +08:00
|
|
|
unfoldAll(client.editorView);
|
2023-06-15 01:27:18 +08:00
|
|
|
},
|
2024-03-02 19:14:27 +08:00
|
|
|
"editor.undo": () => {
|
|
|
|
return undo(client.editorView);
|
|
|
|
},
|
|
|
|
"editor.redo": () => {
|
|
|
|
return redo(client.editorView);
|
|
|
|
},
|
2024-01-26 02:46:08 +08:00
|
|
|
"editor.openSearchPanel": () => {
|
|
|
|
openSearchPanel(client.editorView);
|
|
|
|
},
|
2024-02-28 19:16:51 +08:00
|
|
|
"editor.copyToClipboard": (_ctx, data: string | Blob) => {
|
|
|
|
if (typeof data === "string") {
|
|
|
|
navigator.clipboard.writeText(data);
|
|
|
|
} else {
|
|
|
|
navigator.clipboard.write([new ClipboardItem({ [data.type]: data })]);
|
|
|
|
}
|
|
|
|
},
|
2022-04-04 00:12:16 +08:00
|
|
|
};
|
2022-09-30 22:59:57 +08:00
|
|
|
|
|
|
|
return syscalls;
|
2022-04-04 00:12:16 +08:00
|
|
|
}
|