Vue3+Vite实现动态路由的详细实例代码
项目基本目录
1.首先定义初始默认的路由routes(router.js文件),vue文件使用import引入可以按需加载
import { createRouter, createWebHashHistory } from "vue-router"; import store from '../store/index.js' const routes = [{ path: "/login", component: () => import("../view/Login/index.vue"), children: [], meta: { title: '登录页', hideMenu: true, //加入hideMenu属性,不展示在侧边栏 }, name: "Login", }, { path: "/", component: () => import("../view/Home/index.vue"), meta: { keepalive: true, title: "主页", }, name: 'Home', // hideMenu: true,//不展示在侧边栏 children: [], redirect: '/index' }, ]
2.在store的login.js模块写入调用后端数据的函数(写在vuex的action对象中,方便全局异步调用)
Vuex 允许我们将 store 分割成模块(module),比如登录模块,项目各个业务不相关的模块。每个模块拥有自己的 state、mutation、action、getter、甚至是嵌套子模块(具体可以看主页另一篇博客)
3.执行addRoutes函数获取动态路由 (在router.js文件,点击登录成功后,在全局路由守卫去判断是否登录成功,再调用addRoutes函数)
(1)全局路由守卫逻辑具体可看注释,好处是进入项目之后,刷新页面路由守卫会拦截跳转,可以重新执行addRoutes()获取路由,先获取动态路由,再执行跳转,不然页面会报本地路由找不到(一个小坑)
(代码如下)
router.beforeEach(async (to, from, next) => { //路由守卫 if (store.state.login.token) { //存在token if (to.path == '/login') { //存在token,如果想跳转到登录页,默认有token跳转进项目首页 next({ path: '/' }) } else { //如果存在token,跳转进项目页面,则判断当前后端返回的路由有无长度 //或者有无即将跳转路由的name if (store.getters['login/getRoutes'].length || to.name != null) { next() //有的话直接跳转 } else { //不满足条件的话,重新请求后端动态路由数据 await addRoutes(); //addRoutes()必须加入await!!!!等待逻辑执行完毕获取路由 // 如果 addRoute 未完成,路由守卫会一层一层的执行执行,不加next()可能导致卡死! //直到 addRoute 完成,找到对应的路由 next({ ...to, replace: true }) } } } else { if (to.path == '/login') { next() } else { next('/login') } } })
后端返回的格式
(2)(重点在这,前面的步骤都可以不是重点)Vite使用import.meta.glob动态导入view文件夹所有前端页面,并调用addRoutes()函数执行获取动态路由数据并做处理(代码如下),主要作用是替换掉后端返回的component格式,再addRoute进路由表(看不懂的可以看代码注释或者可以搬进自己的项目打印看看每一步获取的数据)
let asyncRoutes = [] //定义数组接收后端返回的路由 const routeAllPathToCompMap =import.meta.glob(`../view/**/*.vue`); //**为通配符,vite不支持require导入方式,故用import.meta.glob(vite动态导入) /*import.meta.glob * 该方法匹配到的文件默认是懒加载,通过动态导入实现,构建时会分离独立的 chunk,是异步导入,返回的是 Promise * /*import.meta.globEager * 该方法是直接导入所有模块,并且是同步导入,返回结果直接通过 for...in循环就可以操作 */ async function addRoutes() { await store.dispatch('login/getNewRoutes').then((res) => { //获取后端返回的动态路由 if (res.data && res.data.length) { // let homeRouteChildren = []; asyncRoutes = res.data; /* * 1。拿到处理完毕home的children,最终替换掉原来的children,给菜单渲染提供支持 * 2.通过递归,调用router.addRoute,把每一项route插到对应的父route下 */ //服务端配置了路由,但前端还没添加对应文件的路由列表,内容是路由的component值(服务端的) // const unForList = [''] const homeChildren = routes[1].children; const dfs = (parentRouteName = 'Home', asyncRoutes = [], originRouteChildren = []) => { if (!Array.isArray(asyncRoutes)) return []; asyncRoutes.forEach(route => { // if (unForList.includes(route.component)) return; /**后端返回来的路由component是字符串,如component: "view/Index/index.vue", * 前端需要把component: "view/Index/index.vue" 转化为组件对象 * component:() => import("/src/view/Index/index.vue") **/ route.component = routeAllPathToCompMap[`../${route.component}`]; // route.component = () => import(`../${route.component}`); const routeChildren = route.children; router.addRoute(parentRouteName, route); route.children = dfs(route.name, routeChildren) originRouteChildren.push(route) }) return originRouteChildren } // homeRouteChildren = dfs(asyncRoutes) dfs('Home', asyncRoutes, homeChildren) } }); }
最终转化完成,路由数据格式如下
动态路由到此完成
总结
到此这篇关于Vue3+Vite实现动态路由的文章就介绍到这了,更多相关Vue3+Vite动态路由内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
相关文章
vue鼠标移入添加class样式,鼠标移出去除样式(active)实现方法
今天小编就为大家分享一篇vue鼠标移入添加class样式,鼠标移出去除样式(active)实现方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧2018-08-08vue+canvas实现数据实时从上到下刷新瀑布图效果(类似QT的)
这篇文章主要介绍了vue+canvas实现数据实时从上到下刷新瀑布图效果(类似QT的),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧2021-04-04解读element el-upload上传的附件名称不显示 file-list赋值
这篇文章主要介绍了解读element el-upload上传的附件名称不显示 file-list赋值问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教2022-10-10详解Vue的watch中的immediate与watch是什么意思
这篇文章主要介绍了详解Vue的watch中的immediate与watch是什么意思,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧2019-12-12
最新评论