mirror of
https://github.com/chansee97/nova-admin.git
synced 2025-04-05 19:41:59 +08:00
fix: 修改tab逻辑
修改tab逻辑,fixed刷新跳转bug,add静态路由
This commit is contained in:
parent
518689f64d
commit
a0fbe28595
2
.env
2
.env
@ -4,5 +4,7 @@ VITE_BASE_URL=/
|
||||
VITE_APP_TITLE = Ench Admin
|
||||
# 路由模式
|
||||
VITE_HASH_ROUTE = Y
|
||||
# 权限路由模式: static | dynamic
|
||||
VITE_AUTH_ROUTE_MODE=dynamic
|
||||
# 存储前缀
|
||||
VITE_STORAGE_PREFIX = ""
|
||||
|
@ -26,6 +26,7 @@
|
||||
"dependencies": {
|
||||
"axios": "^0.27.2",
|
||||
"pinia": "^2.0.17",
|
||||
"pinia-plugin-persist": "^1.0.0",
|
||||
"vue": "^3.2.37",
|
||||
"vue-router": "^4.1.3"
|
||||
},
|
||||
|
@ -4,6 +4,7 @@
|
||||
:collapsed-width="64"
|
||||
:collapsed-icon-size="24"
|
||||
:indent="20"
|
||||
accordion
|
||||
:options="routesStore.menus"
|
||||
:value="routesStore.activeMenu"
|
||||
@update:value="handleClickMenu"
|
||||
|
@ -28,8 +28,13 @@ export async function createPermissionGuard(
|
||||
// 有登录但是没有路由,初始化路由、侧边菜单等
|
||||
await routeStore.initAuthRoute();
|
||||
// 动态路由加载完回到根路由
|
||||
next({ name: 'appRoot' });
|
||||
return false;
|
||||
if (to.name === 'not-found') {
|
||||
// 动态路由没有加载导致被not-found-page路由捕获,等待权限路由加载好了,回到之前的路由
|
||||
// 若路由是从根路由重定向过来的,重新回到根路由
|
||||
const path = to.redirectedFrom?.fullPath;
|
||||
next({ path, replace: true, query: to.query, hash: to.hash });
|
||||
return false;
|
||||
}
|
||||
}
|
||||
// 权限路由已经加载,仍然未找到,重定向到not-found
|
||||
// if (to.name === 'not-found-page') {
|
||||
|
@ -1,26 +1,7 @@
|
||||
import type { App } from 'vue';
|
||||
import { createRouter, createWebHistory, createWebHashHistory, RouteRecordRaw } from 'vue-router';
|
||||
import { setupRouterGuard } from './guard';
|
||||
import { BasicLayout } from '@/layouts/index';
|
||||
import { constantRoutes } from './routes';
|
||||
|
||||
const routes: RouteRecordRaw[] = [
|
||||
{
|
||||
path: '/',
|
||||
name: 'root',
|
||||
redirect: 'appRoot',
|
||||
component: BasicLayout,
|
||||
children: [...constantRoutes],
|
||||
},
|
||||
{
|
||||
path: '/login',
|
||||
name: 'login',
|
||||
component: () => import('@/views/login/index.vue'), // 注意这里要带上 文件后缀.vue
|
||||
meta: {
|
||||
title: '登录',
|
||||
},
|
||||
},
|
||||
];
|
||||
import { routes } from './routes';
|
||||
|
||||
const { VITE_HASH_ROUTE = 'N', VITE_BASE_URL } = import.meta.env;
|
||||
export const router = createRouter({
|
||||
|
35
src/router/modules/dashboard.ts
Normal file
35
src/router/modules/dashboard.ts
Normal file
@ -0,0 +1,35 @@
|
||||
import { RouteRecordRaw } from 'vue-router';
|
||||
import { BasicLayout } from '@/layouts/index';
|
||||
export const dashboard: RouteRecordRaw = {
|
||||
path: '/dashboard',
|
||||
name: 'dashboard',
|
||||
redirect: '/dashboard/workbench',
|
||||
component: BasicLayout,
|
||||
meta: {
|
||||
title: '分析页',
|
||||
requiresAuth: true,
|
||||
icon: 'icon-park-outline:analysis',
|
||||
},
|
||||
children: [
|
||||
{
|
||||
name: 'dashboard_workbench',
|
||||
path: '/dashboard/workbench',
|
||||
component: () => import('@/views/dashboard/workbench/index.vue'),
|
||||
meta: {
|
||||
title: '工作台',
|
||||
requiresAuth: true,
|
||||
icon: 'icon-park-outline:alarm',
|
||||
},
|
||||
},
|
||||
{
|
||||
name: 'dashboard_monitor',
|
||||
path: '/dashboard/monitor',
|
||||
component: () => import('@/views/dashboard/monitor/index.vue'),
|
||||
meta: {
|
||||
title: '监控页',
|
||||
requiresAuth: true,
|
||||
icon: 'icon-park-outline:anchor',
|
||||
},
|
||||
},
|
||||
],
|
||||
};
|
3
src/router/modules/index.ts
Normal file
3
src/router/modules/index.ts
Normal file
@ -0,0 +1,3 @@
|
||||
import { dashboard } from './dashboard';
|
||||
|
||||
export const staticRoutes = [dashboard];
|
@ -1,34 +1,53 @@
|
||||
import { RouteRecordRaw } from 'vue-router';
|
||||
import { BasicLayout } from '@/layouts/index';
|
||||
|
||||
/* 页面中的一些固定路由,错误页等 */
|
||||
export const constantRoutes = [
|
||||
export const routes: RouteRecordRaw[] = [
|
||||
{
|
||||
path: '/no-found',
|
||||
name: 'not-found',
|
||||
component: () => import('@/views/error/not-found/index.vue'),
|
||||
meta: {
|
||||
title: '找不到页面',
|
||||
icon: 'icon-park-outline:ghost',
|
||||
},
|
||||
path: '/',
|
||||
name: 'root',
|
||||
redirect: 'appRoot',
|
||||
component: BasicLayout,
|
||||
children: [
|
||||
{
|
||||
path: '/no-found',
|
||||
name: 'not-found',
|
||||
component: () => import('@/views/error/not-found/index.vue'),
|
||||
meta: {
|
||||
title: '找不到页面',
|
||||
icon: 'icon-park-outline:ghost',
|
||||
},
|
||||
},
|
||||
{
|
||||
path: '/no-permission',
|
||||
name: 'no-permission',
|
||||
component: () => import('@/views/error/not-permission/index.vue'),
|
||||
meta: {
|
||||
title: '用户无权限',
|
||||
icon: 'icon-park-outline:error',
|
||||
},
|
||||
},
|
||||
{
|
||||
path: '/service-error',
|
||||
name: 'service-error',
|
||||
component: () => import('@/views/error/service-error/index.vue'),
|
||||
meta: {
|
||||
title: '服务器错误',
|
||||
icon: 'icon-park-outline:close-wifi',
|
||||
},
|
||||
},
|
||||
{
|
||||
path: '/:pathMatch(.*)*',
|
||||
redirect: '/no-found',
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
path: '/no-permission',
|
||||
name: 'no-permission',
|
||||
component: () => import('@/views/error/not-permission/index.vue'),
|
||||
path: '/login',
|
||||
name: 'login',
|
||||
component: () => import('@/views/login/index.vue'), // 注意这里要带上 文件后缀.vue
|
||||
meta: {
|
||||
title: '用户无权限',
|
||||
icon: 'icon-park-outline:error',
|
||||
title: '登录',
|
||||
},
|
||||
},
|
||||
{
|
||||
path: '/service-error',
|
||||
name: 'service-error',
|
||||
component: () => import('@/views/error/service-error/index.vue'),
|
||||
meta: {
|
||||
title: '服务器错误',
|
||||
icon: 'icon-park-outline:close-wifi',
|
||||
},
|
||||
},
|
||||
{
|
||||
path: '/:pathMatch(.*)*',
|
||||
redirect: '/no-found',
|
||||
},
|
||||
];
|
||||
|
@ -1,8 +1,10 @@
|
||||
import { createPinia } from 'pinia';
|
||||
import piniaPluginPersist from 'pinia-plugin-persist';
|
||||
import type { App } from 'vue';
|
||||
|
||||
export function setupStore(app: App) {
|
||||
const store = createPinia();
|
||||
store.use(piniaPluginPersist);
|
||||
app.use(store);
|
||||
}
|
||||
export * from './modules';
|
||||
|
@ -41,7 +41,7 @@ export const useAuthStore = defineStore('auth-store', {
|
||||
this.loginLoading = true;
|
||||
const { data } = await fetchLogin({ userName, password });
|
||||
// 处理登录信息
|
||||
await this.handleAfterLogin(data); // TODO 避免any
|
||||
await this.handleAfterLogin(data);
|
||||
|
||||
this.loginLoading = false;
|
||||
},
|
||||
@ -50,9 +50,6 @@ export const useAuthStore = defineStore('auth-store', {
|
||||
async handleAfterLogin(data: Auth.loginToken) {
|
||||
// 将token和userInfo保存下来
|
||||
const catchSuccess = await this.catchUserInfo(data);
|
||||
// 初始化侧边菜单
|
||||
// const { initAuthRoute } = useRouteStore();
|
||||
// await initAuthRoute();
|
||||
|
||||
// 登录写入信息成功
|
||||
if (catchSuccess) {
|
||||
@ -68,9 +65,14 @@ export const useAuthStore = defineStore('auth-store', {
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
// 如果不成功则重置存储
|
||||
this.resetAuthStore();
|
||||
// 登录失败提示
|
||||
window.$notification?.error({
|
||||
title: '登录失败!',
|
||||
content: `验证失败,请检查账号密码`,
|
||||
duration: 3000,
|
||||
});
|
||||
},
|
||||
|
||||
/* 缓存用户信息 */
|
||||
|
@ -4,12 +4,14 @@ import { MenuOption } from 'naive-ui';
|
||||
import { createDynamicRoutes } from '@/router/guard/dynamic';
|
||||
import { router } from '@/router';
|
||||
import { fetchUserRoutes } from '@/service';
|
||||
import { staticRoutes } from '@/router/modules';
|
||||
|
||||
interface RoutesStatus {
|
||||
isInitAuthRoute: boolean;
|
||||
menus: any;
|
||||
userRoutes: AppRoute.Route[];
|
||||
activeMenu: string | null;
|
||||
authRouteMode: ImportMetaEnv['VITE_AUTH_ROUTE_MODE'];
|
||||
}
|
||||
export const useRouteStore = defineStore('route-store', {
|
||||
state: (): RoutesStatus => {
|
||||
@ -18,6 +20,7 @@ export const useRouteStore = defineStore('route-store', {
|
||||
isInitAuthRoute: false,
|
||||
menus: [],
|
||||
activeMenu: null,
|
||||
authRouteMode: import.meta.env.VITE_AUTH_ROUTE_MODE,
|
||||
};
|
||||
},
|
||||
actions: {
|
||||
@ -83,6 +86,14 @@ export const useRouteStore = defineStore('route-store', {
|
||||
// 插入路由表
|
||||
router.addRoute(appRoutes);
|
||||
},
|
||||
/* 初始化静态路由 */
|
||||
async initStaticRoute() {
|
||||
/* 将静态路由转换为侧边菜单 */
|
||||
staticRoutes.forEach((route) => {
|
||||
// 插入路由表
|
||||
router.addRoute(route);
|
||||
});
|
||||
},
|
||||
//* 将返回的路由表渲染成侧边栏 */
|
||||
transformAuthRoutesToMenus(userRoutes: AppRoute.Route[]): MenuOption[] {
|
||||
return userRoutes
|
||||
@ -111,7 +122,11 @@ export const useRouteStore = defineStore('route-store', {
|
||||
},
|
||||
async initAuthRoute() {
|
||||
this.isInitAuthRoute = false;
|
||||
await this.initDynamicRoute();
|
||||
if (this.authRouteMode === 'dynamic') {
|
||||
await this.initDynamicRoute();
|
||||
} else {
|
||||
await this.initStaticRoute();
|
||||
}
|
||||
this.isInitAuthRoute = true;
|
||||
},
|
||||
},
|
||||
|
@ -9,6 +9,7 @@ interface TabState {
|
||||
path: string;
|
||||
}[];
|
||||
tabs: RouteLocationNormalized[];
|
||||
tabWhiteList: string[];
|
||||
currentTab: string;
|
||||
}
|
||||
export const useTabStore = defineStore('tab-store', {
|
||||
@ -22,6 +23,7 @@ export const useTabStore = defineStore('tab-store', {
|
||||
},
|
||||
],
|
||||
tabs: [],
|
||||
tabWhiteList: ['not-found', 'no-permission', 'service-error', 'login'],
|
||||
currentTab: 'dashboard_workbench',
|
||||
};
|
||||
},
|
||||
@ -42,6 +44,10 @@ export const useTabStore = defineStore('tab-store', {
|
||||
if (this.hasExistTab(route.name as string)) {
|
||||
return;
|
||||
}
|
||||
// 如果在白名单内则不添加,错误页等
|
||||
if (this.tabWhiteList.includes(route.name as string)) {
|
||||
return;
|
||||
}
|
||||
this.tabs.push(route);
|
||||
},
|
||||
closeTab(name: string) {
|
||||
@ -56,7 +62,7 @@ export const useTabStore = defineStore('tab-store', {
|
||||
if (this.currentTab === name && !isLast) {
|
||||
// 跳转到后一个标签
|
||||
routerPush(this.tabs[index + 1].path);
|
||||
} else {
|
||||
} else if (this.currentTab === name && isLast) {
|
||||
// 已经是最后一个了,就跳转前一个
|
||||
routerPush(this.tabs[index - 1].path);
|
||||
}
|
||||
@ -85,4 +91,7 @@ export const useTabStore = defineStore('tab-store', {
|
||||
});
|
||||
},
|
||||
},
|
||||
persist: {
|
||||
enabled: true,
|
||||
},
|
||||
});
|
||||
|
2
src/types/env.d.ts
vendored
2
src/types/env.d.ts
vendored
@ -32,6 +32,8 @@ interface ImportMetaEnv {
|
||||
readonly VITE_COMPRESS_TYPE?: 'gzip' | 'brotliCompress' | 'deflate' | 'deflateRaw';
|
||||
/** hash路由模式 */
|
||||
readonly VITE_HASH_ROUTE?: 'Y' | 'N';
|
||||
/** 路由加载模式 */
|
||||
readonly VITE_AUTH_ROUTE_MODE?: 'static' | 'dynamic';
|
||||
/** 本地存储前缀 */
|
||||
readonly VITE_STORAGE_PREFIX?: string;
|
||||
/** 后端服务的环境类型 */
|
||||
|
Loading…
x
Reference in New Issue
Block a user