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

Logo

为开发者提供学习成长、分享交流、生态实践、资源工具等服务,帮助开发者快速成长。

更多推荐