vue-router+vuex addRoutes实现路由动态加载及菜单动态加载

此案例主要实现了一个功能是,在vue实例首次运行时,在加载了login和404两个路由规则,登录成功后,根据登录用户角色权限获取该角色相应菜单权限,生成新的路由规则添加进去。

做过后台管理系统都一定做过这个功能,在对菜单权限进行粗粒度权限控制的时候,通过角色获取菜单后,异步生成菜单,所以一开始拿到需求的时候,我也以为这和平常的没什么不同,不过做起来就发现了很多问题,

1.vue-router的实例,在new vue实例的时候,就加载了,且必须加载,这个时候,登录路由一定要加载,可是这个时候没有登录,无法确定权限
2.路由规则与菜单的同步

解决思路演化,菜单和路由同步,肯定是采用了vuex,一开始的思路的是,在一开始,就把所有的路由规则加载,然后在登录的时候,取得权限路由,对比两个路由,通过修改修改一个权限字段来隐藏菜单,如果在后台页面添加了新菜单规则,路由是按模块加载的不同的文件,这时对路由的文件进行新的读写,虽然可以解决问题,但是如果手动在浏览器地址上路由,依然可以访问,所以在路由的全局钩子上还要做拦截。

这个解决方案虽然解决,但是显的比较复杂,于是就想需找新的方法,重新浏览官方api,发现在2.2.0以后,官方新增了api,addRoutes,专门针对服务端渲染路由,那么这下问题就比较简单了,下面列出实现代码。以下代码不能直接复用,需要根据实际情况修改,只是提供思路

app.js

let permission = JSON.parse(window.sessionStorage.getItem('permission'))
if (permission) {
 store.commit(ADD_MENU, permission)
 router.addRoutes(store.state.menu.items)
}
router.beforeEach((route, redirect, next) => {
 if (state.app.device.isMobile && state.app.sidebar.opened) {
  store.commit(TOGGLE_SIDEBAR, false)
 }
 if (route.path === '/login') {
  window.sessionStorage.removeItem('user')
  window.sessionStorage.removeItem('permission')
  store.commit(ADD_MENU, [])
 }
 let user = JSON.parse(window.sessionStorage.getItem('user'))
 if (!user && route.path !== '/login') {
  next({ path: '/login' })
 } else {
  if (route.name) {
   next()
  } else {
   next({ path: '/nofound' })
  }
 }
}) 

登录的组件login.vue

<template>
 <el-form :model="ruleForm2" :rules="rules2" ref="ruleForm2" label-position="left" label-width="0px"
      class="demo-ruleForm login-container">
  <h3 class="title">系统登录</h3>
  <el-form-item prop="account">
   <el-input type="text" v-model="ruleForm2.account" auto-complete="off" placeholder="账号"></el-input>
  </el-form-item>
  <el-form-item prop="checkPass">
   <el-input type="password" v-model="ruleForm2.checkPass" auto-complete="off" placeholder="密码"></el-input>
  </el-form-item>
  <el-checkbox v-model="checked" checked class="remember">记住密码</el-checkbox>
  <el-form-item style="width:100%;">
   <el-button type="primary" style="width:100%;" @click.native.prevent="handleSubmit2" :loading="logining">登录
   </el-button>
   <!--<el-button @click.native.prevent="handleReset2">重置</el-button>-->
  </el-form-item>
 </el-form>
</template> 

