vue iview实现动态路由和权限验证功能


Posted in Javascript onApril 17, 2018

github上关于vue动态添加路由的例子很多,本项目参考了部分项目后,在iview框架基础上完成了动态路由的动态添加和菜单刷新。为了帮助其他需要的朋友,现分享出实现逻辑,欢迎一起交流学习。

Github地址

iview-dynamicRouter

实现目标

客户端从服务端拿到路由和权限数据后,刷新项目的路由和菜单列表,并进行权限控制。

项目基础

  • 基础框架: iview组件库官方模板项目 iview-admin 的template分支项目,此项目为 iview-admin 的基础框架代码。项目地址:iview-admin 

实现逻辑

动态路由控制加载

一般来说,动态路由控制分为两种:一种是将所有路由数据存储在本地文件中,然后从服务端获取用户的权限信息,在路由跳转时,添加权限判断钩子,如果用户前往的页面不在权限列表内,则禁止跳转。另一种则是本地只存储基本路由,如错误处理页面、无权限控制页面等,而权限路由则从服务器获取,服务器根据用户的权限下发相应的路由数据,客户端利用这些数据进行路由的动态生成和添加,本文采用的是第二种方法。

iview-admin项目将路由分为三种:

  • 不作为Main组件的子页面展示的页面路由,例如login、404、403等错误页面路由;
  • 作为Main组件的子页面展示但是不在左侧菜单显示的路由 otherRouter ,比如首页路由;
  • 作为Main组件的子页面展示并且在左侧菜单显示的路由 appRouter ;

拿到路由数据后,我们主要进行两部分操作,第一部分是遍历数据,利用组件异步加载的方法,加载每个路由节点对应的组件,之后利用 router.addRoutes(routes) 完成路由列表的动态添加;第二部分是因为 iview-admin 框架下的页面标签和面包屑导航,需要遍历appRouter获取路由信息,所以我们也需要将路由数据存入 vuex ,以便全局访问。

需要特别注意的是,如果404页面为静态路由,那么第一次进入页面时,这时动态路由还未加载,找不到路由地址会默认跳转到404错误页,体验很差,所以404路由先不写入路由规则中,和动态路由一起加载。

主要代码实现如下:

数据请求及路由节点生成

//util.js
//生成路由
util.initRouter = function (vm) {
 const constRoutes = [];
 const otherRoutes = [];
 // 404路由需要和动态路由一起注入
 const otherRouter = [{
  path: '/*',
  name: 'error-404',
  meta: {
   title: '404-页面不存在'
  },
  component: 'error-page/404'
 }];
 // 模拟异步请求
 util.ajax('menu.json').then(res => {
  var menuData = res.data;
  util.initRouterNode(constRoutes, menuData);
  util.initRouterNode(otherRoutes, otherRouter);
  // 添加主界面路由
  vm.$store.commit('updateAppRouter', constRoutes.filter(item => item.children.length > 0));
  // 添加全局路由
  vm.$store.commit('updateDefaultRouter', otherRoutes);
  // 刷新界面菜单
  vm.$store.commit('updateMenulist', constRoutes.filter(item => item.children.length > 0));
 });
};
//生成路由节点
util.initRouterNode = function (routers, data) {
 for (var item of data) {
  let menu = Object.assign({}, item);
  menu.component = lazyLoading(menu.component);
  if (item.children && item.children.length > 0) {
   menu.children = [];
   util.initRouterNode(menu.children, item.children);
  }
  //添加权限判断
  meta.permission = menu.permission ? menu.permission : null;
  //添加标题
  meta.title = menu.title ? menu.title : null;
  menu.meta = meta;
 }
};

动态加载组件

//lazyLoading.js
export default (url) =>()=>import(`@/views/${url}.vue`)
Store缓存实现
//app.js
 // 动态添加主界面路由,需要缓存
