1
0
silverbullet/webapp/src/editor.tsx

518 lines
14 KiB
TypeScript
Raw Normal View History

2022-02-24 16:24:49 +00:00
import {
autocompletion,
2022-02-25 14:34:00 +00:00
Completion,
2022-02-24 16:24:49 +00:00
CompletionContext,
completionKeymap,
CompletionResult,
} from "@codemirror/autocomplete";
2022-02-21 08:32:36 +00:00
import { closeBrackets, closeBracketsKeymap } from "@codemirror/closebrackets";
import { indentWithTab, standardKeymap } from "@codemirror/commands";
import { history, historyKeymap } from "@codemirror/history";
2022-02-24 16:24:49 +00:00
import { indentOnInput, syntaxTree } from "@codemirror/language";
2022-02-21 08:32:36 +00:00
import { bracketMatching } from "@codemirror/matchbrackets";
import { searchKeymap } from "@codemirror/search";
2022-02-21 10:27:30 +00:00
import { EditorState, StateField, Transaction } from "@codemirror/state";
2022-02-21 08:32:36 +00:00
import {
drawSelection,
dropCursor,
EditorView,
highlightSpecialChars,
2022-02-25 14:34:00 +00:00
KeyBinding,
2022-02-21 08:32:36 +00:00
keymap,
} from "@codemirror/view";
2022-02-24 16:24:49 +00:00
import React, { useEffect, useReducer } from "react";
2022-02-21 10:27:30 +00:00
import ReactDOM from "react-dom";
2022-02-24 16:24:49 +00:00
import coreManifest from "../../plugins/dist/core.plugin.json";
import { buildContext } from "./buildContext";
2022-02-21 08:32:36 +00:00
import * as commands from "./commands";
2022-02-25 10:27:58 +00:00
import { CommandPalette } from "./components/command_palette";
2022-02-24 16:24:49 +00:00
import { NavigationBar } from "./components/navigation_bar";
2022-02-25 10:27:58 +00:00
import { NuggetNavigator } from "./components/nugget_navigator";
2022-02-24 16:24:49 +00:00
import { StatusBar } from "./components/status_bar";
2022-02-25 14:34:00 +00:00
import { FileSystem } from "./fs";
2022-02-21 08:32:36 +00:00
import { lineWrapper } from "./lineWrapper";
2022-02-21 10:27:30 +00:00
import { markdown } from "./markdown";
2022-02-21 08:32:36 +00:00
import customMarkDown from "./parser";
2022-02-24 16:24:49 +00:00
import { BrowserSystem } from "./plugins/browser_system";
2022-02-25 14:34:00 +00:00
import { Manifest, slashCommandRegexp } from "./plugins/types";
2022-02-22 13:18:37 +00:00
import reducer from "./reducer";
2022-02-21 08:32:36 +00:00
import customMarkdownStyle from "./style";
2022-02-24 16:24:49 +00:00
import dbSyscalls from "./syscalls/db.localstorage";
import editorSyscalls from "./syscalls/editor.browser";
import {
Action,
AppCommand,
AppViewState,
CommandContext,
initialViewState,
2022-02-25 14:34:00 +00:00
NuggetMeta,
2022-02-24 16:24:49 +00:00
} from "./types";
import { safeRun } from "./util";
2022-02-23 13:09:26 +00:00
2022-02-25 10:27:58 +00:00
class NuggetState {
2022-02-23 13:09:26 +00:00
editorState: EditorState;
scrollTop: number;
2022-02-25 14:34:00 +00:00
meta: NuggetMeta;
2022-02-23 13:09:26 +00:00
2022-02-25 14:34:00 +00:00
constructor(editorState: EditorState, scrollTop: number, meta: NuggetMeta) {
2022-02-23 13:09:26 +00:00
this.editorState = editorState;
this.scrollTop = scrollTop;
2022-02-25 14:34:00 +00:00
this.meta = meta;
2022-02-23 13:09:26 +00:00
}
}
2022-02-22 16:36:24 +00:00
2022-02-25 14:34:00 +00:00
const watchInterval = 5000;
2022-02-24 16:24:49 +00:00
export class Editor {
2022-02-23 13:09:26 +00:00
editorView?: EditorView;
viewState: AppViewState;
viewDispatch: React.Dispatch<Action>;
$hashChange?: () => void;
2022-02-25 10:27:58 +00:00
openNuggets: Map<string, NuggetState>;
2022-02-23 13:09:26 +00:00
fs: FileSystem;
2022-02-24 16:24:49 +00:00
editorCommands: Map<string, AppCommand>;
2022-02-23 13:09:26 +00:00
constructor(fs: FileSystem, parent: Element) {
2022-02-24 16:24:49 +00:00
this.editorCommands = new Map();
2022-02-25 10:27:58 +00:00
this.openNuggets = new Map();
2022-02-23 13:09:26 +00:00
this.fs = fs;
this.viewState = initialViewState;
this.viewDispatch = () => {};
this.render(parent);
this.editorView = new EditorView({
state: this.createEditorState(""),
parent: document.getElementById("editor")!,
2022-02-21 08:32:36 +00:00
});
2022-02-23 13:09:26 +00:00
this.addListeners();
2022-02-25 14:34:00 +00:00
this.watch();
2022-02-24 16:24:49 +00:00
}
async init() {
2022-02-25 10:27:58 +00:00
await this.loadNuggetList();
2022-02-24 16:24:49 +00:00
await this.loadPlugins();
2022-02-23 13:09:26 +00:00
this.$hashChange!();
2022-02-24 16:24:49 +00:00
this.focus();
}
async loadPlugins() {
const system = new BrowserSystem("plugin");
system.registerSyscalls(dbSyscalls, editorSyscalls(this));
await system.bootServiceWorker();
console.log("Now loading core plugin");
2022-02-25 14:34:00 +00:00
let mainPlugin = await system.load("core", coreManifest as Manifest);
2022-02-24 16:24:49 +00:00
this.editorCommands = new Map<string, AppCommand>();
2022-02-25 14:34:00 +00:00
const cmds = mainPlugin.manifest!.commands;
2022-02-24 16:24:49 +00:00
for (let name in cmds) {
let cmd = cmds[name];
this.editorCommands.set(name, {
command: cmd,
run: async (arg: CommandContext): Promise<any> => {
2022-02-25 14:34:00 +00:00
return await mainPlugin.invoke(cmd.invoke, [arg]);
2022-02-24 16:24:49 +00:00
},
});
}
this.viewDispatch({
type: "update-commands",
commands: this.editorCommands,
});
2022-02-23 13:09:26 +00:00
}
2022-02-25 14:34:00 +00:00
get currentNugget(): NuggetMeta | undefined {
2022-02-25 10:27:58 +00:00
return this.viewState.currentNugget;
2022-02-21 08:32:36 +00:00
}
2022-02-21 10:27:30 +00:00
createEditorState(text: string): EditorState {
2022-02-22 13:18:37 +00:00
const editor = this;
2022-02-24 16:24:49 +00:00
let commandKeyBindings: KeyBinding[] = [];
for (let def of this.editorCommands.values()) {
if (def.command.key) {
commandKeyBindings.push({
key: def.command.key,
mac: def.command.mac,
run: (): boolean => {
Promise.resolve()
.then(async () => {
await def.run(buildContext(def, this));
})
.catch((e) => console.error(e));
return true;
},
});
}
}
2022-02-21 08:32:36 +00:00
return EditorState.create({
doc: text,
extensions: [
highlightSpecialChars(),
history(),
drawSelection(),
dropCursor(),
indentOnInput(),
customMarkdownStyle,
bracketMatching(),
closeBrackets(),
2022-02-22 16:36:24 +00:00
autocompletion({
2022-02-25 14:34:00 +00:00
override: [
this.nuggetCompleter.bind(this),
this.commandCompleter.bind(this),
],
2022-02-22 16:36:24 +00:00
}),
2022-02-21 08:32:36 +00:00
EditorView.lineWrapping,
lineWrapper([
{ selector: "ATXHeading1", class: "line-h1" },
{ selector: "ATXHeading2", class: "line-h2" },
{ selector: "ListItem", class: "line-li" },
{ selector: "Blockquote", class: "line-blockquote" },
{ selector: "CodeBlock", class: "line-code" },
{ selector: "FencedCode", class: "line-fenced-code" },
]),
keymap.of([
...closeBracketsKeymap,
...standardKeymap,
...searchKeymap,
...historyKeymap,
...completionKeymap,
indentWithTab,
2022-02-24 16:24:49 +00:00
...commandKeyBindings,
2022-02-21 08:32:36 +00:00
{
key: "Ctrl-b",
mac: "Cmd-b",
run: commands.insertMarker("**"),
},
{
key: "Ctrl-i",
mac: "Cmd-i",
run: commands.insertMarker("_"),
},
{
2022-02-25 14:34:00 +00:00
key: "Ctrl-e",
mac: "Cmd-e",
run: (): boolean => {
window.open(location.href, "_blank")!.focus();
2022-02-21 08:32:36 +00:00
return true;
},
},
2022-02-21 10:27:30 +00:00
{
key: "Ctrl-p",
mac: "Cmd-p",
run: (target): boolean => {
2022-02-23 13:09:26 +00:00
this.viewDispatch({ type: "start-navigate" });
2022-02-21 10:27:30 +00:00
return true;
},
},
2022-02-22 13:18:37 +00:00
{
key: "Ctrl-.",
mac: "Cmd-.",
run: (target): boolean => {
2022-02-25 14:34:00 +00:00
this.viewDispatch({
type: "show-palette",
});
2022-02-22 13:18:37 +00:00
return true;
},
},
2022-02-21 08:32:36 +00:00
]),
EditorView.domEventHandlers({
2022-02-21 10:27:30 +00:00
click: this.click.bind(this),
2022-02-21 08:32:36 +00:00
}),
markdown({
base: customMarkDown,
}),
StateField.define({
create: () => null,
2022-02-21 10:27:30 +00:00
update: this.update.bind(this),
2022-02-21 08:32:36 +00:00
}),
],
});
}
2022-02-21 10:27:30 +00:00
2022-02-25 10:27:58 +00:00
nuggetCompleter(ctx: CompletionContext): CompletionResult | null {
2022-02-25 14:34:00 +00:00
let prefix = ctx.matchBefore(/\[\[[\w\s]*/);
2022-02-22 16:36:24 +00:00
if (!prefix) {
return null;
}
return {
from: prefix.from + 2,
2022-02-25 10:27:58 +00:00
options: this.viewState.allNuggets.map((nuggetMeta) => ({
label: nuggetMeta.name,
type: "nugget",
2022-02-22 16:36:24 +00:00
})),
};
}
2022-02-25 14:34:00 +00:00
commandCompleter(ctx: CompletionContext): CompletionResult | null {
let prefix = ctx.matchBefore(slashCommandRegexp);
if (!prefix) {
return null;
}
let options: Completion[] = [];
for (let [name, def] of this.viewState.commands) {
if (!def.command.slashCommand) {
continue;
}
options.push({
label: def.command.slashCommand,
detail: name,
apply: () => {
this.editorView?.dispatch({
changes: {
from: prefix!.from,
to: ctx.pos,
insert: "",
},
});
safeRun(async () => {
def.run(buildContext(def, this));
});
},
});
}
return {
from: prefix.from + 1,
options: options,
};
}
2022-02-21 10:27:30 +00:00
update(value: null, transaction: Transaction): null {
if (transaction.docChanged) {
2022-02-23 13:09:26 +00:00
this.viewDispatch({
2022-02-25 10:27:58 +00:00
type: "nugget-updated",
2022-02-21 10:27:30 +00:00
});
}
return null;
}
click(event: MouseEvent, view: EditorView) {
if (event.metaKey || event.ctrlKey) {
2022-02-22 13:18:37 +00:00
let coords = view.posAtCoords(event)!;
let node = syntaxTree(view.state).resolveInner(coords);
if (node && node.name === "WikiLinkPage") {
2022-02-25 10:27:58 +00:00
let nuggetName = view.state.sliceDoc(node.from, node.to);
this.navigate(nuggetName);
2022-02-22 13:18:37 +00:00
}
2022-02-22 16:36:24 +00:00
if (node && node.name === "TaskMarker") {
let checkBoxText = view.state.sliceDoc(node.from, node.to);
if (checkBoxText === "[x]" || checkBoxText === "[X]") {
view.dispatch({
changes: { from: node.from, to: node.to, insert: "[ ]" },
});
} else {
view.dispatch({
changes: { from: node.from, to: node.to, insert: "[x]" },
});
}
}
2022-02-21 10:27:30 +00:00
return false;
}
}
async save() {
2022-02-23 13:09:26 +00:00
const editorState = this.editorView!.state;
2022-02-25 10:27:58 +00:00
if (!this.currentNugget) {
2022-02-23 13:09:26 +00:00
return;
}
// Write to file system
2022-02-25 14:34:00 +00:00
let nuggetMeta = await this.fs.writeNugget(
this.currentNugget.name,
2022-02-23 13:09:26 +00:00
editorState.sliceDoc()
2022-02-22 13:18:37 +00:00
);
2022-02-23 13:09:26 +00:00
2022-02-25 10:27:58 +00:00
// Update in open nugget cache
this.openNuggets.set(
2022-02-25 14:34:00 +00:00
this.currentNugget.name,
new NuggetState(
editorState,
this.editorView!.scrollDOM.scrollTop,
nuggetMeta
)
2022-02-23 13:09:26 +00:00
);
// Dispatch update to view
2022-02-25 14:34:00 +00:00
this.viewDispatch({ type: "nugget-saved", meta: nuggetMeta });
2022-02-23 13:09:26 +00:00
2022-02-25 10:27:58 +00:00
// If a new nugget was created, let's refresh the nugget list
2022-02-25 14:34:00 +00:00
if (nuggetMeta.created) {
2022-02-25 10:27:58 +00:00
await this.loadNuggetList();
2022-02-22 13:18:37 +00:00
}
}
2022-02-25 10:27:58 +00:00
async loadNuggetList() {
let nuggetsMeta = await this.fs.listNuggets();
2022-02-23 13:09:26 +00:00
this.viewDispatch({
2022-02-25 10:27:58 +00:00
type: "nuggets-listed",
nuggets: nuggetsMeta,
2022-02-22 13:18:37 +00:00
});
2022-02-21 10:27:30 +00:00
}
2022-02-25 14:34:00 +00:00
watch() {
setInterval(() => {
safeRun(async () => {
if (!this.currentNugget) {
return;
}
const currentNuggetName = this.currentNugget.name;
let newNuggetMeta = await this.fs.getMeta(currentNuggetName);
if (
this.currentNugget.lastModified.getTime() <
newNuggetMeta.lastModified.getTime()
) {
console.log("File changed on disk, reloading");
let nuggetData = await this.fs.readNugget(currentNuggetName);
this.openNuggets.set(
newNuggetMeta.name,
new NuggetState(
this.createEditorState(nuggetData.text),
0,
newNuggetMeta
)
);
await this.loadNugget(currentNuggetName);
}
});
}, watchInterval);
}
2022-02-21 10:27:30 +00:00
focus() {
2022-02-23 13:09:26 +00:00
this.editorView!.focus();
2022-02-21 10:27:30 +00:00
}
2022-02-21 12:25:41 +00:00
2022-02-23 13:09:26 +00:00
async navigate(name: string) {
2022-02-21 12:25:41 +00:00
location.hash = encodeURIComponent(name);
}
2022-02-21 08:32:36 +00:00
2022-02-23 13:09:26 +00:00
hashChange() {
Promise.resolve()
.then(async () => {
await this.save();
2022-02-25 10:27:58 +00:00
const nuggetName = decodeURIComponent(location.hash.substring(1));
console.log("Now navigating to", nuggetName);
2022-02-23 13:09:26 +00:00
if (!this.editorView) {
return;
}
2022-02-25 14:34:00 +00:00
await this.loadNugget(nuggetName);
2022-02-23 13:09:26 +00:00
})
.catch((e) => {
console.error(e);
});
}
2022-02-21 08:32:36 +00:00
2022-02-25 14:34:00 +00:00
async loadNugget(nuggetName: string) {
let nuggetState = this.openNuggets.get(nuggetName);
if (!nuggetState) {
let nuggetData = await this.fs.readNugget(nuggetName);
nuggetState = new NuggetState(
this.createEditorState(nuggetData.text),
0,
nuggetData.meta
);
this.openNuggets.set(nuggetName, nuggetState!);
}
this.editorView!.setState(nuggetState!.editorState);
this.editorView!.scrollDOM.scrollTop = nuggetState!.scrollTop;
this.viewDispatch({
type: "nugget-loaded",
meta: nuggetState.meta,
});
}
2022-02-23 13:09:26 +00:00
addListeners() {
this.$hashChange = this.hashChange.bind(this);
window.addEventListener("hashchange", this.$hashChange);
2022-02-22 13:18:37 +00:00
}
2022-02-21 10:27:30 +00:00
2022-02-23 13:09:26 +00:00
dispose() {
if (this.$hashChange) {
window.removeEventListener("hashchange", this.$hashChange);
2022-02-21 12:25:41 +00:00
}
2022-02-23 13:09:26 +00:00
}
2022-02-21 10:27:30 +00:00
2022-02-23 13:09:26 +00:00
ViewComponent(): React.ReactElement {
const [viewState, dispatch] = useReducer(reducer, initialViewState);
this.viewState = viewState;
this.viewDispatch = dispatch;
2022-02-21 08:32:36 +00:00
2022-02-23 13:09:26 +00:00
useEffect(() => {
if (!location.hash) {
this.navigate("start");
2022-02-22 13:18:37 +00:00
}
2022-02-23 13:09:26 +00:00
}, []);
// Auto save
useEffect(() => {
const id = setTimeout(() => {
if (!viewState.isSaved) {
this.save();
}
}, 2000);
return () => {
clearTimeout(id);
};
}, [viewState.isSaved]);
let editor = this;
2022-02-25 10:27:58 +00:00
useEffect(() => {
if (viewState.currentNugget) {
2022-02-25 14:34:00 +00:00
document.title = viewState.currentNugget.name;
2022-02-25 10:27:58 +00:00
}
}, [viewState.currentNugget]);
2022-02-23 13:09:26 +00:00
return (
<>
2022-02-25 10:27:58 +00:00
{viewState.showNuggetNavigator && (
<NuggetNavigator
allNuggets={viewState.allNuggets}
onNavigate={(nugget) => {
2022-02-23 13:09:26 +00:00
dispatch({ type: "stop-navigate" });
editor!.focus();
2022-02-25 10:27:58 +00:00
if (nugget) {
2022-02-23 13:09:26 +00:00
editor
?.save()
.then(() => {
2022-02-25 10:27:58 +00:00
editor!.navigate(nugget);
2022-02-23 13:09:26 +00:00
})
.catch((e) => {
2022-02-25 10:27:58 +00:00
alert("Could not save nugget, not switching");
2022-02-23 13:09:26 +00:00
});
}
}}
/>
)}
{viewState.showCommandPalette && (
<CommandPalette
onTrigger={(cmd) => {
dispatch({ type: "hide-palette" });
editor!.focus();
if (cmd) {
2022-02-24 16:24:49 +00:00
safeRun(async () => {
let result = await cmd.run(buildContext(cmd, editor));
console.log("Result of command", result);
});
2022-02-23 13:09:26 +00:00
}
}}
2022-02-24 16:24:49 +00:00
commands={viewState.commands}
2022-02-23 13:09:26 +00:00
/>
)}
<NavigationBar
2022-02-25 10:27:58 +00:00
currentNugget={viewState.currentNugget}
2022-02-23 13:09:26 +00:00
onClick={() => {
dispatch({ type: "start-navigate" });
2022-02-22 13:18:37 +00:00
}}
/>
2022-02-23 13:09:26 +00:00
<div id="editor"></div>
<StatusBar isSaved={viewState.isSaved} editorView={this.editorView} />
</>
);
}
render(container: ReactDOM.Container) {
const ViewComponent = this.ViewComponent.bind(this);
ReactDOM.render(<ViewComponent />, container);
}
2022-02-21 08:32:36 +00:00
}