version: v5.0.6

This commit is contained in:
XiaoDaiGua-Ray 2024-11-23 12:42:28 +08:00
parent 2c84e3ce4c
commit 852d7ca90a
50 changed files with 918 additions and 664 deletions

View File

@ -1,5 +1,33 @@
# CHANGE LOG
## 5.0.6
## Feats
- 新增 `useChartProvider` 方法,允许注入 `RCharts` 组件配置
- 更新 `echarts` 版本至 `5.5.1`
- 更新 `vue` 版本至 `3.5.13`
- 更新 `@vueuse/core` 版本至 `11.2.0`
- 修改 `SettingDrawer` 组件的 `defaultOptions` 配置项管理方式,现在迁移至 `store.setting` 包中
- 重构 `cache` 工具模块,更有好的类型推导、更少的代码量
- 重构 `precision` 工具模块,更好的类型推导、更少的代码量
- 重写 `updateObjectValue` 方法,现在类型提示更加准确
- 全局使用 `useTemplateRef`, `shallowRef` 方法替代 `ref` 注册模板引用,减少不必要的响应式代理
- 优化 `MenuTag` 组件的关闭按钮样式
- `LockScreen` 组件新增头像展示
- `AppAvatar` 组件现在默认获取 `avatar` 字段为空的时候,展示名字的首字
- 优化 `UnlockScreen` 组件样式,现在会根据主题自动调整背景颜色
- 优化内容区域过度动画效果
## Fixes
- 修复 `404` 页面【返回】按钮不能准确返回的问题
- 修复 `usePagination.getCallback` 方法类型丢失问题;修复该方法获取实时回调不准确的问题
- 修复初始化时,菜单滚动条不能准确滚动到当前激活项的问题
- 修复 `UnlockScreen` 组件在白色主题下,导致样式显示差异问题,现在统一为黑色主题配置覆盖
- 修复 `LockScreen` 组件在退出锁屏时,没有及时更新 `localStorage` 缓存的问题
- 修复 `setupDayjs` 初始化不准确的问题
## 5.0.5
## Feats

View File

