vueRouter动态路由(实现菜单权限控制)

 一、权限控制管理:

对于企业级的项目, 我们可能需要对项目做权限控制管理, 实现不同角色的用户登录项目根据所拥有的权限访问不同的页面内容,此时就需要使用到动态路由来对权限页面做限制。

【使用vue-router实现动态路由,达到实现菜单权限控制的功能】

二、实现思路:

1.动态路由: 将拥有权限的路由单独从路由表系统中提出,视为动态路由,待后续根据用户权限来导入拥有权限的动态路由
2.权限与动态路由绑定: 用户登录之后,需要获取该用户所拥有的权限,该权限前后端需要协调一致,需要对应前端拥有权限的动态路由,后端可以返回动态路由的name来做以区分,也可以在前端单独定义每个动态路由(权限)的标识进行权限的区分(我使用的方法,因为后台的权限分配有自己的逻辑不需要再次改动,只需在前端的动态路由中定义该路由的权限标识即可)
3.获取权限筛选动态路由: 获取到用户所拥有的权限之后,将后端返回的权限标识与动态路由中所定义的权限标识比对,筛选出该用户所拥有的权限(动态路由)
4.添加权限(动态路由): 使用router.addRoute()方法将拥有权限的动态路由添加到路由表系统中实现权限的控制(vue-router4.x版本已废弃router.addRoutes()来对动态路由批量导入)
5.渲染菜单: 获取到拥有权限的动态路由之后,在vuex/pinia中去组合需要渲染菜单的动态路由(+/静态路由),菜单需要在路由中按需定义菜单的名称、图片、icon等等,最后将路由中组合好的菜单循环渲染展示(不能使用router.options来获取所有的路由渲染菜单,因为该方法不能获取到动态路由)
6.退出登录删除动态路由(权限): 用户退出登录时需要删除动态路由(权限),因为如果退出登录时不删除动态路由(权限),紧接着登录其他权限不同的用户,该用户没有上一个用户所拥有的权限,所以无法查看上一个用户所看到的菜单以及页面,但是该用户可以使用浏览器的回退功能或者修改url地址栏的路径去访问上一个用户能访问的页面,这样该用户就可以访问到不属于自己权限的页面,所以在用户退出登录时需要删除动态路由(权限),来保证权限的准确性。删除动态路由使用到router.removeRoute()

三、实现代码:

第一步: 提取动态路由,每个动态路由中的meta对象中定义了要渲染菜单的名称的title和图片imgpermissionCode定义的标识为与后端确定的该用户是否具有该路由菜单的权限。
/***  router/system/index.js*	权限的动态路由*/
export const changeChildRouter = [{path: 'user',name: 'SysUser',component: () => import(/* webpackChunkName:"system" */'../../views/System/UserManage'),meta: {title: '用户管理',permissionCode: 10004,img: require('@/assets/images/Tree/setting-user.png')}},{path: 'department',name: 'SysDepartment',component: () => import(/* webpackChunkName:"system" */'../../views/System/DepartmentManage'),meta: {title: '部门管理',permissionCode: 10404,img: require('@/assets/images/Tree/setting-gis.png')}},{path: 'position',name: 'SysPosition',component: () => import(/* webpackChunkName:"system" */'../../views/System/PositionManage'),meta: {title: '岗位管理',permissionCode: 10204,img: require('@/assets/images/Tree/setting-user.png')}},{path: 'role',name: 'SysRole',component: () => import(/* webpackChunkName:"system" */'../../views/System/RoleManage'),meta: {title: '角色管理',permissionCode: 10104,img: require('@/assets/images/Tree/setting-user.png')}},
]
第二步: 登录之后,获取该登录用户的权限,筛选动态路由。开始有两种想法: 
  • 第一种想法: 当用户登录之后,紧接着在逻辑当中获取该用户的权限,将用户的权限存储到session当中,目的是为了防止用户刷新浏览器权限依然不会丢失。但是安全性不高,用户手动修改session中存储的权限就会导致权限丢失,或者非法获取其他用户的session权限放置未拥有该权限的用户的session中,导致该用户拥有了其他用户不属于自己的权限等等问题,所以我应用了第二种想法
  • 第二种想法: 当用户登录之后,在路由前置守卫router.beforeEach()中去获取权限,获取到权限之后,引入并筛选出该用户权限的动态路由,使用router.addRoute()将该用户的动态路由添加到路由表系统当中。将筛选出到的动态路由(权限)存储到vuex当中。由于vuex做临时存储,所以原理上刷新页面之后vuex中的数据就会丢失,但是在router.beforeEach()路由前置守卫中获取权限就解决了该问题,每次刷新页面都会先执行router.beforeEach() ,所以每次刷新页面都会先去获取该用户的权限,将根据后端返回的权限标识筛选出的动态路由存储到vuex当中。但是想要实现刷新页面动态路由(权限页面)不丢失,不能将404页面定义为静态路由放置在路由表系统当中,需要随权限获取后,动态添加404页面,目的是为了解决刷新权限页面之后权限丢失,在没有重新获取到权限的时候,检测到路由表系统当中不存在该动态路由,所以直接进入404页面的问题。【代码示例如下】
