vue2 创建动态路由
1、在登陆页面使用localstorage。setItem方法,在登陆成功后获取token、menus,其中menus中的某个字段含有登陆后的所有路径,根据自己项目进行修改即可。2、在src/views路径下创建src/views/404/index.vue准备一个404页面。3、在src下创建layout/index.vue作为首页。这些路径根据自己项目进行改动即可。
·
1.准备工作
1、在登陆页面使用localstorage。setItem方法,在登陆成功后获取token、menus,其中menus中的某个字段含有登陆后的所有路径,根据自己项目进行修改即可。
2、在src/views路径下创建src/views/404/index.vue准备一个404页面。
3、在src下创建layout/index.vue作为首页
这些路径根据自己项目进行改动即可
2.在vuex页面
export default new Vuex.Store({
state: {
routerList: []
},
getters: {
},
mutations: {
Set_RouterList (state, data) {
state.routerList = data
},
},
actions: {
},
modules
})
3.在src/utils/中创建.js文件,文件名随自己喜欢就行,我的是permission.js
// 这里的Layout是我项目的首页
import Layout from '@/layout/index'
// 转换标准路由
export const syncRouter = (data) => {
if (!Array.isArray(data)) return []
// console.log(data)
data.forEach((item) => {
// 如果item.component是名为'Layout'的字符串,需要给他改成Layout组件来进行导入,不能直接是字符串
if (item.component === 'Layout') {
item.component = Layout
} else {
// 如果是其他情况,需要用下方的loadComponent()方法也改成对应的导入组件
item.component = loadComponent(item.component)
}
if (item.children?.length) {
// 函数递归
syncRouter(item.children)
}
})
return data
}
const loadComponent = (path) => {
// 这里不能用import(`@/……`) 用了会报错,查资料后发现可以使用require
return resolve => (require([`@/views/${path}`], resolve))
}
4.在router/index.js页面
关于 router.matcher = new VueRouter({}).matcher 参考文章
vue-router 还给路由排了序?解析路由匹配,vue-router Matcher 解析(三)-阿里云开发者社区
import Vue from 'vue'
import VueRouter from 'vue-router'
import Layout from '@/layout/index'
import Login from '../views/login.vue'
import Cookies from 'js-cookie'
import { syncRouter } from '@/utils/permission.js'
import store from '@/store'
import NotFound from '@/views/404/index.vue'
Vue.use(VueRouter)
// 如果有报错,加入如下
const originalPush = VueRouter.prototype.push
VueRouter.prototype.push = function push (location) {
return originalPush.call(this, location).catch((err) => err)
}
const routes = [
{
path: '/login',
name: 'Login',
component: Login
},
{
path: '/',
name: 'Layout',
component: Layout,
redirect: '/dashboard',
children: [
{
path: '/dashboard',
name: 'Dashboard',
component: Home,
meta: {
title: '首页'
}
}
]
}
]
const router = new VueRouter({
routes
})
// 路由钩子
router.beforeEach((to, from, next) => {
const token = Cookies.get('token')
// console.log(token)
if (token) {
if (to.path === '/login') {
if (from.path === '/login') {
next('/')
} else {
next(from.path)
}
} else {
if (!store.state.routerList.length) {
loadMenus(to, next)
} else {
next()
}
}
} else {
Cookies.remove('token')
localStorage.removeItem('user')
store.commit('Set_RouterList', [])
// window.location.reload() 这种方法时刷新页面,了解一下~
if (to.path !== '/login') {
next('/login')
} else {
next()
}
}
}
)
// 合并路由方法
const loadMenus = (to, next) => {
const menus = JSON.parse(localStorage.getItem('menus'))
console.log(menus)
if (!menus) {
Cookies.remove('token')
localStorage.removeItem('user')
// 退出
return next('/login')
}
// 解析后的路由
const newRouter = syncRouter(menus)
// 存储路由
store.commit('Set_RouterList', newRouter)
// 把解析后的路由合并到routes router.addRoute() 方法
router.matcher = new VueRouter({}).matcher
// 创建一个404页面
const not404 = {
path: '*',
name: 'NotFound',
component: NotFound
}
// 把所有的路由组合成一个数组
const routerAll = [...newRouter, ...routes, not404]
routerAll.forEach(route => {
// 把数组中的每一项添加到导出的router上
router.addRoute(route)
})
next({ ...to })
}
export default router
更多推荐
已为社区贡献1条内容
所有评论(0)