import { encodePageURI, type PageRef, parsePageRef, } from "../plug-api/lib/page_ref.ts"; import type { Client } from "./client.ts"; import { cleanPageRef } from "@silverbulletmd/silverbullet/lib/resolve"; import { renderTheTemplate } from "$common/syscalls/template.ts"; import { safeRun } from "../lib/async.ts"; export type PageState = PageRef & { scrollTop?: number; selection?: { anchor: number; head?: number; }; }; export class PathPageNavigator { navigationResolve?: () => void; indexPage!: string; openPages = new Map(); constructor( private client: Client, ) { } async init() { this.indexPage = cleanPageRef( await renderTheTemplate( this.client.config.indexPage, {}, {}, this.client.stateDataStore.functionMap, ), ); } /** * Navigates the client to the given page, this involves: * - Patching the current popstate with current state * - Pushing the new state * - Dispatching a popstate event * @param pageRef to navigate to * @param replaceState whether to update the state in place (rather than to push a new state) */ async navigate( pageRef: PageRef, replaceState = false, ) { if (pageRef.page === this.indexPage) { pageRef.page = ""; } const currentState = this.buildCurrentPageState(); // No need to keep pos and anchor if we already have scrollTop and selection const cleanState = { ...currentState, pos: undefined, anchor: undefined }; this.openPages.set(currentState.page || this.indexPage, cleanState); if (!replaceState) { window.history.replaceState( cleanState, "", `/${encodePageURI(currentState.page)}`, ); window.history.pushState( pageRef, "", `/${encodePageURI(pageRef.page)}`, ); } else { window.history.replaceState( pageRef, "", `/${encodePageURI(pageRef.page)}`, ); } globalThis.dispatchEvent( new PopStateEvent("popstate", { state: pageRef, }), ); await new Promise((resolve) => { this.navigationResolve = resolve; }); this.navigationResolve = undefined; } buildCurrentPageState(): PageState { const pageState: PageState = parsePageRefFromURI(); const mainSelection = this.client.editorView.state.selection.main; pageState.scrollTop = this.client.editorView.scrollDOM.scrollTop; pageState.selection = { head: mainSelection.head, anchor: mainSelection.anchor, }; return pageState; } subscribe( pageLoadCallback: ( pageState: PageState, ) => Promise, ): void { const cb = (event: PopStateEvent) => { safeRun(async () => { const popState = event.state; if (popState) { // This is the usual case if (!popState.page) { popState.page = this.indexPage; } if ( popState.anchor === undefined && popState.pos === undefined && popState.selection === undefined && popState.scrollTop === undefined ) { // Pretty low-context popstate, so let's leverage openPages const openPage = this.openPages.get(popState.page); if (openPage) { popState.selection = openPage.selection; popState.scrollTop = openPage.scrollTop; } } await pageLoadCallback(popState); } else { // This occurs when the page is loaded completely fresh with no browser history around it const pageRef = parsePageRefFromURI(); if (!pageRef.page) { pageRef.page = this.indexPage; } await pageLoadCallback(pageRef); } if (this.navigationResolve) { this.navigationResolve(); } }); }; globalThis.addEventListener("popstate", cb); cb( new PopStateEvent("popstate", { state: this.buildCurrentPageState(), }), ); } } export function parsePageRefFromURI(): PageRef { const pageRef = parsePageRef(decodeURIComponent( location.pathname.substring(1), )); if (location.hash) { pageRef.header = decodeURIComponent(location.hash.substring(1)); } return pageRef; }