vue系列之動(dòng)態(tài)路由詳解【原創(chuàng)】
開題
最近用vue來構(gòu)建了一個(gè)小項(xiàng)目,由于項(xiàng)目是以iframe的形式嵌套在別的項(xiàng)目中的,所以對于登錄的驗(yàn)證就比較的麻煩,索性后端大佬們基于現(xiàn)在的問題提出了解決的方案,在看到他們的解決方案之前,我先畫了一個(gè)比較標(biāo)準(zhǔn)的單系統(tǒng)的解決方案。
本文目錄:
一: 設(shè)想
二: 討論
三:實(shí)現(xiàn)
四:總結(jié)
一: 設(shè)想

簡單解釋下上圖就是:
首先前端從cookie獲取token,如果沒有token就跳轉(zhuǎn)到登錄頁面登錄,登錄驗(yàn)證之后生成token存在數(shù)據(jù)庫中并返回給前端;前端將這個(gè)token保存下來,為了讓在瀏覽器新的tab頁時(shí)不需要登錄,我們前端需要將這個(gè)token保存到cookie之中。
如果用戶已經(jīng)有了token,那么再驗(yàn)證是否有用戶信息,如果沒有去請求用戶信息的接口,后臺讀取用戶的基本信息返回給前端,前端根據(jù)后臺返回的用戶權(quán)限生成固定的路由表用于頁面攔截。
在用戶token和權(quán)限都有的情況下,進(jìn)入自己權(quán)限內(nèi)的頁面并且攜帶token訪問后臺進(jìn)行交互。
用戶在退出時(shí),請求后臺接口,清除token數(shù)據(jù)。
二: 討論
由于公司的項(xiàng)目更加的復(fù)雜,屬于基于原來的系統(tǒng)開發(fā)新的系統(tǒng)模塊,但是這些模塊又為了以后主體功能的更新下次迭代需要保持相對的獨(dú)立性,預(yù)計(jì)以后的老系統(tǒng)只起一個(gè)用戶中心的功能,所以現(xiàn)在是基于實(shí)現(xiàn)單點(diǎn)登錄的能力去迭代更新現(xiàn)在的新的項(xiàng)目。
今天上午對于登錄的實(shí)現(xiàn)進(jìn)行了相關(guān)討論,由于公司項(xiàng)目保密考慮只是單單做相關(guān)的介紹:
現(xiàn)有的老項(xiàng)目將慢慢向用戶中心轉(zhuǎn)換,而以前的新項(xiàng)目需要去這個(gè)用戶中心獲取登錄信息。具體的實(shí)現(xiàn)是:
登錄新項(xiàng)目b.exaplem.com通過session檢測到未登錄時(shí)(這里說下新的項(xiàng)目和老項(xiàng)目在同一個(gè)一級域名下),跳轉(zhuǎn)到a.exaplem.com?returnUrl='b.exaplem.com',在a.exaplem.com下成功登錄后生成一個(gè)ticket給到b.exaplem.com,b.exaplem.com將這個(gè)ticket存在session里面來保持登錄狀態(tài)。
因?yàn)楝F(xiàn)在基本上是先登錄a.exaplem.com然后再去登錄b.exaplem.com,所以當(dāng)我們第一次進(jìn)入b.exaplem.com系統(tǒng)時(shí),b.exaplem.com會(huì)向a.exaplem.com系統(tǒng)發(fā)送請求來獲取ticket,并且生成session來維持登錄狀態(tài)。
三:實(shí)現(xiàn)
當(dāng)時(shí)想通過引入vuex并通過cookie來保存token的狀態(tài),但是由于現(xiàn)在的項(xiàng)目還是后端以session的形式來維持用戶的登錄狀態(tài)所以還是沒有引入vuex。
基本實(shí)現(xiàn)如下:
main.js增加引入的permission.js文件如下:
import Vue from 'vue'
import router from './router'
import { asyncRouterMap, constantRouterMap } from './router'
function hasPermission(roles, route) { //
if (route.meta && route.meta.role) {
return roles.some(role => role === route.meta.role)
} else {
return true
}
}
function filterAsyncRouter(asyncRouterMap, roles) {
const accessedRouters = asyncRouterMap.filter(route => {
if (hasPermission(roles, route)) {
if (route.children && route.children.length) {
route.children = filterAsyncRouter(route.children, roles)
}
return true
}
return false
})
return accessedRouters
}
// 加載頁面之前
router.beforeEach((to, from, next) => {
NProgress.start() // 開啟Progress
if (to.path == '/ContractAduit/Error') {
next()
} else if (!Vue.prototype.hasRoute) {
Vue.prototype.$ajax.get(Vue.prototype.$api.getModuleHost("用戶信息接口地址"), {})
.then(data => {
if (data.code == 1000) {
let menus = data.menu
let roles = menus.map((menu, index) => {
return parseInt(menu.url);
})
const accessedRouters = filterAsyncRouter(asyncRouterMap, roles)
router.addRoutes(accessedRouters)
Vue.prototype.userInfo = {
id: data.id,
realname: data.realname
}
Vue.prototype.hasRoute = true;
next({...to })
} else {
router.push({ name: 'ErrorPageRouter' });
}
})
.catch(err => console.log(err))
} else {
next()
}
});
route.js文件如下:
import Vue from 'vue'
import Router from 'vue-router'
Vue.use(Router)
export const constantRouterMap = [{
path: '/404',
name: 'NoFoundPagetRouter',
component: require('../views/404.page'),
meta: {
title: '404',
}
},
{
path: '/ContractAduit/NoAccess',
name: 'NoAccessPageRouter',
component: require('../views/no-access.page'),
meta: {
title: '無權(quán)限',
}
},
{
path: '/ContractAduit/Error',
name: 'ErrorPageRouter',
component: require('../views/error.page'),
meta: {
title: '內(nèi)部錯(cuò)誤',
}
}
]
export default new Router({
mode: 'history',
routes: constantRouterMap
})
export const asyncRouterMap = [{
path: '/ContractAduit/Supplier/List',
name: 'SupplierListPageRouter',
component: require('../views/supplier/supplier-list.page.vue'),
meta: {
title: '某某列表頁',
role: 10001
}
},
...
{
path: '/',
redirect: '/ContractAduit/Supplier/List',
hidden: true,
meta: {
title: '某某列表頁',
role: 10001
}
}, {
path: '*',
redirect: '/404',
hidden: true
}
]
因?yàn)闆]有引入vuex所以需要在VUE構(gòu)造函數(shù)的原型對象上聲明變量來判斷是否已經(jīng)拉取了用戶的基本信息,因?yàn)槲覀兒蠖说臋?quán)限是配置的頁面級權(quán)限(即不是按照角色來安排哪個(gè)前端頁面來可訪問,而是根據(jù)后臺返回的頁面代碼來判斷哪個(gè)前端頁面可訪問)。
四:總結(jié)
因?yàn)槲覀兊捻?xiàng)目不可能達(dá)到千篇一律的情況,選擇適合自己項(xiàng)目的解決方案才是最重要的,不要為了用某個(gè)技術(shù)而去用某個(gè)技術(shù)。
以上這篇vue系列之動(dòng)態(tài)路由詳解【原創(chuàng)】就是小編分享給大家的全部內(nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
django使用channels2.x實(shí)現(xiàn)實(shí)時(shí)通訊
這篇文章主要介紹了django使用channels2.x實(shí)現(xiàn)實(shí)時(shí)通訊,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-11-11
vue+Java后端進(jìn)行調(diào)試時(shí)解決跨域問題的方式
今天在開發(fā)中遇到有點(diǎn)小問題,vue+Java后端進(jìn)行調(diào)試時(shí)如何解決跨域問題,下面小編給大家分享解決方法,感興趣的朋友一起看看吧2017-10-10
vue 實(shí)現(xiàn)cli3.0中使用proxy進(jìn)行代理轉(zhuǎn)發(fā)
今天小編就為大家分享一篇vue 實(shí)現(xiàn)cli3.0中使用proxy進(jìn)行代理轉(zhuǎn)發(fā),具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-10-10
Vue?3?使用moment設(shè)置顯示時(shí)間格式的問題及解決方法
在Vue?3中,因?yàn)檫^濾器(filter)已經(jīng)被廢棄,取而代之的是全局方法(global?method),本文給大家介紹Vue?3?使用moment設(shè)置顯示時(shí)間格式的問題及解決方法,感興趣的朋友一起看看吧2023-12-12

