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