vue3+vite动态加载路由,本地路由和线上路由匹配方式

 更新时间:2023年06月28日 15:14:00   作者:arno-cai  
这篇文章主要介绍了vue3+vite动态加载路由,本地路由和线上路由匹配方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

使用场景

本地全路由动态匹配线上路由

目的

本地路由方便开发配置,匹配线上路由,进行页面路由控制

内容

// router/constRoutes.js
// ---------------------------------------
// constRoutes 固定需要加载的路由,
// notFound 动态路由加载完成后再添加,否则刷新页面,此时还没有路由,会直接定向到notFound页面
// ---------------------------------------
export const constRoutes = [
  {
    path: '/',
    redirect: '/home'
  },
  {
    path: '/home',
    name: 'home',
    component: () => import('@/views/home/index.vue')
  },
  {
    path: '/login',
    name: 'login',
    component: () => import('@/views/login/index.vue')
  }
]
export const notFound = {
  path: '/:pathMatch(.*)*',
  name: 'notFound',
  component: () => import('@/views/not-found/not-found.vue')
}
// router/asyncRoutes.js
// ---------------------------------------
// 注意:children下的path不需要加'/'
// meta下可以加各种本地你想要的参数
// ---------------------------------------
const Layout = () => import('@/layout/index.vue')
export const asyncRoutes = [
  {
    path: '/monitoringCenter',
    name: 'MonitoringCenter',
    component: Layout,
    meta: { title: '监控中心' },
    children: [
      {
        path: 'carMonitoring',
        name: 'CarMonitoring',
        meta: { title: '车辆监控' },
        children: [
          {
            path: 'positioning',
            name: 'Positioning',
            component: () => import('@/views/monitoringCenter/positioning/index.vue'),
            meta: { title: '实时定位' }
          }
        ]
      },
      {
        path: 'monitorSetting',
        name: 'MonitorSetting',
        meta: { title: '监控设置' },
        children: [
          {
            path: 'ruleManagement',
            name: 'RuleManagement',
            component: () => import('@/views/monitoringCenter/ruleManagement/index.vue'),
            meta: { title: '报警规则' }
          }
        ]
      }
    ]
  },
// router/index.js
// ---------------------------------------
// 先加载constRoutes 固定路由
// ---------------------------------------
import { createRouter, createWebHistory } from 'vue-router'
import { constRoutes } from '@/router/constRoutes'
const router = createRouter({
  history: createWebHistory(import.meta.env.BASE_URL),
  routes: constRoutes
})
export default router
// utils/map-menus.js
// ---------------------------------------
//导出方法,menuList是线上路由,Routes第一次默认是本地路由
// ---------------------------------------
import { asyncRoutes } from '@/router/asyncRoutes'
// 添加redirect重定向
let redirect = ''
// 匹配路由,添加对应线上传回来的参数
export function mapMenusToRoutes(menuList, Routes = asyncRoutes) {
  const routes = []
  for (let item of Routes) {
    const m = menuList.find((menu) => item.name == menu.name)
    if (m) {
    // 线上按钮权限放在attr中
      m.attr && m.attr.length && (item.attr = m.attr)
      item.id = m.id
      // ‘?.'判断是否有children且大于零
      if (item.children?.length) {
          // 添加重定向
        if (item.path.includes('/')) {
          redirect = item.path
          item.redirect = `${redirect}/${item.children[0].path}`
        } else {
          item.redirect = `${redirect}/${item.path}/${item.children[0].path}`
        }
        // 如果有children 则回调并赋值
        item.children = mapMenusToRoutes(m.children, item.children)
      }
      routes.push(item)
    }
  }
  return routes
}

&&&&后面是具体调用方式,因人而异,可自行选择

// stores/login.js
import { defineStore } from 'pinia'
import router from '@/router'
import { notFound } from '@/router/constRoutes'
import localCache from '@/utils/cache'
import { mapMenusToRoutes } from '@/utils/map-menus'
import { login, getInfo } from '@/api/user/login'
export const useLoginStore = defineStore({
  id: 'useLoginStore',
  state: () => ({
    token: '',
    menuList: [],
    routeMenus: []
  }),
  getters: {
    routeMenusGet() {
      return this.routeMenus
    }
  },
  actions: {
    async LoginAction(loginForm) {
      // 1.实现登录逻辑
      const { data: loginData } = await login(loginForm)
      this.token = loginData.token
      // 保存至localStorage,方法随意
      localCache.setCache('token', this.token)
      // 2.跳转至首页
      router.push('/home')
    },
    // router.beforeEach获取菜单树
    async getInfoAction() {
      const infoData = await getInfo({ token: this.token })
      this.menuList = infoData.data.menuList
      this.addRouteMenus()
    },
    // 动态加载路由
    addRouteMenus() {
      this.routeMenus = mapMenusToRoutes(this.menuList)
      // vue3中只有addRoute
      for (let item of this.routeMenus) {
        router.addRoute(item)
      }
      // 最后添加,否则刷新请求线上路由时,还没有加载完,会直接定向到notFound
      router.addRoute(notFound)
    }
  }
})
// src/permission.js
// main.js中引入次文件 import '@/permission'
import router from './router'
import localCache from '@/utils/cache'
import { useLoginStore } from '@/stores/user/login'
router.beforeEach(async (to, from, next) => {
  document.title = `${to.meta.title}`
  if (to.path === '/login') {
      // 清空localStorage
    localCache.clearCache()
    next()
  } else {
    const token = localCache.getCache('token')
    if (!token) {
      next(`/login`)
    } else {
      const loginStore = useLoginStore()
      if (loginStore.routeMenus.length) {
        next()
      } else {
      // 如果没有路由,调用动态加载路由
        try {
          await loginStore.getInfoAction()
          next({ ...to, replace: true })
        } catch (error) {
          next(`/login`)
        }
      }
    }
  }
})

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • vue3中watch和watchEffect的区别

    vue3中watch和watchEffect的区别

    本文主要介绍了vue3中watch和watchEffect的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-05-05
  • vue数据字典取键值方式

    vue数据字典取键值方式

    这篇文章主要介绍了vue数据字典取键值方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-04-04
  • vue循环中调用接口-promise.all();按顺序执行异步处理方式

    vue循环中调用接口-promise.all();按顺序执行异步处理方式

    这篇文章主要介绍了vue循环中调用接口-promise.all();按顺序执行异步处理方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-07-07
  • Vue实现带参数的自定义指令示例

    Vue实现带参数的自定义指令示例

    这篇文章主要为大家介绍了Vue实现带参数的自定义指令示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-01-01
  • Vue中的验证登录状态的实现方法

    Vue中的验证登录状态的实现方法

    这篇文章主要介绍了Vue中的验证登录状态的实现方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2019-03-03
  • vuex管理状态 刷新页面保持不被清空的解决方案

    vuex管理状态 刷新页面保持不被清空的解决方案

    今天小编就为大家分享一篇vuex管理状态 刷新页面保持不被清空的解决方案,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-11-11
  • vue中v-for通过动态绑定class实现触发效果

    vue中v-for通过动态绑定class实现触发效果

    这篇文章主要介绍了vue中v-for通过动态绑定class实现触发效果,本文通过实例代码给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下
    2018-12-12
  • vue 中swiper的使用教程

    vue 中swiper的使用教程

    本文通过实例代码给大家介绍了vue 中swiper的使用,感兴趣的朋友跟随脚本之家小编一起学习吧
    2018-05-05
  • vue.js实现标签页切换效果

    vue.js实现标签页切换效果

    这篇文章主要介绍了vue.js实现标签页切换效果,代码简单易懂,非常不错,具有一定的参考借鉴价值,需要的朋友可以参考下
    2018-06-06
  • Vue自定义树形控件使用详解

    Vue自定义树形控件使用详解

    这篇文章主要为大家详细介绍了Vue自定义树形控件的使用方法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-08-08

最新评论