mirror of
https://github.com/XiaoDaiGua-Ray/ray-template.git
synced 2025-04-06 03:57:49 +08:00
version: v4.7.4
This commit is contained in:
parent
03628890cb
commit
a0f7763778
1
.vscode/settings.json
vendored
1
.vscode/settings.json
vendored
@ -25,6 +25,7 @@
|
|||||||
"domtoimage",
|
"domtoimage",
|
||||||
"EDITMSG",
|
"EDITMSG",
|
||||||
"iife",
|
"iife",
|
||||||
|
"internalkey",
|
||||||
"linebreak",
|
"linebreak",
|
||||||
"macarons",
|
"macarons",
|
||||||
"menutag",
|
"menutag",
|
||||||
|
113
CHANGELOG.md
113
CHANGELOG.md
@ -1,5 +1,118 @@
|
|||||||
# CHANGE LOG
|
# CHANGE LOG
|
||||||
|
|
||||||
|
## 4.7.4
|
||||||
|
|
||||||
|
对于 `RTable`, `RForm`, `RChart` 组件都新增了对应的 `useTable`, `useForm`, `useChart` 方法;让你在业务开发中抛弃注册 `ref` 实例方式调用组件方法。
|
||||||
|
|
||||||
|
补充拓展了 `useModal` 方法,支持 `dad`, `fullscreen` 等拓展配置项。
|
||||||
|
|
||||||
|
```ts
|
||||||
|
import { useTable, useForm } from '@/components'
|
||||||
|
|
||||||
|
const [registerTable, { getTableInstance }] = useTable()
|
||||||
|
const [registerForm, { getFormInstance }] = useForm()
|
||||||
|
|
||||||
|
// 做点什么...
|
||||||
|
```
|
||||||
|
|
||||||
|
## Feats
|
||||||
|
|
||||||
|
- 更新 `vite` 版本至 `5.2.6`
|
||||||
|
- `useDevice` 方法支持自定义 `media` 配置项,用于配置自定义媒体查询尺寸
|
||||||
|
- `RTable` 组件
|
||||||
|
- 新增 `tool` 配置项,配置是否显示工具栏
|
||||||
|
- 优化工具栏渲染逻辑
|
||||||
|
- 新增 `useTable` 方法,用于便捷调用表格方法
|
||||||
|
|
||||||
|
> 该方法比起常见的 `ref` 注册,然后 `tableRef.value.xxx` 的方法获取表格方法更为简洁一点。但是也值得注意的是,需要手动调用一次 `register` 方法,否则会报错;还有值得注意的是,需要注意表格方法的调用时机,需要等待表格注册完成后才能正常调用。如果需要在 `Parent Create` 阶段调用,可以尝试 `nextTick` 包裹一层。
|
||||||
|
|
||||||
|
```tsx
|
||||||
|
import { RTable } from '@/components'
|
||||||
|
import { useTable } from '@/components'
|
||||||
|
|
||||||
|
defineComponent({
|
||||||
|
setup() {
|
||||||
|
const [
|
||||||
|
register,
|
||||||
|
{ getTableInstance, clearFilters, clearSorter, scrollTo, filters, sort },
|
||||||
|
] = useTable()
|
||||||
|
|
||||||
|
const columns = [
|
||||||
|
{
|
||||||
|
title: 'No',
|
||||||
|
key: 'no',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Title',
|
||||||
|
key: 'title',
|
||||||
|
},
|
||||||
|
]
|
||||||
|
const data = [
|
||||||
|
{
|
||||||
|
no: 1,
|
||||||
|
title: 'title',
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
return {
|
||||||
|
register,
|
||||||
|
getTableInstance,
|
||||||
|
clearFilters,
|
||||||
|
clearSorter,
|
||||||
|
scrollTo,
|
||||||
|
filters,
|
||||||
|
sort,
|
||||||
|
columns,
|
||||||
|
data,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
render() {
|
||||||
|
const { columns, data } = this
|
||||||
|
const { register } = this
|
||||||
|
|
||||||
|
return (
|
||||||
|
<RTable columns={columns} data={data} register={register.bind(this)} />
|
||||||
|
)
|
||||||
|
},
|
||||||
|
})
|
||||||
|
```
|
||||||
|
|
||||||
|
- `RForm` 组件
|
||||||
|
- 新增组件,所有行为、方法与 `NForm` 保持一致
|
||||||
|
- `useForm` 方法,使用方法与 `useTable` 几乎一致
|
||||||
|
- `canUseDom`, `isBrowser` 方法统一为函数导出
|
||||||
|
- `RModal` 组件新增 `useModal` 方法
|
||||||
|
- 新增 `useModal` 方法,允许拓展配置 `dad`, `fullscreen` 等配置项。但是由于 `useModal` 生成组件的特殊性,不支持 `memo` 属性配置,其余配置项维持一致
|
||||||
|
> 该方法在当前版本存在一个 bug,`preset = card` 时,不能正确的显示 content,查看相应的 [issues](https://github.com/tusen-ai/naive-ui/issues/5746)。
|
||||||
|
- 重写部分代码,优化组件逻辑,补全 `ts` 类型
|
||||||
|
- `RChart`
|
||||||
|
- 新增 `useChart` 方法,使用方法与 `useTable` 几乎一致
|
||||||
|
- 新增 `usePagination` 方法与 `usePagination.spec` 单元测试模块
|
||||||
|
|
||||||
|
```ts
|
||||||
|
import { usePagination } from '@/hooks'
|
||||||
|
|
||||||
|
const {
|
||||||
|
updatePage,
|
||||||
|
updatePageSize,
|
||||||
|
getItemCount,
|
||||||
|
setItemCount,
|
||||||
|
getPage,
|
||||||
|
setPage,
|
||||||
|
getPageSize,
|
||||||
|
setPageSize,
|
||||||
|
getPagination,
|
||||||
|
getCallback,
|
||||||
|
} = usePagination(
|
||||||
|
() => {
|
||||||
|
// do something...
|
||||||
|
},
|
||||||
|
{
|
||||||
|
// ...options
|
||||||
|
},
|
||||||
|
)
|
||||||
|
```
|
||||||
|
|
||||||
## 4.7.3
|
## 4.7.3
|
||||||
|
|
||||||
补全 `hooks` 包下的单测模块。
|
补全 `hooks` 包下的单测模块。
|
||||||
|
71
__test__/hooks/usePagination.spec.ts
Normal file
71
__test__/hooks/usePagination.spec.ts
Normal file
@ -0,0 +1,71 @@
|
|||||||
|
import { usePagination } from '../../src/hooks/web/usePagination'
|
||||||
|
|
||||||
|
describe('usePagination', () => {
|
||||||
|
let count = 0
|
||||||
|
const defaultOptions = {
|
||||||
|
itemCount: 200,
|
||||||
|
page: 1,
|
||||||
|
pageSize: 10,
|
||||||
|
}
|
||||||
|
|
||||||
|
const {
|
||||||
|
getItemCount,
|
||||||
|
getCallback,
|
||||||
|
getPage,
|
||||||
|
getPageSize,
|
||||||
|
getPagination,
|
||||||
|
setItemCount,
|
||||||
|
setPage,
|
||||||
|
setPageSize,
|
||||||
|
} = usePagination(() => {
|
||||||
|
count++
|
||||||
|
}, defaultOptions)
|
||||||
|
|
||||||
|
it('should get current itemCount', () => {
|
||||||
|
setItemCount(200)
|
||||||
|
|
||||||
|
expect(getItemCount()).toBe(200)
|
||||||
|
|
||||||
|
setItemCount(100)
|
||||||
|
|
||||||
|
expect(getItemCount()).toBe(100)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should get current page', () => {
|
||||||
|
setPage(1)
|
||||||
|
|
||||||
|
expect(getPage()).toBe(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should get current pageSize', () => {
|
||||||
|
setPageSize(10)
|
||||||
|
|
||||||
|
expect(getPageSize()).toBe(10)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should get current pagination', () => {
|
||||||
|
setItemCount(200)
|
||||||
|
|
||||||
|
expect(getPagination()).toMatchObject(defaultOptions)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should update count when page or pageSize changes', () => {
|
||||||
|
count = 0
|
||||||
|
|
||||||
|
setPage(2)
|
||||||
|
|
||||||
|
expect(count).toBe(1)
|
||||||
|
|
||||||
|
setPageSize(20)
|
||||||
|
|
||||||
|
expect(count).toBe(2)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should get callback', () => {
|
||||||
|
count = 0
|
||||||
|
|
||||||
|
getCallback()
|
||||||
|
|
||||||
|
expect(count).toBe(1)
|
||||||
|
})
|
||||||
|
})
|
@ -12,4 +12,5 @@ const canUseDom = () => {
|
|||||||
window.document.createElement
|
window.document.createElement
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
export default canUseDom
|
export default canUseDom
|
||||||
|
@ -5,9 +5,11 @@
|
|||||||
*
|
*
|
||||||
* 如果是浏览器环境,则返回 true,否则返回 false。
|
* 如果是浏览器环境,则返回 true,否则返回 false。
|
||||||
*/
|
*/
|
||||||
const isBrowser = !!(
|
const isBrowser = () =>
|
||||||
typeof window !== 'undefined' &&
|
!!(
|
||||||
window.document &&
|
typeof window !== 'undefined' &&
|
||||||
window.document.createElement
|
window.document &&
|
||||||
)
|
window.document.createElement
|
||||||
|
)
|
||||||
|
|
||||||
export default isBrowser
|
export default isBrowser
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "ray-template",
|
"name": "ray-template",
|
||||||
"private": false,
|
"private": false,
|
||||||
"version": "4.7.3",
|
"version": "4.7.4",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": "^18.0.0 || >=20.0.0",
|
"node": "^18.0.0 || >=20.0.0",
|
||||||
@ -96,7 +96,7 @@
|
|||||||
"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.1.6",
|
"vite": "^5.2.6",
|
||||||
"vite-bundle-analyzer": "0.8.1",
|
"vite-bundle-analyzer": "0.8.1",
|
||||||
"vite-plugin-cdn2": "1.1.0",
|
"vite-plugin-cdn2": "1.1.0",
|
||||||
"vite-plugin-compression": "^0.5.1",
|
"vite-plugin-compression": "^0.5.1",
|
||||||
|
72
pnpm-lock.yaml
generated
72
pnpm-lock.yaml
generated
@ -108,10 +108,10 @@ devDependencies:
|
|||||||
version: 6.21.0(eslint@8.57.0)(typescript@5.2.2)
|
version: 6.21.0(eslint@8.57.0)(typescript@5.2.2)
|
||||||
'@vitejs/plugin-vue':
|
'@vitejs/plugin-vue':
|
||||||
specifier: ^5.0.4
|
specifier: ^5.0.4
|
||||||
version: 5.0.4(vite@5.2.2)(vue@3.4.21)
|
version: 5.0.4(vite@5.2.6)(vue@3.4.21)
|
||||||
'@vitejs/plugin-vue-jsx':
|
'@vitejs/plugin-vue-jsx':
|
||||||
specifier: ^3.1.0
|
specifier: ^3.1.0
|
||||||
version: 3.1.0(vite@5.2.2)(vue@3.4.21)
|
version: 3.1.0(vite@5.2.6)(vue@3.4.21)
|
||||||
'@vitest/ui':
|
'@vitest/ui':
|
||||||
specifier: 1.4.0
|
specifier: 1.4.0
|
||||||
version: 1.4.0(vitest@1.4.0)
|
version: 1.4.0(vitest@1.4.0)
|
||||||
@ -191,8 +191,8 @@ devDependencies:
|
|||||||
specifier: ^0.26.0
|
specifier: ^0.26.0
|
||||||
version: 0.26.0(vue@3.4.21)
|
version: 0.26.0(vue@3.4.21)
|
||||||
vite:
|
vite:
|
||||||
specifier: ^5.1.6
|
specifier: ^5.2.6
|
||||||
version: 5.2.2(@types/node@20.5.1)(sass@1.71.1)
|
version: 5.2.6(@types/node@20.5.1)(sass@1.71.1)
|
||||||
vite-bundle-analyzer:
|
vite-bundle-analyzer:
|
||||||
specifier: 0.8.1
|
specifier: 0.8.1
|
||||||
version: 0.8.1
|
version: 0.8.1
|
||||||
@ -201,31 +201,31 @@ devDependencies:
|
|||||||
version: 1.1.0
|
version: 1.1.0
|
||||||
vite-plugin-compression:
|
vite-plugin-compression:
|
||||||
specifier: ^0.5.1
|
specifier: ^0.5.1
|
||||||
version: 0.5.1(vite@5.2.2)
|
version: 0.5.1(vite@5.2.6)
|
||||||
vite-plugin-ejs:
|
vite-plugin-ejs:
|
||||||
specifier: ^1.7.0
|
specifier: ^1.7.0
|
||||||
version: 1.7.0(vite@5.2.2)
|
version: 1.7.0(vite@5.2.6)
|
||||||
vite-plugin-eslint:
|
vite-plugin-eslint:
|
||||||
specifier: 1.8.1
|
specifier: 1.8.1
|
||||||
version: 1.8.1(eslint@8.57.0)(vite@5.2.2)
|
version: 1.8.1(eslint@8.57.0)(vite@5.2.6)
|
||||||
vite-plugin-imp:
|
vite-plugin-imp:
|
||||||
specifier: ^2.4.0
|
specifier: ^2.4.0
|
||||||
version: 2.4.0(vite@5.2.2)
|
version: 2.4.0(vite@5.2.6)
|
||||||
vite-plugin-inspect:
|
vite-plugin-inspect:
|
||||||
specifier: ^0.8.3
|
specifier: ^0.8.3
|
||||||
version: 0.8.3(vite@5.2.2)
|
version: 0.8.3(vite@5.2.6)
|
||||||
vite-plugin-mock-dev-server:
|
vite-plugin-mock-dev-server:
|
||||||
specifier: 1.4.7
|
specifier: 1.4.7
|
||||||
version: 1.4.7(vite@5.2.2)
|
version: 1.4.7(vite@5.2.6)
|
||||||
vite-plugin-svg-icons:
|
vite-plugin-svg-icons:
|
||||||
specifier: ^2.0.1
|
specifier: ^2.0.1
|
||||||
version: 2.0.1(vite@5.2.2)
|
version: 2.0.1(vite@5.2.6)
|
||||||
vite-svg-loader:
|
vite-svg-loader:
|
||||||
specifier: ^4.0.0
|
specifier: ^4.0.0
|
||||||
version: 4.0.0
|
version: 4.0.0
|
||||||
vite-tsconfig-paths:
|
vite-tsconfig-paths:
|
||||||
specifier: 4.3.2
|
specifier: 4.3.2
|
||||||
version: 4.3.2(typescript@5.2.2)(vite@5.2.2)
|
version: 4.3.2(typescript@5.2.2)(vite@5.2.6)
|
||||||
vitest:
|
vitest:
|
||||||
specifier: 1.4.0
|
specifier: 1.4.0
|
||||||
version: 1.4.0(@types/node@20.5.1)(@vitest/ui@1.4.0)(happy-dom@14.3.1)(sass@1.71.1)
|
version: 1.4.0(@types/node@20.5.1)(@vitest/ui@1.4.0)(happy-dom@14.3.1)(sass@1.71.1)
|
||||||
@ -1816,7 +1816,7 @@ packages:
|
|||||||
resolution: {integrity: sha512-zuVdFrMJiuCDQUMCzQaD6KL28MjnqqN8XnAqiEq9PNm/hCPTSGfrXCOfwj1ow4LFb/tNymJPwsNbVePc1xFqrQ==}
|
resolution: {integrity: sha512-zuVdFrMJiuCDQUMCzQaD6KL28MjnqqN8XnAqiEq9PNm/hCPTSGfrXCOfwj1ow4LFb/tNymJPwsNbVePc1xFqrQ==}
|
||||||
dev: true
|
dev: true
|
||||||
|
|
||||||
/@vitejs/plugin-vue-jsx@3.1.0(vite@5.2.2)(vue@3.4.21):
|
/@vitejs/plugin-vue-jsx@3.1.0(vite@5.2.6)(vue@3.4.21):
|
||||||
resolution: {integrity: sha512-w9M6F3LSEU5kszVb9An2/MmXNxocAnUb3WhRr8bHlimhDrXNt6n6D2nJQR3UXpGlZHh/EsgouOHCsM8V3Ln+WA==}
|
resolution: {integrity: sha512-w9M6F3LSEU5kszVb9An2/MmXNxocAnUb3WhRr8bHlimhDrXNt6n6D2nJQR3UXpGlZHh/EsgouOHCsM8V3Ln+WA==}
|
||||||
engines: {node: ^14.18.0 || >=16.0.0}
|
engines: {node: ^14.18.0 || >=16.0.0}
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
@ -1826,20 +1826,20 @@ packages:
|
|||||||
'@babel/core': 7.24.1
|
'@babel/core': 7.24.1
|
||||||
'@babel/plugin-transform-typescript': 7.23.6(@babel/core@7.24.1)
|
'@babel/plugin-transform-typescript': 7.23.6(@babel/core@7.24.1)
|
||||||
'@vue/babel-plugin-jsx': 1.2.2(@babel/core@7.24.1)
|
'@vue/babel-plugin-jsx': 1.2.2(@babel/core@7.24.1)
|
||||||
vite: 5.2.2(@types/node@20.5.1)(sass@1.71.1)
|
vite: 5.2.6(@types/node@20.5.1)(sass@1.71.1)
|
||||||
vue: 3.4.21(typescript@5.2.2)
|
vue: 3.4.21(typescript@5.2.2)
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
dev: true
|
dev: true
|
||||||
|
|
||||||
/@vitejs/plugin-vue@5.0.4(vite@5.2.2)(vue@3.4.21):
|
/@vitejs/plugin-vue@5.0.4(vite@5.2.6)(vue@3.4.21):
|
||||||
resolution: {integrity: sha512-WS3hevEszI6CEVEx28F8RjTX97k3KsrcY6kvTg7+Whm5y3oYvcqzVeGCU3hxSAn4uY2CLCkeokkGKpoctccilQ==}
|
resolution: {integrity: sha512-WS3hevEszI6CEVEx28F8RjTX97k3KsrcY6kvTg7+Whm5y3oYvcqzVeGCU3hxSAn4uY2CLCkeokkGKpoctccilQ==}
|
||||||
engines: {node: ^18.0.0 || >=20.0.0}
|
engines: {node: ^18.0.0 || >=20.0.0}
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
vite: ^5.0.0
|
vite: ^5.0.0
|
||||||
vue: ^3.2.25
|
vue: ^3.2.25
|
||||||
dependencies:
|
dependencies:
|
||||||
vite: 5.2.2(@types/node@20.5.1)(sass@1.71.1)
|
vite: 5.2.6(@types/node@20.5.1)(sass@1.71.1)
|
||||||
vue: 3.4.21(typescript@5.2.2)
|
vue: 3.4.21(typescript@5.2.2)
|
||||||
dev: true
|
dev: true
|
||||||
|
|
||||||
@ -7886,7 +7886,7 @@ packages:
|
|||||||
debug: 4.3.4
|
debug: 4.3.4
|
||||||
pathe: 1.1.2
|
pathe: 1.1.2
|
||||||
picocolors: 1.0.0
|
picocolors: 1.0.0
|
||||||
vite: 5.2.2(@types/node@20.5.1)(sass@1.71.1)
|
vite: 5.2.6(@types/node@20.5.1)(sass@1.71.1)
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- '@types/node'
|
- '@types/node'
|
||||||
- less
|
- less
|
||||||
@ -7912,7 +7912,7 @@ packages:
|
|||||||
- supports-color
|
- supports-color
|
||||||
dev: true
|
dev: true
|
||||||
|
|
||||||
/vite-plugin-compression@0.5.1(vite@5.2.2):
|
/vite-plugin-compression@0.5.1(vite@5.2.6):
|
||||||
resolution: {integrity: sha512-5QJKBDc+gNYVqL/skgFAP81Yuzo9R+EAf19d+EtsMF/i8kFUpNi3J/H01QD3Oo8zBQn+NzoCIFkpPLynoOzaJg==}
|
resolution: {integrity: sha512-5QJKBDc+gNYVqL/skgFAP81Yuzo9R+EAf19d+EtsMF/i8kFUpNi3J/H01QD3Oo8zBQn+NzoCIFkpPLynoOzaJg==}
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
vite: '>=2.0.0'
|
vite: '>=2.0.0'
|
||||||
@ -7920,21 +7920,21 @@ packages:
|
|||||||
chalk: 4.1.2
|
chalk: 4.1.2
|
||||||
debug: 4.3.4
|
debug: 4.3.4
|
||||||
fs-extra: 10.1.0
|
fs-extra: 10.1.0
|
||||||
vite: 5.2.2(@types/node@20.5.1)(sass@1.71.1)
|
vite: 5.2.6(@types/node@20.5.1)(sass@1.71.1)
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
dev: true
|
dev: true
|
||||||
|
|
||||||
/vite-plugin-ejs@1.7.0(vite@5.2.2):
|
/vite-plugin-ejs@1.7.0(vite@5.2.6):
|
||||||
resolution: {integrity: sha512-JNP3zQDC4mSbfoJ3G73s5mmZITD8NGjUmLkq4swxyahy/W0xuokK9U9IJGXw7KCggq6UucT6hJ0p+tQrNtqTZw==}
|
resolution: {integrity: sha512-JNP3zQDC4mSbfoJ3G73s5mmZITD8NGjUmLkq4swxyahy/W0xuokK9U9IJGXw7KCggq6UucT6hJ0p+tQrNtqTZw==}
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
vite: '>=5.0.0'
|
vite: '>=5.0.0'
|
||||||
dependencies:
|
dependencies:
|
||||||
ejs: 3.1.9
|
ejs: 3.1.9
|
||||||
vite: 5.2.2(@types/node@20.5.1)(sass@1.71.1)
|
vite: 5.2.6(@types/node@20.5.1)(sass@1.71.1)
|
||||||
dev: true
|
dev: true
|
||||||
|
|
||||||
/vite-plugin-eslint@1.8.1(eslint@8.57.0)(vite@5.2.2):
|
/vite-plugin-eslint@1.8.1(eslint@8.57.0)(vite@5.2.6):
|
||||||
resolution: {integrity: sha512-PqdMf3Y2fLO9FsNPmMX+//2BF5SF8nEWspZdgl4kSt7UvHDRHVVfHvxsD7ULYzZrJDGRxR81Nq7TOFgwMnUang==}
|
resolution: {integrity: sha512-PqdMf3Y2fLO9FsNPmMX+//2BF5SF8nEWspZdgl4kSt7UvHDRHVVfHvxsD7ULYzZrJDGRxR81Nq7TOFgwMnUang==}
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
eslint: '>=7'
|
eslint: '>=7'
|
||||||
@ -7944,10 +7944,10 @@ packages:
|
|||||||
'@types/eslint': 8.56.6
|
'@types/eslint': 8.56.6
|
||||||
eslint: 8.57.0
|
eslint: 8.57.0
|
||||||
rollup: 2.79.1
|
rollup: 2.79.1
|
||||||
vite: 5.2.2(@types/node@20.5.1)(sass@1.71.1)
|
vite: 5.2.6(@types/node@20.5.1)(sass@1.71.1)
|
||||||
dev: true
|
dev: true
|
||||||
|
|
||||||
/vite-plugin-imp@2.4.0(vite@5.2.2):
|
/vite-plugin-imp@2.4.0(vite@5.2.6):
|
||||||
resolution: {integrity: sha512-L/6/nvOw+MyNh4UxAlCZHsmKd5MitmHamqqAWB15sbUgVIEz/OQ8jpKr6kkQU0eA/AIe8fkCVbQBlP81ajrqWg==}
|
resolution: {integrity: sha512-L/6/nvOw+MyNh4UxAlCZHsmKd5MitmHamqqAWB15sbUgVIEz/OQ8jpKr6kkQU0eA/AIe8fkCVbQBlP81ajrqWg==}
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
vite: '>= 2.0.0-beta.5'
|
vite: '>= 2.0.0-beta.5'
|
||||||
@ -7959,12 +7959,12 @@ packages:
|
|||||||
chalk: 4.1.2
|
chalk: 4.1.2
|
||||||
param-case: 3.0.4
|
param-case: 3.0.4
|
||||||
pascal-case: 3.1.2
|
pascal-case: 3.1.2
|
||||||
vite: 5.2.2(@types/node@20.5.1)(sass@1.71.1)
|
vite: 5.2.6(@types/node@20.5.1)(sass@1.71.1)
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
dev: true
|
dev: true
|
||||||
|
|
||||||
/vite-plugin-inspect@0.8.3(vite@5.2.2):
|
/vite-plugin-inspect@0.8.3(vite@5.2.6):
|
||||||
resolution: {integrity: sha512-SBVzOIdP/kwe6hjkt7LSW4D0+REqqe58AumcnCfRNw4Kt3mbS9pEBkch+nupu2PBxv2tQi69EQHQ1ZA1vgB/Og==}
|
resolution: {integrity: sha512-SBVzOIdP/kwe6hjkt7LSW4D0+REqqe58AumcnCfRNw4Kt3mbS9pEBkch+nupu2PBxv2tQi69EQHQ1ZA1vgB/Og==}
|
||||||
engines: {node: '>=14'}
|
engines: {node: '>=14'}
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
@ -7983,13 +7983,13 @@ packages:
|
|||||||
perfect-debounce: 1.0.0
|
perfect-debounce: 1.0.0
|
||||||
picocolors: 1.0.0
|
picocolors: 1.0.0
|
||||||
sirv: 2.0.4
|
sirv: 2.0.4
|
||||||
vite: 5.2.2(@types/node@20.5.1)(sass@1.71.1)
|
vite: 5.2.6(@types/node@20.5.1)(sass@1.71.1)
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- rollup
|
- rollup
|
||||||
- supports-color
|
- supports-color
|
||||||
dev: true
|
dev: true
|
||||||
|
|
||||||
/vite-plugin-mock-dev-server@1.4.7(vite@5.2.2):
|
/vite-plugin-mock-dev-server@1.4.7(vite@5.2.6):
|
||||||
resolution: {integrity: sha512-vGNW423fkmMibf0BfYL89n2n4tNKDt51d6Ee14gC1LlLiJAp6jabJBPsjWgU+uMgtp68+1uBb5F1qTlqdAhnoQ==}
|
resolution: {integrity: sha512-vGNW423fkmMibf0BfYL89n2n4tNKDt51d6Ee14gC1LlLiJAp6jabJBPsjWgU+uMgtp68+1uBb5F1qTlqdAhnoQ==}
|
||||||
engines: {node: ^16 || ^18 || >= 20}
|
engines: {node: ^16 || ^18 || >= 20}
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
@ -8011,7 +8011,7 @@ packages:
|
|||||||
mime-types: 2.1.35
|
mime-types: 2.1.35
|
||||||
path-to-regexp: 6.2.1
|
path-to-regexp: 6.2.1
|
||||||
picocolors: 1.0.0
|
picocolors: 1.0.0
|
||||||
vite: 5.2.2(@types/node@20.5.1)(sass@1.71.1)
|
vite: 5.2.6(@types/node@20.5.1)(sass@1.71.1)
|
||||||
ws: 8.16.0
|
ws: 8.16.0
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- bufferutil
|
- bufferutil
|
||||||
@ -8020,7 +8020,7 @@ packages:
|
|||||||
- utf-8-validate
|
- utf-8-validate
|
||||||
dev: true
|
dev: true
|
||||||
|
|
||||||
/vite-plugin-svg-icons@2.0.1(vite@5.2.2):
|
/vite-plugin-svg-icons@2.0.1(vite@5.2.6):
|
||||||
resolution: {integrity: sha512-6ktD+DhV6Rz3VtedYvBKKVA2eXF+sAQVaKkKLDSqGUfnhqXl3bj5PPkVTl3VexfTuZy66PmINi8Q6eFnVfRUmA==}
|
resolution: {integrity: sha512-6ktD+DhV6Rz3VtedYvBKKVA2eXF+sAQVaKkKLDSqGUfnhqXl3bj5PPkVTl3VexfTuZy66PmINi8Q6eFnVfRUmA==}
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
vite: '>=2.0.0'
|
vite: '>=2.0.0'
|
||||||
@ -8033,7 +8033,7 @@ packages:
|
|||||||
pathe: 0.2.0
|
pathe: 0.2.0
|
||||||
svg-baker: 1.7.0
|
svg-baker: 1.7.0
|
||||||
svgo: 2.8.0
|
svgo: 2.8.0
|
||||||
vite: 5.2.2(@types/node@20.5.1)(sass@1.71.1)
|
vite: 5.2.6(@types/node@20.5.1)(sass@1.71.1)
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
dev: true
|
dev: true
|
||||||
@ -8045,7 +8045,7 @@ packages:
|
|||||||
svgo: 3.1.0
|
svgo: 3.1.0
|
||||||
dev: true
|
dev: true
|
||||||
|
|
||||||
/vite-tsconfig-paths@4.3.2(typescript@5.2.2)(vite@5.2.2):
|
/vite-tsconfig-paths@4.3.2(typescript@5.2.2)(vite@5.2.6):
|
||||||
resolution: {integrity: sha512-0Vd/a6po6Q+86rPlntHye7F31zA2URZMbH8M3saAZ/xR9QoGN/L21bxEGfXdWmFdNkqPpRdxFT7nmNe12e9/uA==}
|
resolution: {integrity: sha512-0Vd/a6po6Q+86rPlntHye7F31zA2URZMbH8M3saAZ/xR9QoGN/L21bxEGfXdWmFdNkqPpRdxFT7nmNe12e9/uA==}
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
vite: '*'
|
vite: '*'
|
||||||
@ -8056,14 +8056,14 @@ packages:
|
|||||||
debug: 4.3.4
|
debug: 4.3.4
|
||||||
globrex: 0.1.2
|
globrex: 0.1.2
|
||||||
tsconfck: 3.0.3(typescript@5.2.2)
|
tsconfck: 3.0.3(typescript@5.2.2)
|
||||||
vite: 5.2.2(@types/node@20.5.1)(sass@1.71.1)
|
vite: 5.2.6(@types/node@20.5.1)(sass@1.71.1)
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
- typescript
|
- typescript
|
||||||
dev: true
|
dev: true
|
||||||
|
|
||||||
/vite@5.2.2(@types/node@20.5.1)(sass@1.71.1):
|
/vite@5.2.6(@types/node@20.5.1)(sass@1.71.1):
|
||||||
resolution: {integrity: sha512-FWZbz0oSdLq5snUI0b6sULbz58iXFXdvkZfZWR/F0ZJuKTSPO7v72QPXt6KqYeMFb0yytNp6kZosxJ96Nr/wDQ==}
|
resolution: {integrity: sha512-FPtnxFlSIKYjZ2eosBQamz4CbyrTizbZ3hnGJlh/wMtCrlp1Hah6AzBLjGI5I2urTfNnpovpHdrL6YRuBOPnCA==}
|
||||||
engines: {node: ^18.0.0 || >=20.0.0}
|
engines: {node: ^18.0.0 || >=20.0.0}
|
||||||
hasBin: true
|
hasBin: true
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
@ -8144,7 +8144,7 @@ packages:
|
|||||||
strip-literal: 2.0.0
|
strip-literal: 2.0.0
|
||||||
tinybench: 2.6.0
|
tinybench: 2.6.0
|
||||||
tinypool: 0.8.2
|
tinypool: 0.8.2
|
||||||
vite: 5.2.2(@types/node@20.5.1)(sass@1.71.1)
|
vite: 5.2.6(@types/node@20.5.1)(sass@1.71.1)
|
||||||
vite-node: 1.4.0(@types/node@20.5.1)(sass@1.71.1)
|
vite-node: 1.4.0(@types/node@20.5.1)(sass@1.71.1)
|
||||||
why-is-node-running: 2.2.2
|
why-is-node-running: 2.2.2
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
|
114
src/components/RChart/hooks/useChart.ts
Normal file
114
src/components/RChart/hooks/useChart.ts
Normal file
@ -0,0 +1,114 @@
|
|||||||
|
import type { ECharts } from 'echarts/core'
|
||||||
|
import type { VoidFC } from '@/types'
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 获取 RChart 实例。
|
||||||
|
* 让你能够便捷的调用相关的一些已封装方法。
|
||||||
|
*
|
||||||
|
* @warning
|
||||||
|
* 值得注意的是,必须手动调用 register 方法,否则不能正常使用。
|
||||||
|
* 在使用 相关 hooks 的时候,需要注意生命周期,确保 register 方法已经被调用与表格实例已经被初始化;
|
||||||
|
* 不要在父组件 create 阶段就去调用 hook,如果需要,请使用 nextTick 包裹。
|
||||||
|
*
|
||||||
|
* @example
|
||||||
|
* defineComponent({
|
||||||
|
* setup() {
|
||||||
|
* const [register, { ...Hooks }] = useChart()
|
||||||
|
*
|
||||||
|
* return {
|
||||||
|
* register,
|
||||||
|
* ...Hooks,
|
||||||
|
* }
|
||||||
|
* },
|
||||||
|
* render() {
|
||||||
|
* const { register, ...Hooks } = this
|
||||||
|
*
|
||||||
|
* return <RChart onRegister={register} />
|
||||||
|
* },
|
||||||
|
* })
|
||||||
|
*/
|
||||||
|
const useChart = () => {
|
||||||
|
let echartInst: ECharts
|
||||||
|
let _dispose: VoidFC
|
||||||
|
let _render: VoidFC
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param inst echart instance
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 注册当前 echart 实例,用于使用 useChart hook。
|
||||||
|
*/
|
||||||
|
const register = (inst: ECharts, render: VoidFC, dispose: VoidFC) => {
|
||||||
|
if (inst) {
|
||||||
|
echartInst = inst
|
||||||
|
_dispose = dispose
|
||||||
|
_render = render
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 获取当前 echart 实例。
|
||||||
|
*
|
||||||
|
* 如果未在调用前注册 onRegister 事件,则会抛出异常。
|
||||||
|
*
|
||||||
|
* @example
|
||||||
|
* const [register, { getChartInstance }] = useChart()
|
||||||
|
*
|
||||||
|
* const inst = getChartInstance()
|
||||||
|
*/
|
||||||
|
const getChartInstance = () => {
|
||||||
|
if (!echartInst) {
|
||||||
|
throw new Error(
|
||||||
|
'[useChart]: echart instance is not ready yet. if you are using useChart, please make sure you have called register method in onRegister event.',
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
dispose: _dispose,
|
||||||
|
render: _render,
|
||||||
|
echartInst,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* chart 是否已经销毁。
|
||||||
|
* 如果销毁则返回 true, 否则返回 false。
|
||||||
|
*/
|
||||||
|
const isDispose = () =>
|
||||||
|
!(echartInst && getChartInstance().echartInst.getDom())
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 销毁当前 chart 实例。
|
||||||
|
*/
|
||||||
|
const dispose = () => getChartInstance().dispose.call(null)
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 渲染当前 chart 实例。
|
||||||
|
*/
|
||||||
|
const render = () => getChartInstance().render.call(null)
|
||||||
|
|
||||||
|
return [
|
||||||
|
register,
|
||||||
|
{
|
||||||
|
getChartInstance,
|
||||||
|
isDispose,
|
||||||
|
dispose,
|
||||||
|
render,
|
||||||
|
},
|
||||||
|
] as const
|
||||||
|
}
|
||||||
|
|
||||||
|
export type UseChartReturn = ReturnType<typeof useChart>
|
||||||
|
|
||||||
|
export default useChart
|
@ -1,11 +1,12 @@
|
|||||||
import RChart from './src'
|
import RChart from './src'
|
||||||
import chartProps from './src/props'
|
import chartProps from './src/props'
|
||||||
|
import useChart from './hooks/useChart'
|
||||||
|
|
||||||
import type { ExtractPublicPropTypes } from 'vue'
|
import type { ExtractPublicPropTypes } from 'vue'
|
||||||
|
|
||||||
import type * as RChartType from './src/types'
|
import type * as RChartType from './src/types'
|
||||||
|
import type { UseChartReturn } from './hooks/useChart'
|
||||||
|
|
||||||
export type ChartProps = ExtractPublicPropTypes<typeof chartProps>
|
export type ChartProps = ExtractPublicPropTypes<typeof chartProps>
|
||||||
export type { RChartType }
|
export type { RChartType, UseChartReturn }
|
||||||
|
|
||||||
export { RChart, chartProps }
|
export { RChart, chartProps, useChart }
|
||||||
|
@ -70,14 +70,18 @@ echartThemes.forEach((curr) => {
|
|||||||
*
|
*
|
||||||
* @example
|
* @example
|
||||||
* <template>
|
* <template>
|
||||||
* <RChart :options="options" />
|
* <RChart :options="options" @register="register" />
|
||||||
* </template>
|
* </template>
|
||||||
*
|
*
|
||||||
* <script setup>
|
* <script setup>
|
||||||
* import { RChart } from '@/components'
|
* import { RChart } from '@/components'
|
||||||
|
*
|
||||||
|
* import { useChart } from 'vue'
|
||||||
* import { ref } from 'vue'
|
* import { ref } from 'vue'
|
||||||
*
|
*
|
||||||
|
* const [register, { ...Hooks }] = useChart()
|
||||||
* const options = ref({ ... })
|
* const options = ref({ ... })
|
||||||
|
* </script>
|
||||||
*/
|
*/
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
name: 'RChart',
|
name: 'RChart',
|
||||||
@ -348,6 +352,13 @@ export default defineComponent({
|
|||||||
|
|
||||||
// 初始化完成后移除 intersectionObserver 监听
|
// 初始化完成后移除 intersectionObserver 监听
|
||||||
intersectionObserverReturn?.stop()
|
intersectionObserverReturn?.stop()
|
||||||
|
|
||||||
|
// 注册 register,用于 useChart hook
|
||||||
|
const { onRegister } = props
|
||||||
|
|
||||||
|
if (onRegister && echartInst) {
|
||||||
|
call(onRegister, echartInst, mount, unmount)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (props.intersectionObserver) {
|
if (props.intersectionObserver) {
|
||||||
|
@ -1,3 +1,5 @@
|
|||||||
|
import { loadingOptions } from './utils'
|
||||||
|
|
||||||
import type * as echarts from 'echarts/core' // echarts 核心模块
|
import type * as echarts from 'echarts/core' // echarts 核心模块
|
||||||
import type { PropType, VNode } from 'vue'
|
import type { PropType, VNode } from 'vue'
|
||||||
import type { MaybeArray } from '@/types'
|
import type { MaybeArray } from '@/types'
|
||||||
@ -15,8 +17,7 @@ import type {
|
|||||||
RChartDownloadOptions,
|
RChartDownloadOptions,
|
||||||
} from './types'
|
} from './types'
|
||||||
import type { CardProps, DropdownProps, DropdownOption } from 'naive-ui'
|
import type { CardProps, DropdownProps, DropdownOption } from 'naive-ui'
|
||||||
|
import type { VoidFC } from '@/types'
|
||||||
import { loadingOptions } from './utils'
|
|
||||||
|
|
||||||
const props = {
|
const props = {
|
||||||
/**
|
/**
|
||||||
@ -354,6 +355,20 @@ const props = {
|
|||||||
replaceMerge: [],
|
replaceMerge: [],
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* RChart 注册挂载成功后触发的事件。
|
||||||
|
* 可以结合 useChart 方法中的 register 方法使用,然后便捷的使用 hooks。
|
||||||
|
*
|
||||||
|
* @default null
|
||||||
|
*/
|
||||||
|
onRegister: {
|
||||||
|
type: [Function, Array] as PropType<
|
||||||
|
MaybeArray<(chartInst: ECharts, render: VoidFC, dispose: VoidFC) => void>
|
||||||
|
>,
|
||||||
|
default: null,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
export default props
|
export default props
|
||||||
|
12
src/components/RForm/index.ts
Normal file
12
src/components/RForm/index.ts
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
import RForm from './src/Form'
|
||||||
|
import formProps from './src/props'
|
||||||
|
import useForm from './src/hooks/useForm'
|
||||||
|
|
||||||
|
import type * as RFormType from './src/types'
|
||||||
|
import type { ExtractPublicPropTypes } from 'vue'
|
||||||
|
import type { UseFormReturn } from './src/hooks/useForm'
|
||||||
|
|
||||||
|
export type FormProps = ExtractPublicPropTypes<typeof formProps>
|
||||||
|
export type { RFormType, UseFormReturn }
|
||||||
|
|
||||||
|
export { RForm, formProps, useForm }
|
49
src/components/RForm/src/Form.tsx
Normal file
49
src/components/RForm/src/Form.tsx
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
/**
|
||||||
|
*
|
||||||
|
* @author Ray <https://github.com/XiaoDaiGua-Ray>
|
||||||
|
*
|
||||||
|
* @date 2024-03-27
|
||||||
|
*
|
||||||
|
* @workspace ray-template
|
||||||
|
*
|
||||||
|
* @remark 今天也是元气满满撸代码的一天
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NForm } from 'naive-ui'
|
||||||
|
|
||||||
|
import props from './props'
|
||||||
|
import { call } from '@/utils'
|
||||||
|
|
||||||
|
import type { RFormInst } from './types'
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'RForm',
|
||||||
|
props,
|
||||||
|
setup(props) {
|
||||||
|
const formRef = ref<RFormInst>()
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
// 主动调用 register 方法,满足 useForm 方法正常调用
|
||||||
|
const { onRegister } = props
|
||||||
|
|
||||||
|
if (onRegister && formRef.value) {
|
||||||
|
call(onRegister, formRef.value)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
return {
|
||||||
|
formRef,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
render() {
|
||||||
|
const { $attrs, $props, $slots } = this
|
||||||
|
|
||||||
|
return (
|
||||||
|
<NForm {...$attrs} {...$props} ref="formRef">
|
||||||
|
{{
|
||||||
|
...$slots,
|
||||||
|
}}
|
||||||
|
</NForm>
|
||||||
|
)
|
||||||
|
},
|
||||||
|
})
|
113
src/components/RForm/src/hooks/useForm.ts
Normal file
113
src/components/RForm/src/hooks/useForm.ts
Normal file
@ -0,0 +1,113 @@
|
|||||||
|
import { cloneDeep } from 'lodash-es'
|
||||||
|
|
||||||
|
import type {
|
||||||
|
RFormInst,
|
||||||
|
FormValidateCallback,
|
||||||
|
ShouldRuleBeApplied,
|
||||||
|
RFormRules,
|
||||||
|
} from '../types'
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 获取 RForm 实例。
|
||||||
|
* 让你能够便捷的调用相关的一些已封装方法。
|
||||||
|
*
|
||||||
|
* @warning
|
||||||
|
* 值得注意的是,必须手动调用 register 方法,否则不能正常使用。
|
||||||
|
* 在使用 相关 hooks 的时候,需要注意生命周期,确保 register 方法已经被调用与表格实例已经被初始化;
|
||||||
|
* 不要在父组件 create 阶段就去调用 hook,如果需要,请使用 nextTick 包裹。
|
||||||
|
*
|
||||||
|
* @example
|
||||||
|
* defineComponent({
|
||||||
|
* setup() {
|
||||||
|
* const [register, { ...Hooks }] = useForm()
|
||||||
|
*
|
||||||
|
* return {
|
||||||
|
* register,
|
||||||
|
* ...Hooks,
|
||||||
|
* }
|
||||||
|
* },
|
||||||
|
* render() {
|
||||||
|
* const { register, ...Hooks } = this
|
||||||
|
*
|
||||||
|
* return <RForm onRegister={register} />
|
||||||
|
* },
|
||||||
|
* })
|
||||||
|
*/
|
||||||
|
const useForm = <T extends Record<string, unknown>, R extends RFormRules>(
|
||||||
|
model?: T,
|
||||||
|
rules?: R,
|
||||||
|
) => {
|
||||||
|
const formRef = ref<RFormInst>()
|
||||||
|
|
||||||
|
const register = (inst: RFormInst) => {
|
||||||
|
if (inst) {
|
||||||
|
formRef.value = inst
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const getFormInstance = () => {
|
||||||
|
if (!formRef.value) {
|
||||||
|
throw new Error(
|
||||||
|
'[useForm]: form instance is not ready yet. if you are using useForm, please make sure you have called register method in onRegister event.',
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return formRef.value
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 验证表单,Promise rejection 的返回值类型是 FormValidationError[]。
|
||||||
|
*
|
||||||
|
* @see https://www.naiveui.com/zh-CN/dark/components/form#inline.vue
|
||||||
|
*/
|
||||||
|
const validate = (
|
||||||
|
callback?: FormValidateCallback,
|
||||||
|
shouldRuleBeApplied?: ShouldRuleBeApplied,
|
||||||
|
) => getFormInstance().validate.call(null, callback, shouldRuleBeApplied)
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 还原表单到未验证状态。
|
||||||
|
*
|
||||||
|
* @see https://www.naiveui.com/zh-CN/dark/components/form#Form-Methods
|
||||||
|
*/
|
||||||
|
const restoreValidation = () => getFormInstance().restoreValidation.call(null)
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 获取表项中收集到的值的对象。
|
||||||
|
*
|
||||||
|
* 调用该方法时,需要确保初始化 useForm 方法的时候传入了 model,否则可能有意想不到的问题发生。
|
||||||
|
*/
|
||||||
|
const formModel = () => cloneDeep(model) || ({} as T)
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 获取验证表单项的规则。
|
||||||
|
*
|
||||||
|
* 调用该方法时,需要确保初始化 useForm 方法的时候传入了 rules,否则可能有意想不到的问题发生。
|
||||||
|
*/
|
||||||
|
const formRules = () => cloneDeep(rules) || ({} as R)
|
||||||
|
|
||||||
|
return [
|
||||||
|
register,
|
||||||
|
{
|
||||||
|
getFormInstance,
|
||||||
|
validate,
|
||||||
|
restoreValidation,
|
||||||
|
formModel,
|
||||||
|
formRules,
|
||||||
|
},
|
||||||
|
] as const
|
||||||
|
}
|
||||||
|
|
||||||
|
export type UseFormReturn = ReturnType<typeof useForm>
|
||||||
|
|
||||||
|
export default useForm
|
24
src/components/RForm/src/props.ts
Normal file
24
src/components/RForm/src/props.ts
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
import { formProps } from 'naive-ui'
|
||||||
|
|
||||||
|
import type { MaybeArray } from '@/types'
|
||||||
|
import type { RFormInst } from './types'
|
||||||
|
|
||||||
|
const props = {
|
||||||
|
...formProps,
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* RForm 注册挂载成功后触发的事件。
|
||||||
|
* 可以结合 useForm 方法中的 register 方法使用,然后便捷的使用 hooks。
|
||||||
|
*
|
||||||
|
* @default null
|
||||||
|
*/
|
||||||
|
onRegister: {
|
||||||
|
type: [Function, Array] as PropType<
|
||||||
|
MaybeArray<(formInst: RFormInst) => void>
|
||||||
|
>,
|
||||||
|
default: null,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
export default props
|
13
src/components/RForm/src/types.ts
Normal file
13
src/components/RForm/src/types.ts
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
import type { FormInst, FormItemRule, FormRules } from 'naive-ui'
|
||||||
|
|
||||||
|
export type RFormInst = FormInst
|
||||||
|
|
||||||
|
type FormValidateParameters = Parameters<FormInst['validate']>
|
||||||
|
|
||||||
|
export type FormValidateCallback = FormValidateParameters[0]
|
||||||
|
|
||||||
|
export type ShouldRuleBeApplied = FormValidateParameters[1]
|
||||||
|
|
||||||
|
export interface RFormRules {
|
||||||
|
[itemValidatePath: string]: FormItemRule | Array<FormItemRule> | FormRules
|
||||||
|
}
|
@ -1,8 +1,12 @@
|
|||||||
import RModal from './src/Modal'
|
import RModal from './src/Modal'
|
||||||
import modalProps from './src/props'
|
import modalProps from './src/props'
|
||||||
|
import useModal from './src/hooks/useModal'
|
||||||
|
|
||||||
|
import type * as RModalType from './src/types'
|
||||||
import type { ExtractPublicPropTypes } from 'vue'
|
import type { ExtractPublicPropTypes } from 'vue'
|
||||||
|
import type { UseModalReturn } from './src/hooks/useModal'
|
||||||
|
|
||||||
export type ModalProps = ExtractPublicPropTypes<typeof modalProps>
|
export type ModalProps = ExtractPublicPropTypes<typeof modalProps>
|
||||||
|
export type { RModalType, UseModalReturn }
|
||||||
|
|
||||||
export { RModal, modalProps }
|
export { RModal, modalProps, useModal }
|
||||||
|
@ -15,8 +15,12 @@ import { NModal } from 'naive-ui'
|
|||||||
|
|
||||||
import props from './props'
|
import props from './props'
|
||||||
import { completeSize, uuid } from '@/utils'
|
import { completeSize, uuid } from '@/utils'
|
||||||
import { useWindowSize } from '@vueuse/core'
|
import { setupInteract } from './utils'
|
||||||
import { setupDraggable } from './utils'
|
import {
|
||||||
|
FULLSCREEN_CARD_TYPE_CLASS,
|
||||||
|
R_MODAL_CLASS,
|
||||||
|
CSS_VARS_KEYS,
|
||||||
|
} from './constant'
|
||||||
|
|
||||||
import type interact from 'interactjs'
|
import type interact from 'interactjs'
|
||||||
|
|
||||||
@ -24,11 +28,10 @@ export default defineComponent({
|
|||||||
name: 'RModal',
|
name: 'RModal',
|
||||||
props,
|
props,
|
||||||
setup(props) {
|
setup(props) {
|
||||||
const { height } = useWindowSize()
|
|
||||||
const cssVars = computed(() => ({
|
const cssVars = computed(() => ({
|
||||||
'--r-modal-width': completeSize(props.width ?? 600),
|
[CSS_VARS_KEYS['width']]: completeSize(props.width ?? 600),
|
||||||
'--r-modal-card-width': completeSize(props.cardWidth ?? 600),
|
[CSS_VARS_KEYS['cardWidth']]: completeSize(props.cardWidth ?? 600),
|
||||||
'--r-modal-dialog-width': completeSize(props.dialogWidth ?? 446),
|
[CSS_VARS_KEYS['dialogWidth']]: completeSize(props.dialogWidth ?? 446),
|
||||||
}))
|
}))
|
||||||
const uuidEl = uuid()
|
const uuidEl = uuid()
|
||||||
let intractable: null | ReturnType<typeof interact>
|
let intractable: null | ReturnType<typeof interact>
|
||||||
@ -37,32 +40,10 @@ export default defineComponent({
|
|||||||
x: 0,
|
x: 0,
|
||||||
y: 0,
|
y: 0,
|
||||||
}
|
}
|
||||||
|
// 当前是否为预设 card 类型并且设置了 fullscreen
|
||||||
/**
|
const isFullscreenCardType = computed(
|
||||||
*
|
() => props.preset === 'card' && props.fullscreen,
|
||||||
* 获取当前是否为 card 风格并且为全屏
|
)
|
||||||
*/
|
|
||||||
const isFullscreenCardType = () =>
|
|
||||||
props.preset === 'card' && props.fullscreen
|
|
||||||
|
|
||||||
const setupInteract = () => {
|
|
||||||
const target = document.getElementById(uuidEl)
|
|
||||||
|
|
||||||
if (target) {
|
|
||||||
setupDraggable(target, props.preset, {
|
|
||||||
scheduler: (event) => {
|
|
||||||
const target = event.target
|
|
||||||
|
|
||||||
position.x += event.dx
|
|
||||||
position.y += event.dy
|
|
||||||
|
|
||||||
target.style.transform = `translate(${position.x}px, ${position.y}px)`
|
|
||||||
},
|
|
||||||
}).then((res) => {
|
|
||||||
intractable = res
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
watch(
|
watch(
|
||||||
() => props.show,
|
() => props.show,
|
||||||
@ -73,10 +54,22 @@ export default defineComponent({
|
|||||||
(props.preset === 'card' || props.preset === 'dialog')
|
(props.preset === 'card' || props.preset === 'dialog')
|
||||||
) {
|
) {
|
||||||
nextTick(() => {
|
nextTick(() => {
|
||||||
setupInteract()
|
|
||||||
|
|
||||||
const target = document.getElementById(uuidEl)
|
const target = document.getElementById(uuidEl)
|
||||||
|
|
||||||
|
if (target) {
|
||||||
|
setupInteract(target, {
|
||||||
|
preset: props.preset,
|
||||||
|
x: position.x,
|
||||||
|
y: position.y,
|
||||||
|
dargCallback: (x, y) => {
|
||||||
|
position.x = x
|
||||||
|
position.y = y
|
||||||
|
},
|
||||||
|
}).then((res) => {
|
||||||
|
intractable = res
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
if (props.memo && target) {
|
if (props.memo && target) {
|
||||||
target.style.transform = `translate(${position.x}px, ${position.y}px)`
|
target.style.transform = `translate(${position.x}px, ${position.y}px)`
|
||||||
}
|
}
|
||||||
@ -94,24 +87,22 @@ export default defineComponent({
|
|||||||
|
|
||||||
return {
|
return {
|
||||||
cssVars,
|
cssVars,
|
||||||
height,
|
|
||||||
isFullscreenCardType,
|
isFullscreenCardType,
|
||||||
uuidEl,
|
uuidEl,
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
render() {
|
render() {
|
||||||
const { isFullscreenCardType } = this
|
|
||||||
const { $props, $slots, $attrs } = this
|
const { $props, $slots, $attrs } = this
|
||||||
const { preset, ...$otherProps } = $props
|
const { preset, ...$otherProps } = $props
|
||||||
const { cssVars, height, uuidEl } = this
|
const { cssVars, uuidEl, isFullscreenCardType } = this
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<NModal
|
<NModal
|
||||||
class={[
|
class={[
|
||||||
'r-modal',
|
R_MODAL_CLASS,
|
||||||
isFullscreenCardType() ? 'r-modal__preset-card--fullscreen' : '',
|
isFullscreenCardType ? FULLSCREEN_CARD_TYPE_CLASS : '',
|
||||||
]}
|
]}
|
||||||
style={[cssVars, isFullscreenCardType() ? `height: ${height}px` : '']}
|
style={[cssVars, isFullscreenCardType ? `height: 100vh` : '']}
|
||||||
preset={preset}
|
preset={preset}
|
||||||
{...{
|
{...{
|
||||||
id: uuidEl,
|
id: uuidEl,
|
||||||
|
9
src/components/RModal/src/constant.ts
Normal file
9
src/components/RModal/src/constant.ts
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
export const FULLSCREEN_CARD_TYPE_CLASS = 'r-modal__preset-card--fullscreen'
|
||||||
|
|
||||||
|
export const R_MODAL_CLASS = 'r-modal'
|
||||||
|
|
||||||
|
export const CSS_VARS_KEYS = {
|
||||||
|
width: '--r-modal-width',
|
||||||
|
cardWidth: '--r-modal-card-width',
|
||||||
|
dialogWidth: '--r-modal-dialog-width',
|
||||||
|
}
|
61
src/components/RModal/src/hooks/useModal.ts
Normal file
61
src/components/RModal/src/hooks/useModal.ts
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
import { useModal as useNaiveModal } from 'naive-ui'
|
||||||
|
import { setupInteract } from '../utils'
|
||||||
|
import { queryElements, setStyle, completeSize, setClass } from '@/utils'
|
||||||
|
import { R_MODAL_CLASS, CSS_VARS_KEYS } from '../constant'
|
||||||
|
|
||||||
|
import type { RModalProps } from '../types'
|
||||||
|
|
||||||
|
interface UseModalCreateOptions extends Omit<RModalProps, 'memo'> {}
|
||||||
|
|
||||||
|
const useModal = () => {
|
||||||
|
const { create: naiveCreate, destroyAll: naiveDestroyAll } = useNaiveModal()
|
||||||
|
|
||||||
|
const create = (options: UseModalCreateOptions) => {
|
||||||
|
const { preset, dad, fullscreen, width, cardWidth, dialogWidth } = options
|
||||||
|
const modalReactive = naiveCreate(options)
|
||||||
|
const { key } = modalReactive
|
||||||
|
const cssVars = {
|
||||||
|
[CSS_VARS_KEYS['width']]: completeSize(width ?? 600),
|
||||||
|
[CSS_VARS_KEYS['cardWidth']]: completeSize(cardWidth ?? 600),
|
||||||
|
[CSS_VARS_KEYS['dialogWidth']]: completeSize(dialogWidth ?? 446),
|
||||||
|
}
|
||||||
|
|
||||||
|
nextTick(() => {
|
||||||
|
const [modalElement] =
|
||||||
|
queryElements<HTMLElement>(`[internalkey="${key}"]`) || []
|
||||||
|
|
||||||
|
if (!modalElement) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dad) {
|
||||||
|
setupInteract(modalElement, {
|
||||||
|
preset,
|
||||||
|
x: 0,
|
||||||
|
y: 0,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fullscreen && preset === 'card') {
|
||||||
|
setStyle(modalElement, {
|
||||||
|
width: '100%',
|
||||||
|
height: '100vh',
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
setStyle(modalElement, cssVars)
|
||||||
|
setClass(modalElement, R_MODAL_CLASS)
|
||||||
|
})
|
||||||
|
|
||||||
|
return modalReactive
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
create,
|
||||||
|
destroyAll: naiveDestroyAll,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export type UseModalReturn = ReturnType<typeof useModal>
|
||||||
|
|
||||||
|
export default useModal
|
@ -38,7 +38,7 @@ const props = {
|
|||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* @description
|
* @description
|
||||||
* preset 空时宽度设置。
|
* preset 为空时宽度设置。
|
||||||
*
|
*
|
||||||
* @default 600
|
* @default 600
|
||||||
*/
|
*/
|
||||||
|
53
src/components/RModal/src/types.ts
Normal file
53
src/components/RModal/src/types.ts
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
import type { ModalOptions as NaiveModalOptions } from 'naive-ui'
|
||||||
|
|
||||||
|
export interface RModalProps extends NaiveModalOptions {
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 是否记住上一次的位置。
|
||||||
|
*
|
||||||
|
* @default true
|
||||||
|
*/
|
||||||
|
memo?: boolean
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 是否全屏。
|
||||||
|
*
|
||||||
|
* @default false
|
||||||
|
*/
|
||||||
|
fullscreen?: boolean
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* preset 为空时宽度设置。
|
||||||
|
*
|
||||||
|
* @default 600
|
||||||
|
*/
|
||||||
|
width?: number
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* preset 为 card 时宽度设置。
|
||||||
|
*
|
||||||
|
* @default 600
|
||||||
|
*/
|
||||||
|
cardWidth?: number
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* preset 为 dialog 时宽度设置。
|
||||||
|
*
|
||||||
|
* @default 446
|
||||||
|
*/
|
||||||
|
dialogWidth?: number
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 是否启用拖拽。
|
||||||
|
* 当启用拖拽时,可以通过拖拽 header 部分控制模态框。
|
||||||
|
*
|
||||||
|
* @default false
|
||||||
|
*/
|
||||||
|
dad?: boolean
|
||||||
|
}
|
@ -1,10 +1,18 @@
|
|||||||
import interact from 'interactjs'
|
import interact from 'interactjs'
|
||||||
|
|
||||||
import type { ModalProps } from 'naive-ui'
|
import type { ModalProps } from 'naive-ui'
|
||||||
import type { AnyFC } from '@/types'
|
import type { RModalProps } from './types'
|
||||||
|
|
||||||
interface SetupDraggableOptions {
|
interface SetupDraggableOptions {
|
||||||
scheduler?: AnyFC
|
scheduler?: (event: Interact.DragEvent) => void
|
||||||
|
}
|
||||||
|
|
||||||
|
interface SetupInteractOptions {
|
||||||
|
preset: ModalProps['preset']
|
||||||
|
memo?: RModalProps['memo']
|
||||||
|
x: number
|
||||||
|
y: number
|
||||||
|
dargCallback?: (x: number, y: number, event: Interact.DragEvent) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -12,6 +20,7 @@ interface SetupDraggableOptions {
|
|||||||
* @param bindModal modal 预设元素
|
* @param bindModal modal 预设元素
|
||||||
* @param preset 预设类型
|
* @param preset 预设类型
|
||||||
*
|
*
|
||||||
|
* @description
|
||||||
* 根据预设模态框设置拖拽效果
|
* 根据预设模态框设置拖拽效果
|
||||||
* 但是该效果有且仅有 card, dialog 有效
|
* 但是该效果有且仅有 card, dialog 有效
|
||||||
*
|
*
|
||||||
@ -54,3 +63,39 @@ export const setupDraggable = (
|
|||||||
}, 30)
|
}, 30)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const setupInteract = (
|
||||||
|
target: HTMLElement | string,
|
||||||
|
options: SetupInteractOptions,
|
||||||
|
): Promise<ReturnType<typeof interact>> => {
|
||||||
|
const _target =
|
||||||
|
typeof target === 'string'
|
||||||
|
? (document.querySelector(target) as HTMLElement)
|
||||||
|
: target
|
||||||
|
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
if (_target) {
|
||||||
|
_target.setAttribute('can-drag', 'true')
|
||||||
|
|
||||||
|
const { preset, dargCallback } = options
|
||||||
|
let { x, y } = options
|
||||||
|
|
||||||
|
setupDraggable(_target, preset, {
|
||||||
|
scheduler: (event) => {
|
||||||
|
const target = event.target
|
||||||
|
|
||||||
|
x += event.dx
|
||||||
|
y += event.dy
|
||||||
|
|
||||||
|
target.style.transform = `translate(${x}px, ${y}px)`
|
||||||
|
|
||||||
|
dargCallback?.(x, y, event)
|
||||||
|
},
|
||||||
|
}).then((res) => {
|
||||||
|
resolve(res)
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
reject()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
@ -1,10 +1,12 @@
|
|||||||
import RTable from './src/Table'
|
import RTable from './src/Table'
|
||||||
import tableProps from './src/props'
|
import tableProps from './src/props'
|
||||||
|
import useTable from './src/hooks/useTable'
|
||||||
|
|
||||||
import type * as RTableType from './src/types'
|
import type * as RTableType from './src/types'
|
||||||
|
import type { UseTableReturn } from './src/hooks/useTable'
|
||||||
import type { ExtractPublicPropTypes } from 'vue'
|
import type { ExtractPublicPropTypes } from 'vue'
|
||||||
|
|
||||||
export type TableProps = ExtractPublicPropTypes<typeof tableProps>
|
export type TableProps = ExtractPublicPropTypes<typeof tableProps>
|
||||||
export type { RTableType }
|
export type { RTableType, UseTableReturn }
|
||||||
|
|
||||||
export { RTable, tableProps }
|
export { RTable, tableProps, useTable }
|
||||||
|
@ -30,7 +30,7 @@ export default defineComponent({
|
|||||||
name: 'RTable',
|
name: 'RTable',
|
||||||
props,
|
props,
|
||||||
setup(props, ctx) {
|
setup(props, ctx) {
|
||||||
const { expose } = ctx
|
const { expose, emit } = ctx
|
||||||
|
|
||||||
const rTableInst = ref<DataTableInst>()
|
const rTableInst = ref<DataTableInst>()
|
||||||
const wrapperRef = ref<HTMLElement>()
|
const wrapperRef = ref<HTMLElement>()
|
||||||
@ -161,8 +161,14 @@ export default defineComponent({
|
|||||||
* 处理 toolOptions,合并渲染所有的 toolOptions
|
* 处理 toolOptions,合并渲染所有的 toolOptions
|
||||||
*/
|
*/
|
||||||
const tool = (p: typeof props) => {
|
const tool = (p: typeof props) => {
|
||||||
|
const { tool } = p
|
||||||
|
|
||||||
|
if (!tool) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
const renderDefaultToolOptions = () => (
|
const renderDefaultToolOptions = () => (
|
||||||
<>
|
<NFlex align="center">
|
||||||
<Print {...p} />
|
<Print {...p} />
|
||||||
<Size {...p} onChangeSize={changeTableSize.bind(this)} />
|
<Size {...p} onChangeSize={changeTableSize.bind(this)} />
|
||||||
<Fullscreen />
|
<Fullscreen />
|
||||||
@ -172,25 +178,34 @@ export default defineComponent({
|
|||||||
onPopselectChange={popselectChange.bind(this)}
|
onPopselectChange={popselectChange.bind(this)}
|
||||||
onInitialed={popselectChange.bind(this)}
|
onInitialed={popselectChange.bind(this)}
|
||||||
/>
|
/>
|
||||||
</>
|
</NFlex>
|
||||||
)
|
)
|
||||||
|
|
||||||
if (!props.toolOptions) {
|
if (!props.toolOptions) {
|
||||||
return renderDefaultToolOptions
|
return renderDefaultToolOptions
|
||||||
} else {
|
} else {
|
||||||
if (props.coverTool) {
|
if (props.coverTool) {
|
||||||
return renderToolOptions
|
return <NFlex align="center">{renderToolOptions()}</NFlex>
|
||||||
} else {
|
} else {
|
||||||
return () => (
|
return () => (
|
||||||
<>
|
<NFlex align="center">
|
||||||
{renderDefaultToolOptions()}
|
{renderDefaultToolOptions()}
|
||||||
{renderToolOptions()}
|
{renderToolOptions()}
|
||||||
</>
|
</NFlex>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
// 主动调用 register 方法,满足 useTable 方法正常调用
|
||||||
|
const { onRegister } = props
|
||||||
|
|
||||||
|
if (onRegister && rTableInst.value) {
|
||||||
|
call(onRegister, rTableInst.value)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
provide(config.tableKey, {
|
provide(config.tableKey, {
|
||||||
uuidTable,
|
uuidTable,
|
||||||
uuidWrapper,
|
uuidWrapper,
|
||||||
@ -242,13 +257,13 @@ export default defineComponent({
|
|||||||
default: () => (
|
default: () => (
|
||||||
<>
|
<>
|
||||||
<NDataTable
|
<NDataTable
|
||||||
ref="rTableInst"
|
|
||||||
{...{ id: uuidTable }}
|
{...{ id: uuidTable }}
|
||||||
{...$attrs}
|
{...$attrs}
|
||||||
{...$props}
|
{...$props}
|
||||||
{...propsPopselectValue}
|
{...propsPopselectValue}
|
||||||
rowProps={combineRowProps.bind(this)}
|
rowProps={combineRowProps.bind(this)}
|
||||||
size={privateReactive.size}
|
size={privateReactive.size}
|
||||||
|
ref="rTableInst"
|
||||||
>
|
>
|
||||||
{{
|
{{
|
||||||
...$slots,
|
...$slots,
|
||||||
@ -273,12 +288,8 @@ export default defineComponent({
|
|||||||
header: renderNode(title, {
|
header: renderNode(title, {
|
||||||
defaultElement: <div style="display: none;"></div>,
|
defaultElement: <div style="display: none;"></div>,
|
||||||
}),
|
}),
|
||||||
'header-extra': () => (
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
<NFlex align="center">
|
'header-extra': tool($props as any),
|
||||||
{/* eslint-disable @typescript-eslint/no-explicit-any */}
|
|
||||||
{tool($props as any)}
|
|
||||||
</NFlex>
|
|
||||||
),
|
|
||||||
footer: () => $slots.tableFooter?.(),
|
footer: () => $slots.tableFooter?.(),
|
||||||
action: () => $slots.tableAction?.(),
|
action: () => $slots.tableAction?.(),
|
||||||
}}
|
}}
|
||||||
|
147
src/components/RTable/src/hooks/useTable.ts
Normal file
147
src/components/RTable/src/hooks/useTable.ts
Normal file
@ -0,0 +1,147 @@
|
|||||||
|
import type {
|
||||||
|
RTableInst,
|
||||||
|
CsvOptionsType,
|
||||||
|
FilterState,
|
||||||
|
ScrollToOptions,
|
||||||
|
ColumnKey,
|
||||||
|
SortOrder,
|
||||||
|
} from '../types'
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 获取 RTable 实例。
|
||||||
|
* 让你能够便捷的调用相关的一些已封装方法。
|
||||||
|
*
|
||||||
|
* @warning
|
||||||
|
* 值得注意的是,必须手动调用 register 方法,否则不能正常使用。
|
||||||
|
* 在使用 相关 hooks 的时候,需要注意生命周期,确保 register 方法已经被调用与表格实例已经被初始化;
|
||||||
|
* 不要在父组件 create 阶段就去调用 hook,如果需要,请使用 nextTick 包裹。
|
||||||
|
*
|
||||||
|
* @example
|
||||||
|
* defineComponent({
|
||||||
|
* setup() {
|
||||||
|
* const [register, { ...Hooks }] = useTable()
|
||||||
|
*
|
||||||
|
* return {
|
||||||
|
* register,
|
||||||
|
* ...Hooks,
|
||||||
|
* }
|
||||||
|
* },
|
||||||
|
* render() {
|
||||||
|
* const { register, ...Hooks } = this
|
||||||
|
*
|
||||||
|
* return <RTable onRegister={register} />
|
||||||
|
* },
|
||||||
|
* })
|
||||||
|
*/
|
||||||
|
const useTable = () => {
|
||||||
|
const tableRef = ref<RTableInst>()
|
||||||
|
|
||||||
|
const register = (inst: RTableInst) => {
|
||||||
|
if (inst) {
|
||||||
|
tableRef.value = inst
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 获取 RTable 实例。
|
||||||
|
*/
|
||||||
|
const getTableInstance = () => {
|
||||||
|
if (!tableRef.value) {
|
||||||
|
throw new Error(
|
||||||
|
'[useTable]: table instance is not ready yet. if you are using useTable, please make sure you have called register method in onRegister event.',
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return tableRef.value
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 清空所有 filter 状态。
|
||||||
|
*
|
||||||
|
* @see https://www.naiveui.com/zh-CN/dark/components/data-table#filter-and-sorter
|
||||||
|
*/
|
||||||
|
const clearFilters = () => getTableInstance().clearFilters.call(null)
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 清空所有 sort 状态。
|
||||||
|
*
|
||||||
|
* @see https://www.naiveui.com/zh-CN/dark/components/data-table#filter-and-sorter
|
||||||
|
*/
|
||||||
|
const clearSorter = () => getTableInstance().clearSorter.call(null)
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 下载 CSV。
|
||||||
|
*
|
||||||
|
* @see https://www.naiveui.com/zh-CN/dark/components/data-table#export-csv.vue
|
||||||
|
*/
|
||||||
|
const downloadCsv = (options?: CsvOptionsType) =>
|
||||||
|
getTableInstance().downloadCsv.call(null, options)
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 设定表格当前的过滤器。
|
||||||
|
*
|
||||||
|
* @see https://www.naiveui.com/zh-CN/dark/components/data-table#filter-and-sorter
|
||||||
|
*/
|
||||||
|
const filters = (filters: FilterState | null) =>
|
||||||
|
getTableInstance().filters.call(null, filters)
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 手动设置 page。
|
||||||
|
*
|
||||||
|
* @see https://www.naiveui.com/zh-CN/dark/components/data-table#DataTable-Methods
|
||||||
|
*/
|
||||||
|
const page = (page: number) => getTableInstance().page.call(null, page)
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 滚动内容。
|
||||||
|
*
|
||||||
|
* @see https://www.naiveui.com/zh-CN/dark/components/data-table#DataTable-Methods
|
||||||
|
*/
|
||||||
|
const scrollTo: ScrollToOptions = (options) =>
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
getTableInstance().scrollTo(options as any)
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 设定表格的过滤状态。
|
||||||
|
*
|
||||||
|
* @see https://www.naiveui.com/zh-CN/dark/components/data-table#DataTable-Methods
|
||||||
|
*/
|
||||||
|
const sort = (columnKey: ColumnKey, order: SortOrder) =>
|
||||||
|
getTableInstance().sort.call(null, columnKey, order)
|
||||||
|
|
||||||
|
return [
|
||||||
|
register,
|
||||||
|
{
|
||||||
|
getTableInstance,
|
||||||
|
clearFilters,
|
||||||
|
clearSorter,
|
||||||
|
downloadCsv,
|
||||||
|
filters,
|
||||||
|
page,
|
||||||
|
scrollTo,
|
||||||
|
sort,
|
||||||
|
},
|
||||||
|
] as const
|
||||||
|
}
|
||||||
|
|
||||||
|
export type UseTableReturn = ReturnType<typeof useTable>
|
||||||
|
|
||||||
|
export default useTable
|
@ -14,101 +14,198 @@ import { dataTableProps } from 'naive-ui'
|
|||||||
import type { PropType, VNode } from 'vue'
|
import type { PropType, VNode } from 'vue'
|
||||||
import type { MaybeArray } from '@/types'
|
import type { MaybeArray } from '@/types'
|
||||||
import type { DropdownOption, DataTableColumn } from 'naive-ui'
|
import type { DropdownOption, DataTableColumn } from 'naive-ui'
|
||||||
import type { DownloadCsvTableOptions, PrintTableOptions } from './types'
|
import type {
|
||||||
|
DownloadCsvTableOptions,
|
||||||
|
PrintTableOptions,
|
||||||
|
RTableInst,
|
||||||
|
} from './types'
|
||||||
import type { Recordable } from '@/types'
|
import type { Recordable } from '@/types'
|
||||||
|
|
||||||
const props = {
|
const props = {
|
||||||
...dataTableProps,
|
...dataTableProps,
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 是否启用表格工具栏。
|
||||||
|
*
|
||||||
|
* 当设置为 false 时,不会显示表格工具栏。不论是否配置了 toolOptions。
|
||||||
|
*
|
||||||
|
* @default true
|
||||||
|
*/
|
||||||
|
tool: {
|
||||||
|
type: Boolean,
|
||||||
|
default: true,
|
||||||
|
},
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 下载表格配置项。
|
||||||
|
*
|
||||||
|
* @default {}
|
||||||
|
*/
|
||||||
downloadCsvTableOptions: {
|
downloadCsvTableOptions: {
|
||||||
/**
|
|
||||||
*
|
|
||||||
* 配置下载表格配置项
|
|
||||||
*/
|
|
||||||
type: Object as PropType<DownloadCsvTableOptions>,
|
type: Object as PropType<DownloadCsvTableOptions>,
|
||||||
default: () => ({}),
|
default: () => ({}),
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 表格标题,支持 VNode。
|
||||||
|
*
|
||||||
|
* @default null
|
||||||
|
*/
|
||||||
title: {
|
title: {
|
||||||
/**
|
|
||||||
*
|
|
||||||
* 表格标题
|
|
||||||
* 支持自定义渲染
|
|
||||||
*/
|
|
||||||
type: [String, Number, Object] as PropType<VNode | string | number>,
|
type: [String, Number, Object] as PropType<VNode | string | number>,
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 自定义工具栏配置项。
|
||||||
|
*
|
||||||
|
* @default undefined
|
||||||
|
*/
|
||||||
toolOptions: {
|
toolOptions: {
|
||||||
/** 自定义拓展工具栏 */
|
|
||||||
type: Array as PropType<(VNode | (() => VNode))[]>,
|
type: Array as PropType<(VNode | (() => VNode))[]>,
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 是否覆盖原工具栏功能按钮。
|
||||||
|
*
|
||||||
|
* @default false
|
||||||
|
*/
|
||||||
coverTool: {
|
coverTool: {
|
||||||
/** 当 toolOptions 配置时,是否覆盖原工具栏 */
|
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 右键菜单配置项。
|
||||||
|
*
|
||||||
|
* 基于 NDropdown 实现。
|
||||||
|
*
|
||||||
|
* @default undefined
|
||||||
|
*/
|
||||||
contextMenuOptions: {
|
contextMenuOptions: {
|
||||||
/**
|
|
||||||
*
|
|
||||||
* 右键菜单配置项
|
|
||||||
* 基于 `NDropdown` 实现
|
|
||||||
*/
|
|
||||||
type: Array as PropType<DropdownOption[]>,
|
type: Array as PropType<DropdownOption[]>,
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 是否禁用右键菜单。
|
||||||
|
*
|
||||||
|
* 如果设置为 false 则不会唤起右键菜单。
|
||||||
|
*
|
||||||
|
* @default false
|
||||||
|
*/
|
||||||
disabledContextMenu: {
|
disabledContextMenu: {
|
||||||
/**
|
|
||||||
*
|
|
||||||
* 是否禁用右键菜单
|
|
||||||
* 如果设置为 false 则不会唤起右键菜单
|
|
||||||
*/
|
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 右键菜单点击事件回调。
|
||||||
|
*
|
||||||
|
* 该属性用于配置右键菜单点击事件。
|
||||||
|
*
|
||||||
|
* @default null
|
||||||
|
*/
|
||||||
onContextMenuClick: {
|
onContextMenuClick: {
|
||||||
/** 右键菜单点击 */
|
|
||||||
type: [Function, Array] as PropType<
|
type: [Function, Array] as PropType<
|
||||||
MaybeArray<(key: string | number, option: DropdownOption) => void>
|
MaybeArray<(key: string | number, option: DropdownOption) => void>
|
||||||
>,
|
>,
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 表格容器边框。
|
||||||
|
*
|
||||||
|
* 与表格边框为两个不同配置项。
|
||||||
|
*
|
||||||
|
* @default false
|
||||||
|
*/
|
||||||
wrapperBordered: {
|
wrapperBordered: {
|
||||||
/**
|
|
||||||
*
|
|
||||||
* 表格容器边框
|
|
||||||
* 与表格边框为两个不同配置项
|
|
||||||
*/
|
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 打印表格配置项。
|
||||||
|
*
|
||||||
|
* 基于 dom printDom 实现。
|
||||||
|
*
|
||||||
|
* @default {}
|
||||||
|
*/
|
||||||
printTableOptions: {
|
printTableOptions: {
|
||||||
/**
|
|
||||||
*
|
|
||||||
* 配置打印表格配置项
|
|
||||||
*/
|
|
||||||
type: Object as PropType<PrintTableOptions>,
|
type: Object as PropType<PrintTableOptions>,
|
||||||
default: () => ({}),
|
default: () => ({}),
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 双向绑定列表列配置项事件。
|
||||||
|
*
|
||||||
|
* 当配置为 v-model:columns 时,或者是单独调用 onUpdateColumns 时,该属性生效。
|
||||||
|
* 双向绑定语法糖事件。
|
||||||
|
*
|
||||||
|
* @default null
|
||||||
|
*/
|
||||||
onUpdateColumns: {
|
onUpdateColumns: {
|
||||||
type: [Function, Array] as PropType<
|
type: [Function, Array] as PropType<
|
||||||
MaybeArray<(arr: DataTableColumn[]) => void>
|
MaybeArray<(arr: DataTableColumn[]) => void>
|
||||||
>,
|
>,
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 双向绑定列表列配置项事件。
|
||||||
|
*
|
||||||
|
* 当配置为 v-model:columns 时,或者是单独调用 onUpdateColumns 时,该属性生效。
|
||||||
|
* 双向绑定语法糖事件。
|
||||||
|
*
|
||||||
|
* @default null
|
||||||
|
*/
|
||||||
'onUpdate:columns': {
|
'onUpdate:columns': {
|
||||||
type: [Function, Array] as PropType<
|
type: [Function, Array] as PropType<
|
||||||
MaybeArray<(arr: DataTableColumn[]) => void>
|
MaybeArray<(arr: DataTableColumn[]) => void>
|
||||||
>,
|
>,
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 该属性用于启用右键菜单后被 Table 强行代理后的右键点击事件回调。
|
||||||
|
*
|
||||||
|
* 当右键菜单不启用时,不生效。只需要使用 rowProps 属性配置右键菜单事件即可。
|
||||||
|
*
|
||||||
|
* @default null
|
||||||
|
*/
|
||||||
onContextmenu: {
|
onContextmenu: {
|
||||||
/**
|
|
||||||
*
|
|
||||||
* 该属性用于启用右键菜单后被 Table 强行代理后的右键点击事件回调
|
|
||||||
* 当右键菜单不启用时,不生效。只需要使用 rowProps 属性配置右键菜单事件即可
|
|
||||||
*/
|
|
||||||
type: [Function, Array] as PropType<
|
type: [Function, Array] as PropType<
|
||||||
MaybeArray<(row: Recordable, index: number, e: MouseEvent) => void>
|
MaybeArray<(row: Recordable, index: number, e: MouseEvent) => void>
|
||||||
>,
|
>,
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* RTable 注册挂载成功后触发的事件。
|
||||||
|
* 可以结合 useTable 方法中的 register 方法使用,然后便捷的使用 hooks。
|
||||||
|
*
|
||||||
|
* @default null
|
||||||
|
*/
|
||||||
|
onRegister: {
|
||||||
|
type: [Function, Array] as PropType<
|
||||||
|
MaybeArray<(tableInst: RTableInst) => void>
|
||||||
|
>,
|
||||||
|
default: null,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
export default props
|
export default props
|
||||||
|
@ -42,10 +42,28 @@ export interface C extends DataTableBaseColumn {
|
|||||||
children?: C[]
|
children?: C[]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface RTableInst extends Omit<DataTableInst, 'clearFilter'> {}
|
||||||
|
|
||||||
export type OverridesTableColumn<T = Recordable> = C | DataTableColumn<T>
|
export type OverridesTableColumn<T = Recordable> = C | DataTableColumn<T>
|
||||||
|
|
||||||
export interface TableInst extends Omit<TableProvider, 'wrapperRef'> {
|
export interface TableInst extends Omit<TableProvider, 'wrapperRef'> {
|
||||||
rTableInst: Omit<DataTableInst, 'clearFilter'>
|
rTableInst: RTableInst
|
||||||
}
|
}
|
||||||
|
|
||||||
export type PropsComponentPopselectKeys = 'striped' | 'bordered'
|
export type PropsComponentPopselectKeys = 'striped' | 'bordered'
|
||||||
|
|
||||||
|
type DownloadCsvParameters = Parameters<RTableInst['downloadCsv']>
|
||||||
|
|
||||||
|
export type CsvOptionsType = DownloadCsvParameters[0]
|
||||||
|
|
||||||
|
type FiltersParameters = Parameters<RTableInst['filters']>
|
||||||
|
|
||||||
|
export type FilterState = FiltersParameters[0]
|
||||||
|
|
||||||
|
export type ScrollToOptions = RTableInst['scrollTo']
|
||||||
|
|
||||||
|
type SortParameters = Parameters<RTableInst['sort']>
|
||||||
|
|
||||||
|
export type ColumnKey = SortParameters[0]
|
||||||
|
|
||||||
|
export type SortOrder = SortParameters[1]
|
||||||
|
@ -8,6 +8,7 @@ export * from './RMoreDropdown'
|
|||||||
export * from './RQRCode'
|
export * from './RQRCode'
|
||||||
export * from './RTable'
|
export * from './RTable'
|
||||||
export * from './RTransitionComponent'
|
export * from './RTransitionComponent'
|
||||||
|
export * from './RForm'
|
||||||
|
|
||||||
// 导出自定义组件类型
|
// 导出自定义组件类型
|
||||||
export type * from './RChart/src/types'
|
export type * from './RChart/src/types'
|
||||||
@ -16,3 +17,5 @@ export type * from './RIframe/src/types'
|
|||||||
export type * from './RQRCode/src/types'
|
export type * from './RQRCode/src/types'
|
||||||
export type * from './RTable/src/types'
|
export type * from './RTable/src/types'
|
||||||
export type * from './RTransitionComponent/src/types'
|
export type * from './RTransitionComponent/src/types'
|
||||||
|
export type * from './RForm/src/types'
|
||||||
|
export type * from './RModal/src/types'
|
||||||
|
@ -5,3 +5,4 @@ export * from './useDevice'
|
|||||||
export * from './useElementFullscreen'
|
export * from './useElementFullscreen'
|
||||||
export * from './useDomToImage'
|
export * from './useDomToImage'
|
||||||
export * from './usePrint'
|
export * from './usePrint'
|
||||||
|
export * from './usePagination'
|
||||||
|
@ -19,7 +19,16 @@ import { watchEffectWithTarget } from '@/utils'
|
|||||||
|
|
||||||
import type { UseWindowSizeOptions } from '@vueuse/core'
|
import type { UseWindowSizeOptions } from '@vueuse/core'
|
||||||
|
|
||||||
export interface UseDeviceOptions extends UseWindowSizeOptions {}
|
export interface UseDeviceOptions extends UseWindowSizeOptions {
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 自定义 isTabletOrSmaller 的判断尺寸。
|
||||||
|
*
|
||||||
|
* @default 768
|
||||||
|
*/
|
||||||
|
media?: number
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
@ -40,7 +49,9 @@ export function useDevice(options?: UseDeviceOptions) {
|
|||||||
const isTabletOrSmaller = ref(false)
|
const isTabletOrSmaller = ref(false)
|
||||||
|
|
||||||
const update = () => {
|
const update = () => {
|
||||||
isTabletOrSmaller.value = width.value <= 768
|
const { media = 768 } = options ?? {}
|
||||||
|
|
||||||
|
isTabletOrSmaller.value = width.value <= media
|
||||||
}
|
}
|
||||||
|
|
||||||
watchEffectWithTarget(update)
|
watchEffectWithTarget(update)
|
||||||
|
163
src/hooks/web/usePagination.ts
Normal file
163
src/hooks/web/usePagination.ts
Normal file
@ -0,0 +1,163 @@
|
|||||||
|
import { omit } from '@/utils'
|
||||||
|
|
||||||
|
import type { AnyFC } from '@/types'
|
||||||
|
import type { PaginationProps } from 'naive-ui'
|
||||||
|
|
||||||
|
type OmitKeys =
|
||||||
|
| 'themeOverrides'
|
||||||
|
| 'theme'
|
||||||
|
| 'on-update:page'
|
||||||
|
| 'on-update:page-size'
|
||||||
|
| 'onUpdatePage'
|
||||||
|
| 'onUpdatePageSize'
|
||||||
|
| 'onUpdate:page'
|
||||||
|
| 'onUpdate:page-size'
|
||||||
|
|
||||||
|
export interface UsePaginationOptions extends Omit<PaginationProps, OmitKeys> {}
|
||||||
|
|
||||||
|
const defaultOptions: UsePaginationOptions = {
|
||||||
|
page: 1,
|
||||||
|
pageSize: 10,
|
||||||
|
showSizePicker: true,
|
||||||
|
pageSizes: [10, 20, 50, 100],
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param callback 页码、页条数更新时的回调函数
|
||||||
|
* @param options 配置项
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 便捷分页 hook。
|
||||||
|
*
|
||||||
|
* @warning
|
||||||
|
* callback 暂不支持异步函数。
|
||||||
|
*/
|
||||||
|
export const usePagination = <T extends AnyFC>(
|
||||||
|
callback: T,
|
||||||
|
options?: UsePaginationOptions,
|
||||||
|
) => {
|
||||||
|
if (typeof callback !== 'function') {
|
||||||
|
throw new Error(
|
||||||
|
'[usePagination]: callback expected a function, but got ' +
|
||||||
|
typeof callback,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const omitOptions = omit(options, [
|
||||||
|
'on-update:page',
|
||||||
|
'on-update:page-size',
|
||||||
|
'onUpdatePage',
|
||||||
|
'onUpdatePageSize',
|
||||||
|
'onUpdate:page',
|
||||||
|
'onUpdate:page-size',
|
||||||
|
])
|
||||||
|
const methodsOptions = {
|
||||||
|
onUpdatePage: (page: number) => {
|
||||||
|
paginationRef.page = page
|
||||||
|
|
||||||
|
callback()
|
||||||
|
},
|
||||||
|
onUpdatePageSize: (pageSize: number) => {
|
||||||
|
paginationRef.pageSize = pageSize
|
||||||
|
paginationRef.page = 1
|
||||||
|
|
||||||
|
callback()
|
||||||
|
},
|
||||||
|
}
|
||||||
|
const paginationRef = reactive<PaginationProps>(
|
||||||
|
Object.assign({}, defaultOptions, omitOptions, methodsOptions),
|
||||||
|
)
|
||||||
|
|
||||||
|
const updatePage = paginationRef.onUpdatePage as (page: number) => void
|
||||||
|
const updatePageSize = paginationRef.onUpdatePageSize as (
|
||||||
|
pageSize: number,
|
||||||
|
) => void
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 获取总条数。
|
||||||
|
*/
|
||||||
|
const getItemCount = () => paginationRef.itemCount
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param itemCount 总条数
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 设置总条数。
|
||||||
|
*/
|
||||||
|
const setItemCount = (itemCount: number) => {
|
||||||
|
paginationRef.itemCount = itemCount
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 获取当前页页码。
|
||||||
|
*/
|
||||||
|
const getPage = () => paginationRef.page
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param page 当前页页码
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 设置当前页页码。
|
||||||
|
*
|
||||||
|
* 为了避免响应式丢失,手动调用 updatePage 方法。
|
||||||
|
*/
|
||||||
|
const setPage = (page: number) => {
|
||||||
|
updatePage(page)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 获取每页条数。
|
||||||
|
*/
|
||||||
|
const getPageSize = () => paginationRef.pageSize
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param pageSize 每页条数
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 设置每页条数。
|
||||||
|
*
|
||||||
|
* 为了避免响应式丢失,手动调用 updatePageSize 方法。
|
||||||
|
*/
|
||||||
|
const setPageSize = (pageSize: number) => {
|
||||||
|
updatePageSize(pageSize)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 获取分页配置,通常可以用来传递给 RTable 组件。
|
||||||
|
*/
|
||||||
|
const getPagination = () => paginationRef as UsePaginationOptions
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 获取回调函数。
|
||||||
|
*/
|
||||||
|
const getCallback = callback
|
||||||
|
|
||||||
|
return {
|
||||||
|
updatePage,
|
||||||
|
updatePageSize,
|
||||||
|
getItemCount,
|
||||||
|
setItemCount,
|
||||||
|
getPage,
|
||||||
|
setPage,
|
||||||
|
getPageSize,
|
||||||
|
setPageSize,
|
||||||
|
getPagination,
|
||||||
|
getCallback,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export type UsePaginationReturn = ReturnType<typeof usePagination>
|
@ -24,5 +24,6 @@
|
|||||||
"TemplateHooks": "Template Api",
|
"TemplateHooks": "Template Api",
|
||||||
"Modal": "Modal",
|
"Modal": "Modal",
|
||||||
"ContextMenu": "Right Click Menu",
|
"ContextMenu": "Right Click Menu",
|
||||||
"CacheDemo": "Cache Utils Demo"
|
"CacheDemo": "Cache Utils Demo",
|
||||||
|
"Form": "Form"
|
||||||
}
|
}
|
||||||
|
@ -24,5 +24,6 @@
|
|||||||
"TemplateHooks": "模板内置 Api",
|
"TemplateHooks": "模板内置 Api",
|
||||||
"Modal": "模态框",
|
"Modal": "模态框",
|
||||||
"ContextMenu": "右键菜单",
|
"ContextMenu": "右键菜单",
|
||||||
"CacheDemo": "缓存工具函数"
|
"CacheDemo": "缓存工具函数",
|
||||||
|
"Form": "表单"
|
||||||
}
|
}
|
||||||
|
@ -11,9 +11,6 @@ const cacheDemo: AppRouteRecordRaw = {
|
|||||||
i18nKey: t('menu.CacheDemo'),
|
i18nKey: t('menu.CacheDemo'),
|
||||||
icon: 'other',
|
icon: 'other',
|
||||||
order: 3,
|
order: 3,
|
||||||
extra: {
|
|
||||||
label: 'new',
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -11,6 +11,9 @@ const echart: AppRouteRecordRaw = {
|
|||||||
i18nKey: t('menu.Echart'),
|
i18nKey: t('menu.Echart'),
|
||||||
icon: 'echart',
|
icon: 'echart',
|
||||||
order: 1,
|
order: 1,
|
||||||
|
extra: {
|
||||||
|
label: 'useChart',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
20
src/router/modules/demo/form.ts
Normal file
20
src/router/modules/demo/form.ts
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
import { t } from '@/hooks/web/useI18n'
|
||||||
|
import { LAYOUT } from '@/router/constant'
|
||||||
|
|
||||||
|
import type { AppRouteRecordRaw } from '@/router/types'
|
||||||
|
|
||||||
|
const form: AppRouteRecordRaw = {
|
||||||
|
path: '/form',
|
||||||
|
name: 'FormView',
|
||||||
|
component: () => import('@/views/demo/form'),
|
||||||
|
meta: {
|
||||||
|
i18nKey: t('menu.Form'),
|
||||||
|
icon: 'other',
|
||||||
|
order: 2,
|
||||||
|
extra: {
|
||||||
|
label: 'useForm',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
export default form
|
@ -12,6 +12,9 @@ const mockDemo: AppRouteRecordRaw = {
|
|||||||
icon: 'other',
|
icon: 'other',
|
||||||
order: 3,
|
order: 3,
|
||||||
keepAlive: false,
|
keepAlive: false,
|
||||||
|
extra: {
|
||||||
|
label: 'usePagination',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -11,6 +11,9 @@ const table: AppRouteRecordRaw = {
|
|||||||
i18nKey: t('menu.Table'),
|
i18nKey: t('menu.Table'),
|
||||||
icon: 'other',
|
icon: 'other',
|
||||||
order: 2,
|
order: 2,
|
||||||
|
extra: {
|
||||||
|
label: 'useTable',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -136,9 +136,8 @@ export const piniaMenuStore = defineStore(
|
|||||||
default: () => label.value,
|
default: () => label.value,
|
||||||
}),
|
}),
|
||||||
breadcrumbLabel: label.value,
|
breadcrumbLabel: label.value,
|
||||||
/** 检查该菜单项是否展示 */
|
|
||||||
} as AppMenuOption
|
} as AppMenuOption
|
||||||
/** 合并 icon */
|
/** 合并 icon, extra */
|
||||||
const attr: AppMenuOption = Object.assign({}, route, {
|
const attr: AppMenuOption = Object.assign({}, route, {
|
||||||
icon: createMenuIcon(option),
|
icon: createMenuIcon(option),
|
||||||
extra: createMenuExtra(option),
|
extra: createMenuExtra(option),
|
||||||
|
@ -92,6 +92,8 @@ export type CipherParams = CryptoJS.lib.CipherParams
|
|||||||
|
|
||||||
export type AnyFC<P = any, R = any> = (...args: P[]) => R
|
export type AnyFC<P = any, R = any> = (...args: P[]) => R
|
||||||
|
|
||||||
|
export type VoidFC = (...args: any[]) => void
|
||||||
|
|
||||||
export type PartialCSSStyleDeclaration = Partial<
|
export type PartialCSSStyleDeclaration = Partial<
|
||||||
Record<keyof CSSProperties, string>
|
Record<keyof CSSProperties, string>
|
||||||
>
|
>
|
||||||
|
@ -3,12 +3,25 @@ import './index.scss'
|
|||||||
import { NCard, NSwitch, NFlex, NH2, NButton } from 'naive-ui'
|
import { NCard, NSwitch, NFlex, NH2, NButton } from 'naive-ui'
|
||||||
import { RChart } from '@/components'
|
import { RChart } from '@/components'
|
||||||
|
|
||||||
|
import { useChart } from '@/components'
|
||||||
|
|
||||||
import type { RChartType } from '@/components'
|
import type { RChartType } from '@/components'
|
||||||
|
|
||||||
const Echart = defineComponent({
|
const Echart = defineComponent({
|
||||||
name: 'REchart',
|
name: 'REchart',
|
||||||
setup() {
|
setup() {
|
||||||
const baseChartRef = ref<RChartType.RChartInst>()
|
const [register, { getChartInstance, dispose, render, isDispose }] =
|
||||||
|
useChart()
|
||||||
|
const [
|
||||||
|
register2,
|
||||||
|
{
|
||||||
|
getChartInstance: getChartInstance2,
|
||||||
|
dispose: dispose2,
|
||||||
|
render: render2,
|
||||||
|
isDispose: isDispose2,
|
||||||
|
},
|
||||||
|
] = useChart()
|
||||||
|
|
||||||
const chartLoading = ref(false)
|
const chartLoading = ref(false)
|
||||||
const chartAria = ref(false)
|
const chartAria = ref(false)
|
||||||
const state = reactive({
|
const state = reactive({
|
||||||
@ -179,15 +192,15 @@ const Echart = defineComponent({
|
|||||||
}
|
}
|
||||||
|
|
||||||
const mountChart = () => {
|
const mountChart = () => {
|
||||||
if (!baseChartRef.value?.isDispose()) {
|
if (isDispose()) {
|
||||||
baseChartRef.value?.render()
|
render()
|
||||||
} else {
|
} else {
|
||||||
window.$message.warning('图表已经渲染')
|
window.$message.warning('图表已经渲染')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const unmountChart = () => {
|
const unmountChart = () => {
|
||||||
baseChartRef.value?.dispose()
|
dispose()
|
||||||
}
|
}
|
||||||
|
|
||||||
const updateChartOptions = () => {
|
const updateChartOptions = () => {
|
||||||
@ -203,7 +216,6 @@ const Echart = defineComponent({
|
|||||||
|
|
||||||
return {
|
return {
|
||||||
baseOptions,
|
baseOptions,
|
||||||
baseChartRef,
|
|
||||||
chartLoading,
|
chartLoading,
|
||||||
handleLoadingShow,
|
handleLoadingShow,
|
||||||
chartAria,
|
chartAria,
|
||||||
@ -214,9 +226,16 @@ const Echart = defineComponent({
|
|||||||
mountChart,
|
mountChart,
|
||||||
unmountChart,
|
unmountChart,
|
||||||
updateChartOptions,
|
updateChartOptions,
|
||||||
|
register,
|
||||||
|
register2,
|
||||||
|
dispose2,
|
||||||
|
render2,
|
||||||
|
isDispose2,
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
render() {
|
render() {
|
||||||
|
const { register, register2, dispose2, render2, isDispose2 } = this
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div class="echart">
|
<div class="echart">
|
||||||
<NCard title="chart 组件">
|
<NCard title="chart 组件">
|
||||||
@ -246,6 +265,9 @@ const Echart = defineComponent({
|
|||||||
属性,只有元素在可见范围才会渲染图表,可以滚动查看效果
|
属性,只有元素在可见范围才会渲染图表,可以滚动查看效果
|
||||||
</h3>
|
</h3>
|
||||||
</li>
|
</li>
|
||||||
|
<li>
|
||||||
|
<h3>7. useChart 方法</h3>
|
||||||
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
</NCard>
|
</NCard>
|
||||||
<NCard title="预设 card 风格图表">
|
<NCard title="预设 card 风格图表">
|
||||||
@ -258,8 +280,8 @@ const Echart = defineComponent({
|
|||||||
</NFlex>
|
</NFlex>
|
||||||
<div class="chart--container">
|
<div class="chart--container">
|
||||||
<RChart
|
<RChart
|
||||||
|
onRegister={register}
|
||||||
title="周销售量"
|
title="周销售量"
|
||||||
ref="baseChartRef"
|
|
||||||
autoChangeTheme
|
autoChangeTheme
|
||||||
options={this.baseLineOptions}
|
options={this.baseLineOptions}
|
||||||
showAria={this.chartAria}
|
showAria={this.chartAria}
|
||||||
@ -268,13 +290,30 @@ const Echart = defineComponent({
|
|||||||
</div>
|
</div>
|
||||||
</NCard>
|
</NCard>
|
||||||
<NCard title="不跟随主题切换的暗色主题可视化图,并且手动指定原始主题色">
|
<NCard title="不跟随主题切换的暗色主题可视化图,并且手动指定原始主题色">
|
||||||
<div class="chart--container">
|
<NFlex vertical>
|
||||||
<RChart
|
<NFlex>
|
||||||
autoChangeTheme={false}
|
<NButton
|
||||||
theme="default"
|
onClick={() => {
|
||||||
options={this.baseOptions}
|
if (isDispose2()) {
|
||||||
/>
|
render2()
|
||||||
</div>
|
} else {
|
||||||
|
window.$message.warning('图表已经渲染')
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
渲染
|
||||||
|
</NButton>
|
||||||
|
<NButton onClick={dispose2.bind(this)}>卸载</NButton>
|
||||||
|
</NFlex>
|
||||||
|
<div class="chart--container">
|
||||||
|
<RChart
|
||||||
|
onRegister={register2}
|
||||||
|
autoChangeTheme={false}
|
||||||
|
theme="default"
|
||||||
|
options={this.baseOptions}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</NFlex>
|
||||||
</NCard>
|
</NCard>
|
||||||
<NCard title="加载动画">
|
<NCard title="加载动画">
|
||||||
<NSwitch
|
<NSwitch
|
||||||
|
157
src/views/demo/form/index.tsx
Normal file
157
src/views/demo/form/index.tsx
Normal file
@ -0,0 +1,157 @@
|
|||||||
|
/**
|
||||||
|
*
|
||||||
|
* @author Ray <https://github.com/XiaoDaiGua-Ray>
|
||||||
|
*
|
||||||
|
* @date 2024-03-27
|
||||||
|
*
|
||||||
|
* @workspace ray-template
|
||||||
|
*
|
||||||
|
* @remark 今天也是元气满满撸代码的一天
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { RForm } from '@/components'
|
||||||
|
import {
|
||||||
|
NFormItemGi,
|
||||||
|
NDatePicker,
|
||||||
|
NGrid,
|
||||||
|
NInput,
|
||||||
|
NInputNumber,
|
||||||
|
NFlex,
|
||||||
|
NButton,
|
||||||
|
NRadio,
|
||||||
|
NRadioGroup,
|
||||||
|
} from 'naive-ui'
|
||||||
|
|
||||||
|
import { useForm } from '@/components'
|
||||||
|
|
||||||
|
import type { RFormRules } from '@/components'
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'RFormDemo',
|
||||||
|
setup() {
|
||||||
|
// 使用以下 hooks 的时候,应该注意调用时机
|
||||||
|
const [
|
||||||
|
register,
|
||||||
|
{ getFormInstance, validate, restoreValidation, formModel, formRules },
|
||||||
|
] = useForm(
|
||||||
|
{
|
||||||
|
name: null,
|
||||||
|
age: null,
|
||||||
|
gender: null,
|
||||||
|
date: null,
|
||||||
|
remark: null,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: {
|
||||||
|
required: true,
|
||||||
|
message: '请输入姓名',
|
||||||
|
trigger: ['blur', 'change'],
|
||||||
|
},
|
||||||
|
date: {
|
||||||
|
required: true,
|
||||||
|
message: '请选择日期',
|
||||||
|
trigger: ['blur', 'change'],
|
||||||
|
type: 'number',
|
||||||
|
},
|
||||||
|
gender: {
|
||||||
|
required: true,
|
||||||
|
message: '请选择性别',
|
||||||
|
trigger: 'change',
|
||||||
|
},
|
||||||
|
age: {
|
||||||
|
required: true,
|
||||||
|
message: '请输入年龄',
|
||||||
|
trigger: ['blur', 'change'],
|
||||||
|
type: 'number',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 如果待验证数据类型为: number, array 等,需要手动设置 type 类型。
|
||||||
|
* 具体可以吃查看: async-validator type
|
||||||
|
* @see https://github.com/yiminghe/async-validator?tab=readme-ov-file#type
|
||||||
|
*
|
||||||
|
* 如果你需要自定义验证,可以查看:naive ui custom validation
|
||||||
|
* @see https://www.naiveui.com/zh-CN/dark/components/form#custom-validation.vue
|
||||||
|
*
|
||||||
|
* 如果只是简单的 rules 管理,可以在初始化 useForm 的时候传入第二个参数;
|
||||||
|
* 然后使用 formRules 方法获取到初始化 rules 数据。
|
||||||
|
*/
|
||||||
|
const rules = ref(formRules())
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* 如果只是简单的数据,可以在初始化 useForm 的时候直接传入第一个参数;
|
||||||
|
* 然后使用 formModel 方法获取到初始化 model 数据。
|
||||||
|
*
|
||||||
|
* 动态的复杂数据,不建议使用该方法管理 model;手动的拆分出来是一个更加好的选择。
|
||||||
|
*/
|
||||||
|
const condition = ref(formModel())
|
||||||
|
|
||||||
|
return {
|
||||||
|
register,
|
||||||
|
rules,
|
||||||
|
condition,
|
||||||
|
restoreValidation,
|
||||||
|
formModel,
|
||||||
|
validate,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
render() {
|
||||||
|
const { rules } = this
|
||||||
|
const { register, restoreValidation, formModel, validate } = this
|
||||||
|
|
||||||
|
return (
|
||||||
|
<RForm onRegister={register} rules={rules} model={this.condition}>
|
||||||
|
<NGrid cols={24} xGap={24}>
|
||||||
|
<NFormItemGi label="姓名" path="name" span={12}>
|
||||||
|
<NInput v-model:value={this.condition.name} />
|
||||||
|
</NFormItemGi>
|
||||||
|
<NFormItemGi label="年龄" path="age" span={12}>
|
||||||
|
<NInputNumber
|
||||||
|
v-model:value={this.condition.age}
|
||||||
|
showButton={false}
|
||||||
|
style="width: 100%"
|
||||||
|
/>
|
||||||
|
</NFormItemGi>
|
||||||
|
<NFormItemGi label="出生日期" path="date" span={12}>
|
||||||
|
<NDatePicker
|
||||||
|
v-model:value={this.condition.date}
|
||||||
|
style="width: 100%"
|
||||||
|
/>
|
||||||
|
</NFormItemGi>
|
||||||
|
<NFormItemGi label="性别" path="gender" span={12}>
|
||||||
|
<NRadioGroup v-model:value={this.condition.gender}>
|
||||||
|
<NRadio value="girl">女</NRadio>
|
||||||
|
<NRadio value="man">男</NRadio>
|
||||||
|
</NRadioGroup>
|
||||||
|
</NFormItemGi>
|
||||||
|
<NFormItemGi label="备注信息" span={24}>
|
||||||
|
<NInput type="textarea" v-model:value={this.condition.remark} />
|
||||||
|
</NFormItemGi>
|
||||||
|
<NFormItemGi span={24}>
|
||||||
|
<NFlex justify="flex-end" style="width: 100%">
|
||||||
|
<NButton
|
||||||
|
type="info"
|
||||||
|
onClick={() => {
|
||||||
|
this.condition = formModel()
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
重置表单为初始状态
|
||||||
|
</NButton>
|
||||||
|
<NButton type="warning" onClick={restoreValidation.bind(this)}>
|
||||||
|
移除校验状态
|
||||||
|
</NButton>
|
||||||
|
<NButton type="primary" onClick={() => validate()}>
|
||||||
|
校验
|
||||||
|
</NButton>
|
||||||
|
</NFlex>
|
||||||
|
</NFormItemGi>
|
||||||
|
</NGrid>
|
||||||
|
</RForm>
|
||||||
|
)
|
||||||
|
},
|
||||||
|
})
|
@ -14,6 +14,7 @@ import { RCollapseGrid, RTable } from '@/components'
|
|||||||
|
|
||||||
import { useHookPlusRequest } from '@/axios'
|
import { useHookPlusRequest } from '@/axios'
|
||||||
import { getPersonList } from '@/api/demo/mock/person'
|
import { getPersonList } from '@/api/demo/mock/person'
|
||||||
|
import { usePagination } from '@/hooks'
|
||||||
|
|
||||||
import type { Person } from '@/api/demo/mock/person'
|
import type { Person } from '@/api/demo/mock/person'
|
||||||
|
|
||||||
@ -86,24 +87,23 @@ const MockDemo = defineComponent({
|
|||||||
const condition = reactive({
|
const condition = reactive({
|
||||||
email: null,
|
email: null,
|
||||||
})
|
})
|
||||||
const paginationRef = reactive({
|
|
||||||
page: 1,
|
|
||||||
pageSize: 10,
|
|
||||||
itemCount: 0,
|
|
||||||
pageSizes: [10, 20, 30, 40, 50],
|
|
||||||
showSizePicker: true,
|
|
||||||
onUpdatePage: (page: number) => {
|
|
||||||
paginationRef.page = page
|
|
||||||
|
|
||||||
getPerson()
|
const {
|
||||||
},
|
getPagination,
|
||||||
onUpdatePageSize: (pageSize: number) => {
|
getPage,
|
||||||
paginationRef.pageSize = pageSize
|
getPageSize,
|
||||||
paginationRef.page = 1
|
setItemCount,
|
||||||
|
getCallback,
|
||||||
getPerson()
|
setPage,
|
||||||
},
|
setPageSize,
|
||||||
|
} = usePagination(() => {
|
||||||
|
personFetchRun({
|
||||||
|
page: getPage(),
|
||||||
|
pageSize: getPageSize(),
|
||||||
|
email: condition.email,
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
const paginationRef = getPagination()
|
||||||
const {
|
const {
|
||||||
data: personData,
|
data: personData,
|
||||||
loading: personLoading,
|
loading: personLoading,
|
||||||
@ -111,36 +111,26 @@ const MockDemo = defineComponent({
|
|||||||
} = useHookPlusRequest(getPersonList, {
|
} = useHookPlusRequest(getPersonList, {
|
||||||
defaultParams: [
|
defaultParams: [
|
||||||
{
|
{
|
||||||
page: paginationRef.page,
|
page: getPage(),
|
||||||
pageSize: paginationRef.pageSize,
|
pageSize: getPageSize(),
|
||||||
email: condition.email,
|
email: condition.email,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
onSuccess: (res) => {
|
onSuccess: (res) => {
|
||||||
const { total } = res
|
const { total } = res
|
||||||
|
|
||||||
paginationRef.itemCount = total
|
setItemCount(total)
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
const getPerson = () => {
|
|
||||||
const { pageSize, page } = paginationRef
|
|
||||||
const { email } = condition
|
|
||||||
|
|
||||||
personFetchRun({
|
|
||||||
page,
|
|
||||||
pageSize,
|
|
||||||
email,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
personData,
|
personData,
|
||||||
personLoading,
|
personLoading,
|
||||||
paginationRef,
|
getPagination,
|
||||||
columns,
|
columns,
|
||||||
...toRefs(condition),
|
...toRefs(condition),
|
||||||
getPerson,
|
getCallback,
|
||||||
|
paginationRef,
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
render() {
|
render() {
|
||||||
@ -170,7 +160,7 @@ const MockDemo = defineComponent({
|
|||||||
</>
|
</>
|
||||||
),
|
),
|
||||||
action: () => (
|
action: () => (
|
||||||
<NButton type="primary" onClick={this.getPerson.bind(this)}>
|
<NButton type="primary" onClick={this.getCallback.bind(this)}>
|
||||||
搜索
|
搜索
|
||||||
</NButton>
|
</NButton>
|
||||||
),
|
),
|
||||||
|
@ -12,6 +12,8 @@
|
|||||||
import { RModal } from '@/components'
|
import { RModal } from '@/components'
|
||||||
import { NButton, NCard, NFlex } from 'naive-ui'
|
import { NButton, NCard, NFlex } from 'naive-ui'
|
||||||
|
|
||||||
|
import { useModal } from '@/components'
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
name: 'ModalDemo',
|
name: 'ModalDemo',
|
||||||
setup() {
|
setup() {
|
||||||
@ -20,12 +22,36 @@ export default defineComponent({
|
|||||||
modal2: false,
|
modal2: false,
|
||||||
modal3: false,
|
modal3: false,
|
||||||
})
|
})
|
||||||
|
const { create } = useModal()
|
||||||
|
|
||||||
|
const createCardModal = () => {
|
||||||
|
create({
|
||||||
|
title: '卡片模态框',
|
||||||
|
dad: true,
|
||||||
|
preset: 'card',
|
||||||
|
content: '我可以被拖拽的全屏card模态框',
|
||||||
|
fullscreen: true,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const createDialogModal = () => {
|
||||||
|
create({
|
||||||
|
title: '模态框',
|
||||||
|
content: '内容',
|
||||||
|
preset: 'dialog',
|
||||||
|
dad: true,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
...toRefs(state),
|
...toRefs(state),
|
||||||
|
createCardModal,
|
||||||
|
createDialogModal,
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
render() {
|
render() {
|
||||||
|
const { createCardModal, createDialogModal } = this
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<NFlex vertical>
|
<NFlex vertical>
|
||||||
<NCard title="props">
|
<NCard title="props">
|
||||||
@ -91,6 +117,12 @@ export default defineComponent({
|
|||||||
所有的宽度配置属性都会注入一个对应的 `css variable`,有时候会用上。
|
所有的宽度配置属性都会注入一个对应的 `css variable`,有时候会用上。
|
||||||
</h4>
|
</h4>
|
||||||
</NCard>
|
</NCard>
|
||||||
|
<NCard title="重写 useModal">
|
||||||
|
<NButton onClick={createCardModal.bind(this)}>创建卡片模态框</NButton>
|
||||||
|
<NButton onClick={createDialogModal.bind(this)}>
|
||||||
|
创建对话框模态框
|
||||||
|
</NButton>
|
||||||
|
</NCard>
|
||||||
</NFlex>
|
</NFlex>
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
|
@ -23,20 +23,29 @@ import {
|
|||||||
} from 'naive-ui'
|
} from 'naive-ui'
|
||||||
import { RCollapseGrid, RTable, RIcon, RMoreDropdown } from '@/components'
|
import { RCollapseGrid, RTable, RIcon, RMoreDropdown } from '@/components'
|
||||||
|
|
||||||
|
import { uuid } from '@/utils'
|
||||||
|
import { useTable } from '@/components'
|
||||||
|
|
||||||
import type { DataTableColumns } from 'naive-ui'
|
import type { DataTableColumns } from 'naive-ui'
|
||||||
import type { RTableType } from '@/components'
|
|
||||||
|
|
||||||
type RowData = {
|
type RowData = {
|
||||||
key: number
|
key: number | string
|
||||||
name: string
|
name: string
|
||||||
age: number
|
age: number
|
||||||
address: string
|
address: string
|
||||||
tags: string[]
|
tags: string[]
|
||||||
|
remark: string
|
||||||
}
|
}
|
||||||
|
|
||||||
const TableView = defineComponent({
|
const TableView = defineComponent({
|
||||||
name: 'TableView',
|
name: 'TableView',
|
||||||
setup() {
|
setup() {
|
||||||
|
// 使用以下 hooks 的时候,应该注意调用时机
|
||||||
|
const [
|
||||||
|
register,
|
||||||
|
{ getTableInstance, clearFilters, clearSorter, scrollTo, filters, sort },
|
||||||
|
] = useTable()
|
||||||
|
|
||||||
const baseColumns = [
|
const baseColumns = [
|
||||||
{
|
{
|
||||||
title: 'Name',
|
title: 'Name',
|
||||||
@ -68,7 +77,7 @@ const TableView = defineComponent({
|
|||||||
{
|
{
|
||||||
title: 'Remark',
|
title: 'Remark',
|
||||||
key: 'remark',
|
key: 'remark',
|
||||||
width: 300,
|
width: 100,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: 'Action',
|
title: 'Action',
|
||||||
@ -98,32 +107,7 @@ const TableView = defineComponent({
|
|||||||
const actionColumns = ref<DataTableColumns<RowData>>(
|
const actionColumns = ref<DataTableColumns<RowData>>(
|
||||||
[...baseColumns].map((curr) => ({ ...curr, width: 400 })),
|
[...baseColumns].map((curr) => ({ ...curr, width: 400 })),
|
||||||
)
|
)
|
||||||
const tableData = ref([
|
const tableData = ref<RowData[]>([])
|
||||||
{
|
|
||||||
key: 0,
|
|
||||||
name: 'John Brown',
|
|
||||||
age: 32,
|
|
||||||
address: 'New York No. 1 Lake Park',
|
|
||||||
tags: ['nice', 'developer'],
|
|
||||||
remark: '我是一条很长很长的备注',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
key: 1,
|
|
||||||
name: 'Jim Green',
|
|
||||||
age: 42,
|
|
||||||
address: 'London No. 1 Lake Park',
|
|
||||||
tags: ['wow'],
|
|
||||||
remark: '我是一条很长很长的备注',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
key: 2,
|
|
||||||
name: 'Joe Black',
|
|
||||||
age: 32,
|
|
||||||
address: 'Sidney No. 1 Lake Park',
|
|
||||||
tags: ['cool', 'teacher'],
|
|
||||||
remark: '我是一条很长很长的备注',
|
|
||||||
},
|
|
||||||
])
|
|
||||||
const tableMenuOptions = [
|
const tableMenuOptions = [
|
||||||
{
|
{
|
||||||
label: '编辑',
|
label: '编辑',
|
||||||
@ -140,20 +124,38 @@ const TableView = defineComponent({
|
|||||||
tableLoading: false,
|
tableLoading: false,
|
||||||
})
|
})
|
||||||
|
|
||||||
const handleMenuSelect = (key: string | number) => {
|
const createTableData = () => {
|
||||||
|
for (let i = 0; i < 20; i++) {
|
||||||
|
tableData.value.push({
|
||||||
|
key: uuid(),
|
||||||
|
name: 'John Brown',
|
||||||
|
age: i + 20,
|
||||||
|
address: 'New York No. 1 Lake Park',
|
||||||
|
tags: ['nice', 'developer'],
|
||||||
|
remark: '我是一条很长很长的备注',
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const menuSelect = (key: string | number) => {
|
||||||
window.$message.info(`${key}`)
|
window.$message.info(`${key}`)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
createTableData()
|
||||||
|
|
||||||
return {
|
return {
|
||||||
...toRefs(state),
|
...toRefs(state),
|
||||||
tableData,
|
tableData,
|
||||||
actionColumns,
|
actionColumns,
|
||||||
baseColumns,
|
baseColumns,
|
||||||
tableMenuOptions,
|
tableMenuOptions,
|
||||||
handleMenuSelect,
|
menuSelect,
|
||||||
|
register,
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
render() {
|
render() {
|
||||||
|
const { register } = this
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<NFlex vertical>
|
<NFlex vertical>
|
||||||
<NCard title="RTable">
|
<NCard title="RTable">
|
||||||
@ -199,7 +201,8 @@ const TableView = defineComponent({
|
|||||||
}}
|
}}
|
||||||
</RCollapseGrid>
|
</RCollapseGrid>
|
||||||
<RTable
|
<RTable
|
||||||
scrollX={2000}
|
onRegister={register.bind(this)}
|
||||||
|
scrollX={1000}
|
||||||
title={
|
title={
|
||||||
<NFlex align="center">
|
<NFlex align="center">
|
||||||
<span>标题插槽:</span>
|
<span>标题插槽:</span>
|
||||||
@ -215,7 +218,7 @@ const TableView = defineComponent({
|
|||||||
}}
|
}}
|
||||||
contextMenuOptions={this.tableMenuOptions}
|
contextMenuOptions={this.tableMenuOptions}
|
||||||
loading={this.tableLoading}
|
loading={this.tableLoading}
|
||||||
onContextMenuClick={this.handleMenuSelect.bind(this)}
|
onContextMenuClick={this.menuSelect.bind(this)}
|
||||||
toolOptions={[
|
toolOptions={[
|
||||||
<NPopover>
|
<NPopover>
|
||||||
{{
|
{{
|
||||||
|
@ -18,6 +18,14 @@ export default defineConfig((configEnv) =>
|
|||||||
environment: 'happy-dom',
|
environment: 'happy-dom',
|
||||||
globals: true,
|
globals: true,
|
||||||
poolOptions: {
|
poolOptions: {
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* 如此配置是为避免: Module did not self-register... 错误;
|
||||||
|
* 该错误是一个历史悠久遗留问题,可以查看该 issues:
|
||||||
|
* @see https://github.com/vitest-dev/vitest/issues/740
|
||||||
|
*
|
||||||
|
* 目前暂时没有更好的解决方案,这么做会导致单测执行速度变慢,但是可以避免错误,后续有更好的解决方案会更新。
|
||||||
|
*/
|
||||||
threads: {
|
threads: {
|
||||||
maxThreads: 1,
|
maxThreads: 1,
|
||||||
minThreads: 0,
|
minThreads: 0,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user