/***  permission.js*	权限控制管理*/
import router from "@/router"
import store from '@/store'
import { changeChildRouter } from '@/router/system'// 防止路由无限循环
let routeFlag = false;
// 导航守卫
router.beforeEach(async (to, from, next) => {let token = localStorage.getItem('token');// 如果没有token,并且当前不是登录页,就跳转到登录界面if (!token && !to.path.includes('/login')) {routeFlag = falsenext('/login')} else if (to.path.includes('/login')) {routeFlag = falsenext()} else {if (routeFlag) return next()// 权限控制routeFlag = true// 1. 根据上面定义的 routeFlag 得出: 用户没有获取权限/刷新页面权限丢失,则先调用获取权限的接口const res = await store.dispatch('menu/MenuInfo')// 2. 从后端返回的权限中筛选出权限的标识 roleId, 对应前端动态路由中绑定的权限的标识 permissionCodeconst Liststr = res.map(item => item.roleId)// 3. 系统管理--权限: changeChildRouter 所有的动态路由const systemRoutes = systemAsyncRouter(changeChildRouter, Liststr)// 4. 保存动态路由(权限),为了退出登录之后删除asyncRoutes(systemRoutes)// 5. 添加 404 路由router.addRoute({path: '/:pathMatch(.*)*',name: '404',component: () => import(/* common */'@/views/NotFound'),meta: { title: '404' }},)next({...to,// 重新进入replace: true // 不保存本次进入页面的路由历史记录})}
})// 系统管理——权限
function systemAsyncRouter(routes, codes) {// 1. 根据接口返回的权限找到可以访问的路由const filterRoutes = routes.filter(item => codes.includes(item.meta?.permissionCode))// 2. 添加动态路由addAsyncRouter('系统功能', filterRoutes)// 3. 将动态路由添加到vuex中和静态路由合并渲染侧边栏菜单store.commit('menu/setMenuList', filterRoutes)return filterRoutes
}// 将动态路由添加到路由表系统中
function addAsyncRouter(parentName, asyncRoutesList) {asyncRoutesList.map(item => {// 可能会存在没有定义的组件的父级路由, 目的是渲染菜单时, 该路由只做子级路由的归类(仅用于展示) if (!item.component) return addAsyncRouter(parentName, item.children)router.addRoute(parentName, item) // 添加动态路由})
}// 保存所有的动态路由, 使用展开运算符优化写法,后续有其他的动态路由直接依次传入实参即可
function asyncRoutes(...args) {store.commit('menu/setPermissionRouteList', [].concat(...args))
}

上述代码使用到了router.addRoute()将路由添加至路由表当中,但是在此传递了两个参数:

  1. 第一个参数: 为需要添加的动态路由的一级路由的name
  2. 第二个参数: 动态路由本身

如果只传递了一个动态路由本身的参数,则表示该动态路由为一级路由

第三步: 添加动态路由之后,需要在用户退出的时候调用以下方法删除动态路由以及保存的权限
/***	permission.js*	权限控制管理*/// 退出登录时删除动态路由以及vuex中保存的动态路由,防止下一个用户登录时,回退查看到上一个用户的动态路由(权限)
export function removeAsyncRouter() {// 所有的动态路由const permissionRouteList = store.state.menu.permissionRouteListconst removeRouteFn = (permissionRouteList) => {if (permissionRouteList.length > 0) {permissionRouteList.map(item => {if (!item.component && item.children) return removeRouteFn(item.children)// 删除动态路由,保证用户退出后,下一个用户登录时,不会回退到上一个用户的路由router.removeRoute(item.name)})}}removeRouteFn(permissionRouteList)// 清空动态路由的左侧菜单store.commit('menu/setMenuList', [])// 清空动态路由的权限标识store.commit('menu/setPermissionCodeList', [])
}

 上述代码中使用router.removeRoute()来删除动态路由, 该方法接收一个参数: 路由的name, 即可将该name的路由从路由表当中删除。

第四步: 在vuex当中将动态路由(+/静态路由)组合来渲染权限菜单展示 
/***	store/menu.js*  权限菜单*/import { loadAuthority } from '@/api/loaddata'
import { systemChildRouter } from '@/router/system'
const state = () => ({permissionCodeList: [], // 该用户有权限对应的动态路由的标识permissionRouteList: [], // 动态路由,单独保存一份为了退出登录的时候删除动态路由menuList: [], // 系统管理: 静态路由+动态路由,渲染左侧菜单
})const mutations = {// 添加权限路由对应的权限标识setPermissionCodeList(state, permissionCodeList) {state.permissionCodeList = permissionCodeList},// 动态路由集合setPermissionRouteList(state, permissionRouteList) {state.permissionRouteList = permissionRouteList},// ********** 系统管理:路由权限 ***********setMenuList(state, permissionRouteList) {const menuList = [...systemChildRouter, ...permissionRouteList]},
}const actions = {// 获取该用户的权限项async MenuInfo(ctx) {const res = await loadAuthority.getUserAuthList({ userName: localStorage.getItem('userName') })ctx.commit('setPermissionCodeList', res)return res}
}export default {namespaced: true,state,actions,mutations
}

vuex store中组合了渲染菜单的: 动态路由+静态路由, 由于提前已经将渲染菜单的名称和图片定义在了每个路由当中,所有组合要渲染菜单的路由之后, 即可在vue组件中去循环渲染权限菜单 

第五步: 渲染菜单, 在vue组件计算属性当中获取vuex中组合好的渲染菜单的路由, 在模板中循环渲染展示菜单 
/***	LeftMenu.vue*	系统管理: 权限侧边栏菜单*/<template><a-layout-sider :width="220"><a-menu mode="inline" v-model:selectedKeys="current"><!-- 权限菜单 --><left-menu-item v-for="item in menuList" :key="item.path" :item="item"></left-menu-item></a-menu></a-layout-sider>
</template><script>
import LeftMenuItem from './LeftMenuItem.vue'
export default {name: "SystemMenu",components: {LeftMenuItem},data() {return current: ["help"],};},computed: {menuList() {// 获取渲染菜单的路由数组return this.$store.state.menu.menuList}},
};
</script>

在渲染权限侧边栏菜单的LeftMenu.vue组件中, 引入了一个封装的LeftMenuItem.vue组件, 在该组件中实现了具体的渲染菜单的模板代码, 并且使用LeftMenuItem.vue可以使用递归自身来实现可展开菜单的功能。

/***	LeftMenuItem.vue*	封装渲染菜单组件*/
<template><div><!-- 如果没有children, 则渲染在侧边栏的一级菜单 --><template v-if="!item.children"><a-menu-item :key="item.path"><template #icon><img :src="item.meta.img"></template><router-link :to="item.path">{{ item.meta.title }}</router-link></a-menu-item></template><!-- 如果有children,表明是二级菜单,该一级菜单不提供路由,只是对二级菜单的归类 --><template v-else><a-sub-menu :key="item.path"><template #icon><img :src="item.meta.img"></template><template #title>{{ item.meta.title }}</template><!-- 递归组件,如果子路由里面还有路由,则继续渲染它的子路由对应的菜单 --><left-menu-item v-for="item in item.children" :key="item.path" :item="item"></left-menu-item></a-sub-menu></template></div>
</template><script>
export default {name: "LeftMenuItem",props: {item: {type: Object,required: true},}
}
</script>
<style lang="scss" scoped>
</style>

 递归自身组件: 想要递归自身组件, 需要在组件脚本中导出定义自身的name: 'LeftMenuItem' , 在组件内容即可调用自身的name来实现递归调用组件本身。

 

这样就实现了菜单的权限控制, 拥有不同权限的用户登录之后, 就只会看到属于自己权限的页面~

 

 

 

 

 

 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/307051.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

FFmpeg: 简易ijkplayer播放器实现--06封装打开和关闭stream

文章目录 流程图stream openstream close 流程图 stream open 初始化SDL以允许⾳频输出&#xff1b;初始化帧Frame队列初始化包Packet队列初始化时钟Clock初始化音量创建解复用读取线程read_thread创建视频刷新线程video_refresh_thread int FFPlayer::stream_open(const cha…

Docker 学习笔记(七):介绍 Dockerfile 相关知识,使用 Dockerfile 构建自己的 centos 镜像

一、前言 记录时间 [2024-4-12] 系列文章简摘&#xff1a; Docker学习笔记&#xff08;二&#xff09;&#xff1a;在Linux中部署Docker&#xff08;Centos7下安装docker、环境配置&#xff0c;以及镜像简单使用&#xff09; Docker 学习笔记&#xff08;三&#xff09;&#x…

【MoS2】应变增强的单层MoS2光电探测器

这篇文章的标题是《Strain-Enhanced Large-Area Monolayer MoS2 Photodetectors》&#xff0c;作者是Borna Radatovic等人&#xff0c;发表在《ACS Applied Materials & Interfaces》期刊的2024年第16卷。文章主要研究了应变增强的大面积单层MoS2光电探测器的性能和应用潜力…

【安全】挖矿木马自助清理手册

一、什么是挖矿木马 挖矿木马会占用CPU进行超频运算&#xff0c;从而占用主机大量的CPU资源&#xff0c;严重影响服务器上的其他应用的正常运行。黑客为了得到更多的算力资源&#xff0c;一般都会对全网进行无差别扫描&#xff0c;同时利用SSH爆破和漏洞利用等手段攻击主机。 …

GC垃圾回收

垃圾回收 1、什么是 垃圾回收机制&#xff1a; 理解Java的垃圾回收机制&#xff0c;就要从&#xff1a;“什么时候”&#xff0c;“对什么东西”&#xff0c;“做了什么”三个方面来具体分析。 ​ 第一&#xff1a;“什么时候”即就是GC触发的条件。 ​ GC触发的条件有两种…

相机模型浅析

相机模型 文章目录 相机模型四个坐标系针孔相机模型世界坐标系到相机坐标系相机坐标系到图像坐标系图像坐标到像素坐标 四个坐标系 ①世界坐标系&#xff1a;是客观三维世界的绝对坐标系&#xff0c;也称客观坐标系。因为数码相机安放在三维空间中&#xff0c;我们需要世界坐标…

【opencv】示例-image_alignment.cpp 利用ECC 算法进行图像对齐

affine imshow("image", target_image); imshow("template", template_image); imshow("warped image", warped_image); imshow("error (black: no error)", abs(errorImage) * 255 / max_of_error); homography 这段代码是一个利用EC…

【300套】基于Springboot+Vue的Java毕业设计项目(附源码+演示视频+LW)

大家好&#xff01;我是程序猿老A&#xff0c;感谢您阅读本文&#xff0c;欢迎一键三连哦。 &#x1f9e1;今天给大家分享300的Java毕业设计&#xff0c;基于Springbootvue框架&#xff0c;这些项目都经过精心挑选&#xff0c;涵盖了不同的实战主题和用例&#xff0c;可做毕业…

银行司库系统应用架构介绍

继国务院国资委印发了《关于推动中央企业加快司库体系建设进一步加强资金管理的意见》以及《关于中央企业加快建设世界一流财务管理体系的指导意见》&#xff0c;司库体系建设开始得到了更多重视。其中&#xff0c;作为改革风向标&#xff0c;央企数字化转型及司库建设对整个行…

【YUNBEE云贝-进阶课】MySQL8.0性能优化实战培训

众多已经学习过MySQL 8.0 OCP认证专家的课程的同学们对 MySQL 8.0 的安装部署、体系结构、配置监控、用户管理、主从复制、系统运维、MGR等基础操作和动手实验有了一定的学习基础.很多学员反馈希望更进一步提升技术能力、解决工作中碰到的性能问题。 针对MySQL8.0的数据库性能优…

Ubuntu 20.04 设置开启 root 远程登录连接

Ubuntu默认不设置 root 帐户和密码 Ubuntu默认不设置 root 帐户和密码 Ubuntu默认不设置 root 帐户和密码 如有需要&#xff0c;可在设置中开启允许 root 用户登录。具体操作步骤如下&#xff1a; 操作步骤 1、首先使用普通用户登录 2、设置root密码 macw:~$ sudo passwd …

基于STM32F103单片机的时间同步项目

一、前言 本项目为前一个时间同步项目的更迭版本&#xff0c;由于之前的G031开发板没有外部晶振&#xff0c;从机守时能力几乎没有&#xff0c;5秒以上不同步从机时间就开始飞了。在考虑成本选型后&#xff0c;选择了带有外部有缘晶振的STM32F103C8T6最小单片机&#xff0c;来作…

DRF的认证、权限、限流、序列化、反序列化

DRF的认证、权限、限流、序列化、反序列化 一、认证1、直接用&#xff0c;用户授权2、认证组件源码3、第三方认证djangorestframework-simplejwt 二、权限1. 直接使用&#xff0c;用户权限2.权限组件源码 三、序列化1. 序列化1.1 自定义Serailizer类序列化1.2 在视图APIView中使…

嵌入式驱动学习第七周——GPIO子系统

前言 GPIO子系统是用来统一便捷地访问实现输入输出中断等效果。 嵌入式驱动学习专栏将详细记录博主学习驱动的详细过程&#xff0c;未来预计四个月将高强度更新本专栏&#xff0c;喜欢的可以关注本博主并订阅本专栏&#xff0c;一起讨论一起学习。现在关注就是老粉啦&#xff0…

spring boot整合Redis监听数据变化

一、前言 Redis提供了数据变化的通知事件&#xff0c;可以实时监测key和value的变化&#xff0c;客户端可以通过订阅相关的channel来接收这些通知事件&#xff0c;然后做相应的自定义处理&#xff0c;详细的介绍可以参考官方文档Redis keyspace notifications | Docs 使用Red…

httpsok-快速申请谷歌SSL免费证书

&#x1f525;httpsok-快速申请谷歌SSL免费证书 使用场景&#xff1a; 部署CDN证书、OSS云存储证书证书类型&#xff1a; 单域名 多域名 通配符域名 混合域名证书厂商&#xff1a; ZeroSSL Lets Encrypt Google证书加密类型&#xff1a; ECC、 RSA 一、证书管理 进入 证书管…

【数学建模】2024Mathorcup数学建模C题完整思路与代码论文解析

2024Mathorcup数学应用挑战赛C题|图神经网络的预测模型ARIMA时间序列预测模型人员排班混合整数规划模型|完整代码和论文全解全析 我们已经完成了2024Mathorcup数学建模挑战赛C题的40页完整论文和代码&#xff0c;相关内容可见文末&#xff0c;部分图片如下&#xff1a; 问题分…

[蓝桥杯] 岛屿个数(C语言)

提示&#xff1a; 橙色字体为需要注意部分&#xff0c;红色字体为难点部分&#xff0c;会在文章“重难点解答”部分精讲。 题目链接 蓝桥杯2023年第十四届省赛真题-岛屿个数 - C语言网 题目理解 这道题让我们求岛屿个数&#xff0c;那么我们就应该先弄懂&#xff0c;对于一…

Qt5 编译oracle数据库

库文件 1、Qt源码目录&#xff1a;D:\Qt5\5.15.2\Src\qtbase\src\plugins\sqldrivers\oci 2、oracle客户端SDK: https://www.oracle.com/database/technologies/instant-client/winx64-64-downloads.html 下载各版本中的如下压缩包&#xff0c;一定要版本相同的 将两个压缩包…

【简单讲解如何安装与配置Composer】

&#x1f3a5;博主&#xff1a;程序员不想YY啊 &#x1f4ab;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f917;点赞&#x1f388;收藏⭐再看&#x1f4ab;养成习惯 ✨希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出…