version: v4.8.2

This commit is contained in:
XiaoDaiGua-Ray 2024-04-29 23:29:11 +08:00
parent ae67b7b8b9
commit 1ee3ce0500
92 changed files with 1480 additions and 750 deletions

View File

@ -209,5 +209,78 @@ module.exports = {
message: 'Using deprecated API is not allowed.', message: 'Using deprecated API is not allowed.',
}, },
], ],
'padding-line-between-statements': [
'error',
{
blankLine: 'always',
prev: '*',
next: 'return',
},
{
blankLine: 'always',
prev: '*',
next: 'function',
},
{
blankLine: 'always',
prev: ['const', 'let', 'var'],
next: '*',
},
{
blankLine: 'any',
prev: ['const', 'let', 'var'],
next: ['const', 'let', 'var'],
},
{
blankLine: 'always',
prev: 'directive',
next: '*',
},
{
blankLine: 'any',
prev: 'directive',
next: 'directive',
},
{
blankLine: 'always',
prev: ['case', 'default'],
next: '*',
},
{
blankLine: 'always',
prev: ['break'],
next: '*',
},
{
blankLine: 'always',
prev: ['import'],
next: '*',
},
{
blankLine: 'any',
prev: 'import',
next: 'import',
},
{
blankLine: 'always',
prev: '*',
next: 'export',
},
{
blankLine: 'any',
prev: 'export',
next: 'export',
},
{
blankLine: 'always',
prev: ['function'],
next: '*',
},
{
blankLine: 'always',
prev: ['class'],
next: '*',
},
],
}, },
} }

2
.nvmrc
View File

@ -1 +1 @@
v18.18.2 v20.12.0

View File

@ -1,5 +1,65 @@
# CHANGE LOG # CHANGE LOG
## 4.8.2
破坏性更新!请仔细阅读更新日志!
更新默认 `node` 版本为 `v20.12.0`
并且对于历史一些模块的分包进行了优化,现在更加语意化与清晰。
重构 `axios` 拦截器,使用插件式设计思想,更加强大的拓展性、可维护性!
## Feats
- 更新默认 `node` 版本为 `v20.12.0`
- `RBarcode` 相关
- 新增 `onSuccess`, `onError`, `onFinally` 三个渲染回调
- 新增 `watchText` 配置项,当 `text` 内容变化后,主动更新条形码,默认不启用
- 更新 `vue-tsc` 版本至 `2.0.11`
- `hooks`
- `template`
- 新增 `useContentScroll` 方法,用于滚动 `LayoutContent` 内容区域
- 新增 `useSiderScroll` 方法,用于滚动 `LayoutSider` 侧边栏区域,也就是菜单区域
- `web`
- `usePagination` 类型修复,所有 `get` 方法都会准确的返回对应的类型
- `SiderBarLogo` 组件
- 支持未配置 `icon` 时,允许截取 `title` 第一个字符作为占位
- `router`
- `router types` 包,补充类型,获得更好的类型提示与配置提示
- `combine-raw-route-modules` 方法现在会检查路由配置的 `name`, `path` 属性是否出现重复,如果重复,会直接抛出异常中断流程
- 移除 `name` 配置项作为必选项
- 标记 `components` 配置项为 `deprecated` 描述项
- 增强 `AppRouteRecordRaw` 类型,现在当你显式配置了 `meta.keepAlive``true` 的时候,会提示你 `name` 必填
- `app-config`
- 提供 `MessageProver` 配置
- 限制最大消息数量为 5 条
- 默认不启用手动关闭消息功能
- `RTable` 新增 `cardProps` 配置项,配置外层容器。
- 更新 `vue` 版本至 `3.4.25`
- 更新 `vite` 版本至 `5.2.10`
- 更新 `vite-bundle-analyzer` 版本至 `0.9.4`,新增汇总模式
- 移除 `naive-ui` 自动导入 `hooks`
- `Search` 组件
- 优化分包逻辑
- 现在允许在搜索菜单初始位置进行上下键切换
- 优化搜索菜单样式
- 新增 `positionSelectedMenuItem` 方法,用于定位选中菜单项
- `hooks`
- `template`
- 新增 `useContentScroll` 方法,用于滚动 `LayoutContent` 内容区域
- 新增 `useSiderScroll` 方法,用于滚动 `LayoutSider` 侧边栏区域,也就是菜单区域
- 重写 `AppAvatar` 组件
- `axios`
- `inject` 包重命名为 `axios-interceptor`,并且拆分分包逻辑,现在以插件形式管理拦截器逻辑与注入逻辑
- 新增 `padding-line-between-statements` 规范,强制语句之间的空行
## Fixes
- 修复 `SIDE_BAR_LOGO` 未配置 `icon` 时,侧边栏不显示的问题
- 修复 `RTable` 在配置 `style.height` 时,样式会错乱的问题
- 修复 `MenuTag` 在初始化位置时不能准确滚动到当前标签页的问题
## 4.8.1 ## 4.8.1
## Feats ## Feats

View File

