version: v4.3.3

This commit is contained in:
XiaoDaiGua-Ray 2023-11-17 16:03:07 +08:00
parent 335e456688
commit 94f975eaff
26 changed files with 1287 additions and 973 deletions

View File

@ -1,5 +1,26 @@
# CHANGE LOG # CHANGE LOG
## 4.3.3
紧跟尤大大脚步,更新 `vite` 版本至 `5.0.0` 版本!与此同时,更新了配套所有插件!
更新 ROOT_ROUTE 的一些使用方法,该配置方法与原有的方式不变,但是有一个新的功能点则是,该配置项会传递给 global-variable 的 globalRootRoute 属性。并且更改模板原有获取 path 的方法,改为响应式获取。当你要进行动态的维护 Root Route 的时候,该方法可能可以帮助到你 `useRootRoute`
如果你在更新版本后出现一些奇奇怪怪的问题,不要犹豫,直接删除 `node_modules` 后再重新安装依赖,这是缓存导致的问题。
### Feats
- 更新 `vite` 版本至 `5.0.0`
- 升级所有配套插件
- 升级 ROOT_ROUTE 配置与使用
### Fixes
- 修复不能正确关闭标签页问题
- 修复不能正确识别是否能关闭标签页问题
- 修复 `closeAll` 方法导致标签页闪烁问题
- 修改 useVueRouter 注册时机,避免该方法使用的 HMR 报错问题
## 4.3.2 ## 4.3.2
升级 `vue` 版本至最新 `v3.3.8` 升级 `vue` 版本至最新 `v3.3.8`

View File

@ -26,6 +26,7 @@
- **缓存**:任意深度页面缓存 - **缓存**:任意深度页面缓存
- **SVG**:内置 svg icon 解决方案 - **SVG**:内置 svg icon 解决方案
- **独立的 Data Methods Views**:解耦管理的数据、方法、视图,放心二次开发 - **独立的 Data Methods Views**:解耦管理的数据、方法、视图,放心二次开发
- **模板专属 hooks**:基于模板特性封装的 hooks 让你更加方便的使用模板一些功能
## 🪄 预览地址 ## 🪄 预览地址

View File

@ -15,17 +15,18 @@ A middle and backend template based on vite4.x & ts(x) & pinia & vue3.x
## ✨ Feature ## ✨ Feature
- **Latest Technology Stack**Developed using front-end cutting-edge technologies such as vue3.x/vite4.x/pinia - **Latest Technology Stack**Developed using front-end cutting-edge technologies such as vue3.x/vite4.x/pinia.
- **TypeScript**The language for application-level JavaScript - **TypeScript**The language for application-level JavaScript.
- **App Theme**Configurable themes - **App Theme**Configurable themes.
- **Globalization**Built-in complete internationalization solution - **Globalization**Built-in complete internationalization solution.
- **Mock Data**Built-in Mock data scheme - **Mock Data**Built-in Mock data scheme.
- **Permissions**Built-in complete dynamic routing permission generation solution - **Permissions**Built-in complete dynamic routing permission generation solution.
- **Components**Secondary encapsulation of multiple commonly used components - **Components**Secondary encapsulation of multiple commonly used components.
- **Axios Request**Secondary encapsulation of the axios library, supporting functions such as cancellation, anti-shake, automatic repeat cancellation, etc. - **Axios Request**Secondary encapsulation of the axios library, supporting functions such as cancellation, anti-shake, automatic repeat cancellation, etc.
- **Page Cache**Arbitrarily deep page cache - **Page Cache**Arbitrarily deep page cache.
- **SVG**Built-in svg icon solution - **SVG**Built-in svg icon solution.
- **Standalone Data Methods Views**Decoupled management of data, methods, and views allows for secondary development with confidence - **Standalone Data Methods Views**Decoupled management of data, methods, and views allows for secondary development with confidence.
- **Template Specific Hooks** : Hooks based on the template feature package make it easier to use some of the features of the template.
## 🪄 Preview ## 🪄 Preview

1
cfg.ts
View File

