vue-element-admin实战 | 第二篇: 最小改动接入后台实现根据权限动态加载菜单

简介: vue-element-admin实战 | 第二篇: 最小改动接入后台实现根据权限动态加载菜单

一. 前言

本篇基于 有来商城 youlai-mall 微服务项目,通过对vue-element-admin的权限菜单模块理解个性定制其后台接口,实现对vue-element-admin工程几乎不做改动的情况下,无缝接入后台接口实现动态权限菜单的加载。


在进行接下来的工作前,我们需要对原生的vue-element-admin项目改造,移除mock连通后台接口,具体可参考我这篇文章 vue-element-admin实战 | 第一篇: 移除mock接入后台,搭建有来商城youlai-mall前后端分离管理平台,如果在过程中有遇到问题,欢迎下方留言。


二. 前端调整

至于上文提到的对vue-element-admin几乎不做改动便可实现我们此篇文章的目的是不是我在扯,决定权给各位,我把对vue-element-admin项目改动的地方通过比对工具比对截图放上来。


先声明vue-element-admin此次改动的地方除了一个获取权限菜单的接口之外,剩余的改动全在 src/store/modules/permission.js 文件中。

微信图片_20230709215802.png



看到了吗,可以说仅对vue-element-admin做两处改动,再加上对getRoutes调用后台接口返回的菜单数据的分析,就可以理解和实现动态权限菜单的加载了。


改动代码片段 + 注释说明


import {list as getRoutes} from '@/api/admin/menu'

import Layout from '@/layout'

 

generateRoutes({commit}, roles) {

  return new Promise(resolve => {

    // 请求后台数据替换src/router/index.js的asyncRoutes异步路由

    getRoutes({mode: 3}).then(response => {

      // filterAsyncRoutes方法作权限过滤和数据转换,roles为登录用户角色ID集合,如:[1,2]

      let accessedRoutes = filterAsyncRoutes(response.data, roles)

      commit('SET_ROUTES', accessedRoutes)

      resolve(accessedRoutes)

    })

  })

}

// 递归权限过滤和数据转换

export function filterAsyncRoutes(routes, roles) {

   const res = []

   routes.forEach(route => {

     const tmp = {...route}

     if (hasPermission(roles, tmp)) {

       const component = tmp.component

       if (route.component) {

         if (component == 'Layout') {

           tmp.component = Layout

         } else {

           // 接口组件字符串转换成组件对象

           tmp.component = (resolve) => require([`@/views/${component}`], resolve)

         }

         if (tmp.children) {

           tmp.children = filterAsyncRoutes(tmp.children, roles)

         }

       }

       res.push(tmp)

     }

   })

   return res

}

三. 后端接口

1. 接口数据分析

接下来通过后台接口替换配置在src/router/index.js文件中asyncRoutes异步路由。


首先分析下异步路由的数据结构:

微信图片_20230709215817.png



通过对上图异步路由的数据观察和了解,得出以下几点:


path在子路由中前面没有反斜杠’/’

根菜单的alwaysShow为true

component组件需通过import完成编译时导入,接口只能返回组件路径字符串,所以这里在接口请求完成后必须有一个组件路径字符串到组件对象的转换过程

meta的roles属性对应的是有该路由访问权限角色唯一标识的集合,这里我使用的是角色ID

2. 接口实现

完成以上的数据分析,接下来就是接口的具体实现了,有关SQL和完整代码请点击 有来商城 youlai-mall 拉取即可,以下仅仅贴出关键代码和分析。


SysMenuMapper


获取菜单列表和对应访问权限的角色ID的集合

微信图片_20230709215821.png



SysMenuServiceImpl


将菜单转换成路由,递归生成父子结构树


微信图片_20230709215825.png


SysMenuController


REST对外提供接口


微信图片_20230709215828.png


3. 接口测试

使用接口测试工具测试, http://localhost:9999/youlai-admin/menus?mode=3 ,这里通过网关方式访问,详情请知悉有来商城 youlai-mall 项目。


