You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
115 lines
2.9 KiB
115 lines
2.9 KiB
import { DialogManager } from "@components/Dialog/DialogManager.tsx";
|
|
import type { useAppStore, useMessageStore } from "@core/stores";
|
|
import ConfigPage from "@pages/Config/index.tsx";
|
|
import { Dashboard } from "@pages/Dashboard/index.tsx";
|
|
import MapPage from "@pages/Map/index.tsx";
|
|
import MessagesPage from "@pages/Messages.tsx";
|
|
import NodesPage from "@pages/Nodes/index.tsx";
|
|
import {
|
|
createRootRouteWithContext,
|
|
createRoute,
|
|
createRouter,
|
|
redirect,
|
|
} from "@tanstack/react-router";
|
|
import type { useTranslation } from "react-i18next";
|
|
import { z } from "zod/v4";
|
|
import { App } from "./App.tsx";
|
|
|
|
interface AppContext {
|
|
stores: {
|
|
app: ReturnType<typeof useAppStore>;
|
|
message: ReturnType<typeof useMessageStore>;
|
|
};
|
|
i18n: ReturnType<typeof useTranslation>;
|
|
}
|
|
|
|
export const rootRoute = createRootRouteWithContext<AppContext>()({
|
|
component: () => <App />,
|
|
});
|
|
|
|
const indexRoute = createRoute({
|
|
getParentRoute: () => rootRoute,
|
|
path: "/",
|
|
component: Dashboard,
|
|
loader: () => {
|
|
// Redirect to the broadcast messages page on initial load
|
|
return redirect({ to: "/messages/broadcast/0", replace: true });
|
|
},
|
|
});
|
|
|
|
const messagesRoute = createRoute({
|
|
getParentRoute: () => rootRoute,
|
|
path: "/messages",
|
|
component: MessagesPage,
|
|
beforeLoad: ({ params }) => {
|
|
const DEFAULT_CHANNEL = 0;
|
|
|
|
if (Object.values(params).length === 0) {
|
|
throw redirect({
|
|
to: `/messages/broadcast/${DEFAULT_CHANNEL}`,
|
|
replace: true,
|
|
});
|
|
}
|
|
},
|
|
});
|
|
|
|
export const messagesWithParamsRoute = createRoute({
|
|
getParentRoute: () => rootRoute,
|
|
path: "/messages/$type/$chatId",
|
|
component: MessagesPage,
|
|
parseParams: (params) => ({
|
|
type: z
|
|
.enum(["direct", "broadcast"])
|
|
.refine((val) => val === "direct" || val === "broadcast", {
|
|
message: 'Type must be "direct" or "broadcast".',
|
|
})
|
|
.parse(params.type),
|
|
chatId: z.coerce.number().int().min(0).max(4294967294).parse(params.chatId), // max is 0xffffffff - 1
|
|
}),
|
|
});
|
|
|
|
const mapRoute = createRoute({
|
|
getParentRoute: () => rootRoute,
|
|
path: "/map",
|
|
component: MapPage,
|
|
});
|
|
|
|
const configRoute = createRoute({
|
|
getParentRoute: () => rootRoute,
|
|
path: "/config",
|
|
component: ConfigPage,
|
|
});
|
|
|
|
const nodesRoute = createRoute({
|
|
getParentRoute: () => rootRoute,
|
|
path: "/nodes",
|
|
component: NodesPage,
|
|
});
|
|
|
|
const dialogWithParamsRoute = createRoute({
|
|
getParentRoute: () => rootRoute,
|
|
path: "/dialog/$dialogId",
|
|
component: DialogManager,
|
|
});
|
|
|
|
const routeTree = rootRoute.addChildren([
|
|
indexRoute,
|
|
messagesRoute,
|
|
messagesWithParamsRoute,
|
|
mapRoute,
|
|
configRoute,
|
|
nodesRoute,
|
|
dialogWithParamsRoute,
|
|
]);
|
|
|
|
const router = createRouter({
|
|
routeTree,
|
|
context: {
|
|
stores: {
|
|
app: {} as ReturnType<typeof useAppStore>,
|
|
message: {} as ReturnType<typeof useMessageStore>,
|
|
},
|
|
i18n: {} as ReturnType<typeof import("react-i18next").useTranslation>,
|
|
},
|
|
});
|
|
export { router };
|
|
|