@ -88,7 +88,6 @@ const config: AppConfigExport = {
host: '0.0.0.0', host: '0.0.0.0',
port: 9527, port: 9527,
open: false, open: false,
https: false,
strictPort: false, strictPort: false,
fs: { fs: {
strict: false, strict: false,

View File

@ -1,10 +1,10 @@
{ {
"name": "ray-template", "name": "ray-template",
"private": false, "private": false,
"version": "4.3.2", "version": "4.3.3",
"type": "module", "type": "module",
"engines": { "engines": {
"node": ">=16.0.0", "node": "^18.0.0 || >=20.0.0",
"pnpm": ">=8.0.0" "pnpm": ">=8.0.0"
}, },
"scripts": { "scripts": {
@ -24,74 +24,74 @@
] ]
}, },
"dependencies": { "dependencies": {
"@vueuse/core": "^9.13.0", "@vueuse/core": "^10.6.1",
"awesome-qr": "2.1.5-rc.0", "awesome-qr": "2.1.5-rc.0",
"axios": "^1.2.0", "axios": "^1.5.0",
"clipboard": "^2.0.11", "clipboard": "^2.0.11",
"crypto-js": "^4.1.1", "crypto-js": "^4.1.1",
"currency.js": "^2.0.4", "currency.js": "^2.0.4",
"dayjs": "^1.11.7", "dayjs": "^1.11.10",
"echarts": "^5.4.3", "echarts": "^5.4.3",
"lodash-es": "^4.17.21", "lodash-es": "^4.17.21",
"mockjs": "1.1.0", "mockjs": "1.1.0",
"naive-ui": "^2.35.0", "naive-ui": "^2.35.0",
"pinia": "^2.1.4", "pinia": "^2.1.6",
"pinia-plugin-persistedstate": "^3.1.0", "pinia-plugin-persistedstate": "^3.2.0",
"print-js": "^1.6.0", "print-js": "^1.6.0",
"vue": "^3.3.8", "vue": "^3.3.8",
"vue-hooks-plus": "1.8.5", "vue-hooks-plus": "1.8.5",
"vue-i18n": "^9.2.2", "vue-i18n": "^9.7.0",
"vue-router": "^4.2.4", "vue-router": "^4.2.4",
"xlsx": "^0.18.5" "xlsx": "^0.18.5"
}, },
"devDependencies": { "devDependencies": {
"@babel/core": "^7.20.2", "@babel/core": "^7.23.2",
"@babel/eslint-parser": "^7.19.1", "@babel/eslint-parser": "^7.22.11",
"@commitlint/cli": "^17.4.2", "@commitlint/cli": "^17.7.1",
"@commitlint/config-conventional": "^17.4.2", "@commitlint/config-conventional": "^17.7.0",
"@intlify/unplugin-vue-i18n": "^0.12.1", "@intlify/unplugin-vue-i18n": "^1.5.0",
"@types/crypto-js": "^4.1.1", "@types/crypto-js": "^4.1.1",
"@types/lodash-es": "^4.17.7", "@types/lodash-es": "^4.17.11",
"@types/mockjs": "1.0.7", "@types/mockjs": "1.0.7",
"@typescript-eslint/eslint-plugin": "^5.61.0", "@typescript-eslint/eslint-plugin": "^6.5.0",
"@typescript-eslint/parser": "^5.61.0", "@typescript-eslint/parser": "^6.5.0",
"@vitejs/plugin-vue": "^4.2.3", "@vitejs/plugin-vue": "^4.4.1",
"@vitejs/plugin-vue-jsx": "^3.0.1", "@vitejs/plugin-vue-jsx": "^3.0.2",
"@vue-hooks-plus/resolvers": "1.2.4", "@vue-hooks-plus/resolvers": "1.2.4",
"@vue/eslint-config-prettier": "^7.1.0", "@vue/eslint-config-prettier": "^8.0.0",
"@vue/eslint-config-typescript": "^11.0.3", "@vue/eslint-config-typescript": "^11.0.3",
"autoprefixer": "^10.4.8", "autoprefixer": "^10.4.15",
"depcheck": "^1.4.3", "depcheck": "^1.4.5",
"eslint": "^8.44.0", "eslint": "^8.52.0",
"eslint-config-prettier": "^8.8.0", "eslint-config-prettier": "^9.0.0",
"eslint-config-standard-with-typescript": "^23.0.0", "eslint-config-standard-with-typescript": "^39.0.0",
"eslint-plugin-import": "^2.25.2", "eslint-plugin-import": "^2.29.0",
"eslint-plugin-n": "^15.0.0", "eslint-plugin-n": "^16.2.0",
"eslint-plugin-prettier": "^4.2.1", "eslint-plugin-prettier": "^5.0.1",
"eslint-plugin-promise": "^6.0.0", "eslint-plugin-promise": "^6.1.1",
"eslint-plugin-vue": "^9.15.1", "eslint-plugin-vue": "^9.18.1",
"husky": "^8.0.3", "husky": "^8.0.3",
"lint-staged": "^13.1.0", "lint-staged": "^15.1.0",
"postcss": "^8.1.0", "postcss": "^8.4.31",
"postcss-px-to-viewport-8-plugin": "1.2.2", "postcss-px-to-viewport-8-plugin": "1.2.2",
"prettier": "^2.7.1", "prettier": "^3.0.3",
"rollup-plugin-visualizer": "^5.8.3", "rollup-plugin-visualizer": "^5.9.2",
"sass": "1.54.3", "sass": "1.69.5",
"svg-sprite-loader": "^6.0.11", "svg-sprite-loader": "^6.0.11",
"typescript": "^5.0.2", "typescript": "^5.2.2",
"unplugin-auto-import": "^0.15.0", "unplugin-auto-import": "^0.16.6",
"unplugin-vue-components": "^0.25.1", "unplugin-vue-components": "^0.25.2",
"vite": "^4.4.9", "vite": "^5.0.0",
"vite-plugin-cdn2": "0.12.4", "vite-plugin-cdn2": "0.15.2",
"vite-plugin-compression": "^0.5.1", "vite-plugin-compression": "^0.5.1",
"vite-plugin-ejs": "^1.6.4", "vite-plugin-ejs": "^1.7.0",
"vite-plugin-eslint": "1.8.1", "vite-plugin-eslint": "1.8.1",
"vite-plugin-imp": "^2.3.1", "vite-plugin-imp": "^2.4.0",
"vite-plugin-inspect": "^0.7.26", "vite-plugin-inspect": "^0.7.38",
"vite-plugin-mock-dev-server": "1.3.0", "vite-plugin-mock-dev-server": "1.3.4",
"vite-plugin-svg-icons": "^2.0.1", "vite-plugin-svg-icons": "^2.0.1",
"vite-svg-loader": "^4.0.0", "vite-svg-loader": "^4.0.0",
"vue-tsc": "^1.8.4" "vue-tsc": "^1.8.8"
}, },
"description": "<!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section -->", "description": "<!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section -->",
"main": "index.ts", "main": "index.ts",

1844
pnpm-lock.yaml generated

File diff suppressed because it is too large Load Diff

View File

@ -46,12 +46,15 @@ export const PRE_LOADING_CONFIG: PreloadingConfig = {
/** /**
* *
* *
* , * ROOT_ROUTE Layout Root Path path
* *
* , , * globalRootRoute
*
*
* Root Route 使 useRootRoute
*/ */
export const ROOT_ROUTE: Readonly<RootRoute> = { export const ROOT_ROUTE: RootRoute = {
name: 'Dashboard', name: 'Dashboard',
path: '/dashboard', path: '/dashboard',
} }

View File

@ -13,7 +13,7 @@ import type { AxiosConfig } from '@/types/modules/appConfig'
/** axios 相关配置 */ /** axios 相关配置 */
export const AXIOS_CONFIG: AxiosConfig = { export const AXIOS_CONFIG: AxiosConfig = {
baseURL: '', // `import.meta.env`, baseURL: '', // `const { MODE } = getAppEnvironment()`,
withCredentials: false, // 是否允许跨域携带 `cookie` withCredentials: false, // 是否允许跨域携带 `cookie`
timeout: 5 * 1000, timeout: 5 * 1000,
headers: { headers: {

View File

@ -17,17 +17,20 @@ export const combineDirective = <
>( >(
directiveModules: T, directiveModules: T,
) => { ) => {
const directives = Object.keys(directiveModules).reduce((pre, curr) => { const directives = Object.keys(directiveModules).reduce(
const fc = directiveModules[curr]?.default (pre, curr) => {
const fc = directiveModules[curr]?.default
if (typeof fc === 'function') { if (typeof fc === 'function') {
pre[curr] = fc pre[curr] = fc
return pre return pre
} else { } else {
throw new TypeError(`directiveModules: ${curr} is not function`) throw new TypeError(`directiveModules: ${curr} is not function`)
} }
}, {} as Record<K, CustomDirectiveFC<unknown, unknown>>) },
{} as Record<K, CustomDirectiveFC<unknown, unknown>>,
)
return directives return directives
} }

View File

@ -27,6 +27,9 @@
* setVariable('key', 'value') * setVariable('key', 'value')
*/ */
import { ROOT_ROUTE } from '@/app-config/appConfig'
import { cloneDeep } from 'lodash-es'
import type { AnyFC } from '@/types/modules/utils' import type { AnyFC } from '@/types/modules/utils'
const variableState = reactive({ const variableState = reactive({
@ -34,6 +37,7 @@ const variableState = reactive({
globalDrawerValue: false, // 全局抽屉控制器(小尺寸设备可用) globalDrawerValue: false, // 全局抽屉控制器(小尺寸设备可用)
globalMainLayoutLoad: true, // LayoutContent 区域加载控制器 globalMainLayoutLoad: true, // LayoutContent 区域加载控制器
layoutContentMaximize: false, // LayoutContent 区域全屏控制器 layoutContentMaximize: false, // LayoutContent 区域全屏控制器
globalRootRoute: cloneDeep(ROOT_ROUTE), // 全局根路由配置,同步至 ROOT_ROUTE
}) })
export type VariableState = typeof variableState export type VariableState = typeof variableState

View File

@ -1,9 +1,10 @@
import { useAppMenu } from './useAppMenu' import { useAppMenu } from './useAppMenu'
import { useMainPage } from './useMainPage' import { useMainPage } from './useMainPage'
import { useMenuTag } from './useMenuTag' import { useMenuTag } from './useMenuTag'
import { useRootRoute } from './useRootRoute'
export type { MaximizeOptions } from './useMainPage' export type { MaximizeOptions } from './useMainPage'
export type { Target } from './useAppMenu' export type { Target } from './useAppMenu'
export type { CloseMenuTag } from './useMenuTag' export type { CloseMenuTag } from './useMenuTag'
export { useAppMenu, useMainPage, useMenuTag } export { useAppMenu, useMainPage, useMenuTag, useRootRoute }

View File

@ -10,7 +10,8 @@
*/ */
import { useMenuGetters, useMenuActions } from '@/store' import { useMenuGetters, useMenuActions } from '@/store'
import { ROOT_ROUTE } from '@/app-config/appConfig' import { redirectRouterToDashboard } from '@/router/helper/routerCopilot'
import { useRootRoute } from '@/hooks/template/index'
import type { MenuTagOptions, Key } from '@/types/modules/app' import type { MenuTagOptions, Key } from '@/types/modules/app'
@ -24,7 +25,7 @@ export function useMenuTag() {
emptyMenuTagOptions, emptyMenuTagOptions,
setMenuTagOptions, setMenuTagOptions,
} = useMenuActions() } = useMenuActions()
const { path } = ROOT_ROUTE const { getRootPath } = useRootRoute()
/** /**
* *
@ -169,9 +170,16 @@ export function useMenuTag() {
const normal = normalMenuTagOption(target, 'close') const normal = normalMenuTagOption(target, 'close')
if (normal) { if (normal) {
const { option } = normal const { index } = normal
changeMenuModelValue(option.key, option) spliceMenTagOptions(index)
if (getMenuKey.value !== getRootPath.value) {
const length = getMenuTagOptions.value.length
const tag = getMenuTagOptions.value[length - 1]
changeMenuModelValue(tag.key as string, tag)
}
} }
} }
@ -180,13 +188,8 @@ export function useMenuTag() {
* root path * root path
*/ */
const closeAll = () => { const closeAll = () => {
const option = getMenuTagOptions.value.find((curr) => curr.key === path)
if (option) {
changeMenuModelValue(path, option)
}
emptyMenuTagOptions() emptyMenuTagOptions()
redirectRouterToDashboard()
} }
/** /**

View File

@ -0,0 +1,48 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-11-17
*
* @workspace ray-template
*
* @remark
*/
import { setVariable, getVariableToRefs } from '@/global-variable/index'
import type { DeepMutable } from '@/types/modules/helper'
export function useRootRoute() {
const globalRootRoute = getVariableToRefs('globalRootRoute')
/**
*
* @remark
*/
const getRootRoute = computed(() => globalRootRoute.value)
/**
*
* @remark path
*/
const getRootPath = computed(() => globalRootRoute.value.path)
/**
*
* @remark name
*/
const getRootName = computed(() => globalRootRoute.value.name)
const setRootRoute = (route: DeepMutable<typeof globalRootRoute.value>) => {
setVariable(
'globalRootRoute',
Object.assign({}, globalRootRoute.value, route),
)
}
return {
getRootRoute,
getRootPath,
getRootName,
setRootRoute,
}
}

View File

@ -1,14 +0,0 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-11-16
*
* @workspace ray-template
*
* @remark
*/
export function useWebFullscreen() {
//
}

View File

@ -42,12 +42,12 @@ import CloseLeft from '@/icons/close_left.svg?component'
import { useMenuGetters, useMenuActions } from '@/store' import { useMenuGetters, useMenuActions } from '@/store'
import { uuid } from '@/utils/basic' import { uuid } from '@/utils/basic'
import { hasClass } from '@/utils/element' import { hasClass } from '@/utils/element'
import { ROOT_ROUTE } from '@/app-config/appConfig'
import { queryElements } from '@use-utils/element' import { queryElements } from '@use-utils/element'
import { renderNode } from '@/utils/vue/index' import { renderNode } from '@/utils/vue/index'
import { useMainPage } from '@/hooks/template/index' import { useMainPage } from '@/hooks/template/index'
import { useMenuTag } from '@/hooks/template/index' import { useMenuTag } from '@/hooks/template/index'
import { throttle } from 'lodash-es' import { throttle } from 'lodash-es'
import { useRootRoute } from '@/hooks/template/index'
import type { ScrollbarInst } from 'naive-ui' import type { ScrollbarInst } from 'naive-ui'
import type { MenuTagOptions, AppMenuOption } from '@/types/modules/app' import type { MenuTagOptions, AppMenuOption } from '@/types/modules/app'
@ -59,7 +59,7 @@ export default defineComponent({
const { getMenuKey, getMenuTagOptions } = useMenuGetters() const { getMenuKey, getMenuTagOptions } = useMenuGetters()
const { changeMenuModelValue } = useMenuActions() const { changeMenuModelValue } = useMenuActions()
const { path } = ROOT_ROUTE const { getRootPath } = useRootRoute()
const { reload, maximize } = useMainPage() const { reload, maximize } = useMainPage()
const { const {
close, close,
@ -124,7 +124,7 @@ export default defineComponent({
const uuidScrollBar = uuid(16) // scroll bar uuid const uuidScrollBar = uuid(16) // scroll bar uuid
const actionMap = { const actionMap = {
closeCurrentPage: () => { closeCurrentPage: () => {
getMenuKey.value !== path && close(currentContextmenuIndex) getMenuKey.value !== getRootPath.value && close(currentContextmenuIndex)
}, },
reloadCurrentPage: () => { reloadCurrentPage: () => {
reload() reload()
@ -262,20 +262,12 @@ export default defineComponent({
*/ */
const setDisabledAccordionToIndex = () => { const setDisabledAccordionToIndex = () => {
const length = getMenuTagOptions.value.length - 1 const length = getMenuTagOptions.value.length - 1
const { closeable } =
getMenuTagOptions.value[currentContextmenuIndex] ??
({} as MenuTagOptions)
// 是否需要禁用关闭当前标签页 // 是否需要禁用关闭当前标签页
if (getMenuKey.value === path) { setMoreOptionsDisabled('closeCurrentPage', !closeable ?? false)
setMoreOptionsDisabled('closeCurrentPage', true)
} else {
const isRoot = moreOptions.value[currentContextmenuIndex]
// 避免 isRoot 为 undefined
if (isRoot?.key === 'closeCurrentPage') {
setMoreOptionsDisabled('closeCurrentPage', true)
} else {
setMoreOptionsDisabled('closeCurrentPage', false)
}
}
// 是否需要禁用关闭右侧标签页 // 是否需要禁用关闭右侧标签页
if (currentContextmenuIndex === length) { if (currentContextmenuIndex === length) {
@ -310,7 +302,10 @@ export default defineComponent({
/** 仅有 getMenuTagOptions 长度大于 1 并且非 root path 时, 才激活关闭按钮 */ /** 仅有 getMenuTagOptions 长度大于 1 并且非 root path 时, 才激活关闭按钮 */
const menuTagMouseenter = (option: MenuTagOptions) => { const menuTagMouseenter = (option: MenuTagOptions) => {
if (getMenuTagOptions.value.length > 1 && option.key !== path) { if (
getMenuTagOptions.value.length > 1 &&
option.key !== getRootPath.value
) {
option.closeable = true option.closeable = true
} }
} }
@ -360,21 +355,19 @@ export default defineComponent({
/** 如果有且只有一个标签页时, 禁止全部关闭操作 */ /** 如果有且只有一个标签页时, 禁止全部关闭操作 */
watch( watch(
() => getMenuTagOptions.value, () => getMenuTagOptions.value,
(newData, oldData) => { (ndata, odata) => {
// 当 menuTagOptions 长度为 1时禁用所有 canDisabledOptions 匹配的项 // 当 menuTagOptions 长度为 1时禁用所有 canDisabledOptions 匹配的项
moreOptions.value.forEach((curr) => { moreOptions.value.forEach((curr) => {
if (canDisabledOptions.includes(curr.key)) { if (canDisabledOptions.includes(curr.key)) {
newData.length > 1 ndata.length > 1 ? (curr.disabled = false) : (curr.disabled = true)
? (curr.disabled = false)
: (curr.disabled = true)
} }
}) })
// 更新当前激活标签定位 // 更新当前激活标签定位
if (oldData?.length) { if (odata?.length) {
if (newData.length > oldData?.length) { if (ndata.length > odata?.length) {
updateScrollBarPosition() updateScrollBarPosition()
} else if (newData.length === oldData?.length) { } else if (ndata.length === odata?.length) {
positionMenuTag() positionMenuTag()
} }
} }
@ -422,7 +415,7 @@ export default defineComponent({
}, },
render() { render() {
const { iconConfig } = this const { iconConfig } = this
const { maximize } = this const { maximize, closeCurrentMenuTag } = this
return ( return (
<NLayoutHeader> <NLayoutHeader>
@ -475,7 +468,7 @@ export default defineComponent({
key={curr.key} key={curr.key}
strong strong
closable={curr.closeable} closable={curr.closeable}
onClose={this.closeCurrentMenuTag.bind(this, idx)} onClose={closeCurrentMenuTag.bind(this, idx)}
type={curr.key === this.getMenuKey ? 'primary' : 'default'} type={curr.key === this.getMenuKey ? 'primary' : 'default'}
bordered={false} bordered={false}
{...{ {...{

View File

@ -21,11 +21,12 @@
*/ */
import { getStorage } from '@/utils/cache' import { getStorage } from '@/utils/cache'
import { APP_CATCH_KEY, ROOT_ROUTE } from '@/app-config/appConfig' import { APP_CATCH_KEY } from '@/app-config/appConfig'
import { redirectRouterToDashboard } from '@/router/helper/routerCopilot' import { redirectRouterToDashboard } from '@/router/helper/routerCopilot'
import { WHITE_ROUTES } from '@/app-config/routerConfig' import { WHITE_ROUTES } from '@/app-config/routerConfig'
import { validRole } from '@/router/helper/routerCopilot' import { validRole } from '@/router/helper/routerCopilot'
import { isValueType } from '@/utils/basic' import { isValueType } from '@/utils/basic'
import { useRootRoute } from '@/hooks/template/index'
import type { Router, RouteLocationNormalized } from 'vue-router' import type { Router, RouteLocationNormalized } from 'vue-router'
import type { AppRouteMeta } from '@/router/type' import type { AppRouteMeta } from '@/router/type'
@ -33,6 +34,7 @@ import type { AppRouteMeta } from '@/router/type'
/** 路由守卫 */ /** 路由守卫 */
export const permissionRouter = (router: Router) => { export const permissionRouter = (router: Router) => {
const { beforeEach } = router const { beforeEach } = router
const { getRootPath } = useRootRoute()
const isToLogin = ( const isToLogin = (
to: RouteLocationNormalized, to: RouteLocationNormalized,
@ -44,7 +46,7 @@ export const permissionRouter = (router: Router) => {
const catchRoutePath = getStorage( const catchRoutePath = getStorage(
'menuKey', 'menuKey',
'sessionStorage', 'sessionStorage',
ROOT_ROUTE.path, getRootPath.value,
) )
const { meta, name } = to const { meta, name } = to

View File

@ -12,10 +12,10 @@
import { permissionRouter } from './permission' import { permissionRouter } from './permission'
import { SETUP_ROUTER_ACTION, SUPER_ADMIN } from '@/app-config/routerConfig' import { SETUP_ROUTER_ACTION, SUPER_ADMIN } from '@/app-config/routerConfig'
import { useVueRouter } from '@/hooks/web/index' import { useVueRouter } from '@/hooks/web/index'
import { ROOT_ROUTE } from '@/app-config/appConfig'
import { setStorage } from '@/utils/cache' import { setStorage } from '@/utils/cache'
import { getAppEnvironment } from '@/utils/basic' import { getAppEnvironment } from '@/utils/basic'
import { useSigningGetters } from '@/store' import { useSigningGetters } from '@/store'
import { useRootRoute } from '@/hooks/template/index'
import type { Router } from 'vue-router' import type { Router } from 'vue-router'
import type { AppRouteMeta } from '@/router/type' import type { AppRouteMeta } from '@/router/type'
@ -132,9 +132,9 @@ export const redirectRouterToDashboard = (isReplace = true) => {
const { router } = useVueRouter() const { router } = useVueRouter()
const { push, replace } = router const { push, replace } = router
const { path } = ROOT_ROUTE const { getRootPath } = useRootRoute()
setStorage('menuKey', path) setStorage('menuKey', getRootPath.value)
isReplace ? replace(path) : push(path) isReplace ? replace(getRootPath.value) : push(getRootPath.value)
} }

View File

@ -35,7 +35,7 @@ export const setupRouter = async (app: App<Element>) => {
router = await createVueRouter() router = await createVueRouter()
vueRouterRegister(router) vueRouterRegister(router)
useVueRouter()
app.use(router) app.use(router)
// 等待 router 挂载后,初始化 useRouter 方法
useVueRouter()
} }

View File

@ -1,27 +1,31 @@
import Layout from '@/layout/index' import Layout from '@/layout/index'
import { appExpandRoutes } from './appRouteModules' import { appExpandRoutes } from './appRouteModules'
import { ROOT_ROUTE } from '@/app-config/appConfig' import { useRootRoute } from '@/hooks/template/index'
export default async () => [ export default async () => {
/** const { getRootPath } = useRootRoute()
*
* Login return [
* /**
*/ *
{ * Login
path: '/', *
name: 'login', */
component: () => import('@/views/login/index'), {
}, path: '/',
/** name: 'login',
* component: () => import('@/views/login/index'),
* App Layout },
*/ /**
{ *
path: '/', * App Layout
name: 'layout', */
redirect: ROOT_ROUTE.path, {
component: Layout, path: '/',
children: appExpandRoutes, name: 'layout',
}, redirect: getRootPath.value,
] component: Layout,
children: appExpandRoutes,
},
]
}

View File

@ -10,7 +10,7 @@
*/ */
import { piniaMenuStore } from '../modules/menu' import { piniaMenuStore } from '../modules/menu'
import { ROOT_ROUTE } from '@/app-config/appConfig' import { useRootRoute } from '@/hooks/template/index'
export const useMenuGetters = () => { export const useMenuGetters = () => {
const variable = piniaMenuStore() const variable = piniaMenuStore()
@ -35,16 +35,16 @@ export const useMenuGetters = () => {
* @remark * @remark
*/ */
const getMenuTagOptions = computed(() => { const getMenuTagOptions = computed(() => {
const { path } = ROOT_ROUTE const { getRootPath } = useRootRoute()
return variable.menuTagOptions.map((curr, _idx, currentArray) => { return variable.menuTagOptions.map((curr, _idx, currentArray) => {
if (curr.key === getMenuKey.value && curr.key !== path) { if (curr.key === getMenuKey.value && curr.key !== getRootPath.value) {
curr.closeable = true curr.closeable = true
} else { } else {
curr.closeable = false curr.closeable = false
} }
if (curr.key === path) { if (curr.key === getRootPath.value) {
curr.closeable = false curr.closeable = false
} }

View File

@ -11,10 +11,11 @@
/** 本方法感谢 <https://yunkuangao.me/> 的支持 */ /** 本方法感谢 <https://yunkuangao.me/> 的支持 */
import { APP_MENU_CONFIG, ROOT_ROUTE } from '@/app-config/appConfig' import { APP_MENU_CONFIG } from '@/app-config/appConfig'
import RIcon from '@/components/RIcon/index' import RIcon from '@/components/RIcon/index'
import { isValueType } from '@/utils/basic' import { isValueType } from '@/utils/basic'
import { getStorage } from '@/utils/cache' import { getStorage } from '@/utils/cache'
import { useRootRoute } from '@/hooks/template/index'
import type { import type {
AppMenuOption, AppMenuOption,
@ -171,13 +172,13 @@ export const hasMenuIcon = (option: AppMenuOption) => {
return () => icon return () => icon
} }
/** 获取缓存的 menu key, 如果未获取到则使用 ROOT_ROUTE path 当作默认激活路由菜单 */ /** 获取缓存的 menu key, 如果未获取到则使用 getRootPath 当作默认激活路由菜单 */
export const getCatchMenuKey = () => { export const getCatchMenuKey = () => {
const { path: rootPath } = ROOT_ROUTE const { getRootPath } = useRootRoute()
const cacheMenuKey = getStorage<AppMenuKey>( const cacheMenuKey = getStorage<AppMenuKey>(
'menuKey', 'menuKey',
'sessionStorage', 'sessionStorage',
rootPath, getRootPath.value,
) )
return cacheMenuKey return cacheMenuKey

View File

@ -13,3 +13,15 @@ export type ConditionalPick<Base, Condition> = Pick<
export type Recordable<T = any> = Record<string, T> export type Recordable<T = any> = Record<string, T>
export type ValueOf<T extends object> = T[keyof T] export type ValueOf<T extends object> = T[keyof T]
export type Mutable<T> = {
-readonly [P in keyof T]: T[P]
}
export type DeepMutable<T> = {
-readonly [P in keyof T]: T[P] extends ReadonlyArray<infer U>
? Array<DeepMutable<U>>
: T[P] extends object
? DeepMutable<T[P]>
: T[P]
}

View File

@ -2,10 +2,10 @@ import { NForm, NFormItem, NInput, NButton } from 'naive-ui'
import { setStorage } from '@/utils/cache' import { setStorage } from '@/utils/cache'
import { useI18n } from '@/hooks/web/index' import { useI18n } from '@/hooks/web/index'
import { APP_CATCH_KEY, ROOT_ROUTE } from '@/app-config/appConfig' import { APP_CATCH_KEY } from '@/app-config/appConfig'
import { useVueRouter } from '@/hooks/web/index'
import { setVariable, getVariableToRefs } from '@/global-variable/index' import { setVariable, getVariableToRefs } from '@/global-variable/index'
import { useSigningActions, useSigningGetters } from '@/store' import { useSigningActions } from '@/store'
import { useRootRoute } from '@/hooks/template/index'
import type { FormInst } from 'naive-ui' import type { FormInst } from 'naive-ui'
@ -16,7 +16,7 @@ export default defineComponent({
const { t } = useI18n() const { t } = useI18n()
const { signing } = useSigningActions() const { signing } = useSigningActions()
const { path } = ROOT_ROUTE const { getRootPath } = useRootRoute()
const globalSpinning = getVariableToRefs('globalSpinning') const globalSpinning = getVariableToRefs('globalSpinning')
const useSigningForm = () => ({ const useSigningForm = () => ({
@ -24,7 +24,7 @@ export default defineComponent({
pwd: '123456', pwd: '123456',
}) })
const { router } = useVueRouter() const router = useRouter()
const signingForm = ref(useSigningForm()) const signingForm = ref(useSigningForm())
const rules = { const rules = {
@ -57,7 +57,7 @@ export default defineComponent({
setStorage(APP_CATCH_KEY.token, 'tokenValue') setStorage(APP_CATCH_KEY.token, 'tokenValue')
setStorage(APP_CATCH_KEY.signing, res.data) setStorage(APP_CATCH_KEY.signing, res.data)
router.push(path) router.push(getRootPath.value)
}, 2 * 1000) }, 2 * 1000)
} }
}) })

View File

@ -80,6 +80,7 @@ const Login = defineComponent({
<NDropdown <NDropdown
options={LOCAL_OPTIONS} options={LOCAL_OPTIONS}
onSelect={(key) => this.updateLocale(key)} onSelect={(key) => this.updateLocale(key)}
trigger="click"
> >
<RIcon <RIcon
customClassName="login-icon" customClassName="login-icon"

View File

@ -1,6 +1,7 @@
/* eslint-disable */ /* eslint-disable */
/* prettier-ignore */ /* prettier-ignore */
// @ts-nocheck // @ts-nocheck
// noinspection JSUnusedGlobalSymbols
// Generated by unplugin-auto-import // Generated by unplugin-auto-import
export {} export {}
declare global { declare global {