updateAppRouter (state, routes) {
 state.routers.push(...routes);
 router.addRoutes(routes);
},
// 动态添加全局路由,不需要缓存
updateDefaultRouter (state, routes) {
 router.addRoutes(routes);
},
// 接受前台数组,刷新菜单
updateMenulist (state, routes) {
 state.menuList = routes;
}

最后在main.js中进行调用

//main.js
 mounted () {
 // 调用方法,动态生成路由
 util.initRouter(this);
 }

权限控制

同动态路由实现方法类似,操作权限控制也一般也分为两种,第一种是页面显示时不控制权限,所有的操作,比如按钮全部展现,然后在操作发起时,进行权限判断,如果用户拥有该操作的权限,则通过,否则提醒用户无权限,第二种则是在页面加载的时候,就进行权限判断,无权限的操作不进行显示。本人更喜欢第二种方法,这样不会对用户进行误导,个人认为用户看到的应该就行可操作的,不然点下按钮再提示无权限的感觉一定很不爽。

本项目的思路来源见参考博文,原博主的具体思路是:在路由结构的meta字段中,添加用户操作权限列表,然后注册全局指令,当节点初次渲染时,判断该页面是否存在权限,如果存在,并且传入的参数不在权限列表中,则直接删除该节点。

主要代码实现如下:

在路由数据中添加 permission 字段,存放权限列表

//menu.json,模拟异步请求数据
[
 {
 "path": "/groupOne",
 "icon": "ios-folder",
 "name": "system_index",
 "title": "groupOne",
 "component": "Main",
 "children": [
  {
  "path": "pageOne",
  "icon": "ios-paper-outline",
  "name": "pageOne",
  "title": "pageOne",
  "component": "group/page1/page1",
  "permission":["del"]
  },
  ...
 ]
 }
]

在遍历生成路由节点时,将 permission 字段数据存入路由节点 meta 属性中

//util.js
//生成路由节点
util.initRouterNode = function (routers, data) {
 for (var item of data) {
  ....
  //添加权限判断
  meta.permission = menu.permission ? menu.permission : null;
  ...
 }
};

定义全局命令组件,读取路由 permission 属性值获得权限列表,如果该不权限在权限列表中,则删除节点

//hasPermission.js 

const hasPermission = {
 install (Vue, options) {
  Vue.directive('hasPermission', {
   bind (el, binding, vnode) {
    let permissionList = vnode.context.$route.meta.permission;
    if (permissionList && permissionList.length && !permissionList.includes(binding.value)) {
     el.parentNode.removeChild(el);
    }
   }
  });
 }
};
export default hasPermission;

权限组件使用示例:

<template>
 <div>
  <h1>page1</h1>
  <Button v-hasPermission="'add'">添加</Button>
  <Button v-hasPermission="'edit'">修改</Button>
  <Button v-hasPermission="'del'">删除</Button>
 </div>
</template>

全局注册组件

// main.js
import hasPermission from '@/libs/hasPermission.js';
Vue.use(hasPermission);

这种权限控制方法的优点就是,不管是管理配置还是页面处理逻辑都相对简单,没有很多重复的代码判断和节点处理,在参考对比了网上几种实现方式后,个人比较推荐这一种方法。

页面标签和面包屑导航

在我看来,页面标签和面包屑都属于系统中锦上添花的页面相关控件,提高页面管理的便捷性,在iview官方admin项目中已经实现了这两个组件。所以这个项目中,只是将其进行移植,实现了组件功能,没有深入了解,感兴趣的可以仔细研究。

总结

以上所述是小编给大家介绍的vue iview实现动态路由和权限验证功能,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

