fix: 修改tab逻辑

修改tab逻辑,fixed刷新跳转bug,add静态路由
This commit is contained in:
chen.home 2022-08-21 20:00:29 +08:00
parent 518689f64d
commit a0fbe28595
13 changed files with 132 additions and 55 deletions

2
.env
View File

@ -4,5 +4,7 @@ VITE_BASE_URL=/
VITE_APP_TITLE = Ench Admin VITE_APP_TITLE = Ench Admin
# 路由模式 # 路由模式
VITE_HASH_ROUTE = Y VITE_HASH_ROUTE = Y
# 权限路由模式: static dynamic
VITE_AUTH_ROUTE_MODE=dynamic
# 存储前缀 # 存储前缀
VITE_STORAGE_PREFIX = "" VITE_STORAGE_PREFIX = ""

View File

@ -26,6 +26,7 @@
"dependencies": { "dependencies": {
"axios": "^0.27.2", "axios": "^0.27.2",
"pinia": "^2.0.17", "pinia": "^2.0.17",
"pinia-plugin-persist": "^1.0.0",
"vue": "^3.2.37", "vue": "^3.2.37",
"vue-router": "^4.1.3" "vue-router": "^4.1.3"
}, },

View File

@ -4,6 +4,7 @@
:collapsed-width="64" :collapsed-width="64"
:collapsed-icon-size="24" :collapsed-icon-size="24"
:indent="20" :indent="20"
accordion
:options="routesStore.menus" :options="routesStore.menus"
:value="routesStore.activeMenu" :value="routesStore.activeMenu"
@update:value="handleClickMenu" @update:value="handleClickMenu"

View File

@ -28,8 +28,13 @@ export async function createPermissionGuard(
// 有登录但是没有路由,初始化路由、侧边菜单等 // 有登录但是没有路由,初始化路由、侧边菜单等
await routeStore.initAuthRoute(); await routeStore.initAuthRoute();
// 动态路由加载完回到根路由 // 动态路由加载完回到根路由
next({ name: 'appRoot' }); if (to.name === 'not-found') {
return false; // 动态路由没有加载导致被not-found-page路由捕获等待权限路由加载好了回到之前的路由
// 若路由是从根路由重定向过来的,重新回到根路由
const path = to.redirectedFrom?.fullPath;
next({ path, replace: true, query: to.query, hash: to.hash });
return false;
}
} }
// 权限路由已经加载仍然未找到重定向到not-found // 权限路由已经加载仍然未找到重定向到not-found
// if (to.name === 'not-found-page') { // if (to.name === 'not-found-page') {

View File

@ -1,26 +1,7 @@
import type { App } from 'vue'; import type { App } from 'vue';
import { createRouter, createWebHistory, createWebHashHistory, RouteRecordRaw } from 'vue-router'; import { createRouter, createWebHistory, createWebHashHistory, RouteRecordRaw } from 'vue-router';
import { setupRouterGuard } from './guard'; import { setupRouterGuard } from './guard';
import { BasicLayout } from '@/layouts/index'; import { routes } from './routes';
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: '登录',
},
},
];
const { VITE_HASH_ROUTE = 'N', VITE_BASE_URL } = import.meta.env; const { VITE_HASH_ROUTE = 'N', VITE_BASE_URL } = import.meta.env;
export const router = createRouter({ export const router = createRouter({

View 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',
},
},
],
};

View File

@ -0,0 +1,3 @@
import { dashboard } from './dashboard';
export const staticRoutes = [dashboard];

View File

