hippofish/packages/client/src/nirax.ts

296 lines
6.7 KiB
TypeScript
Raw Normal View History

2022-06-21 07:18:06 +02:00
// NIRAX --- A lightweight router
2023-01-13 05:40:33 +01:00
import { EventEmitter } from "eventemitter3";
2024-04-23 15:30:55 +02:00
import type { Component } from "vue";
2023-10-31 12:19:05 +01:00
import { shallowRef } from "vue";
2024-03-30 17:30:50 +01:00
import { safeURIDecode } from "@/scripts/safe-uri-decode";
import { pleaseLogin } from "@/scripts/please-login";
2024-04-12 05:31:11 +02:00
export interface RouteDef {
path: string;
component: Component;
query?: Record<string, string>;
loginRequired?: boolean;
name?: string;
hash?: string;
globalCacheKey?: string;
children?: RouteDef[];
2023-09-02 01:27:33 +02:00
}
2023-01-13 05:40:33 +01:00
type ParsedPath = (
| string
| {
name: string;
startsWith?: string;
wildcard?: boolean;
optional?: boolean;
}
)[];
2023-09-02 01:27:33 +02:00
export interface Resolved {
2023-01-13 05:40:33 +01:00
route: RouteDef;
props: Map<string, string>;
child?: Resolved;
2023-09-02 01:27:33 +02:00
}
function parsePath(path: string): ParsedPath {
const res = [] as ParsedPath;
2024-04-23 15:30:55 +02:00
// biome-ignore lint/style/noParameterAssign: assign it intentionally
path = path.substring(1);
2023-01-13 05:40:33 +01:00
for (const part of path.split("/")) {
if (part.includes(":")) {
const prefix = part.substring(0, part.indexOf(":"));
const placeholder = part.substring(part.indexOf(":") + 1);
const wildcard = placeholder.includes("(*)");
const optional = placeholder.endsWith("?");
res.push({
2023-01-13 05:40:33 +01:00
name: placeholder.replace("(*)", "").replace("?", ""),
startsWith: prefix !== "" ? prefix : undefined,
wildcard,
optional,
});
2022-06-23 18:26:15 +02:00
} else if (part.length !== 0) {
res.push(part);
}
}
return res;
}
export class Router extends EventEmitter<{
change: (ctx: {
beforePath: string;
path: string;
resolved: Resolved;
key: string;
}) => void;
2023-09-02 01:27:33 +02:00
replace: (ctx: { path: string; key: string }) => void;
push: (ctx: {
beforePath: string;
path: string;
route: RouteDef | null;
props: Map<string, string> | null;
key: string;
}) => void;
2022-07-05 15:25:27 +02:00
same: () => void;
}> {
private routes: RouteDef[];
2024-04-23 15:30:55 +02:00
public current!: Resolved; // It is assigned in this.navigate
public currentRef = shallowRef<Resolved>();
public currentRoute = shallowRef<RouteDef>();
private currentPath: string;
private currentKey = Date.now().toString();
2024-04-23 15:30:55 +02:00
public navHook: ((path: string, flag?: unknown) => boolean) | null = null;
2023-01-13 05:40:33 +01:00
constructor(routes: Router["routes"], currentPath: Router["currentPath"]) {
super();
this.routes = routes;
this.currentPath = currentPath;
this.navigate(currentPath, null, false);
}
2024-04-23 15:30:55 +02:00
public resolve(_path: string): Resolved | null {
2024-03-30 17:30:50 +01:00
let queryString: string | null = null;
let hash: string | null = null;
2024-04-23 15:30:55 +02:00
let path = _path;
2023-01-13 05:40:33 +01:00
if (path[0] === "/") path = path.substring(1);
if (path.includes("#")) {
hash = path.substring(path.indexOf("#") + 1);
path = path.substring(0, path.indexOf("#"));
}
2023-01-13 05:40:33 +01:00
if (path.includes("?")) {
queryString = path.substring(path.indexOf("?") + 1);
path = path.substring(0, path.indexOf("?"));
}
2023-01-13 05:40:33 +01:00
if (_DEV_) console.log("Routing: ", path, queryString);
function check(routes: RouteDef[], _parts: string[]): Resolved | null {
2023-01-13 05:40:33 +01:00
forEachRouteLoop: for (const route of routes) {
let parts = [..._parts];
const props = new Map<string, string>();
2023-11-16 21:18:19 +01:00
for (const p of parsePath(route.path)) {
2023-01-13 05:40:33 +01:00
if (typeof p === "string") {
if (p === parts[0]) {
parts.shift();
} else {
continue forEachRouteLoop;
}
} else {
if (parts[0] == null && !p.optional) {
continue forEachRouteLoop;
}
if (p.wildcard) {
if (parts.length !== 0) {
2023-01-13 05:40:33 +01:00
props.set(p.name, safeURIDecode(parts.join("/")));
parts = [];
}
2023-11-16 21:18:19 +01:00
break;
2022-06-21 07:12:39 +02:00
} else {
if (p.startsWith) {
2023-01-13 05:40:33 +01:00
if (parts[0] == null || !parts[0].startsWith(p.startsWith))
continue forEachRouteLoop;
2023-01-13 05:40:33 +01:00
props.set(
p.name,
safeURIDecode(parts[0].substring(p.startsWith.length)),
);
parts.shift();
} else {
if (parts[0]) {
props.set(p.name, safeURIDecode(parts[0]));
}
parts.shift();
2022-07-13 11:28:04 +02:00
}
2022-06-21 07:12:39 +02:00
}
}
}
if (parts.length === 0) {
if (route.children) {
const child = check(route.children, []);
if (child) {
return {
route,
props,
child,
};
} else {
2023-11-16 21:18:19 +01:00
continue;
}
}
if (route.hash != null && hash != null) {
props.set(route.hash, safeURIDecode(hash));
}
2023-01-13 05:40:33 +01:00
if (route.query != null && queryString != null) {
2024-04-23 15:30:55 +02:00
// const queryObject = [
// ...new URLSearchParams(queryString).entries(),
// ].reduce((obj, entry) => ({ ...obj, [entry[0]]: entry[1] }), {});
const queryObject: Record<string, string> = Object.assign(
{},
...[...new URLSearchParams(queryString).entries()].map(
(entry) => ({ [entry[0]]: entry[1] }),
),
);
2023-01-13 05:40:33 +01:00
for (const q in route.query) {
const as = route.query[q];
if (queryObject[q]) {
props.set(as, safeURIDecode(queryObject[q]));
}
}
}
2023-01-13 05:40:33 +01:00
return {
route,
props,
};
} else {
if (route.children) {
const child = check(route.children, parts);
if (child) {
return {
route,
props,
child,
};
} else {
}
} else {
}
}
}
return null;
}
2023-01-13 05:40:33 +01:00
const _parts = path.split("/").filter((part) => part.length !== 0);
return check(this.routes, _parts);
}
2023-01-13 05:40:33 +01:00
private navigate(
path: string,
key: string | null | undefined,
emitChange = true,
) {
const beforePath = this.currentPath;
this.currentPath = path;
const res = this.resolve(this.currentPath);
if (res == null) {
2023-01-13 05:40:33 +01:00
throw new Error(`no route found for: ${path}`);
}
if (res.route.loginRequired) {
2023-01-13 05:40:33 +01:00
pleaseLogin("/");
}
const isSamePath = beforePath === path;
2024-04-23 15:30:55 +02:00
// biome-ignore lint/style/noParameterAssign: assign it intentionally
if (isSamePath && key == null) key = this.currentKey;
this.current = res;
this.currentRef.value = res;
this.currentRoute.value = res.route;
this.currentKey = res.route.globalCacheKey ?? key ?? path;
if (emitChange) {
2023-01-13 05:40:33 +01:00
this.emit("change", {
beforePath,
path,
resolved: res,
key: this.currentKey,
});
}
return res;
}
public getCurrentPath() {
return this.currentPath;
}
public getCurrentKey() {
return this.currentKey;
}
2024-04-23 15:30:55 +02:00
public push(path: string, flag?: unknown) {
2022-07-05 15:25:27 +02:00
const beforePath = this.currentPath;
if (path === beforePath) {
2023-01-13 05:40:33 +01:00
this.emit("same");
2022-07-05 15:25:27 +02:00
return;
}
2022-06-28 10:59:23 +02:00
if (this.navHook) {
const cancel = this.navHook(path, flag);
2022-06-28 10:59:23 +02:00
if (cancel) return;
}
const res = this.navigate(path, null);
2023-01-13 05:40:33 +01:00
this.emit("push", {
beforePath,
path,
route: res.route,
props: res.props,
key: this.currentKey,
});
}
public replace(path: string, key?: string | null, emitEvent = true) {
this.navigate(path, key);
if (emitEvent) {
2023-01-13 05:40:33 +01:00
this.emit("replace", {
path,
key: this.currentKey,
});
}
}
}