<script>
 import NProgress from 'nprogress'
 import { mapActions, mapGetters } from 'vuex'
 export default {
  data () {
   return {
    logining: false,
    ruleForm2: {
     account: 'admin',
     checkPass: '123456'
    },
    rules2: {
     account: [
      {required: true, message: '请输入账号', trigger: 'blur'}
      // { validator: validaePass }
     ],
     checkPass: [
      {required: true, message: '请输入密码', trigger: 'blur'}
      // { validator: validaePass2 }
     ]
    },
    checked: true
   }
  },
  computed: {
   ...mapGetters([
    'menuitems',
    'isLoadRoutes'
    // ...
   ])
  },
  methods: {
   handleReset2 () {
    this.$refs.ruleForm2.resetFields()
   },
   handleSubmit2 (ev) {
    this.$refs.ruleForm2.validate((valid) => {
     if (valid) {
      this.logining = true
      NProgress.start()
      let loginParams = {loginName: this.ruleForm2.account, password: this.ruleForm2.checkPass}
      this.$http.post('/api/privilege/user/login', loginParams).then(resp => {
       this.logining = false
       NProgress.done()
       let {message, data} = resp.data 

       if (message === 'fail') {
        this.$notify({
         title: '错误',
         message: message,
         type: 'error'
        })
       } else {
        window.sessionStorage.setItem('user', JSON.stringify(data.user))
        window.sessionStorage.setItem('permission', JSON.stringify(data.permission))
        this.addMenu(data.permission)
        if (!this.isLoadRoutes) {
         this.$router.addRoutes(this.menuitems)
         this.loadRoutes()
        }
        this.$router.push('/system/office')
       }
      })
     } else {
      console.log('error submit!!')
      return false
     }
    })
   }, 

   ...mapActions([
    'addMenu',
    'loadRoutes'
   ])
  }
 } 

</script> 

<style lang="scss" scoped>
 .login-container {
  /*box-shadow: 0 0px 8px 0 rgba(0, 0, 0, 0.06), 0 1px 0px 0 rgba(0, 0, 0, 0.02);*/
  -webkit-border-radius: 5px;
  border-radius: 5px;
  -moz-border-radius: 5px;
  background-clip: padding-box;
  margin-bottom: 20px;
  background-color: #F9FAFC;
  margin: 180px auto;
  border: 2px solid #8492A6;
  width: 350px;
  padding: 35px 35px 15px 35px; 

 .title {
  margin: 0px auto 40px auto;
  text-align: center;
  color: #505458;
 } 

 .remember {
  margin: 0px 0px 35px 0px;
 } 

 }
</style> 

关键点解释

computed: {
   ...mapGetters([
    'menuitems',
    'isLoadRoutes'
    // ...
   ])
  }, 

这里是从vuex取得两个对象,menuitems是菜单对象,isLoadRoutes是用来判断是否是第一次登录,用来排除重复加载路由规则

...mapActions([
    'addMenu',
    'loadRoutes'
   ]) 

这里是从vuex取得两个方法,一个是添加菜单,一个更改loadRoutes的值

this.$router.addRoutes(this.menuitems) 

这是关键api,动态的向router实例中添加路由规则

menu模块的state与mutations

const state = {
 items: [
 ],
 isLoadRoutes: false
} 

const mutations = {
 [types.EXPAND_MENU] (state, menuItem) {
  if (menuItem.index > -1) {
   if (state.items[menuItem.index] && state.items[menuItem.index].meta) {
    state.items[menuItem.index].meta.expanded = menuItem.expanded
   }
  } else if (menuItem.item && 'expanded' in menuItem.item.meta) {
   menuItem.item.meta.expanded = menuItem.expanded
  }
 },
 [types.ADD_MENU] (state, menuItems) {
  if (menuItems.length === 0) {
   state.items = []
  } else {
   generateMenuItems(state.items, menuItems)
  }
 },
 [types.LOAD_ROUTES] (state) {
  state.isLoadRoutes = !state.isLoadRoutes
 }
}

路由配置文件router.js

import Vue from 'vue'
import Router from 'vue-router'
import menuModule from 'vuex-store/modules/menu'
Vue.use(Router) 

export default new Router({
 mode: 'hash', // Demo is living in GitHub.io, so required!
 linkActiveClass: 'is-active',
 scrollBehavior: () => ({ y: 0 }),
 routes: [
  {
   path: '/login',
   component: require('../Login.vue'),
   meta: {
    expanded: false,
    show: false
   },
   name: 'Login'
  },
  {
   path: '/',
   component: require('../views/Home.vue'),
   meta: {
    expanded: false,
    show: false
   },
   children: [
    { path: '/nofound', component: require('../404.vue'), name: 'NOFOUND', meta: {show: false} }
   ]
  },
  ...generateRoutesFromMenu(menuModule.state.items)
 ]
}) 

// Menu should have 2 levels.
function generateRoutesFromMenu (menu = [], routes = []) {
 for (let i = 0, l = menu.length; i < l; i++) {
  let item = menu[i]
  if (item.path) {
   routes.push(item)
  }
 }
 return routes
}

vuex

