路由在前端开发当中很常见,频繁配置也是一项比较劳累的工作。自动化注入路由的开发模式,可以大大减少开发的时间。本文借助vite的import.meta.glob对文件进行全局加载,这种方式大大提高开发效率。
同步加载模式
先看看完整的代码
const modules = import.meta.glob('../../views/**/*.vue', { eager: true, });
export const childrenRouteModuleList: RouteRecordRaw[] = [];
Object.keys(modules).forEach((file) => {
// @ts-ignore
const component = modules[file].default || {};
const route = component.route
if (route) {
const path = toLower((file.match(/^.*?views(.*?).vue$/) || [])[1] || "") // 生成路由path
// route meta 塞入默认的中文名称
route.meta = route.meta || {}
route.meta.title = route.meta.title || route.cname
component.name = component.name || path.split("/").join("")
const _route = {
name: component.name,
path: path,
component,
...route,
}
childrenRouteModuleList.push(_route)
}
});
const modules = import.meta.glob('../../views/**/*.vue', { eager: true, });
这种加载方式可以一次性加载所有的
vue
组件,但是很多情况下这个页面可能不是我们需要的页面。这个时候需要进行一些过滤。
过滤不必要的文件
指定特定文件名为入口页面
const modules = import.meta.glob('../../views/**/index.vue', { eager: true, });
例如这种写法,只有index.vue的页面被加载进来
在文件当中定义一个标识,用于识别是否为路由文件
ts
defineOptions({
route: {
meta: {
}
}
})
然后在循环操作当中,读取这个标识来过滤文件。
Object.keys(modules).forEach((file) => {
// @ts-ignore
const component = modules[file].default || {};
const route = component.route
if (route) {
// ... 如果route 存在,则定义一些信息
// route meta 塞入一些信息
route.meta = route.meta || {}
route.meta.title = route.meta.title || route.cname
}
});
同步方式比较好的地方是,我们在页面当中,直接进行定义一些路由信息,没有必要再操作单独的路由文件。不好的地方在于,打包之后,会将
vue
文件打包到一块,使得首次加载页面的时间拉长。
异步的方式
上边批量加载的函数去掉{ eager: true, }
这个配置,就变成异步加载了。
由于异步模块不会直接返回一个组件的原始信息,二是类似()=> import('xxx.vue')
这样的形式返回,我们再用defineOptions
的形式去单文件定义信息变得困难。这样以来如果需要自定义路由就有点困难了。这个时候,可以提供自定义路由的方式,来进行一些没办法实现的内容。
示例代码如下
↓ 会员用户可见内容 ↓
ts
const modules = import.meta.glob('../../views/**/index.vue');
// 自定义路由,自动识别目录下方xx.route.ts文件
const routes = import.meta.glob('../../views/**/*.route.ts', { eager: true, });
export const childrenRouteModuleList: RouteRecordRaw[] = [];
// 加入到路由集合中
Object.keys(routes).forEach((file) => {
// @ts-ignore
childrenRouteModuleList.push(routes[file].default || {})
})
Object.keys(modules).forEach((file) => {
// 根据文件名生成路径path
const path = toLower((file.match(/^.*?views\/(.*?)\/index.vue$/) || [])[1] || "")
const _route = {
// 路径驼峰命名生成路由名称
name: path.split("/").map((words, index) => index > 0 ? upperFirst(words) : words).join(""),
path: path,
component: modules[file] as Component,
}
childrenRouteModuleList.push(_route)
});
路由当中使用
js
export const RootRoute: RouteRecordRaw = {
path: '/',
name: 'Root',
redirect: PageEnum.BASE_HOME,
meta: {
title: 'Root',
},
component: LAYOUT,
children: childrenRouteModuleList,
}
export const basicRoutes = [
RootRoute,
// ... 其它路由文件导入
];
const router = createRouter({
history: createWebHistory(),
routes: basicRoutes as unknown as RouteRecordRaw[],
scrollBehavior(to, from, savedPosition) {
if (savedPosition) {
return savedPosition
} else {
return { top: 0 }
}
},
});