基于iview-admin實(shí)現(xiàn)動(dòng)態(tài)路由的示例代碼
iview-admin是一個(gè)基于vue和iview組件庫實(shí)現(xiàn)的管理后臺(tái)前端,本文基于iview-admin最新版本,實(shí)現(xiàn)基于權(quán)限的動(dòng)態(tài)路由加載。
本文代碼可參見:https://github.com/MayBeWrong/iview-admin-dynamic-router
背景:
動(dòng)態(tài)路由:vue的路由,可通過new Router傳入路由數(shù)組定義實(shí)現(xiàn),也可以通過router.addRoutes實(shí)現(xiàn)。通過router.addRoutes動(dòng)態(tài)傳入路由定義的方式,稱之為動(dòng)態(tài)路由。路由數(shù)據(jù)可以全部保存在后臺(tái)數(shù)據(jù)庫中,也可以將路由配置在前端,后端返回給前端路由權(quán)限信息,然后匹配過濾,進(jìn)行加載。本文就這兩種方式分別進(jìn)行介紹,并且給出實(shí)現(xiàn)參考。
目標(biāo):
基于iview-admin最新代碼,實(shí)現(xiàn)兩種不同的路由動(dòng)態(tài)加載方式:
- 路由(導(dǎo)航菜單)數(shù)據(jù)全部存儲(chǔ)在后臺(tái)
- 路由數(shù)據(jù)配置在前端,后臺(tái)只存儲(chǔ)權(quán)限信息
注意:本文通過Mock模擬后端接口
方式1:路由(導(dǎo)航菜單)數(shù)據(jù)全部存儲(chǔ)在后臺(tái)
定義路由數(shù)據(jù)結(jié)構(gòu)體,在文件中:src/mock/data.js
export const routersData = [{ path: '/pet',//訪問路徑 name: 'Pet',//路由的名字,這個(gè)與i18n有關(guān),需要唯一 meta: { title: '寵物',//標(biāo)題 hideInMenu: false,//是否在左側(cè)導(dǎo)航菜單隱藏 icon: 'logo-freebsd-devil'//圖標(biāo) }, component: 'components/main',//組件文件路徑,不需要Import children: [{//嵌套路由 path: 'cat', name: 'Cat', meta: { title: '貓咪', hideInMenu: false, icon: 'ios-cloudy-night' }, component: 'view/pet/cat/Cat.vue' }, { path: 'dog', name: 'Dog', meta: { hideInMenu: false, title: '狗娃', icon: 'ios-color-filter' }, component: 'view/pet/dog/Dog.vue' }, { path: 'pig', name: 'Pig', meta: { hideInMenu: false, title: '豬啊', icon: 'ios-contact' }, component: 'view/pet/pig/Pig.vue', children: [ { path: 'female', name: 'Female', meta: { hideInMenu: false, title: '母豬', icon: 'ios-contact' }, component: 'view/pet/pig/Pig.vue', }, { path: 'male', name: 'Male', meta: { hideInMenu: false, title: '公豬', icon: 'ios-contact' }, component: 'view/pet/pig/Pig.vue', } ] }]}]
暴露ajax調(diào)用接口:src/mock/index.js,中增加:
Mock.mock(/\/sys\/routers/, routersData)
實(shí)現(xiàn)一個(gè)ajax調(diào)用:src/api/routers.js中增加:
export const getRouterReq = (access) => { return axios.request({ url: '/sys/routers', params: { access }, method: 'get' })}
1、在store中定義動(dòng)態(tài)路由相關(guān)邏輯,修改:src/store/module/app.js
引入ajax請(qǐng)求:
import {getRouterReq} from '@/api/routers'
定義兩個(gè)state,如下
state: { ..... routers: [],//拿到的路由數(shù)據(jù) hasGetRouter: false//是否已經(jīng)拿過路由數(shù)據(jù) },
同步增加mutations:
mutations:{ ...... //設(shè)置路由數(shù)據(jù) setRouters(state, routers) { state.routers = routers }, //設(shè)置是否已經(jīng)拿過路由 setHasGetRouter(state, status) { state.hasGetRouter = status }......}
增加一個(gè)action:
action:{ ........ getRouters({commit}) { return new Promise((resolve, reject) => { try { getRouterReq().then(res => { let routers = backendMenusToRouters(res.data) commit('setRouters', routers) commit('setHasGetRouter', true) resolve(routers) }).catch(err => { reject(err) }) } catch (error) { reject(error) } }) }, ........ }
此處用到了一個(gè)函數(shù):backendMenusToRouters,這個(gè)函數(shù)定義在src/libs/util.js中,用來對(duì)后端返回的路由數(shù)據(jù)遞歸處理,行程vue的路由。
export const backendMenusToRouters = (menus) => { let routers = [] forEach(menus, (menu) => { // 將后端數(shù)據(jù)轉(zhuǎn)換成路由數(shù)據(jù) let route = backendMenuToRoute(menu) // 如果后端數(shù)據(jù)有下級(jí),則遞歸處理下級(jí) if (menu.children && menu.children.length !== 0) { route.children = backendMenusToRouters(menu.children) } routers.push(route) }) return routers }
修改src/router/index.js,增加動(dòng)態(tài)路由加入邏輯,主要方法:
const initRouters = (store) => { //這個(gè)人登錄了已經(jīng) if (store.state.user.hasGetInfo) { //路由加載過了 if (store.state.app.hasGetRouter && store.state.app.routers && store.state.app.routers.length > 0) { console.log("已經(jīng)加載過了路由") } else { //加載路由 console.log("開始加載路由權(quán)限...") store.dispatch('getUserMenus').then(routers => { //此處routers已經(jīng)是按照權(quán)限過濾后的路由了,沒權(quán)限的,不加入路由,無法訪問 //路由重置一下把404放最后 const newRouter = new Router({ routes, mode: config.routerModel }) router.matcher = newRouter.matcher; //把404加最后面,如果用router.push({name:'xxxx'})這種的話,404頁面可能空白,用path:'/aa/bb' router.addRoutes(routers.concat([{ path: '*', name: 'error_404', meta: { hideInMenu: true }, component: () => import(/* webpackChunkName: "404" */'@/view/error-page/404.vue') }])) }).finally(() => { }) } }}
每次路由加載之前,都會(huì)判斷是否已經(jīng)初始化過系統(tǒng)路由,如果沒有,則初始化。
至此,動(dòng)態(tài)路由基本實(shí)現(xiàn)。文章可能有遺漏和不足,歡迎探討。第二種實(shí)現(xiàn)方式
具體實(shí)現(xiàn),請(qǐng)參見: https://github.com/MayBeWrong/iview-admin-dynamic-router
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Vue使用Composition?API生成計(jì)算屬性computed
這篇文章主要為大家詳細(xì)介紹了Vue如何使用Composition?API實(shí)現(xiàn)生成計(jì)算屬性computed,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-06-06vue代理請(qǐng)求之Request?failed?with?status?code?404問題及解決
這篇文章主要介紹了vue代理請(qǐng)求之Request?failed?with?status?code?404問題及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-07-07Vue3新增時(shí)自動(dòng)獲取當(dāng)前時(shí)間的操作方法
這篇文章主要介紹了Vue3新增時(shí)自動(dòng)獲取當(dāng)前時(shí)間的操作方法,本文通過實(shí)例代碼圖文相結(jié)合給大家介紹的非常詳細(xì),感興趣的朋友跟隨小編一起看看吧2024-07-07vue學(xué)習(xí)筆記五:在vue項(xiàng)目里面使用引入公共方法詳解
這篇文章主要介紹了在vue項(xiàng)目里面使用引入公共方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04ant design vue中日期選擇框混合時(shí)間選擇器的用法說明
這篇文章主要介紹了ant design vue中日期選擇框混合時(shí)間選擇器的用法說明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-10-10vue中element-ui不能修改el-input框,或是不能修改某些值問題
這篇文章主要介紹了vue中element-ui不能修改el-input框,或是不能修改某些值問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-10-10Vue3父子組件傳參有關(guān)sync修飾符的用法詳解
這篇文章主要給大家介紹關(guān)于前端Vue3父子組件傳參有關(guān)sync修飾符的用法詳細(xì)解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助2021-09-09IE11下處理Promise及Vue的單項(xiàng)數(shù)據(jù)流問題
最近我開發(fā)的公司的競(jìng)賽網(wǎng)站被發(fā)現(xiàn)在IE11下排行榜無數(shù)據(jù),但是在其他瀏覽器沒問題,我然后打開控制臺(tái)一看,發(fā)現(xiàn)有問題,糾結(jié)了半天才搗騰好,下面小編把方案分享處理,供大家選擇2019-07-07vue基于Teleport實(shí)現(xiàn)Modal組件
Teleport 提供了一種干凈的方法,允許我們控制在 DOM 中哪個(gè)父節(jié)點(diǎn)下渲染了 HTML,而不必求助于全局狀態(tài)或?qū)⑵洳鸱譃閮蓚€(gè)組件。2021-05-05