import Vue from 'vue'
import Vuex from 'vuex'
import * as actions from './actions'
import * as getters from './getters' 

import menu from './modules/menu' 

Vue.use(Vuex) 

const store = new Vuex.Store({
 strict: true, // process.env.NODE_ENV !== 'development',
 actions,
 getters,
 modules: {
  menu
 },
 mutations: {
 }
}) 

export default store 

actions

export const addMenu = ({ commit }, menuItems) => {
 if (menuItems.length > 0) {
  commit(types.ADD_MENU, menuItems)
 }
} 

export const loadRoutes = ({ commit }) => {
 commit(types.LOAD_ROUTES)
} 

getters

const menuitems = state => state.menu.items
const isLoadRoutes = state => state.menu.isLoadRoutes
export {
 menuitems,
 isLoadRoutes
} 

mutations_type.js

export const ADD_MENU = 'ADD_MENU' 

export const LOAD_ROUTES = 'LOAD_ROUTES' 

因为上面的代码不能直接运行,再次梳理一下思路,

1.创建vue实例的时候,将vuex和vue-router加载,这个时候,vue-router只有登录规则和404规则

2.vuex中state管理的状态对象有,菜单对象menuitems,是否加载过路由loadRoutes ,并提供相应的getters与actions当然还有一些其他的,这里没有列举

3.然后在登录组件中,登录成功后,将服务端传回来之后,调用actions更改state.menuitems,并且中间有格式化的过程,这个过程的代码没有贴出来,主要是由于不同的表涉和服务端返回的数据不一样,,

4.然后调用addRoutes和actions更改已经加载过路由的方法

5.然后为了防止用户直接手动按f5刷新页面,这个时候会重新构建vue实例,而又没有重新登录,所以vuex里面的东西会清空,所以将登录后的数据存放在sessionStroage中,在刷新页面,重新构建vue实例的时候,会有判断

6.之后会渲染侧边栏组件,列出菜单,数据就可以根据state.menuitems来就可以了,我这里没有贴我的,实际根据自己的需求来

