Skip to content

Routing

Routes are owned by features, not by apps. The application shell composes each feature's routes export into a single router instance.

1. The Router Instance

@vh5/app-shell creates the router once. Adapter apps do not configure routes.

ts
// packages/app-shell/src/router/index.ts
import { createRouter, createWebHistory } from "vue-router";
import { mergeRouteModules } from "@vh5/utils";
import { authRoutes } from "@vh5/feature-auth";
import { homeRoutes } from "@vh5/feature-home";
import { productRoutes } from "@vh5/feature-product";
import { userRoutes } from "@vh5/feature-user";
import { BasicLayout } from "../layouts/BasicLayout.vue";

const featureRoutes = mergeRouteModules([...homeRoutes, ...productRoutes, ...userRoutes]);

export const router = createRouter({
  history: createWebHistory(import.meta.env.BASE_URL),
  routes: [
    { path: "/", component: BasicLayout, children: featureRoutes },
    ...authRoutes, // /login, /forgot
    { path: "/:pathMatch(.*)*", component: () => import("../views/NotFound.vue") },
  ],
});

2. Defining a Feature's Routes

ts
// packages/features/product/src/routes.ts
import type { RouteRecordRaw } from "vue-router";

export const productRoutes: RouteRecordRaw[] = [
  {
    path: "product",
    name: "product-list",
    component: () => import("./views/List.vue"),
    meta: {
      title: "Products",
      authority: ["user", "admin"], // role-based access
      tab: true, // show in BasicLayout tabbar
    },
  },
  {
    path: "product/:id",
    name: "product-detail",
    component: () => import("./views/Detail.vue"),
    meta: { title: "Product Detail", authority: ["user", "admin"] },
  },
];

meta keys recognised by the shell:

KeyTypePurpose
titlestringDocument title + navbar title
authoritystring[]Roles allowed to access (omit ⇒ public for authenticated)
publicbooleanAccessible without login
tabbooleanRender entry in the bottom tab bar
keepAlivebooleanWrap the view in <KeepAlive>

3. Permission Pipeline

A single global beforeEach guard is registered in @vh5/app-shell/router/guards.ts:

ts
router.beforeEach(async (to) => {
  startProgress();

  if (to.meta.public) return true;

  const auth = useAuthStore();
  if (!auth.isAuthenticated) {
    return { name: "login", query: { redirect: to.fullPath } };
  }

  if (Array.isArray(to.meta.authority) && !to.meta.authority.some(auth.hasRole)) {
    return { name: "forbidden" };
  }

  return true;
});

router.afterEach(() => stopProgress());

Frontend filtering is delegated to generateRoutesByFrontend() from @vh5/utils for screens that show a dynamic menu (e.g. an admin section).

4. Type-safe Names

unplugin-vue-router generates types/typed-router.d.ts. Inside features prefer named navigation:

ts
router.push({ name: "product-detail", params: { id } });

5. Backend-Driven Routes (Optional)

For apps that load their menu from the server:

ts
import { generateRoutesByBackend } from "@vh5/utils";
import { fetchMenuListAsync } from "@vh5/services";

const routes = await generateRoutesByBackend({
  fetchMenuListAsync,
  layoutMap: { BasicLayout },
  pageMap: import.meta.glob("@/features/**/views/*.vue"),
});

router.addRoute({ path: "/", component: BasicLayout, children: routes });

The same meta.authority filtering applies to backend routes.

6. Layout & Title

BasicLayout provides:

  • top navbar with meta.title
  • bottom tab bar built from routes whose meta.tab === true
  • a slot for the active view

Document title is updated by the shell:

ts
watchEffect(() => {
  const title = router.currentRoute.value.meta?.title as string | undefined;
  useTitle(title ? `${title} - Vue H5 Template` : "Vue H5 Template");
});

Released under the MIT License.