前言

之前用vue+ant-design-vue写了一个动态路由的页面,更新看一下不能用了555~~~
之前用的组件版本不知道了,回退也不知道哪个版本合适,就是用"vue": "^3.2.13" , "vue-router": "^4.0.3","vuex": "^4.0.0",ant-design-vue": "^3.2.5"重新写一个吧。
本文章是看了其它杂七杂八的博客,自己排错后编写下,不容易啊

实现

  1. 首先在store\index.js文件编写
import { createStore } from 'vuex'


export default createStore({
  state: {
    menu_lists: []  //菜单
  },
  getters: {
    account(state) {
      return state.menu_lists  // 读取菜单列表
    }
  },
  mutations: {
    // 增加菜单
    menuAdd(state, n) {
      if (state.menu_lists.length == 0) {
        state.menu_lists.push(n)
      } else {
        if (state.menu_lists.some(menu => menu.id!= n.id)) {
          state.menu_lists.push(n)
        }
      }
    },
    // 清空菜单
    menuDelect(state) {
      state.menu_lists.length = 0
    }
  },
  actions: {
    menu_add({ commit }, data) {
      commit('menuAdd', data)
    },
	// 登出时调用将菜单数据删除
    menu_delect({ commit }) {
      commit('menuDelect')
    }
  },
  modules: {
  }
})


  1. 接着在App.vue编写

原因: 刷新时,动态路由需要重新挂载到路由实例, 但是在App.vue中调用init方法去初始化,并不能解决,因为App.vue属于路由的根,还未进入就被通配符拦截到404页面了, 我就在根上退出时将菜单保存在sessionStorage

// 创建完毕状态
  created() {
//在页面加载时读取sessionStorage里的状态信息
    if (sessionStorage.getItem("store")) {
      this.$store.replaceState(
        Object.assign(
          {},
          this.$store.state,
          JSON.parse(sessionStorage.getItem("store"))
        )
      );
    }
    //在页面刷新时将vuex里的信息保存到sessionStorage里
    window.addEventListener("beforeunload", () => {
      sessionStorage.removeItem("store");
      sessionStorage.setItem("store", JSON.stringify(this.$store.state));
    });
 }
  1. 读取后端菜单文件进行处理下

根据实际修改
vueRouter4中移除了addRouters,所以只能通过addRouter进行路由的动态添加

import { useRouter } from "vue-router";
import { useStore } from "vuex";
export default defineComponent({
	setup() {
	const store = useStore();
	const router = useRouter();
		// 路由数据重新封装
    function routerPackag(routers) {
      let accessedRouters = routers.filter((itemRouter) => {
        if (itemRouter.component != "Layout") {
          //处理组件---重点
          router.addRoute("base", {
            path: `/${itemRouter.path}`,
            name: itemRouter.name,
            component: () => import(`@/${itemRouter.component}`),
          });
          // 通过sessionStorage排查菜单是否存储,避免刷新后重复添加
          if (!sessionStorage.getItem("store")) {
            store.dispatch("menu_add", itemRouter);
          }
        }
        //存在子集
        if (itemRouter.children && itemRouter.children.length) {
          routerPackag(itemRouter.children);
        }
        return true;
      });
      return accessedRouters;
    }
   }
)}
  1. 主要的来了,可以main或者router\index编写(我是在router\index编写的)

1、刷新404:将匹配全部为定义路径到404的路由从静态路由表中去除,在动态权限路由都添加了之后在添加。
2、刷新白屏:如果是在路由守卫router.beforeEach中检测并用router.addRoute添加的路由,则需要将动态权限路由添加后的next()放行,改为next({ path: ${to.path} })触发新导航。

import { createRouter, createWebHashHistory } from 'vue-router'
import store from "../store";
import { ref } from 'vue'
const routes = [
  {
    path: '/login',
    name: 'login',
    // route level code-splitting
    // this generates a separate chunk (about.[hash].js) for this route
    // which is lazy-loaded when the route is visited.
    component: () => import( /* webpackChunkName: "Login" */ '../views/ant_login.vue'),
    meta: {
      requireAuth: false,
    },
  },
  {
    path: '/',
    name: 'base',

    component: () => import( /* webpackChunkName: "Login" */ '../views/ant_base.vue'),
    meta: {
      requireAuth: true,
    },
    children: [
      {
        path: 'index',
        name: 'home',
        redirect: "/map",
        component: () => import( /* webpackChunkName: "Login" */ '../views/ant_home.vue'),
      }
    ]
  },
  {
    name: "NotFont",
    path: '/:pathMatch(.*)*',
    component: () => import('../components/NotFont.vue'),
    alias: '/404', // 别名
    hideMenu: true
  }
]

const router = createRouter({
  history: createWebHashHistory(), //createWebHashHistory是hash模式
  // 页面刷新白屏问题
  // mode取值说明:
  // histroy:URL就像正常的 url,示例:http://localhost:8080/home
  // hash:默认值,会多一个“#”,示例:http://localhost:8080/#/home
  // abstract”:url不变示例:http://localhost:8080
  // mode: 'history',
  base: process.env.BASE_URL,
  routes
})
// 下面全局前置路由守卫可在main文件编写
const registerRouteFresh = ref(true) // 定义标识,记录路由是否添加
router.beforeEach(async (to, from, next) => {
  if (registerRouteFresh.value && store.state.menu_lists.length > 0) {
    router.removeRoute("NotFont")
    await store.state.menu_lists.forEach(e => {
      // 
      // 处理组件---重点
      router.addRoute("base", {
        path: `/${e.path}`,
        name: e.name,
        component: () => import(`@/${e.component}`),
      });
    })
    registerRouteFresh.value = false
    // next({ ...to, replace: true })
    next({
      path: `${to.path}`
    })
  } else {
    router.addRoute(router.options.routes[2])
    next()
  }
})

// 全局后置钩子-常用于结束动画等
router.afterEach(() => {
  //不接受next
});
export default router

登出页面需要清除缓存

import { useStore } from "vuex";
export default defineComponent({
	setup() {
	const store = useStore()
	function Logout() {
	// 将vuex的菜单数据删除
	store.dispatch("menu_delect");
   	window.sessionStorage.clear()
   }
)}

排错过程

心累不说看了那些博客了真是大海捞个针,博客太杂了,有的写的next({ …to, replace: true })我就想知道你是咋成功的,哎,排的有好的但不实用,排到垃圾就不想说了,连使用的组件都没有就光把一段代码粘贴上去,累累累😫

期望

码友们有啥更简单的方式实现前端动态路由刷新后能正常显示刷新前页面欢迎评论分享

Logo

华为开发者空间,是为全球开发者打造的专属开发空间,汇聚了华为优质开发资源及工具,致力于让每一位开发者拥有一台云主机,基于华为根生态开发、创新。

更多推荐