后面有时间会在github上上传完整代码。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • vue用addRoutes实现动态路由的示例

    之前在基于Vue实现后台系统权限控制一文中提到路由权限的实现思路,因为不喜欢在每次路由跳转的before钩子里做判断,所以在初始化Vue实例前对路由做了筛选,再用实际路由初始化Vue实例,代价是登录页需要从Vue实例中独立出来,实现上倒没什么问题,不过这种做法需要在登录和首页之间通过url跳转,感觉总是不太"优雅",实际上只要能在登录后动态修改当前实例的路由就行了,之前确实没办法,但vue-router 2.2版本新增了一个router.addRoutes(routes)方法,让动态路

  • vue-router+vuex addRoutes实现路由动态加载及菜单动态加载

    此案例主要实现了一个功能是,在vue实例首次运行时,在加载了login和404两个路由规则,登录成功后,根据登录用户角色权限获取该角色相应菜单权限,生成新的路由规则添加进去. 做过后台管理系统都一定做过这个功能,在对菜单权限进行粗粒度权限控制的时候,通过角色获取菜单后,异步生成菜单,所以一开始拿到需求的时候,我也以为这和平常的没什么不同,不过做起来就发现了很多问题, 1.vue-router的实例,在new vue实例的时候,就加载了,且必须加载,这个时候,登录路由一定要加载,可是这个时候没有登

  • vue router+vuex实现首页登录验证判断逻辑

    首页登录逻辑要求在页面上判断是否获取到登录token ,没有获取到则跳转到登录页.登录成功后,跳转到前一个页面. 1.vue router 路由判断首先我们想到的是router.beforeEach 前置导航守卫 ,这个方法接受三个参数 to from next . to参数为即将跳转的路由路径,from为当前导航正要离开的路由,next方法用来resolve这个钩子. 下面以工作中写的一个判断为为例子: router.beforeEach(async (to, from, next) => {

  • 基于vue,vue-router, vuex及addRoutes进行权限控制问题

    基于vuex, vue-router,vuex的权限控制教程,完整代码地址见 https://github.com/linrunzheng/vue-permission-control 接下来让我们模拟一个普通用户打开网站的过程,一步一步的走完整个流程. 首先从打开本地的服务localhost:8080开始,我们知道打开后会进入login页面,那么判断的依据是什么. 首先是token. 没有登陆的用户是获取不到token的,而登陆后的角色我们会将token存到local或者seesionStor

  • web前端vue之vuex单独一文件使用方式实例详解

    Vuex 是什么? Vuex 是一个专为 Vue.js 应用程序开发的状态管理模式.它采用集中式存储管理应用的所有组件的状态,并以相应的规则保证状态以一种可预测的方式发生变化.Vuex 也集成到 Vue 的官方调试工具 devtools extension,提供了诸如零配置的 time-travel 调试.状态快照导入导出等高级调试功能. 上次我用了一个加减的例子为大家讲解vuex的基本的使用方式,和在什么样的情况下使用.上次还是在一个组件内把这个例子简单的展示了下,这次我把vuex抽离出来一个

  • 基于Vue、Vuex、Vue-router实现的购物商城(原生切换动画)效果

    效果图如下所示: 在线地址: github.czero.cn/fancy 点击下载安卓apk安装包 源码地址: github.com/czero1995/f- 项目主架构 使用的库 vue-cli (vue+webpack脚手架) vue-router(路由跳转) vuex(状态管理) axios(数据请求) mock.js(模拟后台数据) vue-touch(手势判断) fastclick(解决移动端浏览器 300 毫秒点击延迟问题) vue-lazyload(图片懒加载) swiper(轮播

  • Vue Router中应用中间件的方法

    中间件是我们在软件开发中的一个古老而强大的概念,当我们在应用程序中使用路由相关模式时,它非常有用. 如果您不太了解中间件的含义,Nodejs框架Express里的中间件可以帮助您了解它们的工作原理. 但是,中间件仅适用于后端吗? 不,当应用程序中有路由时,中间件在前端或后端中就会非常常见.比如现在流行的单页应用程序. 有一些示例可以说明,何时可以使用中间件: 不允许未登录用户访问您的网页. 仅允许某些类型的用户查看页面(角色:管理员,作者等) 数据采集. 重置设置或清理存储空间. 限制访问用户的

  • Vue Router 实现动态路由和常见问题及解决方法

    个人理解:动态路由不同于常见的静态路由,可以根据不同的「因素」而改变站点路由列表.常见的动态路由大都是用来实现:多用户权限系统不同用户展示不同导航菜单. 如何利用Vue Router 实现动态路由 Vue项目实现动态路由的方式大体可分为两种: 前端将全部路由规定好,登录时根据用户角色权限来动态展示路由: 路由存储在数据库中,前端通过接口获取当前用户对应路由列表并进行渲染: 第一种方式在很多Vue UI Admin上都实现了,可以去读一下他们的源码理解具体的实现思路,这里就不过多展开.第二种方式现

  • vue router学习之动态路由和嵌套路由详解

    本文主要参考:https://router.vuejs.org/zh-cn/essentials/nested-routes.html 本文的阅读前提是已经能够搭建一个vue前台程序并且运行.如果还么有搭建可以参考文章: http://www.jb51.net/article/111650.htm 好,下面上货. 首先介绍一下动态路由. 动态路由按照我的理解,就是说能够进行页面的跳转,比如说:下面的这个页面中: <template> <div id="app">

  • 全面解析vue router 基本使用(动态路由,嵌套路由)

    路由,其实就是指向的意思,当我点击页面上的home按钮时,页面中就要显示home的内容,如果点击页面上的about 按钮,页面中就要显示about 的内容.Home按钮  => home 内容, about按钮 => about 内容,也可以说是一种映射. 所以在页面上有两个部分,一个是点击部分,一个是点击之后,显示内容的部分. 点击之后,怎么做到正确的对应,比如,我点击home 按钮,页面中怎么就正好能显示home的内容.这就要在js 文件中配置路由. 路由中有三个基本的概念 route,

  • vue router动态路由下让每个子路由都是独立组件的解决方案

    vue-router 之动态路由 vue-router官网上面是这样说的 // 带查询参数,变成 /register?plan=private router.push({ path: 'register', query: { plan: 'private' }}) 然后,我就这样写了: this.$router.push({path:'manage', query: {id: 'tasklist'}})1 结果很明显,失败了.然后我就默默的再次看了一下官网,结果发现了这句话 // 命名的路由 r

随机推荐