silverbullet/web/client.ts

1238 lines
35 KiB
TypeScript
Raw Normal View History

// Third party web dependencies
2022-04-25 16:33:38 +08:00
import {
Compartment,
CompletionContext,
2022-04-25 16:33:38 +08:00
CompletionResult,
2022-04-04 21:25:07 +08:00
EditorView,
2023-05-29 15:53:49 +08:00
gitIgnoreCompiler,
SyntaxNode,
syntaxTree,
} from "./deps.ts";
import { Space } from "../common/space.ts";
2024-02-09 04:12:23 +08:00
import { FilterOption } from "../type/web.ts";
import { EventHook } from "../common/hooks/event.ts";
import { AppCommand } from "$common/hooks/command.ts";
2024-01-24 21:44:39 +08:00
import {
PageState,
parsePageRefFromURI,
PathPageNavigator,
} from "./navigator.ts";
2023-07-14 19:58:16 +08:00
2024-02-09 04:12:23 +08:00
import { AppViewState, BuiltinSettings } from "../type/web.ts";
2024-02-29 22:23:05 +08:00
import type {
AppEvent,
CompleteEvent,
SlashCompletions,
} from "../plug-api/types.ts";
import { 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 {
ISyncService,
NoSyncSyncService,
pageSyncInterval,
SyncService,
} from "./sync_service.ts";
import { simpleHash } from "$lib/crypto.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 { FilteredSpacePrimitives } from "$common/spaces/filtered_space_primitives.ts";
import { encodePageRef, validatePageName } from "$sb/lib/page_ref.ts";
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 "$sb/lib/resolve.ts";
import { SpacePrimitives } from "$common/spaces/space_primitives.ts";
2024-02-29 22:23:05 +08:00
import { 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";
import {
EncryptedSpacePrimitives,
} from "$common/spaces/encrypted_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";
import { PageRef } from "../plug-api/lib/page_ref.ts";
import { ReadOnlySpacePrimitives } from "$common/spaces/ro_space_primitives.ts";
import { KvPrimitives } from "$lib/data/kv_primitives.ts";
import { builtinFunctions } from "$lib/builtin_query_functions.ts";
import { ensureAndLoadSettingsAndIndex } from "$common/settings.ts";
import { LimitedMap } from "$lib/limited_map.ts";
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 {
interface Window {
// Injected via index.html
silverBulletConfig: {
spaceFolderPath: string;
2023-08-30 23:25:54 +08:00
syncOnly: boolean;
readOnly: boolean;
clientEncryption: boolean;
enableSpaceScript: boolean;
};
2023-07-14 22:56:20 +08:00
client: Client;
}
}
// history.scrollRestoration = "manual";
2023-07-14 22:56:20 +08:00
export class Client {
clientSystem!: ClientSystem;
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
plugSpaceRemotePrimitives!: PlugSpacePrimitives;
httpSpacePrimitives!: HttpSpacePrimitives;
2023-07-27 17:41:44 +08:00
space!: Space;
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);
// Track if plugs have been updated since sync cycle
fullSyncCompleted = false;
2022-12-16 19:44:04 +08:00
syncService!: ISyncService;
2023-07-27 17:41:44 +08:00
settings!: BuiltinSettings;
2023-07-14 19:44:30 +08:00
// Event bus used to communicate between components
eventHook!: EventHook;
2023-07-14 20:22:26 +08:00
ui!: MainUI;
stateDataStore!: DataStore;
spaceKV?: KvPrimitives;
mq!: DataStoreMQ;
2023-07-14 19:44:30 +08:00
2024-01-24 21:44:39 +08:00
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() {
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
// Event hook
this.eventHook = new EventHook();
// 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,
window.silverBulletConfig.readOnly,
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,
(path) => {
// TODO: At some point we should remove the data.db exception here
return path !== "data.db" &&
// Exclude all plug space primitives paths
!this.plugSpaceRemotePrimitives.isLikelyHandled(path) ||
// 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.loadSettings();
await this.loadCaches();
// Pinging a 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,
);
}
2023-07-27 21:25:33 +08:00
await this.loadPlugs();
await this.clientSystem.loadSpaceScripts();
await this.initNavigator();
await this.initSync();
2022-11-24 19:04:00 +08:00
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
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
this.updatePageListCache().catch(console.error);
2023-07-27 21:25:33 +08:00
}
async loadSettings() {
2024-01-25 18:42:36 +08:00
this.settings = await ensureAndLoadSettingsAndIndex(
this.space.spacePrimitives,
);
this.ui.viewDispatch({
type: "settings-loaded",
settings: this.settings,
});
}
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) => {
// console.log("Operations", 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-02-28 17:55:25 +08:00
// Let's load space scripts, 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
});
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
});
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.showProgress(
Math.round(status.filesProcessed / status.totalFiles * 100),
);
});
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 | undefined = pageState.pos;
if (pageState.anchor) {
console.log("Navigating to anchor", pageState.anchor);
const pageText = this.editorView.state.sliceDoc();
2024-01-25 21:51:40 +08:00
// This is somewhat of a simplistic way to find the anchor, but it works for now
pos = pageText.indexOf(`$${pageState.anchor}`);
if (pos === -1) {
return this.flashNotification(
`Could not find anchor $${pageState.anchor}`,
"error",
);
}
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) {
// setTimeout(() => {
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);
await this.stateDataStore.set(
["client", "lastOpenedPage"],
pageState.page,
);
});
2023-08-16 17:40:31 +08:00
if (location.hash === "#boot") {
(async () => {
// Cold start PWA load
const lastPage = await this.stateDataStore.get([
"client",
"lastOpenedPage",
]);
2023-08-16 17:40:31 +08:00
if (lastPage) {
await this.navigate(lastPage);
}
})().catch(console.error);
}
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 (window.silverBulletConfig.clientEncryption) {
console.log("Enabling encryption");
const encryptedSpacePrimitives = new EncryptedSpacePrimitives(
this.httpSpacePrimitives,
);
remoteSpacePrimitives = encryptedSpacePrimitives;
let loggedIn = false;
// First figure out if we're online & if the key file exists, if not we need to initialize the space
try {
if (!await encryptedSpacePrimitives.init()) {
console.log(
"Space not initialized, will ask for password to initialize",
);
alert(
"You appear to be accessing a new space with encryption enabled, you will now be asked to create a password",
);
const password = prompt("Choose a password");
if (!password) {
alert("Cannot do anything without a password, reloading");
location.reload();
throw new Error("Not initialized");
}
const password2 = prompt("Confirm password");
if (password !== password2) {
alert("Passwords don't match, reloading");
location.reload();
throw new Error("Not initialized");
}
await encryptedSpacePrimitives.setup(password);
// this.stateDataStore.set(["encryptionKey"], password);
await this.stateDataStore.set(
["spaceSalt"],
encryptedSpacePrimitives.spaceSalt,
);
loggedIn = true;
}
} catch (e: any) {
if (e.message === "Offline") {
console.log(
"Offline, will assume encryption space is initialized, fetching salt from data store",
);
await encryptedSpacePrimitives.init(
await this.stateDataStore.get(["spaceSalt"]),
);
}
}
if (!loggedIn) {
// Let's ask for the password
try {
await encryptedSpacePrimitives.login(
prompt("Password")!,
);
await this.stateDataStore.set(
["spaceSalt"],
encryptedSpacePrimitives.spaceSalt,
);
} catch (e: any) {
console.log("Got this error", e);
if (e.message === "Incorrect password") {
alert("Incorrect password");
location.reload();
}
throw e;
}
}
}
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.settings) {
await this.loadSettings();
}
2023-08-26 14:31:51 +08:00
if (typeof this.settings?.spaceIgnore === "string") {
fileFilterFn = gitIgnoreCompiler(this.settings.spaceIgnore).accepts;
} 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,
2024-03-14 20:02:34 +08:00
_localChange: boolean,
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
2023-12-22 22:55:50 +08:00
// Caching a list of known pages for the wiki_link highlighter (that checks if a page exists)
this.eventHook.addLocalListener("page:saved", (pageName: string) => {
// Make sure this page is in the list of known pages
this.clientSystem.allKnownPages.add(pageName);
2023-12-22 22:55:50 +08:00
});
this.eventHook.addLocalListener("page:deleted", (pageName: string) => {
this.clientSystem.allKnownPages.delete(pageName);
2023-12-22 22:55:50 +08:00
});
this.eventHook.addLocalListener(
"file:listed",
(allFiles: FileMeta[]) => {
// Update list of known pages
this.clientSystem.allKnownPages.clear();
allFiles.forEach((f) => {
if (f.name.endsWith(".md")) {
this.clientSystem.allKnownPages.add(f.name.slice(0, -3));
}
});
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);
}
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,
);
resolve();
})
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" | "template") {
2023-12-22 01:37:50 +08:00
// Then show the page navigator
this.ui.viewDispatch({ type: "start-navigate", mode });
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", {});
2023-12-22 22:55:50 +08:00
this.ui.viewDispatch({
type: "update-page-list",
allPages,
});
2023-12-22 01:37:50 +08:00
}
2023-07-27 23:02:53 +08:00
private progressTimeout?: number;
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,
);
}
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);
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
}
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-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;
}
}
2023-12-22 22:55:50 +08:00
// console.log("Compeltion result", actualResult);
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);
}
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.settings.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}/${encodePageRef(pageRef)}`,
);
const win = globalThis.open(
`${location.origin}/${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
};
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,
});
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", "settings"];
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);
}
}
type WidgetCacheItem = {
height: number;
html: string;
buttons?: CodeWidgetButton[];
banner?: string;
};