1172 lines
34 KiB
TypeScript
1172 lines
34 KiB
TypeScript
// Third party web dependencies
|
|
import {
|
|
Compartment,
|
|
CompletionContext,
|
|
CompletionResult,
|
|
EditorView,
|
|
gitIgnoreCompiler,
|
|
SyntaxNode,
|
|
syntaxTree,
|
|
} from "../common/deps.ts";
|
|
import { Space } from "./space.ts";
|
|
import { FilterOption } from "./types.ts";
|
|
import { ensureAndLoadSettingsAndIndex } from "../common/util.ts";
|
|
import { EventHook } from "../plugos/hooks/event.ts";
|
|
import { AppCommand } from "./hooks/command.ts";
|
|
import {
|
|
PageState,
|
|
parsePageRefFromURI,
|
|
PathPageNavigator,
|
|
} from "./navigator.ts";
|
|
|
|
import { AppViewState, BuiltinSettings } from "./types.ts";
|
|
|
|
import type { AppEvent, CompleteEvent } from "../plug-api/app_event.ts";
|
|
import { throttle } from "$sb/lib/async.ts";
|
|
import { PlugSpacePrimitives } from "../common/spaces/plug_space_primitives.ts";
|
|
import { EventedSpacePrimitives } from "../common/spaces/evented_space_primitives.ts";
|
|
import {
|
|
ISyncService,
|
|
NoSyncSyncService,
|
|
pageSyncInterval,
|
|
SyncService,
|
|
} from "./sync_service.ts";
|
|
import { simpleHash } from "../common/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.ts";
|
|
import { ClientSystem } from "./client_system.ts";
|
|
import { createEditorState } from "./editor_state.ts";
|
|
import { MainUI } from "./editor_ui.tsx";
|
|
import { cleanPageRef } from "$sb/lib/resolve.ts";
|
|
import { SpacePrimitives } from "../common/spaces/space_primitives.ts";
|
|
import { CodeWidgetButton, FileMeta, PageMeta } from "$sb/types.ts";
|
|
import { DataStore } from "../plugos/lib/datastore.ts";
|
|
import { IndexedDBKvPrimitives } from "../plugos/lib/indexeddb_kv_primitives.ts";
|
|
import { DataStoreMQ } from "../plugos/lib/mq.datastore.ts";
|
|
import { DataStoreSpacePrimitives } from "../common/spaces/datastore_space_primitives.ts";
|
|
import {
|
|
EncryptedSpacePrimitives,
|
|
} from "../common/spaces/encrypted_space_primitives.ts";
|
|
|
|
import {
|
|
ensureSpaceIndex,
|
|
markFullSpaceIndexComplete,
|
|
} from "../common/space_index.ts";
|
|
import { LimitedMap } from "$sb/lib/limited_map.ts";
|
|
import { renderHandlebarsTemplate } from "../common/syscalls/handlebars.ts";
|
|
import { buildQueryFunctions } from "../common/query_functions.ts";
|
|
import { PageRef } from "$sb/lib/page.ts";
|
|
const frontMatterRegex = /^---\n(([^\n]|\n)*?)---\n/;
|
|
|
|
const autoSaveInterval = 1000;
|
|
|
|
declare global {
|
|
interface Window {
|
|
// Injected via index.html
|
|
silverBulletConfig: {
|
|
spaceFolderPath: string;
|
|
syncOnly: boolean;
|
|
clientEncryption: boolean;
|
|
};
|
|
client: Client;
|
|
}
|
|
}
|
|
|
|
// history.scrollRestoration = "manual";
|
|
|
|
export class Client {
|
|
system!: ClientSystem;
|
|
editorView!: EditorView;
|
|
keyHandlerCompartment?: Compartment;
|
|
|
|
private pageNavigator!: PathPageNavigator;
|
|
|
|
private dbPrefix: string;
|
|
|
|
plugSpaceRemotePrimitives!: PlugSpacePrimitives;
|
|
// localSpacePrimitives!: FilteredSpacePrimitives;
|
|
httpSpacePrimitives!: HttpSpacePrimitives;
|
|
space!: Space;
|
|
|
|
saveTimeout?: number;
|
|
debouncedUpdateEvent = throttle(() => {
|
|
this.eventHook
|
|
.dispatchEvent("editor:updated")
|
|
.catch((e) => console.error("Error dispatching editor:updated event", e));
|
|
}, 1000);
|
|
|
|
// Track if plugs have been updated since sync cycle
|
|
fullSyncCompleted = false;
|
|
|
|
syncService!: ISyncService;
|
|
settings!: BuiltinSettings;
|
|
|
|
// Event bus used to communicate between components
|
|
eventHook!: EventHook;
|
|
|
|
ui!: MainUI;
|
|
stateDataStore!: DataStore;
|
|
spaceDataStore!: DataStore;
|
|
mq!: DataStoreMQ;
|
|
|
|
// Used by the "wiki link" highlighter to check if a page exists
|
|
public allKnownPages = new Set<string>();
|
|
onLoadPageRef: PageRef;
|
|
|
|
constructor(
|
|
private parent: Element,
|
|
public syncMode = false,
|
|
) {
|
|
if (!syncMode) {
|
|
this.fullSyncCompleted = true;
|
|
}
|
|
// Generate a semi-unique prefix for the database so not to reuse databases for different space paths
|
|
this.dbPrefix = "" + simpleHash(window.silverBulletConfig.spaceFolderPath);
|
|
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,
|
|
buildQueryFunctions(this.allKnownPages),
|
|
);
|
|
|
|
// Setup message queue
|
|
this.mq = new DataStoreMQ(this.stateDataStore);
|
|
|
|
setInterval(() => {
|
|
// Timeout after 5s, retries 3 times, otherwise drops the message (no DLQ)
|
|
this.mq.requeueTimeouts(5000, 3, true).catch(console.error);
|
|
}, 20000); // Look to requeue every 20s
|
|
|
|
// Event hook
|
|
this.eventHook = new EventHook();
|
|
|
|
// Instantiate a PlugOS system
|
|
this.system = new ClientSystem(
|
|
this,
|
|
this.mq,
|
|
this.stateDataStore,
|
|
this.eventHook,
|
|
);
|
|
|
|
const localSpacePrimitives = await this.initSpace();
|
|
|
|
this.syncService = this.syncMode
|
|
? new SyncService(
|
|
localSpacePrimitives,
|
|
this.plugSpaceRemotePrimitives,
|
|
this.stateDataStore,
|
|
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);
|
|
|
|
this.ui = new MainUI(this);
|
|
this.ui.render(this.parent);
|
|
|
|
this.editorView = new EditorView({
|
|
state: createEditorState(this, "", "", false),
|
|
parent: document.getElementById("sb-editor")!,
|
|
});
|
|
|
|
this.focus();
|
|
|
|
this.system.init();
|
|
|
|
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();
|
|
} catch (e: any) {
|
|
if (e.message === "Not authenticated") {
|
|
console.warn("Not authenticated, redirecting to auth page");
|
|
return;
|
|
}
|
|
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;
|
|
}
|
|
console.warn(
|
|
"Could not reach remote server, we're offline or the server is down",
|
|
e,
|
|
);
|
|
}
|
|
|
|
await this.loadPlugs();
|
|
this.initNavigator();
|
|
await this.initSync();
|
|
|
|
this.loadCustomStyles().catch(console.error);
|
|
|
|
await this.dispatchAppEvent("editor:init");
|
|
|
|
setInterval(() => {
|
|
try {
|
|
this.syncService.syncFile(`${this.currentPage}.md`).catch((e: any) => {
|
|
console.error("Interval sync error", e);
|
|
});
|
|
} catch (e: any) {
|
|
console.error("Interval sync error", e);
|
|
}
|
|
}, pageSyncInterval);
|
|
|
|
this.updatePageListCache().catch(console.error);
|
|
}
|
|
|
|
async loadSettings() {
|
|
this.settings = await ensureAndLoadSettingsAndIndex(
|
|
this.space.spacePrimitives,
|
|
);
|
|
this.ui.viewDispatch({
|
|
type: "settings-loaded",
|
|
settings: this.settings,
|
|
});
|
|
}
|
|
|
|
private async initSync() {
|
|
this.syncService.start();
|
|
|
|
// We're still booting, if a initial sync has already been completed we know this is the initial sync
|
|
const initialSync = !await this.syncService.hasInitialSyncCompleted();
|
|
|
|
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;
|
|
|
|
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.system.system).catch(
|
|
console.error,
|
|
);
|
|
} else {
|
|
// This was the initial sync, let's mark a full index as completed
|
|
await markFullSpaceIndexComplete(this.stateDataStore);
|
|
}
|
|
}
|
|
if (operations) {
|
|
// Likely initial sync so let's show visually that we're synced now
|
|
this.showProgress(100);
|
|
}
|
|
|
|
this.ui.viewDispatch({ type: "sync-change", syncSuccess: true });
|
|
});
|
|
this.eventHook.addLocalListener("sync:error", (_name) => {
|
|
this.ui.viewDispatch({ type: "sync-change", syncSuccess: 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.showProgress(
|
|
Math.round(status.filesProcessed / status.totalFiles * 100),
|
|
);
|
|
});
|
|
this.eventHook.addLocalListener(
|
|
"file:synced",
|
|
(meta: FileMeta, direction: string) => {
|
|
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));
|
|
}
|
|
},
|
|
);
|
|
}
|
|
|
|
private navigateWithinPage(pageState: PageState) {
|
|
// Did we end up doing anything in terms of internal navigation?
|
|
let adjustedPosition = false;
|
|
|
|
// Was a particular scroll position persisted?
|
|
if (
|
|
pageState.scrollTop !== undefined &&
|
|
!(pageState.scrollTop === 0 &&
|
|
(pageState.pos !== undefined || pageState.anchor !== undefined ||
|
|
pageState.header !== undefined))
|
|
) {
|
|
setTimeout(() => {
|
|
console.log("Kicking off scroll to", pageState.scrollTop);
|
|
this.editorView.scrollDOM.scrollTop = pageState.scrollTop!;
|
|
});
|
|
adjustedPosition = true;
|
|
}
|
|
|
|
// Was a particular cursor/selection set?
|
|
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();
|
|
|
|
// 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;
|
|
}
|
|
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 initNavigator() {
|
|
this.pageNavigator = new PathPageNavigator(this);
|
|
|
|
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,
|
|
);
|
|
});
|
|
|
|
if (location.hash === "#boot") {
|
|
(async () => {
|
|
// Cold start PWA load
|
|
const lastPage = await this.stateDataStore.get([
|
|
"client",
|
|
"lastOpenedPage",
|
|
]);
|
|
if (lastPage) {
|
|
await this.navigate(lastPage);
|
|
}
|
|
})().catch(console.error);
|
|
}
|
|
}
|
|
|
|
async initSpace(): Promise<SpacePrimitives> {
|
|
this.httpSpacePrimitives = new HttpSpacePrimitives(
|
|
location.origin,
|
|
window.silverBulletConfig.spaceFolderPath,
|
|
);
|
|
|
|
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;
|
|
}
|
|
}
|
|
}
|
|
|
|
this.plugSpaceRemotePrimitives = new PlugSpacePrimitives(
|
|
remoteSpacePrimitives,
|
|
this.system.namespaceHook,
|
|
this.syncMode ? undefined : "client",
|
|
);
|
|
|
|
let fileFilterFn: (s: string) => boolean = () => true;
|
|
|
|
let localSpacePrimitives: SpacePrimitives | undefined;
|
|
|
|
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();
|
|
|
|
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,
|
|
buildQueryFunctions(this.allKnownPages),
|
|
),
|
|
),
|
|
this.plugSpaceRemotePrimitives,
|
|
),
|
|
this.eventHook,
|
|
),
|
|
(meta) => fileFilterFn(meta.name),
|
|
// Run when a list of files has been retrieved
|
|
async () => {
|
|
if (!this.settings) {
|
|
await this.loadSettings();
|
|
}
|
|
|
|
if (typeof this.settings?.spaceIgnore === "string") {
|
|
fileFilterFn = gitIgnoreCompiler(this.settings.spaceIgnore).accepts;
|
|
} else {
|
|
fileFilterFn = () => true;
|
|
}
|
|
},
|
|
);
|
|
} else {
|
|
localSpacePrimitives = new EventedSpacePrimitives(
|
|
this.plugSpaceRemotePrimitives,
|
|
this.eventHook,
|
|
);
|
|
}
|
|
|
|
this.space = new Space(
|
|
localSpacePrimitives,
|
|
this.eventHook,
|
|
);
|
|
|
|
this.eventHook.addLocalListener(
|
|
"file:changed",
|
|
(
|
|
path: string,
|
|
_localChange?: boolean,
|
|
oldHash?: number,
|
|
newHash?: number,
|
|
) => {
|
|
// Only reload when watching the current page (to avoid reloading when switching pages)
|
|
if (
|
|
this.space.watchInterval && `${this.currentPage}.md` === path
|
|
) {
|
|
console.log(
|
|
"Page changed elsewhere, reloading. Old hash",
|
|
oldHash,
|
|
"new hash",
|
|
newHash,
|
|
);
|
|
this.flashNotification("Page changed elsewhere, reloading");
|
|
this.reloadPage();
|
|
}
|
|
},
|
|
);
|
|
|
|
// 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.allKnownPages.add(pageName);
|
|
});
|
|
|
|
this.eventHook.addLocalListener("page:deleted", (pageName: string) => {
|
|
this.allKnownPages.delete(pageName);
|
|
});
|
|
|
|
this.eventHook.addLocalListener(
|
|
"file:listed",
|
|
(allFiles: FileMeta[]) => {
|
|
// Update list of known pages
|
|
this.allKnownPages.clear();
|
|
allFiles.forEach((f) => {
|
|
if (f.name.endsWith(".md")) {
|
|
this.allKnownPages.add(f.name.slice(0, -3));
|
|
}
|
|
});
|
|
},
|
|
);
|
|
|
|
this.space.watch();
|
|
|
|
return localSpacePrimitives;
|
|
}
|
|
|
|
get currentPage(): string {
|
|
return this.ui.viewState.currentPage !== undefined
|
|
? this.ui.viewState.currentPage
|
|
: this.onLoadPageRef.page; // best effort
|
|
}
|
|
|
|
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 (
|
|
!this.ui.viewState.unsavedChanges ||
|
|
this.ui.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(async (meta) => {
|
|
this.ui.viewDispatch({ type: "page-saved" });
|
|
await this.dispatchAppEvent(
|
|
"editor:pageSaved",
|
|
this.currentPage,
|
|
meta,
|
|
);
|
|
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 : autoSaveInterval,
|
|
);
|
|
});
|
|
}
|
|
|
|
flashNotification(message: string, type: "info" | "error" = "info") {
|
|
const id = Math.floor(Math.random() * 1000000);
|
|
this.ui.viewDispatch({
|
|
type: "show-notification",
|
|
notification: {
|
|
id,
|
|
type,
|
|
message,
|
|
date: new Date(),
|
|
},
|
|
});
|
|
setTimeout(
|
|
() => {
|
|
this.ui.viewDispatch({
|
|
type: "dismiss-notification",
|
|
id: id,
|
|
});
|
|
},
|
|
type === "info" ? 4000 : 5000,
|
|
);
|
|
}
|
|
|
|
startPageNavigate(mode: "page" | "template") {
|
|
// Then show the page navigator
|
|
this.ui.viewDispatch({ type: "start-navigate", mode });
|
|
this.updatePageListCache().catch(console.error);
|
|
}
|
|
|
|
async updatePageListCache() {
|
|
console.log("Updating page list cache");
|
|
const allPages = await this.system.queryObjects<PageMeta>("page", {});
|
|
this.ui.viewDispatch({
|
|
type: "update-page-list",
|
|
allPages,
|
|
});
|
|
}
|
|
|
|
private progressTimeout?: number;
|
|
showProgress(progressPerc: number) {
|
|
this.ui.viewDispatch({
|
|
type: "set-progress",
|
|
progressPerc,
|
|
});
|
|
if (this.progressTimeout) {
|
|
clearTimeout(this.progressTimeout);
|
|
}
|
|
this.progressTimeout = setTimeout(
|
|
() => {
|
|
this.ui.viewDispatch({
|
|
type: "set-progress",
|
|
});
|
|
},
|
|
10000,
|
|
);
|
|
}
|
|
|
|
filterBox(
|
|
label: string,
|
|
options: FilterOption[],
|
|
helpText = "",
|
|
placeHolder = "",
|
|
): Promise<FilterOption | undefined> {
|
|
return new Promise((resolve) => {
|
|
this.ui.viewDispatch({
|
|
type: "show-filterbox",
|
|
label,
|
|
options,
|
|
placeHolder,
|
|
helpText,
|
|
onSelect: (option: any) => {
|
|
this.ui.viewDispatch({ type: "hide-filterbox" });
|
|
this.focus();
|
|
resolve(option);
|
|
},
|
|
});
|
|
});
|
|
}
|
|
|
|
prompt(
|
|
message: string,
|
|
defaultValue = "",
|
|
): Promise<string | undefined> {
|
|
return new Promise((resolve) => {
|
|
this.ui.viewDispatch({
|
|
type: "show-prompt",
|
|
message,
|
|
defaultValue,
|
|
callback: (value: string | undefined) => {
|
|
this.ui.viewDispatch({ type: "hide-prompt" });
|
|
this.focus();
|
|
resolve(value);
|
|
},
|
|
});
|
|
});
|
|
}
|
|
|
|
confirm(
|
|
message: string,
|
|
): Promise<boolean> {
|
|
return new Promise((resolve) => {
|
|
this.ui.viewDispatch({
|
|
type: "show-confirm",
|
|
message,
|
|
callback: (value: boolean) => {
|
|
this.ui.viewDispatch({ type: "hide-confirm" });
|
|
this.focus();
|
|
resolve(value);
|
|
},
|
|
});
|
|
});
|
|
}
|
|
|
|
async loadPlugs() {
|
|
await this.system.reloadPlugsFromSpace(this.space);
|
|
await this.eventHook.dispatchEvent("system:ready");
|
|
await this.dispatchAppEvent("plugs:loaded");
|
|
}
|
|
|
|
rebuildEditorState() {
|
|
const editorView = this.editorView;
|
|
console.log("Rebuilding editor state");
|
|
|
|
if (this.currentPage) {
|
|
editorView.setState(
|
|
createEditorState(
|
|
this,
|
|
this.currentPage,
|
|
editorView.state.sliceDoc(),
|
|
this.ui.viewState.currentPageMeta?.perm === "ro",
|
|
),
|
|
);
|
|
if (editorView.contentDOM) {
|
|
this.tweakEditorDOM(
|
|
editorView.contentDOM,
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Code completion support
|
|
async completeWithEvent(
|
|
context: CompletionContext,
|
|
eventName: AppEvent,
|
|
): Promise<CompletionResult | 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);
|
|
|
|
const parentNodes: string[] = [];
|
|
const sTree = syntaxTree(editorState);
|
|
const currentNode = sTree.resolveInner(selection.from);
|
|
if (currentNode) {
|
|
let node: SyntaxNode | null = currentNode;
|
|
do {
|
|
if (node.name === "FencedCode" || node.name === "FrontMatter") {
|
|
const body = editorState.sliceDoc(node.from + 3, node.to - 3);
|
|
parentNodes.push(`${node.name}:${body}`);
|
|
} else {
|
|
parentNodes.push(node.name);
|
|
}
|
|
node = node.parent;
|
|
} while (node);
|
|
}
|
|
|
|
const results = await this.dispatchAppEvent(eventName, {
|
|
pageName: this.currentPage,
|
|
linePrefix,
|
|
pos: selection.from,
|
|
parentNodes,
|
|
} 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",
|
|
);
|
|
console.error("Previously had", actualResult, "now also got", result);
|
|
return null;
|
|
}
|
|
actualResult = result;
|
|
}
|
|
}
|
|
// console.log("Compeltion result", actualResult);
|
|
return actualResult;
|
|
}
|
|
|
|
editorComplete(
|
|
context: CompletionContext,
|
|
): Promise<CompletionResult | null> {
|
|
return this.completeWithEvent(context, "editor:complete");
|
|
}
|
|
|
|
miniEditorComplete(
|
|
context: CompletionContext,
|
|
): Promise<CompletionResult | null> {
|
|
return this.completeWithEvent(context, "minieditor:complete");
|
|
}
|
|
|
|
async reloadPage() {
|
|
console.log("Reloading page");
|
|
clearTimeout(this.saveTimeout);
|
|
await this.loadPage(this.currentPage);
|
|
}
|
|
|
|
focus() {
|
|
const viewState = this.ui.viewState;
|
|
if (
|
|
[
|
|
viewState.showCommandPalette,
|
|
viewState.showPageNavigator,
|
|
viewState.showFilterBox,
|
|
viewState.showConfirm,
|
|
viewState.showPrompt,
|
|
].some(Boolean)
|
|
) {
|
|
// Some other modal UI element is visible, don't focus editor now
|
|
return;
|
|
}
|
|
this.editorView.focus();
|
|
}
|
|
|
|
async navigate(
|
|
pageRef: PageRef,
|
|
replaceState = false,
|
|
newWindow = false,
|
|
) {
|
|
if (!pageRef.page) {
|
|
pageRef.page = cleanPageRef(
|
|
renderHandlebarsTemplate(this.settings.indexPage, {}, {}),
|
|
);
|
|
}
|
|
|
|
try {
|
|
validatePageName(pageRef.page);
|
|
} catch (e: any) {
|
|
return this.flashNotification(e.message, "error");
|
|
}
|
|
|
|
if (newWindow) {
|
|
console.log(
|
|
"Navigating to new page in new window",
|
|
`${location.origin}/${encodePageRef(pageRef)}`,
|
|
);
|
|
const win = window.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;
|
|
const previousPage = this.currentPage;
|
|
|
|
// Persist current page state and nicely close page
|
|
if (previousPage) {
|
|
// this.openPages.saveState(previousPage);
|
|
this.space.unwatchPage(previousPage);
|
|
if (previousPage !== pageName) {
|
|
await this.save(true);
|
|
}
|
|
}
|
|
|
|
this.ui.viewDispatch({
|
|
type: "page-loading",
|
|
name: pageName,
|
|
});
|
|
|
|
// Fetch next page to open
|
|
let doc;
|
|
try {
|
|
doc = await this.space.readPage(pageName);
|
|
} catch (e: any) {
|
|
if (e.message.includes("Not found")) {
|
|
// Not found, new page
|
|
console.log("Page doesn't exist, creating new page:", pageName);
|
|
// Initialize page
|
|
doc = {
|
|
text: "",
|
|
meta: {
|
|
ref: pageName,
|
|
tags: ["page"],
|
|
name: pageName,
|
|
lastModified: "",
|
|
created: "",
|
|
perm: "rw",
|
|
} as PageMeta,
|
|
};
|
|
this.system.system.invokeFunction("template.newPage", [pageName]).then(
|
|
() => {
|
|
this.focus();
|
|
},
|
|
).catch(
|
|
console.error,
|
|
);
|
|
} else {
|
|
this.flashNotification(
|
|
`Could not load page ${pageName}: ${e.message}`,
|
|
"error",
|
|
);
|
|
if (previousPage) {
|
|
this.ui.viewDispatch({
|
|
type: "page-loading",
|
|
name: previousPage,
|
|
});
|
|
}
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
this.ui.viewDispatch({
|
|
type: "page-loaded",
|
|
meta: doc.meta,
|
|
});
|
|
|
|
const editorState = createEditorState(
|
|
this,
|
|
pageName,
|
|
doc.text,
|
|
doc.meta.perm === "ro",
|
|
);
|
|
editorView.setState(editorState);
|
|
if (editorView.contentDOM) {
|
|
this.tweakEditorDOM(editorView.contentDOM);
|
|
}
|
|
this.space.watchPage(pageName);
|
|
|
|
// Note: these events are dispatched asynchronously deliberately (not waiting for results)
|
|
if (loadingDifferentPage) {
|
|
this.eventHook.dispatchEvent("editor:pageLoaded", pageName, previousPage)
|
|
.catch(
|
|
console.error,
|
|
);
|
|
} else {
|
|
this.eventHook.dispatchEvent("editor:pageReloaded", pageName).catch(
|
|
console.error,
|
|
);
|
|
}
|
|
}
|
|
|
|
tweakEditorDOM(contentDOM: HTMLElement) {
|
|
contentDOM.spellcheck = true;
|
|
contentDOM.setAttribute("autocorrect", "on");
|
|
contentDOM.setAttribute("autocapitalize", "on");
|
|
}
|
|
|
|
async loadCustomStyles() {
|
|
if (this.settings.customStyles) {
|
|
const accumulatedCSS: string[] = [];
|
|
let customStylePages = this.settings.customStyles;
|
|
if (!Array.isArray(customStylePages)) {
|
|
customStylePages = [customStylePages];
|
|
}
|
|
for (const customStylesPage of customStylePages) {
|
|
try {
|
|
const { text: stylesText } = await this.space.readPage(
|
|
cleanPageRef(customStylesPage),
|
|
);
|
|
const cssBlockRegex = /```css([^`]+)```/;
|
|
const match = cssBlockRegex.exec(stylesText);
|
|
if (!match) {
|
|
return;
|
|
}
|
|
accumulatedCSS.push(match[1]);
|
|
} catch (e: any) {
|
|
console.error("Failed to load custom styles", e);
|
|
}
|
|
}
|
|
document.getElementById("custom-styles")!.innerHTML = accumulatedCSS.join(
|
|
"\n\n",
|
|
);
|
|
}
|
|
}
|
|
|
|
async runCommandByName(name: string, args?: any[]) {
|
|
const cmd = this.ui.viewState.commands.get(name);
|
|
if (cmd) {
|
|
if (args) {
|
|
await cmd.run(args);
|
|
} else {
|
|
await cmd.run();
|
|
}
|
|
} else {
|
|
throw new Error(`Command ${name} not found`);
|
|
}
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
// Widget and image height caching
|
|
private widgetCache = new LimitedMap<WidgetCacheItem>(100); // bodyText -> WidgetCacheItem
|
|
private widgetHeightCache = new LimitedMap<number>(100); // bodytext -> height
|
|
|
|
async loadCaches() {
|
|
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");
|
|
}, 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");
|
|
}, 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;
|
|
};
|