silverbullet/web/navigator.ts

98 lines
2.1 KiB
TypeScript
Raw Permalink Normal View History

import { safeRun } from "../common/util.ts";
function encodePageUrl(name: string): string {
return name;
}
function decodePageUrl(url: string): string {
return url;
}
2022-03-28 21:25:05 +08:00
export class PathPageNavigator {
navigationResolve?: () => void;
2022-08-02 18:43:39 +08:00
constructor(readonly indexPage: string, readonly root: string = "") {}
2022-07-22 19:44:28 +08:00
2024-01-02 18:01:34 +08:00
async navigate(
page: string,
pos?: number | string | undefined,
replaceState = false,
) {
2022-08-02 18:43:39 +08:00
let encodedPage = encodePageUrl(page);
if (page === this.indexPage) {
encodedPage = "";
}
if (replaceState) {
window.history.replaceState(
2024-01-02 18:01:34 +08:00
{ page },
page,
`${this.root}/${encodedPage}`,
);
} else {
window.history.pushState(
2024-01-02 18:01:34 +08:00
{ page },
page,
`${this.root}/${encodedPage}`,
);
}
globalThis.dispatchEvent(
2022-04-01 21:02:35 +08:00
new PopStateEvent("popstate", {
state: { page, pos },
}),
2022-03-28 21:25:05 +08:00
);
await new Promise<void>((resolve) => {
this.navigationResolve = resolve;
});
this.navigationResolve = undefined;
}
2022-03-28 21:25:05 +08:00
subscribe(
2024-01-02 18:01:34 +08:00
pageLoadCallback: (
pageName: string,
pos: number | string | undefined,
) => Promise<void>,
2022-03-28 21:25:05 +08:00
): 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(),
2024-01-02 18:01:34 +08:00
event?.state?.pos,
2022-04-04 21:25:07 +08:00
);
2022-04-01 21:03:12 +08:00
if (this.navigationResolve) {
this.navigationResolve();
}
2022-04-01 21:03:12 +08:00
});
};
globalThis.addEventListener("popstate", cb);
cb();
}
2022-08-30 16:44:20 +08:00
decodeURI(): [string, number | string] {
const [page, pos] = decodeURI(
location.pathname.substring(this.root.length + 1),
2023-11-03 19:01:33 +08:00
).split(/[@$]/);
2022-08-30 16:44:20 +08:00
if (pos) {
if (pos.match(/^\d+$/)) {
return [page, +pos];
} else {
return [page, pos];
}
2022-04-25 00:06:34 +08:00
} else {
2022-08-30 16:44:20 +08:00
return [page, 0];
2022-04-25 00:06:34 +08:00
}
}
getCurrentPage(): string {
2022-08-02 18:43:39 +08:00
return decodePageUrl(this.decodeURI()[0]) || this.indexPage;
}
2022-08-30 16:44:20 +08:00
getCurrentPos(): number | string {
2022-07-22 19:44:28 +08:00
// console.log("Pos", this.decodeURI()[1]);
2022-04-25 00:06:34 +08:00
return this.decodeURI()[1];
}
}