当前位置: 代码网 > it编程>前端脚本>Vue.js > vue3动态路由解决刷新页面空白或跳转404问题

vue3动态路由解决刷新页面空白或跳转404问题

2025年02月13日 Vue.js 我要评论
前言开发后台管理系统时,有时后端会根据权限返回不同的菜单列表,前端需要异步获取数据然后实时生成路由配置。在vue3项目中,我们使用pinia、vue-router实现动态路由,关键步骤如下:异步请求获

前言

开发后台管理系统时,有时后端会根据权限返回不同的菜单列表,前端需要异步获取数据然后实时生成路由配置。

在vue3项目中,我们使用pinia、vue-router实现动态路由,关键步骤如下:

  • 异步请求获取路由接口数据;
  • pinia状态管理保存路由信息;
  • vue-router实现路由配置;
  • 动态添加路由。

1 异步请求获取路由接口数据

// /src/api/route.js
export const getroutelist = () => {
  // 模拟异步请求
  return new promise((resolve) => {
    settimeout(() => {
      resolve([
        {
          name: "home",
          path: "/home",
          meta: { title: "首页" },
        },
        {
          name: "user",
          path: "/user",
          meta: { title: "用户" },
        },
      ])
    }, 1000)
  })
}

2 pinia状态管理保存路由信息

// /src/store/route.js
import { definestore } from "pinia"

export const useroutestore = definestore("route", {
  state: () => ({
    routelist: sessionstorage.getitem("routelist")
      ? json.parse(sessionstorage.getitem("routelist"))
      : [],
    isupdate: false,
  }),
  actions: {
    updateroutelist(routelist) {
      this.routelist = routelist
      sessionstorage.setitem("routelist", json.stringify(routelist))
      this.isupdate = true
    },
  },
  getters: {},
})

3 vue-router实现路由配置

import { createrouter, createwebhashhistory } from "vue-router"

// 定义基本路由配置
const routes = [
  {
    path: "/:pathmatch(.*)",
    meta: { title: "404" },
    name: "404",
    component: () => import("@/views/error/404.vue"),
  },
  {
    path: "/login",
    name: "login",
    component: () => import("@/views/login/login.vue"),
    meta: { title: "登录" },
  },
  {
    path: "/",
    name: "layout",
    component: () => import("@/views/layout/layout.vue"),
    redirect: "/home",
    children: [], // 初始时没有子路由
  },
]

export const router = createrouter({
  history: createwebhashhistory(import.meta.env.base_url),
  routes,
})

// 路由守卫,用于处理路由跳转前的逻辑
router.beforeeach(async (to, from, next) => {
  // 判断是否已登录且没有 token,未登录时重定向到登录页
  const token = localstorage.getitem("token")
  if (to.path !== "/login" && !token) {
    return next({ name: "login" })
  }
  next()
})

4 动态添加路由

核心代码

router.addroute("layout", {
	path: item.path,
	name: item.name,
	component: () => import(`@/views/${item.name}/index.vue`),
})

完整代码

// /src/router/index.js
import { createrouter, createwebhashhistory } from "vue-router"
import { useroutestore } from "@/store/route"
import { getroutelist } from "@/api/route"

// 定义基本路由配置
const routes = [
  {
    path: "/:pathmatch(.*)",
    meta: { title: "404" },
    name: "404",
    component: () => import("@/views/error/404.vue"),
  },
  {
    path: "/login",
    name: "login",
    component: () => import("@/views/login/login.vue"),
    meta: { title: "登录" },
  },
  {
    path: "/",
    name: "layout",
    component: () => import("@/views/layout/layout.vue"),
    redirect: "/home",
    children: [], // 初始时没有子路由
  },
]

export const router = createrouter({
  history: createwebhashhistory(import.meta.env.base_url),
  routes,
})

// 添加动态路由
const addroute = () => {
  const routestore = useroutestore()
  if (routestore.isupdate) {
    routestore.routelist.foreach((item) => {
      if (!router.hasroute(item.name)) {
        router.addroute("layout", {
          path: item.path,
          name: item.name,
          component: () => import(`@/views/${item.name}/index.vue`),
        })
      }
    })
    routestore.isupdate = false
  }
}

// 初始化路由
export const initrouter = async () => {
  let routelist = sessionstorage.getitem("routelist")
    ? json.parse(sessionstorage.getitem("routelist"))
    : await getroutelist()
  const routestore = useroutestore()
  routestore.updateroutelist(routelist)
  addroute()
}

// 路由守卫,用于处理路由跳转前的逻辑
router.beforeeach(async (to, from, next) => {
  // 添加动态路由
  addroute()
  // 判断是否已登录且没有 token,未登录时重定向到登录页
  const token = localstorage.getitem("token")
  if (to.path !== "/login" && !token) {
    return next({ name: "login" })
  }
  next()
})

注意:动态添加路由后刷新页面会跳转404页面,因为在进路由守卫之前,程序已经进行了路由匹配,结果就是没匹配到任何内容。

解决方案:在router注册之前调用initrouter函数初始化路由。

// main.js
import "./assets/css/main.css"

import { createapp } from "vue"
import { createpinia } from "pinia"
import app from "./app.vue"
import { initrouter, router } from "./router"

const app = createapp(app)

const call = async () => {
  app.use(createpinia())
  await initrouter()
  app.use(router)

  app.mount("#app")
}
call()

到此这篇关于vue3动态路由解决刷新页面空白或跳转404问题的文章就介绍到这了,更多相关vue3刷新页面空白或404内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!

(0)

相关文章:

版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。

发表评论

验证码:
Copyright © 2017-2025  代码网 保留所有权利. 粤ICP备2024248653号
站长QQ:2386932994 | 联系邮箱:2386932994@qq.com