mirror of
https://github.com/chansee97/nova-admin.git
synced 2025-04-06 03:57:54 +08:00
feat(projects): 添加路由拦截判断权限
This commit is contained in:
parent
c646819b23
commit
4acb525777
@ -13,7 +13,7 @@ const userInfo = {
|
|||||||
role: 'admin',
|
role: 'admin',
|
||||||
password: '123456',
|
password: '123456',
|
||||||
token,
|
token,
|
||||||
permissions: [
|
userRoutes: [
|
||||||
{
|
{
|
||||||
name: 'dashboard',
|
name: 'dashboard',
|
||||||
path: '/dashboard',
|
path: '/dashboard',
|
||||||
|
@ -9,10 +9,10 @@ async function setupApp() {
|
|||||||
setupAssets();
|
setupAssets();
|
||||||
// 创建vue实例
|
// 创建vue实例
|
||||||
const app = createApp(App);
|
const app = createApp(App);
|
||||||
// 安装router
|
|
||||||
await setupRouter(app);
|
|
||||||
// 安装pinia全局状态库
|
// 安装pinia全局状态库
|
||||||
setupStore(app);
|
setupStore(app);
|
||||||
|
// 安装router
|
||||||
|
await setupRouter(app);
|
||||||
// 挂载
|
// 挂载
|
||||||
app.mount('#app');
|
app.mount('#app');
|
||||||
}
|
}
|
||||||
|
@ -1,25 +1,20 @@
|
|||||||
import type { Router } from 'vue-router';
|
import type { Router } from 'vue-router';
|
||||||
import { getToken } from '@/utils/auth';
|
|
||||||
|
import { createPermissionGuard } from './permission';
|
||||||
|
|
||||||
const appTitle = import.meta.env.VITE_APP_TITLE;
|
const appTitle = import.meta.env.VITE_APP_TITLE;
|
||||||
|
|
||||||
export function setupRouterGuard(router: Router) {
|
export function setupRouterGuard(router: Router) {
|
||||||
const isLogin = Boolean(getToken());
|
router.beforeEach(async (to, from, next) => {
|
||||||
|
// 开始 loadingBar
|
||||||
router.beforeEach((to, _from, next) => {
|
window.$loadingBar?.start();
|
||||||
// 登录鉴权
|
// 权限操作
|
||||||
if (!isLogin) {
|
await createPermissionGuard(to, from, next);
|
||||||
if (to.name === 'login') {
|
});
|
||||||
next();
|
router.afterEach((to) => {
|
||||||
} else {
|
|
||||||
const redirect = to.fullPath;
|
|
||||||
next({ path: '/login', query: { redirect } });
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
// 修改网页标题
|
// 修改网页标题
|
||||||
document.title = `${to.meta.title}——${appTitle}`;
|
document.title = `${to.meta.title}——${appTitle}`;
|
||||||
next();
|
// 结束 loadingBar
|
||||||
|
window.$loadingBar?.finish();
|
||||||
});
|
});
|
||||||
// router.afterEach((_to) => {});
|
|
||||||
}
|
}
|
||||||
|
37
src/router/guard/permission.ts
Normal file
37
src/router/guard/permission.ts
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
import { RouteLocationNormalized, NavigationGuardNext } from 'vue-router';
|
||||||
|
import { getToken } from '@/utils/auth';
|
||||||
|
import { useRouteStore, useAuthStore } from '@/store';
|
||||||
|
|
||||||
|
export async function createPermissionGuard(
|
||||||
|
to: RouteLocationNormalized,
|
||||||
|
from: RouteLocationNormalized,
|
||||||
|
next: NavigationGuardNext,
|
||||||
|
) {
|
||||||
|
const isLogin = Boolean(getToken());
|
||||||
|
const routeStore = useRouteStore();
|
||||||
|
|
||||||
|
// 判断路由有无进行初始化
|
||||||
|
if (!routeStore.isInitAuthRoute) {
|
||||||
|
// 没有初始化路由 => 登录鉴权
|
||||||
|
// 登录鉴权
|
||||||
|
if (!isLogin) {
|
||||||
|
if (to.name === 'login') {
|
||||||
|
next();
|
||||||
|
} else {
|
||||||
|
const redirect = to.fullPath;
|
||||||
|
next({ path: '/login', query: { redirect } });
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 有登录但是没有路由,初始化路由等
|
||||||
|
await routeStore.initAuthRoute();
|
||||||
|
}
|
||||||
|
// 权限路由已经加载,仍然未找到,重定向到not-found
|
||||||
|
// if (to.name === 'not-found-page') {
|
||||||
|
// next({ name: 'not-found-page', replace: true });
|
||||||
|
// }
|
||||||
|
|
||||||
|
// next({ name: 'root' });
|
||||||
|
next();
|
||||||
|
}
|
@ -8,7 +8,7 @@ const routes: RouteRecordRaw[] = [
|
|||||||
{
|
{
|
||||||
path: '/',
|
path: '/',
|
||||||
name: 'root',
|
name: 'root',
|
||||||
redirect: '/test1',
|
redirect: '/test/test1',
|
||||||
component: BasicLayout,
|
component: BasicLayout,
|
||||||
children: [
|
children: [
|
||||||
{
|
{
|
||||||
|
@ -1,6 +1,14 @@
|
|||||||
// import { BasicLayout } from '@/layouts/index';
|
// import { BasicLayout } from '@/layouts/index';
|
||||||
|
|
||||||
export const constantRoutes = [
|
export const constantRoutes = [
|
||||||
|
{
|
||||||
|
path: '/no-found',
|
||||||
|
name: 'not-found',
|
||||||
|
component: () => import('@/views/inherit-page/not-found/index.vue'),
|
||||||
|
meta: {
|
||||||
|
title: '找不到页面',
|
||||||
|
},
|
||||||
|
},
|
||||||
{
|
{
|
||||||
path: '/no-permission',
|
path: '/no-permission',
|
||||||
name: 'no-permission',
|
name: 'no-permission',
|
||||||
@ -19,10 +27,10 @@ export const constantRoutes = [
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
path: '/:pathMatch(.*)*',
|
path: '/:pathMatch(.*)*',
|
||||||
name: '404',
|
name: 'not-found-page',
|
||||||
component: () => import('@/views/inherit-page/not-found/index.vue'),
|
component: () => import('@/views/inherit-page/not-found/index.vue'),
|
||||||
meta: {
|
meta: {
|
||||||
title: '错误404',
|
title: '找不到页面',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
@ -50,7 +50,8 @@ export const useAuthStore = defineStore('auth-store', {
|
|||||||
// 等待数据写入完成
|
// 等待数据写入完成
|
||||||
const catchSuccess = await this.catchUserInfo(data);
|
const catchSuccess = await this.catchUserInfo(data);
|
||||||
// 初始化侧边菜单
|
// 初始化侧边菜单
|
||||||
await this.setRouterStore(data.permissions);
|
const { setMenus } = useRouteStore();
|
||||||
|
await setMenus();
|
||||||
// 登录写入信息成功
|
// 登录写入信息成功
|
||||||
if (catchSuccess) {
|
if (catchSuccess) {
|
||||||
// 进行重定向跳转
|
// 进行重定向跳转
|
||||||
@ -65,7 +66,9 @@ export const useAuthStore = defineStore('auth-store', {
|
|||||||
});
|
});
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
// 如果不成功写到后面
|
|
||||||
|
// 如果不成功则重置存储
|
||||||
|
this.resetAuthStore();
|
||||||
},
|
},
|
||||||
|
|
||||||
/* 缓存用户信息 */
|
/* 缓存用户信息 */
|
||||||
@ -82,11 +85,5 @@ export const useAuthStore = defineStore('auth-store', {
|
|||||||
|
|
||||||
return catchSuccess;
|
return catchSuccess;
|
||||||
},
|
},
|
||||||
|
|
||||||
/* 将路由表等信息推送到routeStore */
|
|
||||||
async setRouterStore(permissions: Auth.UserInfoPermissions[]) {
|
|
||||||
const { setMenus } = useRouteStore();
|
|
||||||
setMenus(permissions);
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
@ -1,2 +1,3 @@
|
|||||||
export * from './app';
|
export * from './app';
|
||||||
export * from './auth';
|
export * from './auth';
|
||||||
|
export * from './route';
|
||||||
|
@ -1,23 +1,29 @@
|
|||||||
import { defineStore } from 'pinia';
|
import { defineStore } from 'pinia';
|
||||||
import { renderIcon } from '@/utils/icon';
|
import { renderIcon, getUserInfo } from '@/utils';
|
||||||
import type { MenuOption } from 'naive-ui';
|
import type { MenuOption } from 'naive-ui';
|
||||||
|
// import { useAuthStore } from './auth';
|
||||||
|
|
||||||
interface RuutesStatus {
|
// const authStore = useAuthStore();
|
||||||
|
|
||||||
|
interface RoutesStatus {
|
||||||
|
isInitAuthRoute: boolean;
|
||||||
menus: any;
|
menus: any;
|
||||||
}
|
}
|
||||||
export const useRouteStore = defineStore('route-store', {
|
export const useRouteStore = defineStore('route-store', {
|
||||||
state: (): RuutesStatus => {
|
state: (): RoutesStatus => {
|
||||||
return {
|
return {
|
||||||
|
isInitAuthRoute: false,
|
||||||
menus: [],
|
menus: [],
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
actions: {
|
actions: {
|
||||||
setMenus(data: Auth.UserInfoPermissions[]) {
|
async setMenus() {
|
||||||
this.menus = this.transformAuthRoutesToMenus(data);
|
const { userRoutes } = getUserInfo();
|
||||||
|
this.menus = this.transformAuthRoutesToMenus(userRoutes);
|
||||||
},
|
},
|
||||||
// 将返回的路由表渲染成侧边栏
|
// 将返回的路由表渲染成侧边栏
|
||||||
transformAuthRoutesToMenus(data: Auth.UserInfoPermissions[]): MenuOption[] {
|
transformAuthRoutesToMenus(userRoutes: Auth.UserInfoPermissions[]): MenuOption[] {
|
||||||
return data.map((item) => {
|
return userRoutes.map((item) => {
|
||||||
const target: MenuOption = {
|
const target: MenuOption = {
|
||||||
label: item.meta.title,
|
label: item.meta.title,
|
||||||
key: item.path,
|
key: item.path,
|
||||||
@ -33,5 +39,10 @@ export const useRouteStore = defineStore('route-store', {
|
|||||||
return target;
|
return target;
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
|
async initAuthRoute() {
|
||||||
|
await this.setMenus();
|
||||||
|
this.isInitAuthRoute = true;
|
||||||
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
2
src/types/business.d.ts
vendored
2
src/types/business.d.ts
vendored
@ -26,7 +26,7 @@ declare namespace Auth {
|
|||||||
/* token */
|
/* token */
|
||||||
token: string;
|
token: string;
|
||||||
/* 权限路由 */
|
/* 权限路由 */
|
||||||
permissions: UserInfoPermissions[];
|
userRoutes: UserInfoPermissions[];
|
||||||
}
|
}
|
||||||
interface UserInfoPermissions {
|
interface UserInfoPermissions {
|
||||||
name: string;
|
name: string;
|
||||||
|
4
src/types/route.d.ts
vendored
4
src/types/route.d.ts
vendored
@ -35,10 +35,12 @@ declare namespace AppRoute {
|
|||||||
/* 是否开启页面缓存 */
|
/* 是否开启页面缓存 */
|
||||||
keepAlive?: boolean;
|
keepAlive?: boolean;
|
||||||
/* 有些路由我们并不想在菜单中显示,比如某些编辑页面。 */
|
/* 有些路由我们并不想在菜单中显示,比如某些编辑页面。 */
|
||||||
hideMenu?: boolean;
|
hide?: boolean;
|
||||||
/* 菜单排序。 */
|
/* 菜单排序。 */
|
||||||
order?: number;
|
order?: number;
|
||||||
/* 嵌套外链 */
|
/* 嵌套外链 */
|
||||||
herf?: string;
|
herf?: string;
|
||||||
|
/** 当前路由需要选中的菜单项(用于跳转至不在左侧菜单显示的路由且需要高亮某个菜单的情况) */
|
||||||
|
activeMenu?: RouteKey;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user