diff --git a/apps/web-antd/src/apis/modules/user.ts b/apps/web-antd/src/apis/modules/user.ts index c39090d1d..75d515b46 100644 --- a/apps/web-antd/src/apis/modules/user.ts +++ b/apps/web-antd/src/apis/modules/user.ts @@ -1,6 +1,7 @@ -import type { UserApiType } from '@/apis/types'; import type { UserInfo } from '@vben/types'; +import type { UserApiType } from '@/apis/types'; + import { request } from '@/forward'; /** diff --git a/apps/web-antd/src/app.vue b/apps/web-antd/src/app.vue index 692545e03..bce3c2aae 100644 --- a/apps/web-antd/src/app.vue +++ b/apps/web-antd/src/app.vue @@ -1,13 +1,14 @@ + + diff --git a/internal/lint-configs/eslint-config/package.json b/internal/lint-configs/eslint-config/package.json index 8706d4c52..877c7f75d 100644 --- a/internal/lint-configs/eslint-config/package.json +++ b/internal/lint-configs/eslint-config/package.json @@ -41,9 +41,9 @@ "eslint-config-prettier": "^9.1.0", "eslint-plugin-eslint-comments": "^3.2.0", "eslint-plugin-i": "^2.29.1", - "eslint-plugin-jsdoc": "^48.2.7", + "eslint-plugin-jsdoc": "^48.2.8", "eslint-plugin-jsonc": "^2.16.0", - "eslint-plugin-n": "^17.7.0", + "eslint-plugin-n": "^17.8.0", "eslint-plugin-no-only-tests": "^3.1.0", "eslint-plugin-perfectionist": "^2.10.0", "eslint-plugin-prettier": "^5.1.3", diff --git a/internal/lint-configs/eslint-config/src/configs/perfectionist.ts b/internal/lint-configs/eslint-config/src/configs/perfectionist.ts index 1c45d3296..86814fa43 100644 --- a/internal/lint-configs/eslint-config/src/configs/perfectionist.ts +++ b/internal/lint-configs/eslint-config/src/configs/perfectionist.ts @@ -22,31 +22,38 @@ export async function perfectionist(): Promise { { 'custom-groups': { type: { - vben: 'vue', - vue: ['vue', 'vue-*', '@vue*'], + vben: 'vben', + vue: 'vue', }, value: { - vben: 'vben', - vue: ['@vben-*', '@vben-core/*'], + vben: ['@vben*', '@vben/*', '@vben-core/*'], + vue: ['vue', 'vue-*', '@vue*'], }, }, groups: [ - 'side-effect', - 'type', - 'vue', + ['external-type', 'builtin-type', 'type'], + ['parent-type', 'sibling-type', 'index-type'], + ['internal-type'], 'builtin', + 'vue', 'vben', 'external', - 'internal-type', 'internal', ['parent', 'sibling', 'index'], + 'side-effect', + 'side-effect-style', 'style', 'object', 'unknown', - 'type', - ['parent-type', 'sibling-type', 'index-type'], ], - 'internal-pattern': ['@/layouts/**', '@/router/**', '@/views/**'], + 'internal-pattern': [ + '@/layouts/**', + '@/apis/**', + '@/forward/**', + '@/router/**', + '@/views/**', + '#/**', + ], 'newlines-between': 'always', order: 'asc', type: 'natural', diff --git a/internal/tailwind-config/package.json b/internal/tailwind-config/package.json index 61fdb67c9..9c21f576a 100644 --- a/internal/tailwind-config/package.json +++ b/internal/tailwind-config/package.json @@ -54,12 +54,12 @@ "@tailwindcss/nesting": "0.0.0-insiders.565cd3e", "@tailwindcss/typography": "^0.5.13", "autoprefixer": "^10.4.19", - "cssnano": "^7.0.1", + "cssnano": "^7.0.2", "postcss": "^8.4.38", "postcss-antd-fixes": "^0.2.0", "postcss-import": "^16.1.0", "postcss-preset-env": "^9.5.14", - "tailwindcss": "^3.4.3", + "tailwindcss": "^3.4.4", "tailwindcss-animate": "^1.0.7" }, "devDependencies": { diff --git a/internal/tailwind-config/src/index.ts b/internal/tailwind-config/src/index.ts index 0c2e85dd6..ef2ee70d9 100644 --- a/internal/tailwind-config/src/index.ts +++ b/internal/tailwind-config/src/index.ts @@ -2,10 +2,11 @@ import type { Config } from 'tailwindcss'; import path from 'node:path'; +import { fs, getPackagesSync } from '@vben/node-utils'; + import { addDynamicIconSelectors } from '@iconify/tailwind'; import formsPlugin from '@tailwindcss/forms'; import typographyPlugin from '@tailwindcss/typography'; -import { fs, getPackagesSync } from '@vben/node-utils'; import animate from 'tailwindcss-animate'; import { enterAnimationPlugin } from './plugins/entry'; diff --git a/internal/vite-config/src/config/application.ts b/internal/vite-config/src/config/application.ts index 770a8370e..522541960 100644 --- a/internal/vite-config/src/config/application.ts +++ b/internal/vite-config/src/config/application.ts @@ -1,5 +1,7 @@ import type { UserConfig } from 'vite'; +import type { DefineApplicationOptions } from '../typing'; + import { resolve } from 'node:path'; import { defineConfig, loadEnv, mergeConfig } from 'vite'; @@ -7,8 +9,6 @@ import { defineConfig, loadEnv, mergeConfig } from 'vite'; import { getApplicationConditionPlugins } from '../plugins'; import { getCommonConfig } from './common'; -import type { DefineApplicationOptions } from '../typing'; - function defineApplicationConfig(options: DefineApplicationOptions = {}) { return defineConfig(async ({ command, mode }) => { const { application = {}, vite = {} } = options; diff --git a/internal/vite-config/src/config/index.ts b/internal/vite-config/src/config/index.ts index 547ceffab..6cded5597 100644 --- a/internal/vite-config/src/config/index.ts +++ b/internal/vite-config/src/config/index.ts @@ -1,11 +1,11 @@ +import type { DefineConfig } from '../typing'; + import { existsSync } from 'node:fs'; import { join } from 'node:path'; import { defineApplicationConfig } from './application'; import { defineLibraryConfig } from './library'; -import type { DefineConfig } from '../typing'; - export * from './application'; export * from './library'; diff --git a/internal/vite-config/src/config/library.ts b/internal/vite-config/src/config/library.ts index 18e74d01b..fca61d140 100644 --- a/internal/vite-config/src/config/library.ts +++ b/internal/vite-config/src/config/library.ts @@ -1,13 +1,14 @@ import type { UserConfig } from 'vite'; +import type { DefineLibraryOptions } from '../typing'; + import { readPackageJSON } from '@vben/node-utils'; + import { defineConfig, mergeConfig } from 'vite'; import { getLibraryConditionPlugins } from '../plugins'; import { getCommonConfig } from './common'; -import type { DefineLibraryOptions } from '../typing'; - function defineLibraryConfig(options: DefineLibraryOptions = {}) { return defineConfig(async ({ command, mode }) => { const root = process.cwd(); diff --git a/internal/vite-config/src/plugins/extra-app-config.ts b/internal/vite-config/src/plugins/extra-app-config.ts index acea14643..13d98ec95 100644 --- a/internal/vite-config/src/plugins/extra-app-config.ts +++ b/internal/vite-config/src/plugins/extra-app-config.ts @@ -3,6 +3,7 @@ import { generatorContentHash, readPackageJSON, } from '@vben/node-utils'; + import { type PluginOption } from 'vite'; import { getEnvConfig } from '../utils/env'; diff --git a/internal/vite-config/src/plugins/index.ts b/internal/vite-config/src/plugins/index.ts index 553f7262d..49495ae03 100644 --- a/internal/vite-config/src/plugins/index.ts +++ b/internal/vite-config/src/plugins/index.ts @@ -1,9 +1,17 @@ import type { PluginOption } from 'vite'; +import type { + ApplicationPluginOptions, + CommonPluginOptions, + ConditionPlugin, + LibraryPluginOptions, +} from '../typing'; + import { join } from 'node:path'; -import viteVueI18nPlugin from '@intlify/unplugin-vue-i18n/vite'; import { getPackage } from '@vben/node-utils'; + +import viteVueI18nPlugin from '@intlify/unplugin-vue-i18n/vite'; import viteVue from '@vitejs/plugin-vue'; import viteVueJsx from '@vitejs/plugin-vue-jsx'; import { visualizer as viteVisualizerPlugin } from 'rollup-plugin-visualizer'; @@ -19,13 +27,6 @@ import { viteExtraAppConfigPlugin } from './extra-app-config'; import { viteImportMapPlugin } from './importmap'; import { viteInjectAppLoadingPlugin } from './inject-app-loading'; -import type { - ApplicationPluginOptions, - CommonPluginOptions, - ConditionPlugin, - LibraryPluginOptions, -} from '../typing'; - /** * 获取条件成立的 vite 插件 * @param conditionPlugins diff --git a/internal/vite-config/src/plugins/inject-app-loading/index.ts b/internal/vite-config/src/plugins/inject-app-loading/index.ts index f5d8f2527..7aedc3e91 100644 --- a/internal/vite-config/src/plugins/inject-app-loading/index.ts +++ b/internal/vite-config/src/plugins/inject-app-loading/index.ts @@ -2,6 +2,7 @@ import { join } from 'node:path'; import { fileURLToPath } from 'node:url'; import { fs } from '@vben/node-utils'; + import { type PluginOption } from 'vite'; /** diff --git a/internal/vite-config/src/plugins/inject-app-loading/loading-antd.html b/internal/vite-config/src/plugins/inject-app-loading/loading-antd.html index cf7e9676d..44bdd91ad 100644 --- a/internal/vite-config/src/plugins/inject-app-loading/loading-antd.html +++ b/internal/vite-config/src/plugins/inject-app-loading/loading-antd.html @@ -29,7 +29,7 @@ .loading.hidden { visibility: hidden; opacity: 0; - transition: all 1s ease-out; + transition: all 0.6s ease-out; } .loading .dots { diff --git a/internal/vite-config/src/plugins/inject-app-loading/loading.html b/internal/vite-config/src/plugins/inject-app-loading/loading.html index a947cd825..8833fb3ca 100644 --- a/internal/vite-config/src/plugins/inject-app-loading/loading.html +++ b/internal/vite-config/src/plugins/inject-app-loading/loading.html @@ -23,7 +23,7 @@ .loading.hidden { visibility: hidden; opacity: 0; - transition: all 1s ease-out; + transition: all 0.6s ease-out; } .dark .loading { diff --git a/internal/vite-config/src/utils/env.ts b/internal/vite-config/src/utils/env.ts index c08c3cfc7..ee5adc24e 100644 --- a/internal/vite-config/src/utils/env.ts +++ b/internal/vite-config/src/utils/env.ts @@ -1,6 +1,7 @@ import { join } from 'node:path'; import { fs } from '@vben/node-utils'; + import dotenv from 'dotenv'; /** diff --git a/package.json b/package.json index 62456c280..49fdc9799 100644 --- a/package.json +++ b/package.json @@ -48,7 +48,7 @@ "@changesets/cli": "^2.27.5", "@ls-lint/ls-lint": "^2.2.3", "@types/jsdom": "^21.1.7", - "@types/node": "^20.14.1", + "@types/node": "^20.14.2", "@vben/commitlint-config": "workspace:*", "@vben/eslint-config": "workspace:*", "@vben/lint-staged-config": "workspace:*", @@ -76,7 +76,7 @@ "node": ">=18.7.0", "pnpm": ">=8.5.0" }, - "packageManager": "pnpm@9.1.4", + "packageManager": "pnpm@9.2.0", "pnpm": { "overrides": { "@ctrl/tinycolor": "4.1.0", diff --git a/packages/@vben-core/forward/helpers/src/generator-menus.ts b/packages/@vben-core/forward/helpers/src/generator-menus.ts index 1a5901752..ae0d2ca66 100644 --- a/packages/@vben-core/forward/helpers/src/generator-menus.ts +++ b/packages/@vben-core/forward/helpers/src/generator-menus.ts @@ -1,7 +1,7 @@ import type { ExRouteRecordRaw, MenuRecordRaw } from '@vben-core/typings'; +import type { RouteRecordRaw, Router } from 'vue-router'; import { mapTree } from '@vben-core/toolkit'; -import type { RouteRecordRaw, Router } from 'vue-router'; /** * 根据 routes 生成菜单列表 diff --git a/packages/@vben-core/forward/helpers/src/generator-routes.ts b/packages/@vben-core/forward/helpers/src/generator-routes.ts index c1e2674b0..5e787566c 100644 --- a/packages/@vben-core/forward/helpers/src/generator-routes.ts +++ b/packages/@vben-core/forward/helpers/src/generator-routes.ts @@ -1,16 +1,30 @@ -import { filterTree } from '@vben-core/toolkit'; import type { RouteRecordRaw } from 'vue-router'; + +import { filterTree, mapTree } from '@vben-core/toolkit'; /** * 动态生成路由 */ async function generatorRoutes( routes: RouteRecordRaw[], roles: string[], + forbiddenPage?: RouteRecordRaw['component'], ): Promise { // 根据角色标识过滤路由表,判断当前用户是否拥有指定权限 - return filterTree(routes, (route) => { + const finalRoutes = filterTree(routes, (route) => { return hasVisible(route) && hasAuthority(route, roles); }); + + if (!forbiddenPage) { + return finalRoutes; + } + + // 如果有禁止访问的页面,将禁止访问的页面替换为403页面 + return mapTree(finalRoutes, (route) => { + if (menuHasVisibleWithForbidden(route)) { + route.component = forbiddenPage; + } + return route; + }); } /** @@ -24,9 +38,10 @@ function hasAuthority(route: RouteRecordRaw, access: string[]) { if (!authority) { return true; } - return access.some((value) => { - return authority.includes(value); - }); + return ( + access.some((value) => authority.includes(value)) || + menuHasVisibleWithForbidden(route) + ); } /** @@ -37,4 +52,12 @@ function hasVisible(route?: RouteRecordRaw) { return !route?.meta?.hideInMenu; } +/** + * 判断路由是否在菜单中显示,但是访问会被重定向到403 + * @param route + */ +function menuHasVisibleWithForbidden(route: RouteRecordRaw) { + return !!route.meta?.menuVisibleWithForbidden; +} + export { generatorRoutes, hasAuthority, hasVisible }; diff --git a/packages/@vben-core/forward/helpers/src/merge-route-modules.test.ts b/packages/@vben-core/forward/helpers/src/merge-route-modules.test.ts index 591ffd26b..40d9243c4 100644 --- a/packages/@vben-core/forward/helpers/src/merge-route-modules.test.ts +++ b/packages/@vben-core/forward/helpers/src/merge-route-modules.test.ts @@ -1,11 +1,11 @@ import type { RouteRecordRaw } from 'vue-router'; +import type { RouteModuleType } from './merge-route-modules'; + import { describe, expect, it } from 'vitest'; import { mergeRouteModules } from './merge-route-modules'; -import type { RouteModuleType } from './merge-route-modules'; - describe('mergeRouteModules', () => { it('should merge route modules correctly', () => { const routeModules: Record = { diff --git a/packages/@vben-core/forward/preferences/src/index.ts b/packages/@vben-core/forward/preferences/src/index.ts index 75305555e..011b814ef 100644 --- a/packages/@vben-core/forward/preferences/src/index.ts +++ b/packages/@vben-core/forward/preferences/src/index.ts @@ -1,9 +1,9 @@ import type { Flatten } from '@vben-core/typings'; -import { preferencesManager } from './preferences'; - import type { Preferences } from './types'; +import { preferencesManager } from './preferences'; + // 偏好设置(带有层级关系) const preferences: Preferences = preferencesManager.getPreferences(); diff --git a/packages/@vben-core/forward/preferences/src/preferences.ts b/packages/@vben-core/forward/preferences/src/preferences.ts index b835bee7e..19b27bda1 100644 --- a/packages/@vben-core/forward/preferences/src/preferences.ts +++ b/packages/@vben-core/forward/preferences/src/preferences.ts @@ -4,6 +4,10 @@ import type { FlattenObjectKeys, } from '@vben-core/typings'; +import type { Preferences } from './types'; + +import { markRaw, reactive, watch } from 'vue'; + import { StorageManager } from '@vben-core/cache'; import { flattenObject, nestedObject } from '@vben-core/helpers'; import { convertToHslCssVar, merge } from '@vben-core/toolkit'; @@ -14,12 +18,9 @@ import { useCssVar, useDebounceFn, } from '@vueuse/core'; -import { markRaw, reactive, watch } from 'vue'; import { defaultPreferences } from './config'; -import type { Preferences } from './types'; - const STORAGE_KEY = 'preferences'; const STORAGE_KEY_LOCALE = `${STORAGE_KEY}-locale`; const STORAGE_KEY_THEME = `${STORAGE_KEY}-theme`; diff --git a/packages/@vben-core/forward/preferences/src/use-preferences.ts b/packages/@vben-core/forward/preferences/src/use-preferences.ts index 5b4ff90bd..21fad0ad2 100644 --- a/packages/@vben-core/forward/preferences/src/use-preferences.ts +++ b/packages/@vben-core/forward/preferences/src/use-preferences.ts @@ -1,7 +1,7 @@ -import { diff } from '@vben-core/toolkit'; - import { computed } from 'vue'; +import { diff } from '@vben-core/toolkit'; + import { isDarkTheme, preferencesManager } from './preferences'; function usePreferences() { diff --git a/packages/@vben-core/forward/request/src/request-client/request-client.ts b/packages/@vben-core/forward/request/src/request-client/request-client.ts index 3e8554bf3..3d85c9577 100644 --- a/packages/@vben-core/forward/request/src/request-client/request-client.ts +++ b/packages/@vben-core/forward/request/src/request-client/request-client.ts @@ -6,6 +6,8 @@ import type { InternalAxiosRequestConfig, } from 'axios'; +import type { MakeAuthorizationFn, RequestClientOptions } from './types'; + import { merge } from '@vben-core/toolkit'; import axios from 'axios'; @@ -15,8 +17,6 @@ import { FileDownloader } from './modules/downloader'; import { InterceptorManager } from './modules/interceptor'; import { FileUploader } from './modules/uploader'; -import type { MakeAuthorizationFn, RequestClientOptions } from './types'; - class RequestClient { private instance: AxiosInstance; private makeAuthorization: MakeAuthorizationFn | undefined; diff --git a/packages/@vben-core/forward/stores/src/modules/access.ts b/packages/@vben-core/forward/stores/src/modules/access.ts index 34fe35712..ff1a0ca56 100644 --- a/packages/@vben-core/forward/stores/src/modules/access.ts +++ b/packages/@vben-core/forward/stores/src/modules/access.ts @@ -1,5 +1,4 @@ import type { MenuRecordRaw } from '@vben-core/typings'; - import type { RouteRecordRaw } from 'vue-router'; import { acceptHMRUpdate, defineStore } from 'pinia'; diff --git a/packages/@vben-core/forward/stores/src/modules/tabs.test.ts b/packages/@vben-core/forward/stores/src/modules/tabs.test.ts index 255c11150..9cb17ce63 100644 --- a/packages/@vben-core/forward/stores/src/modules/tabs.test.ts +++ b/packages/@vben-core/forward/stores/src/modules/tabs.test.ts @@ -1,6 +1,7 @@ +import { createRouter, createWebHistory } from 'vue-router'; + import { createPinia, setActivePinia } from 'pinia'; import { beforeEach, describe, expect, it, vi } from 'vitest'; -import { createRouter, createWebHistory } from 'vue-router'; import { useTabsStore } from './tabs'; diff --git a/packages/@vben-core/forward/stores/src/modules/tabs.ts b/packages/@vben-core/forward/stores/src/modules/tabs.ts index 01ad18847..d9e962f86 100644 --- a/packages/@vben-core/forward/stores/src/modules/tabs.ts +++ b/packages/@vben-core/forward/stores/src/modules/tabs.ts @@ -1,10 +1,12 @@ -import { startProgress, stopProgress } from '@vben-core/toolkit'; -import { TabItem } from '@vben-core/typings'; import type { RouteRecordNormalized, Router } from 'vue-router'; -import { acceptHMRUpdate, defineStore } from 'pinia'; import { toRaw } from 'vue'; +import { startProgress, stopProgress } from '@vben-core/toolkit'; +import { TabItem } from '@vben-core/typings'; + +import { acceptHMRUpdate, defineStore } from 'pinia'; + /** * @zh_CN 克隆路由,防止路由被修改 * @param route diff --git a/packages/@vben-core/shared/iconify/src/factory.ts b/packages/@vben-core/shared/iconify/src/factory.ts index 09d3ba46f..b4323cd3e 100644 --- a/packages/@vben-core/shared/iconify/src/factory.ts +++ b/packages/@vben-core/shared/iconify/src/factory.ts @@ -1,6 +1,7 @@ -import { Icon } from '@iconify/vue'; import { defineComponent, h } from 'vue'; +import { Icon } from '@iconify/vue'; + function createIcon(name: string) { return defineComponent({ setup(props, { attrs }) { diff --git a/packages/@vben-core/shared/typings/src/vue-router.d.ts b/packages/@vben-core/shared/typings/src/vue-router.d.ts index 47787eb7c..e1d4b8f2a 100644 --- a/packages/@vben-core/shared/typings/src/vue-router.d.ts +++ b/packages/@vben-core/shared/typings/src/vue-router.d.ts @@ -72,6 +72,10 @@ interface RouteMeta { * 路由是否已经加载过 */ loaded?: boolean; + /** + * 菜单可以看到,但是访问会被重定向到403 + */ + menuVisibleWithForbidden?: boolean; /** * 用于路由->菜单排序 */ @@ -80,6 +84,7 @@ interface RouteMeta { * 外链-跳转路径 */ target?: string; + /** * 标题名称 */ diff --git a/packages/@vben-core/shared/typings/vue-router.d.ts b/packages/@vben-core/shared/typings/vue-router.d.ts index abe9f9265..239f315ef 100644 --- a/packages/@vben-core/shared/typings/vue-router.d.ts +++ b/packages/@vben-core/shared/typings/vue-router.d.ts @@ -1,7 +1,7 @@ -import 'vue-router'; - import type { RouteMeta as IRouteMeta } from '@vben-core/typings'; +import 'vue-router'; + declare module 'vue-router' { interface RouteMeta extends IRouteMeta {} } diff --git a/packages/@vben-core/uikit/layout-ui/src/components/layout-content.vue b/packages/@vben-core/uikit/layout-ui/src/components/layout-content.vue index 96b93b76b..7d3aef763 100644 --- a/packages/@vben-core/uikit/layout-ui/src/components/layout-content.vue +++ b/packages/@vben-core/uikit/layout-ui/src/components/layout-content.vue @@ -2,7 +2,6 @@ import type { ContentCompactType } from '@vben-core/typings'; import type { CSSProperties } from 'vue'; - import { computed } from 'vue'; interface Props { diff --git a/packages/@vben-core/uikit/layout-ui/src/components/layout-footer.vue b/packages/@vben-core/uikit/layout-ui/src/components/layout-footer.vue index b4d68616f..07a690170 100644 --- a/packages/@vben-core/uikit/layout-ui/src/components/layout-footer.vue +++ b/packages/@vben-core/uikit/layout-ui/src/components/layout-footer.vue @@ -1,10 +1,9 @@