2022-04-21 20:16:40 +08:00
|
|
|
import { safeRun } from "../silverbullet-common/util";
|
2022-03-20 16:56:28 +08:00
|
|
|
|
|
|
|
function encodePageUrl(name: string): string {
|
|
|
|
return name.replaceAll(" ", "_");
|
|
|
|
}
|
|
|
|
|
|
|
|
function decodePageUrl(url: string): string {
|
|
|
|
return url.replaceAll("_", " ");
|
|
|
|
}
|
|
|
|
|
2022-03-28 21:25:05 +08:00
|
|
|
export class PathPageNavigator {
|
|
|
|
navigationResolve?: () => void;
|
|
|
|
|
|
|
|
async navigate(page: string, pos?: number) {
|
2022-04-01 21:02:35 +08:00
|
|
|
window.history.pushState({ page, pos }, page, `/${encodePageUrl(page)}`);
|
|
|
|
window.dispatchEvent(
|
|
|
|
new PopStateEvent("popstate", {
|
|
|
|
state: { page, pos },
|
|
|
|
})
|
2022-03-28 21:25:05 +08:00
|
|
|
);
|
|
|
|
await new Promise<void>((resolve) => {
|
2022-03-20 16:56:28 +08:00
|
|
|
this.navigationResolve = resolve;
|
|
|
|
});
|
|
|
|
this.navigationResolve = undefined;
|
|
|
|
}
|
2022-03-28 21:25:05 +08:00
|
|
|
|
|
|
|
subscribe(
|
|
|
|
pageLoadCallback: (pageName: string, pos: number) => Promise<void>
|
|
|
|
): void {
|
2022-04-01 21:03:12 +08:00
|
|
|
const cb = (event?: PopStateEvent) => {
|
|
|
|
const gotoPage = this.getCurrentPage();
|
|
|
|
if (!gotoPage) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
safeRun(async () => {
|
2022-04-04 21:25:07 +08:00
|
|
|
await pageLoadCallback(
|
|
|
|
this.getCurrentPage(),
|
|
|
|
event?.state && event.state.pos
|
|
|
|
);
|
2022-04-01 21:03:12 +08:00
|
|
|
if (this.navigationResolve) {
|
|
|
|
this.navigationResolve();
|
2022-03-20 16:56:28 +08:00
|
|
|
}
|
2022-04-01 21:03:12 +08:00
|
|
|
});
|
|
|
|
};
|
|
|
|
window.addEventListener("popstate", cb);
|
2022-03-20 16:56:28 +08:00
|
|
|
cb();
|
|
|
|
}
|
|
|
|
|
|
|
|
getCurrentPage(): string {
|
2022-04-10 17:04:07 +08:00
|
|
|
let [page] = decodeURI(location.pathname).substring(1).split("@");
|
2022-03-28 21:25:05 +08:00
|
|
|
return decodePageUrl(page);
|
2022-03-20 16:56:28 +08:00
|
|
|
}
|
|
|
|
|
2022-03-28 21:25:05 +08:00
|
|
|
getCurrentPos(): number {
|
2022-04-10 17:04:07 +08:00
|
|
|
let [, pos] = decodeURI(location.pathname).substring(1).split("@");
|
2022-03-28 21:25:05 +08:00
|
|
|
return +pos || 0;
|
2022-03-20 16:56:28 +08:00
|
|
|
}
|
|
|
|
}
|