import { autocompletion, CompletionContext, completionKeymap, CompletionResult, } from "@codemirror/autocomplete"; import { closeBrackets, closeBracketsKeymap } from "@codemirror/closebrackets"; import { indentWithTab, standardKeymap } from "@codemirror/commands"; import { history, historyKeymap } from "@codemirror/history"; import { indentOnInput, syntaxTree } from "@codemirror/language"; import { bracketMatching } from "@codemirror/matchbrackets"; import { searchKeymap } from "@codemirror/search"; import { EditorState, StateField, Transaction } from "@codemirror/state"; import { KeyBinding } from "@codemirror/view"; import { drawSelection, dropCursor, EditorView, highlightSpecialChars, keymap, } from "@codemirror/view"; import React, { useEffect, useReducer } from "react"; import ReactDOM from "react-dom"; import coreManifest from "../../plugins/dist/core.plugin.json"; import { buildContext } from "./buildContext"; import * as commands from "./commands"; import { CommandPalette } from "./components/command_palette"; import { NavigationBar } from "./components/navigation_bar"; import { NuggetNavigator } from "./components/nugget_navigator"; import { StatusBar } from "./components/status_bar"; import { FileSystem, HttpFileSystem } from "./fs"; import { lineWrapper } from "./lineWrapper"; import { markdown } from "./markdown"; import customMarkDown from "./parser"; import { BrowserSystem } from "./plugins/browser_system"; import { Manifest } from "./plugins/types"; import reducer from "./reducer"; import customMarkdownStyle from "./style"; import dbSyscalls from "./syscalls/db.localstorage"; import editorSyscalls from "./syscalls/editor.browser"; import { Action, AppCommand, AppViewState, CommandContext, initialViewState, } from "./types"; import { safeRun } from "./util"; class NuggetState { editorState: EditorState; scrollTop: number; constructor(editorState: EditorState, scrollTop: number) { this.editorState = editorState; this.scrollTop = scrollTop; } } export class Editor { editorView?: EditorView; viewState: AppViewState; viewDispatch: React.Dispatch; $hashChange?: () => void; openNuggets: Map; fs: FileSystem; editorCommands: Map; constructor(fs: FileSystem, parent: Element) { this.editorCommands = new Map(); this.openNuggets = new Map(); this.fs = fs; this.viewState = initialViewState; this.viewDispatch = () => {}; this.render(parent); this.editorView = new EditorView({ state: this.createEditorState(""), parent: document.getElementById("editor")!, }); this.addListeners(); } async init() { await this.loadNuggetList(); await this.loadPlugins(); this.$hashChange!(); this.focus(); } async loadPlugins() { const system = new BrowserSystem("plugin"); system.registerSyscalls(dbSyscalls, editorSyscalls(this)); await system.bootServiceWorker(); console.log("Now loading core plugin"); let mainCartridge = await system.load("core", coreManifest as Manifest); this.editorCommands = new Map(); const cmds = mainCartridge.manifest!.commands; for (let name in cmds) { let cmd = cmds[name]; this.editorCommands.set(name, { command: cmd, run: async (arg: CommandContext): Promise => { return await mainCartridge.invoke(cmd.invoke, [arg]); }, }); } this.viewDispatch({ type: "update-commands", commands: this.editorCommands, }); } get currentNugget(): string | undefined { return this.viewState.currentNugget; } createEditorState(text: string): EditorState { const editor = this; 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; }, }); } } return EditorState.create({ doc: text, extensions: [ highlightSpecialChars(), history(), drawSelection(), dropCursor(), indentOnInput(), customMarkdownStyle, bracketMatching(), closeBrackets(), autocompletion({ override: [this.nuggetCompleter.bind(this)], }), 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, ...commandKeyBindings, { key: "Ctrl-b", mac: "Cmd-b", run: commands.insertMarker("**"), }, { key: "Ctrl-i", mac: "Cmd-i", run: commands.insertMarker("_"), }, { key: "Ctrl-s", mac: "Cmd-s", run: (target: EditorView): boolean => { Promise.resolve() .then(async () => { console.log("Saving"); await this.save(); }) .catch((e) => console.error(e)); return true; }, }, { key: "Ctrl-p", mac: "Cmd-p", run: (target): boolean => { this.viewDispatch({ type: "start-navigate" }); return true; }, }, { key: "Ctrl-.", mac: "Cmd-.", run: (target): boolean => { this.viewDispatch({ type: "show-palette" }); return true; }, }, ]), EditorView.domEventHandlers({ click: this.click.bind(this), }), markdown({ base: customMarkDown, }), StateField.define({ create: () => null, update: this.update.bind(this), }), ], }); } nuggetCompleter(ctx: CompletionContext): CompletionResult | null { let prefix = ctx.matchBefore(/\[\[\w*/); if (!prefix) { return null; } // TODO: Lots of optimization potential here // TODO: put something in the cm-completionIcon-nugget style return { from: prefix.from + 2, options: this.viewState.allNuggets.map((nuggetMeta) => ({ label: nuggetMeta.name, type: "nugget", })), }; } update(value: null, transaction: Transaction): null { if (transaction.docChanged) { this.viewDispatch({ type: "nugget-updated", }); } return null; } click(event: MouseEvent, view: EditorView) { if (event.metaKey || event.ctrlKey) { let coords = view.posAtCoords(event)!; let node = syntaxTree(view.state).resolveInner(coords); if (node && node.name === "WikiLinkPage") { let nuggetName = view.state.sliceDoc(node.from, node.to); this.navigate(nuggetName); } 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]" }, }); } } return false; } } async save() { const editorState = this.editorView!.state; if (!this.currentNugget) { return; } // Write to file system const created = await this.fs.writeNugget( this.currentNugget, editorState.sliceDoc() ); // Update in open nugget cache this.openNuggets.set( this.currentNugget, new NuggetState(editorState, this.editorView!.scrollDOM.scrollTop) ); // Dispatch update to view this.viewDispatch({ type: "nugget-saved" }); // If a new nugget was created, let's refresh the nugget list if (created) { await this.loadNuggetList(); } } async loadNuggetList() { let nuggetsMeta = await this.fs.listNuggets(); this.viewDispatch({ type: "nuggets-listed", nuggets: nuggetsMeta, }); } focus() { this.editorView!.focus(); } async navigate(name: string) { location.hash = encodeURIComponent(name); } hashChange() { Promise.resolve() .then(async () => { await this.save(); const nuggetName = decodeURIComponent(location.hash.substring(1)); console.log("Now navigating to", nuggetName); if (!this.editorView) { return; } let nuggetState = this.openNuggets.get(nuggetName); if (!nuggetState) { let text = await this.fs.readNugget(nuggetName); nuggetState = new NuggetState(this.createEditorState(text), 0); } this.openNuggets.set(nuggetName, nuggetState!); this.editorView!.setState(nuggetState!.editorState); this.editorView.scrollDOM.scrollTop = nuggetState!.scrollTop; this.viewDispatch({ type: "nugget-loaded", name: nuggetName, }); }) .catch((e) => { console.error(e); }); } addListeners() { this.$hashChange = this.hashChange.bind(this); window.addEventListener("hashchange", this.$hashChange); } dispose() { if (this.$hashChange) { window.removeEventListener("hashchange", this.$hashChange); } } ViewComponent(): React.ReactElement { const [viewState, dispatch] = useReducer(reducer, initialViewState); this.viewState = viewState; this.viewDispatch = dispatch; useEffect(() => { if (!location.hash) { this.navigate("start"); } }, []); // Auto save useEffect(() => { const id = setTimeout(() => { if (!viewState.isSaved) { this.save(); } }, 2000); return () => { clearTimeout(id); }; }, [viewState.isSaved]); let editor = this; useEffect(() => { if (viewState.currentNugget) { document.title = viewState.currentNugget; } }, [viewState.currentNugget]); return ( <> {viewState.showNuggetNavigator && ( { dispatch({ type: "stop-navigate" }); editor!.focus(); if (nugget) { editor ?.save() .then(() => { editor!.navigate(nugget); }) .catch((e) => { alert("Could not save nugget, not switching"); }); } }} /> )} {viewState.showCommandPalette && ( { dispatch({ type: "hide-palette" }); editor!.focus(); if (cmd) { safeRun(async () => { let result = await cmd.run(buildContext(cmd, editor)); console.log("Result of command", result); }); } }} commands={viewState.commands} /> )} { dispatch({ type: "start-navigate" }); }} />
); } render(container: ReactDOM.Container) { const ViewComponent = this.ViewComponent.bind(this); ReactDOM.render(, container); } } let ed = new Editor( new HttpFileSystem("http://localhost:2222/fs"), document.getElementById("root")! ); ed.loadPlugins().catch((e) => { console.error(e); }); safeRun(async () => { await ed.init(); }); // @ts-ignore window.editor = ed;