[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-aFuycpuK-1600336598229)(https://i.loli.net/2020/09/16/PYgeyuFOCpt9Kk2.png)]


完整返回数据,看看是不是很匹配asyncRoutes异步路由的数据格式了


{

   "code": "00000",

   "data": [{

       "path": "/admin",

       "component": "Layout",

       "alwaysShow": true,

       "name": "系统管理",

       "meta": {

           "title": "系统管理",

           "icon": "documentation",

           "roles": [2, 1]

       },

       "children": [{

           "path": "user",

           "component": "admin/user",

           "alwaysShow": false,

           "name": "用户管理",

           "meta": {

               "title": "用户管理",

               "icon": "user",

               "roles": [1]

           }

       }, {

           "path": "role",

           "component": "admin/role",

           "alwaysShow": false,

           "name": "角色管理",

           "meta": {

               "title": "角色管理",

               "icon": "peoples",

               "roles": [2, 1]

           }

       }, {

           "path": "dept",

           "component": "admin/dept",

           "alwaysShow": false,

           "name": "部门管理",

           "meta": {

               "title": "部门管理",

               "icon": "tree",

               "roles": [1, 2]

           }

       }, {

           "path": "menu",

           "component": "admin/menu",

           "alwaysShow": false,

           "name": "菜单管理",

           "meta": {

               "title": "菜单管理",

               "icon": "tree-table",

               "roles": [1, 2]

           }

       }, {

           "path": "dict",

           "component": "admin/dict",

           "alwaysShow": false,

           "name": "字典管理",

           "meta": {

               "title": "字典管理",

               "icon": "education",

               "roles": [1, 2]

           }

       }]

   }],

   "msg": "一切ok"

}


最后要做的就是将组件(component)路径字符串转换成组件对象即可,再次贴出在上文permission.js改造的代码:


if (component == 'Layout') {

   tmp.component = Layout

 } else {

   // 接口组件字符串转换成组件对象

   tmp.component = (resolve) => require([`@/views/${component}`], resolve)

 }


有一点需要注意的是上面组件动态导入不能使用import,如下:


tmp.component = () => import(`@/views/${component}`)

1

不然会报以下错误


Error: Cannot find module '@/views/***'

1

原因是webpack不支持import动态导入了,但是之前确实是可以的,所以有些让人想不通,这里使用require动态导入组件即可。


4. 菜单权限测试

参看当前用户拥有的角色ID为2

微信图片_20230709215958.png



用户管理要求只有角色ID为1才能有其访问权限


[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-VCh7ZKj6-1600336598232)(https://i.loli.net/2020/09/16/nXpj6TfF4zbBV9o.png)]


测试下用户登录控制台能否看到“用户管理”菜单


微信图片_20230709220002.png


怎么样,只拥有角色ID为2的角色是看不到需要角色ID为1才能访问的“用户管理”菜单的,也就证明了我们成功通过接入后台接口实现了权限菜单的动态加载的目的。


四. 结语

其实有个问题值得去思考下的。为什么我们在从后台获取全部菜单列表的时候需要关联查询出有对应访问权限的角色ID的集合,而不是说动态传入当前登录用户的信息去查询该用户拥有的菜单集合呢?这其实是两种实现方式,区别在于第一种每个用户都要获取全部菜单数据,而第二种方式针对每个用户加载其拥有权限的菜单数据。单看的话第二种方式肯定优于第一种,但这里我们选用的是第一种加载全部菜单数据。如果你结合缓存去看的话自然就想的通了,所有用户共享缓存的同一份菜单数据,还是说缓存针对每个用户存放一份菜单缓存又或者用户每次都去请求数据库。后面计划把菜单数据存放到缓存Redis中然后做一个分析比较。


附完整代码:


有来商城后端:youlai-mall


有来商城前端:youlai-mall-admin-web


写了这么多其实也是想给自己的项目打个广告,更希望对大家有所帮助,对于技术人来说少走弯路真的很重要,喜欢的朋友给个star,对我来说这份帮助更是自己继续下去的动力,所以谢谢了。有啥问题下方留言,或直接联系我(微信号:haoxianrui)。


相关文章
|
3月前
|
存储 JavaScript 前端开发
前端开发:Vue.js入门与实战
【10月更文挑战第9天】前端开发:Vue.js入门与实战
|
3月前
|
JavaScript 前端开发
Vue开发必备:$nextTick方法的理解与实战场景
Vue开发必备:$nextTick方法的理解与实战场景
271 1
|
2月前
|
JavaScript UED
"Vue实战技巧大揭秘:一招解决路由跳转页面不回顶部难题,让你的单页面应用用户体验飙升!"
【10月更文挑战第23天】在Vue单页面应用中,点击路由跳转时,默认情况下页面不会自动滚动到顶部,这可能影响用户体验。本文通过一个新闻网站的案例,介绍了如何使用Vue-router的全局前置守卫和`scrollBehavior`方法,实现路由跳转时页面自动滚动到顶部的功能,提升用户浏览体验。
130 0
|
4月前
|
JavaScript 前端开发 UED
组件库实战 | 用vue3+ts实现全局Header和列表数据渲染ColumnList
该文章详细介绍了如何使用Vue3结合TypeScript来开发全局Header组件和列表数据渲染组件ColumnList,并提供了从设计到实现的完整步骤指导。
|
5月前
|
JavaScript
Vue学习之--------消息订阅和发布、基础知识和实战应用(2022/8/24)
这篇文章介绍了Vue中消息订阅和发布的机制,作为组件间通信的一种方式,特别适用于任意组件间的通信。文章通过基础知识讲解和具体的代码实例,展示了如何使用`pubsub-js`库来安装、订阅、发布消息,并在组件销毁前取消订阅以避免内存泄漏。
Vue学习之--------消息订阅和发布、基础知识和实战应用(2022/8/24)
|
5月前
|
JavaScript 前端开发 开发者
Vue学习之--------深入理解Vuex、原理详解、实战应用(2022/9/1)
这篇文章详细介绍了Vuex的基本概念、使用场景、安装配置、基本用法、实际应用案例以及注意事项,通过一个数字累加器的实战示例,帮助开发者深入理解Vuex的原理和应用。
|
5月前
|
JavaScript
Vue中组件化编码使用(实战练习一)
这篇文章是关于Vue中组件化编码的实战练习,介绍了组件化编码的流程、组件间参数传递的方法以及如何通过组件配合完成一个需求。内容包括了组件拆分、动态组件实现、交互绑定事件的步骤,以及使用props进行父子组件通信的注意事项。文章还提供了一个待办事项列表的静态页面效果和相关代码示例,包括TheFooter.vue、TheHeader.vue、TheItem.vue、TheList.vue和App.vue等组件的代码。
Vue中组件化编码使用(实战练习一)
|
5月前
|
JavaScript
Vue中组件化编码 完成任务的添加、删除、统计、勾选需求(实战练习三完结)
这篇文章展示了如何在Vue中实现任务添加、删除、勾选和统计功能,通过组件间参数传递和方法实现,提供了App.vue、TheList.vue、TheItem.vue、TheHeader.vue和TheFooter.vue组件的主要代码示例。
|
6月前
|
存储 JavaScript API
Vue 全局状态管理新宠:Pinia实战指南
 随着Vue.js项目的日益复杂,高效的状态管理变得至关重要。Pinia作为Vue.js官方推荐的新一代状态管理库,以其简洁的API和强大的功能脱颖而出。本文将带您快速上手Pinia,从安装到应用,轻松实现Vue.js项目的全局状态管理,提升开发效率和项目可维护性。
|
5月前
|
数据采集 JavaScript
Vue中组件化编码使用、实现组件之间的参数传递(实战练习二)
这篇文章介绍了Vue中组件化编码的实现和组件间参数传递的方法,通过实战练习展示了从App.vue向TheList.vue和TheItem.vue传递数据的流程和代码示例。