feat: customized user home page

新增自定义的用户首页(可以每个用户都不相同)
This commit is contained in:
无木 2021-07-13 14:10:31 +08:00
parent f5e31febbd
commit 0a3683a186
9 changed files with 62 additions and 6 deletions

View File

@ -2,6 +2,8 @@
- **NoticeList** 添加分页、超长自动省略、标题点击事件、标题删除线等功能 - **NoticeList** 添加分页、超长自动省略、标题点击事件、标题删除线等功能
- **MixSider** 优化 Mix 菜单布局时 底部折叠按钮 的样式,与其它菜单布局时的风格保持一致 - **MixSider** 优化 Mix 菜单布局时 底部折叠按钮 的样式,与其它菜单布局时的风格保持一致
- 可以为不同的用户指定不同的后台首页:
- 在`getUserInfo`接口返回的用户信息中增加`homePath`字段(可选)即可为当前用户定制首页路径
### 🐛 Bug Fixes ### 🐛 Bug Fixes
@ -18,6 +20,7 @@
- 修复左侧混合菜单的悬停触发逻辑 - 修复左侧混合菜单的悬停触发逻辑
- 修复顶栏菜单在显示包含需要隐藏的菜单项目时出错的问题 - 修复顶栏菜单在显示包含需要隐藏的菜单项目时出错的问题
- 修复悬停触发模式下左侧混合菜单会在没有子菜单且被激活时直接跳转路由 - 修复悬停触发模式下左侧混合菜单会在没有子菜单且被激活时直接跳转路由
- **Breadcrumb** 修复带有重定向的菜单点击无法跳转的问题
- **其它** - **其它**
- 修复菜单默认折叠的配置不起作用的问题 - 修复菜单默认折叠的配置不起作用的问题
- 修复`safari`浏览器报错导致网站打不开 - 修复`safari`浏览器报错导致网站打不开

View File

