// Third party web dependencies import { autocompletion, cLanguage, closeBrackets, closeBracketsKeymap, CompletionContext, completionKeymap, CompletionResult, cppLanguage, csharpLanguage, dartLanguage, drawSelection, dropCursor, EditorSelection, EditorState, EditorView, gitIgnoreCompiler, highlightSpecialChars, history, historyKeymap, indentOnInput, indentWithTab, javaLanguage, javascriptLanguage, jsonLanguage, KeyBinding, keymap, kotlinLanguage, LanguageDescription, LanguageSupport, markdown, objectiveCLanguage, objectiveCppLanguage, postgresqlLanguage, protobufLanguage, pythonLanguage, runScopeHandlers, rustLanguage, scalaLanguage, searchKeymap, shellLanguage, sqlLanguage, standardKeymap, StreamLanguage, syntaxHighlighting, syntaxTree, tomlLanguage, typescriptLanguage, ViewPlugin, ViewUpdate, xmlLanguage, yamlLanguage, } from "../common/deps.ts"; import { SilverBulletHooks } from "../common/manifest.ts"; import { loadMarkdownExtensions, MDExt, } from "../common/markdown_parser/markdown_ext.ts"; import buildMarkdown from "../common/markdown_parser/parser.ts"; import { Space } from "./space.ts"; import { markdownSyscalls } from "./syscalls/markdown.ts"; import { FilterOption, PageMeta } from "./types.ts"; import { isMacLike, parseYamlSettings, safeRun } from "../common/util.ts"; import { createSandbox } from "../plugos/environments/webworker_sandbox.ts"; import { EventHook } from "../plugos/hooks/event.ts"; import assetSyscalls from "../plugos/syscalls/asset.ts"; import { eventSyscalls } from "../plugos/syscalls/event.ts"; import { System } from "../plugos/system.ts"; import { cleanModePlugins } from "./cm_plugins/clean.ts"; import { CollabState } from "./cm_plugins/collab.ts"; import { attachmentExtension, pasteLinkExtension, } from "./cm_plugins/editor_paste.ts"; import { inlineImagesPlugin } from "./cm_plugins/inline_image.ts"; import { lineWrapper } from "./cm_plugins/line_wrapper.ts"; import { smartQuoteKeymap } from "./cm_plugins/smart_quotes.ts"; import { Confirm, Prompt } from "./components/basic_modals.tsx"; import { CommandPalette } from "./components/command_palette.tsx"; import { FilterList } from "./components/filter.tsx"; import { PageNavigator } from "./components/page_navigator.tsx"; import { Panel } from "./components/panel.tsx"; import { TopBar } from "./components/top_bar.tsx"; import { BookIcon, HomeIcon, preactRender, TerminalIcon, useEffect, useReducer, vim, yUndoManagerKeymap, } from "./deps.ts"; import { AppCommand, CommandHook } from "./hooks/command.ts"; import { SlashCommandHook } from "./hooks/slash_command.ts"; import { PathPageNavigator } from "./navigator.ts"; import reducer from "./reducer.ts"; import customMarkdownStyle from "./style.ts"; import { collabSyscalls } from "./syscalls/collab.ts"; import { editorSyscalls } from "./syscalls/editor.ts"; import { spaceSyscalls } from "./syscalls/space.ts"; import { systemSyscalls } from "./syscalls/system.ts"; import { Action, AppViewState, BuiltinSettings, initialViewState, } from "./types.ts"; import type { AppEvent, ClickEvent, CompleteEvent, } from "../plug-api/app_event.ts"; import { CodeWidgetHook } from "./hooks/code_widget.ts"; import { throttle } from "../common/async_util.ts"; import { readonlyMode } from "./cm_plugins/readonly.ts"; import { PageNamespaceHook } from "../common/hooks/page_namespace.ts"; import { CronHook } from "../plugos/hooks/cron.ts"; import { pageIndexSyscalls } from "./syscalls/index.ts"; import { storeSyscalls } from "../plugos/syscalls/store.dexie_browser.ts"; import { PlugSpacePrimitives } from "../common/spaces/plug_space_primitives.ts"; import { IndexedDBSpacePrimitives } from "../common/spaces/indexeddb_space_primitives.ts"; import { FileMetaSpacePrimitives } from "../common/spaces/file_meta_space_primitives.ts"; import { EventedSpacePrimitives } from "../common/spaces/evented_space_primitives.ts"; import { clientStoreSyscalls } from "./syscalls/clientStore.ts"; import { sandboxFetchSyscalls } from "./syscalls/fetch.ts"; import { shellSyscalls } from "./syscalls/shell.ts"; import { SyncService } from "./sync_service.ts"; import { yamlSyscalls } from "./syscalls/yaml.ts"; import { simpleHash } from "../common/crypto.ts"; import { DexieKVStore } from "../plugos/lib/kv_store.dexie.ts"; import { SyncStatus } from "../common/spaces/sync.ts"; import { HttpSpacePrimitives } from "../common/spaces/http_space_primitives.ts"; import { FallbackSpacePrimitives } from "../common/spaces/fallback_space_primitives.ts"; import { syncSyscalls } from "./syscalls/sync.ts"; import { FilteredSpacePrimitives } from "../common/spaces/filtered_space_primitives.ts"; import { globToRegExp } from "https://deno.land/std@0.189.0/path/glob.ts"; const frontMatterRegex = /^---\n(([^\n]|\n)*?)---\n/; class PageState { constructor( readonly scrollTop: number, readonly selection: EditorSelection, ) {} } const saveInterval = 1000; declare global { interface Window { // Injected via index.html silverBulletConfig: { spaceFolderPath: string; syncEndpoint: string; }; editor: Editor; } } // TODO: Oh my god, need to refactor this export class Editor { readonly commandHook: CommandHook; readonly slashCommandHook: SlashCommandHook; openPages = new Map(); editorView?: EditorView; viewState: AppViewState = initialViewState; viewDispatch: (action: Action) => void = () => {}; space: Space; remoteSpacePrimitives: HttpSpacePrimitives; pageNavigator?: PathPageNavigator; eventHook: EventHook; codeWidgetHook: CodeWidgetHook; saveTimeout: any; debouncedUpdateEvent = throttle(() => { this.eventHook .dispatchEvent("editor:updated") .catch((e) => console.error("Error dispatching editor:updated event", e)); }, 1000); system: System; mdExtensions: MDExt[] = []; // Track if plugs have been updated since sync cycle private plugsUpdated = false; fullSyncCompleted = false; // Runtime state (that doesn't make sense in viewState) collabState?: CollabState; syncService: SyncService; settings?: BuiltinSettings; kvStore: DexieKVStore; constructor( parent: Element, ) { const runtimeConfig = window.silverBulletConfig; // Instantiate a PlugOS system const system = new System(); this.system = system; // Generate a semi-unique prefix for the database so not to reuse databases for different space paths const dbPrefix = "" + simpleHash(runtimeConfig.spaceFolderPath); // Attach the page namespace hook const namespaceHook = new PageNamespaceHook(); system.addHook(namespaceHook); // Event hook this.eventHook = new EventHook(); system.addHook(this.eventHook); // Cron hook const cronHook = new CronHook(system); system.addHook(cronHook); const indexSyscalls = pageIndexSyscalls( `${dbPrefix}_page_index`, globalThis.indexedDB, ); this.kvStore = new DexieKVStore( `${dbPrefix}_store`, "data", globalThis.indexedDB, ); const storeCalls = storeSyscalls(this.kvStore); // Setup space this.remoteSpacePrimitives = new HttpSpacePrimitives( runtimeConfig.syncEndpoint, runtimeConfig.spaceFolderPath, true, ); const plugSpacePrimitives = new PlugSpacePrimitives( // Using fallback space primitives here to allow (by default) local reads to "fall through" to HTTP when files aren't synced yet new FallbackSpacePrimitives( new IndexedDBSpacePrimitives( `${dbPrefix}_space`, globalThis.indexedDB, ), this.remoteSpacePrimitives, ), namespaceHook, ); let fileFilterFn: (s: string) => boolean = () => true; const localSpacePrimitives = new FilteredSpacePrimitives( new FileMetaSpacePrimitives( new EventedSpacePrimitives( plugSpacePrimitives, this.eventHook, ), indexSyscalls, ), (meta) => fileFilterFn(meta.name), async () => { await this.loadSettings(); if (typeof this.settings?.spaceIgnore === "string") { fileFilterFn = gitIgnoreCompiler(this.settings.spaceIgnore).accepts; } else { fileFilterFn = () => true; } }, ); this.space = new Space(localSpacePrimitives, this.kvStore); this.space.watch(); this.syncService = new SyncService( localSpacePrimitives, this.remoteSpacePrimitives, this.kvStore, this.eventHook, (path) => { // Do not sync the current page if it's in collab mode (server will will handle persistence) // console.log("Checking", path); if (this.collabState && path === `${this.currentPage}.md`) { console.log("Collab mode, not syncing current page", path); return false; } // TODO: At some point we should remove the data.db exception here return path !== "data.db" && !plugSpacePrimitives.isLikelyHandled(path); }, ); // Code widget hook this.codeWidgetHook = new CodeWidgetHook(); this.system.addHook(this.codeWidgetHook); // Command hook this.commandHook = new CommandHook(); this.commandHook.on({ commandsUpdated: (commandMap) => { this.viewDispatch({ type: "update-commands", commands: commandMap, }); }, }); this.system.addHook(this.commandHook); // Slash command hook this.slashCommandHook = new SlashCommandHook(this); this.system.addHook(this.slashCommandHook); this.render(parent); this.editorView = new EditorView({ state: this.createEditorState("", "", false), parent: document.getElementById("sb-editor")!, }); // Syscalls available to all plugs this.system.registerSyscalls( [], eventSyscalls(this.eventHook), editorSyscalls(this), spaceSyscalls(this), systemSyscalls(this, this.system), markdownSyscalls(buildMarkdown(this.mdExtensions)), assetSyscalls(this.system), collabSyscalls(this), yamlSyscalls(), storeCalls, indexSyscalls, syncSyscalls(this.syncService), // LEGACY clientStoreSyscalls(storeCalls), ); // Syscalls that require some additional permissions this.system.registerSyscalls( ["fetch"], sandboxFetchSyscalls(this.remoteSpacePrimitives), ); this.system.registerSyscalls( ["shell"], shellSyscalls(this.remoteSpacePrimitives), ); // Make keyboard shortcuts work even when the editor is in read only mode or not focused globalThis.addEventListener("keydown", (ev) => { if (!this.editorView?.hasFocus) { if ((ev.target as any).closest(".cm-editor")) { // In some cm element, let's back out return; } if (runScopeHandlers(this.editorView!, ev, "editor")) { ev.preventDefault(); } } }); globalThis.addEventListener("touchstart", (ev) => { // Launch the page picker on a two-finger tap if (ev.touches.length === 2) { ev.stopPropagation(); ev.preventDefault(); this.viewDispatch({ type: "start-navigate" }); } // Launch the command palette using a three-finger tap if (ev.touches.length === 3) { ev.stopPropagation(); ev.preventDefault(); this.viewDispatch({ type: "show-palette", context: this.getContext() }); } }); this.eventHook.addLocalListener("plug:changed", async (fileName) => { console.log("Plug updated, reloading:", fileName); system.unload(fileName); await system.load( // await this.space.readFile(fileName, "utf8"), new URL(`/.fs/${fileName}`, location.href), createSandbox, ); this.plugsUpdated = true; }); } get currentPage(): string | undefined { return this.viewState.currentPage; } async init() { this.focus(); this.space.on({ pageChanged: (meta) => { // Only reload when watching the current page (to avoid reloading when switching pages and in collab mode) if (this.space.watchInterval && this.currentPage === meta.name) { console.log("Page changed elsewhere, reloading"); this.flashNotification("Page changed elsewhere, reloading"); this.reloadPage(); } }, pageListUpdated: (pages) => { this.viewDispatch({ type: "pages-listed", pages: pages, }); }, }); // Load settings this.settings = await this.loadSettings(); this.pageNavigator = new PathPageNavigator( this.settings.indexPage, ); await this.reloadPlugs(); this.pageNavigator.subscribe(async (pageName, pos: number | string) => { console.log("Now navigating to", pageName); if (!this.editorView) { return; } const stateRestored = await this.loadPage(pageName); if (pos) { if (typeof pos === "string") { console.log("Navigating to anchor", pos); // We're going to look up the anchor through a direct page store query... const posLookup = await this.system.localSyscall( "core", "index.get", [ pageName, `a:${pageName}:${pos}`, ], ); if (!posLookup) { return this.flashNotification( `Could not find anchor @${pos}`, "error", ); } else { pos = +posLookup; } } this.editorView.dispatch({ selection: { anchor: pos }, scrollIntoView: true, }); } else if (!stateRestored) { // Somewhat ad-hoc way to determine if the document contains frontmatter and if so, putting the cursor _after it_. const pageText = this.editorView.state.sliceDoc(); // Default the cursor to be at position 0 let initialCursorPos = 0; const match = frontMatterRegex.exec(pageText); if (match) { // Frontmatter found, put cursor after it initialCursorPos = match[0].length; } // By default scroll to the top this.editorView.scrollDOM.scrollTop = 0; this.editorView.dispatch({ selection: { anchor: initialCursorPos }, // And then scroll down if required scrollIntoView: true, }); } }); this.loadCustomStyles().catch(console.error); // Kick off background sync this.syncService.start(); this.eventHook.addLocalListener("sync:success", async (operations) => { // console.log("Operations", operations); if (operations > 0) { // Update the page list await this.space.updatePageList(); } if (operations !== undefined) { // "sync:success" is called with a number of operations only from syncSpace(), not from syncing individual pages this.fullSyncCompleted = true; } if (this.plugsUpdated) { // To register new commands, update editor state based on new plugs this.rebuildEditorState(); this.dispatchAppEvent("editor:pageLoaded", this.currentPage); if (operations) { // Likely initial sync so let's show visually that we're synced now this.flashNotification(`Synced ${operations} files`, "info"); } } // Reset for next sync cycle this.plugsUpdated = false; this.viewDispatch({ type: "sync-change", synced: true }); }); this.eventHook.addLocalListener("sync:error", (name) => { this.viewDispatch({ type: "sync-change", synced: false }); }); this.eventHook.addLocalListener("sync:conflict", (name) => { this.flashNotification( `Sync: conflict detected for ${name} - conflict copy created`, "error", ); }); this.eventHook.addLocalListener("sync:progress", (status: SyncStatus) => { this.flashNotification( `Sync: ${ Math.round(status.filesProcessed / status.totalFiles * 10000) / 100 }% — processed ${status.filesProcessed} out of ${status.totalFiles}`, "info", ); }); await this.dispatchAppEvent("editor:init"); } async loadSettings(): Promise { let settingsText: string | undefined; try { settingsText = (await this.space.readPage("SETTINGS")).text; } catch (e: any) { console.log("No SETTINGS page, falling back to default"); settingsText = "```yaml\nindexPage: index\n```\n"; } const settings = parseYamlSettings(settingsText!) as BuiltinSettings; if (!settings.indexPage) { settings.indexPage = "index"; } return settings; } save(immediate = false): Promise { return new Promise((resolve, reject) => { if (this.saveTimeout) { clearTimeout(this.saveTimeout); } this.saveTimeout = setTimeout( () => { if (this.currentPage) { if ( !this.viewState.unsavedChanges || this.viewState.uiOptions.forcedROMode ) { // No unsaved changes, or read-only mode, not gonna save return resolve(); } console.log("Saving page", this.currentPage); this.space .writePage( this.currentPage, this.editorView!.state.sliceDoc(0), true, ) .then(() => { this.viewDispatch({ type: "page-saved" }); resolve(); }) .catch((e) => { this.flashNotification( "Could not save page, retrying again in 10 seconds", "error", ); this.saveTimeout = setTimeout(this.save.bind(this), 10000); reject(e); }); } else { resolve(); } }, immediate ? 0 : saveInterval, ); }); } flashNotification(message: string, type: "info" | "error" = "info") { const id = Math.floor(Math.random() * 1000000); this.viewDispatch({ type: "show-notification", notification: { id, type, message, date: new Date(), }, }); setTimeout( () => { this.viewDispatch({ type: "dismiss-notification", id: id, }); }, type === "info" ? 4000 : 5000, ); } filterBox( label: string, options: FilterOption[], helpText = "", placeHolder = "", ): Promise { return new Promise((resolve) => { this.viewDispatch({ type: "show-filterbox", label, options, placeHolder, helpText, onSelect: (option: any) => { this.viewDispatch({ type: "hide-filterbox" }); this.focus(); resolve(option); }, }); }); } prompt( message: string, defaultValue = "", ): Promise { return new Promise((resolve) => { this.viewDispatch({ type: "show-prompt", message, defaultValue, callback: (value: string | undefined) => { this.viewDispatch({ type: "hide-prompt" }); this.focus(); resolve(value); }, }); }); } confirm( message: string, ): Promise { return new Promise((resolve) => { this.viewDispatch({ type: "show-confirm", message, callback: (value: boolean) => { this.viewDispatch({ type: "hide-confirm" }); this.focus(); resolve(value); }, }); }); } dispatchAppEvent(name: AppEvent, data?: any): Promise { return this.eventHook.dispatchEvent(name, data); } createEditorState( pageName: string, text: string, readOnly: boolean, ): EditorState { const commandKeyBindings: KeyBinding[] = []; for (const def of this.commandHook.editorCommands.values()) { if (def.command.key) { commandKeyBindings.push({ key: def.command.key, mac: def.command.mac, run: (): boolean => { if (def.command.contexts) { const context = this.getContext(); if (!context || !def.command.contexts.includes(context)) { return false; } } Promise.resolve() .then(def.run) .catch((e: any) => { console.error(e); this.flashNotification( `Error running command: ${e.message}`, "error", ); }) .then(() => { // Always be focusing the editor after running a command editor.focus(); }); return true; }, }); } } // deno-lint-ignore no-this-alias const editor = this; let touchCount = 0; return EditorState.create({ doc: this.collabState ? this.collabState.ytext.toString() : text, extensions: [ // Not using CM theming right now, but some extensions depend on the "dark" thing EditorView.theme({}, { dark: this.viewState.uiOptions.darkMode }), // Enable vim mode, or not [...editor.viewState.uiOptions.vimMode ? [vim({ status: true })] : []], [ ...readOnly || editor.viewState.uiOptions.forcedROMode ? [readonlyMode()] : [], ], // The uber markdown mode markdown({ base: buildMarkdown(this.mdExtensions), codeLanguages: [ LanguageDescription.of({ name: "yaml", alias: ["meta", "data", "embed"], support: new LanguageSupport(StreamLanguage.define(yamlLanguage)), }), LanguageDescription.of({ name: "javascript", alias: ["js"], support: new LanguageSupport(javascriptLanguage), }), LanguageDescription.of({ name: "typescript", alias: ["ts"], support: new LanguageSupport(typescriptLanguage), }), LanguageDescription.of({ name: "sql", alias: ["sql"], support: new LanguageSupport(StreamLanguage.define(sqlLanguage)), }), LanguageDescription.of({ name: "postgresql", alias: ["pgsql", "postgres"], support: new LanguageSupport( StreamLanguage.define(postgresqlLanguage), ), }), LanguageDescription.of({ name: "rust", alias: ["rs"], support: new LanguageSupport(StreamLanguage.define(rustLanguage)), }), LanguageDescription.of({ name: "css", support: new LanguageSupport(StreamLanguage.define(sqlLanguage)), }), LanguageDescription.of({ name: "python", alias: ["py"], support: new LanguageSupport( StreamLanguage.define(pythonLanguage), ), }), LanguageDescription.of({ name: "protobuf", alias: ["proto"], support: new LanguageSupport( StreamLanguage.define(protobufLanguage), ), }), LanguageDescription.of({ name: "shell", alias: ["sh", "bash", "zsh", "fish"], support: new LanguageSupport( StreamLanguage.define(shellLanguage), ), }), LanguageDescription.of({ name: "swift", support: new LanguageSupport(StreamLanguage.define(rustLanguage)), }), LanguageDescription.of({ name: "toml", support: new LanguageSupport(StreamLanguage.define(tomlLanguage)), }), LanguageDescription.of({ name: "json", support: new LanguageSupport(StreamLanguage.define(jsonLanguage)), }), LanguageDescription.of({ name: "xml", support: new LanguageSupport(StreamLanguage.define(xmlLanguage)), }), LanguageDescription.of({ name: "c", support: new LanguageSupport(StreamLanguage.define(cLanguage)), }), LanguageDescription.of({ name: "cpp", alias: ["c++", "cxx"], support: new LanguageSupport(StreamLanguage.define(cppLanguage)), }), LanguageDescription.of({ name: "java", support: new LanguageSupport(StreamLanguage.define(javaLanguage)), }), LanguageDescription.of({ name: "csharp", alias: ["c#", "cs"], support: new LanguageSupport( StreamLanguage.define(csharpLanguage), ), }), LanguageDescription.of({ name: "scala", alias: ["sc"], support: new LanguageSupport( StreamLanguage.define(scalaLanguage), ), }), LanguageDescription.of({ name: "kotlin", alias: ["kt", "kts"], support: new LanguageSupport( StreamLanguage.define(kotlinLanguage), ), }), LanguageDescription.of({ name: "objc", alias: ["objective-c", "objectivec"], support: new LanguageSupport( StreamLanguage.define(objectiveCLanguage), ), }), LanguageDescription.of({ name: "objcpp", alias: [ "objc++", "objective-cpp", "objectivecpp", "objective-c++", "objectivec++", ], support: new LanguageSupport( StreamLanguage.define(objectiveCppLanguage), ), }), LanguageDescription.of({ name: "dart", support: new LanguageSupport(StreamLanguage.define(dartLanguage)), }), ], addKeymap: true, }), syntaxHighlighting(customMarkdownStyle(this.mdExtensions)), autocompletion({ override: [ this.editorComplete.bind(this), this.slashCommandHook.slashCommandCompleter.bind( this.slashCommandHook, ), ], }), inlineImagesPlugin(this.space), highlightSpecialChars(), history(), drawSelection(), dropCursor(), indentOnInput(), ...cleanModePlugins(this), EditorView.lineWrapping, lineWrapper([ { selector: "ATXHeading1", class: "sb-line-h1" }, { selector: "ATXHeading2", class: "sb-line-h2" }, { selector: "ATXHeading3", class: "sb-line-h3" }, { selector: "ATXHeading4", class: "sb-line-h4" }, { selector: "ListItem", class: "sb-line-li", nesting: true }, { selector: "Blockquote", class: "sb-line-blockquote" }, { selector: "Task", class: "sb-line-task" }, { selector: "CodeBlock", class: "sb-line-code" }, { selector: "FencedCode", class: "sb-line-fenced-code" }, { selector: "Comment", class: "sb-line-comment" }, { selector: "BulletList", class: "sb-line-ul" }, { selector: "OrderedList", class: "sb-line-ol" }, { selector: "TableHeader", class: "sb-line-tbl-header" }, { selector: "FrontMatter", class: "sb-frontmatter" }, ]), keymap.of([ ...smartQuoteKeymap, ...closeBracketsKeymap, ...standardKeymap, ...searchKeymap, ...historyKeymap, ...completionKeymap, ...(this.collabState ? yUndoManagerKeymap : []), indentWithTab, ...commandKeyBindings, { key: "Ctrl-k", mac: "Cmd-k", run: (): boolean => { this.viewDispatch({ type: "start-navigate" }); this.space.updatePageList(); return true; }, }, { key: "Ctrl-/", mac: "Cmd-/", run: (): boolean => { this.viewDispatch({ type: "show-palette", context: this.getContext(), }); return true; }, }, { key: "Ctrl-.", mac: "Cmd-.", run: (): boolean => { this.viewDispatch({ type: "show-palette", context: this.getContext(), }); return true; }, }, ]), EditorView.domEventHandlers({ // This may result in duplicated touch events on mobile devices touchmove: (event: TouchEvent, view: EditorView) => { touchCount++; }, touchend: (event: TouchEvent, view: EditorView) => { if (touchCount === 0) { safeRun(async () => { const touch = event.changedTouches.item(0)!; const clickEvent: ClickEvent = { page: pageName, ctrlKey: event.ctrlKey, metaKey: event.metaKey, altKey: event.altKey, pos: view.posAtCoords({ x: touch.clientX, y: touch.clientY, })!, }; await this.dispatchAppEvent("page:click", clickEvent); }); } touchCount = 0; }, mousedown: (event: MouseEvent, view: EditorView) => { safeRun(async () => { const pos = view.posAtCoords(event); if (!pos) { return; } const potentialClickEvent: ClickEvent = { page: pageName, ctrlKey: event.ctrlKey, metaKey: event.metaKey, altKey: event.altKey, pos: view.posAtCoords({ x: event.x, y: event.y, })!, }; // Make sure tags are clicked without moving the cursor there if (!event.altKey && event.target instanceof Element) { const parentA = event.target.closest("a"); if (parentA) { event.stopPropagation(); event.preventDefault(); await this.dispatchAppEvent( "page:click", potentialClickEvent, ); return; } } const distanceX = event.x - view.coordsAtPos(pos)!.left; // What we're trying to determine here is if the click occured anywhere near the looked up position // this may not be the case with locations that expand signifcantly based on live preview (such as links), we don't want any accidental clicks // Fixes #357 if (distanceX <= view.defaultCharacterWidth) { await this.dispatchAppEvent("page:click", potentialClickEvent); } }); }, }), ViewPlugin.fromClass( class { update(update: ViewUpdate): void { if (update.docChanged) { editor.viewDispatch({ type: "page-changed" }); editor.debouncedUpdateEvent(); editor.save().catch((e) => console.error("Error saving", e)); } } }, ), pasteLinkExtension, attachmentExtension(this), closeBrackets(), ...[this.collabState ? this.collabState.collabExtension() : []], ], }); } async reloadPlugs() { console.log("Loading plugs"); await this.space.updatePageList(); await this.system.unloadAll(); console.log("(Re)loading plugs"); await Promise.all((await this.space.listPlugs()).map(async (plugName) => { try { await this.system.load( new URL(`/.fs/${plugName}`, location.href), createSandbox, ); } catch (e: any) { console.error("Could not load plug", plugName, "error:", e.message); } })); this.rebuildEditorState(); await this.dispatchAppEvent("plugs:loaded"); } rebuildEditorState() { const editorView = this.editorView; console.log("Rebuilding editor state"); // Load all syntax extensions this.mdExtensions = loadMarkdownExtensions(this.system); // And reload the syscalls to use the new syntax extensions this.system.registerSyscalls( [], markdownSyscalls(buildMarkdown(this.mdExtensions)), ); if (editorView && this.currentPage) { // And update the editor if a page is loaded this.saveState(this.currentPage); editorView.setState( this.createEditorState( this.currentPage, editorView.state.sliceDoc(), this.viewState.currentPageMeta?.perm === "ro", ), ); if (editorView.contentDOM) { this.tweakEditorDOM( editorView.contentDOM, ); } this.restoreState(this.currentPage); } } // Code completion support private async completeWithEvent( context: CompletionContext, eventName: AppEvent, ): Promise { const editorState = context.state; const selection = editorState.selection.main; const line = editorState.doc.lineAt(selection.from); const linePrefix = line.text.slice(0, selection.from - line.from); const results = await this.dispatchAppEvent(eventName, { linePrefix, pos: selection.from, } as CompleteEvent); let actualResult = null; for (const result of results) { if (result) { if (actualResult) { console.error( "Got completion results from multiple sources, cannot deal with that", ); return null; } actualResult = result; } } return actualResult; } editorComplete( context: CompletionContext, ): Promise { return this.completeWithEvent(context, "editor:complete"); } miniEditorComplete( context: CompletionContext, ): Promise { return this.completeWithEvent(context, "minieditor:complete"); } async reloadPage() { console.log("Reloading page"); clearTimeout(this.saveTimeout); await this.loadPage(this.currentPage!); } focus() { this.editorView!.focus(); } async navigate( name: string, pos?: number | string, replaceState = false, newWindow = false, ) { if (!name) { name = this.settings!.indexPage; } if (newWindow) { const win = window.open(`${location.origin}/${name}`, "_blank"); if (win) { win.focus(); } return; } await this.pageNavigator!.navigate(name, pos, replaceState); } async loadPage(pageName: string): Promise { const loadingDifferentPage = pageName !== this.currentPage; const editorView = this.editorView; if (!editorView) { return false; } const previousPage = this.currentPage; // Persist current page state and nicely close page if (previousPage) { this.saveState(previousPage); this.space.unwatchPage(previousPage); if (previousPage !== pageName) { await this.save(true); // And stop the collab session if (this.collabState) { this.stopCollab(); } } } this.viewDispatch({ type: "page-loading", name: pageName, }); // Fetch next page to open let doc; try { doc = await this.space.readPage(pageName); } catch (e: any) { // Not found, new page console.log("Creating new page", pageName); doc = { text: "", meta: { name: pageName, lastModified: 0, perm: "rw" } as PageMeta, }; } const editorState = this.createEditorState( pageName, doc.text, doc.meta.perm === "ro", ); editorView.setState(editorState); if (editorView.contentDOM) { this.tweakEditorDOM(editorView.contentDOM); } const stateRestored = this.restoreState(pageName); this.space.watchPage(pageName); this.viewDispatch({ type: "page-loaded", meta: doc.meta, }); // Note: these events are dispatched asynchronously deliberately (not waiting for results) if (loadingDifferentPage) { this.eventHook.dispatchEvent("editor:pageLoaded", pageName).catch( console.error, ); } else { this.eventHook.dispatchEvent("editor:pageReloaded", pageName).catch( console.error, ); } return stateRestored; } tweakEditorDOM(contentDOM: HTMLElement) { contentDOM.spellcheck = true; contentDOM.setAttribute("autocorrect", "on"); contentDOM.setAttribute("autocapitalize", "on"); } async loadCustomStyles() { try { const { text: stylesText } = await this.space.readPage("STYLES"); const cssBlockRegex = /```css([^`]+)```/; const match = cssBlockRegex.exec(stylesText); if (!match) { return; } const css = match[1]; document.getElementById("custom-styles")!.innerHTML = css; } catch { // Nuthin' } } private restoreState(pageName: string): boolean { const pageState = this.openPages.get(pageName); const editorView = this.editorView!; if (pageState) { // Restore state editorView.scrollDOM.scrollTop = pageState!.scrollTop; try { editorView.dispatch({ selection: pageState.selection, scrollIntoView: true, }); } catch { // This is fine, just go to the top editorView.dispatch({ selection: { anchor: 0 }, scrollIntoView: true, }); } } else { editorView.scrollDOM.scrollTop = 0; editorView.dispatch({ selection: { anchor: 0 }, scrollIntoView: true, }); } editorView.focus(); return !!pageState; } private saveState(currentPage: string) { this.openPages.set( currentPage, new PageState( this.editorView!.scrollDOM.scrollTop, this.editorView!.state.selection, ), ); } ViewComponent() { const [viewState, dispatch] = useReducer(reducer, initialViewState); this.viewState = viewState; this.viewDispatch = dispatch; // deno-lint-ignore no-this-alias const editor = this; useEffect(() => { if (viewState.currentPage) { document.title = viewState.currentPage; } }, [viewState.currentPage]); useEffect(() => { if (editor.editorView) { editor.tweakEditorDOM( editor.editorView.contentDOM, ); } }, [viewState.uiOptions.forcedROMode]); useEffect(() => { this.rebuildEditorState(); this.dispatchAppEvent("editor:modeswitch"); }, [viewState.uiOptions.vimMode]); useEffect(() => { document.documentElement.dataset.theme = viewState.uiOptions.darkMode ? "dark" : "light"; }, [viewState.uiOptions.darkMode]); useEffect(() => { // Need to dispatch a resize event so that the top_bar can pick it up globalThis.dispatchEvent(new Event("resize")); }, [viewState.panels]); return ( <> {viewState.showPageNavigator && ( { dispatch({ type: "stop-navigate" }); setTimeout(() => { editor.focus(); }); if (page) { safeRun(async () => { await editor.navigate(page); }); } }} /> )} {viewState.showCommandPalette && ( { dispatch({ type: "hide-palette" }); setTimeout(() => { editor.focus(); }); if (cmd) { dispatch({ type: "command-run", command: cmd.command.name }); cmd .run() .catch((e: any) => { console.error("Error running command", e.message); }) .then(() => { // Always be focusing the editor after running a command editor.focus(); }); } }} commands={this.getCommandsByContext(viewState)} vimMode={viewState.uiOptions.vimMode} darkMode={viewState.uiOptions.darkMode} completer={this.miniEditorComplete.bind(this)} recentCommands={viewState.recentCommands} /> )} {viewState.showFilterBox && ( )} {viewState.showPrompt && ( { dispatch({ type: "hide-prompt" }); viewState.promptCallback!(value); }} /> )} {viewState.showConfirm && ( { dispatch({ type: "hide-confirm" }); viewState.confirmCallback!(value); }} /> )} { if (!newName) { // Always move cursor to the start of the page editor.editorView?.dispatch({ selection: { anchor: 0 }, }); editor.focus(); return; } console.log("Now renaming page to...", newName); await editor.system.loadedPlugs.get("core")!.invoke( "renamePage", [{ page: newName }], ); editor.focus(); }} actionButtons={[ { icon: HomeIcon, description: `Go home (Alt-h)`, callback: () => { editor.navigate(""); }, }, { icon: BookIcon, description: `Open page (${isMacLike() ? "Cmd-k" : "Ctrl-k"})`, callback: () => { dispatch({ type: "start-navigate" }); this.space.updatePageList(); }, }, { icon: TerminalIcon, description: `Run command (${isMacLike() ? "Cmd-/" : "Ctrl-/"})`, callback: () => { dispatch({ type: "show-palette", context: this.getContext() }); }, }, ]} rhs={!!viewState.panels.rhs.mode && (
)} lhs={!!viewState.panels.lhs.mode && (
)} />
{!!viewState.panels.lhs.mode && ( )}
{!!viewState.panels.rhs.mode && ( )}
{!!viewState.panels.modal.mode && (
)} {!!viewState.panels.bhs.mode && (
)} ); } async runCommandByName(name: string, ...args: any[]) { const cmd = this.viewState.commands.get(name); if (cmd) { await cmd.run(); } else { throw new Error(`Command ${name} not found`); } } render(container: Element) { const ViewComponent = this.ViewComponent.bind(this); preactRender(, container); } private getCommandsByContext( state: AppViewState, ): Map { const commands = new Map(state.commands); for (const [k, v] of state.commands.entries()) { if ( v.command.contexts && (!state.showCommandPaletteContext || !v.command.contexts.includes(state.showCommandPaletteContext)) ) { commands.delete(k); } } return commands; } private getContext(): string | undefined { const state = this.editorView!.state; const selection = state.selection.main; if (selection.empty) { return syntaxTree(state).resolveInner(selection.from).type.name; } return; } startCollab(serverUrl: string, token: string, username: string) { if (this.collabState) { // Clean up old collab state this.collabState.stop(); } const initialText = this.editorView!.state.sliceDoc(); this.collabState = new CollabState( serverUrl, this.currentPage!, token, username, this.syncService, ); this.rebuildEditorState(); // Don't watch for local changes in this mode this.space.unwatch(); } stopCollab() { if (this.collabState) { this.collabState.stop(); this.collabState = undefined; this.rebuildEditorState(); } // Start file watching again this.space.watch(); } }