@ -13,6 +13,7 @@ import { mount } from '@vue/test-utils'
*
* const text = wrapper.find('div').text() // hello
*/
// eslint-disable-next-line @typescript-eslint/no-unsafe-function-type
const createRefElement = (slots?: Record<string, Function>) => {
const wrapper = mount(
defineComponent({

View File

@ -1,7 +1,7 @@
{
"name": "ray-template",
"private": false,
"version": "5.0.5",
"version": "5.0.6",
"type": "module",
"engines": {
"node": "^18.0.0 || >=20.0.0",
@ -35,12 +35,12 @@
"dependencies": {
"@logicflow/core": "2.0.6",
"@logicflow/extension": "2.0.10",
"@vueuse/core": "^11.1.0",
"@vueuse/core": "^11.2.0",
"axios": "^1.7.5",
"clipboard": "^2.0.11",
"currency.js": "^2.0.4",
"dayjs": "^1.11.10",
"echarts": "^5.5.0",
"echarts": "^5.5.1",
"html-to-image": "1.11.11",
"interactjs": "1.10.26",
"jsbarcode": "3.11.6",
@ -50,7 +50,7 @@
"pinia": "^2.2.4",
"pinia-plugin-persistedstate": "^4.1.1",
"print-js": "^1.6.0",
"vue": "^3.5.12",
"vue": "^3.5.13",
"vue-demi": "0.14.6",
"vue-hooks-plus": "2.2.1",
"vue-i18n": "^9.13.1",

319
pnpm-lock.yaml generated
View File

@ -15,8 +15,8 @@ importers:
specifier: 2.0.10
version: 2.0.10(@logicflow/core@2.0.6)
'@vueuse/core':
specifier: ^11.1.0
version: 11.1.0(vue@3.5.12(typescript@5.6.3))
specifier: ^11.2.0
version: 11.2.0(vue@3.5.13(typescript@5.6.3))
axios:
specifier: ^1.7.5
version: 1.7.5
@ -30,8 +30,8 @@ importers:
specifier: ^1.11.10
version: 1.11.10
echarts:
specifier: ^5.5.0
version: 5.5.0
specifier: ^5.5.1
version: 5.5.1
html-to-image:
specifier: 1.11.11
version: 1.11.11
@ -49,31 +49,31 @@ importers:
version: 1.1.0
naive-ui:
specifier: ^2.40.1
version: 2.40.1(vue@3.5.12(typescript@5.6.3))
version: 2.40.1(vue@3.5.13(typescript@5.6.3))
pinia:
specifier: ^2.2.4
version: 2.2.4(typescript@5.6.3)(vue@3.5.12(typescript@5.6.3))
version: 2.2.4(typescript@5.6.3)(vue@3.5.13(typescript@5.6.3))
pinia-plugin-persistedstate:
specifier: ^4.1.1
version: 4.1.1(pinia@2.2.4(typescript@5.6.3)(vue@3.5.12(typescript@5.6.3)))(rollup@4.20.0)(webpack-sources@3.2.3)
version: 4.1.1(pinia@2.2.4(typescript@5.6.3)(vue@3.5.13(typescript@5.6.3)))(rollup@4.20.0)(webpack-sources@3.2.3)
print-js:
specifier: ^1.6.0
version: 1.6.0
vue:
specifier: ^3.5.12
version: 3.5.12(typescript@5.6.3)
specifier: ^3.5.13
version: 3.5.13(typescript@5.6.3)
vue-demi:
specifier: 0.14.6
version: 0.14.6(vue@3.5.12(typescript@5.6.3))
version: 0.14.6(vue@3.5.13(typescript@5.6.3))
vue-hooks-plus:
specifier: 2.2.1
version: 2.2.1(vue@3.5.12(typescript@5.6.3))
version: 2.2.1(vue@3.5.13(typescript@5.6.3))
vue-i18n:
specifier: ^9.13.1
version: 9.13.1(vue@3.5.12(typescript@5.6.3))
version: 9.13.1(vue@3.5.13(typescript@5.6.3))
vue-router:
specifier: ^4.3.2
version: 4.3.2(vue@3.5.12(typescript@5.6.3))
version: 4.3.2(vue@3.5.13(typescript@5.6.3))
vue3-next-qrcode:
specifier: 2.0.10
version: 2.0.10(typescript@5.6.3)
@ -89,7 +89,7 @@ importers:
version: 1.10.21
'@intlify/unplugin-vue-i18n':
specifier: ^4.0.0
version: 4.0.0(rollup@4.20.0)(vue-i18n@9.13.1(vue@3.5.12(typescript@5.6.3)))
version: 4.0.0(rollup@4.20.0)(vue-i18n@9.13.1(vue@3.5.13(typescript@5.6.3)))
'@types/crypto-js':
specifier: ^4.2.2
version: 4.2.2
@ -110,10 +110,10 @@ importers:
version: 8.13.0(eslint@8.57.0)(typescript@5.6.3)
'@vitejs/plugin-vue':
specifier: ^5.1.0
version: 5.1.0(vite@5.4.3(@types/node@20.5.1)(sass@1.71.1))(vue@3.5.12(typescript@5.6.3))
version: 5.1.0(vite@5.4.3(@types/node@20.5.1)(sass@1.71.1))(vue@3.5.13(typescript@5.6.3))
'@vitejs/plugin-vue-jsx':
specifier: ^4.0.1
version: 4.0.1(vite@5.4.3(@types/node@20.5.1)(sass@1.71.1))(vue@3.5.12(typescript@5.6.3))
version: 4.0.1(vite@5.4.3(@types/node@20.5.1)(sass@1.71.1))(vue@3.5.13(typescript@5.6.3))
'@vitest/ui':
specifier: 1.4.0
version: 1.4.0(vitest@1.5.2)
@ -125,7 +125,7 @@ importers:
version: 12.0.0(eslint-plugin-vue@9.25.0(eslint@8.57.0))(eslint@8.57.0)(typescript@5.6.3)
'@vue/test-utils':
specifier: 2.4.3
version: 2.4.3(@vue/server-renderer@3.5.12(vue@3.5.12(typescript@5.6.3)))(vue@3.5.12(typescript@5.6.3))
version: 2.4.3(@vue/server-renderer@3.5.13(vue@3.5.13(typescript@5.6.3)))(vue@3.5.13(typescript@5.6.3))
autoprefixer:
specifier: ^10.4.16
version: 10.4.16(postcss@8.4.38)
@ -182,10 +182,10 @@ importers:
version: 5.6.3
unplugin-auto-import:
specifier: ^0.18.2
version: 0.18.2(@nuxt/kit@3.13.2(rollup@4.20.0)(webpack-sources@3.2.3))(@vueuse/core@11.1.0(vue@3.5.12(typescript@5.6.3)))(rollup@4.20.0)(webpack-sources@3.2.3)
version: 0.18.2(@nuxt/kit@3.13.2(rollup@4.20.0)(webpack-sources@3.2.3))(@vueuse/core@11.2.0(vue@3.5.13(typescript@5.6.3)))(rollup@4.20.0)(webpack-sources@3.2.3)
unplugin-vue-components:
specifier: ^0.27.4
version: 0.27.4(@babel/parser@7.25.8)(@nuxt/kit@3.13.2(rollup@4.20.0)(webpack-sources@3.2.3))(rollup@4.20.0)(vue@3.5.12(typescript@5.6.3))(webpack-sources@3.2.3)
version: 0.27.4(@babel/parser@7.26.2)(@nuxt/kit@3.13.2(rollup@4.20.0)(webpack-sources@3.2.3))(rollup@4.20.0)(vue@3.5.13(typescript@5.6.3))(webpack-sources@3.2.3)
vite:
specifier: ^5.4.3
version: 5.4.3(@types/node@20.5.1)(sass@1.71.1)
@ -375,6 +375,11 @@ packages:
engines: {node: '>=6.0.0'}
hasBin: true
'@babel/parser@7.26.2':
resolution: {integrity: sha512-DWMCZH9WA4Maitz2q21SRKHo9QXZxkDsbNZoVD62gusNtNBBqDg9i7uOhASfTfIGNzW+O+r7+jAlM8dwphcJKQ==}
engines: {node: '>=6.0.0'}
hasBin: true
'@babel/plugin-syntax-jsx@7.24.1':
resolution: {integrity: sha512-2eCtxZXf+kbkMIsXS4poTvT4Yu5rXiRa+9xGVT56raghjmBTKMpFNc9R4IDiB4emao9eO22Ox7CxuJG7BgExqA==}
engines: {node: '>=6.9.0'}
@ -1317,6 +1322,9 @@ packages:
'@vue/compiler-core@3.5.12':
resolution: {integrity: sha512-ISyBTRMmMYagUxhcpyEH0hpXRd/KqDU4ymofPgl2XAkY9ZhQ+h0ovEZJIiPop13UmR/54oA2cgMDjgroRelaEw==}
'@vue/compiler-core@3.5.13':
resolution: {integrity: sha512-oOdAkwqUfW1WqpwSYJce06wvt6HljgY3fGeM9NcVA1HaYOij3mZG9Rkysn0OHuyUAGMbEbARIpsG+LPVlBJ5/Q==}
'@vue/compiler-dom@3.4.21':
resolution: {integrity: sha512-IZC6FKowtT1sl0CR5DpXSiEB5ayw75oT2bma1BEhV7RRR1+cfwLrxc2Z8Zq/RGFzJ8w5r9QtCOvTjQgdn0IKmA==}
@ -1326,6 +1334,9 @@ packages:
'@vue/compiler-dom@3.5.12':
resolution: {integrity: sha512-9G6PbJ03uwxLHKQ3P42cMTi85lDRvGLB2rSGOiQqtXELat6uI4n8cNz9yjfVHRPIu+MsK6TE418Giruvgptckg==}
'@vue/compiler-dom@3.5.13':
resolution: {integrity: sha512-ZOJ46sMOKUjO3e94wPdCzQ6P1Lx/vhp2RSvfaab88Ajexs0AHeV0uasYhi99WPaogmBlRHNRuly8xV75cNTMDA==}
'@vue/compiler-sfc@3.4.21':
resolution: {integrity: sha512-me7epoTxYlY+2CUM7hy9PCDdpMPfIwrOvAXud2Upk10g4YLv9UBW7kL798TvMeDhPthkZ0CONNrK2GoeI1ODiQ==}
@ -1335,6 +1346,9 @@ packages:
'@vue/compiler-sfc@3.5.12':
resolution: {integrity: sha512-2k973OGo2JuAa5+ZlekuQJtitI5CgLMOwgl94BzMCsKZCX/xiqzJYzapl4opFogKHqwJk34vfsaKpfEhd1k5nw==}
'@vue/compiler-sfc@3.5.13':
resolution: {integrity: sha512-6VdaljMpD82w6c2749Zhf5T9u5uLBWKnVue6XWxprDobftnletJ8+oel7sexFfM3qIxNmVE7LSFGTpv6obNyaQ==}
'@vue/compiler-ssr@3.4.21':
resolution: {integrity: sha512-M5+9nI2lPpAsgXOGQobnIueVqc9sisBFexh5yMIMRAPYLa7+5wEJs8iqOZc1WAa9WQbx9GR2twgznU8LTIiZ4Q==}
@ -1344,6 +1358,9 @@ packages:
'@vue/compiler-ssr@3.5.12':
resolution: {integrity: sha512-eLwc7v6bfGBSM7wZOGPmRavSWzNFF6+PdRhE+VFJhNCgHiF8AM7ccoqcv5kBXA2eWUfigD7byekvf/JsOfKvPA==}
'@vue/compiler-ssr@3.5.13':
resolution: {integrity: sha512-wMH6vrYHxQl/IybKJagqbquvxpWCuVYpoUJfCqFZwa/JY1GdATAQ+TgVtgrwwMZ0D07QhA99rs/EAAWfvG6KpA==}
'@vue/compiler-vue2@2.7.16':
resolution: {integrity: sha512-qYC3Psj9S/mfu9uVi5WvNZIzq+xnXMhOwbTFKKDD7b1lhpnn71jXSFdTQ+WsIEk0ONCd7VV2IMm7ONl6tbQ86A==}
@ -1378,19 +1395,19 @@ packages:
typescript:
optional: true
'@vue/reactivity@3.5.12':
resolution: {integrity: sha512-UzaN3Da7xnJXdz4Okb/BGbAaomRHc3RdoWqTzlvd9+WBR5m3J39J1fGcHes7U3za0ruYn/iYy/a1euhMEHvTAg==}
'@vue/reactivity@3.5.13':
resolution: {integrity: sha512-NaCwtw8o48B9I6L1zl2p41OHo/2Z4wqYGGIK1Khu5T7yxrn+ATOixn/Udn2m+6kZKB/J7cuT9DbWWhRxqixACg==}
'@vue/runtime-core@3.5.12':
resolution: {integrity: sha512-hrMUYV6tpocr3TL3Ad8DqxOdpDe4zuQY4HPY3X/VRh+L2myQO8MFXPAMarIOSGNu0bFAjh1yBkMPXZBqCk62Uw==}
'@vue/runtime-core@3.5.13':
resolution: {integrity: sha512-Fj4YRQ3Az0WTZw1sFe+QDb0aXCerigEpw418pw1HBUKFtnQHWzwojaukAs2X/c9DQz4MQ4bsXTGlcpGxU/RCIw==}
'@vue/runtime-dom@3.5.12':
resolution: {integrity: sha512-q8VFxR9A2MRfBr6/55Q3umyoN7ya836FzRXajPB6/Vvuv0zOPL+qltd9rIMzG/DbRLAIlREmnLsplEF/kotXKA==}
'@vue/runtime-dom@3.5.13':
resolution: {integrity: sha512-dLaj94s93NYLqjLiyFzVs9X6dWhTdAlEAciC3Moq7gzAc13VJUdCnjjRurNM6uTLFATRHexHCTu/Xp3eW6yoog==}
'@vue/server-renderer@3.5.12':
resolution: {integrity: sha512-I3QoeDDeEPZm8yR28JtY+rk880Oqmj43hreIBVTicisFTx/Dl7JpG72g/X7YF8hnQD3IFhkky5i2bPonwrTVPg==}
'@vue/server-renderer@3.5.13':
resolution: {integrity: sha512-wAi4IRJV/2SAW3htkTlB+dHeRmpTiVIK1OGLWV1yeStVSebSQQOwGwIq0D3ZIoBj2C2qpgz5+vX9iEBkTdk5YA==}
peerDependencies:
vue: 3.5.12
vue: 3.5.13
'@vue/shared@3.4.21':
resolution: {integrity: sha512-PuJe7vDIi6VYSinuEbUIQgMIRZGgM8e4R+G+/dQTk0X1NEdvgvvgv7m+rfmDH1gZzyA1OjjoWskvHlfRNfQf3g==}
@ -1401,6 +1418,9 @@ packages:
'@vue/shared@3.5.12':
resolution: {integrity: sha512-L2RPSAwUFbgZH20etwrXyVyCBu9OxRSi8T/38QsvnkJyvq2LufW2lDCOzm7t/U9C1mkhJGWYfCuFBCmIuNivrg==}
'@vue/shared@3.5.13':
resolution: {integrity: sha512-/hnE/qP5ZoGpol0a5mDi45bOd7t3tjYJBjsgCsivow7D48cJeV5l05RD82lPqi7gRiphZM37rnhW1l6ZoCNNnQ==}
'@vue/test-utils@2.4.3':
resolution: {integrity: sha512-F4K7mF+ad++VlTrxMJVRnenKSJmO6fkQt2wpRDiKDesQMkfpniGWsqEi/JevxGBo2qEkwwjvTUAoiGJLNx++CA==}
peerDependencies:
@ -1410,14 +1430,14 @@ packages:
'@vue/server-renderer':
optional: true
'@vueuse/core@11.1.0':
resolution: {integrity: sha512-P6dk79QYA6sKQnghrUz/1tHi0n9mrb/iO1WTMk/ElLmTyNqgDeSZ3wcDf6fRBGzRJbeG1dxzEOvLENMjr+E3fg==}
'@vueuse/core@11.2.0':
resolution: {integrity: sha512-JIUwRcOqOWzcdu1dGlfW04kaJhW3EXnnjJJfLTtddJanymTL7lF1C0+dVVZ/siLfc73mWn+cGP1PE1PKPruRSA==}
'@vueuse/metadata@11.1.0':
resolution: {integrity: sha512-l9Q502TBTaPYGanl1G+hPgd3QX5s4CGnpXriVBR5fEZ/goI6fvDaVmIl3Td8oKFurOxTmbXvBPSsgrd6eu6HYg==}
'@vueuse/metadata@11.2.0':
resolution: {integrity: sha512-L0ZmtRmNx+ZW95DmrgD6vn484gSpVeRbgpWevFKXwqqQxW9hnSi2Ppuh2BzMjnbv4aJRiIw8tQatXT9uOB23dQ==}
'@vueuse/shared@11.1.0':
resolution: {integrity: sha512-YUtIpY122q7osj+zsNMFAfMTubGz0sn5QzE5gPzAIiCmtt2ha3uQUY1+JPyL4gRCTsLPX82Y9brNbo/aqlA91w==}
'@vueuse/shared@11.2.0':
resolution: {integrity: sha512-VxFjie0EanOudYSgMErxXfq6fo8vhr5ICI+BuE3I9FnX7ePllEsVrRQ7O6Q1TLgApeLuPKcHQxAXpP+KnlrJsg==}
'@xn-sakina/rml-darwin-arm64@2.3.0':
resolution: {integrity: sha512-3CxaA3NRBo6pd9i6Ih5FL+3qmCrYt4nlc1dAw+VhvyUImkSt1tt9WVvm955i2YJVEjQydgsE+U1xhxKJnFa8Hg==}
@ -2227,8 +2247,8 @@ packages:
eastasianwidth@0.2.0:
resolution: {integrity: sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==}
echarts@5.5.0:
resolution: {integrity: sha512-rNYnNCzqDAPCr4m/fqyUFv7fD9qIsd50S6GDFgO1DxZhncCsNsG7IfUlAlvZe5oSEQxtsjnHiUuppzccry93Xw==}
echarts@5.5.1:
resolution: {integrity: sha512-Fce8upazaAXUVUVsjgV6mBnGuqgO+JNDlcgF79Dksy4+wgGpQB2lmYoO4TSweFg/mZITdpGHomw/cNBJZj1icA==}
editorconfig@1.0.4:
resolution: {integrity: sha512-L9Qe08KWTlqYMVvMcTIvMAdl1cDUubzRNYL+WfA4bLDMHe4nemKkpmYzkznE1FwLKu0EEmy6obgQKzMJrg4x9Q==}
@ -3333,6 +3353,9 @@ packages:
magic-string@0.30.11:
resolution: {integrity: sha512-+Wri9p0QHMy+545hKww7YAu5NyzF8iomPL/RQazugQ9+Ez4Ic3mERMd8ZTX5rfK944j+560ZJi8iAwgak1Ac7A==}
magic-string@0.30.12:
resolution: {integrity: sha512-Ea8I3sQMVXr8JhN4z+H/d8zwo+tYDgHE9+5G4Wnrwhs0gaK9fXTKx0Tw5Xwsd/bCPTTZNRAdpyzvoeORe9LYpw==}
magic-string@0.30.8:
resolution: {integrity: sha512-ISQTe55T2ao7XtlAStud6qwYPZjE4GK1S/BeVPus4jrq6JuOnQ00YKQC581RWhR122W7msZV263KzVeLoqidyQ==}
engines: {node: '>=12'}
@ -3594,8 +3617,8 @@ packages:
object-inspect@1.13.1:
resolution: {integrity: sha512-5qoj1RUiKOMsCCNLV1CBiPYE10sziTsnmNxkAI/rZhiD63CF7IqdFGC/XzjWjpSgLf0LxXX3bDFIh0E18f6UhQ==}
object-inspect@1.13.2:
resolution: {integrity: sha512-IRZSRuzJiynemAXPYtPe5BoI/RESNYR7TYm50MC5Mqbd3Jmw5y790sErYw3V6SryFJD64b74qQQs9wn5Bg/k3g==}
object-inspect@1.13.3:
resolution: {integrity: sha512-kDCGIbxkDSXE3euJZZXzc6to7fCrKHNI/hSRQnRuQ+BWjFNzZwiFF8fj/6o2t2G9/jTj8PSIYTfCLelLZEeRpA==}
engines: {node: '>= 0.4'}
object-keys@1.1.1:
@ -3826,6 +3849,10 @@ packages:
resolution: {integrity: sha512-56rxCq7G/XfB4EkXq9Egn5GCqugWvDFjafDOThIdMBsI15iqPqR5r15TfSr1YPYeEI19YeaXMCbY6u88Y76GLQ==}
engines: {node: ^10 || ^12 || >=14}
postcss@8.4.49:
resolution: {integrity: sha512-OCVPnIObs4N29kxTjzLfUryOkvZEq+pf8jTF0lg8E7uETuWHA+v7j3c/xJmiqpX450191LlmZfUKkXxkTry7nA==}
engines: {node: ^10 || ^12 || >=14}
posthtml-parser@0.2.1:
resolution: {integrity: sha512-nPC53YMqJnc/+1x4fRYFfm81KV2V+G9NZY+hTohpYg64Ay7NemWWcV4UWuy/SgMupqQ3kJ88M/iRfZmSnxT+pw==}
@ -4850,8 +4877,8 @@ packages:
vue3-next-qrcode@2.0.10:
resolution: {integrity: sha512-YwdQ1YjbHzcUsA7Vj5ziQLoX7oQZGWDZNaAadLu7EZzV9UU8Dd2IXAxGobF7dIHamgz5mPiOgH1Xb3qO8bCGDA==}
vue@3.5.12:
resolution: {integrity: sha512-CLVZtXtn2ItBIi/zHZ0Sg1Xkb7+PU32bJJ8Bmy7ts3jxXTcbfsEfBivFYYWz1Hur+lalqGAh65Coin0r+HRUfg==}
vue@3.5.13:
resolution: {integrity: sha512-wmeiSMxkZCSc+PM2w2VRsOYAZC8GdipNFRTsLSfodVqI9mbejKeXEGr8SckuLnrQPGe3oJN5c3K0vpoU9q/wCQ==}
peerDependencies:
typescript: '*'
peerDependenciesMeta:
@ -4988,8 +5015,8 @@ packages:
resolution: {integrity: sha512-9bnSc/HEW2uRy67wc+T8UwauLuPJVn28jb+GtJY16iiKWyvmYJRXVT4UamsAEGQfPohgr2q4Tq0sQbQlxTfi1g==}
engines: {node: '>=12.20'}
zrender@5.5.0:
resolution: {integrity: sha512-O3MilSi/9mwoovx77m6ROZM7sXShR/O/JIanvzTwjN3FORfLSr81PsUGd7jlaYOeds9d8tw82oP44+3YucVo+w==}
zrender@5.6.0:
resolution: {integrity: sha512-uzgraf4njmmHAbEUxMJ8Oxg+P3fT04O+9p7gY+wJRVxo8Ge+KmYv0WJev945EH4wFuc4OY2NLXz46FZrWS9xJg==}
snapshots:
@ -5207,6 +5234,10 @@ snapshots:
dependencies:
'@babel/types': 7.26.0
'@babel/parser@7.26.2':
dependencies:
'@babel/types': 7.26.0
'@babel/plugin-syntax-jsx@7.24.1(@babel/core@7.24.7)':
dependencies:
'@babel/core': 7.24.7
@ -5400,9 +5431,9 @@ snapshots:
dependencies:
css-render: 0.15.14
'@css-render/vue3-ssr@0.15.14(vue@3.5.12(typescript@5.6.3))':
'@css-render/vue3-ssr@0.15.14(vue@3.5.13(typescript@5.6.3))':
dependencies:
vue: 3.5.12(typescript@5.6.3)
vue: 3.5.13(typescript@5.6.3)
'@emotion/hash@0.8.0': {}
@ -5585,7 +5616,7 @@ snapshots:
'@interactjs/types@1.10.26': {}
'@intlify/bundle-utils@8.0.0(vue-i18n@9.13.1(vue@3.5.12(typescript@5.6.3)))':
'@intlify/bundle-utils@8.0.0(vue-i18n@9.13.1(vue@3.5.13(typescript@5.6.3)))':
dependencies:
'@intlify/message-compiler': 9.13.1
'@intlify/shared': 9.13.1
@ -5597,7 +5628,7 @@ snapshots:
source-map-js: 1.2.0
yaml-eslint-parser: 1.2.2
optionalDependencies:
vue-i18n: 9.13.1(vue@3.5.12(typescript@5.6.3))
vue-i18n: 9.13.1(vue@3.5.13(typescript@5.6.3))
'@intlify/core-base@9.13.1':
dependencies:
@ -5611,9 +5642,9 @@ snapshots:
'@intlify/shared@9.13.1': {}
'@intlify/unplugin-vue-i18n@4.0.0(rollup@4.20.0)(vue-i18n@9.13.1(vue@3.5.12(typescript@5.6.3)))':
'@intlify/unplugin-vue-i18n@4.0.0(rollup@4.20.0)(vue-i18n@9.13.1(vue@3.5.13(typescript@5.6.3)))':
dependencies:
'@intlify/bundle-utils': 8.0.0(vue-i18n@9.13.1(vue@3.5.12(typescript@5.6.3)))
'@intlify/bundle-utils': 8.0.0(vue-i18n@9.13.1(vue@3.5.13(typescript@5.6.3)))
'@intlify/shared': 9.13.1
'@rollup/pluginutils': 5.1.0(rollup@4.20.0)
'@vue/compiler-sfc': 3.4.27
@ -5626,7 +5657,7 @@ snapshots:
source-map-js: 1.2.0
unplugin: 1.10.1
optionalDependencies:
vue-i18n: 9.13.1(vue@3.5.12(typescript@5.6.3))
vue-i18n: 9.13.1(vue@3.5.13(typescript@5.6.3))
transitivePeerDependencies:
- rollup
- supports-color
@ -6072,20 +6103,20 @@ snapshots:
'@ungap/structured-clone@1.2.0': {}
'@vitejs/plugin-vue-jsx@4.0.1(vite@5.4.3(@types/node@20.5.1)(sass@1.71.1))(vue@3.5.12(typescript@5.6.3))':
'@vitejs/plugin-vue-jsx@4.0.1(vite@5.4.3(@types/node@20.5.1)(sass@1.71.1))(vue@3.5.13(typescript@5.6.3))':
dependencies:
'@babel/core': 7.24.7
'@babel/plugin-transform-typescript': 7.24.8(@babel/core@7.24.7)
'@vue/babel-plugin-jsx': 1.2.2(@babel/core@7.24.7)
vite: 5.4.3(@types/node@20.5.1)(sass@1.71.1)
vue: 3.5.12(typescript@5.6.3)
vue: 3.5.13(typescript@5.6.3)
transitivePeerDependencies:
- supports-color
'@vitejs/plugin-vue@5.1.0(vite@5.4.3(@types/node@20.5.1)(sass@1.71.1))(vue@3.5.12(typescript@5.6.3))':
'@vitejs/plugin-vue@5.1.0(vite@5.4.3(@types/node@20.5.1)(sass@1.71.1))(vue@3.5.13(typescript@5.6.3))':
dependencies:
vite: 5.4.3(@types/node@20.5.1)(sass@1.71.1)
vue: 3.5.12(typescript@5.6.3)
vue: 3.5.13(typescript@5.6.3)
'@vitest/expect@1.5.2':
dependencies:
@ -6199,6 +6230,14 @@ snapshots:
estree-walker: 2.0.2
source-map-js: 1.2.1
'@vue/compiler-core@3.5.13':
dependencies:
'@babel/parser': 7.26.2
'@vue/shared': 3.5.13
entities: 4.5.0
estree-walker: 2.0.2
source-map-js: 1.2.1
'@vue/compiler-dom@3.4.21':
dependencies:
'@vue/compiler-core': 3.4.21
@ -6214,6 +6253,11 @@ snapshots:
'@vue/compiler-core': 3.5.12
'@vue/shared': 3.5.12
'@vue/compiler-dom@3.5.13':
dependencies:
'@vue/compiler-core': 3.5.13
'@vue/shared': 3.5.13
'@vue/compiler-sfc@3.4.21':
dependencies:
'@babel/parser': 7.24.1
@ -6250,6 +6294,18 @@ snapshots:
postcss: 8.4.47
source-map-js: 1.2.1
'@vue/compiler-sfc@3.5.13':
dependencies:
'@babel/parser': 7.26.2
'@vue/compiler-core': 3.5.13
'@vue/compiler-dom': 3.5.13
'@vue/compiler-ssr': 3.5.13
'@vue/shared': 3.5.13
estree-walker: 2.0.2
magic-string: 0.30.12
postcss: 8.4.49
source-map-js: 1.2.1
'@vue/compiler-ssr@3.4.21':
dependencies:
'@vue/compiler-dom': 3.4.21
@ -6265,6 +6321,11 @@ snapshots:
'@vue/compiler-dom': 3.5.12
'@vue/shared': 3.5.12
'@vue/compiler-ssr@3.5.13':
dependencies:
'@vue/compiler-dom': 3.5.13
'@vue/shared': 3.5.13
'@vue/compiler-vue2@2.7.16':
dependencies:
de-indent: 1.0.2
@ -6308,27 +6369,27 @@ snapshots:
optionalDependencies:
typescript: 5.6.3
'@vue/reactivity@3.5.12':
'@vue/reactivity@3.5.13':
dependencies:
'@vue/shared': 3.5.12
'@vue/shared': 3.5.13
'@vue/runtime-core@3.5.12':
'@vue/runtime-core@3.5.13':
dependencies:
'@vue/reactivity': 3.5.12
'@vue/shared': 3.5.12
'@vue/reactivity': 3.5.13
'@vue/shared': 3.5.13
'@vue/runtime-dom@3.5.12':
'@vue/runtime-dom@3.5.13':
dependencies:
'@vue/reactivity': 3.5.12
'@vue/runtime-core': 3.5.12
'@vue/shared': 3.5.12
'@vue/reactivity': 3.5.13
'@vue/runtime-core': 3.5.13
'@vue/shared': 3.5.13
csstype: 3.1.3
'@vue/server-renderer@3.5.12(vue@3.5.12(typescript@5.6.3))':
'@vue/server-renderer@3.5.13(vue@3.5.13(typescript@5.6.3))':
dependencies:
'@vue/compiler-ssr': 3.5.12
'@vue/shared': 3.5.12
vue: 3.5.12(typescript@5.6.3)
'@vue/compiler-ssr': 3.5.13
'@vue/shared': 3.5.13
vue: 3.5.13(typescript@5.6.3)
'@vue/shared@3.4.21': {}
@ -6336,29 +6397,31 @@ snapshots:
'@vue/shared@3.5.12': {}
'@vue/test-utils@2.4.3(@vue/server-renderer@3.5.12(vue@3.5.12(typescript@5.6.3)))(vue@3.5.12(typescript@5.6.3))':
'@vue/shared@3.5.13': {}
'@vue/test-utils@2.4.3(@vue/server-renderer@3.5.13(vue@3.5.13(typescript@5.6.3)))(vue@3.5.13(typescript@5.6.3))':
dependencies:
js-beautify: 1.14.11
vue: 3.5.12(typescript@5.6.3)
vue: 3.5.13(typescript@5.6.3)
vue-component-type-helpers: 1.8.26
optionalDependencies:
'@vue/server-renderer': 3.5.12(vue@3.5.12(typescript@5.6.3))
'@vue/server-renderer': 3.5.13(vue@3.5.13(typescript@5.6.3))
'@vueuse/core@11.1.0(vue@3.5.12(typescript@5.6.3))':
'@vueuse/core@11.2.0(vue@3.5.13(typescript@5.6.3))':
dependencies:
'@types/web-bluetooth': 0.0.20
'@vueuse/metadata': 11.1.0
'@vueuse/shared': 11.1.0(vue@3.5.12(typescript@5.6.3))
vue-demi: 0.14.10(vue@3.5.12(typescript@5.6.3))
'@vueuse/metadata': 11.2.0
'@vueuse/shared': 11.2.0(vue@3.5.13(typescript@5.6.3))
vue-demi: 0.14.10(vue@3.5.13(typescript@5.6.3))
transitivePeerDependencies:
- '@vue/composition-api'
- vue
'@vueuse/metadata@11.1.0': {}
'@vueuse/metadata@11.2.0': {}
'@vueuse/shared@11.1.0(vue@3.5.12(typescript@5.6.3))':
'@vueuse/shared@11.2.0(vue@3.5.13(typescript@5.6.3))':
dependencies:
vue-demi: 0.14.10(vue@3.5.12(typescript@5.6.3))
vue-demi: 0.14.10(vue@3.5.13(typescript@5.6.3))
transitivePeerDependencies:
- '@vue/composition-api'
- vue
@ -7183,10 +7246,10 @@ snapshots:
eastasianwidth@0.2.0: {}
echarts@5.5.0:
echarts@5.5.1:
dependencies:
tslib: 2.3.0
zrender: 5.5.0
zrender: 5.6.0
editorconfig@1.0.4:
dependencies:
@ -7256,7 +7319,7 @@ snapshots:
is-string: 1.0.7
is-typed-array: 1.1.13
is-weakref: 1.0.2
object-inspect: 1.13.2
object-inspect: 1.13.3
object-keys: 1.1.1
object.assign: 4.1.5
regexp.prototype.flags: 1.5.2
@ -8421,6 +8484,10 @@ snapshots:
dependencies:
'@jridgewell/sourcemap-codec': 1.5.0
magic-string@0.30.12:
dependencies:
'@jridgewell/sourcemap-codec': 1.5.0
magic-string@0.30.8:
dependencies:
'@jridgewell/sourcemap-codec': 1.5.0
@ -8610,10 +8677,10 @@ snapshots:
arrify: 2.0.1
minimatch: 3.1.2
naive-ui@2.40.1(vue@3.5.12(typescript@5.6.3)):
naive-ui@2.40.1(vue@3.5.13(typescript@5.6.3)):
dependencies:
'@css-render/plugin-bem': 0.15.14(css-render@0.15.14)
'@css-render/vue3-ssr': 0.15.14(vue@3.5.12(typescript@5.6.3))
'@css-render/vue3-ssr': 0.15.14(vue@3.5.13(typescript@5.6.3))
'@types/katex': 0.16.7
'@types/lodash': 4.17.6
'@types/lodash-es': 4.17.12
@ -8628,10 +8695,10 @@ snapshots:
lodash-es: 4.17.21
seemly: 0.3.8
treemate: 0.3.11
vdirs: 0.1.8(vue@3.5.12(typescript@5.6.3))
vooks: 0.2.12(vue@3.5.12(typescript@5.6.3))
vue: 3.5.12(typescript@5.6.3)
vueuc: 0.4.64(vue@3.5.12(typescript@5.6.3))
vdirs: 0.1.8(vue@3.5.13(typescript@5.6.3))
vooks: 0.2.12(vue@3.5.13(typescript@5.6.3))
vue: 3.5.13(typescript@5.6.3)
vueuc: 0.4.64(vue@3.5.13(typescript@5.6.3))
nanoid@3.3.7: {}
@ -8710,7 +8777,7 @@ snapshots:
object-inspect@1.13.1: {}
object-inspect@1.13.2: {}
object-inspect@1.13.3: {}
object-keys@1.1.1: {}
@ -8858,25 +8925,25 @@ snapshots:
pidtree@0.6.0: {}
pinia-plugin-persistedstate@4.1.1(pinia@2.2.4(typescript@5.6.3)(vue@3.5.12(typescript@5.6.3)))(rollup@4.20.0)(webpack-sources@3.2.3):
pinia-plugin-persistedstate@4.1.1(pinia@2.2.4(typescript@5.6.3)(vue@3.5.13(typescript@5.6.3)))(rollup@4.20.0)(webpack-sources@3.2.3):
dependencies:
'@nuxt/kit': 3.13.2(rollup@4.20.0)(webpack-sources@3.2.3)
deep-pick-omit: 1.2.1
defu: 6.1.4
destr: 2.0.3
optionalDependencies:
pinia: 2.2.4(typescript@5.6.3)(vue@3.5.12(typescript@5.6.3))
pinia: 2.2.4(typescript@5.6.3)(vue@3.5.13(typescript@5.6.3))
transitivePeerDependencies:
- magicast
- rollup
- supports-color
- webpack-sources
pinia@2.2.4(typescript@5.6.3)(vue@3.5.12(typescript@5.6.3)):
pinia@2.2.4(typescript@5.6.3)(vue@3.5.13(typescript@5.6.3)):
dependencies:
'@vue/devtools-api': 6.6.3
vue: 3.5.12(typescript@5.6.3)
vue-demi: 0.14.10(vue@3.5.12(typescript@5.6.3))
vue: 3.5.13(typescript@5.6.3)
vue-demi: 0.14.10(vue@3.5.13(typescript@5.6.3))
optionalDependencies:
typescript: 5.6.3
@ -8934,6 +9001,12 @@ snapshots:
picocolors: 1.1.0
source-map-js: 1.2.1
postcss@8.4.49:
dependencies:
nanoid: 3.3.7
picocolors: 1.1.1
source-map-js: 1.2.1
posthtml-parser@0.2.1:
dependencies:
htmlparser2: 3.10.1
@ -9741,7 +9814,7 @@ snapshots:
unpipe@1.0.0: {}
unplugin-auto-import@0.18.2(@nuxt/kit@3.13.2(rollup@4.20.0)(webpack-sources@3.2.3))(@vueuse/core@11.1.0(vue@3.5.12(typescript@5.6.3)))(rollup@4.20.0)(webpack-sources@3.2.3):
unplugin-auto-import@0.18.2(@nuxt/kit@3.13.2(rollup@4.20.0)(webpack-sources@3.2.3))(@vueuse/core@11.2.0(vue@3.5.13(typescript@5.6.3)))(rollup@4.20.0)(webpack-sources@3.2.3):
dependencies:
'@antfu/utils': 0.7.10
'@rollup/pluginutils': 5.1.0(rollup@4.20.0)
@ -9753,12 +9826,12 @@ snapshots:
unplugin: 1.13.1(webpack-sources@3.2.3)
optionalDependencies:
'@nuxt/kit': 3.13.2(rollup@4.20.0)(webpack-sources@3.2.3)
'@vueuse/core': 11.1.0(vue@3.5.12(typescript@5.6.3))
'@vueuse/core': 11.2.0(vue@3.5.13(typescript@5.6.3))
transitivePeerDependencies:
- rollup
- webpack-sources
unplugin-vue-components@0.27.4(@babel/parser@7.25.8)(@nuxt/kit@3.13.2(rollup@4.20.0)(webpack-sources@3.2.3))(rollup@4.20.0)(vue@3.5.12(typescript@5.6.3))(webpack-sources@3.2.3):
unplugin-vue-components@0.27.4(@babel/parser@7.26.2)(@nuxt/kit@3.13.2(rollup@4.20.0)(webpack-sources@3.2.3))(rollup@4.20.0)(vue@3.5.13(typescript@5.6.3))(webpack-sources@3.2.3):
dependencies:
'@antfu/utils': 0.7.10
'@rollup/pluginutils': 5.1.0(rollup@4.20.0)
@ -9770,9 +9843,9 @@ snapshots:
minimatch: 9.0.5
mlly: 1.7.1
unplugin: 1.13.1(webpack-sources@3.2.3)
vue: 3.5.12(typescript@5.6.3)
vue: 3.5.13(typescript@5.6.3)
optionalDependencies:
'@babel/parser': 7.25.8
'@babel/parser': 7.26.2
'@nuxt/kit': 3.13.2(rollup@4.20.0)(webpack-sources@3.2.3)
transitivePeerDependencies:
- rollup
@ -9858,10 +9931,10 @@ snapshots:
vary@1.1.2: {}
vdirs@0.1.8(vue@3.5.12(typescript@5.6.3)):
vdirs@0.1.8(vue@3.5.13(typescript@5.6.3)):
dependencies:
evtd: 0.2.4
vue: 3.5.12(typescript@5.6.3)
vue: 3.5.13(typescript@5.6.3)
vite-bundle-analyzer@0.9.4:
dependencies:
@ -10031,22 +10104,22 @@ snapshots:
- supports-color
- terser
vooks@0.2.12(vue@3.5.12(typescript@5.6.3)):
vooks@0.2.12(vue@3.5.13(typescript@5.6.3)):
dependencies:
evtd: 0.2.4
vue: 3.5.12(typescript@5.6.3)
vue: 3.5.13(typescript@5.6.3)
vscode-uri@3.0.8: {}
vue-component-type-helpers@1.8.26: {}
vue-demi@0.14.10(vue@3.5.12(typescript@5.6.3)):
vue-demi@0.14.10(vue@3.5.13(typescript@5.6.3)):
dependencies:
vue: 3.5.12(typescript@5.6.3)
vue: 3.5.13(typescript@5.6.3)
vue-demi@0.14.6(vue@3.5.12(typescript@5.6.3)):
vue-demi@0.14.6(vue@3.5.13(typescript@5.6.3)):
dependencies:
vue: 3.5.12(typescript@5.6.3)
vue: 3.5.13(typescript@5.6.3)
vue-eslint-parser@9.3.2(eslint@8.57.0):
dependencies:
@ -10074,7 +10147,7 @@ snapshots:
transitivePeerDependencies:
- supports-color
vue-hooks-plus@2.2.1(vue@3.5.12(typescript@5.6.3)):
vue-hooks-plus@2.2.1(vue@3.5.13(typescript@5.6.3)):
dependencies:
'@types/js-cookie': 3.0.6
'@vue/devtools-api': 6.6.3
@ -10083,19 +10156,19 @@ snapshots:
qs: 6.12.1
query-string: 7.1.3
screenfull: 5.2.0
vue: 3.5.12(typescript@5.6.3)
vue: 3.5.13(typescript@5.6.3)
vue-i18n@9.13.1(vue@3.5.12(typescript@5.6.3)):
vue-i18n@9.13.1(vue@3.5.13(typescript@5.6.3)):
dependencies:
'@intlify/core-base': 9.13.1
'@intlify/shared': 9.13.1
'@vue/devtools-api': 6.6.1
vue: 3.5.12(typescript@5.6.3)
vue: 3.5.13(typescript@5.6.3)
vue-router@4.3.2(vue@3.5.12(typescript@5.6.3)):
vue-router@4.3.2(vue@3.5.13(typescript@5.6.3)):
dependencies:
'@vue/devtools-api': 6.6.1
vue: 3.5.12(typescript@5.6.3)
vue: 3.5.13(typescript@5.6.3)
vue-tsc@2.1.10(typescript@5.6.3):
dependencies:
@ -10107,30 +10180,30 @@ snapshots:
vue3-next-qrcode@2.0.10(typescript@5.6.3):
dependencies:
js-binary-schema-parser: 2.0.3
vue: 3.5.12(typescript@5.6.3)
vue: 3.5.13(typescript@5.6.3)
transitivePeerDependencies:
- typescript
vue@3.5.12(typescript@5.6.3):
vue@3.5.13(typescript@5.6.3):
dependencies:
'@vue/compiler-dom': 3.5.12
'@vue/compiler-sfc': 3.5.12
'@vue/runtime-dom': 3.5.12
'@vue/server-renderer': 3.5.12(vue@3.5.12(typescript@5.6.3))
'@vue/shared': 3.5.12
'@vue/compiler-dom': 3.5.13
'@vue/compiler-sfc': 3.5.13
'@vue/runtime-dom': 3.5.13
'@vue/server-renderer': 3.5.13(vue@3.5.13(typescript@5.6.3))
'@vue/shared': 3.5.13
optionalDependencies:
typescript: 5.6.3
vueuc@0.4.64(vue@3.5.12(typescript@5.6.3)):
vueuc@0.4.64(vue@3.5.13(typescript@5.6.3)):
dependencies:
'@css-render/vue3-ssr': 0.15.14(vue@3.5.12(typescript@5.6.3))
'@css-render/vue3-ssr': 0.15.14(vue@3.5.13(typescript@5.6.3))
'@juggle/resize-observer': 3.4.0
css-render: 0.15.14
evtd: 0.2.4
seemly: 0.3.8
vdirs: 0.1.8(vue@3.5.12(typescript@5.6.3))
vooks: 0.2.12(vue@3.5.12(typescript@5.6.3))
vue: 3.5.12(typescript@5.6.3)
vdirs: 0.1.8(vue@3.5.13(typescript@5.6.3))
vooks: 0.2.12(vue@3.5.13(typescript@5.6.3))
vue: 3.5.13(typescript@5.6.3)
webidl-conversions@7.0.0: {}
@ -10243,6 +10316,6 @@ snapshots:
yocto-queue@1.0.0: {}
zrender@5.5.0:
zrender@5.6.0:
dependencies:
tslib: 2.3.0

View File

@ -54,7 +54,14 @@ const AppAvatar = defineComponent({
objectFit="cover"
round
size={avatarSize}
/>
>
{{
default: () =>
getSigningCallback.avatar
? null
: getSigningCallback?.name?.[0],
}}
</NAvatar>
{getSigningCallback?.name}
</NFlex>
</NButton>

View File

@ -1,16 +1,18 @@
import { NInput, NForm, NFormItem, NButton } from 'naive-ui'
import AppAvatar from '@/app-components/app/AppAvatar'
import useAppLockScreen from '@/app-components/app/AppLockScreen/appLockVar'
import { rules, useCondition } from '@/app-components/app/AppLockScreen/shared'
import { useSettingGetters, useSettingActions } from '@/store'
import { useSettingActions } from '@/store'
import { useTemplateRef } from 'vue'
import type { FormInst, InputInst } from 'naive-ui'
const LockScreen = defineComponent({
name: 'LockScreen',
setup() {
const formInstRef = ref<FormInst | null>(null)
const inputInstRef = ref<InputInst | null>(null)
const formInstRef = useTemplateRef<FormInst | null>('formInstRef')
const inputInstRef = useTemplateRef<InputInst | null>('inputInstRef')
const { setLockAppScreen } = useAppLockScreen()
const { updateSettingState } = useSettingActions()
@ -19,7 +21,6 @@ const LockScreen = defineComponent({
lockCondition: useCondition(),
})
/** 锁屏 */
const lockScreen = () => {
formInstRef.value?.validate((error) => {
if (!error) {
@ -48,6 +49,11 @@ const LockScreen = defineComponent({
return (
<div class="app-lock-screen__content">
<div class="app-lock-screen__input">
<AppAvatar
avatarSize={52}
style="pointer-events: none;margin: 24px 0;"
vertical
/>
<NForm
ref="formInstRef"
model={this.lockCondition}

View File

@ -8,14 +8,15 @@ import { useSigningActions, useSettingActions } from '@/store'
import { rules, useCondition } from '@/app-components/app/AppLockScreen/shared'
import useAppLockScreen from '@/app-components/app/AppLockScreen/appLockVar'
import { useDevice } from '@/hooks'
import { useTemplateRef } from 'vue'
import type { FormInst, InputInst } from 'naive-ui'
export default defineComponent({
name: 'UnlockScreen',
setup() {
const formRef = ref<FormInst | null>(null)
const inputInstRef = ref<InputInst | null>(null)
const formRef = useTemplateRef<FormInst | null>('formRef')
const inputInstRef = useTemplateRef<InputInst | null>('inputInstRef')
const { logout } = useSigningActions()
const { updateSettingState } = useSettingActions()
@ -24,13 +25,13 @@ export default defineComponent({
const HH_MM_FORMAT = 'HH:mm'
const AM_PM_FORMAT = 'A'
const YY_MM_DD_FORMAT = 'YY年MM月DD日'
const YY_MM_DD_FORMAT = 'YYYY-MM-DD'
const DDD_FORMAT = 'ddd'
const state = reactive({
lockCondition: useCondition(),
HH_MM: dayjs().format(HH_MM_FORMAT),
AM_PM: dayjs().locale('en').format(AM_PM_FORMAT),
AM_PM: dayjs().format(AM_PM_FORMAT),
YY_MM_DD: dayjs().format(YY_MM_DD_FORMAT),
DDD: dayjs().format(DDD_FORMAT),
})
@ -43,7 +44,6 @@ export default defineComponent({
state.DDD = dayjs().format(DDD_FORMAT)
}, 86_400_000)
/** 退出登陆并且回到登陆页 */
const backToSigning = () => {
window.$dialog.warning({
title: '警告',
@ -51,15 +51,14 @@ export default defineComponent({
positiveText: '确定',
negativeText: '取消',
onPositiveClick: () => {
logout()
setTimeout(() => {
updateSettingState('lockScreenSwitch', false)
})
setTimeout(() => {
logout()
}, 100)
},
})
}
/** 解锁 */
const unlockScreen = () => {
formRef.value?.validate((error) => {
if (!error) {
@ -71,6 +70,11 @@ export default defineComponent({
})
}
onMounted(() => {
nextTick(() => {
inputInstRef.value?.focus()
})
})
onBeforeUnmount(() => {
clearInterval(dayInterval)
clearInterval(yearInterval)
@ -126,7 +130,6 @@ export default defineComponent({
placeholder="请输入解锁密码"
clearable
minlength={6}
maxlength={12}
onKeydown={(e: KeyboardEvent) => {
if (e.code === 'Enter') {
unlockScreen()
@ -153,11 +156,8 @@ export default defineComponent({
</NForm>
</div>
<div class="app-lock-screen__unlock__content-date">
<div class="current-date">
{HH_MM}&nbsp;<span>{AM_PM}</span>
</div>
<div class="current-year">
{YY_MM_DD}&nbsp;<span>{DDD}</span>
{YY_MM_DD}&nbsp;<span>{DDD}</span>&nbsp;<span>{AM_PM}</span>
</div>
</div>
</div>

View File

@ -36,7 +36,7 @@
width: 100%;
height: 100%;
@include flexCenter;
font-size: 320px;
font-size: 16.67rem;
gap: 80px;
z-index: 0;
@ -85,9 +85,23 @@
& .current-year,
& .current-date span {
font-size: 1.5rem;
font-size: 1.875rem;
line-height: 2.25rem;
}
}
}
}
}
.ray-template--light {
.app-lock-screen__unlock__content-bg__wrapper {
background-color: #fff !important;
}
.app-lock-screen__unlock__content-bg {
& .left,
& .right {
background-color: rgba(244, 244, 245, 1) !important;
}
}
}

View File

@ -28,18 +28,10 @@ export default defineComponent({
if (version !== cacheVersion) {
modalShow.value = true
setStorage<string>(
APP_CATCH_KEY.appVersionProvider,
version,
'localStorage',
)
setStorage(APP_CATCH_KEY.appVersionProvider, version, 'localStorage')
}
} else {
setStorage<string>(
APP_CATCH_KEY.appVersionProvider,
version,
'localStorage',
)
setStorage(APP_CATCH_KEY.appVersionProvider, version, 'localStorage')
}
return {

View File

@ -4,7 +4,7 @@ import type { Ref } from 'vue'
/**
*
* @description
* ref
* shallowRef
*
* , 使(scrollTo)
*
@ -16,12 +16,12 @@ import type { Ref } from 'vue'
* })
*/
export const LAYOUT_CONTENT_REF: Readonly<Ref<LayoutInst | null>> =
ref<LayoutInst | null>(null)
shallowRef<LayoutInst | null>(null)
/**
*
* @description
* ref
* shallowRef
*
*
* 使使 nextTick() dom
@ -31,7 +31,7 @@ export const LAYOUT_CONTENT_REF: Readonly<Ref<LayoutInst | null>> =
* })
*/
export const LAYOUT_SIDER_REF: Readonly<Ref<LayoutInst | null>> =
ref<LayoutInst | null>(null)
shallowRef<LayoutInst | null>(null)
export const SETUP_ROUTER_ACTION = {
/** 是否启用路由切换时顶部加载条 */

View File

@ -16,7 +16,7 @@ import type {
* request instance ,
*/
const requestHeaderToken = (ins: RequestInterceptorConfig, mode: string) => {
const token = getStorage<string>(APP_CATCH_KEY.token, 'localStorage')
const token = getStorage<string | null>(APP_CATCH_KEY.token, 'localStorage')
if (ins.url) {
// TODO: 根据 url 不同是否设置 token

View File

@ -40,14 +40,14 @@ function useRequest<
fetchOptions: AppRawRequestConfig<Response>,
option?: UseRequestOptions<Response, HookPlusParams, HookPlusPlugin>,
) {
const fc = () => {
const fn = () => {
const cb = request<Response>(fetchOptions)
return cb
}
const hooks = useHookPlusRequest<Response, HookPlusParams>(
fc,
fn,
Object.assign({}, option),
)

View File

@ -0,0 +1,6 @@
import type { InjectionKey } from 'vue'
import type { ChartProviderOptions } from './hooks/useChartProvider'
export const USE_CHART_PROVIDER_KEY: InjectionKey<
Partial<ChartProviderOptions>
> = Symbol('USE_CHART_PROVIDER_KEY')

View File

@ -0,0 +1,18 @@
import { USE_CHART_PROVIDER_KEY } from '../config'
import type { ChartTheme } from '../types'
export interface ChartProviderOptions {
theme: ChartTheme
}
/**
*
* @param
*
* @description
* chart
*/
export const useChartProvider = (options: Partial<ChartProviderOptions>) => {
provide(USE_CHART_PROVIDER_KEY, options)
}

View File

@ -36,6 +36,7 @@ import {
import { RMoreDropdown } from '@/components'
import { useSettingGetters } from '@/store'
import { useTemplateRef } from 'vue'
import { USE_CHART_PROVIDER_KEY } from './config'
import type { WatchStopHandle } from 'vue'
import type { AnyFC } from '@/types'
@ -124,6 +125,7 @@ export default defineComponent({
const __catch = {
aria: props.showAria,
}
const chartProvideOptions = inject(USE_CHART_PROVIDER_KEY, {})
/**
*
@ -174,10 +176,19 @@ export default defineComponent({
* echartTheme 使
*/
const updateChartTheme = () => {
const { theme: providerTheme } = chartProvideOptions || {}
if (echartInstanceRef.value) {
destroyChart()
}
// 如果配置了全局配置主题,则忽略后面所有逻辑
if (providerTheme) {
renderChart(providerTheme)
return
}
if (props.theme === 'default') {
props.autoChangeTheme ? renderChart('dark') : renderChart('')

View File

@ -40,7 +40,7 @@ const useForm = <T extends Recordable, R extends RFormRules>(
model?: T,
rules?: R,
) => {
const formRef = ref<RFormInst>()
const formRef = shallowRef<RFormInst>()
const register = (inst: RFormInst) => {
if (inst) {

View File

@ -41,7 +41,7 @@ import type { PrintDomOptions } from '@/utils'
* })
*/
const useTable = () => {
const tableRef = ref<RTableInst>()
const tableRef = shallowRef<RTableInst>()
let extra = {} as TableProvider
const register: UseTableRegister = (inst, extraInfo) => {

View File

@ -1,6 +1,10 @@
import dayjs from 'dayjs'
import { DEFAULT_DAYJS_LOCAL } from '@/app-config'
import 'dayjs/locale/zh-cn'
import { getStorage } from '@/utils'
import { APP_CATCH_KEY, DAYJS_LOCAL_MAP } from '@/app-config'
import type { SettingState } from '@/store/modules/setting/types'
/**
*
@ -10,5 +14,16 @@ import 'dayjs/locale/zh-cn'
* dayjs
*/
export const setupDayjs = () => {
dayjs.locale(DEFAULT_DAYJS_LOCAL)
const { localeLanguage } = getStorage<SettingState>(
APP_CATCH_KEY.appPiniaSettingStore,
'localStorage',
{
defaultValue: {} as SettingState,
},
)
const local =
DAYJS_LOCAL_MAP[localeLanguage as keyof typeof DAYJS_LOCAL_MAP] ||
DEFAULT_DAYJS_LOCAL
dayjs.locale(local)
}

View File

@ -8,10 +8,10 @@ export const combineDirective = <
) => {
const directives = Object.keys(directiveModules).reduce(
(pre, curr) => {
const fc = directiveModules[curr]?.default
const fn = directiveModules[curr]?.default
if (typeof fc === 'function') {
pre[curr as K] = fc
if (typeof fn === 'function') {
pre[curr as K] = fn
return pre
} else {

View File

@ -23,10 +23,11 @@ export interface UseContextmenuCoordinateOptions {
*
* @param target
*
* NDropdown 使
* @description
* NDropdown 使
*
* @example
* const target = ref<HTMLElement | null>(null)
* const target = useTemplateRef<HTMLElement | null>('target')
* const { x, y, show, stop } = useContextmenuCoordinate(target)
*
* stop
@ -44,6 +45,13 @@ export const useContextmenuCoordinate = (
const show = ref(false) // 是否显示右键菜单
const { clickOutside } = options ?? {}
/**
*
* @param value
*
* @description
*
*/
const updateShow = (value: boolean) => {
show.value = value
}
@ -52,8 +60,9 @@ export const useContextmenuCoordinate = (
*
* @param evt
*
*
*
* @description
*
*
*/
const bindContextMenuEvent = (evt: Event) => {
evt.preventDefault()
@ -73,7 +82,8 @@ export const useContextmenuCoordinate = (
if (clickOutside) {
/**
*
*
* @description
*
*/
onClickOutside(target as MaybeElementRef<MaybeElement>, (detectIframe) => {
clickOutside(detectIframe)
@ -82,7 +92,8 @@ export const useContextmenuCoordinate = (
/**
*
* ref dom
* @description
* ref dom
*/
const cleanupContextmenu = useEventListener(
target,
@ -90,9 +101,11 @@ export const useContextmenuCoordinate = (
bindContextMenuEvent,
options,
)
/**
*
* ref dom
* @description
* ref dom
*/
const cleanupClick = useEventListener(target, 'click', () => {
updateShow(false)
@ -100,8 +113,9 @@ export const useContextmenuCoordinate = (
/**
*
*
*
* @description
*
*
*/
const stop = () => {
cleanupContextmenu()

View File

@ -10,17 +10,17 @@ export type CloseMenuTag = Key | MenuTagOptions
/**
*
* @param target key
* @param fc
* @param fn
*
*
*/
const normalMenuTagOption = (target: CloseMenuTag, fc: string) => {
const normalMenuTagOption = (target: CloseMenuTag, fn: string) => {
const { getMenuTagOptions } = useMenuGetters()
if (typeof target === 'number') {
// 判断是否为 NaN
if (isNaN(target)) {
console.warn(`${fc}: The ${target} is NaN, expect number.`)
console.warn(`${fn}: The ${target} is NaN, expect number.`)
return
}
@ -28,7 +28,7 @@ const normalMenuTagOption = (target: CloseMenuTag, fc: string) => {
// 判断是否超出当前标签页列表最大长度或者是否为负数
if (target > getMenuTagOptions.value.length || target < -1) {
console.warn(
`${fc}: The incoming index ${target} did not match the corresponding item.`,
`${fn}: The incoming index ${target} did not match the corresponding item.`,
)
return
@ -50,7 +50,7 @@ const normalMenuTagOption = (target: CloseMenuTag, fc: string) => {
index,
}
: console.warn(
`${fc}: The incoming key ${target} did not match the corresponding item.`,
`${fn}: The incoming key ${target} did not match the corresponding item.`,
)
} else {
const { fullPath } = target
@ -60,7 +60,7 @@ const normalMenuTagOption = (target: CloseMenuTag, fc: string) => {
if (index === -1) {
console.warn(
`${fc}: The incoming menuTag option ${target.fullPath} did not match the corresponding item.`,
`${fn}: The incoming menuTag option ${target.fullPath} did not match the corresponding item.`,
)
return

View File

@ -85,7 +85,7 @@ const domToImageMethods = {
/**
*
* @param target ref dom
* @param target useTemplateRef dom
* @param options html-to-image options
*
* @see https://github.com/bubkoo/html-to-image
@ -99,7 +99,7 @@ const domToImageMethods = {
* 使 jpeg
*
* @example
* const refDom = ref<HTMLElement>()
* const refDom = useTemplateRef<HTMLElement>('refDom')
* const { create, stop } = useDomToImage(refDom, {
* beforeCreate: (element) => { ... },
* created: (element, result) => { ... },

View File

@ -82,7 +82,7 @@ const styleElement = document.createElement('style')
* <div ref="refDom" />
* </template>
* <script lang="ts" setup>
* const refDom = ref<HTMLElement>()
* const refDom = useTemplateRef<HTMLElement>('refDom')
* const { enter, exit, toggleFullscreen } = useElementFullscreen(refDom, { UseElementFullscreenOptions })
*
* enter() // 进入全屏

View File

@ -17,7 +17,7 @@ type OmitKeys =
export interface UsePaginationOptions extends Omit<PaginationProps, OmitKeys> {}
const defaultOptions: UsePaginationOptions = {
const DEFAULT_OPTIONS: UsePaginationOptions = {
page: 1,
pageSize: 10,
showSizePicker: true,
@ -36,16 +36,8 @@ export const usePagination = <T extends AnyFC>(
callback?: T,
options?: UsePaginationOptions,
) => {
const callbackRef = ref(callback)
const omitOptions = omit(options, [
'on-update:page',
'on-update:page-size',
'onUpdatePage',
'onUpdatePageSize',
'onUpdate:page',
'onUpdate:page-size',
])
const methodsOptions = {
const callbackRef = shallowRef(callback)
const paginationMethods = {
onUpdatePage: (page: number) => {
paginationRef.value.page = page
@ -53,16 +45,30 @@ export const usePagination = <T extends AnyFC>(
},
onUpdatePageSize: (pageSize: number) => {
paginationRef.value.pageSize = pageSize
paginationRef.value.page = 1
paginationRef.value.page = DEFAULT_OPTIONS.page
callbackRef.value?.()
},
}
const paginationRef = ref<PaginationProps>(
Object.assign({}, defaultOptions, omitOptions, methodsOptions),
)
// 使用 computed 优化配置合并
const mergedOptions = computed(() => ({
...DEFAULT_OPTIONS,
...omit(options, [
'on-update:page',
'on-update:page-size',
'onUpdatePage',
'onUpdatePageSize',
'onUpdate:page',
'onUpdate:page-size',
]),
...paginationMethods,
}))
const paginationRef = ref<PaginationProps>(mergedOptions.value)
// 更新分页页数
const updatePage = paginationRef.value.onUpdatePage as (page: number) => void
// 更新分页每页条数
const updatePageSize = paginationRef.value.onUpdatePageSize as (
pageSize: number,
) => void
@ -137,7 +143,7 @@ export const usePagination = <T extends AnyFC>(
* @description
*
*/
const getCallback = callback
const getCallback = callbackRef.value as T
/**
*
@ -149,9 +155,8 @@ export const usePagination = <T extends AnyFC>(
* @example
* setCallback(() => {})
*/
const setCallback = (callback: T) => {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
callbackRef.value = callback as any
const setCallback = (callback: AnyFC) => {
callbackRef.value = callback
}
/**
@ -165,8 +170,9 @@ export const usePagination = <T extends AnyFC>(
const resetPagination = () => {
const { pageSizes } = paginationRef.value
paginationRef.value.page = 1
paginationRef.value.pageSize = (pageSizes?.[0] as number) || 10
paginationRef.value.page = DEFAULT_OPTIONS.page
paginationRef.value.pageSize =
(pageSizes?.[0] as number) || DEFAULT_OPTIONS.pageSize
}
effectDispose(() => {

View File

@ -16,16 +16,16 @@ export type UsePrintTarget<T = unknown> =
/**
*
* @param target ref dom
* @param target useTemplateRef dom
* @param options print-js options
*
* @see https://printjs.crabbly.com/
*
* @description
* print-js usePrint ref Dom
* print-js usePrint useTemplateRef Dom
*
* @example
* const refDom = ref<HTMLElement>()
* const refDom = useTemplateRef<HTMLElement>('refDom')
*
* const { print } = usePrint(refDom, {})
*

View File

@ -6,7 +6,7 @@ import { RIcon } from '@/components'
import { isValueType, renderNode } from '@/utils'
import { useSettingGetters } from '@/store'
export const SIDER_BAR_LOGO = ref<HTMLElement>()
export const SIDER_BAR_LOGO = shallowRef<HTMLElement>()
export default defineComponent({
name: 'SiderBarLogo',

View File

@ -7,6 +7,7 @@ import { LAYOUT_SIDER_REF } from '@/app-config'
import { useDevice } from '@/hooks'
import { getVariableToRefs, setVariable } from '@/global-variable'
import { useMenuGetters, useMenuActions, useSettingGetters } from '@/store'
import { positionSelectedMenuItem } from '@/utils'
import type { MenuInst } from 'naive-ui'
import type { NaiveMenuOptions } from '@/types'
@ -15,7 +16,8 @@ import type { AppMenuOption } from '@/types'
export default defineComponent({
name: 'AppMenu',
setup() {
const menuRef = ref<MenuInst | null>(null)
// 这里使用 shallowRef 而不是 useTemplateRef 是因为在这里有一个特殊情况,会导致一个 readonly 的警告
const menuRef = shallowRef<MenuInst | null>()
const { changeMenuModelValue, collapsedMenu, updateMenuState } =
useMenuActions()
@ -23,8 +25,6 @@ export default defineComponent({
const { getMenuOptions, getCollapsed, getMenuKey } = useMenuGetters()
const modelMenuKey = computed({
get: () => {
// eslint-disable-next-line vue/no-async-in-computed-properties
setTimeout(() => {
/**
*
* @description
@ -33,7 +33,10 @@ export default defineComponent({
*
*
*/
// eslint-disable-next-line vue/no-async-in-computed-properties
setTimeout(() => {
showMenuOption()
positionSelectedMenuItem()
}, 300)
return getMenuKey.value

View File

@ -25,46 +25,36 @@ $menuTagWrapperWidth: 76px;
}
// 激活标签页关闭按钮样式
.menu-tag {
.menu-tag__btn {
.menu-tag__btn-icon--hidden {
display: none !important;
}
.menu-tag .menu-tag__btn {
transition:
color 0.3s var(--n-bezier),
background-color 0.3s var(--n-bezier),
opacity 0.3s var(--n-bezier),
border-color 0.3s var(--n-bezier),
width 0.3s var(--n-bezier);
.menu-tag__btn-icon {
display: inline;
margin-left: 0;
width: 0;
height: 0;
transition: all 0.3s var(--r-bezier);
overflow: hidden;
.n-button__icon {
opacity: 0;
& .ray-icon {
width: 11px !important;
height: 11px !important;
}
width: 0;
height: var(--n-icon-size);
transition: all 0.3s var(--r-bezier);
margin-left: 0px;
transform: translateX(4px);
}
&:hover {
.menu-tag__btn-icon {
width: 14px;
height: 14px;
margin-left: 5px;
font-size: 12px;
background-color: rgba(0, 0, 0, 0.12);
border-radius: 50%;
padding: 1px;
transition: all 0.3s var(--r-bezier);
.n-button__icon {
opacity: 1;
display: flex;
justify-content: center;
align-items: center;
}
width: var(--n-icon-size);
transition: all 0.3s var(--r-bezier);
}
}
}
.menu-tag__btn-icon:hover {
filter: brightness(0.8);
}
// 设置 dropdown animate svg 尺寸
.menu-tag__dropdown {
& .menu-tag__icon {

View File

@ -517,6 +517,7 @@ export default defineComponent({
}}
size="small"
focusable={false}
iconPlacement="right"
>
{{
default: () => (
@ -533,15 +534,17 @@ export default defineComponent({
},
}}
</span>
<NIcon
class="menu-tag__btn-icon"
</>
),
icon: () => (
<RIcon
customClassName="menu-tag__btn-icon"
name="close"
size="15"
{...{
onMousedown: closeCurrentMenuTag.bind(this, idx),
}}
>
<RIcon name="close" size="14" />
</NIcon>
</>
/>
),
}}
</NButton>

View File

@ -2,85 +2,8 @@ import type { SettingState } from '@/store/modules/setting/types'
import type { InjectionKey, Reactive } from 'vue'
import type { DebouncedFunc } from 'lodash-es'
type OmitKeys =
| 'contentTransition'
| 'watermarkSwitch'
| 'keepAliveConfig'
| 'menuConfig'
| 'menuTagSwitch'
| 'breadcrumbSwitch'
| 'copyrightSwitch'
| 'drawerPlacement'
| 'colorWeakness'
| 'watermarkConfig'
| 'dynamicDocumentTitle'
interface Config extends Pick<SettingState, OmitKeys> {}
// SettingDrawer 默认配置系统配置项
const defaultSettingConfig: Readonly<Partial<Config>> = {
contentTransition: 'scale',
watermarkSwitch: false,
keepAliveConfig: {
maxKeepAliveLength: 10,
setupKeepAlive: true,
keepAliveExclude: [],
},
menuConfig: {
collapsedWidth: 64,
collapsedMode: 'width',
collapsedIconSize: 16,
collapsedIndent: 24,
accordion: false,
menuSiderBarLogo: true,
iconSize: 16,
menuWidth: 272,
inverted: false,
nativeScrollbar: false,
},
menuTagSwitch: true,
breadcrumbSwitch: true,
copyrightSwitch: true,
drawerPlacement: 'right',
colorWeakness: false,
watermarkConfig: {
content: 'Trying be better~',
fontSize: 16,
lineHeight: 16,
width: 384,
height: 384,
xOffset: 12,
yOffset: 60,
rotate: -15,
xGap: 0,
yGap: 0,
cross: true,
},
dynamicDocumentTitle: true,
}
/**
*
* @description
*
*/
export const getDefaultSettingConfig = (): Partial<Config> => {
return defaultSettingConfig
}
/**
*
* @param config
*
* @description
*
*/
export const updateDefaultSettingConfig = (config: Partial<SettingState>) => {
Object.assign(defaultSettingConfig, config)
}
interface SettingDrawerInjectKey extends SettingState {
throttleSetupAppMenu: DebouncedFunc<() => Promise<void>>
throttleSetupAppMenu: DebouncedFunc<() => void>
}
export const SETTING_DRAWER_INJECT_KEY: Reactive<

View File

@ -15,10 +15,11 @@ import SegmentViewsWatermark from './segment-views/Watermark'
import SegmentViewsCustomMenu from './segment-views/CustomMenu'
import { useSettingGetters, useSettingActions, useMenuActions } from '@/store'
import { getDefaultSettingConfig, SETTING_DRAWER_INJECT_KEY } from './constant'
import { SETTING_DRAWER_INJECT_KEY } from './constant'
import { forIn, throttle } from 'lodash-es'
import { drawerProps } from 'naive-ui'
import { useModal } from '@/components'
import { getDefaultSettingConfig } from '@/store/modules/setting/constant'
import type { SettingState } from '@/store/modules/setting/types'

View File

@ -29,9 +29,9 @@ import { useSettingGetters } from '@/store'
export default defineComponent({
name: 'RLayout',
setup() {
const layoutSiderBarRef = ref<HTMLElement>() // 顶部操作栏 ref
const layoutMenuTagRef = ref<HTMLElement>() // 标签页 ref
const layoutFooterRef = ref<HTMLElement>() // 底部版权 ref
const layoutSiderBarRef = shallowRef<HTMLElement>() // 顶部操作栏 shallowRef
const layoutMenuTagRef = shallowRef<HTMLElement>() // 标签页 shallowRef
const layoutFooterRef = shallowRef<HTMLElement>() // 底部版权 shallowRef
const { getMenuTagSwitch, getCopyrightSwitch } = useSettingGetters()
const { getLockAppScreen } = useAppLockScreen()

View File

@ -5,7 +5,8 @@ export const useKeepAliveGetters = () => {
/**
*
* @remark name
* @@description
* name
*/
const getKeepAliveInclude = computed(() => variable.keepAliveInclude)

View File

@ -6,22 +6,29 @@ export const useMenuGetters = () => {
/**
*
* @remark
* @description
*
*/
const getMenuOptions = computed(() => variable.options)
/**
*
* @remark
* @description
*
*/
const getBreadcrumbOptions = computed(() => variable.breadcrumbOptions)
/**
*
* @remark key
* @description
* key
*/
const getMenuKey = computed(() => variable.menuKey)
/**
*
* @remark
* @description
*
*/
const getMenuTagOptions = computed(() => {
const { getRootPath } = useAppRoot()
@ -44,14 +51,18 @@ export const useMenuGetters = () => {
return curr
})
})
/**
*
* @remark
* @description
*
*/
const getCurrentMenuOption = computed(() => variable.currentMenuOption)
/**
*
* @remark
* @description
*
*/
const getCollapsed = computed(() => variable.collapsed)

View File

@ -1,20 +1,3 @@
/**
*
* menu pinia store
*
* :
* - BreadcrumbMenuTagMenuMenu
* - BreadcrumbMenuTagMenuMenu vue-router routers,
*
* (sessionStorage):
* - breadcrumbOptions
* - menuKey
* - menuTagOptions
*
* :
* - parseAndFindMatchingNodes: 如果需要反向查找完整的父子节点使
*/
import { NEllipsis } from 'naive-ui'
import { setStorage, equalRouterPath, updateObjectValue } from '@/utils'
@ -33,10 +16,9 @@ import { APP_CATCH_KEY } from '@/app-config'
import { pick } from 'lodash-es'
import { pickRouteRecordNormalizedConstant } from './constant'
import type { AppMenuOption, MenuTagOptions } from '@/types'
import type { AppMenuOption, MenuTagOptions, AnyFC } from '@/types'
import type { MenuState } from '@/store/modules/menu/types'
import type { LocationQuery } from 'vue-router'
import type { UpdateMenuState } from './types'
let cachePreNormal: AppMenuOption | undefined = void 0
@ -101,12 +83,17 @@ export const piniaMenuStore = defineStore(
*
* @param key menu state key
* @param value updated value
* @param cb callback function
*
* @description
* menu state key
*/
const updateMenuState: UpdateMenuState = (key, value, cb) => {
updateObjectValue(menuState, key, value, cb)
const updateMenuState = <T extends keyof MenuState>(
key: T,
value: Partial<MenuState[T]>,
cb?: AnyFC,
) => {
updateObjectValue(menuState, key, value as MenuState[T], cb)
}
/**
@ -169,10 +156,8 @@ export const piniaMenuStore = defineStore(
*
*/
const setBreadcrumbOptions = (key: string | number) => {
menuState.breadcrumbOptions = parseAndFindMatchingNodes(
menuState.options,
'fullPath',
key,
menuState.breadcrumbOptions = unref(
parseAndFindMatchingNodes(menuState.options, 'fullPath', key),
)
}
@ -254,22 +239,22 @@ export const piniaMenuStore = defineStore(
const { sameLevel } = meta
/** 更新缓存队列 */
// 更新缓存队列
setKeepAliveInclude(option)
/** 更新浏览器标题 */
// 更新浏览器标题
updateDocumentTitle(option)
// 如果不为 sameLevel则会执行更新覆盖更新面包屑、添加标签菜单、更新缓存
if (!sameLevel) {
/** 更新标签菜单 */
// 更新标签菜单
setMenuTagOptionsWhenMenuValueChange(key, option)
/** 更新面包屑 */
// 更新面包屑
setBreadcrumbOptions(key)
menuState.menuKey = key
menuState.currentMenuOption = option
/** 缓存菜单 key(sessionStorage) */
// 缓存菜单 key(sessionStorage)
setStorage(APP_CATCH_KEY.appMenuKey, key)
} else {
// 使用 pick 提取仅需要的字段,避免 vue 抛错空引用,导致性能损耗

View File

@ -8,11 +8,3 @@ export interface MenuState {
breadcrumbOptions: AppMenuOption[]
currentMenuOption: AppMenuOption | null
}
type PickUpdateKeys = 'collapsed' | 'currentMenuOption'
export type UpdateMenuState = <T extends keyof Pick<MenuState, PickUpdateKeys>>(
key: T,
value: Partial<MenuState[T]>,
cb?: AnyFC,
) => void

View File

@ -0,0 +1,89 @@
import type { SettingState } from './types'
type OmitKeys =
| 'contentTransition'
| 'watermarkSwitch'
| 'keepAliveConfig'
| 'menuConfig'
| 'menuTagSwitch'
| 'breadcrumbSwitch'
| 'copyrightSwitch'
| 'drawerPlacement'
| 'colorWeakness'
| 'watermarkConfig'
| 'dynamicDocumentTitle'
interface Config extends Pick<SettingState, OmitKeys> {}
// SettingDrawer 默认配置系统配置项
const defaultSettingConfig: Readonly<Config> = {
// 切换过渡效果
contentTransition: 'scale',
// 水印开关
watermarkSwitch: false,
// 缓存动态设置
keepAliveConfig: {
maxKeepAliveLength: 10,
setupKeepAlive: true,
keepAliveExclude: [],
},
// 菜单配置
menuConfig: {
collapsedWidth: 64,
collapsedMode: 'width',
collapsedIconSize: 16,
collapsedIndent: 24,
accordion: false,
menuSiderBarLogo: true,
iconSize: 16,
menuWidth: 272,
inverted: false,
nativeScrollbar: false,
},
// 多标签页开关
menuTagSwitch: true,
// 面包屑开关
breadcrumbSwitch: true,
// 底部区域开关
copyrightSwitch: true,
// 默认设置出现位置
drawerPlacement: 'right',
// 色弱模式
colorWeakness: false,
// 水印
watermarkConfig: {
content: 'Trying be better~',
fontSize: 16,
lineHeight: 16,
width: 384,
height: 384,
xOffset: 12,
yOffset: 60,
rotate: -15,
xGap: 0,
yGap: 0,
cross: true,
},
// 动态标题
dynamicDocumentTitle: true,
}
/**
*
* @description
*
*/
export const getDefaultSettingConfig = (): Readonly<Config> => {
return defaultSettingConfig
}
/**
*
* @param config
*
* @description
*
*/
export const updateDefaultSettingConfig = (config: Partial<SettingState>) => {
Object.assign(defaultSettingConfig, config)
}

View File

@ -2,6 +2,7 @@ import { getAppDefaultLanguage } from '@/locales/utils'
import { colorToRgba, setStorage, updateObjectValue, setStyle } from '@/utils'
import { useI18n, useDayjs } from '@/hooks'
import { APP_CATCH_KEY, APP_THEME, GLOBAL_CLASS_NAMES } from '@/app-config'
import { getDefaultSettingConfig } from './constant'
import type { SettingState } from '@/store/modules/setting/types'
import type { LocalKey } from '@/hooks'
@ -17,8 +18,6 @@ export const piniaSettingStore = defineStore(
const { locale: dayjsLocal } = useDayjs()
const settingState = reactive<SettingState>({
// 默认设置出现位置
drawerPlacement: 'right',
// 默认主题色
primaryColorOverride: {
common: {
@ -27,37 +26,14 @@ export const piniaSettingStore = defineStore(
primaryColorPressed: primaryColor,
},
},
// true 为黑夜主题, false 为明亮主题
// 内部使用用于判断是否为黑夜主题为了兼容历史遗留版本true 为黑夜主题,false 为明亮主题
_appTheme: false,
// 当前主题样式
appTheme: 'light',
// 多标签页开关
menuTagSwitch: true,
// 面包屑开关
breadcrumbSwitch: true,
// 默认国际化语言
localeLanguage: getAppDefaultLanguage(),
// 锁屏开关
lockScreenSwitch: false,
// 底部区域开关
copyrightSwitch: true,
// 切换过渡效果
contentTransition: 'scale',
// 水印开关
watermarkSwitch: false,
// 水印
watermarkConfig: {
content: 'Trying be better~',
fontSize: 16,
lineHeight: 16,
width: 384,
height: 384,
xOffset: 12,
xGap: 0,
yGap: 0,
yOffset: 60,
rotate: -15,
cross: true,
},
// 根路由信息
appRootRoute: {
name: 'Dashboard',
@ -70,29 +46,7 @@ export const piniaSettingStore = defineStore(
url: '/dashboard',
jumpType: 'station',
},
// 缓存动态设置
keepAliveConfig: {
setupKeepAlive: true,
keepAliveExclude: [],
maxKeepAliveLength: 10,
},
// 菜单配置
menuConfig: {
collapsedWidth: 64,
collapsedMode: 'width',
collapsedIconSize: 16,
collapsedIndent: 24,
accordion: false,
menuSiderBarLogo: true,
iconSize: 16,
menuWidth: 272,
inverted: false,
nativeScrollbar: false,
},
// 色弱模式
colorWeakness: false,
// 动态标题
dynamicDocumentTitle: true,
...getDefaultSettingConfig(),
})
// 修改当前语言
@ -150,7 +104,7 @@ export const piniaSettingStore = defineStore(
value: Partial<V[T]>,
cb?: C,
) => {
updateObjectValue(settingState, key, value, cb)
updateObjectValue(settingState, key, value as V[T], cb)
}
const toggleColorWeakness = (bool: boolean) => {

View File

@ -90,6 +90,7 @@ export const piniaSigningStore = defineStore(
removeStorage(token, 'localStorage')
removeStorage(signing, 'localStorage')
removeStorage(appMenuKey, 'localStorage')
removeStorage(APP_CATCH_KEY.isAppLockScreen, 'localStorage')
// 关闭所有侧边栏标签
closeAll()

View File

@ -22,21 +22,27 @@
}
/* scale-transform */
.scale-transform-enter-active,
.scale-transform-enter-active {
transition:
transform 0.35s cubic-bezier(0.33, 1, 0.68, 1),
opacity 0.25s cubic-bezier(0.33, 1, 0.68, 1);
transition-delay: 0.25s;
}
.scale-transform-leave-active {
transition:
transform 0.4s var(--r-bezier),
opacity 0.45s var(--r-bezier);
transform 0.35s cubic-bezier(0.32, 0, 0.67, 0),
opacity 0.25s ease-out;
}
.scale-transform-enter-from {
opacity: 0;
transform: scale(0.92);
transform: scale(0.95);
}
.scale-transform-leave-to {
opacity: 0;
transform: scale(1.06);
transform: scale(1.03);
}
/* opacity-transform */
@ -54,35 +60,49 @@
}
/* fade-bottom-transform */
.fade-bottom-transform-enter-active,
.fade-bottom-transform-enter-active {
transition:
transform 0.35s cubic-bezier(0.33, 1, 0.68, 1),
opacity 0.25s cubic-bezier(0.33, 1, 0.68, 1);
transition-delay: 0.25s;
}
.fade-bottom-transform-leave-active {
transition:
opacity 0.55s var(--r-bezier),
transform 0.45s var(--r-bezier);
transform 0.35s cubic-bezier(0.32, 0, 0.67, 0),
opacity 0.25s ease-out;
}
.fade-bottom-transform-enter-from {
opacity: 0;
transform: translateY(-10%);
transform: translateY(-15px);
}
.fade-bottom-transform-leave-to {
opacity: 0;
transform: translateY(10%);
transform: translateY(15px);
}
/* fade-scale-transform */
.fade-scale-transform-leave-active,
.fade-scale-transform-enter-active {
transition: all 0.48s var(--r-bezier);
transition:
transform 0.35s cubic-bezier(0.33, 1, 0.68, 1),
opacity 0.25s cubic-bezier(0.33, 1, 0.68, 1);
transition-delay: 0.25s;
}
.fade-scale-transform-leave-active {
transition:
transform 0.25s cubic-bezier(0.32, 0, 0.67, 0),
opacity 0.25s ease-out;
}
.fade-scale-transform-enter-from {
opacity: 0;
transform: scale(1.2);
transform: scale(1.1);
}
.fade-scale-transform-leave-to {
opacity: 0;
transform: scale(0.8);
transform: scale(0.85);
}

View File

@ -209,6 +209,31 @@ export const downloadAnyFile = (
})
}
/**
*
* @param file file object
*
* @description
* File base64
*
* @example
* const base64 = await fileToBase64(file) // 'data:image/png;base64,...'
*/
export const fileToBase64 = (file: File) => {
return new Promise<string>((resolve, reject) => {
const reader = new FileReader()
reader.onload = () => {
resolve(reader.result as string)
}
reader.onerror = (error) => {
reject(error)
}
reader.readAsDataURL(file)
})
}
/**
*
* @param func
@ -254,7 +279,7 @@ export const isPromise = <T>(value: unknown): value is Promise<T> => {
/**
*
* @param fc
* @param fn
* @param errorCallback
* @param args
*
@ -266,14 +291,14 @@ export const isPromise = <T>(value: unknown): value is Promise<T> => {
* callWithErrorHandling((x: number) => { throw new Error('error') }, (error) => { console.log(error) }, [123]) => undefined
*/
export const callWithErrorHandling = <T extends AnyFC, E extends Error>(
fc: T,
fn: T,
errorCallback: AnyFC<E, void>,
args?: Parameters<T>,
) => {
let result: ReturnType<T> | undefined
try {
result = args ? fc(...args) : fc()
result = args ? fn(...args) : fn()
} catch (error) {
errorCallback(error as E)
}
@ -299,16 +324,16 @@ export const callWithAsyncErrorHandling = async <
T extends AnyFC,
E extends Error,
>(
fc: T,
fn: T,
errorCallback: (error: E) => void,
args?: Parameters<T>,
): Promise<ReturnType<T> | undefined> => {
try {
if (!isPromise(fc)) {
return Promise.resolve(callWithErrorHandling(fc, errorCallback, args))
if (!isPromise(fn)) {
return Promise.resolve(callWithErrorHandling(fn, errorCallback, args))
}
return await fc(...(args as Parameters<T>))
return await fn(...(args as Parameters<T>))
} catch (error) {
errorCallback(error as E)

View File

@ -3,14 +3,31 @@ import { APP_CATCH_KEY_PREFIX } from '@/app-config'
import type { StorageLike, StorageOptions, RemoveStorageFC } from '@/types'
/**
*
* @param key key
* @param storageType
* @param type
* @returns Storage
*/
const getStorageInstance = (type: StorageLike = 'sessionStorage'): Storage =>
type === 'localStorage' ? window.localStorage : window.sessionStorage
/**
*
* @param key
* @param options
* @returns
*/
const getKeyWithPrefix = (key: string, options?: StorageOptions): string => {
const { prefix, prefixKey } = options ?? {}
return prefix ? (prefixKey || APP_CATCH_KEY_PREFIX) + key : key
}
/**
* @param key key
* @param storageType
* @param options
*
* @description
* key
*
* sessionStorage
*
* @example
@ -21,23 +38,20 @@ function hasStorage(
storageType: StorageLike = 'sessionStorage',
options?: StorageOptions,
) {
const { prefix, prefixKey } = options ?? {}
const _prefix = prefix ? prefixKey || APP_CATCH_KEY_PREFIX : ''
const storage =
storageType === 'localStorage' ? window.localStorage : window.sessionStorage
const prefixedKey = getKeyWithPrefix(key, options)
const storage = getStorageInstance(storageType)
return !!Object.keys(storage).find((curr) => curr === _prefix + key)
return Object.keys(storage).includes(prefixedKey)
}
/**
*
* @param key key
* @param type
* @param key key
* @param value
* @param storageType
* @param options
*
* @description
* sessionStorage
*
* key
*
* @example
@ -52,22 +66,16 @@ function setStorage<T = unknown>(
options?: StorageOptions,
) {
if (!key) {
console.error(
`[setStorage]: Failed to set stored data: key ${key} is empty`,
)
console.error('[setStorage]: Failed to set stored data: key is empty')
return
}
const { prefix, prefixKey } = options ?? {}
const _prefix = prefix ? prefixKey || APP_CATCH_KEY_PREFIX : ''
const prefixedKey = getKeyWithPrefix(key, options)
const storage = getStorageInstance(storageType)
try {
const waitCacheValue = JSON.stringify(value)
storageType === 'localStorage'
? window.localStorage.setItem(_prefix + key, waitCacheValue)
: window.sessionStorage.setItem(_prefix + key, waitCacheValue)
storage.setItem(prefixedKey, JSON.stringify(value))
} catch (error) {
console.error(
`[setStorage]: Failed to set stored data for key '${key}'`,
@ -79,26 +87,23 @@ function setStorage<T = unknown>(
function getStorage<T = unknown>(
key: string,
storageType: StorageLike,
options?: StorageOptions<T>,
options: StorageOptions<T> & { defaultValue: T },
): T
function getStorage<T = unknown>(
key: string,
storageType?: StorageLike,
options?: StorageOptions<T>,
options?: Omit<StorageOptions<T>, 'defaultValue'>,
): T | null
/**
*
* @param key key
* @param type
* @param key key
* @param storageType
* @param options
*
* @description
*
*
* sessionStorage
*
* key
*
* @example
@ -111,20 +116,14 @@ function getStorage<T = unknown>(
storageType: StorageLike = 'sessionStorage',
options?: StorageOptions<T>,
): T | null {
const { prefix, prefixKey, defaultValue } = options ?? {}
const _prefix = prefix ? prefixKey || APP_CATCH_KEY_PREFIX : ''
const prefixedKey = getKeyWithPrefix(key, options)
const storage = getStorageInstance(storageType)
const { defaultValue } = options ?? {}
try {
const data =
storageType === 'localStorage'
? window.localStorage.getItem(_prefix + key)
: window.sessionStorage.getItem(_prefix + key)
const data = storage.getItem(prefixedKey)
if (data === null) {
return defaultValue ?? null
}
return JSON.parse(data) as T
return data === null ? defaultValue ?? null : (JSON.parse(data) as T)
} catch (error) {
console.error(
`[getStorage]: Failed to get stored data for key '${key}'`,
@ -136,16 +135,13 @@ function getStorage<T = unknown>(
}
/**
*
* @param key key
* @param type
* @param storageType
* @param options
*
* @description
*
*
* sessionStorage
*
* __all____all_sessionStorage____all_localStorage__ key
* sessionStorage localStorage
*
@ -153,63 +149,47 @@ function getStorage<T = unknown>(
* removeStorage('__all__', 'all') // 清空所有缓存
* removeStorage('__all_sessionStorage__', 'sessionStorage') // 清空 sessionStorage 缓存
* removeStorage('__all_localStorage__', 'localStorage') // 清空 localStorage 缓存
* removeStorage('signing', 'sessionStorage' || 'localStorage') // 清空 session 中 signing 缓存字段
* removeStorage('signing', 'sessionStorage' || 'localStorage') // 清空 session 或者 localStorage 中 signing 缓存字段
*/
const removeStorage: RemoveStorageFC = (key, storageType, options) => {
if (!key) {
console.error(
`[removeStorage]: Failed to remove stored data: key ${key} is empty or undefined`,
)
console.error('[removeStorage]: Failed to remove stored data: key is empty')
return
}
const { prefix, prefixKey } = options ?? {}
const _prefix = prefix ? prefixKey || APP_CATCH_KEY_PREFIX : ''
const localStorageKeys = Object.keys(window.localStorage)
const sessionStorageKeys = Object.keys(window.sessionStorage)
const prefixedKey = getKeyWithPrefix(key, options)
const localStorage = window.localStorage
const sessionStorage = window.sessionStorage
const remove = (isAll: boolean, removeType?: StorageLike) => {
const keys = isAll
? [...sessionStorageKeys, ...localStorageKeys]
: removeType === 'localStorage'
? localStorageKeys
: sessionStorageKeys
keys.forEach((curr) => {
if (key === '__all__') {
window.sessionStorage.removeItem(_prefix + curr)
window.localStorage.removeItem(_prefix + curr)
} else {
removeType === 'localStorage'
? window.localStorage.removeItem(_prefix + curr)
: window.sessionStorage.removeItem(_prefix + curr)
}
})
/**
* @param storage
*/
const removeFromStorage = (storage: Storage) => {
Object.keys(storage).forEach((k) =>
storage.removeItem(getKeyWithPrefix(k, options)),
)
}
switch (key) {
case '__all__':
remove(true)
removeFromStorage(localStorage)
removeFromStorage(sessionStorage)
break
case '__all_sessionStorage__':
remove(false, 'sessionStorage')
removeFromStorage(sessionStorage)
break
case '__all_localStorage__':
remove(false, 'localStorage')
removeFromStorage(localStorage)
break
default:
storageType === 'localStorage'
? window.localStorage.removeItem(_prefix + key)
: window.sessionStorage.removeItem(_prefix + key)
break
getStorageInstance(storageType as StorageLike).removeItem(prefixedKey)
}
}

View File

@ -300,6 +300,7 @@ export const removeStyle = (
*
* @param color
* @param alpha
* @param defaultColor
*
* @description
* rgba rgba
@ -309,20 +310,50 @@ export const removeStyle = (
* colorToRgba('rgb(18, 54, 50)', 0.8) // rgba(18, 54, 50, 0.8)
* colorToRgba('#ee4f12', 0.3) // rgba(238, 79, 18, 0.3)
* colorToRgba('rgba(238, 79, 18, 0.3)', 0.3) // rgba(238, 79, 18, 0.3)
* colorToRgba('not a color', 0.3) // not a color
* colorToRgba('not a color', 0.3, '#000') // #000
* colorToRgba('hsl(0, 100%, 50%)', 0.5) // hsla(0, 100%, 50%, 0.5)
* colorToRgba('not a color') // not a color
*/
export const colorToRgba = (color: string, alpha = 1) => {
if (color.includes('rgba')) {
return color
export const colorToRgba = (
color: string,
alpha = 1,
defaultColor?: string,
) => {
if (!color) {
return defaultColor || color
}
if (color.includes('rgb')) {
// 处理 rgba 格式 - 替换原有的 alpha 值
if (color.includes('rgba')) {
return color.replace(
/rgba\((.*?),(.*?),(.*?),.*?\)/,
`rgba($1,$2,$3,${alpha})`,
)
}
// 处理 rgb 格式
if (color.includes('rgb(')) {
return color.replace('rgb', 'rgba').replace(')', `, ${alpha})`)
}
if (color.includes('#')) {
// 处理 hsla 格式 - 替换原有的 alpha 值
if (color.includes('hsla')) {
return color.replace(
/hsla\((.*?),(.*?),(.*?),.*?\)/,
`hsla($1,$2,$3,${alpha})`,
)
}
// 处理 hsl 格式
if (color.includes('hsl(')) {
return color.replace('hsl', 'hsla').replace(')', `, ${alpha})`)
}
// 处理 hex 格式
if (color.startsWith('#')) {
const hex = color.replace('#', '')
try {
switch (hex.length) {
case 3:
return `rgba(${parseInt(hex[0] + hex[0], 16)}, ${parseInt(hex[1] + hex[1], 16)}, ${parseInt(hex[2] + hex[2], 16)}, ${alpha})`
@ -331,14 +362,18 @@ export const colorToRgba = (color: string, alpha = 1) => {
return `rgba(${parseInt(hex.slice(0, 2), 16)}, ${parseInt(hex.slice(2, 4), 16)}, ${parseInt(hex.slice(4, 6), 16)}, ${alpha})`
case 8:
return `rgba(${parseInt(hex.slice(0, 2), 16)}, ${parseInt(hex.slice(2, 4), 16)}, ${parseInt(hex.slice(4, 6), 16)}, ${(parseInt(hex.slice(6, 8), 16) / 255).toFixed(2)})`
return `rgba(${parseInt(hex.slice(0, 2), 16)}, ${parseInt(hex.slice(2, 4), 16)}, ${parseInt(hex.slice(4, 6), 16)}, ${alpha})`
default:
return color
return defaultColor || color
}
} catch {
return defaultColor || color
}
}
return color
// 返回原始字符串或默认值
return defaultColor || color
}
/**

View File

@ -3,8 +3,12 @@ import { useSiderScroll } from '@/hooks'
import { unrefElement } from '@/utils'
import { SIDER_BAR_LOGO } from '@/layout/components/Menu/components/SiderBarLogo'
const MENU_ITEM_SELECTED = '.n-menu-item-content--selected' // 菜单激活样式 class name
const MENU_ITEM = 'n-menu-item'
// 菜单激活样式 class name
const MENU_ITEM_SELECTED = '.n-menu-item-content--selected'
// 菜单项 class name
const MENU_ITEM = '.n-menu-item'
// 折叠菜单激活样式 class name
const MENU_COLLAPSED_ITEM_SELECTED = '.n-menu-item-content--child-active'
/**
*
@ -20,36 +24,61 @@ const MENU_ITEM = 'n-menu-item'
* positionSelectedMenuItem() // 滚动到当前激活菜单项的位置
*/
export const positionSelectedMenuItem = () => {
// 获取 Sider 元素
const siderEl = unrefElement(LAYOUT_SIDER_REF as Ref<HTMLElement>)
const selectedEl = siderEl?.querySelector<HTMLElement>(MENU_ITEM_SELECTED)
// 获取激活的菜单项元素
const selectedEl =
siderEl?.querySelector<HTMLElement>(MENU_ITEM_SELECTED) ||
siderEl?.querySelector<HTMLElement>(MENU_COLLAPSED_ITEM_SELECTED)
// 获取 Sider 栏 Logo 元素
const siderBarLogoEl = unrefElement(SIDER_BAR_LOGO)
// Sider 栏 Logo 高度
let siderBarLogoHeight: number = 0
// 获取菜单项元素
const menuItemEl = siderEl?.querySelector<HTMLElement>(MENU_ITEM)
// 获取 Sider 栏 Logo 高度
if (siderBarLogoEl) {
const { height } = siderBarLogoEl.getBoundingClientRect()
siderBarLogoHeight = height
}
// 如果激活的菜单项元素和 Sider 元素存在,则滚动 Sider 栏
if (selectedEl && siderEl) {
// 获取元素相对于视口的位置信息
const selectedRect = selectedEl.getBoundingClientRect()
const siderRect = siderEl.getBoundingClientRect()
// 检查 selectedEl 是否在 sider 的可视区域内
const isInViewport =
selectedRect.top >= siderRect.top + siderBarLogoHeight &&
selectedRect.bottom <= siderRect.bottom
// 如果不在可视区域内,才执行滚动
if (!isInViewport) {
// 获取 Sider 滚动条
const siderScroll = useSiderScroll()
const { top: siderTop } = siderEl.getBoundingClientRect()
const { top: selectedTop } = selectedEl.getBoundingClientRect()
// 获取 Sider 元素当前滚动条的位置
const siderScrollTop = siderEl.scrollTop
// 获取菜单项元素的 margin-top 值,如果菜单项元素不存在,则使用默认值 6测试得出 6px 是最佳值)
const menuItemMarginTop = menuItemEl
? parseInt(window.getComputedStyle(menuItemEl).marginTop)
: 6
siderScroll({
// 滚动到激活的菜单项位置,减去 Sider 栏 Logo 高度和菜单项的 margin-top 值
top:
selectedTop -
siderTop +
selectedRect.top -
siderRect.top +
siderScrollTop -
siderBarLogoHeight -
menuItemMarginTop,
// 水平滚动到 0 位置
left: 0,
// 平滑滚动
behavior: 'smooth',
})
}
}
}

View File

@ -21,7 +21,6 @@ import { cloneDeep } from 'lodash-es'
import { isValueType } from '@/utils'
import type { Options } from 'currency.js'
import type { AnyFC } from '@/types'
export type CurrencyArguments = string | number | currency
@ -35,7 +34,7 @@ export interface CurrencyOptions extends Options {
const defaultOptions: Partial<CurrencyOptions> = {
precision: 8,
decimal: '.',
}
} as const
// currency.js 原型属性集合
const currencyPrototypeKeys = [
's',
@ -52,7 +51,7 @@ const currencyPrototypeKeys = [
'multiply',
'subtract',
'toString',
]
] as const
/**
*
@ -66,23 +65,28 @@ const currencyPrototypeKeys = [
const basic = (
valueOptions: CurrencyArguments[],
dividend: CurrencyArguments,
cb: AnyFC,
) => {
cb: (
pre: currency,
curr: CurrencyArguments,
idx: number,
arr: CurrencyArguments[],
) => currency,
): currency => {
// 如果 valueOptions 为空,则返回 0
if (!valueOptions?.length) {
return 0
return currency(0, defaultOptions)
}
// 如果 valueOptions 长度为 1则返回 valueOptions[0]
if (valueOptions.length === 1) {
return currency(valueOptions[0], defaultOptions)
}
const result = valueOptions.reduce((pre, curr, idx, arr) => {
pre = cb?.(pre, curr, idx, arr)
// 初始值
const initialValue = currency(dividend, defaultOptions)
return pre
}, dividend)
return result
// 计算
return valueOptions.reduce(cb, initialValue)
}
/**
@ -98,16 +102,14 @@ const basic = (
* isCurrency({ s: 1, intValue: 1, p: 1, value: 1 }) // false
* isCurrency(currency(1)) // true
*/
export const isCurrency = (value: unknown) => {
if (typeof value === 'string' || typeof value === 'number') {
export const isCurrency = (value: unknown): value is currency => {
// 如果 value 不是对象类型,则返回 false
if (!isValueType<object>(value, 'Object')) {
return false
}
if (isValueType<object>(value, 'Object')) {
return currencyPrototypeKeys.every((key) => Reflect.has(value, key))
}
return false
// 如果 value 对象的属性在 currencyPrototypeKeys 中,则返回 true
return currencyPrototypeKeys.every((key) => key in value)
}
/**
@ -124,13 +126,16 @@ export const isCurrency = (value: unknown) => {
* format(0.1, { symbol: '¥' }) // ¥0.1
*/
export const format = (value: CurrencyArguments, options?: CurrencyOptions) => {
const assignOptions = Object.assign({}, defaultOptions, options)
// 合并默认配置和传入配置
const assignOptions = { ...defaultOptions, ...options }
// 将 value 转换为 currency 对象
const v = currency(value, assignOptions)
// 获取 type 配置,如果 type 不存在,则使用 'number'
const { type = 'number' } = assignOptions
// 如果 type 为 'number',则返回 value 的值,否则返回 value 的字符串
return type === 'number' ? v.value : v.toString()
}
/**
*
* @description
@ -140,14 +145,19 @@ export const format = (value: CurrencyArguments, options?: CurrencyOptions) => {
* format(add(0.1, 0.2)) // 0.3
* format(add(0.2, 0.33)) // 0.53
*/
export const add = (...args: CurrencyArguments[]) => {
export const add = (...args: CurrencyArguments[]): currency => {
// 如果 args 为空,则返回 0
if (!args.length) {
return currency(0, defaultOptions)
}
// 如果 args 长度为 1则返回 args[0] 加上 0
if (args.length === 1) {
return currency(args[0], defaultOptions).add(0)
}
return basic(args, 0, (pre, curr) => {
return currency(pre, defaultOptions).add(curr)
})
// 计算
return basic(args, 0, (pre, curr) => pre.add(curr))
}
/**
@ -159,27 +169,27 @@ export const add = (...args: CurrencyArguments[]) => {
* format(subtract(0.1, 0.12312)) // -0.02
* format(subtract(0.2, 0.33)) // -0.13
*/
export const subtract = (...args: CurrencyArguments[]) => {
export const subtract = (...args: CurrencyArguments[]): currency => {
// 如果 args 为空,则返回 0
if (!args.length) {
return currency(0, defaultOptions)
}
// 如果 args 长度为 1则返回 args[0] 减去 0
if (args.length === 1) {
return currency(args[0], defaultOptions).subtract(0)
}
// 如果 args 长度为 2则返回 args[0] 减去 args[1]
if (args.length === 2) {
const [one, two] = args
return currency(one, defaultOptions).subtract(two)
}
const cloneDeepArgs = cloneDeep(args)
const dividend = cloneDeepArgs.shift() as CurrencyArguments
const [first, ...rest] = args
if (!cloneDeepArgs.length) {
return dividend
}
return basic(cloneDeepArgs, dividend, (pre, curr) => {
return currency(pre, defaultOptions).subtract(curr)
})
return basic(rest, first, (pre, curr) => pre.subtract(curr))
}
/**
@ -192,10 +202,12 @@ export const subtract = (...args: CurrencyArguments[]) => {
* format(multiply(0.2, 0.33)) // 0.07
*/
export const multiply = (...args: CurrencyArguments[]) => {
// 如果 args 长度为 1则返回 args[0] 乘以 1
if (args.length === 1) {
return currency(args[0], defaultOptions).multiply(1)
}
// 计算
return basic(args, 1, (pre, curr) => {
return currency(pre, defaultOptions).multiply(curr)
})
@ -211,17 +223,21 @@ export const multiply = (...args: CurrencyArguments[]) => {
* format(divide(0.2, 0.33)) // 0.61
*/
export const divide = (...args: CurrencyArguments[]) => {
// 如果 args 为空,则返回 0
if (args.length === 1) {
return currency(args[0], defaultOptions).divide(1)
}
// 如果 args 长度为 2则返回 args[0] 除以 args[1]
if (args.length === 2) {
const [one, two] = args
return currency(one, defaultOptions).divide(two)
}
// 深度克隆 args
const cloneDeepArgs = cloneDeep(args)
// 获取 dividend
const dividend = cloneDeepArgs.shift() as CurrencyArguments
return basic(cloneDeepArgs, dividend, (pre, curr) => {
@ -245,15 +261,18 @@ export const distribute = (
length: number,
options?: CurrencyOptions,
) => {
// 如果 length 小于等于 1则返回一个包含 value 的数组
if (length <= 1) {
return [value ? value : 0]
} else {
// 如果 value 为 undefined null则返回一个长度为 length 的 0 数组
if (!value) {
return new Array(length).fill(0)
}
}
const assignOptions = Object.assign({}, defaultOptions, options)
// 合并配置项
const assignOptions = { ...defaultOptions, ...options }
const result = currency(value, assignOptions)
.distribute(length)

View File

@ -22,27 +22,34 @@ import type { Recordable, AnyFC } from '@/types'
export const updateObjectValue = <
Target extends Recordable,
Key extends keyof Target,
Value extends Partial<Target[Key]>,
Callback extends AnyFC,
Value extends Target[Key], // 移除 Partial使类型更准确
Callback extends AnyFC = AnyFC, // 添加默认类型
>(
targetObject: Target,
key: Key,
value: Value,
callback?: Callback,
) => {
): void => {
if (!targetObject || typeof targetObject !== 'object') {
console.warn(
`[updateObjectValue]: targetObject must be an object, expected ${typeof targetObject}`,
`[updateObjectValue]: targetObject must be an object, received ${typeof targetObject}`,
)
return
}
if (Object.hasOwn(targetObject, key)) {
typeof value === 'object'
? (targetObject[key] = Object.assign({}, targetObject[key], value))
: (targetObject[key] = value)
if (!Object.hasOwn(targetObject, key)) {
return
}
if (typeof value === 'object' && value !== null) {
targetObject[key] = {
...targetObject[key],
...value,
} as Target[Key]
} else {
targetObject[key] = value
}
callback?.()
}
}

View File

@ -4,7 +4,7 @@ import type { AnyFC } from '@/types'
/**
*
* @param fc effect
* @param fn effect
*
* @description
* true effect false effect
@ -16,9 +16,9 @@ import type { AnyFC } from '@/types'
* effectDispose(watchStop)
* effectDispose(watchEffectStop)
*/
export function effectDispose<T extends AnyFC>(fc: T) {
export function effectDispose<T extends AnyFC>(fn: T) {
if (getCurrentScope()) {
onScopeDispose(fc)
onScopeDispose(fn)
return true
}

View File

@ -5,7 +5,7 @@ import type { AnyFC } from '@/types'
/**
*
* @param fc
* @param fn
* @param watchOptions watchEffect
*
* @description
@ -19,10 +19,10 @@ import type { AnyFC } from '@/types'
* watchEffectWithTarget(watcher)
*/
export function watchEffectWithTarget<T extends AnyFC>(
fc: T,
fn: T,
watchOptions?: WatchOptionsBase,
) {
const stop = watchEffect(fc, watchOptions)
const stop = watchEffect(fn, watchOptions)
effectDispose(stop)
}

View File

@ -12,9 +12,7 @@ import { NResult, NButton, NFlex } from 'naive-ui'
import { redirectRouterToDashboard } from '@/router/utils'
import { resultProps } from 'naive-ui'
import { getStorage } from '@/utils'
import { useVueRouter } from '@/hooks'
import { APP_CATCH_KEY } from '@/app-config'
import { useSettingGetters } from '@/store'
import type { ResultProps } from 'naive-ui'
@ -24,16 +22,13 @@ const PageResult = defineComponent({
...resultProps,
},
setup() {
const { router } = useVueRouter()
const { replace } = useRouter()
const goBack = () => {
const { appMenuKey } = APP_CATCH_KEY
const key = getStorage(appMenuKey, 'sessionStorage', {
defaultValue: '',
})
const { getAppRootRoute } = useSettingGetters()
if (key) {
router.replace(key)
if (getAppRootRoute.value?.path) {
replace(getAppRootRoute.value.path)
}
}

View File

@ -1,16 +1,18 @@
import { NForm, NFormItem, NInput, NButton } from 'naive-ui'
import { RForm } from '@/components'
import { setStorage } from '@/utils'
import { useI18n, useAppRoot } from '@/hooks'
import { APP_CATCH_KEY } from '@/app-config'
import { useSigningActions } from '@/store'
import { useForm } from '@/components'
import type { FormInst } from 'naive-ui'
export default defineComponent({
name: 'RSigning',
setup() {
const loginFormRef = ref<FormInst>()
const [register, { validate }] = useForm()
const { t } = useI18n()
const { signing } = useSigningActions()
@ -40,8 +42,7 @@ export default defineComponent({
/** 普通登陆形式 */
const handleLogin = () => {
loginFormRef.value?.validate((valid) => {
if (!valid) {
validate().then(() => {
loading.value = true
signing(signingForm.value)
@ -62,23 +63,22 @@ export default defineComponent({
.catch(() => {
window.$message.error('不可以这样哟, 不可以哟')
})
}
})
}
return {
signingForm,
loginFormRef,
register,
handleLogin,
rules,
loading,
}
},
render() {
const { $t, loading } = this
const { $t, loading, register } = this
return (
<NForm model={this.signingForm} ref="loginFormRef" rules={this.rules}>
<RForm model={this.signingForm} onRegister={register} rules={this.rules}>
<NFormItem label={$t('views.login.index.Name')} path="name">
<NInput
v-model:value={this.signingForm.name}
@ -102,7 +102,7 @@ export default defineComponent({
>
{$t('views.login.index.Login')}
</NButton>
</NForm>
</RForm>
)
},
})