Javascript 相关文章推荐
javascript各浏览器中option元素的表现差异
Apr 07 Javascript
javascript中的注释使用与注意事项小结
Sep 20 Javascript
通过百度地图获取公交线路的站点坐标的js代码
May 11 Javascript
中文字符串截取的js函数代码
Apr 17 Javascript
AngularJS基础 ng-options 指令详解
Aug 02 Javascript
JS无缝滚动效果实现方法分析
Dec 21 Javascript
AngularJS ng-repeat指令及Ajax的应用实例分析
Jul 06 Javascript
Angularjs cookie 操作实例详解
Sep 27 Javascript
vue实现商品加减计算总价的实例代码
Aug 12 Javascript
谈谈React中的Render Props模式
Dec 06 Javascript
搭建一个Koa后端项目脚手架的方法步骤
May 30 Javascript
vue router 动态路由清除方式
May 25 Vue.js
基于VuePress 轻量级静态网站生成器的实现方法
Apr 17 #Javascript
Vue-cropper 图片裁剪的基本原理及思路讲解
Apr 17 #Javascript
js闭包学习心得总结
Apr 17 #Javascript
Vue使用json-server进行后端数据模拟功能
Apr 17 #Javascript
js实现点击按钮复制文本功能
Jul 20 #Javascript
Element-UI Table组件上添加列拖拽效果实现方法
Apr 14 #Javascript
React中的render何时执行过程
Apr 13 #Javascript
You might like
删除及到期域名的查看(抢域名必备哦)
2008/05/14 PHP
ThinkPHP框架实现的邮箱激活功能示例
2018/06/15 PHP
Ajax请求PHP后台接口返回信息的实例代码
2018/08/21 PHP
Laravel路由研究之domain解决多域名问题的方法示例
2019/04/04 PHP
PHP 实现 WebSocket 协议原理与应用详解
2020/04/22 PHP
LazyLoad 延迟加载(按需加载)
2010/05/31 Javascript
Jquery拖拽并简单保存的实现代码
2010/11/28 Javascript
基于JQuery的浮动DIV显示提示信息并自动隐藏
2011/02/11 Javascript
关于textarea提交的内容无法换行的解决办法
2013/04/09 Javascript
JavaScript返回上一页的三种方法及区别介绍
2015/07/04 Javascript
Angular.js如何从PHP读取后台数据
2016/03/24 Javascript
简单实现js上传文件功能
2017/08/21 Javascript
JS实现浏览上传文件的代码
2017/08/23 Javascript
vue用addRoutes实现动态路由的示例
2017/09/15 Javascript
深入浅出理解JavaScript闭包的功能与用法
2018/08/01 Javascript
JavaScript基于数组实现的栈与队列操作示例
2018/12/22 Javascript
layui字体图标 loading图标静止不旋转的解决方法
2019/09/23 Javascript
[01:58]DOTA2上海特级锦标赛现场采访:RTZ这个ID到底好不好
2016/03/25 DOTA
使用BeautifulSoup爬虫程序获取百度搜索结果的标题和url示例
2014/01/19 Python
python实现在pandas.DataFrame添加一行
2018/04/04 Python
解决pytorch-yolov3 train 报错的问题
2020/02/18 Python
python实现音乐播放器 python实现花框音乐盒子
2020/02/25 Python
tensorflow从ckpt和从.pb文件读取变量的值方式
2020/05/26 Python
Python切片列表字符串如何实现切换
2020/08/06 Python
Django filter动态过滤与排序实现过程解析
2020/11/26 Python
德国Discount-Apotheke中文官网:DC德式康线上药房
2020/02/18 全球购物
为什么UNION ALL比UNION快
2016/03/17 面试题
厂长助理岗位职责
2013/12/27 职场文书
七夕情人节促销方案
2014/06/07 职场文书
六一儿童节活动总结
2014/08/27 职场文书
股东合作协议书
2014/09/12 职场文书
年度考核个人总结
2015/03/06 职场文书
python批量更改目录名/文件名的方法
2021/04/18 Python
探讨Java中的深浅拷贝问题
2021/06/26 Java/Android
特别篇动画《总之就是非常可爱 ~制服~》PV公开,2022年夏季播出
2022/04/04 日漫
《杜鹃的婚约》OP主题曲「凸凹」无字幕影像公开
2022/04/08 日漫