@ -48,7 +48,7 @@ A `completely free`, `efficient`, `feature complete` and based on vite5. x & ts(
- `Multi-terminal adaptation:` support pc, phone, pad - `Multi-terminal adaptation:` support pc, phone, pad
- `Documentation:` complete documentation - `Documentation:` complete documentation
- `Mock data:` built-in Mock data solution - `Mock data:` built-in Mock data solution
- `Axios request:` secondary encapsulation of axios library, support: cancel, jitter, automatic repeat cancellation and other functions - `Axios request:` the plug-in design is used to encapsulate the axios library interceptor twice, which makes the interceptor more flexible
- `SVG:` built-in svg icon solution - `SVG:` built-in svg icon solution
- `Hooks:` based on the template characteristics of the encapsulated hooks to make it easier to use some functions of the template - `Hooks:` based on the template characteristics of the encapsulated hooks to make it easier to use some functions of the template
- `TypeScript:` provide a complete type - `TypeScript:` provide a complete type
@ -57,12 +57,10 @@ A `completely free`, `efficient`, `feature complete` and based on vite5. x & ts(
## 👀 Preview ## 👀 Preview
- [Preview](https://xiaodaigua-ray.github.io/ray-template/#/) - [Preview](https://xiaodaigua-ray.github.io/ray-template/#/)
- [Preview(Acceleration address)](https://ray-template.yunkuangao.com/#/)
## 📌 Documentation ## 📌 Documentation
- [Documentation](https://xiaodaigua-ray.github.io/ray-template-doc/) - [Documentation](https://xiaodaigua-ray.github.io/ray-template-doc/)
- [Documentation(Acceleration address)](https://ray-template.yunkuangao.com/ray-template-doc/)
## 🔋 Change Log ## 🔋 Change Log
@ -90,9 +88,6 @@ A `completely free`, `efficient`, `feature complete` and based on vite5. x & ts(
```sh ```sh
# github # github
git clone https://github.com/XiaoDaiGua-Ray/ray-template.git git clone https://github.com/XiaoDaiGua-Ray/ray-template.git
# If your download speed is very slow, you can switch to the proxy address below
git clone https://mirror.ghproxy.com/https://github.com/XiaoDaiGua-Ray/ray-template.git
``` ```
### Pull dependencies ### Pull dependencies

View File

@ -48,7 +48,7 @@
- `多端适配:`支持 pc, phone, pad - `多端适配:`支持 pc, phone, pad
- `文档:`完善的文档 - `文档:`完善的文档
- `Mock 数据:`内置 Mock 数据方案 - `Mock 数据:`内置 Mock 数据方案
- `Axios 请求:`二次封装 axios 库,支持:取消、防抖、自动重复取消等功能 - `Axios 请求:`采用插件式设计二次封装 axios 库拦截器,让拦截器更加灵活
- `SVG`内置 svg icon 解决方案 - `SVG`内置 svg icon 解决方案
- `Hooks`基于模板特性封装的 hooks 让你更加方便的使用模板一些功能 - `Hooks`基于模板特性封装的 hooks 让你更加方便的使用模板一些功能
- `TypeScript`提供完整的类型 - `TypeScript`提供完整的类型
@ -57,12 +57,10 @@
## 👀 预览地址 ## 👀 预览地址
- [点击预览](https://xiaodaigua-ray.github.io/ray-template/#/) - [点击预览](https://xiaodaigua-ray.github.io/ray-template/#/)
- [点击预览(加速地址)](https://ray-template.yunkuangao.com/#/)
## 📌 文档地址 ## 📌 文档地址
- [文档](https://xiaodaigua-ray.github.io/ray-template-doc/) - [文档](https://xiaodaigua-ray.github.io/ray-template-doc/)
- [文档(加速地址)](https://ray-template.yunkuangao.com/ray-template-doc/)
## 🔋 更新日志 ## 🔋 更新日志
@ -90,9 +88,6 @@
```sh ```sh
# github # github
git clone https://github.com/XiaoDaiGua-Ray/ray-template.git git clone https://github.com/XiaoDaiGua-Ray/ray-template.git
# 如果你的下载速度很慢,可以切换到下面的代理地址
git clone https://mirror.ghproxy.com/https://github.com/XiaoDaiGua-Ray/ray-template.git
``` ```
### 拉取依赖 ### 拉取依赖

View File

@ -63,12 +63,4 @@ describe('usePagination', () => {
expect(count).toBe(2) expect(count).toBe(2)
}) })
it('should get callback', () => {
count = 0
getCallback()
expect(count).toBe(1)
})
}) })

View File

@ -1,7 +1,7 @@
{ {
"name": "ray-template", "name": "ray-template",
"private": false, "private": false,
"version": "4.8.1", "version": "4.8.2",
"type": "module", "type": "module",
"engines": { "engines": {
"node": "^18.0.0 || >=20.0.0", "node": "^18.0.0 || >=20.0.0",
@ -49,9 +49,9 @@
"pinia": "^2.1.7", "pinia": "^2.1.7",
"pinia-plugin-persistedstate": "^3.2.0", "pinia-plugin-persistedstate": "^3.2.0",
"print-js": "^1.6.0", "print-js": "^1.6.0",
"vue": "^3.4.21", "vue": "^3.4.25",
"vue-demi": "0.14.6", "vue-demi": "0.14.6",
"vue-hooks-plus": "1.8.8", "vue-hooks-plus": "1.9.0",
"vue-i18n": "^9.9.0", "vue-i18n": "^9.9.0",
"vue-router": "^4.3.0" "vue-router": "^4.3.0"
}, },
@ -80,7 +80,7 @@
"eslint-config-standard-with-typescript": "^43.0.0", "eslint-config-standard-with-typescript": "^43.0.0",
"eslint-plugin-prettier": "^5.1.3", "eslint-plugin-prettier": "^5.1.3",
"eslint-plugin-promise": "^6.1.1", "eslint-plugin-promise": "^6.1.1",
"eslint-plugin-vue": "^9.18.1", "eslint-plugin-vue": "^9.25.0",
"happy-dom": "14.3.1", "happy-dom": "14.3.1",
"husky": "8.0.3", "husky": "8.0.3",
"lint-staged": "^15.1.0", "lint-staged": "^15.1.0",
@ -92,8 +92,8 @@
"typescript": "^5.2.2", "typescript": "^5.2.2",
"unplugin-auto-import": "^0.17.5", "unplugin-auto-import": "^0.17.5",
"unplugin-vue-components": "^0.26.0", "unplugin-vue-components": "^0.26.0",
"vite": "^5.2.8", "vite": "^5.2.10",
"vite-bundle-analyzer": "0.8.1", "vite-bundle-analyzer": "0.9.4",
"vite-plugin-cdn2": "1.1.0", "vite-plugin-cdn2": "1.1.0",
"vite-plugin-compression": "^0.5.1", "vite-plugin-compression": "^0.5.1",
"vite-plugin-ejs": "^1.7.0", "vite-plugin-ejs": "^1.7.0",
@ -104,8 +104,8 @@
"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",
"vite-tsconfig-paths": "4.3.2", "vite-tsconfig-paths": "4.3.2",
"vitest": "1.4.0", "vitest": "1.5.2",
"vue-tsc": "^1.8.27" "vue-tsc": "^2.0.11"
}, },
"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",

771
pnpm-lock.yaml generated

File diff suppressed because it is too large Load Diff

View File

@ -19,27 +19,26 @@
import './index.scss' import './index.scss'
import { NAvatar, NFlex } from 'naive-ui' import { NAvatar, NButton, NFlex } from 'naive-ui'
import { avatarProps, flexProps } from 'naive-ui' import { avatarProps } from 'naive-ui'
import { APP_CATCH_KEY } from '@/app-config' import { APP_CATCH_KEY } from '@/app-config'
import { getStorage } from '@/utils' import { getStorage } from '@/utils'
import type { PropType } from 'vue' import type { PropType } from 'vue'
import type { AvatarProps, SpaceProps } from 'naive-ui' import type { AvatarProps, FlexProps } from 'naive-ui'
import type { SigningCallback } from '@/store/modules/signing/types' import type { SigningCallback } from '@/store/modules/signing/types'
const AppAvatar = defineComponent({ const AppAvatar = defineComponent({
name: 'AppAvatar', name: 'AppAvatar',
props: { props: {
...avatarProps, ...avatarProps,
...flexProps,
cursor: { cursor: {
type: String, type: String,
default: 'auto', default: 'auto',
}, },
spaceSize: { spaceSize: {
type: [String, Number] as PropType<SpaceProps['size']>, type: [String, Number, Array] as PropType<FlexProps['size']>,
default: 'medium', default: 'medium',
}, },
avatarSize: { avatarSize: {
@ -49,39 +48,27 @@ const AppAvatar = defineComponent({
}, },
setup(props) { setup(props) {
const signing = getStorage<SigningCallback>(APP_CATCH_KEY.signing) const signing = getStorage<SigningCallback>(APP_CATCH_KEY.signing)
const cssVars = computed(() => {
const vars = {
'--app-avatar-cursor': props.cursor,
}
return vars
})
return { return {
signing, signing,
cssVars,
} }
}, },
render() { render() {
const { signing, cssVars, spaceSize, avatarSize, $props } = this const { signing, avatarSize, spaceSize, $props } = this
return ( return (
<NFlex <NButton quaternary strong>
class="app-avatar" <NFlex align="center" size={spaceSize}>
{...this.$props} <NAvatar
style={cssVars} {...($props as AvatarProps)}
size={spaceSize} src={signing?.avatar}
> objectFit="cover"
<NAvatar round
// eslint-disable-next-line prettier/prettier, @typescript-eslint/no-explicit-any size={avatarSize}
{...($props as any)} />
src={signing?.avatar} {signing?.name}
objectFit="cover" </NFlex>
round </NButton>
size={avatarSize}
/>
<div class="app-avatar__name">{signing?.name}</div>
</NFlex>
) )
}, },
}) })

View File

@ -12,7 +12,6 @@
/** 锁屏界面 */ /** 锁屏界面 */
import { NInput, NForm, NFormItem, NButton } from 'naive-ui' import { NInput, NForm, NFormItem, NButton } from 'naive-ui'
import AppAvatar from '@/app-components/app/AppAvatar'
import useAppLockScreen from '@/app-components/app/AppLockScreen/appLockVar' import useAppLockScreen from '@/app-components/app/AppLockScreen/appLockVar'
import { rules, useCondition } from '@/app-components/app/AppLockScreen/shared' import { rules, useCondition } from '@/app-components/app/AppLockScreen/shared'
@ -61,7 +60,6 @@ const LockScreen = defineComponent({
render() { render() {
return ( return (
<div class="app-lock-screen__input"> <div class="app-lock-screen__input">
<AppAvatar vertical align="center" avatarSize={52} />
<NForm <NForm
ref="formInstRef" ref="formInstRef"
model={this.lockCondition} model={this.lockCondition}

View File

@ -120,7 +120,7 @@ export default defineComponent({
</div> </div>
</div> </div>
<div class="app-lock-screen__unlock__content-avatar"> <div class="app-lock-screen__unlock__content-avatar">
<AppAvatar vertical align="center" avatarSize={52} /> <AppAvatar avatarSize={52} style="pointer-events: none;" />
</div> </div>
<div class="app-lock-screen__unlock__content-input"> <div class="app-lock-screen__unlock__content-input">
<NForm ref="formRef" model={this.lockCondition} rules={rules}> <NForm ref="formRef" model={this.lockCondition} rules={rules}>

View File

@ -31,6 +31,7 @@ import {
import { getNaiveLocales } from '@/locales/utils' import { getNaiveLocales } from '@/locales/utils'
import { useSettingGetters } from '@/store' import { useSettingGetters } from '@/store'
import { MESSAGE_PROVIDER } from '@/app-config'
export default defineComponent({ export default defineComponent({
name: 'GlobalProvider', name: 'GlobalProvider',
@ -92,7 +93,7 @@ export default defineComponent({
dateLocale={localePackage.dateLocal} dateLocale={localePackage.dateLocal}
> >
<NLoadingBarProvider> <NLoadingBarProvider>
<NMessageProvider> <NMessageProvider {...MESSAGE_PROVIDER}>
<NDialogProvider> <NDialogProvider>
<NModalProvider> <NModalProvider>
<NNotificationProvider> <NNotificationProvider>

View File

@ -13,6 +13,20 @@
import type { LayoutSideBarLogo, PreloadingConfig } from '@/types' import type { LayoutSideBarLogo, PreloadingConfig } from '@/types'
import type { AppMenuConfig, AppKeepAlive } from '@/types' import type { AppMenuConfig, AppKeepAlive } from '@/types'
import type { MessageProviderProps } from 'naive-ui'
/**
*
* @description
* MessageProver
* Message
*
* @see https://www.naiveui.com/zh-CN/dark/components/message#MessageProvider-Props
*/
export const MESSAGE_PROVIDER: MessageProviderProps = {
max: 5,
closable: false,
}
/** /**
* *
@ -57,12 +71,12 @@ export const PRE_LOADING_CONFIG: PreloadingConfig = {
/** /**
* *
* icon: LOGO , `RIcon` () * icon: LOGO `RIcon` ()4.8.2 VNode
* title: LOGO * title: LOGO
* url: 点击跳转地址, , * url: 点击跳转地址
* jumpType: 跳转类型(station: 项目内跳转, outsideStation: 新页面打开) * jumpType: 跳转类型(station: 项目内跳转 outsideStation: 新页面打开)
* *
* , LOGO * LOGO
*/ */
export const SIDE_BAR_LOGO: LayoutSideBarLogo | undefined = { export const SIDE_BAR_LOGO: LayoutSideBarLogo | undefined = {
icon: 'ray', icon: 'ray',

View File

@ -12,9 +12,11 @@
/** vue-router 相关配置入口 */ /** vue-router 相关配置入口 */
import type { LayoutInst } from 'naive-ui' import type { LayoutInst } from 'naive-ui'
import type { Ref } from 'vue'
/** /**
* *
* @description
* ref * ref
* *
* , 使(scrollTo) * , 使(scrollTo)
@ -22,13 +24,27 @@ import type { LayoutInst } from 'naive-ui'
* *
* , , dom * , , dom
* @example * @example
* ```ts
* nextTick().then(() => { * nextTick().then(() => {
* LAYOUT_CONTENT_REF.value?.scrollTo() * LAYOUT_CONTENT_REF.value?.scrollTo()
* }) * })
* ```
*/ */
export const LAYOUT_CONTENT_REF = ref<LayoutInst | null>(null) export const LAYOUT_CONTENT_REF: Readonly<Ref<LayoutInst | null>> =
ref<LayoutInst | null>(null)
/**
*
* @description
* ref
*
*
* 使使 nextTick() dom
* @example
* nextTick().then(() => {
* LAYOUT_SIDER_REF.value?.scrollTo({ top: 0 })
* })
*/
export const LAYOUT_SIDER_REF: Readonly<Ref<LayoutInst | null>> =
ref<LayoutInst | null>(null)
export const SETUP_ROUTER_ACTION = { export const SETUP_ROUTER_ACTION = {
/** 是否启用路由切换时顶部加载条 */ /** 是否启用路由切换时顶部加载条 */
@ -39,6 +55,7 @@ export const SETUP_ROUTER_ACTION = {
/** /**
* *
* @description
* () * ()
* *
* *
@ -50,6 +67,7 @@ export const WHITE_ROUTES: string[] = ['RLogin', 'ErrorPage', 'RayTemplateDoc']
/** /**
* *
* @description
* *
* , * ,
*/ */

View File

@ -0,0 +1,30 @@
import { axiosCanceler } from '@/axios/utils/interceptor'
import type { FetchFunction, FetchErrorFunction } from '@/axios/types'
/**
*
* @param ins
* @param mode
*
* @description
*
*/
const injectRequestCanceler: FetchFunction = (ins, mode) => {
axiosCanceler.removePendingRequest(ins) // 检查是否存在重复请求, 若存在则取消已发的请求
axiosCanceler.addPendingRequest(ins) // 把当前的请求信息添加到 pendingRequest 表中
}
/**
*
* @param error
* @param mode
*
* @description
*
*/
const requestErrorCanceler: FetchErrorFunction = (error, mode) => {
axiosCanceler.removePendingRequest(error) // 移除请求拦截器
}
export { injectRequestCanceler, requestErrorCanceler }

View File

@ -0,0 +1,39 @@
import { appendRequestHeaders } from '@/axios/utils/append-request-headers'
import { APP_CATCH_KEY } from '@/app-config'
import { getStorage } from '@/utils'
import type { RequestInterceptorConfig, FetchFunction } from '@/axios/types'
/**
*
* token token key value
*
*
*
* request instance ,
*/
const requestHeaderToken = (ins: RequestInterceptorConfig, mode: string) => {
const token = getStorage<string>(APP_CATCH_KEY.token)
if (ins.url) {
// TODO: 根据 url 不同是否设置 token
}
return {
key: 'X-TOKEN',
value: token,
}
}
/** 注入请求头信息 */
const injectRequestHeaders: FetchFunction = (ins, mode) => {
appendRequestHeaders(ins, [
requestHeaderToken(ins, mode),
{
key: 'Demo-Header-Key',
value: 'Demo Header Value',
},
])
}
export { injectRequestHeaders }

View File

@ -0,0 +1,39 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-06-06
*
* @workspace ray-template
*
* @remark
*/
/**
*
*
* ,
* ,
*
* , 便
*
* injectRequestCanceler requestErrorCanceler axios request interceptor
*/
import { injectRequestCanceler, requestErrorCanceler } from './plugins/cancel'
import { injectRequestHeaders } from './plugins/request-headers'
/**
*
*
*
*/
export default {
// 请求正常
implementRequestInterceptorArray: [
injectRequestHeaders,
injectRequestCanceler,
],
// 请求错误
implementRequestInterceptorErrorArray: [requestErrorCanceler],
}

View File

@ -0,0 +1,29 @@
import { axiosCanceler } from '@/axios/utils/interceptor'
import type { FetchFunction, FetchErrorFunction } from '@/axios/types'
/**
*
* @param ins
* @param mode
*
* @description
*
*/
const injectResponseCanceler: FetchFunction = (ins, mode) => {
axiosCanceler.removePendingRequest(ins)
}
/**
*
* @param error
* @param mode
*
* @description
*
*/
const responseErrorCanceler: FetchErrorFunction = (error, mode) => {
axiosCanceler.removePendingRequest(error)
}
export { injectResponseCanceler, responseErrorCanceler }

View File

@ -20,38 +20,7 @@
* injectResponseCanceler responseErrorCanceler axios response interceptor * injectResponseCanceler responseErrorCanceler axios response interceptor
*/ */
import { axiosCanceler } from '@/axios/utils/interceptor' import { injectResponseCanceler, responseErrorCanceler } from './plugins/cancel'
import type {
ResponseInterceptorConfig,
FetchFunction,
FetchErrorFunction,
} from '@/axios/types'
import type { Recordable } from '@/types'
/**
*
* @param ins
* @param mode
*
* @description
*
*/
const injectResponseCanceler: FetchFunction = (ins, mode) => {
axiosCanceler.removePendingRequest(ins)
}
/**
*
* @param error
* @param mode
*
* @description
*
*/
const responseErrorCanceler: FetchErrorFunction = (error, mode) => {
axiosCanceler.removePendingRequest(error)
}
/** /**
* *

View File

@ -1,105 +0,0 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-06-06
*
* @workspace ray-template
*
* @remark
*/
/**
*
*
* ,
* ,
*
* , 便
*
* injectRequestCanceler requestErrorCanceler axios request interceptor
*/
import { axiosCanceler } from '@/axios/utils/interceptor'
import { appendRequestHeaders } from '@/axios/utils/axios-copilot'
import { APP_CATCH_KEY } from '@/app-config'
import { getStorage } from '@/utils'
import type {
RequestInterceptorConfig,
FetchFunction,
FetchErrorFunction,
} from '@/axios/types'
import type { Recordable } from '@/types'
/**
*
* token token key value
*
*
*
* request instance ,
*/
const requestHeaderToken = (ins: RequestInterceptorConfig, mode: string) => {
const token = getStorage<string>(APP_CATCH_KEY.token)
if (ins.url) {
// TODO: 根据 url 不同是否设置 token
}
return {
key: 'X-TOKEN',
value: token,
}
}
/** 注入请求头信息 */
const injectRequestHeaders: FetchFunction = (ins, mode) => {
appendRequestHeaders(ins, [
requestHeaderToken(ins, mode),
{
key: 'Demo-Header-Key',
value: 'Demo Header Value',
},
])
}
/**
*
* @param ins
* @param mode
*
* @description
*
*/
const injectRequestCanceler: FetchFunction = (ins, mode) => {
axiosCanceler.removePendingRequest(ins) // 检查是否存在重复请求, 若存在则取消已发的请求
axiosCanceler.addPendingRequest(ins) // 把当前的请求信息添加到 pendingRequest 表中
}
/**
*
* @param error
* @param mode
*
* @description
*
*/
const requestErrorCanceler: FetchErrorFunction = (error, mode) => {
axiosCanceler.removePendingRequest(error) // 移除请求拦截器
}
/**
*
*
*
*/
export default {
// 请求正常
implementRequestInterceptorArray: [
injectRequestHeaders,
injectRequestCanceler,
],
// 请求错误
implementRequestInterceptorErrorArray: [requestErrorCanceler],
}

View File

@ -22,11 +22,11 @@ import { useAxiosInterceptor } from '@/axios/utils/interceptor'
import { import {
setupResponseInterceptor, setupResponseInterceptor,
setupResponseErrorInterceptor, setupResponseErrorInterceptor,
} from '@/axios/inject/response' } from '@/axios/axios-interceptor/response'
import { import {
setupRequestInterceptor, setupRequestInterceptor,
setupRequestErrorInterceptor, setupRequestErrorInterceptor,
} from '@/axios/inject/request' } from '@/axios/axios-interceptor/request'
import type { AxiosInstanceExpand } from './types' import type { AxiosInstanceExpand } from './types'

View File

@ -24,11 +24,26 @@ export interface RequestHeaderOptions {
} }
export interface CancelConfig { export interface CancelConfig {
/**
*
* @description
*
*/
cancel?: boolean cancel?: boolean
} }
export interface AppRawRequestConfig<T = any> extends AxiosRequestConfig<T> { export interface AppRawRequestConfig<T = any> extends AxiosRequestConfig<T> {
/**
*
* @description
*
*/
cancelConfig?: CancelConfig cancelConfig?: CancelConfig
/**
*
* @description
*
*/
__CANCELER_TAG_RAY_TEMPLATE__?: '__CANCELER_TAG_RAY_TEMPLATE__' __CANCELER_TAG_RAY_TEMPLATE__?: '__CANCELER_TAG_RAY_TEMPLATE__'
} }

View File

@ -18,6 +18,17 @@
import type { AppRawRequestConfig, CancelerParams } from '@/axios/types' import type { AppRawRequestConfig, CancelerParams } from '@/axios/types'
/**
*
* @class RequestCanceler
*
* @description
* signal
* generateRequestKey key
*
* cancelConfig.cancel true
* __CANCELER_TAG_RAY_TEMPLATE__
*/
export default class RequestCanceler { export default class RequestCanceler {
private pendingRequest: Map<string, AbortController> private pendingRequest: Map<string, AbortController>
@ -25,7 +36,19 @@ export default class RequestCanceler {
this.pendingRequest = new Map<string, AbortController>() this.pendingRequest = new Map<string, AbortController>()
} }
/** 是否需要加入取消请求表中 */ /**
*
* @param config config
*
* @description
* signal
*
* cancelConfig false signal
* cancelConfig true signal
*
* @example
* const bool = isAppending(config) // true or false
*/
private isAppending(config: AppRawRequestConfig | CancelerParams) { private isAppending(config: AppRawRequestConfig | CancelerParams) {
return config.cancelConfig?.cancel ?? true return config.cancelConfig?.cancel ?? true
} }
@ -33,9 +56,12 @@ export default class RequestCanceler {
/** /**
* *
* @param config config * @param config config
* @returns key
* *
* @remark config request key * @description
* key
*
* @example
* const key = generateRequestKey(config) // string
*/ */
private generateRequestKey(config: AppRawRequestConfig | CancelerParams) { private generateRequestKey(config: AppRawRequestConfig | CancelerParams) {
const { method, url } = config const { method, url } = config
@ -52,7 +78,15 @@ export default class RequestCanceler {
* *
* @param config axios request config * @param config axios request config
* *
* @remark signal , * @description
* pendingRequest map
* signal
*
* cancelConfig.cancel false
* __CANCELER_TAG_RAY_TEMPLATE__
*
* @example
* addPendingRequest(config)
*/ */
addPendingRequest(config: AppRawRequestConfig | CancelerParams) { addPendingRequest(config: AppRawRequestConfig | CancelerParams) {
if (this.isAppending(config)) { if (this.isAppending(config)) {
@ -77,7 +111,11 @@ export default class RequestCanceler {
* *
* @param config axios request config * @param config axios request config
* *
* @remark , map generateRequestKey value * @description
* pendingRequest map
*
* @example
* removePendingRequest(config)
*/ */
removePendingRequest(config: AppRawRequestConfig | CancelerParams) { removePendingRequest(config: AppRawRequestConfig | CancelerParams) {
const requestKey = this.generateRequestKey(config) const requestKey = this.generateRequestKey(config)
@ -88,7 +126,17 @@ export default class RequestCanceler {
} }
} }
/** 取消所有请求 */ /**
*
* @description
* pendingRequest map
*
*
* cancelConfig.cancel false
*
* @example
* cancelAllRequest()
*/
cancelAllRequest() { cancelAllRequest() {
this.pendingRequest.forEach((curr) => { this.pendingRequest.forEach((curr) => {
curr.abort() curr.abort()

View File

@ -0,0 +1,31 @@
import type { RawAxiosRequestHeaders, AxiosRequestConfig } from 'axios'
import type { RequestHeaderOptions } from '../types'
/**
*
* @param instance axios instance
* @param options axios headers options
*
* @description
* axios
*
* @example
* appendRequestHeaders(inst, [
* {
* key: 'Demo-Header-Key',
* value: 'Demo Header Value',
* }
* ])
*/
export const appendRequestHeaders = <T = unknown>(
instance: AxiosRequestConfig<T>,
options: RequestHeaderOptions[],
) => {
if (instance) {
const requestHeaders = instance.headers as RawAxiosRequestHeaders
options.forEach((curr) => {
requestHeaders[curr.key] = curr.value
})
}
}

View File

@ -54,6 +54,7 @@ const errorImplement: ErrorImplementQueue = {
implementRequestInterceptorErrorArray: [], implementRequestInterceptorErrorArray: [],
implementResponseInterceptorErrorArray: [], implementResponseInterceptorErrorArray: [],
} }
/** 取消器实例 */ /** 取消器实例 */
export const axiosCanceler = new RequestCanceler() export const axiosCanceler = new RequestCanceler()

View File

@ -15,7 +15,9 @@ import { NSpin } from 'naive-ui'
import barcode from 'jsbarcode' import barcode from 'jsbarcode'
import props from './props' import props from './props'
import { completeSize } from '@/utils' import { completeSize, call } from '@/utils'
import type { WatchStopHandle } from 'vue'
export default defineComponent({ export default defineComponent({
name: 'RBarcode', name: 'RBarcode',
@ -30,20 +32,54 @@ export default defineComponent({
return cssVar return cssVar
}) })
let watchStop: WatchStopHandle
const barcodeRender = () => { const barcodeRender = () => {
const { format, text = '', options } = props try {
const { format, text, options, onSuccess } = props
const assignOptions = Object.assign({}, options, { const assignOptions = Object.assign({}, options, {
format, format,
}) })
barcode(barcodeRef.value, text, assignOptions) barcode(
barcodeRef.value,
text !== void 0 && text !== null ? text.toString() : '',
assignOptions,
)
if (onSuccess) {
call(onSuccess, text, format, options)
}
} catch (e) {
const { onError } = props
if (onError) {
call(onError, e as Error)
}
} finally {
const { onFinally } = props
if (onFinally) {
call(onFinally)
}
}
} }
watchEffect(() => {
if (props.watchText) {
watchStop = watch(() => props.text, barcodeRender)
} else {
watchStop?.()
}
})
onMounted(() => { onMounted(() => {
barcodeRender() barcodeRender()
}) })
onBeforeUnmount(() => {
watchStop?.()
})
return { return {
barcodeRef, barcodeRef,

View File

@ -1,5 +1,6 @@
import type { RBarcodeRender, RBarcodeOptions, RBarcodeFormat } from './types' import type { RBarcodeRender, RBarcodeOptions, RBarcodeFormat } from './types'
import type { PropType } from 'vue' import type { PropType } from 'vue'
import type { MaybeArray } from '@/types'
const props = { const props = {
/** /**
@ -82,6 +83,57 @@ const props = {
type: String as PropType<RBarcodeFormat>, type: String as PropType<RBarcodeFormat>,
default: () => 'CODE128', default: () => 'CODE128',
}, },
/**
*
* @description
* text text
*
* 使text
*
* @default false
*/
watchText: {
type: Boolean,
default: false,
},
/**
*
* @description
*
*
* @default undefined
*/
onSuccess: {
type: [Function, Array] as PropType<
MaybeArray<
(
currentText: string | undefined,
format: RBarcodeFormat,
option: RBarcodeOptions,
) => void
>
>,
},
/**
*
* @description
*
*
* @default undefined
*/
onError: {
type: [Function, Array] as PropType<MaybeArray<(err: Error) => void>>,
},
/**
*
* @description
*
*
* @default undefined
*/
onFinally: {
type: [Function, Array] as PropType<MaybeArray<() => void>>,
},
} as const } as const
export default props export default props

View File

@ -437,6 +437,7 @@ export default defineComponent({
props.setChartOptions, props.setChartOptions,
defaultChartOptions, defaultChartOptions,
) )
// 如果 options 发生变动更新 echarts // 如果 options 发生变动更新 echarts
echartInst?.setOption(options, setOpt) echartInst?.setOption(options, setOpt)
}, },

View File

@ -161,7 +161,7 @@ const props = {
* @default 100% * @default 100%
*/ */
width: { width: {
type: String, type: [String, Number] as PropType<string | number>,
default: '100%', default: '100%',
}, },
/** /**
@ -174,7 +174,7 @@ const props = {
* @default 100% * @default 100%
*/ */
height: { height: {
type: String, type: [String, Number] as PropType<string | number>,
default: '100%', default: '100%',
}, },
/** /**

View File

@ -6,6 +6,7 @@ import type {
ShouldRuleBeApplied, ShouldRuleBeApplied,
RFormRules, RFormRules,
} from '../types' } from '../types'
import type { Recordable } from '@/types'
/** /**
* *
@ -35,7 +36,7 @@ import type {
* }, * },
* }) * })
*/ */
const useForm = <T extends Record<string, unknown>, R extends RFormRules>( const useForm = <T extends Recordable, R extends RFormRules>(
model?: T, model?: T,
rules?: R, rules?: R,
) => { ) => {

View File

@ -53,7 +53,7 @@ const readGIFAsArrayBuffer = (url: string): Promise<GIFBuffer> => {
} }
export default defineComponent({ export default defineComponent({
name: 'RayQRcode', name: 'RQrcode',
props, props,
setup(props, ctx) { setup(props, ctx) {
const { expose } = ctx const { expose } = ctx

View File

@ -75,6 +75,7 @@ export default defineComponent({
segmentWidthVar = '100%' segmentWidthVar = '100%'
break break
case 'fitContent': case 'fitContent':
segmentWidthVar = 'fit-content' segmentWidthVar = 'fit-content'

View File

@ -28,6 +28,7 @@ import type { C as CType, PropsComponentPopselectKeys } from './types'
export default defineComponent({ export default defineComponent({
name: 'RTable', name: 'RTable',
inheritAttrs: false,
props, props,
setup(props, ctx) { setup(props, ctx) {
const { expose, emit } = ctx const { expose, emit } = ctx
@ -241,15 +242,18 @@ export default defineComponent({
$slots, $slots,
propsPopselectValue, propsPopselectValue,
} = this } = this
const { class: className } = $attrs
const { tool, combineRowProps, contextMenuSelect } = this const { tool, combineRowProps, contextMenuSelect } = this
return ( return (
<NCard <NCard
ref="wrapperRef" {...$props.cardProps}
bordered={wrapperBordered}
{...{ {...{
id: uuidWrapper, id: uuidWrapper,
}} }}
ref="wrapperRef"
bordered={wrapperBordered}
class={className}
> >
{{ {{
default: () => ( default: () => (

View File

@ -18,11 +18,21 @@ import type {
DownloadCsvTableOptions, DownloadCsvTableOptions,
PrintTableOptions, PrintTableOptions,
RTableInst, RTableInst,
RTableCardProps,
} from './types' } from './types'
import type { Recordable } from '@/types' import type { Recordable } from '@/types'
const props = { const props = {
...dataTableProps, ...dataTableProps,
/**
*
* @description
* props NCard
*/
cardProps: {
type: Object as PropType<RTableCardProps>,
default: () => ({}),
},
/** /**
* *
* @description * @description

View File

@ -6,11 +6,16 @@ import type {
DataTableInst, DataTableInst,
DataTableColumn, DataTableColumn,
DataTableBaseColumn, DataTableBaseColumn,
CardProps,
} from 'naive-ui' } from 'naive-ui'
import type { VNode } from 'vue' import type { VNode, CSSProperties } from 'vue'
import type { Recordable } from '@/types' import type { Recordable } from '@/types'
import type { PrintDomOptions } from '@/utils/dom' import type { PrintDomOptions } from '@/utils/dom'
export interface RTableCardProps extends CardProps {
style?: CSSProperties
}
export type TableActionIcon = string | (() => VNode) export type TableActionIcon = string | (() => VNode)
export type DropdownMixedOption = export type DropdownMixedOption =

View File

@ -6,3 +6,5 @@ export * from './useSiderBar'
export * from './useAppNavigation' export * from './useAppNavigation'
export * from './useAppRoot' export * from './useAppRoot'
export * from './useBadge' export * from './useBadge'
export * from './useSiderScroll'
export * from './useContentScroll'

View File

@ -0,0 +1,28 @@
import { LAYOUT_CONTENT_REF } from '@/app-config'
/**
*
* @description
*
*
* @see https://www.naiveui.com/zh-CN/dark/components/layout#scroll-to.vue
*
* @example
* const contentScrollTo = useContentScroll()
*
* contentScrollTo({ top: 0, left: 0, behavior: 'smooth' }) // 滚动内容区域
* contentScrollTo(10, 10) // x, y 方向滚动到 10 位置
*/
export const useContentScroll = () => {
const contentInst = LAYOUT_CONTENT_REF.value
const { scrollTo: contentScrollTo } = contentInst || {}
if (!contentScrollTo) {
throw new Error(
`[useContentScroll]: LAYOUT_CONTENT_REF is not ready yet. please wait component mounted!`,
)
}
return contentScrollTo
}

View File

@ -0,0 +1,27 @@
import { LAYOUT_SIDER_REF } from '@/app-config'
/**
*
* @description
*
*
* @see https://www.naiveui.com/zh-CN/dark/components/layout#scroll-to.vue
*
* @example
* const siderScrollTo = useSiderScroll()
*
* siderScrollTo({ top: 0, behavior: 'smooth' }) // y 方向侧边栏滚动
* siderScrollTo(0, 0) // x, y 方向滚动到 0 位置
*/
export const useSiderScroll = () => {
const siderInst = LAYOUT_SIDER_REF.value
const { scrollTo: siderScrollTo } = siderInst || {}
if (!siderScrollTo) {
throw new Error(
`[useSiderScroll]: LAYOUT_SIDER_REF is not ready yet. please wait component mounted!`,
)
}
return siderScrollTo
}

View File

@ -1,4 +1,5 @@
import { omit } from 'lodash-es' import { omit } from 'lodash-es'
import { effectDispose } from '@/utils'
import type { AnyFC } from '@/types' import type { AnyFC } from '@/types'
import type { PaginationProps } from 'naive-ui' import type { PaginationProps } from 'naive-ui'
@ -32,16 +33,10 @@ const defaultOptions: UsePaginationOptions = {
* 便 hook * 便 hook
*/ */
export const usePagination = <T extends AnyFC>( export const usePagination = <T extends AnyFC>(
callback: T, callback?: T,
options?: UsePaginationOptions, options?: UsePaginationOptions,
) => { ) => {
if (typeof callback !== 'function') { const callbackRef = ref(callback)
throw new Error(
'[usePagination]: callback expected a function, but got ' +
typeof callback,
)
}
const omitOptions = omit(options, [ const omitOptions = omit(options, [
'on-update:page', 'on-update:page',
'on-update:page-size', 'on-update:page-size',
@ -54,13 +49,13 @@ export const usePagination = <T extends AnyFC>(
onUpdatePage: (page: number) => { onUpdatePage: (page: number) => {
paginationRef.value.page = page paginationRef.value.page = page
callback() callbackRef.value?.()
}, },
onUpdatePageSize: (pageSize: number) => { onUpdatePageSize: (pageSize: number) => {
paginationRef.value.pageSize = pageSize paginationRef.value.pageSize = pageSize
paginationRef.value.page = 1 paginationRef.value.page = 1
callback() callbackRef.value?.()
}, },
} }
const paginationRef = ref<PaginationProps>( const paginationRef = ref<PaginationProps>(
@ -77,7 +72,7 @@ export const usePagination = <T extends AnyFC>(
* @description * @description
* *
*/ */
const getItemCount = () => paginationRef.value.itemCount const getItemCount = () => paginationRef.value.itemCount as number
/** /**
* *
@ -95,7 +90,7 @@ export const usePagination = <T extends AnyFC>(
* @description * @description
* *
*/ */
const getPage = () => paginationRef.value.page const getPage = () => paginationRef.value.page as number
/** /**
* *
@ -115,7 +110,7 @@ export const usePagination = <T extends AnyFC>(
* @description * @description
* *
*/ */
const getPageSize = () => paginationRef.value.pageSize const getPageSize = () => paginationRef.value.pageSize as number
/** /**
* *
@ -144,6 +139,25 @@ export const usePagination = <T extends AnyFC>(
*/ */
const getCallback = callback const getCallback = callback
/**
*
* @param callback
*
* @description
*
*
* @example
* setCallback(() => {})
*/
const setCallback = (callback: T) => {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
callbackRef.value = callback as any
}
effectDispose(() => {
callbackRef.value = void 0
})
return [ return [
paginationRef as Ref<UsePaginationOptions>, paginationRef as Ref<UsePaginationOptions>,
{ {
@ -157,6 +171,7 @@ export const usePagination = <T extends AnyFC>(
setPageSize, setPageSize,
getPagination, getPagination,
getCallback, getCallback,
setCallback,
}, },
] as const ] as const
} }

View File

@ -11,9 +11,18 @@
import './index.scss' import './index.scss'
import { NEllipsis, NPopover } from 'naive-ui' import { NEllipsis, NTooltip } from 'naive-ui'
import { RIcon } from '@/components' import { RIcon } from '@/components'
import { isValueType } from '@/utils'
/**
*
* @description
* Logo ref
*/
export const SIDER_BAR_LOGO = ref<HTMLElement>()
export default defineComponent({ export default defineComponent({
name: 'SiderBarLogo', name: 'SiderBarLogo',
props: { props: {
@ -36,7 +45,7 @@ export default defineComponent({
* - station: 模板内跳转 * - station: 模板内跳转
* - outsideStation: 新开页面跳转 * - outsideStation: 新开页面跳转
*/ */
const handleSideBarLogoClick = () => { const sideBarLogoClick = () => {
if (sideBarLogo && sideBarLogo.url) { if (sideBarLogo && sideBarLogo.url) {
sideBarLogo.jumpType === 'station' sideBarLogo.jumpType === 'station'
? router.push(sideBarLogo.url) ? router.push(sideBarLogo.url)
@ -44,44 +53,72 @@ export default defineComponent({
} }
} }
const TemplateLogo = ({ cursor }: { cursor: string }) => ( const TemplateLogo = ({ cursor }: { cursor: string }) => {
<RIcon name={sideBarLogo!.icon as string} size="30" cursor={cursor} /> if (typeof sideBarLogo.icon === 'string') {
) return (
<RIcon name={sideBarLogo!.icon as string} size="30" cursor={cursor} />
)
}
if (isValueType<object>(sideBarLogo.icon, 'Object')) {
return <sideBarLogo.icon />
}
}
return { return {
sideBarLogo, sideBarLogo,
handleSideBarLogoClick, sideBarLogoClick,
TemplateLogo, TemplateLogo,
} }
}, },
render() { render() {
return this.sideBarLogo?.icon && this.sideBarLogo?.title ? ( const { sideBarLogo, collapsed, TemplateLogo, sideBarLogoClick } = this
return sideBarLogo?.title ? (
<div <div
class={[ class={[
'ray-menu__logo', 'ray-menu__logo',
this.sideBarLogo?.url ? 'ray-menu__logo-url' : null, sideBarLogo?.url ? 'ray-menu__logo-url' : null,
]} ]}
onClick={this.handleSideBarLogoClick.bind(this)} onClick={sideBarLogoClick.bind(this)}
ref={SIDER_BAR_LOGO}
> >
{this.sideBarLogo?.icon ? ( {sideBarLogo?.icon ? (
this.collapsed ? ( collapsed ? (
<NPopover placement="right"> <NTooltip placement="right">
{{ {{
trigger: () => <this.TemplateLogo cursor="pointer" />, trigger: () =>
default: () => this.sideBarLogo?.title, TemplateLogo({
cursor: 'pointer',
}),
default: () => sideBarLogo.title,
}} }}
</NPopover> </NTooltip>
) : ( ) : (
<this.TemplateLogo cursor="pointer" /> TemplateLogo({
cursor: 'pointer',
})
) )
) : collapsed ? (
<NTooltip placement="right">
{{
trigger: () => (
<h1 class="n-menu-item-content">
{sideBarLogo.title[0] || null}
</h1>
),
default: () => sideBarLogo.title,
}}
</NTooltip>
) : null} ) : null}
<h1 <h1
class={[ class={[
!this.collapsed ? 'ray-menu__logo-title--open' : null, !collapsed ? 'ray-menu__logo-title--open' : null,
'ray-menu__logo-title', 'ray-menu__logo-title',
'class="n-menu-item-content"',
]} ]}
> >
<NEllipsis>{this.sideBarLogo?.title}</NEllipsis> <NEllipsis>{sideBarLogo.title}</NEllipsis>
</h1> </h1>
</div> </div>
) : null ) : null

View File

@ -14,7 +14,7 @@ import './index.scss'
import { NMenu, NLayoutSider, NDrawer } from 'naive-ui' import { NMenu, NLayoutSider, NDrawer } from 'naive-ui'
import SiderBarLogo from './components/SiderBarLogo' import SiderBarLogo from './components/SiderBarLogo'
import { APP_MENU_CONFIG } from '@/app-config' import { APP_MENU_CONFIG, LAYOUT_SIDER_REF } from '@/app-config'
import { useDevice } from '@/hooks' import { useDevice } from '@/hooks'
import { getVariableToRefs, setVariable } from '@/global-variable' import { getVariableToRefs, setVariable } from '@/global-variable'
import { useMenuGetters, useMenuActions } from '@/store' import { useMenuGetters, useMenuActions } from '@/store'
@ -33,7 +33,16 @@ export default defineComponent({
const modelMenuKey = computed({ const modelMenuKey = computed({
get: () => { get: () => {
// eslint-disable-next-line vue/no-async-in-computed-properties
nextTick().then(() => { nextTick().then(() => {
/**
*
* @description
* eslint computed 使
* computed get
*
*
*/
showMenuOption() showMenuOption()
}) })
@ -55,14 +64,15 @@ export default defineComponent({
/** /**
* *
* * @description
*
*/ */
const showMenuOption = () => { const showMenuOption = () => {
const key = modelMenuKey.value as string const key = modelMenuKey.value
nextTick().then(() => { if (key !== void 0 && key !== null) {
menuRef.value?.showOption?.(key) nextTick(() => menuRef.value?.showOption?.(key))
}) }
} }
const BasicMenu = () => ( const BasicMenu = () => (
@ -73,6 +83,7 @@ export default defineComponent({
collapsedWidth={APP_MENU_CONFIG.menuCollapsedWidth} collapsedWidth={APP_MENU_CONFIG.menuCollapsedWidth}
onUpdateCollapsed={collapsedMenu.bind(this)} onUpdateCollapsed={collapsedMenu.bind(this)}
nativeScrollbar={false} nativeScrollbar={false}
ref={LAYOUT_SIDER_REF}
> >
<SiderBarLogo collapsed={getCollapsed.value} /> <SiderBarLogo collapsed={getCollapsed.value} />
<NMenu <NMenu

View File

@ -381,6 +381,11 @@ export default defineComponent({
positionMenuTag() positionMenuTag()
} }
} }
// 初始化时,滚动到当前激活标签页
if (odata === void 0) {
positionMenuTag()
}
}, },
{ {
immediate: true, immediate: true,

View File

@ -28,10 +28,16 @@ import {
NFlex, NFlex,
NSpin, NSpin,
NCard, NCard,
NText,
} from 'naive-ui' } from 'naive-ui'
import { RIcon } from '@/components' import { RIcon } from '@/components'
import { queryElements, setClass, removeClass } from '@/utils' import {
queryElements,
setClass,
removeClass,
positionSelectedMenuItem,
} from '@/utils'
import { throttle, pick } from 'lodash-es' import { throttle, pick } from 'lodash-es'
import { useMenuActions } from '@/store' import { useMenuActions } from '@/store'
import { validMenuItemShow } from '@/router/utils' import { validMenuItemShow } from '@/router/utils'
@ -91,6 +97,7 @@ export default defineComponent({
let preSearchElementIndex = searchElementIndex let preSearchElementIndex = searchElementIndex
const { isTabletOrSmaller } = useDevice() const { isTabletOrSmaller } = useDevice()
const loading = ref(false) const loading = ref(false)
const ACTIVE_CLASS = 'content-item--active' // 激活样式 class name
/** 初始化一些值 */ /** 初始化一些值 */
const resetSearchSomeValue = () => { const resetSearchSomeValue = () => {
@ -175,6 +182,7 @@ export default defineComponent({
modelShow.value = false modelShow.value = false
changeMenuModelValue(option.fullPath, option) changeMenuModelValue(option.fullPath, option)
setTimeout(positionSelectedMenuItem, 300)
} }
} }
} }
@ -183,7 +191,6 @@ export default defineComponent({
const autoFocusingSearchItem = () => { const autoFocusingSearchItem = () => {
const currentOption = state.searchOptions[searchElementIndex] // 获取当前搜索项 const currentOption = state.searchOptions[searchElementIndex] // 获取当前搜索项
const preOption = state.searchOptions[preSearchElementIndex] // 获取上一搜索项 const preOption = state.searchOptions[preSearchElementIndex] // 获取上一搜索项
const activeClass = 'content-item--active' // 激活样式 class name
if (currentOption) { if (currentOption) {
nextTick().then(() => { nextTick().then(() => {
@ -197,13 +204,13 @@ export default defineComponent({
if (preSearchElementOptions?.length) { if (preSearchElementOptions?.length) {
const [el] = preSearchElementOptions const [el] = preSearchElementOptions
removeClass(el, activeClass) removeClass(el, ACTIVE_CLASS)
} }
if (searchElementOptions?.length) { if (searchElementOptions?.length) {
const [el] = searchElementOptions const [el] = searchElementOptions
setClass(el, activeClass) setClass(el, ACTIVE_CLASS)
} }
}) })
} }
@ -225,13 +232,19 @@ export default defineComponent({
/** 更新索引 */ /** 更新索引 */
const updateIndex = (type: 'up' | 'down') => { const updateIndex = (type: 'up' | 'down') => {
if (type === 'up') { if (type === 'up') {
searchElementIndex = searchElementIndex -= 1
searchElementIndex - 1 < 0 ? 0 : searchElementIndex - 1
} else if (type === 'down') { if (searchElementIndex < 0) {
searchElementIndex = searchElementIndex = state.searchOptions.length - 1
searchElementIndex + 1 >= state.searchOptions.length }
? state.searchOptions.length - 1 }
: searchElementIndex + 1
if (type === 'down') {
searchElementIndex += 1
if (searchElementIndex >= state.searchOptions.length) {
searchElementIndex = 0
}
} }
} }
@ -256,10 +269,12 @@ export default defineComponent({
updateIndex('up') updateIndex('up')
break break
case 'ArrowDown': case 'ArrowDown':
updateIndex('down') updateIndex('down')
break break
case 'Enter': case 'Enter':
// eslint-disable-next-line no-case-declarations // eslint-disable-next-line no-case-declarations
const option = state.searchOptions[searchElementIndex] const option = state.searchOptions[searchElementIndex]
@ -386,7 +401,7 @@ export default defineComponent({
justify="center" justify="center"
class="global-search__empty-content" class="global-search__empty-content"
> >
<NText></NText>
</NFlex> </NFlex>
), ),
}} }}

View File

@ -1,8 +1,8 @@
import TooltipIcon from './TooltipIcon' import TooltipIcon from './TooltipIcon'
import SettingDrawer from './SettingDrawer' import SettingDrawer from './SettingDrawer'
import Breadcrumb from './Breadcrumb' import Breadcrumb from './Breadcrumb'
import GlobalSearch from './GlobalSearch' import GlobalSearch from '../../Search/GlobalSearch'
import GlobalSearchButton from './GlobalSearchButton' import GlobalSearchButton from '../../Search/GlobalSearchButton'
export { export {
TooltipIcon, TooltipIcon,

View File

@ -16,3 +16,7 @@
.ray-template--dark .layout-header { .ray-template--dark .layout-header {
box-shadow: 0 1px 2px $layoutShadowColorDark; box-shadow: 0 1px 2px $layoutShadowColorDark;
} }
.override-button__layout {
padding: 0 9px;
}

View File

@ -19,7 +19,7 @@
import './index.scss' import './index.scss'
import { NLayoutHeader, NFlex, NDropdown } from 'naive-ui' import { NLayoutHeader, NFlex, NDropdown, NButton, NScrollbar } from 'naive-ui'
import { RIcon } from '@/components' import { RIcon } from '@/components'
import { import {
TooltipIcon, TooltipIcon,
@ -146,8 +146,9 @@ export default defineComponent({
class="layout-header__method" class="layout-header__method"
align="center" align="center"
justify="space-between" justify="space-between"
wrap={false}
> >
<NFlex align="center"> <NFlex align="center" wrap={false}>
{leftIconOptions.map((curr) => ( {leftIconOptions.map((curr) => (
<TooltipIcon <TooltipIcon
key={curr.name} key={curr.name}
@ -161,7 +162,7 @@ export default defineComponent({
))} ))}
{getBreadcrumbSwitch ? <Breadcrumb /> : null} {getBreadcrumbSwitch ? <Breadcrumb /> : null}
</NFlex> </NFlex>
<NFlex align="center" size={[16, 0]}> <NFlex justify="end" align="center" size={[0, 0]} wrap={false}>
{isRenderVNode( {isRenderVNode(
<GlobalSearchButton <GlobalSearchButton
onClick={(e) => { onClick={(e) => {
@ -172,34 +173,40 @@ export default defineComponent({
/>, />,
)} )}
{rightTooltipIconOptions.map((curr) => ( {rightTooltipIconOptions.map((curr) => (
<TooltipIcon <NButton
class="override-button__layout"
quaternary
size="medium"
key={curr.name} key={curr.name}
iconName={curr.name}
tooltipText={
isRef(curr.tooltip) ? curr.tooltip.value : curr.tooltip
}
customClassName={curr.iconClass}
onClick={toolIconClick.bind(this, curr.name)} onClick={toolIconClick.bind(this, curr.name)}
/> >
<RIcon name={curr.name} size={18} cursor="pointer" />
</NButton>
))} ))}
<NDropdown <NDropdown
options={LOCAL_OPTIONS} options={LOCAL_OPTIONS}
onSelect={(key: string | number) => updateLocale(String(key))} onSelect={(key: string | number) => updateLocale(String(key))}
trigger="click" trigger="click"
> >
<RIcon <NButton quaternary class="override-button__layout">
customClassName="layout-header__method--icon" <RIcon
name="language" customClassName="layout-header__method--icon"
size="18" name="language"
cursor="pointer" size="18"
/> cursor="pointer"
/>
</NButton>
</NDropdown> </NDropdown>
<NDropdown <NDropdown
options={createAvatarOptions()} options={createAvatarOptions()}
onSelect={avatarDropdownClick.bind(this)} onSelect={avatarDropdownClick.bind(this)}
trigger="click" trigger="click"
> >
<AppAvatar avatarSize="small" align="center" cursor="pointer" /> <AppAvatar
avatarSize={24}
spaceSize={[8, 0]}
class="override-button__layout"
/>
</NDropdown> </NDropdown>
</NFlex> </NFlex>
</NFlex> </NFlex>

View File

@ -74,15 +74,18 @@ export default defineComponent({
globalMainLayoutLoad, globalMainLayoutLoad,
layoutContentMaximize, layoutContentMaximize,
layoutContentSpinning, layoutContentSpinning,
maximize,
spinning,
themeOverridesSpin,
getContentTransition,
} = this } = this
const { maximize } = this
return ( return (
<NSpin <NSpin
show={this.spinning || !globalMainLayoutLoad || layoutContentSpinning} show={spinning || !globalMainLayoutLoad || layoutContentSpinning}
description="loading..." description="loading..."
size="large" size="large"
themeOverrides={this.themeOverridesSpin} themeOverrides={themeOverridesSpin}
class={[ class={[
layoutContentMaximize layoutContentMaximize
? 'r-layout-full__viewer-content--maximize' ? 'r-layout-full__viewer-content--maximize'
@ -105,7 +108,7 @@ export default defineComponent({
{globalMainLayoutLoad ? ( {globalMainLayoutLoad ? (
<RTransitionComponent <RTransitionComponent
class="content-wrapper" class="content-wrapper"
transitionPropName={this.getContentTransition + '-transform'} transitionPropName={getContentTransition + '-transform'}
/> />
) : null} ) : null}
</NSpin> </NSpin>

View File

@ -55,7 +55,7 @@ export default defineComponent({
const { getLockAppScreen } = this const { getLockAppScreen } = this
return !getLockAppScreen() ? ( return !getLockAppScreen() ? (
<NLayout class="r-layout-full" style={[cssVarsRef]} hasSider> <NLayout class="r-layout-full" style={[cssVarsRef]} hasSider embedded>
<Menu /> <Menu />
<NLayoutContent class="r-layout-full__viewer"> <NLayoutContent class="r-layout-full__viewer">
<HeaderWrapper ref="layoutSiderBarRef" /> <HeaderWrapper ref="layoutSiderBarRef" />

View File

@ -5,7 +5,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const dashboard: AppRouteRecordRaw = { const dashboard: AppRouteRecordRaw = {
path: '/dashboard', path: '/dashboard',
name: 'RDashboard',
component: () => import('@/views/dashboard'), component: () => import('@/views/dashboard'),
meta: { meta: {
i18nKey: t('menu.Dashboard'), i18nKey: t('menu.Dashboard'),

View File

@ -5,7 +5,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const barcode: AppRouteRecordRaw = { const barcode: AppRouteRecordRaw = {
path: 'barcode', path: 'barcode',
name: 'Barcode',
component: () => import('@/views/demo/BarcodeDemo'), component: () => import('@/views/demo/BarcodeDemo'),
meta: { meta: {
i18nKey: t('menu.Barcode'), i18nKey: t('menu.Barcode'),

View File

@ -5,7 +5,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const cacheDemo: AppRouteRecordRaw = { const cacheDemo: AppRouteRecordRaw = {
path: '/cache-demo', path: '/cache-demo',
name: 'CacheDemo',
component: () => import('@/views/demo/cache-demo/index'), component: () => import('@/views/demo/cache-demo/index'),
meta: { meta: {
i18nKey: t('menu.CacheDemo'), i18nKey: t('menu.CacheDemo'),

View File

@ -4,7 +4,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const contextMenu: AppRouteRecordRaw = { const contextMenu: AppRouteRecordRaw = {
path: '/context-menu', path: '/context-menu',
name: 'ContextMenuDemo',
component: () => import('@/views/demo/context-menu/index'), component: () => import('@/views/demo/context-menu/index'),
meta: { meta: {
i18nKey: t('menu.ContextMenu'), i18nKey: t('menu.ContextMenu'),

View File

@ -5,7 +5,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const directive: AppRouteRecordRaw = { const directive: AppRouteRecordRaw = {
path: '/directive', path: '/directive',
name: 'RDirective',
component: () => import('@/views/demo/directive/index'), component: () => import('@/views/demo/directive/index'),
meta: { meta: {
i18nKey: t('menu.Directive'), i18nKey: t('menu.Directive'),

View File

@ -5,7 +5,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const echart: AppRouteRecordRaw = { const echart: AppRouteRecordRaw = {
path: '/echart', path: '/echart',
name: 'REchart',
component: () => import('@/views/demo/echart/index'), component: () => import('@/views/demo/echart/index'),
meta: { meta: {
i18nKey: t('menu.Echart'), i18nKey: t('menu.Echart'),

View File

@ -5,7 +5,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const form: AppRouteRecordRaw = { const form: AppRouteRecordRaw = {
path: '/form', path: '/form',
name: 'FormView',
component: () => import('@/views/demo/form'), component: () => import('@/views/demo/form'),
meta: { meta: {
i18nKey: t('menu.Form'), i18nKey: t('menu.Form'),

View File

@ -4,7 +4,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const iframe: AppRouteRecordRaw = { const iframe: AppRouteRecordRaw = {
path: '/iframe', path: '/iframe',
name: 'IframeDemo',
component: () => import('@/views/demo/iframe/index'), component: () => import('@/views/demo/iframe/index'),
meta: { meta: {
icon: 'other', icon: 'other',

View File

@ -5,7 +5,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const mockDemo: AppRouteRecordRaw = { const mockDemo: AppRouteRecordRaw = {
path: '/mock-demo', path: '/mock-demo',
name: 'MockDemo',
component: () => import('@/views/demo/mock-demo/index'), component: () => import('@/views/demo/mock-demo/index'),
meta: { meta: {
i18nKey: t('menu.Mock'), i18nKey: t('menu.Mock'),

View File

@ -5,7 +5,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const mockDemo: AppRouteRecordRaw = { const mockDemo: AppRouteRecordRaw = {
path: '/modal-demo', path: '/modal-demo',
name: 'ModalDemo',
component: () => import('@/views/demo/modal-demo/index'), component: () => import('@/views/demo/modal-demo/index'),
meta: { meta: {
i18nKey: t('menu.Modal'), i18nKey: t('menu.Modal'),

View File

@ -5,7 +5,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const multiMenu: AppRouteRecordRaw = { const multiMenu: AppRouteRecordRaw = {
path: '/multi', path: '/multi',
name: 'MultiMenu',
component: LAYOUT, component: LAYOUT,
meta: { meta: {
i18nKey: t('menu.MultiMenu'), i18nKey: t('menu.MultiMenu'),

View File

@ -5,7 +5,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const precision: AppRouteRecordRaw = { const precision: AppRouteRecordRaw = {
path: '/precision', path: '/precision',
name: 'CalculatePrecision',
component: () => import('@/views/demo/precision/index'), component: () => import('@/views/demo/precision/index'),
meta: { meta: {
i18nKey: t('menu.CalculatePrecision'), i18nKey: t('menu.CalculatePrecision'),

View File

@ -5,7 +5,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const qrcode: AppRouteRecordRaw = { const qrcode: AppRouteRecordRaw = {
path: '/qrcode', path: '/qrcode',
name: 'RQRCode',
component: () => import('@/views/demo/qrcode/index'), component: () => import('@/views/demo/qrcode/index'),
meta: { meta: {
i18nKey: t('menu.QRCode'), i18nKey: t('menu.QRCode'),

View File

@ -5,7 +5,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const rely: AppRouteRecordRaw = { const rely: AppRouteRecordRaw = {
path: 'rely-about', path: 'rely-about',
name: 'RelyAbout',
component: () => import('@/views/demo/rely/views/rely-about/index'), component: () => import('@/views/demo/rely/views/rely-about/index'),
meta: { meta: {
i18nKey: t('menu.RelyAbout'), i18nKey: t('menu.RelyAbout'),

View File

@ -5,7 +5,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const routerDemo: AppRouteRecordRaw = { const routerDemo: AppRouteRecordRaw = {
path: '/router-demo', path: '/router-demo',
name: 'RouterDemoRoot',
component: LAYOUT, component: LAYOUT,
meta: { meta: {
i18nKey: t('menu.RouterDemo'), i18nKey: t('menu.RouterDemo'),
@ -15,7 +14,6 @@ const routerDemo: AppRouteRecordRaw = {
children: [ children: [
{ {
path: 'router-demo-home', path: 'router-demo-home',
name: 'RouterDemoHome',
component: () => component: () =>
import('@/views/demo/router-demo/router-demo-home/index'), import('@/views/demo/router-demo/router-demo-home/index'),
meta: { meta: {
@ -24,7 +22,6 @@ const routerDemo: AppRouteRecordRaw = {
}, },
{ {
path: 'router-demo-detail', path: 'router-demo-detail',
name: 'RouterDemoDetail',
component: () => component: () =>
import('@/views/demo/router-demo/router-demo-detail/index'), import('@/views/demo/router-demo/router-demo-detail/index'),
meta: { meta: {

View File

@ -11,7 +11,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const scrollReveal: AppRouteRecordRaw = { const scrollReveal: AppRouteRecordRaw = {
path: '/scroll-reveal', path: '/scroll-reveal',
name: 'ScrollReveal',
component: () => import('@/views/demo/scroll-reveal/index'), component: () => import('@/views/demo/scroll-reveal/index'),
meta: { meta: {
i18nKey: t('menu.scrollReveal'), i18nKey: t('menu.scrollReveal'),

View File

@ -5,7 +5,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const segment: AppRouteRecordRaw = { const segment: AppRouteRecordRaw = {
path: '/segment', path: '/segment',
name: 'RAxios',
component: () => import('@/views/demo/segment'), component: () => import('@/views/demo/segment'),
meta: { meta: {
i18nKey: t('menu.Segment'), i18nKey: t('menu.Segment'),

View File

@ -5,7 +5,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const previewSVGIcons: AppRouteRecordRaw = { const previewSVGIcons: AppRouteRecordRaw = {
path: '/svg-icons', path: '/svg-icons',
name: 'PreviewSVGIcons',
component: () => import('@/views/demo/svg-icons/index'), component: () => import('@/views/demo/svg-icons/index'),
meta: { meta: {
i18nKey: t('menu.SvgIcon'), i18nKey: t('menu.SvgIcon'),

View File

@ -5,7 +5,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const table: AppRouteRecordRaw = { const table: AppRouteRecordRaw = {
path: '/table', path: '/table',
name: 'TableView',
component: () => import('@/views/demo/table/index'), component: () => import('@/views/demo/table/index'),
meta: { meta: {
i18nKey: t('menu.Table'), i18nKey: t('menu.Table'),

View File

@ -5,7 +5,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const axios: AppRouteRecordRaw = { const axios: AppRouteRecordRaw = {
path: '/template-hooks', path: '/template-hooks',
name: 'TemplateHooks',
component: () => import('@/views/demo/template-hooks/index'), component: () => import('@/views/demo/template-hooks/index'),
meta: { meta: {
i18nKey: t('menu.TemplateHooks'), i18nKey: t('menu.TemplateHooks'),

View File

@ -5,7 +5,6 @@ import type { AppRouteRecordRaw } from '@/router/types'
const error404: AppRouteRecordRaw = { const error404: AppRouteRecordRaw = {
path: '/:catchAll(.*)', path: '/:catchAll(.*)',
name: 'ErrorPage',
component: () => import('@/views/error/views/Error404'), component: () => import('@/views/error/views/Error404'),
meta: { meta: {
i18nKey: t('menu.Error'), i18nKey: t('menu.Error'),

View File

@ -11,39 +11,202 @@ export type Component<T = any> =
| (() => Promise<T>) | (() => Promise<T>)
export interface AppMenuExtraOptions { export interface AppMenuExtraOptions {
/**
*
* @description
*
*/
label?: string | number label?: string | number
/**
*
* @description
*
*/
icon?: VNode icon?: VNode
/**
*
* @description
*
*/
type?: TagProps['type'] type?: TagProps['type']
/**
*
* @description
*
*/
show?: boolean show?: boolean
/**
*
* @description
* i18n
*/
i18nLabel?: string i18nLabel?: string
} }
export interface AppRouteMeta { export interface AppRouteMeta {
/**
*
* @description
* i18n
*/
i18nKey?: string i18nKey?: string
/**
*
* @description
*
* string使 RIcon icons
* VNode 使
*/
icon?: string | VNode icon?: string | VNode
/**
*
* @description
*
*/
windowOpen?: string windowOpen?: string
/**
*
* @description
*
*
*/
role?: (string | number)[] role?: (string | number)[]
/**
*
* @description
*
* 404, 500
*/
hidden?: boolean hidden?: boolean
/**
*
* @description
*
*/
noLocalTitle?: string | number noLocalTitle?: string | number
/**
*
* @description
*
*
*/
ignoreAutoResetScroll?: boolean ignoreAutoResetScroll?: boolean
/**
*
* @description
*
*/
order?: number order?: number
/**
*
* @description
*
*
*
* @see https://xiaodaigua-ray.github.io/ray-template-doc/ray-template-docs/common-problem/keep-alive.html
*/
keepAlive?: boolean keepAlive?: boolean
/**
*
* @description
*
* true MenuTag
*/
sameLevel?: boolean sameLevel?: boolean
/**
*
* @description
*
*
*/
env?: string | string[] env?: string | string[]
/**
*
* @description
*
*/
extra?: AppMenuExtraOptions extra?: AppMenuExtraOptions
} }
// @ts-ignore // @ts-ignore
export interface AppRouteRecordRaw extends Omit<RouteRecordRaw, 'meta'> { interface BaseAppRouteRecordRaw extends Omit<RouteRecordRaw, 'meta'> {
name: string /**
*
* @description
* name vue-router 使
*
*
*
*/
name?: string
/**
*
* @description
*
*/
meta: AppRouteMeta meta: AppRouteMeta
/**
*
* @description
*
* 使 LAYOUT
*
* @example
* {
* path: '/dashboard',
* component: LAYOUT,
* children: [ ... ]
* }
*/
component?: Component | string component?: Component | string
/**
*
* @description
*
*
* @deprecated
* 使
*/
components?: Component components?: Component
/**
*
* @description
*
*/
children?: AppRouteRecordRaw[] children?: AppRouteRecordRaw[]
/**
*
* @description
* props
*/
props?: Recordable props?: Recordable
/**
*
* @description
* search hash
*
*/
fullPath?: string fullPath?: string
} }
interface KeepAliveAppRouteRecordRaw extends BaseAppRouteRecordRaw {
meta: AppRouteMeta & {
keepAlive: true
}
name: string
}
interface NonKeepAliveAppRouteRecordRaw extends BaseAppRouteRecordRaw {
meta: AppRouteMeta & {
keepAlive?: false
}
name?: string
}
export type AppRouteRecordRaw =
| KeepAliveAppRouteRecordRaw
| NonKeepAliveAppRouteRecordRaw
export interface RouteModules { export interface RouteModules {
[propName: string]: { [propName: string]: {
default: AppRouteRecordRaw default: AppRouteRecordRaw

View File

@ -4,12 +4,14 @@ import type { AppRouteRecordRaw, RouteModules } from '@/router/types'
* *
* @returns * @returns
* *
* @remark , ts route module views * @description
* ts route module views
* modules ts 使
* *
* , modules , import.meta.glob * modules import.meta.glob
* URL * URL
* *
* modules ts , 使 * name
*/ */
export const combineRawRouteModules = () => { export const combineRawRouteModules = () => {
const modulesFiles: RouteModules = import.meta.glob( const modulesFiles: RouteModules = import.meta.glob(
@ -19,18 +21,18 @@ export const combineRawRouteModules = () => {
}, },
) )
const modules = Object.keys(modulesFiles).reduce((modules, modulePath) => { const modules = Object.keys(modulesFiles).reduce((pre, curr) => {
const route = modulesFiles[modulePath].default const module = modulesFiles[curr].default
if (route) { if (module) {
modules.push(route) pre.push(module)
} else { } else {
throw new Error( throw new Error(
'router helper combine: an exception occurred while parsing the routing file!', `[combineRawRouteModules]: ${curr} module must export default.`,
) )
} }
return modules return pre
}, [] as AppRouteRecordRaw[]) }, [] as AppRouteRecordRaw[])
return modules return modules

View File

@ -332,6 +332,7 @@ export const piniaMenuStore = defineStore(
let fullPath = `${ let fullPath = `${
parentPath.endsWith('/') ? parentPath : parentPath + '/' parentPath.endsWith('/') ? parentPath : parentPath + '/'
}${curr.path}` }${curr.path}`
// 使用正则表达式替换重复的 '/' // 使用正则表达式替换重复的 '/'
fullPath = fullPath.replace(/\/+/g, '/') fullPath = fullPath.replace(/\/+/g, '/')

View File

@ -1,9 +1,9 @@
import type { VNode } from 'vue' import type { VNode } from 'vue'
import type { AppRouteRecordRaw, AppRouteMeta } from '@/router/types' import type { AppRouteMeta } from '@/router/types'
export type Key = string | number export type Key = string | number
export interface AppMenuOption extends AppRouteRecordRaw { export interface AppMenuOption {
name: string name: string
key: Key key: Key
path: string path: string

View File

@ -7,9 +7,10 @@ import type {
} from 'vite' } from 'vite'
import type { Recordable } from '@/types' import type { Recordable } from '@/types'
import type { GlobalThemeOverrides } from 'naive-ui' import type { GlobalThemeOverrides } from 'naive-ui'
import type { VNode } from 'vue'
export interface LayoutSideBarLogo { export interface LayoutSideBarLogo {
icon?: string icon?: string | VNode
title?: string title?: string
url?: string url?: string
jumpType?: 'station' | 'outsideStation' jumpType?: 'station' | 'outsideStation'

View File

@ -162,6 +162,7 @@ export const downloadAnyFile = (
try { try {
if (typeof data === 'string') { if (typeof data === 'string') {
downloadBase64File(data, fileName) downloadBase64File(data, fileName)
return resolve() return resolve()
} }
@ -200,6 +201,8 @@ export const downloadAnyFile = (
document.body.appendChild(link) document.body.appendChild(link)
link.click() link.click()
return resolve()
} catch (error) { } catch (error) {
return reject(error) return reject(error)
} }

View File

@ -171,6 +171,7 @@ const removeStorage: RemoveStorageFC = (key, storageType, options) => {
console.error( console.error(
`[removeStorage]: Failed to remove stored data: key ${key} is empty or undefined`, `[removeStorage]: Failed to remove stored data: key ${key} is empty or undefined`,
) )
return return
} }
@ -185,6 +186,7 @@ const removeStorage: RemoveStorageFC = (key, storageType, options) => {
: removeType === 'localStorage' : removeType === 'localStorage'
? localStorageKeys ? localStorageKeys
: sessionStorageKeys : sessionStorageKeys
keys.forEach((curr) => { keys.forEach((curr) => {
if (key === '__all__') { if (key === '__all__') {
window.sessionStorage.removeItem(_prefix + curr) window.sessionStorage.removeItem(_prefix + curr)

View File

@ -1,3 +1,5 @@
import { positionSelectedMenuItem } from './position-selected-menu-item'
export * from './basic' export * from './basic'
export * from './cache' export * from './cache'
export * from './dom' export * from './dom'
@ -5,3 +7,4 @@ export * from './element'
export * from './precision' export * from './precision'
export * from './vue' export * from './vue'
export * from './app' export * from './app'
export { positionSelectedMenuItem }

View File

@ -0,0 +1,55 @@
import { LAYOUT_SIDER_REF } from '@/app-config'
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'
/**
*
* @description
* Sider
*
* NMenu
*
*
* 使 setTimeout
*
* @example
* positionSelectedMenuItem() // 滚动到当前激活菜单项的位置
*/
export const positionSelectedMenuItem = () => {
const siderEl = unrefElement(LAYOUT_SIDER_REF as Ref<HTMLElement>)
const selectedEl = siderEl?.querySelector<HTMLElement>(MENU_ITEM_SELECTED)
const siderBarLogoEl = unrefElement(SIDER_BAR_LOGO)
let siderBarLogoHeight: number = 0
const menuItemEl = siderEl?.querySelector<HTMLElement>(MENU_ITEM)
if (siderBarLogoEl) {
const { height } = siderBarLogoEl.getBoundingClientRect()
siderBarLogoHeight = height
}
if (selectedEl && siderEl) {
const siderScroll = useSiderScroll()
const { top: siderTop } = siderEl.getBoundingClientRect()
const { top: selectedTop } = selectedEl.getBoundingClientRect()
const siderScrollTop = siderEl.scrollTop
const menuItemMarginTop = menuItemEl
? parseInt(window.getComputedStyle(menuItemEl).marginTop)
: 6
siderScroll({
top:
selectedTop -
siderTop +
siderScrollTop -
siderBarLogoHeight -
menuItemMarginTop,
left: 0,
behavior: 'smooth',
})
}
}

View File

@ -10,7 +10,15 @@
*/ */
import { RBarcode } from '@/components' import { RBarcode } from '@/components'
import { NAlert, NCard, NFlex, NGrid, NGridItem, NSwitch } from 'naive-ui' import {
NAlert,
NCard,
NFlex,
NGrid,
NGridItem,
NInput,
NSwitch,
} from 'naive-ui'
export default defineComponent({ export default defineComponent({
name: 'BarcodeDemo', name: 'BarcodeDemo',
@ -19,10 +27,12 @@ export default defineComponent({
width: 4, width: 4,
} }
const loading = ref(false) const loading = ref(false)
const text = ref('RayTemplate')
return { return {
baseOptions, baseOptions,
loading, loading,
text,
} }
}, },
render() { render() {
@ -119,6 +129,14 @@ export default defineComponent({
</NFlex> </NFlex>
</NCard> </NCard>
</NGridItem> </NGridItem>
<NGridItem span={1}>
<NCard title="watchText 主动监听 text 变化">
<NFlex vertical>
<NInput v-model:value={this.text} />
<RBarcode text={this.text} watchText />
</NFlex>
</NCard>
</NGridItem>
</NGrid> </NGrid>
) )
}, },

View File

@ -22,7 +22,7 @@ import {
NRadioGroup, NRadioGroup,
} from 'naive-ui' } from 'naive-ui'
import { useForm } from '@/components' import { useForm, useModal } from '@/components'
import type { RFormRules } from '@/components' import type { RFormRules } from '@/components'
@ -138,6 +138,8 @@ export default defineComponent({
type="info" type="info"
onClick={() => { onClick={() => {
this.condition = formModel() this.condition = formModel()
restoreValidation()
}} }}
> >

View File

@ -158,7 +158,7 @@ const MockDemo = defineComponent({
</> </>
), ),
action: () => ( action: () => (
<NButton type="primary" onClick={this.getCallback.bind(this)}> <NButton type="primary" onClick={this.getCallback?.bind(this)}>
</NButton> </NButton>
), ),

View File

@ -47,6 +47,7 @@ const CalculatePrecision = defineComponent({
) )
}) })
} }
updateDistributeValue() updateDistributeValue()
return { return {

View File

@ -69,11 +69,7 @@
"useAttrs": true, "useAttrs": true,
"useCssModule": true, "useCssModule": true,
"useCssVars": true, "useCssVars": true,
"useDialog": true,
"useLink": true, "useLink": true,
"useLoadingBar": true,
"useMessage": true,
"useNotification": true,
"useRoute": true, "useRoute": true,
"useRouter": true, "useRouter": true,
"useSlots": true, "useSlots": true,
@ -81,7 +77,6 @@
"watchEffect": true, "watchEffect": true,
"watchPostEffect": true, "watchPostEffect": true,
"watchSyncEffect": true, "watchSyncEffect": true,
"NEllipsis": true,
"ExtractDefaultPropTypes": true, "ExtractDefaultPropTypes": true,
"ExtractPropTypes": true, "ExtractPropTypes": true,
"ExtractPublicPropTypes": true "ExtractPublicPropTypes": true

View File

@ -6,7 +6,6 @@
export {} export {}
declare global { declare global {
const EffectScope: typeof import('vue')['EffectScope'] const EffectScope: typeof import('vue')['EffectScope']
const NEllipsis: typeof import('naive-ui')['NEllipsis']
const acceptHMRUpdate: typeof import('pinia')['acceptHMRUpdate'] const acceptHMRUpdate: typeof import('pinia')['acceptHMRUpdate']
const computed: typeof import('vue')['computed'] const computed: typeof import('vue')['computed']
const createApp: typeof import('vue')['createApp'] const createApp: typeof import('vue')['createApp']
@ -67,11 +66,7 @@ declare global {
const useAttrs: typeof import('vue')['useAttrs'] const useAttrs: typeof import('vue')['useAttrs']
const useCssModule: typeof import('vue')['useCssModule'] const useCssModule: typeof import('vue')['useCssModule']
const useCssVars: typeof import('vue')['useCssVars'] const useCssVars: typeof import('vue')['useCssVars']
const useDialog: typeof import('naive-ui')['useDialog']
const useLink: typeof import('vue-router')['useLink'] const useLink: typeof import('vue-router')['useLink']
const useLoadingBar: typeof import('naive-ui')['useLoadingBar']
const useMessage: typeof import('naive-ui')['useMessage']
const useNotification: typeof import('naive-ui')['useNotification']
const useRoute: typeof import('vue-router')['useRoute'] const useRoute: typeof import('vue-router')['useRoute']
const useRouter: typeof import('vue-router')['useRouter'] const useRouter: typeof import('vue-router')['useRouter']
const useSlots: typeof import('vue')['useSlots'] const useSlots: typeof import('vue')['useSlots']

View File

@ -47,7 +47,7 @@ import type { BuildOptions } from 'vite'
const config: AppConfigExport = { const config: AppConfigExport = {
/** 公共基础路径配置, 如果为空则会默认以 '/' 填充 */ /** 公共基础路径配置, 如果为空则会默认以 '/' 填充 */
base: '/ray-template/', // base: '/ray-template/',
/** 配置首屏加载信息 */ /** 配置首屏加载信息 */
preloadingConfig: PRE_LOADING_CONFIG, preloadingConfig: PRE_LOADING_CONFIG,
/** 默认主题色(不可省略, 必填), 也用于 ejs 注入 */ /** 默认主题色(不可省略, 必填), 也用于 ejs 注入 */

View File

@ -17,7 +17,7 @@ import viteVeI18nPlugin from '@intlify/unplugin-vue-i18n/vite'
import viteInspect from 'vite-plugin-inspect' import viteInspect from 'vite-plugin-inspect'
import viteSvgLoader from 'vite-svg-loader' import viteSvgLoader from 'vite-svg-loader'
import vitePluginImp from 'vite-plugin-imp' import vitePluginImp from 'vite-plugin-imp'
import { analyzer } from 'vite-bundle-analyzer' import { analyzer, adapter } from 'vite-bundle-analyzer'
import viteCompression from 'vite-plugin-compression' import viteCompression from 'vite-plugin-compression'
import { ViteEjsPlugin as viteEjsPlugin } from 'vite-plugin-ejs' import { ViteEjsPlugin as viteEjsPlugin } from 'vite-plugin-ejs'
import viteAutoImport from 'unplugin-auto-import/vite' import viteAutoImport from 'unplugin-auto-import/vite'
@ -25,15 +25,19 @@ import viteEslint from 'vite-plugin-eslint'
import mockDevServerPlugin from 'vite-plugin-mock-dev-server' import mockDevServerPlugin from 'vite-plugin-mock-dev-server'
import { createSvgIconsPlugin } from 'vite-plugin-svg-icons' import { createSvgIconsPlugin } from 'vite-plugin-svg-icons'
import unpluginViteComponents from 'unplugin-vue-components/vite' import unpluginViteComponents from 'unplugin-vue-components/vite'
import { cdn as viteCDNPlugin } from 'vite-plugin-cdn2'
import { cdnResolve, svgIconResolve } from './vite-helper' import { cdn as viteCDNPlugin } from 'vite-plugin-cdn2'
import { NaiveUiResolver } from 'unplugin-vue-components/resolvers' import { NaiveUiResolver } from 'unplugin-vue-components/resolvers'
import { cdnResolve, svgIconResolve } from './vite-helper'
import config from './vite.custom.config' import config from './vite.custom.config'
import type { PluginOption } from 'vite' import type { PluginOption } from 'vite'
function pathResolve(dir: string) {
return path.resolve(__dirname, dir)
}
// 仅适用于报告模式report // 仅适用于报告模式report
function onlyReportOptions(mode: string): PluginOption[] { function onlyReportOptions(mode: string): PluginOption[] {
if (mode !== 'report') { if (mode !== 'report') {
@ -41,10 +45,12 @@ function onlyReportOptions(mode: string): PluginOption[] {
} }
return [ return [
analyzer({ adapter(
analyzerMode: 'server', // 以默认服务器代理打开文件 analyzer({
openAnalyzer: true, // 以默认服务器代理打开文件 analyzerMode: 'server', // 以默认服务器代理打开文件
}), openAnalyzer: true, // 以默认服务器代理打开文件
}),
),
] ]
} }
@ -107,7 +113,7 @@ function onlyBuildOptions(mode: string): PluginOption[] {
// 仅适用于开发模式 // 仅适用于开发模式
function onlyDevOptions(mode: string): PluginOption[] { function onlyDevOptions(mode: string): PluginOption[] {
if (mode === 'development') { if (mode !== 'development') {
return [] return []
} }
@ -127,7 +133,7 @@ function baseOptions(mode: string): PluginOption[] {
compositionOnly: true, compositionOnly: true,
forceStringify: true, forceStringify: true,
defaultSFCLang: 'json', defaultSFCLang: 'json',
include: [path.resolve(__dirname, '../locales/**')], include: [pathResolve('../locales/**')],
}), }),
viteAutoImport({ viteAutoImport({
eslintrc: { eslintrc: {
@ -141,19 +147,7 @@ function baseOptions(mode: string): PluginOption[] {
/\.md$/, // .md /\.md$/, // .md
], ],
dts: './unplugin/auto-imports.d.ts', dts: './unplugin/auto-imports.d.ts',
imports: [ imports: ['vue', 'vue-router', 'pinia'],
'vue',
'vue-router',
'pinia',
{
'naive-ui': [
'useDialog',
'useMessage',
'useNotification',
'useLoadingBar',
],
},
],
resolvers: [NaiveUiResolver()], resolvers: [NaiveUiResolver()],
}), }),
unpluginViteComponents({ unpluginViteComponents({

View File

@ -9,11 +9,11 @@ export default defineConfig((configEnv) =>
defineConfig({ defineConfig({
plugins: [tsconfigPaths()], plugins: [tsconfigPaths()],
test: { test: {
include: ['**/__test__/**/*.(spec).(ts|tsx)'], include: ['./__test__/**/*.(spec).(ts|tsx)'],
exclude: [ exclude: [
...configDefaults.exclude, ...configDefaults.exclude,
'**/src/**', '**/src/**',
'**/__test__/utils/**/*', './__test__/utils/**/*',
], ],
environment: 'happy-dom', environment: 'happy-dom',
globals: true, globals: true,