diff --git a/.vscode/settings.json b/.vscode/settings.json index 11990044..0fa44d74 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -5,7 +5,7 @@ "i18n-ally.namespace": true, "i18n-ally.pathMatcher": "{locale}/{namespaces}.{ext}", "i18n-ally.enabledParsers": ["json"], - "i18n-ally.sourceLanguage": "en", + "i18n-ally.sourceLanguage": "zh-CN", "i18n-ally.displayLanguage": "zh-CN", "i18n-ally.enabledFrameworks": ["vue", "react"] } diff --git a/CHANGELOG.md b/CHANGELOG.md index 7d7068a0..1a2d8541 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,12 @@ # CHANGE LOG +## 4.0.2 + +### Feats + +- 新增平级路由配置(router meta)配置项,sameLevel 允许你将子路由标记为平级模式,跳转时不会出发菜单、标签页更新,仅会更新面包屑 +- 修改路由菜单显示、隐藏逻辑,现在仅会针对权限的验证匹配选择是否加入菜单列表中 + ## 4.0.1 ### Feats diff --git a/src/layout/components/Menu/index.tsx b/src/layout/components/Menu/index.tsx index 7e5ec5c5..3e46b446 100644 --- a/src/layout/components/Menu/index.tsx +++ b/src/layout/components/Menu/index.tsx @@ -22,7 +22,7 @@ const LayoutMenu = defineComponent({ const modelMenuKey = computed({ get: () => { nextTick().then(() => { - menuRef.value?.showOption?.(menuStore.menuKey as string) + showMenuOption() }) return menuStore.menuKey @@ -44,6 +44,14 @@ const LayoutMenu = defineComponent({ } } + const showMenuOption = () => { + const key = modelMenuKey.value as string + + nextTick().then(() => { + menuRef.value?.showOption?.(key) + }) + } + return { modelMenuKey, changeMenuModelValue, diff --git a/src/layout/components/SiderBar/Components/Breadcrumb/index.tsx b/src/layout/components/SiderBar/Components/Breadcrumb/index.tsx index accdd8ad..e9eedf4c 100644 --- a/src/layout/components/SiderBar/Components/Breadcrumb/index.tsx +++ b/src/layout/components/SiderBar/Components/Breadcrumb/index.tsx @@ -22,7 +22,12 @@ import { NDropdown, NBreadcrumb, NBreadcrumbItem } from 'naive-ui' import { useMenu } from '@/store' -import type { DropdownOption } from 'naive-ui' +import type { DropdownOption, MenuOption } from 'naive-ui' +import type { + AppMenuOption, + MenuTagOptions, + AppMenuKey, +} from '@/types/modules/app' const Breadcrumb = defineComponent({ name: 'RBreadcrumb', @@ -30,7 +35,8 @@ const Breadcrumb = defineComponent({ const menuStore = useMenu() const { changeMenuModelValue } = menuStore - const modelBreadcrumbOptions = computed(() => menuStore.breadcrumbOptions) + const { breadcrumbOptions } = storeToRefs(menuStore) + const modelBreadcrumbOptions = computed(() => breadcrumbOptions.value) const handleDropdownSelect = ( key: string | number, @@ -39,16 +45,26 @@ const Breadcrumb = defineComponent({ changeMenuModelValue(key, option) } + const handleBreadcrumbItemClick = (option: AppMenuOption) => { + if (!option.children?.length) { + changeMenuModelValue(option.key, option as unknown as MenuOption) + } + } + return { modelBreadcrumbOptions, handleDropdownSelect, + handleBreadcrumbItemClick, } }, render() { return ( {this.modelBreadcrumbOptions.map((curr) => ( - + { - setupAppMenu() - }) return { windowHeight, diff --git a/src/locales/lang/en-US/menu.json b/src/locales/lang/en-US/menu.json index be9dff9d..3fc2dfac 100644 --- a/src/locales/lang/en-US/menu.json +++ b/src/locales/lang/en-US/menu.json @@ -15,5 +15,6 @@ "Office_Presentation": "Presentation", "Office_Spreadsheet": "Spreadsheet", "CalculatePrecision": "Precision", - "Directive": "Directive" + "Directive": "Directive", + "RouterDemo": "Same Level Router Demo" } diff --git a/src/locales/lang/zh-CN/menu.json b/src/locales/lang/zh-CN/menu.json index c47ec7af..139029e8 100644 --- a/src/locales/lang/zh-CN/menu.json +++ b/src/locales/lang/zh-CN/menu.json @@ -15,5 +15,6 @@ "Office_Presentation": "演示", "Office_Spreadsheet": "表格", "CalculatePrecision": "数字精度", - "Directive": "指令" + "Directive": "指令", + "RouterDemo": "平层路由详情" } diff --git a/src/router/README.md b/src/router/README.md index a22efcc4..b1c66ad7 100644 --- a/src/router/README.md +++ b/src/router/README.md @@ -99,6 +99,7 @@ interface RouteMeta { noLocalTitle?: string | number ignoreAutoResetScroll?: boolean keepAlive?: boolean + sameLevel?: boolean } ``` @@ -114,4 +115,5 @@ hidden: 是否显示 noLocalTitle: 不使用国际化渲染 Menu Titile ignoreAutoResetScroll: 该页面内容区域自动初始化滚动条位置 keepAlive: 是否缓存该页面(需要配置 APP_KEEP_ALIVE setupKeepAlive 属性为 true 启用才有效) +sameLevel: 是否标记该路由为平级模式 ``` diff --git a/src/router/helper/permission.ts b/src/router/helper/permission.ts index bca0982d..a002ae46 100644 --- a/src/router/helper/permission.ts +++ b/src/router/helper/permission.ts @@ -32,6 +32,7 @@ import type { RouteLocationNormalized, } from 'vue-router' import type { AppMenuOption } from '@/types/modules/app' +import type { AppRouteMeta } from '@/router/type' export const permissionRouter = (router: Router) => { const { beforeEach } = router @@ -39,9 +40,10 @@ export const permissionRouter = (router: Router) => { beforeEach((to, from, next) => { const token = getCache(APP_CATCH_KEY.token) const route = getCache('menuKey') || ROOT_ROUTE.path + const { meta } = to if (token !== null) { - if (validMenuItemShow(to as unknown as AppMenuOption)) { + if (validRole(meta as AppRouteMeta)) { if (to.path === '/' || from.path === '/login') { if (route !== 'no') { next(route) diff --git a/src/router/helper/routerCopilot.ts b/src/router/helper/routerCopilot.ts index 71d4458c..b109a536 100644 --- a/src/router/helper/routerCopilot.ts +++ b/src/router/helper/routerCopilot.ts @@ -66,31 +66,10 @@ export const validRole = (meta: AppRouteMeta) => { * 如果你仅仅是希望校验是否满足权限, 应该使用另一个方法 validRole */ export const validMenuItemShow = (option: AppMenuOption) => { - const { meta, name } = option - const hidden = - meta?.hidden === undefined || meta?.hidden === false ? false : meta?.hidden + const { meta = {} } = option + const { hidden } = meta - // 如果是超级管理员(预设为 admin), 则根据其菜单栏(hidden)字段判断是否显示 - if (validRole(meta)) { - return true && !hidden - } else { - // 如果为基础路由, 不进行鉴权则根据其菜单栏(hidden)字段判断是否显示 - if (WHITE_ROUTES.includes(name)) { - return true && !hidden - } - - // 如果 role 为 undefind 或者空数组, 则认为该路由不做权限过滤 - if (!meta?.role || !meta.role?.length) { - return true && !hidden - } - - // 判断权限是否匹配和菜单栏(hidden)字段判断是否显示 - if (meta?.role && meta.role.length) { - return validRole(meta) && !hidden - } - - return true && !hidden - } + return hidden === undefined || hidden === false ? true : false } /** diff --git a/src/router/modules/axios.ts b/src/router/modules/axios.ts index de2d671a..9245ca89 100644 --- a/src/router/modules/axios.ts +++ b/src/router/modules/axios.ts @@ -11,7 +11,6 @@ const axios: AppRouteRecordRaw = { icon: 'axios', order: 3, keepAlive: true, - hidden: false, }, } diff --git a/src/router/modules/router-demo.ts b/src/router/modules/router-demo.ts new file mode 100644 index 00000000..d28acc22 --- /dev/null +++ b/src/router/modules/router-demo.ts @@ -0,0 +1,37 @@ +import { t } from '@/locales/useI18n' +import { LAYOUT } from '@/router/constant/index' + +import type { AppRouteRecordRaw } from '@/router/type' + +const routerDemo: AppRouteRecordRaw = { + path: '/router-demo', + name: 'RouterDemoRoot', + component: LAYOUT, + meta: { + i18nKey: t('menu.RouterDemo'), + icon: 'other', + order: 3, + }, + children: [ + { + path: 'router-demo-home', + name: 'RouterDemoHome', + component: () => import('@/views/router-demo/router-demo-home/index'), + meta: { + noLocalTitle: '人员信息', + }, + }, + { + path: 'router-demo-detail', + name: 'RouterDemoDetail', + component: () => import('@/views/router-demo/router-demo-detail/index'), + meta: { + noLocalTitle: '信息详情', + hidden: true, + sameLevel: true, + }, + }, + ], +} + +export default routerDemo diff --git a/src/router/routes.ts b/src/router/routes.ts index 37bc076d..b21daa7a 100644 --- a/src/router/routes.ts +++ b/src/router/routes.ts @@ -18,10 +18,10 @@ export default () => [ component: Layout, children: expandRoutes(getAppRawRoutes()), }, - { - path: '/:catchAll(.*)', - name: 'errorPage', - component: Layout, - redirect: '/error', - }, + // { + // path: '/:catchAll(.*)', + // name: 'errorPage', + // component: Layout, + // redirect: '/error', + // }, ] diff --git a/src/router/type.ts b/src/router/type.ts index 90731dbd..ddc179a7 100644 --- a/src/router/type.ts +++ b/src/router/type.ts @@ -19,6 +19,7 @@ export interface AppRouteMeta { ignoreAutoResetScroll?: boolean order?: number keepAlive?: boolean + sameLevel?: boolean } // @ts-ignore diff --git a/src/store/modules/menu/index.ts b/src/store/modules/menu/index.ts index 25e74ea5..c11fd790 100644 --- a/src/store/modules/menu/index.ts +++ b/src/store/modules/menu/index.ts @@ -25,22 +25,26 @@ import { NEllipsis } from 'naive-ui' import { getCache, setCache } from '@/utils/cache' -import { validMenuItemShow } from '@/router/helper/routerCopilot' +import { validMenuItemShow, validRole } from '@/router/helper/routerCopilot' import { parseAndFindMatchingNodes, - matchMenuOption, updateDocumentTitle, hasMenuIcon, getCatchMenuKey, } from './helper' import { useI18n } from '@/locales/useI18n' import { getAppRawRoutes } from '@/router/routeModules' +import { expandRoutes } from '@/router/helper/expandRoutes' import { useKeepAlive } from '@/store' import { useVueRouter } from '@/router/helper/useVueRouter' import type { MenuOption } from 'naive-ui' import type { AppRouteMeta, AppRouteRecordRaw } from '@/router/type' -import type { AppMenuOption, MenuTagOptions } from '@/types/modules/app' +import type { + AppMenuOption, + MenuTagOptions, + AppMenuKey, +} from '@/types/modules/app' import type { MenuState } from '@/store/modules/menu/type' export const useMenu = defineStore( @@ -58,6 +62,7 @@ export const useMenu = defineStore( menuTagOptions: [], // tag 标签菜单 breadcrumbOptions: [], // 面包屑菜单 }) + const isSetupAppMenuLock = ref(true) /** * @@ -77,80 +82,28 @@ export const useMenu = defineStore( /** * - * @param key 菜单更新后的 key - * @param option 菜单当前 option 项 + * 设置面包屑 * - * @remark 修改 `menu key` 后的回调函数 - * @remark 修改后, 缓存当前选择 key 并且存储标签页与跳转页面(router push 操作) + * 如果识别到为平级模式, 则会自动追加一层面包屑 */ - const changeMenuModelValue = (key: string | number, option: MenuOption) => { - const { meta, path } = option as unknown as AppRouteRecordRaw + const setBreadcrumbOptions = (key: string | number, option: MenuOption) => { + const { meta } = option as unknown as AppRouteRecordRaw - if (meta.windowOpen) { - window.open(meta.windowOpen) - } else { - // 防止重复点击做重复操作处理 - if (menuState.menuKey !== key) { - matchMenuOption( - option as unknown as MenuTagOptions, - menuState.menuKey, - menuState.menuTagOptions, - ) - updateDocumentTitle(option as unknown as AppMenuOption) - setKeepAliveInclude(option as unknown as AppMenuOption) + menuState.breadcrumbOptions = getCompleteRoutePath(menuState.options, key) - menuState.breadcrumbOptions = parseAndFindMatchingNodes( - menuState.options, - 'key', - key, - ) // 获取面包屑 + if (meta.sameLevel) { + nextTick().then(() => { + const fd = menuState.breadcrumbOptions.find((curr) => { + return curr.path === option.path + }) - /** 是否为根路由 */ - if (!String(key).startsWith('/')) { - /** 如果不是根路由, 则拼接完整路由并跳转 */ - const _path = getCompleteRoutePath(menuState.options, key) - .map((curr) => curr.key) - .join('/') - - router.push(_path) - } else { - /** 根路由直接跳转 */ - router.push(path) + if (!fd) { + menuState.breadcrumbOptions.push(option as unknown as AppMenuOption) } - - menuState.menuKey = key - - /** 缓存菜单 key(sessionStorage) */ - setCache('menuKey', key) - } + }) } } - /** - * - * @param path 路由地址 - * - * @remark 监听路由地址变化更新菜单状态 - * @remark 递归查找匹配项 - */ - const updateMenuKeyWhenRouteUpdate = (path: string) => { - const matchMenuItem = (options: MenuOption[]) => { - for (const i of options) { - if (i?.children?.length) { - matchMenuItem(i.children) - } - - if (path === i.path) { - changeMenuModelValue(i.path, i) - - break - } - } - } - - matchMenuItem(menuState.options as MenuOption[]) - } - /** * * @param optins menu tag option(s) @@ -168,6 +121,98 @@ export const useMenu = defineStore( : (menuState.menuTagOptions = arr) } + /** 当 url 地址发生变化触发 menuTagOptions 更新 */ + const setMenuTagOptionsWhenMenuValueChange = ( + key: string | number, + option: MenuOption, + ) => { + const tag = menuState.menuTagOptions.find((curr) => curr.path === key) + + if (!tag) { + menuState.menuTagOptions.push(option as unknown as MenuTagOptions) + } + } + + /** + * + * @param key 菜单更新后的 key + * @param option 菜单当前 option 项 + * + * @remark 修改 `menu key` 后的回调函数 + * @remark 修改后, 缓存当前选择 key 并且存储标签页与跳转页面(router push 操作) + */ + const changeMenuModelValue = (key: string | number, option: MenuOption) => { + const { meta, path } = option as unknown as AppRouteRecordRaw + + if (meta.windowOpen) { + window.open(meta.windowOpen) + } else { + /** + * + * key 以 `/` 开头, 则说明为根路由, 直接跳转 + * key 开头未匹配到 `/`, 则需要获取到完整路由后再进行跳转 + * + * 但是, 缓存 key 都以当前点击 key 为准 + */ + if (!String(key).startsWith('/')) { + /** 如果不是根路由, 则拼接完整路由并跳转 */ + const _path = getCompleteRoutePath(menuState.options, key) + .map((curr) => curr.key) + .join('/') + + router.push(_path) + } else { + /** 根路由直接跳转 */ + router.push(path) + } + + /** 检查是否为根路由 */ + const count = (path.match(new RegExp('/', 'g')) || []).length + + /** 更新浏览器标题 */ + updateDocumentTitle(option as unknown as AppMenuOption) + /** 更新缓存队列 */ + setKeepAliveInclude(option as unknown as AppMenuOption) + + if (!meta.sameLevel || (meta.sameLevel && count === 1)) { + /** 更新标签菜单 */ + setMenuTagOptionsWhenMenuValueChange(key, option) + /** 更新面包屑 */ + setBreadcrumbOptions(key, option) + + menuState.menuKey = key + /** 缓存菜单 key(sessionStorage) */ + setCache('menuKey', key) + } else { + setBreadcrumbOptions(menuState.menuKey || '', option) + } + } + } + + /** + * + * @param path 路由地址 + * + * @remark 监听路由地址变化更新菜单状态 + * @remark 递归查找匹配项 + */ + const updateMenuKeyWhenRouteUpdate = (path: string) => { + const appRawRoutes = expandRoutes(getAppRawRoutes()) + const count = (path.match(new RegExp('/', 'g')) || []).length + const fd = appRawRoutes.find((curr) => curr.path === path) + let combinePath = path + + if (count > 1) { + const splitPath = path.split('/').filter((curr) => curr) + + combinePath = splitPath[splitPath.length - 1] + } + + if (fd) { + changeMenuModelValue(combinePath, fd as unknown as MenuOption) + } + } + /** * * @remark 初始化菜单列表, 并且按照权限过滤 @@ -191,6 +236,9 @@ export const useMenu = defineStore( default: () => label.value, }), breadcrumbLabel: label.value, + /** 检查该菜单项是否展示 */ + show: + meta.hidden === false || meta.hidden === undefined ? true : false, } as AppMenuOption /** 合并 icon */ const attr: AppMenuOption = Object.assign({}, route, { @@ -198,15 +246,14 @@ export const useMenu = defineStore( }) if (option.path === getCatchMenuKey()) { - /** 设置菜单标签 */ - setMenuTagOptions(attr) /** 设置浏览器标题 */ updateDocumentTitle(attr) + setMenuTagOptionsWhenMenuValueChange( + option.path, + attr as unknown as MenuOption, + ) } - /** 检查该菜单项是否展示 */ - attr.show = validMenuItemShow(option) - return attr } @@ -214,9 +261,9 @@ export const useMenu = defineStore( const catchArr: AppMenuOption[] = [] for (const curr of routes) { - if (curr.children?.length && validMenuItemShow(curr)) { + if (curr.children?.length) { curr.children = resolveRoutes(curr.children, index++) - } else if (!validMenuItemShow(curr)) { + } else if (!validRole(curr.meta)) { /** 如果校验失败, 则不会添加进 menu options */ continue } @@ -234,15 +281,6 @@ export const useMenu = defineStore( ) resolve() - - /** 初始化后渲染面包屑 */ - nextTick(() => { - menuState.breadcrumbOptions = parseAndFindMatchingNodes( - menuState.options, - 'key', - menuState.menuKey as string, - ) - }) }) } @@ -274,11 +312,28 @@ export const useMenu = defineStore( menuState.menuTagOptions = [] } + /** + * + * 初始化系统菜单列表 + * 该方法仅执行一次 + */ + const setupPiniaMenuStore = async () => { + if (isSetupAppMenuLock.value) { + await setupAppMenu() + + isSetupAppMenuLock.value = false + } + } + /** 监听路由变化并且更新路由菜单与菜单标签 */ watch( () => route.fullPath, - (newData) => { - updateMenuKeyWhenRouteUpdate(newData) + async (newData) => { + const reg = /^([^?]+)/ + const match = newData.match(reg)?.[1] + + await setupPiniaMenuStore() + updateMenuKeyWhenRouteUpdate(match || '') }, { immediate: true, diff --git a/src/views/router-demo/router-demo-detail/index.tsx b/src/views/router-demo/router-demo-detail/index.tsx new file mode 100644 index 00000000..266326f9 --- /dev/null +++ b/src/views/router-demo/router-demo-detail/index.tsx @@ -0,0 +1,26 @@ +/** + * + * @author Ray + * + * @date 2023-06-30 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +import { NCard, NSpace } from 'naive-ui' + +const RouterDemoDetail = defineComponent({ + name: 'RouterDemoDetail', + + render() { + return ( + + 我是平层路由详情页面 + + ) + }, +}) + +export default RouterDemoDetail diff --git a/src/views/router-demo/router-demo-home/index.tsx b/src/views/router-demo/router-demo-home/index.tsx new file mode 100644 index 00000000..80c20c1b --- /dev/null +++ b/src/views/router-demo/router-demo-home/index.tsx @@ -0,0 +1,94 @@ +/** + * + * @author Ray + * + * @date 2023-06-30 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +import { NSpace, NDataTable, NButton } from 'naive-ui' + +import { useVueRouter } from '@/router/helper/useVueRouter' + +import type { DataTableColumns } from 'naive-ui' + +export interface RowData { + key: string | number + name: string + phone: string + address: string +} + +const RouterDemoHome = defineComponent({ + name: 'RouterDemoHome', + setup() { + const { router } = useVueRouter() + + const columns: DataTableColumns = [ + { + title: '姓名', + key: 'name', + }, + { + title: '地址', + key: 'address', + }, + { + title: '联系方式', + key: 'phone', + }, + { + title: '操作', + key: '', + render: (row) => { + return ( + + { + router.push({ + path: '/router-demo/router-demo-detail', + query: { + row: JSON.stringify(row), + }, + }) + }} + > + 详情 + + + ) + }, + }, + ] + const dataSource: RowData[] = [] + + for (let i = 0; i < 10; i++) { + dataSource.push({ + name: '张三', + address: 'New York No. 1 Lake Park', + phone: '010-121212', + key: i, + }) + } + + return { + dataSource, + columns, + } + }, + render() { + return ( + + + + ) + }, +}) + +export default RouterDemoHome diff --git a/tsconfig.json b/tsconfig.json index 08bc3c13..f2511d9b 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -43,9 +43,6 @@ "src/**/*.tsx", "src/**/*.ts", "src/**/*.vue", - "src/*.ts", - "src/*.vue", - "src/*", "components.d.ts", "auto-imports.d.ts", "src/types/global.d.ts"