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";
|
2022-06-20 10:38:49 +02:00
|
|
|
|
2024-04-12 05:31:11 +02:00
|
|
|
export interface RouteDef {
|
2022-06-20 10:38:49 +02:00
|
|
|
path: string;
|
|
|
|
component: Component;
|
|
|
|
query?: Record<string, string>;
|
2022-06-29 11:26:06 +02:00
|
|
|
loginRequired?: boolean;
|
2022-06-20 10:38:49 +02:00
|
|
|
name?: string;
|
2022-06-29 09:00:00 +02:00
|
|
|
hash?: string;
|
2022-06-20 10:38:49 +02:00
|
|
|
globalCacheKey?: string;
|
2022-07-20 12:59:27 +02:00
|
|
|
children?: RouteDef[];
|
2023-09-02 01:27:33 +02:00
|
|
|
}
|
2022-06-20 10:38:49 +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
|
|
|
}
|
2022-07-20 12:59:27 +02:00
|
|
|
|
2022-06-20 10:38:49 +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
|
2022-06-20 10:38:49 +02:00
|
|
|
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("?");
|
2022-06-20 10:38:49 +02:00
|
|
|
res.push({
|
2023-01-13 05:40:33 +01:00
|
|
|
name: placeholder.replace("(*)", "").replace("?", ""),
|
|
|
|
startsWith: prefix !== "" ? prefix : undefined,
|
2022-06-20 10:38:49 +02:00
|
|
|
wildcard,
|
|
|
|
optional,
|
|
|
|
});
|
2022-06-23 18:26:15 +02:00
|
|
|
} else if (part.length !== 0) {
|
2022-06-20 10:38:49 +02:00
|
|
|
res.push(part);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
export class Router extends EventEmitter<{
|
|
|
|
change: (ctx: {
|
|
|
|
beforePath: string;
|
|
|
|
path: string;
|
2022-07-20 12:59:27 +02:00
|
|
|
resolved: Resolved;
|
|
|
|
key: string;
|
|
|
|
}) => void;
|
2023-09-02 01:27:33 +02:00
|
|
|
replace: (ctx: { path: string; key: string }) => void;
|
2022-06-20 10:38:49 +02:00
|
|
|
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;
|
2022-06-20 10:38:49 +02:00
|
|
|
}> {
|
|
|
|
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>();
|
2022-06-20 10:38:49 +02:00
|
|
|
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;
|
2022-06-20 10:38:49 +02:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
constructor(routes: Router["routes"], currentPath: Router["currentPath"]) {
|
2022-06-20 10:38:49 +02:00
|
|
|
super();
|
|
|
|
|
|
|
|
this.routes = routes;
|
|
|
|
this.currentPath = currentPath;
|
2022-07-20 12:59:27 +02:00
|
|
|
this.navigate(currentPath, null, false);
|
2022-06-20 10:38:49 +02:00
|
|
|
}
|
|
|
|
|
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("#"));
|
2022-06-29 09:00:00 +02:00
|
|
|
}
|
2023-01-13 05:40:33 +01:00
|
|
|
if (path.includes("?")) {
|
|
|
|
queryString = path.substring(path.indexOf("?") + 1);
|
|
|
|
path = path.substring(0, path.indexOf("?"));
|
2022-06-20 10:38:49 +02:00
|
|
|
}
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
if (_DEV_) console.log("Routing: ", path, queryString);
|
2022-06-20 10:38:49 +02:00
|
|
|
|
2022-07-20 12:59:27 +02:00
|
|
|
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];
|
2022-07-20 12:59:27 +02:00
|
|
|
const props = new Map<string, string>();
|
2022-06-20 10:38:49 +02:00
|
|
|
|
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") {
|
2022-07-20 12:59:27 +02:00
|
|
|
if (p === parts[0]) {
|
|
|
|
parts.shift();
|
|
|
|
} else {
|
|
|
|
continue forEachRouteLoop;
|
2022-06-20 10:38:49 +02:00
|
|
|
}
|
|
|
|
} else {
|
2022-07-20 12:59:27 +02:00
|
|
|
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("/")));
|
2022-07-20 12:59:27 +02:00
|
|
|
parts = [];
|
|
|
|
}
|
2023-11-16 21:18:19 +01:00
|
|
|
break;
|
2022-06-21 07:12:39 +02:00
|
|
|
} else {
|
2022-07-20 12:59:27 +02:00
|
|
|
if (p.startsWith) {
|
2023-01-13 05:40:33 +01:00
|
|
|
if (parts[0] == null || !parts[0].startsWith(p.startsWith))
|
|
|
|
continue forEachRouteLoop;
|
2022-07-20 12:59:27 +02:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
props.set(
|
|
|
|
p.name,
|
|
|
|
safeURIDecode(parts[0].substring(p.startsWith.length)),
|
|
|
|
);
|
2022-07-20 12:59:27 +02:00
|
|
|
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
|
|
|
}
|
2022-06-20 10:38:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-20 12:59:27 +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;
|
2022-07-20 12:59:27 +02:00
|
|
|
}
|
|
|
|
}
|
2022-06-20 10:38:49 +02:00
|
|
|
|
2022-07-20 12:59:27 +02:00
|
|
|
if (route.hash != null && hash != null) {
|
|
|
|
props.set(route.hash, safeURIDecode(hash));
|
|
|
|
}
|
2023-01-13 05:40:33 +01:00
|
|
|
|
2022-07-20 12:59:27 +02: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
|
|
|
|
2022-07-20 12:59:27 +02: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
|
|
|
|
2022-07-20 12:59:27 +02:00
|
|
|
return {
|
|
|
|
route,
|
|
|
|
props,
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
if (route.children) {
|
|
|
|
const child = check(route.children, parts);
|
|
|
|
if (child) {
|
|
|
|
return {
|
|
|
|
route,
|
|
|
|
props,
|
|
|
|
child,
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
}
|
|
|
|
} else {
|
2022-06-20 10:38:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-06-29 09:00:00 +02:00
|
|
|
|
2022-07-20 12:59:27 +02:00
|
|
|
return null;
|
2022-06-20 10:38:49 +02:00
|
|
|
}
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
const _parts = path.split("/").filter((part) => part.length !== 0);
|
2022-07-20 12:59:27 +02:00
|
|
|
|
|
|
|
return check(this.routes, _parts);
|
2022-06-20 10:38:49 +02:00
|
|
|
}
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
private navigate(
|
|
|
|
path: string,
|
|
|
|
key: string | null | undefined,
|
|
|
|
emitChange = true,
|
|
|
|
) {
|
2022-06-20 10:38:49 +02:00
|
|
|
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}`);
|
2022-06-20 10:38:49 +02:00
|
|
|
}
|
|
|
|
|
2022-06-29 11:26:06 +02:00
|
|
|
if (res.route.loginRequired) {
|
2023-01-13 05:40:33 +01:00
|
|
|
pleaseLogin("/");
|
2022-06-29 11:26:06 +02:00
|
|
|
}
|
|
|
|
|
2022-06-20 10:38:49 +02:00
|
|
|
const isSamePath = beforePath === path;
|
2024-04-23 15:30:55 +02:00
|
|
|
// biome-ignore lint/style/noParameterAssign: assign it intentionally
|
2022-06-20 10:38:49 +02:00
|
|
|
if (isSamePath && key == null) key = this.currentKey;
|
2022-07-20 12:59:27 +02:00
|
|
|
this.current = res;
|
|
|
|
this.currentRef.value = res;
|
2022-06-20 10:38:49 +02:00
|
|
|
this.currentRoute.value = res.route;
|
2022-07-20 12:59:27 +02:00
|
|
|
this.currentKey = res.route.globalCacheKey ?? key ?? path;
|
2022-06-20 10:38:49 +02:00
|
|
|
|
2022-07-20 12:59:27 +02:00
|
|
|
if (emitChange) {
|
2023-01-13 05:40:33 +01:00
|
|
|
this.emit("change", {
|
2022-06-20 10:38:49 +02:00
|
|
|
beforePath,
|
|
|
|
path,
|
2022-07-20 12:59:27 +02:00
|
|
|
resolved: res,
|
2022-06-20 10:38:49 +02:00
|
|
|
key: this.currentKey,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-07-20 12:59:27 +02:00
|
|
|
return res;
|
2022-06-20 10:38:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
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) {
|
2022-07-16 22:12:22 +02:00
|
|
|
const cancel = this.navHook(path, flag);
|
2022-06-28 10:59:23 +02:00
|
|
|
if (cancel) return;
|
|
|
|
}
|
2022-07-20 12:59:27 +02:00
|
|
|
const res = this.navigate(path, null);
|
2023-01-13 05:40:33 +01:00
|
|
|
this.emit("push", {
|
2022-06-20 10:38:49 +02:00
|
|
|
beforePath,
|
|
|
|
path,
|
2022-07-20 12:59:27 +02:00
|
|
|
route: res.route,
|
|
|
|
props: res.props,
|
2022-06-20 10:38:49 +02:00
|
|
|
key: this.currentKey,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-07-20 12:59:27 +02:00
|
|
|
public replace(path: string, key?: string | null, emitEvent = true) {
|
2022-06-20 10:38:49 +02:00
|
|
|
this.navigate(path, key);
|
2022-07-20 12:59:27 +02:00
|
|
|
if (emitEvent) {
|
2023-01-13 05:40:33 +01:00
|
|
|
this.emit("replace", {
|
2022-07-20 12:59:27 +02:00
|
|
|
path,
|
|
|
|
key: this.currentKey,
|
|
|
|
});
|
|
|
|
}
|
2022-06-20 10:38:49 +02:00
|
|
|
}
|
|
|
|
}
|