silverbullet/web/client.ts

1282 lines
37 KiB
TypeScript
Raw Normal View History

import type {
CompletionContext,
CompletionResult,
} from "@codemirror/autocomplete";
import type { Compartment } from "@codemirror/state";
import { EditorView } from "@codemirror/view";
import { syntaxTree } from "@codemirror/language";
import { compile as gitIgnoreCompiler } from "gitignore-parser";
2024-07-30 23:33:33 +08:00
import type { SyntaxNode } from "@lezer/common";
import { Space } from "../common/space.ts";
import type { FilterOption } from "@silverbulletmd/silverbullet/type/client";
import { EventHook } from "../common/hooks/event.ts";
2024-07-30 23:33:33 +08:00
import type { AppCommand } from "$lib/command.ts";
2024-01-24 21:44:39 +08:00
import {
2024-07-30 23:33:33 +08:00
type PageState,
2024-01-24 21:44:39 +08:00
parsePageRefFromURI,
PathPageNavigator,
} from "./navigator.ts";
2023-07-14 19:58:16 +08:00
import type { AppViewState } from "./type.ts";
2024-02-29 22:23:05 +08:00
import type {
AppEvent,
CompleteEvent,
SlashCompletions,
} from "../plug-api/types.ts";
2024-07-30 23:33:33 +08:00
import type { StyleObject } from "../plugs/index/style.ts";
import { throttle } from "$lib/async.ts";
import { PlugSpacePrimitives } from "$common/spaces/plug_space_primitives.ts";
import { EventedSpacePrimitives } from "$common/spaces/evented_space_primitives.ts";
2023-08-30 03:17:29 +08:00
import {
2024-07-30 23:33:33 +08:00
type ISyncService,
2023-08-30 03:17:29 +08:00
NoSyncSyncService,
pageSyncInterval,
SyncService,
} from "./sync_service.ts";
import { simpleHash } from "$lib/crypto.ts";
2024-07-30 23:33:33 +08:00
import type { 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 { FilteredSpacePrimitives } from "$common/spaces/filtered_space_primitives.ts";
import {
encodePageRef,
validatePageName,
} from "@silverbulletmd/silverbullet/lib/page_ref";
2023-07-14 19:44:30 +08:00
import { ClientSystem } from "./client_system.ts";
2023-07-14 19:58:16 +08:00
import { createEditorState } from "./editor_state.ts";
2023-07-14 20:22:26 +08:00
import { MainUI } from "./editor_ui.tsx";
import { cleanPageRef } from "@silverbulletmd/silverbullet/lib/resolve";
2024-07-30 23:33:33 +08:00
import type { SpacePrimitives } from "$common/spaces/space_primitives.ts";
import type {
CodeWidgetButton,
FileMeta,
PageMeta,
} from "../plug-api/types.ts";
import { DataStore } from "$lib/data/datastore.ts";
import { IndexedDBKvPrimitives } from "$lib/data/indexeddb_kv_primitives.ts";
import { DataStoreMQ } from "$lib/data/mq.datastore.ts";
import { DataStoreSpacePrimitives } from "$common/spaces/datastore_space_primitives.ts";
2024-02-28 23:22:28 +08:00
import { ensureSpaceIndex } from "$common/space_index.ts";
import { renderTheTemplate } from "$common/syscalls/template.ts";
2024-07-30 23:33:33 +08:00
import type { PageRef } from "../plug-api/lib/page_ref.ts";
import { ReadOnlySpacePrimitives } from "$common/spaces/ro_space_primitives.ts";
2024-07-30 23:33:33 +08:00
import type { KvPrimitives } from "$lib/data/kv_primitives.ts";
import { builtinFunctions } from "$lib/builtin_query_functions.ts";
import {
ensureAndLoadSettingsAndIndex,
updateObjectDecorators,
} from "../common/config.ts";
import { LimitedMap } from "$lib/limited_map.ts";
2024-05-28 02:33:41 +08:00
import { plugPrefix } from "$common/spaces/constants.ts";
2024-07-07 02:29:38 +08:00
import { lezerToParseTree } from "$common/markdown_parser/parse_tree.ts";
import { findNodeMatching } from "@silverbulletmd/silverbullet/lib/tree";
2024-07-25 21:18:58 +08:00
import type { LinkObject } from "../plugs/index/page_links.ts";
2024-08-15 22:39:06 +08:00
import type { Config, ConfigContainer } from "../type/config.ts";
2024-05-28 02:33:41 +08:00
2023-07-14 19:44:30 +08:00
const frontMatterRegex = /^---\n(([^\n]|\n)*?)---\n/;
2023-07-14 19:44:30 +08:00
const autoSaveInterval = 1000;
declare global {
2024-07-30 21:17:34 +08:00
// deno-lint-ignore no-var
var silverBulletConfig: {
spaceFolderPath: string;
syncOnly: boolean;
readOnly: boolean;
enableSpaceScript: boolean;
};
// deno-lint-ignore no-var
var client: Client;
}
2024-07-26 02:29:13 +08:00
type WidgetCacheItem = {
height: number;
html: string;
buttons?: CodeWidgetButton[];
banner?: string;
};
export class Client implements ConfigContainer {
2024-07-26 02:29:13 +08:00
// Event bus used to communicate between components
eventHook = new EventHook();
space!: Space;
config!: Config;
2024-07-26 02:29:13 +08:00
clientSystem!: ClientSystem;
2024-07-26 02:29:13 +08:00
plugSpaceRemotePrimitives!: PlugSpacePrimitives;
httpSpacePrimitives!: HttpSpacePrimitives;
ui!: MainUI;
stateDataStore!: DataStore;
spaceKV?: KvPrimitives;
mq!: DataStoreMQ;
// CodeMirror editor
editorView!: EditorView;
keyHandlerCompartment?: Compartment;
2023-07-27 17:41:44 +08:00
private pageNavigator!: PathPageNavigator;
2023-07-27 17:41:44 +08:00
private dbPrefix: string;
2022-12-22 23:20:05 +08:00
2023-07-27 17:41:44 +08:00
saveTimeout?: number;
2022-04-04 21:25:07 +08:00
debouncedUpdateEvent = throttle(() => {
this.eventHook
.dispatchEvent("editor:updated")
.catch((e) => console.error("Error dispatching editor:updated event", e));
2022-04-04 21:25:07 +08:00
}, 1000);
2024-07-26 02:29:13 +08:00
// Sync related stuff
// Track if plugs have been updated since sync cycle
fullSyncCompleted = false;
syncService!: ISyncService;
2024-07-26 02:29:13 +08:00
private onLoadPageRef: PageRef;
2023-12-22 22:55:50 +08:00
2022-08-02 18:43:39 +08:00
constructor(
private parent: Element,
public syncMode: boolean,
private readOnlyMode: boolean,
2022-08-02 18:43:39 +08:00
) {
if (!syncMode) {
this.fullSyncCompleted = true;
}
// Generate a semi-unique prefix for the database so not to reuse databases for different space paths
2023-07-27 17:41:44 +08:00
this.dbPrefix = "" + simpleHash(window.silverBulletConfig.spaceFolderPath);
2024-01-24 21:44:39 +08:00
this.onLoadPageRef = parsePageRefFromURI();
}
/**
* Initialize the client
* This is a separated from the constructor to allow for async initialization
*/
async init() {
2024-07-26 02:29:13 +08:00
// Setup the state data store
const stateKvPrimitives = new IndexedDBKvPrimitives(
`${this.dbPrefix}_state`,
);
await stateKvPrimitives.init();
this.stateDataStore = new DataStore(stateKvPrimitives);
// Setup message queue
this.mq = new DataStoreMQ(this.stateDataStore);
2023-08-11 00:32:41 +08:00
2023-07-14 19:44:30 +08:00
// Instantiate a PlugOS system
this.clientSystem = new ClientSystem(
2023-07-14 19:44:30 +08:00
this,
2023-08-11 00:32:41 +08:00
this.mq,
this.stateDataStore,
2023-07-14 19:44:30 +08:00
this.eventHook,
this.readOnlyMode,
2023-07-14 19:44:30 +08:00
);
const localSpacePrimitives = await this.initSpace();
2023-08-30 03:17:29 +08:00
this.syncService = this.syncMode
? new SyncService(
localSpacePrimitives,
this.plugSpaceRemotePrimitives,
this.stateDataStore,
2023-08-30 03:17:29 +08:00
this.eventHook,
2024-07-26 02:29:13 +08:00
(path) => { // isSyncCandidate
// Exclude all plug space primitives paths
return !this.plugSpaceRemotePrimitives.isLikelyHandled(path) ||
2023-08-30 03:17:29 +08:00
// Except federated ones
path.startsWith("!");
},
)
: new NoSyncSyncService(this.space);
2023-07-14 20:22:26 +08:00
this.ui = new MainUI(this);
this.ui.render(this.parent);
2022-10-26 00:50:07 +08:00
this.editorView = new EditorView({
2023-07-14 19:58:16 +08:00
state: createEditorState(this, "", "", false),
2022-07-22 19:44:28 +08:00
parent: document.getElementById("sb-editor")!,
});
this.focus();
this.clientSystem.init();
2022-11-24 19:04:00 +08:00
await this.loadCaches();
2024-07-26 02:29:13 +08:00
// Let's ping the remote space to ensure we're authenticated properly, if not will result in a redirect to auth page
try {
await this.httpSpacePrimitives.ping();
2023-07-27 18:37:39 +08:00
} catch (e: any) {
if (e.message === "Not authenticated") {
console.warn("Not authenticated, redirecting to auth page");
return;
}
2023-11-25 21:30:31 +08:00
if (e.message.includes("Offline") && !this.syncMode) {
// Offline and not in sync mode, this is not going to fly.
this.flashNotification(
"Could not reach remote server, going to reload in a few seconds",
"error",
);
setTimeout(() => {
location.reload();
}, 5000);
throw e;
}
2023-07-27 17:41:44 +08:00
console.warn(
2023-07-27 18:37:39 +08:00
"Could not reach remote server, we're offline or the server is down",
e,
);
}
// Load plugs
2023-07-27 21:25:33 +08:00
await this.loadPlugs();
// Load config (after the plugs, specifically the 'index' plug is loaded)
await this.loadConfig();
await this.clientSystem.loadSpaceScripts();
await this.initNavigator();
await this.initSync();
2022-11-24 19:04:00 +08:00
2024-07-26 02:29:13 +08:00
// We can load custom styles async
2023-07-27 17:41:44 +08:00
this.loadCustomStyles().catch(console.error);
2023-07-27 21:25:33 +08:00
await this.dispatchAppEvent("editor:init");
2023-08-08 02:42:52 +08:00
2024-07-26 02:29:13 +08:00
// Regularly sync the currently open file
2023-08-08 02:42:52 +08:00
setInterval(() => {
2023-08-10 00:07:01 +08:00
try {
2024-01-24 21:44:39 +08:00
this.syncService.syncFile(`${this.currentPage}.md`).catch((e: any) => {
2023-08-10 00:07:01 +08:00
console.error("Interval sync error", e);
});
} catch (e: any) {
2023-08-09 23:22:42 +08:00
console.error("Interval sync error", e);
2023-08-10 00:07:01 +08:00
}
2023-08-08 02:42:52 +08:00
}, pageSyncInterval);
2023-12-22 22:55:50 +08:00
2024-07-26 02:29:13 +08:00
// Let's update the local page list cache asynchronously
2023-12-22 22:55:50 +08:00
this.updatePageListCache().catch(console.error);
2023-07-27 21:25:33 +08:00
}
async loadConfig() {
this.config = await ensureAndLoadSettingsAndIndex(
2024-01-25 18:42:36 +08:00
this.space.spacePrimitives,
this.clientSystem.system,
2024-01-25 18:42:36 +08:00
);
updateObjectDecorators(this.config, this.stateDataStore);
2024-01-25 18:42:36 +08:00
this.ui.viewDispatch({
type: "config-loaded",
config: this.config,
2024-01-25 18:42:36 +08:00
});
this.clientSystem.slashCommandHook.buildAllCommands(
this.clientSystem.system,
);
2024-08-04 02:59:53 +08:00
this.eventHook.dispatchEvent("config:loaded", this.config);
}
private async initSync() {
2023-07-27 17:41:44 +08:00
this.syncService.start();
// We're still booting, if a initial sync has already been completed we know this is the initial sync
2024-03-12 03:56:29 +08:00
let initialSync = !await this.syncService.hasInitialSyncCompleted();
2023-07-27 17:41:44 +08:00
this.eventHook.addLocalListener("sync:success", async (operations) => {
if (operations > 0) {
// Update the page list
2023-12-22 22:55:50 +08:00
await this.space.updatePageList();
2023-07-27 17:41:44 +08:00
}
if (operations !== undefined) {
// "sync:success" is called with a number of operations only from syncSpace(), not from syncing individual pages
this.fullSyncCompleted = true;
console.log("Full sync completed");
// A full sync just completed
if (!initialSync) {
// If this was NOT the initial sync let's check if we need to perform a space reindex
ensureSpaceIndex(this.stateDataStore, this.clientSystem.system).catch(
console.error,
);
2024-03-12 03:56:29 +08:00
} else { // initialSync
2024-07-26 02:29:13 +08:00
// Let's load space scripts again, which probably weren't loaded before
await this.clientSystem.loadSpaceScripts();
2024-02-28 17:55:25 +08:00
console.log(
"Initial sync completed, now need to do a full space index to ensure all pages are indexed using any custom space script indexers",
);
ensureSpaceIndex(this.stateDataStore, this.clientSystem.system).catch(
console.error,
);
2024-03-12 03:56:29 +08:00
initialSync = false;
}
2023-07-27 17:41:44 +08:00
}
if (operations) {
// Likely initial sync so let's show visually that we're synced now
this.showProgress(100);
2023-07-27 17:41:44 +08:00
}
2023-08-16 17:40:31 +08:00
this.ui.viewDispatch({ type: "sync-change", syncSuccess: true });
2023-07-27 17:41:44 +08:00
});
2024-07-26 02:29:13 +08:00
2023-07-27 17:41:44 +08:00
this.eventHook.addLocalListener("sync:error", (_name) => {
2023-08-16 17:40:31 +08:00
this.ui.viewDispatch({ type: "sync-change", syncSuccess: false });
2023-07-27 17:41:44 +08:00
});
2024-07-26 02:29:13 +08:00
2023-07-27 17:41:44 +08:00
this.eventHook.addLocalListener("sync:conflict", (name) => {
this.flashNotification(
`Sync: conflict detected for ${name} - conflict copy created`,
"error",
);
});
2024-07-26 02:29:13 +08:00
2023-07-27 17:41:44 +08:00
this.eventHook.addLocalListener("sync:progress", (status: SyncStatus) => {
this.showProgress(
Math.round(status.filesProcessed / status.totalFiles * 100),
);
});
2024-07-26 02:29:13 +08:00
2023-08-30 03:17:29 +08:00
this.eventHook.addLocalListener(
"file:synced",
(meta: FileMeta, direction: string) => {
2023-08-08 02:42:52 +08:00
if (meta.name.endsWith(".md") && direction === "secondary->primary") {
// We likely polled the currently open page which trigggered a local update, let's update the editor accordingly
this.space.getPageMeta(meta.name.slice(0, -3));
}
},
2023-08-30 03:17:29 +08:00
);
2023-07-27 17:41:44 +08:00
}
private navigateWithinPage(pageState: PageState) {
// Did we end up doing anything in terms of internal navigation?
let adjustedPosition = false;
// Was a particular scroll position persisted?
2024-01-24 21:44:39 +08:00
if (
pageState.scrollTop !== undefined &&
!(pageState.scrollTop === 0 &&
2024-01-25 21:51:40 +08:00
(pageState.pos !== undefined || pageState.anchor !== undefined ||
pageState.header !== undefined))
2024-01-24 21:44:39 +08:00
) {
setTimeout(() => {
this.editorView.scrollDOM.scrollTop = pageState.scrollTop!;
});
adjustedPosition = true;
}
// Was a particular cursor/selection set?
2024-01-25 21:51:40 +08:00
if (
pageState.selection?.anchor && !pageState.pos && !pageState.anchor &&
!pageState.header
) { // Only do this if we got a specific cursor position
console.log("Changing cursor position to", pageState.selection);
this.editorView.dispatch({
selection: pageState.selection,
});
adjustedPosition = true;
}
// Was there a pos or anchor set?
let pos: number | { line: number; column: number } | undefined =
pageState.pos;
if (pageState.anchor) {
console.log("Navigating to anchor", pageState.anchor);
const pageText = this.editorView.state.sliceDoc();
2024-07-07 02:29:38 +08:00
const sTree = syntaxTree(this.editorView.state);
const tree = lezerToParseTree(pageText, sTree.topNode);
2024-07-07 02:29:38 +08:00
const foundNode = findNodeMatching(tree, (node) => {
if (
node.type === "NamedAnchor" &&
node.children![0].text === `$${pageState.anchor}`
) {
return true;
}
return false;
});
if (!foundNode) {
return this.flashNotification(
`Could not find anchor $${pageState.anchor}`,
"error",
);
2024-07-07 02:29:38 +08:00
} else {
pos = foundNode.from;
}
adjustedPosition = true;
}
2024-01-25 21:51:40 +08:00
if (pageState.header) {
console.log("Navigating to header", pageState.header);
const pageText = this.editorView.state.sliceDoc();
// This is somewhat of a simplistic way to find the header, but it works for now
pos = pageText.indexOf(`# ${pageState.header}\n`) + 2;
if (pos === -1) {
return this.flashNotification(
`Could not find header "${pageState.header}"`,
"error",
);
}
adjustedPosition = true;
}
if (pos !== undefined) {
// Translate line and column number to position in text
if (pos instanceof Object) {
// CodeMirror already keeps information about lines
const cmLine = this.editorView.state.doc.line(pos.line);
// How much to move inside the line, column number starts from 1
const offset = Math.max(0, Math.min(cmLine.length, pos.column - 1));
pos = cmLine.from + offset;
}
this.editorView.dispatch({
selection: { anchor: pos! },
effects: EditorView.scrollIntoView(pos!, {
y: "start",
yMargin: 5,
}),
});
adjustedPosition = true;
}
// If not: just put the cursor at the top of the page, right after the frontmatter
if (!adjustedPosition) {
// 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,
});
}
}
private async initNavigator() {
this.pageNavigator = new PathPageNavigator(this);
2023-07-27 17:41:44 +08:00
await this.pageNavigator.init();
this.pageNavigator.subscribe(async (pageState) => {
console.log("Now navigating to", pageState);
await this.loadPage(pageState.page);
// Setup scroll position, cursor position, etc
this.navigateWithinPage(pageState);
2024-07-26 02:29:13 +08:00
// Persist this page as the last opened page, we'll use this for cold start PWA loads
await this.stateDataStore.set(
["client", "lastOpenedPage"],
pageState.page,
);
});
2023-08-16 17:40:31 +08:00
if (location.hash === "#boot" && this.config.pwaOpenLastPage !== false) {
2024-07-26 02:29:13 +08:00
// Cold start PWA load
const lastPage = await this.stateDataStore.get([
"client",
"lastOpenedPage",
]);
if (lastPage) {
console.log("Navigating to last opened page", lastPage);
await this.navigate({ page: lastPage });
}
2023-08-16 17:40:31 +08:00
}
2023-07-27 17:41:44 +08:00
}
async initSpace(): Promise<SpacePrimitives> {
this.httpSpacePrimitives = new HttpSpacePrimitives(
2023-07-27 17:41:44 +08:00
location.origin,
window.silverBulletConfig.spaceFolderPath,
);
2023-01-22 22:48:12 +08:00
let remoteSpacePrimitives: SpacePrimitives = this.httpSpacePrimitives;
if (this.readOnlyMode) {
remoteSpacePrimitives = new ReadOnlySpacePrimitives(
remoteSpacePrimitives,
);
}
2023-07-27 17:41:44 +08:00
this.plugSpaceRemotePrimitives = new PlugSpacePrimitives(
remoteSpacePrimitives,
this.clientSystem.namespaceHook,
this.syncMode ? undefined : "client",
2023-07-27 17:41:44 +08:00
);
2023-07-27 17:41:44 +08:00
let fileFilterFn: (s: string) => boolean = () => true;
2023-08-26 14:31:51 +08:00
let localSpacePrimitives: SpacePrimitives | undefined;
2023-08-30 03:17:29 +08:00
if (this.syncMode) {
// We'll store the space files in a separate data store
const spaceKvPrimitives = new IndexedDBKvPrimitives(
`${this.dbPrefix}_synced_space`,
);
await spaceKvPrimitives.init();
this.spaceKV = spaceKvPrimitives;
2023-08-26 14:31:51 +08:00
localSpacePrimitives = new FilteredSpacePrimitives(
new EventedSpacePrimitives(
// 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 DataStoreSpacePrimitives(
new DataStore(
spaceKvPrimitives,
{},
),
),
this.plugSpaceRemotePrimitives,
2023-07-27 17:41:44 +08:00
),
this.eventHook,
2023-07-27 17:41:44 +08:00
),
2023-08-26 14:31:51 +08:00
(meta) => fileFilterFn(meta.name),
// Run when a list of files has been retrieved
async () => {
if (!this.config) {
await this.loadConfig();
}
if (typeof this.config?.spaceIgnore === "string") {
fileFilterFn = gitIgnoreCompiler(this.config.spaceIgnore).accepts;
2023-08-26 14:31:51 +08:00
} else {
fileFilterFn = () => true;
}
},
);
} else {
2023-08-27 17:02:24 +08:00
localSpacePrimitives = new EventedSpacePrimitives(
this.plugSpaceRemotePrimitives,
this.eventHook,
);
2023-08-26 14:31:51 +08:00
}
this.space = new Space(
localSpacePrimitives,
this.eventHook,
);
2023-07-27 17:41:44 +08:00
2024-03-14 20:02:34 +08:00
let lastSaveTimestamp: number | undefined;
this.eventHook.addLocalListener("editor:pageSaving", () => {
lastSaveTimestamp = Date.now();
});
2023-11-12 17:33:27 +08:00
this.eventHook.addLocalListener(
"file:changed",
(
path: string,
_localChange: boolean,
2024-03-14 20:02:34 +08:00
oldHash: number,
newHash: number,
2023-11-12 17:33:27 +08:00
) => {
// Only reload when watching the current page (to avoid reloading when switching pages)
if (
2024-03-14 20:02:34 +08:00
this.space.watchInterval && `${this.currentPage}.md` === path &&
// Avoid reloading if the page was just saved (5s window)
(!lastSaveTimestamp || (lastSaveTimestamp < Date.now() - 5000))
2023-11-12 17:33:27 +08:00
) {
console.log(
"Page changed elsewhere, reloading. Old hash",
oldHash,
"new hash",
newHash,
);
2024-03-14 20:02:34 +08:00
console.log(
"Last save timestamp",
lastSaveTimestamp,
"now",
Date.now(),
);
2023-11-12 17:33:27 +08:00
this.flashNotification("Page changed elsewhere, reloading");
this.reloadPage();
}
},
);
2023-08-27 17:02:24 +08:00
2024-05-28 02:33:41 +08:00
// Caching a list of known files for the wiki_link highlighter (that checks if a file exists)
2024-07-26 02:29:13 +08:00
// And keeping it up to date as we go
2024-05-28 02:33:41 +08:00
this.eventHook.addLocalListener("file:changed", (fileName: string) => {
// Make sure this file is in the list of known pages
this.clientSystem.allKnownFiles.add(fileName);
2023-12-22 22:55:50 +08:00
});
2024-05-28 02:33:41 +08:00
this.eventHook.addLocalListener("file:deleted", (fileName: string) => {
this.clientSystem.allKnownFiles.delete(fileName);
2023-12-22 22:55:50 +08:00
});
this.eventHook.addLocalListener(
"file:listed",
(allFiles: FileMeta[]) => {
// Update list of known pages
2024-05-28 02:33:41 +08:00
this.clientSystem.allKnownFiles.clear();
allFiles.forEach((f) => {
2024-05-28 02:33:41 +08:00
if (!f.name.startsWith(plugPrefix)) {
this.clientSystem.allKnownFiles.add(f.name);
}
});
2023-12-22 22:55:50 +08:00
},
);
2023-07-27 17:41:44 +08:00
this.space.watch();
2023-08-26 14:31:51 +08:00
return localSpacePrimitives;
}
2024-01-24 21:44:39 +08:00
get currentPage(): string {
return this.ui.viewState.currentPage !== undefined
? this.ui.viewState.currentPage
: this.onLoadPageRef.page; // best effort
2023-07-27 17:41:44 +08:00
}
2023-07-27 21:25:33 +08:00
dispatchAppEvent(name: AppEvent, ...args: any[]): Promise<any[]> {
return this.eventHook.dispatchEvent(name, ...args);
}
2024-07-26 02:29:13 +08:00
// Save the current page
save(immediate = false): Promise<void> {
return new Promise((resolve, reject) => {
if (this.saveTimeout) {
clearTimeout(this.saveTimeout);
}
this.saveTimeout = setTimeout(
() => {
if (this.currentPage) {
if (
2023-07-14 20:22:26 +08:00
!this.ui.viewState.unsavedChanges ||
this.ui.viewState.uiOptions.forcedROMode
) {
2022-10-23 02:23:54 +08:00
// No unsaved changes, or read-only mode, not gonna save
2022-10-22 01:02:00 +08:00
return resolve();
}
console.log("Saving page", this.currentPage);
2024-03-02 22:21:36 +08:00
this.dispatchAppEvent(
"editor:pageSaving",
this.currentPage,
);
this.space
.writePage(
this.currentPage,
2023-07-27 17:41:44 +08:00
this.editorView.state.sliceDoc(0),
true,
)
.then(async (meta) => {
2023-07-14 20:22:26 +08:00
this.ui.viewDispatch({ type: "page-saved" });
await this.dispatchAppEvent(
"editor:pageSaved",
this.currentPage,
meta,
);
2024-07-14 17:29:43 +08:00
// At this all the essential stuff is done, let's proceed
resolve();
// In the background we'll fetch any enriched meta data, if any
2024-07-13 20:55:35 +08:00
const enrichedMeta = await this.clientSystem.getObjectByRef<
PageMeta
>(
this.currentPage,
"page",
this.currentPage,
);
2024-07-14 17:29:43 +08:00
if (enrichedMeta) {
this.ui.viewDispatch({
type: "update-current-page-meta",
meta: enrichedMeta,
});
}
})
2022-07-19 23:21:11 +08:00
.catch((e) => {
this.flashNotification(
"Could not save page, retrying again in 10 seconds",
"error",
2022-07-19 23:21:11 +08:00
);
this.saveTimeout = setTimeout(this.save.bind(this), 10000);
reject(e);
});
} else {
resolve();
}
},
2023-07-14 19:44:30 +08:00
immediate ? 0 : autoSaveInterval,
);
});
}
flashNotification(message: string, type: "info" | "error" = "info") {
const id = Math.floor(Math.random() * 1000000);
2023-07-14 20:22:26 +08:00
this.ui.viewDispatch({
type: "show-notification",
notification: {
id,
type,
message,
date: new Date(),
},
});
setTimeout(
() => {
2023-07-14 20:22:26 +08:00
this.ui.viewDispatch({
type: "dismiss-notification",
id: id,
});
},
type === "info" ? 4000 : 5000,
);
}
startPageNavigate(mode: "page" | "meta" | "all") {
2023-12-22 01:37:50 +08:00
// Then show the page navigator
this.ui.viewDispatch({ type: "start-navigate", mode });
2024-07-26 02:29:13 +08:00
// And update the page list cache asynchronously
2023-12-22 22:55:50 +08:00
this.updatePageListCache().catch(console.error);
}
async updatePageListCache() {
console.log("Updating page list cache");
if (!this.clientSystem.system.loadedPlugs.has("index")) {
console.warn("Index plug not loaded, cannot update page list cache");
return;
}
const allPages = await this.clientSystem.queryObjects<PageMeta>("page", {});
2024-07-25 21:18:58 +08:00
const allBrokenLinkPages = (await this.clientSystem.queryObjects<
LinkObject
>("link", {
filter: ["and", ["attr", "toPage"], ["not", ["call", "pageExists", [[
"attr",
"toPage",
]]]]],
select: [{ name: "toPage" }],
})).map((link): PageMeta => ({
ref: link.toPage!,
tag: "page",
_isBrokenLink: true,
name: link.toPage!,
created: "",
lastModified: "",
perm: "rw",
}));
2023-12-22 22:55:50 +08:00
this.ui.viewDispatch({
type: "update-page-list",
2024-07-25 21:18:58 +08:00
allPages: allPages.concat(allBrokenLinkPages),
2023-12-22 22:55:50 +08:00
});
2023-12-22 01:37:50 +08:00
}
2024-07-26 02:29:13 +08:00
// Progress circle handling
2023-07-27 23:02:53 +08:00
private progressTimeout?: number;
2024-07-26 02:29:13 +08:00
2023-06-15 02:58:08 +08:00
showProgress(progressPerc: number) {
2023-07-14 20:22:26 +08:00
this.ui.viewDispatch({
2023-06-15 02:58:08 +08:00
type: "set-progress",
progressPerc,
});
if (this.progressTimeout) {
clearTimeout(this.progressTimeout);
}
this.progressTimeout = setTimeout(
() => {
2023-07-14 20:22:26 +08:00
this.ui.viewDispatch({
2023-06-15 02:58:08 +08:00
type: "set-progress",
});
},
10000,
);
}
2024-07-26 02:29:13 +08:00
// Various UI elements
filterBox(
label: string,
options: FilterOption[],
helpText = "",
placeHolder = "",
): Promise<FilterOption | undefined> {
return new Promise((resolve) => {
2023-07-14 20:22:26 +08:00
this.ui.viewDispatch({
type: "show-filterbox",
label,
options,
placeHolder,
helpText,
onSelect: (option: any) => {
2023-07-14 20:22:26 +08:00
this.ui.viewDispatch({ type: "hide-filterbox" });
this.focus();
resolve(option);
},
});
});
}
2022-12-21 23:08:51 +08:00
prompt(
message: string,
defaultValue = "",
): Promise<string | undefined> {
return new Promise((resolve) => {
2023-07-14 20:22:26 +08:00
this.ui.viewDispatch({
2022-12-21 23:08:51 +08:00
type: "show-prompt",
message,
defaultValue,
callback: (value: string | undefined) => {
2023-07-14 20:22:26 +08:00
this.ui.viewDispatch({ type: "hide-prompt" });
2022-12-21 23:08:51 +08:00
this.focus();
resolve(value);
},
});
});
}
confirm(
message: string,
): Promise<boolean> {
return new Promise((resolve) => {
2023-07-14 20:22:26 +08:00
this.ui.viewDispatch({
2022-12-21 23:08:51 +08:00
type: "show-confirm",
message,
callback: (value: boolean) => {
2023-07-14 20:22:26 +08:00
this.ui.viewDispatch({ type: "hide-confirm" });
2022-12-21 23:08:51 +08:00
this.focus();
resolve(value);
},
});
});
}
2023-07-27 21:25:33 +08:00
async loadPlugs() {
await this.clientSystem.reloadPlugsFromSpace(this.space);
await this.eventHook.dispatchEvent("system:ready");
2022-07-11 15:08:22 +08:00
await this.dispatchAppEvent("plugs:loaded");
2022-04-27 01:04:36 +08:00
}
rebuildEditorState() {
const editorView = this.editorView;
console.log("Rebuilding editor state");
2023-07-27 21:25:33 +08:00
if (this.currentPage) {
2022-03-31 23:25:34 +08:00
editorView.setState(
2023-07-14 19:58:16 +08:00
createEditorState(
this,
2023-01-16 18:28:59 +08:00
this.currentPage,
editorView.state.sliceDoc(),
2023-07-14 20:22:26 +08:00
this.ui.viewState.currentPageMeta?.perm === "ro",
2023-01-16 18:28:59 +08:00
),
2022-03-31 23:25:34 +08:00
);
if (editorView.contentDOM) {
2022-05-17 17:53:17 +08:00
this.tweakEditorDOM(
editorView.contentDOM,
);
}
2022-03-31 23:25:34 +08:00
}
}
// Code completion support
2023-11-06 16:14:16 +08:00
async completeWithEvent(
context: CompletionContext,
eventName: AppEvent,
): Promise<CompletionResult | SlashCompletions | null> {
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);
2024-07-26 02:29:13 +08:00
// Build up list of parent nodes, some completions need this
2023-08-02 03:35:19 +08:00
const parentNodes: string[] = [];
const sTree = syntaxTree(editorState);
const currentNode = sTree.resolveInner(selection.from);
2023-08-02 03:35:19 +08:00
if (currentNode) {
let node: SyntaxNode | null = currentNode;
do {
2023-12-22 01:37:50 +08:00
if (node.name === "FencedCode" || node.name === "FrontMatter") {
2023-10-04 23:14:24 +08:00
const body = editorState.sliceDoc(node.from + 3, node.to - 3);
2023-12-22 01:37:50 +08:00
parentNodes.push(`${node.name}:${body}`);
} else {
parentNodes.push(node.name);
}
2023-08-02 03:35:19 +08:00
node = node.parent;
} while (node);
2023-08-02 03:35:19 +08:00
}
2024-07-26 02:29:13 +08:00
// Dispatch the event
const results = await this.dispatchAppEvent(eventName, {
2024-01-24 21:44:39 +08:00
pageName: this.currentPage,
linePrefix,
pos: selection.from,
2023-08-02 03:35:19 +08:00
parentNodes,
} as CompleteEvent);
2024-07-26 02:29:13 +08:00
// Merge results
2024-02-23 20:42:02 +08:00
let currentResult: CompletionResult | null = null;
for (const result of results) {
2024-02-23 20:42:02 +08:00
if (!result) {
continue;
}
if (currentResult) {
// Let's see if we can merge results
if (currentResult.from !== result.from) {
console.error(
"Got completion results from multiple sources with different `from` locators, cannot deal with that",
);
console.error(
2024-02-23 20:42:02 +08:00
"Previously had",
currentResult,
"now also got",
result,
);
return null;
2024-02-23 20:42:02 +08:00
} else {
// Merge
currentResult = {
from: result.from,
options: [...currentResult.options, ...result.options],
};
}
2024-02-23 20:42:02 +08:00
} else {
currentResult = result;
}
}
2024-02-23 20:42:02 +08:00
return currentResult;
}
editorComplete(
context: CompletionContext,
): Promise<CompletionResult | null> {
return this.completeWithEvent(context, "editor:complete") as Promise<
CompletionResult | null
>;
}
miniEditorComplete(
context: CompletionContext,
): Promise<CompletionResult | null> {
return this.completeWithEvent(context, "minieditor:complete") as Promise<
CompletionResult | null
>;
}
2022-10-11 00:19:08 +08:00
async reloadPage() {
console.log("Reloading page");
2022-10-11 00:19:08 +08:00
clearTimeout(this.saveTimeout);
2024-01-24 21:44:39 +08:00
await this.loadPage(this.currentPage);
}
2024-07-26 02:29:13 +08:00
// Focus the editor
focus() {
2023-07-24 15:36:33 +08:00
const viewState = this.ui.viewState;
if (
[
viewState.showCommandPalette,
viewState.showPageNavigator,
viewState.showFilterBox,
viewState.showConfirm,
viewState.showPrompt,
].some(Boolean)
) {
2024-02-28 19:16:51 +08:00
// console.log("not focusing");
2023-07-24 15:36:33 +08:00
// Some other modal UI element is visible, don't focus editor now
return;
}
2023-07-27 17:41:44 +08:00
this.editorView.focus();
}
async navigate(
pageRef: PageRef,
replaceState = false,
newWindow = false,
) {
if (!pageRef.page) {
pageRef.page = cleanPageRef(
await renderTheTemplate(
this.config.indexPage,
{},
{},
builtinFunctions,
),
);
2022-08-02 18:43:39 +08:00
}
try {
validatePageName(pageRef.page);
} catch (e: any) {
return this.flashNotification(e.message, "error");
}
if (newWindow) {
2024-01-24 21:44:39 +08:00
console.log(
"Navigating to new page in new window",
`${location.origin}/${encodeURIComponent(encodePageRef(pageRef))}`,
2024-01-24 21:44:39 +08:00
);
const win = globalThis.open(
`${location.origin}/${encodeURIComponent(encodePageRef(pageRef))}`,
"_blank",
);
if (win) {
win.focus();
}
return;
}
await this.pageNavigator!.navigate(
pageRef,
replaceState,
);
this.focus();
}
async loadPage(pageName: string) {
const loadingDifferentPage = pageName !== this.currentPage;
const editorView = this.editorView;
2022-09-06 22:33:00 +08:00
const previousPage = this.currentPage;
// Persist current page state and nicely close page
2022-09-06 22:33:00 +08:00
if (previousPage) {
// this.openPages.saveState(previousPage);
2022-09-06 22:33:00 +08:00
this.space.unwatchPage(previousPage);
2022-10-11 00:19:08 +08:00
if (previousPage !== pageName) {
await this.save(true);
}
}
2023-07-14 20:22:26 +08:00
this.ui.viewDispatch({
2022-09-12 20:50:37 +08:00
type: "page-loading",
name: pageName,
});
// Fetch next page to open
let doc;
try {
doc = await this.space.readPage(pageName);
} catch (e: any) {
2023-07-27 17:41:44 +08:00
if (e.message.includes("Not found")) {
// Not found, new page
console.log("Page doesn't exist, creating new page:", pageName);
// Initialize page
2023-07-27 17:41:44 +08:00
doc = {
text: "",
2023-11-06 16:14:16 +08:00
meta: {
ref: pageName,
tags: ["page"],
name: pageName,
lastModified: "",
created: "",
perm: "rw",
} as PageMeta,
2023-07-27 17:41:44 +08:00
};
2024-07-26 02:29:13 +08:00
// Create new page based on a template
this.clientSystem.system.invokeFunction("template.newPage", [pageName])
.then(
() => {
this.focus();
},
).catch(
console.error,
);
2023-07-27 17:41:44 +08:00
} else {
2023-10-06 00:24:12 +08:00
this.flashNotification(
`Could not load page ${pageName}: ${e.message}`,
"error",
);
if (previousPage) {
this.ui.viewDispatch({
type: "page-loading",
name: previousPage,
});
}
return false;
2023-07-27 17:41:44 +08:00
}
}
this.ui.viewDispatch({
type: "page-loaded",
meta: doc.meta,
});
2024-07-14 17:29:43 +08:00
// Fetch (possibly) enriched meta data asynchronously
2024-07-26 02:29:13 +08:00
this.clientSystem.getObjectByRef<
2024-07-14 17:29:43 +08:00
PageMeta
>(
this.currentPage,
"page",
this.currentPage,
).then((enrichedMeta) => {
if (!enrichedMeta) {
// Nothing in the store, revert to default
enrichedMeta = doc.meta;
}
const bodyEl = this.parent.parentElement;
if (bodyEl) {
bodyEl.removeAttribute("class");
2024-07-31 19:06:10 +08:00
if (enrichedMeta.pageDecoration?.cssClasses) {
bodyEl.className = enrichedMeta.pageDecoration.cssClasses.join(" ")
.replaceAll(/[^a-zA-Z0-9-_ ]/g, "");
}
}
2024-07-14 17:29:43 +08:00
this.ui.viewDispatch({
type: "update-current-page-meta",
meta: enrichedMeta,
});
}).catch(console.error);
2023-07-14 19:58:16 +08:00
const editorState = createEditorState(
this,
2023-01-16 18:28:59 +08:00
pageName,
doc.text,
doc.meta.perm === "ro",
);
editorView.setState(editorState);
if (editorView.contentDOM) {
2023-01-16 18:28:59 +08:00
this.tweakEditorDOM(editorView.contentDOM);
}
this.space.watchPage(pageName);
2022-11-24 23:55:30 +08:00
// Note: these events are dispatched asynchronously deliberately (not waiting for results)
if (loadingDifferentPage) {
this.eventHook.dispatchEvent("editor:pageLoaded", pageName, previousPage)
.catch(
console.error,
);
2022-07-18 22:48:36 +08:00
} else {
2022-11-24 23:55:30 +08:00
this.eventHook.dispatchEvent("editor:pageReloaded", pageName).catch(
console.error,
);
}
}
2023-01-16 18:28:59 +08:00
tweakEditorDOM(contentDOM: HTMLElement) {
2022-05-09 16:45:36 +08:00
contentDOM.spellcheck = true;
contentDOM.setAttribute("autocorrect", "on");
contentDOM.setAttribute("autocapitalize", "on");
}
async loadCustomStyles() {
const spaceStyles = await this.clientSystem.queryObjects<StyleObject>(
"space-style",
{},
);
if (!spaceStyles) {
return;
}
// Sort stylesheets (last declared styles take precedence)
// Order is 1: Imported styles, 2: Other styles, 3: customStyles from Settings
const sortOrder = ["library", "user", "config"];
spaceStyles.sort((a, b) =>
sortOrder.indexOf(a.origin) - sortOrder.indexOf(b.origin)
);
const accumulatedCSS: string[] = [];
for (const s of spaceStyles) {
accumulatedCSS.push(s.style);
}
const customStylesContent = accumulatedCSS.join("\n\n");
this.ui.viewDispatch({
type: "set-ui-option",
key: "customStyles",
value: customStylesContent,
});
document.getElementById("custom-styles")!.innerHTML = customStylesContent;
}
2023-12-22 20:22:25 +08:00
async runCommandByName(name: string, args?: any[]) {
2023-07-14 20:22:26 +08:00
const cmd = this.ui.viewState.commands.get(name);
2022-07-11 15:08:22 +08:00
if (cmd) {
if (args) {
await cmd.run(args);
} else {
await cmd.run();
}
2022-07-11 15:08:22 +08:00
} else {
throw new Error(`Command ${name} not found`);
}
}
2023-07-14 20:22:26 +08:00
getCommandsByContext(
state: AppViewState,
): Map<string, AppCommand> {
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;
}
2023-07-14 19:58:16 +08:00
getContext(): string | undefined {
2023-07-27 17:41:44 +08:00
const state = this.editorView.state;
const selection = state.selection.main;
if (selection.empty) {
return syntaxTree(state).resolveInner(selection.from).type.name;
}
return;
}
// Widget and image height caching
private widgetCache = new LimitedMap<WidgetCacheItem>(100); // bodyText -> WidgetCacheItem
private widgetHeightCache = new LimitedMap<number>(100); // bodytext -> height
async loadCaches() {
2024-01-02 18:32:57 +08:00
const [widgetHeightCache, widgetCache] = await this
.stateDataStore.batchGet([[
"cache",
"widgetHeight",
], ["cache", "widgets"]]);
this.widgetHeightCache = new LimitedMap(100, widgetHeightCache || {});
this.widgetCache = new LimitedMap(100, widgetCache || {});
}
debouncedWidgetHeightCacheFlush = throttle(() => {
this.stateDataStore.set(
["cache", "widgetHeight"],
this.widgetHeightCache.toJSON(),
)
.catch(
console.error,
);
// console.log("Flushed widget height cache to store");
2023-12-30 03:03:54 +08:00
}, 2000);
setCachedWidgetHeight(bodyText: string, height: number) {
this.widgetHeightCache.set(bodyText, height);
this.debouncedWidgetHeightCacheFlush();
}
getCachedWidgetHeight(bodyText: string): number {
return this.widgetHeightCache.get(bodyText) ?? -1;
}
debouncedWidgetCacheFlush = throttle(() => {
this.stateDataStore.set(["cache", "widgets"], this.widgetCache.toJSON())
.catch(
console.error,
);
console.log("Flushed widget cache to store");
2023-12-30 03:03:54 +08:00
}, 2000);
setWidgetCache(key: string, cacheItem: WidgetCacheItem) {
this.widgetCache.set(key, cacheItem);
this.debouncedWidgetCacheFlush();
}
getWidgetCache(key: string): WidgetCacheItem | undefined {
return this.widgetCache.get(key);
}
}