@ -11,6 +11,7 @@ export function createFakeUserList() {
desc: 'manager', desc: 'manager',
password: '123456', password: '123456',
token: 'fakeToken1', token: 'fakeToken1',
homePath: '/dashboard/analysis',
roles: [ roles: [
{ {
roleName: 'Super Admin', roleName: 'Super Admin',
@ -26,6 +27,7 @@ export function createFakeUserList() {
avatar: 'https://q1.qlogo.cn/g?b=qq&nk=339449197&s=640', avatar: 'https://q1.qlogo.cn/g?b=qq&nk=339449197&s=640',
desc: 'tester', desc: 'tester',
token: 'fakeToken2', token: 'fakeToken2',
homePath: '/dashboard/workbench',
roles: [ roles: [
{ {
roleName: 'Tester', roleName: 'Tester',

View File

@ -17,6 +17,7 @@
import { useMenuSetting } from '/@/hooks/setting/useMenuSetting'; import { useMenuSetting } from '/@/hooks/setting/useMenuSetting';
import { useDesign } from '/@/hooks/web/useDesign'; import { useDesign } from '/@/hooks/web/useDesign';
import { PageEnum } from '/@/enums/pageEnum'; import { PageEnum } from '/@/enums/pageEnum';
import { useUserStore } from '/@/store/modules/user';
const props = { const props = {
/** /**
@ -39,6 +40,7 @@
setup(props) { setup(props) {
const { prefixCls } = useDesign('app-logo'); const { prefixCls } = useDesign('app-logo');
const { getCollapsedShowTitle } = useMenuSetting(); const { getCollapsedShowTitle } = useMenuSetting();
const userStore = useUserStore();
const { title } = useGlobSetting(); const { title } = useGlobSetting();
const go = useGo(); const go = useGo();
@ -56,7 +58,7 @@
]); ]);
function goHome() { function goHome() {
go(PageEnum.BASE_HOME); go(userStore.getUserInfo.homePath || PageEnum.BASE_HOME);
} }
return { return {

View File

@ -7,8 +7,12 @@ import { useUserStoreWithOut } from '/@/store/modules/user';
import { PAGE_NOT_FOUND_ROUTE } from '/@/router/routes/basic'; import { PAGE_NOT_FOUND_ROUTE } from '/@/router/routes/basic';
import { RootRoute } from '/@/router/routes';
const LOGIN_PATH = PageEnum.BASE_LOGIN; const LOGIN_PATH = PageEnum.BASE_LOGIN;
const ROOT_PATH = RootRoute.path;
const whitePathList: PageEnum[] = [LOGIN_PATH]; const whitePathList: PageEnum[] = [LOGIN_PATH];
export function createPermissionGuard(router: Router) { export function createPermissionGuard(router: Router) {
@ -17,7 +21,17 @@ export function createPermissionGuard(router: Router) {
router.beforeEach(async (to, from, next) => { router.beforeEach(async (to, from, next) => {
// Jump to the 404 page after processing the login // Jump to the 404 page after processing the login
if (from.path === LOGIN_PATH && to.name === PAGE_NOT_FOUND_ROUTE.name) { if (from.path === LOGIN_PATH && to.name === PAGE_NOT_FOUND_ROUTE.name) {
next(PageEnum.BASE_HOME); next(userStore.getUserInfo.homePath || PageEnum.BASE_HOME);
return;
}
if (
from.path === ROOT_PATH &&
to.path === PageEnum.BASE_HOME &&
userStore.getUserInfo.homePath &&
userStore.getUserInfo.homePath !== PageEnum.BASE_HOME
) {
next(userStore.getUserInfo.homePath);
return; return;
} }

View File

@ -19,7 +19,7 @@ const dashboard: AppRouteModule = {
name: 'Analysis', name: 'Analysis',
component: () => import('/@/views/dashboard/analysis/index.vue'), component: () => import('/@/views/dashboard/analysis/index.vue'),
meta: { meta: {
affix: true, // affix: true,
title: t('routes.dashboard.analysis'), title: t('routes.dashboard.analysis'),
}, },
}, },

View File

@ -13,6 +13,7 @@ import { getRawRoute } from '/@/utils';
import { MULTIPLE_TABS_KEY } from '/@/enums/cacheEnum'; import { MULTIPLE_TABS_KEY } from '/@/enums/cacheEnum';
import projectSetting from '/@/settings/projectSetting'; import projectSetting from '/@/settings/projectSetting';
import { useUserStore } from '/@/store/modules/user';
export interface MultipleTabState { export interface MultipleTabState {
cacheTabList: Set<string>; cacheTabList: Set<string>;
@ -181,7 +182,8 @@ export const useMultipleTabStore = defineStore({
if (index === 0) { if (index === 0) {
// There is only one tab, then jump to the homepage, otherwise jump to the right tab // There is only one tab, then jump to the homepage, otherwise jump to the right tab
if (this.tabList.length === 1) { if (this.tabList.length === 1) {
toTarget = PageEnum.BASE_HOME; const userStore = useUserStore();
toTarget = userStore.getUserInfo.homePath || PageEnum.BASE_HOME;
} else { } else {
// Jump to the right tab // Jump to the right tab
const page = this.tabList[index + 1]; const page = this.tabList[index + 1];

View File

@ -22,6 +22,7 @@ import { getMenuList } from '/@/api/sys/menu';
import { getPermCode } from '/@/api/sys/user'; import { getPermCode } from '/@/api/sys/user';
import { useMessage } from '/@/hooks/web/useMessage'; import { useMessage } from '/@/hooks/web/useMessage';
import { PageEnum } from '/@/enums/pageEnum';
interface PermissionState { interface PermissionState {
// Permission code list // Permission code list
@ -117,6 +118,32 @@ export const usePermissionStore = defineStore({
return !ignoreRoute; return !ignoreRoute;
}; };
/**
* @description pathroutes中的affix标记
* */
const patchHomeAffix = (routes: AppRouteRecordRaw[]) => {
if (!routes || routes.length === 0) return;
const homePath = userStore.getUserInfo.homePath || PageEnum.BASE_HOME;
function patcher(routes: AppRouteRecordRaw[], parentPath = '') {
if (parentPath) parentPath = parentPath + '/';
routes.forEach((route: AppRouteRecordRaw) => {
const { path, children } = route;
const currentPath = path.startsWith('/') ? path : parentPath + path;
if (currentPath === homePath) {
route.meta = Object.assign({}, route.meta, { affix: true });
throw new Error('end');
}
children && children.length > 0 && patcher(children, currentPath);
});
}
try {
patcher(routes);
} catch (e) {
// 已处理完毕跳出循环
}
return;
};
switch (permissionMode) { switch (permissionMode) {
case PermissionModeEnum.ROLE: case PermissionModeEnum.ROLE:
routes = filter(asyncRoutes, routeFilter); routes = filter(asyncRoutes, routeFilter);
@ -176,6 +203,7 @@ export const usePermissionStore = defineStore({
} }
routes.push(ERROR_LOG_ROUTE); routes.push(ERROR_LOG_ROUTE);
patchHomeAffix(routes);
return routes; return routes;
}, },
}, },

View File

@ -88,13 +88,15 @@ export const useUserStore = defineStore({
const sessionTimeout = this.sessionTimeout; const sessionTimeout = this.sessionTimeout;
sessionTimeout && this.setSessionTimeout(false); sessionTimeout && this.setSessionTimeout(false);
!sessionTimeout && goHome && (await router.replace(PageEnum.BASE_HOME)); !sessionTimeout &&
goHome &&
(await router.replace(userInfo.homePath || PageEnum.BASE_HOME));
return userInfo; return userInfo;
} catch (error) { } catch (error) {
return Promise.reject(error); return Promise.reject(error);
} }
}, },
async getUserInfoAction() { async getUserInfoAction(): Promise<UserInfo> {
const userInfo = await getUserInfo(); const userInfo = await getUserInfo();
const { roles } = userInfo; const { roles } = userInfo;
const roleList = roles.map((item) => item.value) as RoleEnum[]; const roleList = roles.map((item) => item.value) as RoleEnum[];

3
types/store.d.ts vendored
View File

@ -1,5 +1,6 @@
import { ErrorTypeEnum } from '/@/enums/exceptionEnum'; import { ErrorTypeEnum } from '/@/enums/exceptionEnum';
import { MenuModeEnum, MenuTypeEnum } from '/@/enums/menuEnum'; import { MenuModeEnum, MenuTypeEnum } from '/@/enums/menuEnum';
import { RoleInfo } from '/@/api/sys/model/userModel';
// Lock screen information // Lock screen information
export interface LockInfo { export interface LockInfo {
@ -35,6 +36,8 @@ export interface UserInfo {
realName: string; realName: string;
avatar: string; avatar: string;
desc?: string; desc?: string;
homePath?: string;
roles: RoleInfo[];
} }
export interface BeforeMiniState { export interface BeforeMiniState {