feat: support the dynamic introduction and sorting of routes
parent
c5eb0841a5
commit
30f7472d26
|
@ -10,11 +10,14 @@ import { useTitle } from '@vueuse/core';
|
||||||
|
|
||||||
import { dynamicRoutes } from '@/router/routes';
|
import { dynamicRoutes } from '@/router/routes';
|
||||||
|
|
||||||
|
// 不需要权限的页面白名单
|
||||||
|
const WHITE_ROUTE_NAMES = new Set<string>([]);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 通用守卫配置
|
* 通用守卫配置
|
||||||
* @param router
|
* @param router
|
||||||
*/
|
*/
|
||||||
function configCommonGuard(router: Router) {
|
function setupCommonGuard(router: Router) {
|
||||||
// 记录已经加载的页面
|
// 记录已经加载的页面
|
||||||
const loadedPaths = new Set<string>();
|
const loadedPaths = new Set<string>();
|
||||||
|
|
||||||
|
@ -44,28 +47,11 @@ function configCommonGuard(router: Router) {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// 不需要权限的页面白名单
|
|
||||||
const WHITE_ROUTE_NAMES = new Set<string>([]);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 跳转登录页面
|
|
||||||
* @param to
|
|
||||||
*/
|
|
||||||
function loginPageMeta(to: RouteLocationNormalized) {
|
|
||||||
return {
|
|
||||||
path: LOGIN_PATH,
|
|
||||||
// 如不需要,直接删除 query
|
|
||||||
query: { redirect: encodeURIComponent(to.fullPath) },
|
|
||||||
// 携带当前跳转的页面,登录后重新跳转该页面
|
|
||||||
replace: true,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 权限访问守卫配置
|
* 权限访问守卫配置
|
||||||
* @param router
|
* @param router
|
||||||
*/
|
*/
|
||||||
function configAccessGuard(router: Router) {
|
function setupAccessGuard(router: Router) {
|
||||||
router.beforeEach(async (to, from) => {
|
router.beforeEach(async (to, from) => {
|
||||||
const accessStore = useAccessStore();
|
const accessStore = useAccessStore();
|
||||||
const accessToken = accessStore.getAccessToken;
|
const accessToken = accessStore.getAccessToken;
|
||||||
|
@ -123,7 +109,19 @@ function configAccessGuard(router: Router) {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
export { configAccessGuard };
|
/**
|
||||||
|
* 登录页面信息
|
||||||
|
* @param to
|
||||||
|
*/
|
||||||
|
function loginPageMeta(to: RouteLocationNormalized) {
|
||||||
|
return {
|
||||||
|
path: LOGIN_PATH,
|
||||||
|
// 如不需要,直接删除 query
|
||||||
|
query: { redirect: encodeURIComponent(to.fullPath) },
|
||||||
|
// 携带当前跳转的页面,登录后重新跳转该页面
|
||||||
|
replace: true,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 项目守卫配置
|
* 项目守卫配置
|
||||||
|
@ -131,9 +129,9 @@ export { configAccessGuard };
|
||||||
*/
|
*/
|
||||||
function createRouterGuard(router: Router) {
|
function createRouterGuard(router: Router) {
|
||||||
/** 通用 */
|
/** 通用 */
|
||||||
configCommonGuard(router);
|
setupCommonGuard(router);
|
||||||
/** 权限访问 */
|
/** 权限访问 */
|
||||||
configAccessGuard(router);
|
setupAccessGuard(router);
|
||||||
}
|
}
|
||||||
|
|
||||||
export { createRouterGuard };
|
export { createRouterGuard };
|
||||||
|
|
|
@ -4,7 +4,7 @@ import { traverseTreeValues } from '@vben/utils';
|
||||||
import { createRouter, createWebHashHistory } from 'vue-router';
|
import { createRouter, createWebHashHistory } from 'vue-router';
|
||||||
|
|
||||||
import { createRouterGuard } from './guard';
|
import { createRouterGuard } from './guard';
|
||||||
import { staticRoutes } from './routes';
|
import { routes } from './routes';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @zh_CN 创建vue-router实例
|
* @zh_CN 创建vue-router实例
|
||||||
|
@ -12,16 +12,14 @@ import { staticRoutes } from './routes';
|
||||||
const router = createRouter({
|
const router = createRouter({
|
||||||
history: createWebHashHistory(import.meta.env.VITE_PUBLIC_PATH),
|
history: createWebHashHistory(import.meta.env.VITE_PUBLIC_PATH),
|
||||||
// 应该添加到路由的初始路由列表。
|
// 应该添加到路由的初始路由列表。
|
||||||
routes: staticRoutes,
|
routes,
|
||||||
scrollBehavior: (to, from, savedPosition) => {
|
scrollBehavior: (_to, _from, savedPosition) => {
|
||||||
if (to.path !== from.path) {
|
// if (to.path !== from.path) {
|
||||||
setTimeout(() => {
|
// const app = document.querySelector('#app');
|
||||||
const app = document.querySelector('#app');
|
// if (app) {
|
||||||
if (app) {
|
// app.scrollTop = 0;
|
||||||
app.scrollTop = 0;
|
// }
|
||||||
}
|
// }
|
||||||
});
|
|
||||||
}
|
|
||||||
return savedPosition || { left: 0, top: 0 };
|
return savedPosition || { left: 0, top: 0 };
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
@ -34,9 +32,9 @@ function resetRoutes() {
|
||||||
const staticRouteNames = traverseTreeValues<
|
const staticRouteNames = traverseTreeValues<
|
||||||
RouteRecordRaw,
|
RouteRecordRaw,
|
||||||
RouteRecordName | undefined
|
RouteRecordName | undefined
|
||||||
>(staticRoutes, (route) => {
|
>(routes, (route) => {
|
||||||
// 这些路由需要指定 name,防止在路由重置时,不能删除没有指定 name 的路由
|
// 这些路由需要指定 name,防止在路由重置时,不能删除没有指定 name 的路由
|
||||||
if (!route.name) {
|
if (import.meta.env.DEV && !route.name) {
|
||||||
console.warn(
|
console.warn(
|
||||||
`The route with the path ${route.path} needs to specify the field name.`,
|
`The route with the path ${route.path} needs to specify the field name.`,
|
||||||
);
|
);
|
||||||
|
@ -45,8 +43,8 @@ function resetRoutes() {
|
||||||
});
|
});
|
||||||
|
|
||||||
const { getRoutes, hasRoute, removeRoute } = router;
|
const { getRoutes, hasRoute, removeRoute } = router;
|
||||||
const routes = getRoutes();
|
const allRoutes = getRoutes();
|
||||||
routes.forEach(({ name }) => {
|
allRoutes.forEach(({ name }) => {
|
||||||
// 存在于路由表且非白名单才需要删除
|
// 存在于路由表且非白名单才需要删除
|
||||||
if (name && !staticRouteNames.includes(name) && hasRoute(name)) {
|
if (name && !staticRouteNames.includes(name) && hasRoute(name)) {
|
||||||
removeRoute(name);
|
removeRoute(name);
|
||||||
|
|
|
@ -2,7 +2,7 @@ import type { RouteRecordRaw } from 'vue-router';
|
||||||
|
|
||||||
import { BasicLayout } from '@/layouts';
|
import { BasicLayout } from '@/layouts';
|
||||||
|
|
||||||
export const nestedRoutes: RouteRecordRaw[] = [
|
const routes: RouteRecordRaw[] = [
|
||||||
{
|
{
|
||||||
component: BasicLayout,
|
component: BasicLayout,
|
||||||
meta: {
|
meta: {
|
||||||
|
@ -69,3 +69,5 @@ export const nestedRoutes: RouteRecordRaw[] = [
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
|
export default routes;
|
|
@ -2,7 +2,7 @@ import type { RouteRecordRaw } from 'vue-router';
|
||||||
|
|
||||||
import { BasicLayout, IFrameView } from '@/layouts';
|
import { BasicLayout, IFrameView } from '@/layouts';
|
||||||
|
|
||||||
export const outsideRoutes: RouteRecordRaw[] = [
|
const routes: RouteRecordRaw[] = [
|
||||||
{
|
{
|
||||||
component: BasicLayout,
|
component: BasicLayout,
|
||||||
meta: {
|
meta: {
|
||||||
|
@ -35,3 +35,5 @@ export const outsideRoutes: RouteRecordRaw[] = [
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
|
export default routes;
|
|
@ -2,11 +2,12 @@ import type { RouteRecordRaw } from 'vue-router';
|
||||||
|
|
||||||
import { BasicLayout } from '@/layouts';
|
import { BasicLayout } from '@/layouts';
|
||||||
|
|
||||||
const rootRoutes: RouteRecordRaw[] = [
|
const routes: RouteRecordRaw[] = [
|
||||||
{
|
{
|
||||||
component: BasicLayout,
|
component: BasicLayout,
|
||||||
meta: {
|
meta: {
|
||||||
hideChildrenInMenu: true,
|
hideChildrenInMenu: true,
|
||||||
|
orderNo: -1,
|
||||||
title: '首页',
|
title: '首页',
|
||||||
},
|
},
|
||||||
name: 'Home',
|
name: 'Home',
|
||||||
|
@ -26,4 +27,4 @@ const rootRoutes: RouteRecordRaw[] = [
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
export { rootRoutes };
|
export default routes;
|
|
@ -5,7 +5,7 @@ import { BasicLayout, IFrameView } from '@/layouts';
|
||||||
import { VBEN_GITHUB_URL } from '@vben/constants';
|
import { VBEN_GITHUB_URL } from '@vben/constants';
|
||||||
import { $t } from '@vben/locales/helper';
|
import { $t } from '@vben/locales/helper';
|
||||||
|
|
||||||
export const vbenRoutes: RouteRecordRaw[] = [
|
const routes: RouteRecordRaw[] = [
|
||||||
{
|
{
|
||||||
component: BasicLayout,
|
component: BasicLayout,
|
||||||
meta: {
|
meta: {
|
||||||
|
@ -49,3 +49,5 @@ export const vbenRoutes: RouteRecordRaw[] = [
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
|
export default routes;
|
|
@ -1,24 +1,28 @@
|
||||||
|
import { mergeRouteModules } from '@vben-core/helpers';
|
||||||
import type { RouteRecordRaw } from 'vue-router';
|
import type { RouteRecordRaw } from 'vue-router';
|
||||||
|
|
||||||
import { essentialRoutes } from './_essential';
|
import { essentialRoutes } from './_essential';
|
||||||
import { nestedRoutes } from './modules/nested';
|
|
||||||
import { outsideRoutes } from './modules/outside';
|
const dynamicRouteFiles = import.meta.glob('./dynamic/**/*.ts', {
|
||||||
import { rootRoutes } from './modules/root';
|
eager: true,
|
||||||
import { vbenRoutes } from './modules/vben';
|
});
|
||||||
|
|
||||||
|
const staticRouteFiles = import.meta.glob('./static/**/*.ts', { eager: true });
|
||||||
|
|
||||||
|
const externalRouteFiles = import.meta.glob('./external/**/*.ts', {
|
||||||
|
eager: true,
|
||||||
|
});
|
||||||
|
|
||||||
/** 动态路由 */
|
/** 动态路由 */
|
||||||
const dynamicRoutes: RouteRecordRaw[] = [
|
const dynamicRoutes: RouteRecordRaw[] = mergeRouteModules(dynamicRouteFiles);
|
||||||
// 根路由
|
|
||||||
...rootRoutes,
|
|
||||||
...nestedRoutes,
|
|
||||||
...outsideRoutes,
|
|
||||||
...vbenRoutes,
|
|
||||||
];
|
|
||||||
|
|
||||||
/** 排除在主框架外的路由,这些路由没有菜单和顶部及其他框架内容 */
|
|
||||||
const externalRoutes: RouteRecordRaw[] = [];
|
|
||||||
|
|
||||||
/** 静态路由列表,访问这些页面可以不需要权限 */
|
/** 静态路由列表,访问这些页面可以不需要权限 */
|
||||||
const staticRoutes: RouteRecordRaw[] = [...essentialRoutes];
|
const staticRoutes: RouteRecordRaw[] = mergeRouteModules(staticRouteFiles);
|
||||||
|
|
||||||
export { dynamicRoutes, externalRoutes, staticRoutes };
|
/** 排除在主框架外的路由,这些路由没有菜单和顶部及其他框架内容 */
|
||||||
|
const externalRoutes: RouteRecordRaw[] = mergeRouteModules(externalRouteFiles);
|
||||||
|
|
||||||
|
/** 路由列表,由基本路由+静态路由组成 */
|
||||||
|
const routes: RouteRecordRaw[] = [...essentialRoutes, ...staticRoutes];
|
||||||
|
|
||||||
|
export { dynamicRoutes, externalRoutes, routes };
|
||||||
|
|
|
@ -1,8 +1,16 @@
|
||||||
import { describe, expect, it, vi } from 'vitest';
|
import { describe, expect, it, vi } from 'vitest';
|
||||||
|
|
||||||
import { generatorMenus } from './generator-menus'; // 替换为您的实际路径
|
import { generatorMenus } from './generator-menus'; // 替换为您的实际路径
|
||||||
import type { RouteRecordRaw } from 'vue-router';
|
import {
|
||||||
|
type RouteRecordRaw,
|
||||||
|
type Router,
|
||||||
|
createRouter,
|
||||||
|
createWebHistory,
|
||||||
|
} from 'vue-router';
|
||||||
|
|
||||||
|
// Nested route setup to test child inclusion and hideChildrenInMenu functionality
|
||||||
|
|
||||||
|
describe('generatorMenus', () => {
|
||||||
// 模拟路由数据
|
// 模拟路由数据
|
||||||
const mockRoutes = [
|
const mockRoutes = [
|
||||||
{
|
{
|
||||||
|
@ -33,9 +41,6 @@ const mockRouter = {
|
||||||
]),
|
]),
|
||||||
};
|
};
|
||||||
|
|
||||||
// Nested route setup to test child inclusion and hideChildrenInMenu functionality
|
|
||||||
|
|
||||||
describe('generatorMenus', () => {
|
|
||||||
it('the correct menu list should be generated according to the route', async () => {
|
it('the correct menu list should be generated according to the route', async () => {
|
||||||
const expectedMenus = [
|
const expectedMenus = [
|
||||||
{
|
{
|
||||||
|
@ -138,8 +143,6 @@ describe('generatorMenus', () => {
|
||||||
mockRoutesWithRedirect,
|
mockRoutesWithRedirect,
|
||||||
mockRouter as any,
|
mockRouter as any,
|
||||||
);
|
);
|
||||||
console.log(111, menus);
|
|
||||||
|
|
||||||
expect(menus).toEqual([
|
expect(menus).toEqual([
|
||||||
// Assuming your generatorMenus function excludes redirect routes from the menu
|
// Assuming your generatorMenus function excludes redirect routes from the menu
|
||||||
{
|
{
|
||||||
|
@ -168,4 +171,60 @@ describe('generatorMenus', () => {
|
||||||
},
|
},
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const routes: any = [
|
||||||
|
{
|
||||||
|
meta: { orderNo: 2, title: 'Home' },
|
||||||
|
name: 'home',
|
||||||
|
path: '/',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
meta: { orderNo: 1, title: 'About' },
|
||||||
|
name: 'about',
|
||||||
|
path: '/about',
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const router: Router = createRouter({
|
||||||
|
history: createWebHistory(),
|
||||||
|
routes,
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should generate menu list with correct order', async () => {
|
||||||
|
const menus = await generatorMenus(routes, router);
|
||||||
|
const expectedMenus = [
|
||||||
|
{
|
||||||
|
badge: undefined,
|
||||||
|
badgeType: undefined,
|
||||||
|
badgeVariants: undefined,
|
||||||
|
icon: undefined,
|
||||||
|
name: 'About',
|
||||||
|
orderNo: 1,
|
||||||
|
parent: undefined,
|
||||||
|
parents: undefined,
|
||||||
|
path: '/about',
|
||||||
|
children: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
badge: undefined,
|
||||||
|
badgeType: undefined,
|
||||||
|
badgeVariants: undefined,
|
||||||
|
icon: undefined,
|
||||||
|
name: 'Home',
|
||||||
|
orderNo: 2,
|
||||||
|
parent: undefined,
|
||||||
|
parents: undefined,
|
||||||
|
path: '/',
|
||||||
|
children: [],
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
expect(menus).toEqual(expectedMenus);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should handle empty routes', async () => {
|
||||||
|
const emptyRoutes: any[] = [];
|
||||||
|
const menus = await generatorMenus(emptyRoutes, router);
|
||||||
|
expect(menus).toEqual([]);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -17,7 +17,7 @@ async function generatorMenus(
|
||||||
router.getRoutes().map(({ name, path }) => [name, path]),
|
router.getRoutes().map(({ name, path }) => [name, path]),
|
||||||
);
|
);
|
||||||
|
|
||||||
const menus = mapTree<ExRouteRecordRaw, MenuRecordRaw>(routes, (route) => {
|
let menus = mapTree<ExRouteRecordRaw, MenuRecordRaw>(routes, (route) => {
|
||||||
// 路由表的路径写法有多种,这里从router获取到最终的path并赋值
|
// 路由表的路径写法有多种,这里从router获取到最终的path并赋值
|
||||||
const path = finalRoutesMap[route.name as string] ?? route.path;
|
const path = finalRoutesMap[route.name as string] ?? route.path;
|
||||||
|
|
||||||
|
@ -65,6 +65,8 @@ async function generatorMenus(
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// 对菜单进行排序
|
||||||
|
menus = menus.sort((a, b) => (a.orderNo || 999) - (b.orderNo || 999));
|
||||||
return menus;
|
return menus;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
export * from './flatten-object';
|
export * from './flatten-object';
|
||||||
export * from './generator-menus';
|
export * from './generator-menus';
|
||||||
export * from './generator-routes';
|
export * from './generator-routes';
|
||||||
|
export * from './merge-route-modules';
|
||||||
export * from './nested-object';
|
export * from './nested-object';
|
||||||
|
|
|
@ -0,0 +1,68 @@
|
||||||
|
import type { RouteRecordRaw } from 'vue-router';
|
||||||
|
|
||||||
|
import { describe, expect, it } from 'vitest';
|
||||||
|
|
||||||
|
import { mergeRouteModules } from './merge-route-modules';
|
||||||
|
|
||||||
|
import type { RouteModuleType } from './merge-route-modules';
|
||||||
|
|
||||||
|
describe('mergeRouteModules', () => {
|
||||||
|
it('should merge route modules correctly', () => {
|
||||||
|
const routeModules: Record<string, RouteModuleType> = {
|
||||||
|
'./dynamic-routes/about.ts': {
|
||||||
|
default: [
|
||||||
|
{
|
||||||
|
component: () => Promise.resolve({ template: '<div>About</div>' }),
|
||||||
|
name: 'About',
|
||||||
|
path: '/about',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
'./dynamic-routes/home.ts': {
|
||||||
|
default: [
|
||||||
|
{
|
||||||
|
component: () => Promise.resolve({ template: '<div>Home</div>' }),
|
||||||
|
name: 'Home',
|
||||||
|
path: '/',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
const expectedRoutes: RouteRecordRaw[] = [
|
||||||
|
{
|
||||||
|
component: expect.any(Function),
|
||||||
|
name: 'About',
|
||||||
|
path: '/about',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
component: expect.any(Function),
|
||||||
|
name: 'Home',
|
||||||
|
path: '/',
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const mergedRoutes = mergeRouteModules(routeModules);
|
||||||
|
expect(mergedRoutes).toEqual(expectedRoutes);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should handle empty modules', () => {
|
||||||
|
const routeModules: Record<string, RouteModuleType> = {};
|
||||||
|
const expectedRoutes: RouteRecordRaw[] = [];
|
||||||
|
|
||||||
|
const mergedRoutes = mergeRouteModules(routeModules);
|
||||||
|
expect(mergedRoutes).toEqual(expectedRoutes);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should handle modules with no default export', () => {
|
||||||
|
const routeModules: Record<string, RouteModuleType> = {
|
||||||
|
'./dynamic-routes/empty.ts': {
|
||||||
|
default: [],
|
||||||
|
},
|
||||||
|
};
|
||||||
|
const expectedRoutes: RouteRecordRaw[] = [];
|
||||||
|
|
||||||
|
const mergedRoutes = mergeRouteModules(routeModules);
|
||||||
|
expect(mergedRoutes).toEqual(expectedRoutes);
|
||||||
|
});
|
||||||
|
});
|
|
@ -0,0 +1,28 @@
|
||||||
|
import type { RouteRecordRaw } from 'vue-router';
|
||||||
|
|
||||||
|
// 定义模块类型
|
||||||
|
interface RouteModuleType {
|
||||||
|
default: RouteRecordRaw[];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 合并动态路由模块的默认导出
|
||||||
|
* @param routeModules 动态导入的路由模块对象
|
||||||
|
* @returns 合并后的路由配置数组
|
||||||
|
*/
|
||||||
|
function mergeRouteModules(
|
||||||
|
routeModules: Record<string, unknown>,
|
||||||
|
): RouteRecordRaw[] {
|
||||||
|
const mergedRoutes: RouteRecordRaw[] = [];
|
||||||
|
|
||||||
|
for (const routeModule of Object.values(routeModules)) {
|
||||||
|
const moduleRoutes = (routeModule as RouteModuleType)?.default ?? [];
|
||||||
|
mergedRoutes.push(...moduleRoutes);
|
||||||
|
}
|
||||||
|
|
||||||
|
return mergedRoutes;
|
||||||
|
}
|
||||||
|
|
||||||
|
export { mergeRouteModules };
|
||||||
|
|
||||||
|
export type { RouteModuleType };
|
|
@ -42,7 +42,6 @@ interface RouteMeta {
|
||||||
* @default false
|
* @default false
|
||||||
*/
|
*/
|
||||||
hideInMenu?: boolean;
|
hideInMenu?: boolean;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 当前路由在标签页不展现
|
* 当前路由在标签页不展现
|
||||||
* @default false
|
* @default false
|
||||||
|
|
Loading…
Reference in New Issue