@ -1,34 +1,53 @@
import { RouteRecordRaw } from 'vue-router';
import { BasicLayout } from '@/layouts/index';
/* 页面中的一些固定路由,错误页等 */ /* 页面中的一些固定路由,错误页等 */
export const constantRoutes = [ export const routes: RouteRecordRaw[] = [
{ {
path: '/no-found', path: '/',
name: 'not-found', name: 'root',
component: () => import('@/views/error/not-found/index.vue'), redirect: 'appRoot',
meta: { component: BasicLayout,
title: '找不到页面', children: [
icon: 'icon-park-outline:ghost', {
}, 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', path: '/login',
name: 'no-permission', name: 'login',
component: () => import('@/views/error/not-permission/index.vue'), component: () => import('@/views/login/index.vue'), // 注意这里要带上 文件后缀.vue
meta: { meta: {
title: '用户无权限', 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',
},
]; ];

View File

@ -1,8 +1,10 @@
import { createPinia } from 'pinia'; import { createPinia } from 'pinia';
import piniaPluginPersist from 'pinia-plugin-persist';
import type { App } from 'vue'; import type { App } from 'vue';
export function setupStore(app: App) { export function setupStore(app: App) {
const store = createPinia(); const store = createPinia();
store.use(piniaPluginPersist);
app.use(store); app.use(store);
} }
export * from './modules'; export * from './modules';

View File

@ -41,7 +41,7 @@ export const useAuthStore = defineStore('auth-store', {
this.loginLoading = true; this.loginLoading = true;
const { data } = await fetchLogin({ userName, password }); const { data } = await fetchLogin({ userName, password });
// 处理登录信息 // 处理登录信息
await this.handleAfterLogin(data); // TODO 避免any await this.handleAfterLogin(data);
this.loginLoading = false; this.loginLoading = false;
}, },
@ -50,9 +50,6 @@ export const useAuthStore = defineStore('auth-store', {
async handleAfterLogin(data: Auth.loginToken) { async handleAfterLogin(data: Auth.loginToken) {
// 将token和userInfo保存下来 // 将token和userInfo保存下来
const catchSuccess = await this.catchUserInfo(data); const catchSuccess = await this.catchUserInfo(data);
// 初始化侧边菜单
// const { initAuthRoute } = useRouteStore();
// await initAuthRoute();
// 登录写入信息成功 // 登录写入信息成功
if (catchSuccess) { if (catchSuccess) {
@ -68,9 +65,14 @@ export const useAuthStore = defineStore('auth-store', {
}); });
return; return;
} }
// 如果不成功则重置存储 // 如果不成功则重置存储
this.resetAuthStore(); this.resetAuthStore();
// 登录失败提示
window.$notification?.error({
title: '登录失败!',
content: `验证失败,请检查账号密码`,
duration: 3000,
});
}, },
/* 缓存用户信息 */ /* 缓存用户信息 */

View File

@ -4,12 +4,14 @@ import { MenuOption } from 'naive-ui';
import { createDynamicRoutes } from '@/router/guard/dynamic'; import { createDynamicRoutes } from '@/router/guard/dynamic';
import { router } from '@/router'; import { router } from '@/router';
import { fetchUserRoutes } from '@/service'; import { fetchUserRoutes } from '@/service';
import { staticRoutes } from '@/router/modules';
interface RoutesStatus { interface RoutesStatus {
isInitAuthRoute: boolean; isInitAuthRoute: boolean;
menus: any; menus: any;
userRoutes: AppRoute.Route[]; userRoutes: AppRoute.Route[];
activeMenu: string | null; activeMenu: string | null;
authRouteMode: ImportMetaEnv['VITE_AUTH_ROUTE_MODE'];
} }
export const useRouteStore = defineStore('route-store', { export const useRouteStore = defineStore('route-store', {
state: (): RoutesStatus => { state: (): RoutesStatus => {
@ -18,6 +20,7 @@ export const useRouteStore = defineStore('route-store', {
isInitAuthRoute: false, isInitAuthRoute: false,
menus: [], menus: [],
activeMenu: null, activeMenu: null,
authRouteMode: import.meta.env.VITE_AUTH_ROUTE_MODE,
}; };
}, },
actions: { actions: {
@ -83,6 +86,14 @@ export const useRouteStore = defineStore('route-store', {
// 插入路由表 // 插入路由表
router.addRoute(appRoutes); router.addRoute(appRoutes);
}, },
/* 初始化静态路由 */
async initStaticRoute() {
/* 将静态路由转换为侧边菜单 */
staticRoutes.forEach((route) => {
// 插入路由表
router.addRoute(route);
});
},
//* 将返回的路由表渲染成侧边栏 */ //* 将返回的路由表渲染成侧边栏 */
transformAuthRoutesToMenus(userRoutes: AppRoute.Route[]): MenuOption[] { transformAuthRoutesToMenus(userRoutes: AppRoute.Route[]): MenuOption[] {
return userRoutes return userRoutes
@ -111,7 +122,11 @@ export const useRouteStore = defineStore('route-store', {
}, },
async initAuthRoute() { async initAuthRoute() {
this.isInitAuthRoute = false; this.isInitAuthRoute = false;
await this.initDynamicRoute(); if (this.authRouteMode === 'dynamic') {
await this.initDynamicRoute();
} else {
await this.initStaticRoute();
}
this.isInitAuthRoute = true; this.isInitAuthRoute = true;
}, },
}, },

View File

@ -9,6 +9,7 @@ interface TabState {
path: string; path: string;
}[]; }[];
tabs: RouteLocationNormalized[]; tabs: RouteLocationNormalized[];
tabWhiteList: string[];
currentTab: string; currentTab: string;
} }
export const useTabStore = defineStore('tab-store', { export const useTabStore = defineStore('tab-store', {
@ -22,6 +23,7 @@ export const useTabStore = defineStore('tab-store', {
}, },
], ],
tabs: [], tabs: [],
tabWhiteList: ['not-found', 'no-permission', 'service-error', 'login'],
currentTab: 'dashboard_workbench', currentTab: 'dashboard_workbench',
}; };
}, },
@ -42,6 +44,10 @@ export const useTabStore = defineStore('tab-store', {
if (this.hasExistTab(route.name as string)) { if (this.hasExistTab(route.name as string)) {
return; return;
} }
// 如果在白名单内则不添加,错误页等
if (this.tabWhiteList.includes(route.name as string)) {
return;
}
this.tabs.push(route); this.tabs.push(route);
}, },
closeTab(name: string) { closeTab(name: string) {
@ -56,7 +62,7 @@ export const useTabStore = defineStore('tab-store', {
if (this.currentTab === name && !isLast) { if (this.currentTab === name && !isLast) {
// 跳转到后一个标签 // 跳转到后一个标签
routerPush(this.tabs[index + 1].path); routerPush(this.tabs[index + 1].path);
} else { } else if (this.currentTab === name && isLast) {
// 已经是最后一个了,就跳转前一个 // 已经是最后一个了,就跳转前一个
routerPush(this.tabs[index - 1].path); 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
View File

@ -32,6 +32,8 @@ interface ImportMetaEnv {
readonly VITE_COMPRESS_TYPE?: 'gzip' | 'brotliCompress' | 'deflate' | 'deflateRaw'; readonly VITE_COMPRESS_TYPE?: 'gzip' | 'brotliCompress' | 'deflate' | 'deflateRaw';
/** hash路由模式 */ /** hash路由模式 */
readonly VITE_HASH_ROUTE?: 'Y' | 'N'; readonly VITE_HASH_ROUTE?: 'Y' | 'N';
/** 路由加载模式 */
readonly VITE_AUTH_ROUTE_MODE?: 'static' | 'dynamic';
/** 本地存储前缀 */ /** 本地存储前缀 */
readonly VITE_STORAGE_PREFIX?: string; readonly VITE_STORAGE_PREFIX?: string;
/** 后端服务的环境类型 */ /** 后端服务的环境类型 */