version: v4.8.0

This commit is contained in:
XiaoDaiGua-Ray 2024-04-13 15:24:51 +08:00
parent 85f7f28dc4
commit e95c06b009
165 changed files with 1981 additions and 807 deletions

View File

@ -202,5 +202,12 @@ module.exports = {
message: 'Disallow using key as a custom attribute', message: 'Disallow using key as a custom attribute',
}, },
], ],
'no-restricted-syntax': [
'error',
{
selector: "CallExpression[callee.property.name='deprecated']",
message: 'Using deprecated API is not allowed.',
},
],
}, },
} }

View File

@ -19,6 +19,7 @@
"alias-skip.allowedsuffix": ["ts", "tsx"], "alias-skip.allowedsuffix": ["ts", "tsx"],
"alias-skip.rootpath": "package.json", "alias-skip.rootpath": "package.json",
"cSpell.words": [ "cSpell.words": [
"bezier",
"Clickoutside", "Clickoutside",
"commitmsg", "commitmsg",
"datetimerange", "datetimerange",

View File

@ -1,5 +1,44 @@
# CHANGE LOG # CHANGE LOG
## 4.8.0
全局破坏性更新。移除了很多包、方法,请谨慎更新。
## Feats
- 移除无意义依赖包
- `RTable` 组件
- 强制约束使用 `useTable` 方法操作实例,移除 `expose` 暴露
- 新增 `useCheckedRowKeys` 方法,用于操作表格选中行,该方法仅适用于选中行操作(多选、单选)
- `RForm`, `RChart` 强制约束使用对应 `useForm`, `useChart` 方法操作实例,移除 `expose` 暴露
- 优化 `usePagination` 方法修改 `paginationRef` 值类型,使用 `Ref` 签名类型
- `eslint` 规则新增禁用被标记弃用方法
- 移除 `omit`, `pick` 方法,使用 `lodash-es` 包替代
- 新增 `RSegment` 分段器组件
> 由于是基于 [NTabs](https://www.naiveui.com/zh-CN/dark/components/tabs) 组件二开所以也继承了该组件的一些特性bug
- `svg icon` 支持分包管理
- `vite-helper` 包中所有方法进行拆分,约定按照功能模块进行拆分。如果该方法属于 `vite` 插件,按照 `xx-xx-plugin` 方式命名。
- 统一所有模块的文件命名
- `class`, `hooks` 方法统一为小驼峰命名
> 示例: `MyClass`, `useMyHooks`
- `component` 的文件夹与组件名称统一为大驼峰命名
> 示例: `MyComponent`, `ChildComponent`
- `utils`, `custom function` 统一为蛇形命名
> 示例: `custom-function`, `custom-utils`
- 新增 `v-ripple` 水波纹指令
- 新增 `v-lazy-show` 惰性 `v-show` 指令
- `GlobalSearchButton` 组件样式优化
- `useElementFullscreen` 方法
- 移除缓存 `transition` 样式还原方式,使用 `options.transition` 方式配置,默认为 `all 0.3s var(--r-bezier)`
- 优化 `LayoutContent` 网页最大化动画效果
- `global-variable` 新增 `getVariable` 方法,允许解构获取全局响应式变量
- 移除 `SiderBar` 组件的 `tip` 提示
- 补充了一些注释
## Fixes
- 修复 `RTable C` 组件对于 `columns type` 项无法有效兼容问题
## 4.7.5 ## 4.7.5
## Feats ## Feats

View File

@ -1,4 +1,4 @@
import { prefixCacheKey } from '../../src/utils/app/prefixCacheKey' import { prefixCacheKey } from '../../src/utils/app/prefix-cache-key'
describe('prefixCacheKey', () => { describe('prefixCacheKey', () => {
it('should return the key with the default prefix', () => { it('should return the key with the default prefix', () => {

View File

@ -1,39 +0,0 @@
import { omit } from '../../src/utils/basic'
describe('omit', () => {
it('should omit key from object', () => {
const obj = { a: 1, b: 2, c: 3 }
const result = omit(obj, 'b')
expect(result).toEqual({ a: 1, c: 3 })
})
it('should omit key from the array argument', () => {
const obj = { a: 1, b: 2, c: 3 }
const result = omit(obj, ['a', 'c'])
expect(result).toEqual({ b: 2 })
})
it('should return empty object if no keys are provided', () => {
const obj = { a: 1, b: 2, c: 3 }
const result = omit(obj, Object.keys(obj))
expect(result).toEqual({})
})
it('should return empty object if object is empty', () => {
const obj = {}
const result = omit(obj, 'a', 'b')
expect(result).toEqual({})
})
it('an empty object should be returned if null or undefined is passed', () => {
const result1 = omit(null)
const result2 = omit(void 0)
expect(result1).toEqual({})
expect(result2).toEqual({})
})
})

View File

@ -1,25 +0,0 @@
import { pick } from '../../src/utils/basic'
describe('pick', () => {
it('should pick keys from object', () => {
const obj = { a: 1, b: 2, c: 3 }
const result = pick(obj, 'a', 'c')
expect(result).toEqual({ a: 1, c: 3 })
})
it('should pick keys from the array argument', () => {
const obj = { a: 1, b: 2, c: 3 }
const result = pick(obj, ['a', 'c'])
expect(result).toEqual({ a: 1, c: 3 })
})
it('an empty object should be returned if null or undefined is passed', () => {
const result1 = pick(null)
const result2 = pick(void 0)
expect(result1).toEqual({})
expect(result2).toEqual({})
})
})

View File

@ -49,6 +49,6 @@ describe('useElementFullscreen', async () => {
await nextTick() await nextTick()
expect(div.style.transition).toBe('') expect(!div.style.transition).not.toBe(true)
}) })
}) })

View File

@ -1,4 +1,4 @@
import { effectDispose } from '../../src/utils/vue/effectDispose' import { effectDispose } from '../../src/utils/vue/effect-dispose'
describe('effectDispose', () => { describe('effectDispose', () => {
it('should return false if getCurrentScope is null', () => { it('should return false if getCurrentScope is null', () => {

View File

@ -1,4 +1,4 @@
import { renderNode } from '../../src/utils/vue/renderNode' import { renderNode } from '../../src/utils/vue/render-node'
import createRefElement from '../utils/createRefElement' import createRefElement from '../utils/createRefElement'
describe('renderNode', () => { describe('renderNode', () => {

View File

@ -1,7 +1,7 @@
{ {
"name": "ray-template", "name": "ray-template",
"private": false, "private": false,
"version": "4.7.5", "version": "4.8.0",
"type": "module", "type": "module",
"engines": { "engines": {
"node": "^18.0.0 || >=20.0.0", "node": "^18.0.0 || >=20.0.0",
@ -37,7 +37,6 @@
"awesome-qr": "2.1.5-rc.0", "awesome-qr": "2.1.5-rc.0",
"axios": "^1.6.7", "axios": "^1.6.7",
"clipboard": "^2.0.11", "clipboard": "^2.0.11",
"crypto-js": "^4.1.1",
"currency.js": "^2.0.4", "currency.js": "^2.0.4",
"dayjs": "^1.11.10", "dayjs": "^1.11.10",
"dom-to-image": "2.6.0", "dom-to-image": "2.6.0",
@ -53,11 +52,9 @@
"vue-demi": "0.14.6", "vue-demi": "0.14.6",
"vue-hooks-plus": "1.8.8", "vue-hooks-plus": "1.8.8",
"vue-i18n": "^9.9.0", "vue-i18n": "^9.9.0",
"vue-router": "^4.2.5" "vue-router": "^4.3.0"
}, },
"devDependencies": { "devDependencies": {
"@babel/core": "^7.23.9",
"@babel/eslint-parser": "^7.23.3",
"@commitlint/cli": "^17.7.1", "@commitlint/cli": "^17.7.1",
"@commitlint/config-conventional": "^17.7.0", "@commitlint/config-conventional": "^17.7.0",
"@interactjs/types": "1.10.21", "@interactjs/types": "1.10.21",
@ -71,7 +68,6 @@
"@vitejs/plugin-vue": "^5.0.4", "@vitejs/plugin-vue": "^5.0.4",
"@vitejs/plugin-vue-jsx": "^3.1.0", "@vitejs/plugin-vue-jsx": "^3.1.0",
"@vitest/ui": "1.4.0", "@vitest/ui": "1.4.0",
"@vue-hooks-plus/resolvers": "1.2.4",
"@vue/eslint-config-prettier": "^9.0.0", "@vue/eslint-config-prettier": "^9.0.0",
"@vue/eslint-config-typescript": "^12.0.0", "@vue/eslint-config-typescript": "^12.0.0",
"@vue/test-utils": "2.4.3", "@vue/test-utils": "2.4.3",
@ -80,8 +76,6 @@
"eslint": "^8.56.0", "eslint": "^8.56.0",
"eslint-config-prettier": "^9.1.0", "eslint-config-prettier": "^9.1.0",
"eslint-config-standard-with-typescript": "^43.0.0", "eslint-config-standard-with-typescript": "^43.0.0",
"eslint-plugin-import": "^2.29.0",
"eslint-plugin-n": "^16.6.2",
"eslint-plugin-prettier": "^5.1.3", "eslint-plugin-prettier": "^5.1.3",
"eslint-plugin-promise": "^6.1.1", "eslint-plugin-promise": "^6.1.1",
"eslint-plugin-vue": "^9.18.1", "eslint-plugin-vue": "^9.18.1",

80
pnpm-lock.yaml generated
View File

@ -17,9 +17,6 @@ dependencies:
clipboard: clipboard:
specifier: ^2.0.11 specifier: ^2.0.11
version: 2.0.11 version: 2.0.11
crypto-js:
specifier: ^4.1.1
version: 4.2.0
currency.js: currency.js:
specifier: ^2.0.4 specifier: ^2.0.4
version: 2.0.4 version: 2.0.4
@ -66,16 +63,10 @@ dependencies:
specifier: ^9.9.0 specifier: ^9.9.0
version: 9.9.0(vue@3.4.21) version: 9.9.0(vue@3.4.21)
vue-router: vue-router:
specifier: ^4.2.5 specifier: ^4.3.0
version: 4.3.0(vue@3.4.21) version: 4.3.0(vue@3.4.21)
devDependencies: devDependencies:
'@babel/core':
specifier: ^7.23.9
version: 7.24.1
'@babel/eslint-parser':
specifier: ^7.23.3
version: 7.23.10(@babel/core@7.24.1)(eslint@8.57.0)
'@commitlint/cli': '@commitlint/cli':
specifier: ^17.7.1 specifier: ^17.7.1
version: 17.8.1 version: 17.8.1
@ -115,9 +106,6 @@ devDependencies:
'@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)
'@vue-hooks-plus/resolvers':
specifier: 1.2.4
version: 1.2.4(vue-hooks-plus@1.8.8)
'@vue/eslint-config-prettier': '@vue/eslint-config-prettier':
specifier: ^9.0.0 specifier: ^9.0.0
version: 9.0.0(eslint@8.57.0)(prettier@3.2.5) version: 9.0.0(eslint@8.57.0)(prettier@3.2.5)
@ -142,12 +130,6 @@ devDependencies:
eslint-config-standard-with-typescript: eslint-config-standard-with-typescript:
specifier: ^43.0.0 specifier: ^43.0.0
version: 43.0.0(@typescript-eslint/eslint-plugin@6.21.0)(eslint-plugin-import@2.29.1)(eslint-plugin-n@16.6.2)(eslint-plugin-promise@6.1.1)(eslint@8.57.0)(typescript@5.2.2) version: 43.0.0(@typescript-eslint/eslint-plugin@6.21.0)(eslint-plugin-import@2.29.1)(eslint-plugin-n@16.6.2)(eslint-plugin-promise@6.1.1)(eslint@8.57.0)(typescript@5.2.2)
eslint-plugin-import:
specifier: ^2.29.0
version: 2.29.1(@typescript-eslint/parser@6.21.0)(eslint@8.57.0)
eslint-plugin-n:
specifier: ^16.6.2
version: 16.6.2(eslint@8.57.0)
eslint-plugin-prettier: eslint-plugin-prettier:
specifier: ^5.1.3 specifier: ^5.1.3
version: 5.1.3(eslint-config-prettier@9.1.0)(eslint@8.57.0)(prettier@3.2.5) version: 5.1.3(eslint-config-prettier@9.1.0)(eslint@8.57.0)(prettier@3.2.5)
@ -288,20 +270,6 @@ packages:
- supports-color - supports-color
dev: true dev: true
/@babel/eslint-parser@7.23.10(@babel/core@7.24.1)(eslint@8.57.0):
resolution: {integrity: sha512-3wSYDPZVnhseRnxRJH6ZVTNknBz76AEnyC+AYYhasjP3Yy23qz0ERR7Fcd2SHmYuSFJ2kY9gaaDd3vyqU09eSw==}
engines: {node: ^10.13.0 || ^12.13.0 || >=14.0.0}
peerDependencies:
'@babel/core': ^7.11.0
eslint: ^7.5.0 || ^8.0.0
dependencies:
'@babel/core': 7.24.1
'@nicolo-ribaudo/eslint-scope-5-internals': 5.1.1-v1
eslint: 8.57.0
eslint-visitor-keys: 2.1.0
semver: 6.3.1
dev: true
/@babel/generator@7.24.1: /@babel/generator@7.24.1:
resolution: {integrity: sha512-DfCRfZsBcrPEHUfuBMgbJ1Ut01Y/itOs+hY2nFLgqsqXd52/iSiVq5TITtUasIUgm+IIKdY2/1I7auiQOEeC9A==} resolution: {integrity: sha512-DfCRfZsBcrPEHUfuBMgbJ1Ut01Y/itOs+hY2nFLgqsqXd52/iSiVq5TITtUasIUgm+IIKdY2/1I7auiQOEeC9A==}
engines: {node: '>=6.9.0'} engines: {node: '>=6.9.0'}
@ -1391,12 +1359,6 @@ packages:
- supports-color - supports-color
dev: false dev: false
/@nicolo-ribaudo/eslint-scope-5-internals@5.1.1-v1:
resolution: {integrity: sha512-54/JRvkLIzzDWshCWfuhadfrfZVPiElY8Fcgmg1HroEly/EDSszzhBAsarCux+D/kOslTRquNzuyGSmUSTTHGg==}
dependencies:
eslint-scope: 5.1.1
dev: true
/@nodelib/fs.scandir@2.1.5: /@nodelib/fs.scandir@2.1.5:
resolution: {integrity: sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==} resolution: {integrity: sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==}
engines: {node: '>= 8'} engines: {node: '>= 8'}
@ -1616,6 +1578,7 @@ packages:
/@types/js-cookie@3.0.6: /@types/js-cookie@3.0.6:
resolution: {integrity: sha512-wkw9yd1kEXOPnvEeEV1Go1MmxtBJL0RR79aOTAApecWFVu7w0NNXNqhcWgvw2YgZDYadliXkl14pa3WXw5jlCQ==} resolution: {integrity: sha512-wkw9yd1kEXOPnvEeEV1Go1MmxtBJL0RR79aOTAApecWFVu7w0NNXNqhcWgvw2YgZDYadliXkl14pa3WXw5jlCQ==}
dev: false
/@types/json-schema@7.0.15: /@types/json-schema@7.0.15:
resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==} resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==}
@ -1916,16 +1879,6 @@ packages:
path-browserify: 1.0.1 path-browserify: 1.0.1
dev: true dev: true
/@vue-hooks-plus/resolvers@1.2.4(vue-hooks-plus@1.8.8):
resolution: {integrity: sha512-RNvBDq2YDEVT3uDjPsWRpFZ4YJR/qfMGXNoMIDvBFWy/s2tvRWBaYmJILWXIqgY3gYeKoxy6TxGsh1mjsEhiRw==}
engines: {node: '>=14'}
peerDependencies:
vue-hooks-plus: ^1.5.2
dependencies:
local-pkg: 0.4.3
vue-hooks-plus: 1.8.8(vue@3.4.21)
dev: true
/@vue/babel-helper-vue-transform-on@1.2.2: /@vue/babel-helper-vue-transform-on@1.2.2:
resolution: {integrity: sha512-nOttamHUR3YzdEqdM/XXDyCSdxMA9VizUKoroLX6yTyRtggzQMHXcmwh8a7ZErcJttIBIc9s68a1B8GZ+Dmvsw==} resolution: {integrity: sha512-nOttamHUR3YzdEqdM/XXDyCSdxMA9VizUKoroLX6yTyRtggzQMHXcmwh8a7ZErcJttIBIc9s68a1B8GZ+Dmvsw==}
dev: true dev: true
@ -3086,10 +3039,6 @@ packages:
which: 2.0.2 which: 2.0.2
dev: true dev: true
/crypto-js@4.2.0:
resolution: {integrity: sha512-KALDyEYgpY+Rlob/iriUtjV6d5Eq+Y191A5g4UqLAi8CyGP9N1+FdVbkc1SxKc2r4YAYqG8JzO2KGL+AizD70Q==}
dev: false
/css-render@0.15.12: /css-render@0.15.12:
resolution: {integrity: sha512-eWzS66patiGkTTik+ipO9qNGZ+uNuGyTmnz6/+EJIiFg8+3yZRpnMwgFo8YdXhQRsiePzehnusrxVvugNjXzbw==} resolution: {integrity: sha512-eWzS66patiGkTTik+ipO9qNGZ+uNuGyTmnz6/+EJIiFg8+3yZRpnMwgFo8YdXhQRsiePzehnusrxVvugNjXzbw==}
dependencies: dependencies:
@ -4047,14 +3996,6 @@ packages:
- supports-color - supports-color
dev: true dev: true
/eslint-scope@5.1.1:
resolution: {integrity: sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==}
engines: {node: '>=8.0.0'}
dependencies:
esrecurse: 4.3.0
estraverse: 4.3.0
dev: true
/eslint-scope@7.2.2: /eslint-scope@7.2.2:
resolution: {integrity: sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==} resolution: {integrity: sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==}
engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
@ -4063,11 +4004,6 @@ packages:
estraverse: 5.3.0 estraverse: 5.3.0
dev: true dev: true
/eslint-visitor-keys@2.1.0:
resolution: {integrity: sha512-0rSmRBzXgDzIsD6mGdJgevzgezI534Cer5L/vyMX0kHzT/jiB43jRhd9YUlMGYLQy2zprNmoT8qasCGtY+QaKw==}
engines: {node: '>=10'}
dev: true
/eslint-visitor-keys@3.4.3: /eslint-visitor-keys@3.4.3:
resolution: {integrity: sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==} resolution: {integrity: sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==}
engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
@ -4149,11 +4085,6 @@ packages:
estraverse: 5.3.0 estraverse: 5.3.0
dev: true dev: true
/estraverse@4.3.0:
resolution: {integrity: sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==}
engines: {node: '>=4.0'}
dev: true
/estraverse@5.3.0: /estraverse@5.3.0:
resolution: {integrity: sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==} resolution: {integrity: sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==}
engines: {node: '>=4.0'} engines: {node: '>=4.0'}
@ -4354,6 +4285,7 @@ packages:
/filter-obj@1.1.0: /filter-obj@1.1.0:
resolution: {integrity: sha512-8rXg1ZnX7xzy2NGDVkBVaAy+lSlPNwad13BtgSlLuxfIslyt5Vg64U7tFcCt4WS1R0hvtnQybT/IyCkGZ3DpXQ==} resolution: {integrity: sha512-8rXg1ZnX7xzy2NGDVkBVaAy+lSlPNwad13BtgSlLuxfIslyt5Vg64U7tFcCt4WS1R0hvtnQybT/IyCkGZ3DpXQ==}
engines: {node: '>=0.10.0'} engines: {node: '>=0.10.0'}
dev: false
/find-up@4.1.0: /find-up@4.1.0:
resolution: {integrity: sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==} resolution: {integrity: sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==}
@ -5331,6 +5263,7 @@ packages:
/js-cookie@3.0.5: /js-cookie@3.0.5:
resolution: {integrity: sha512-cEiJEAEoIbWfCZYKWhVwFuvPX1gETRYPw6LlaTKoxD3s2AkXzkCjnp6h0V77ozyqj0jakteJ4YqDJT830+lVGw==} resolution: {integrity: sha512-cEiJEAEoIbWfCZYKWhVwFuvPX1gETRYPw6LlaTKoxD3s2AkXzkCjnp6h0V77ozyqj0jakteJ4YqDJT830+lVGw==}
engines: {node: '>=14'} engines: {node: '>=14'}
dev: false
/js-tokens@4.0.0: /js-tokens@4.0.0:
resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==}
@ -6543,6 +6476,7 @@ packages:
filter-obj: 1.1.0 filter-obj: 1.1.0
split-on-first: 1.1.0 split-on-first: 1.1.0
strict-uri-encode: 2.0.0 strict-uri-encode: 2.0.0
dev: false
/queue-microtask@1.2.3: /queue-microtask@1.2.3:
resolution: {integrity: sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==} resolution: {integrity: sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==}
@ -6836,6 +6770,7 @@ packages:
/screenfull@5.2.0: /screenfull@5.2.0:
resolution: {integrity: sha512-9BakfsO2aUQN2K9Fdbj87RJIEZ82Q9IGim7FqM5OsebfoFC6ZHXgDq/KvniuLTPdeM8wY2o6Dj3WQ7KeQCj3cA==} resolution: {integrity: sha512-9BakfsO2aUQN2K9Fdbj87RJIEZ82Q9IGim7FqM5OsebfoFC6ZHXgDq/KvniuLTPdeM8wY2o6Dj3WQ7KeQCj3cA==}
engines: {node: '>=0.10.0'} engines: {node: '>=0.10.0'}
dev: false
/scule@1.3.0: /scule@1.3.0:
resolution: {integrity: sha512-6FtHJEvt+pVMIB9IBY+IcCJ6Z5f1iQnytgyfKMhDKgmzYG+TeH/wx1y3l27rshSbLiSanrR9ffZDrEsmjlQF2g==} resolution: {integrity: sha512-6FtHJEvt+pVMIB9IBY+IcCJ6Z5f1iQnytgyfKMhDKgmzYG+TeH/wx1y3l27rshSbLiSanrR9ffZDrEsmjlQF2g==}
@ -7083,6 +7018,7 @@ packages:
/split-on-first@1.1.0: /split-on-first@1.1.0:
resolution: {integrity: sha512-43ZssAJaMusuKWL8sKUBQXHWOpq8d6CfN/u1p4gUzfJkM05C8rxTmYrkIPTXapZpORA6LkkzcUulJ8FqA7Uudw==} resolution: {integrity: sha512-43ZssAJaMusuKWL8sKUBQXHWOpq8d6CfN/u1p4gUzfJkM05C8rxTmYrkIPTXapZpORA6LkkzcUulJ8FqA7Uudw==}
engines: {node: '>=6'} engines: {node: '>=6'}
dev: false
/split-string@3.1.0: /split-string@3.1.0:
resolution: {integrity: sha512-NzNVhJDYpwceVVii8/Hu6DKfD2G+NrQHlS/V/qgv763EYudVwEcMQNxd2lh+0VrUByXN/oJkl5grOhYWvQUYiw==} resolution: {integrity: sha512-NzNVhJDYpwceVVii8/Hu6DKfD2G+NrQHlS/V/qgv763EYudVwEcMQNxd2lh+0VrUByXN/oJkl5grOhYWvQUYiw==}
@ -7135,6 +7071,7 @@ packages:
/strict-uri-encode@2.0.0: /strict-uri-encode@2.0.0:
resolution: {integrity: sha512-QwiXZgpRcKkhTj2Scnn++4PKtWsH0kpzZ62L2R6c/LUVYv7hVnZqcg2+sMuT6R7Jusu1vviK/MFsu6kNJfWlEQ==} resolution: {integrity: sha512-QwiXZgpRcKkhTj2Scnn++4PKtWsH0kpzZ62L2R6c/LUVYv7hVnZqcg2+sMuT6R7Jusu1vviK/MFsu6kNJfWlEQ==}
engines: {node: '>=4'} engines: {node: '>=4'}
dev: false
/string-argv@0.3.2: /string-argv@0.3.2:
resolution: {integrity: sha512-aqD2Q0144Z+/RqG52NeHEkZauTAUWJO8c6yTftGJKO3Tja5tUgIfmIl6kExvhtxSDP7fXB6DvzkfMpCd/F3G+Q==} resolution: {integrity: sha512-aqD2Q0144Z+/RqG52NeHEkZauTAUWJO8c6yTftGJKO3Tja5tUgIfmIl6kExvhtxSDP7fXB6DvzkfMpCd/F3G+Q==}
@ -8230,6 +8167,7 @@ packages:
query-string: 7.1.3 query-string: 7.1.3
screenfull: 5.2.0 screenfull: 5.2.0
vue: 3.4.21(typescript@5.2.2) vue: 3.4.21(typescript@5.2.2)
dev: false
/vue-i18n@9.9.0(vue@3.4.21): /vue-i18n@9.9.0(vue@3.4.21):
resolution: {integrity: sha512-xQ5SxszUAqK5n84N+uUyHH/PiQl9xZ24FOxyAaNonmOQgXeN+rD9z/6DStOpOxNFQn4Cgcquot05gZc+CdOujA==} resolution: {integrity: sha512-xQ5SxszUAqK5n84N+uUyHH/PiQl9xZ24FOxyAaNonmOQgXeN+rD9z/6DStOpOxNFQn4Cgcquot05gZc+CdOujA==}

View File

@ -1,2 +1,16 @@
export * from './valid/validAppRootPath' import { validAppRootPath } from './valid/valid-app-root-path'
export * from './valid/validLocal' import { validLocal } from './valid/valid-local'
/**
*
* @description
* ray-template
*/
export const setupRayTemplateCore = async () => {
if (!__DEV__) {
return
}
await validAppRootPath()
await validLocal()
}

View File

@ -11,10 +11,6 @@ import { useVueRouter } from '@/hooks'
* getRoutes * getRoutes
*/ */
export const validAppRootPath = async () => { export const validAppRootPath = async () => {
if (!__DEV__) {
return
}
const { getAppRootRoute } = useSettingGetters() const { getAppRootRoute } = useSettingGetters()
const { const {
router: { getRoutes }, router: { getRoutes },

View File

@ -92,10 +92,6 @@ const validDefaultDayjsLocal = () => {
* localConfig * localConfig
*/ */
export const validLocal = async () => { export const validLocal = async () => {
if (!__DEV__) {
return
}
validSystemDefaultLocal() validSystemDefaultLocal()
validSystemFallbackLocale() validSystemFallbackLocale()
validDayjsLocalMap() validDayjsLocalMap()

View File

@ -1,6 +1,6 @@
export * from './appConfig' export * from './app-config'
export * from './designConfig' export * from './design-config'
export * from './localConfig' export * from './local-config'
export * from './regexConfig' export * from './regex-config'
export * from './requestConfig' export * from './request-config'
export * from './routerConfig' export * from './router-config'

View File

@ -21,7 +21,7 @@
*/ */
import { axiosCanceler } from '@/axios/utils/interceptor' import { axiosCanceler } from '@/axios/utils/interceptor'
import { appendRequestHeaders } from '@/axios/utils/axiosCopilot' import { appendRequestHeaders } from '@/axios/utils/axios-copilot'
import { APP_CATCH_KEY } from '@/app-config' import { APP_CATCH_KEY } from '@/app-config'
import { getStorage } from '@/utils' import { getStorage } from '@/utils'

View File

@ -461,12 +461,7 @@ export default defineComponent({
} }
}) })
expose({ expose()
echart: echartInstanceRef,
dispose: unmount,
render: mount,
isDispose,
})
onBeforeMount(async () => { onBeforeMount(async () => {
// 注册 echarts 组件与渲染器 // 注册 echarts 组件与渲染器

View File

@ -381,6 +381,6 @@ const props = {
>, >,
default: null, default: null,
}, },
} } as const
export default props export default props

View File

@ -19,7 +19,7 @@ import type { RFormInst } from './types'
export default defineComponent({ export default defineComponent({
name: 'RForm', name: 'RForm',
props, props,
setup(props) { setup(props, { expose }) {
const formRef = ref<RFormInst>() const formRef = ref<RFormInst>()
onMounted(() => { onMounted(() => {
@ -31,6 +31,8 @@ export default defineComponent({
} }
}) })
expose()
return { return {
formRef, formRef,
} }

View File

@ -19,6 +19,6 @@ const props = {
>, >,
default: null, default: null,
}, },
} } as const
export default props export default props

View File

@ -21,6 +21,6 @@ const props = {
type: String, type: String,
default: 'default', default: 'default',
}, },
} } as const
export default props export default props

View File

@ -305,6 +305,6 @@ const props = {
type: [Function, Array] as PropType<MaybeArray<() => void>>, type: [Function, Array] as PropType<MaybeArray<() => void>>,
default: null, default: null,
}, },
} } as const
export default props export default props

View File

@ -0,0 +1,10 @@
import RSegment from './src/Segment'
import segmentProps from './src/props'
import type { ExtractPublicPropTypes } from 'vue'
import type { RSegmentOptions } from './src/types'
export type SegmentProps = ExtractPublicPropTypes<typeof segmentProps>
export type { RSegmentOptions }
export { RSegment, segmentProps }

View File

@ -0,0 +1,133 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2024-04-10
*
* @workspace ray-template
*
* @remark
*/
import './index.scss'
import { NTabs, NTab, NPopover, NFlex } from 'naive-ui'
import props from './props'
import { themeOverrides } from './constant'
import { completeSize, isValueType } from '@/utils'
import type { TabsProps } from 'naive-ui'
import type { RSegmentOptions } from './types'
const iconSegmentTab = (option: RSegmentOptions) => {
const { icon, label } = option
if (icon) {
return (
<NFlex align="center" wrap={false} size="small">
<icon />
<div>{label}</div>
</NFlex>
)
}
return label
}
const popoverSegmentTab = (option: RSegmentOptions) => {
if (typeof option.popover === 'string') {
return (
<NPopover>
{{
trigger: iconSegmentTab(option),
default: () => option.popover,
}}
</NPopover>
)
}
if (isValueType<object>(option.popover, 'Object')) {
const { popover } = option
const { label, ...parameters } = popover
return (
<NPopover {...parameters}>
{{
trigger: iconSegmentTab(option),
default: () => label,
}}
</NPopover>
)
}
}
export default defineComponent({
name: 'RSegment',
props,
setup(props, { expose }) {
const cssVars = computed(() => {
const { width: propsWidth } = props
let segmentWidthVar = ''
switch (propsWidth) {
case 'block':
segmentWidthVar = '100%'
break
case 'fitContent':
segmentWidthVar = 'fit-content'
break
default:
segmentWidthVar =
typeof propsWidth === 'number'
? completeSize(propsWidth)
: 'fit-content'
}
return {
'--r-segment-width': segmentWidthVar,
}
})
expose()
return {
cssVars,
}
},
render() {
const { $props, options, cssVars, themeOverrides: _themeOverrides } = this
return (
<NTabs
{...($props as TabsProps)}
ref="segmentRef"
style={[cssVars]}
class="r-segment"
type="segment"
animated
themeOverrides={Object.assign({}, themeOverrides, _themeOverrides)}
>
{options.map((curr) => {
return (
<NTab
key={curr.key}
name={curr.key}
tab={
!curr.popover ? iconSegmentTab(curr) : popoverSegmentTab(curr)
}
disabled={curr.disabled}
>
{{
...curr.slots,
}}
</NTab>
)
})}
</NTabs>
)
},
})

View File

@ -0,0 +1,26 @@
export const OMIT_TABS_PROPS_KEYS = [
'addTabClass',
'addTabStyle',
'paneClass',
'paneStyle',
'paneWrapperStyle',
'tabClass',
'tabStyle',
'type',
'label',
'addable',
'closable',
'onAdd',
'onClose',
'placement',
'animated',
'justifyContent',
'builtinThemeOverrides',
'displayDirective',
] as const
export const themeOverrides = {
tabPaddingSmallSegment: '4px 7px',
tabPaddingMediumSegment: '6px 11px',
tabPaddingLargeSegment: '8px 11px',
}

View File

@ -0,0 +1,3 @@
.r-segment {
width: var(--r-segment-width);
}

View File

@ -0,0 +1,29 @@
import { tabsProps } from 'naive-ui'
import { omit } from 'lodash-es'
import { OMIT_TABS_PROPS_KEYS } from './constant'
import type { RSegmentOptions, RSegmentWidth } from './types'
import type { PropType } from 'vue'
/**
*
* @description
* vue props omit
*
*/
const segmentProps = omit(
{
...tabsProps,
options: {
type: Array as PropType<RSegmentOptions[]>,
default: () => [],
},
width: {
type: [Number, String] as PropType<RSegmentWidth>,
default: 'fitContent',
},
},
...OMIT_TABS_PROPS_KEYS,
)
export default segmentProps

View File

@ -0,0 +1,28 @@
import type { TabsProps, TabPaneProps, PopoverProps } from 'naive-ui'
import type { OMIT_TABS_PROPS_KEYS } from './constant'
import type { VNode, VNodeChild, ExtractPublicPropTypes } from 'vue'
export type OmitTabsPropsKeys = (typeof OMIT_TABS_PROPS_KEYS)[number]
export interface RSegmentPopover extends ExtractPublicPropTypes<PopoverProps> {
label: string
}
export interface RSegmentOptions {
label: string | VNode | (() => VNodeChild)
key: string | number
displayDirective?: TabPaneProps['displayDirective']
disabled?: boolean
slots?: {
default?: () => VNode | string | number
}
popover?: string | RSegmentPopover
icon?: VNode
}
export type RSegmentWidth = number | 'block' | 'fitContent'
export interface RSegmentProps extends Omit<TabsProps, OmitTabsPropsKeys> {
options?: RSegmentOptions
width?: RSegmentWidth
}

View File

@ -1,12 +1,14 @@
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 useTable from './src/hooks/useTable'
import useCheckedRowKeys from './src/hooks/useCheckedRowKeys'
import type * as RTableType from './src/types' import type * as RTableType from './src/types'
import type { UseTableReturn } from './src/hooks/useTable' import type { UseTableReturn } from './src/hooks/useTable'
import type { UseCheckedRowKeysReturn } from './src/hooks/useCheckedRowKeys'
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, UseTableReturn } export type { RTableType, UseTableReturn, UseCheckedRowKeysReturn }
export { RTable, tableProps, useTable } export { RTable, tableProps, useTable, useCheckedRowKeys }

View File

@ -211,11 +211,7 @@ export default defineComponent({
uuidWrapper, uuidWrapper,
wrapperRef, wrapperRef,
}) })
expose({ expose()
rTableInst,
uuidTable,
uuidWrapper,
})
return { return {
uuidWrapper, uuidWrapper,

View File

@ -24,6 +24,7 @@ import { RIcon } from '@/components'
import { config } from '../shared' import { config } from '../shared'
import props from '../props' import props from '../props'
import { call } from '@/utils' import { call } from '@/utils'
import { cloneDeep, isEmpty } from 'lodash-es'
import type { TreeOption, TreeDropInfo } from 'naive-ui' import type { TreeOption, TreeDropInfo } from 'naive-ui'
import type { C } from '../types' import type { C } from '../types'
@ -109,11 +110,22 @@ export default defineComponent({
}, },
}, },
setup(props) { setup(props) {
let tableColumnType: C
// 深拷贝 columns 避免修改源数据 // 深拷贝 columns 避免修改源数据
const treeDataSource = computed({ const treeDataSource = computed({
get: () => { get: () => {
const cloneColumns = cloneDeep(props.columns).filter((curr) => {
if (curr.type) {
tableColumnType = curr as unknown as C
return false
}
return true
})
// eslint-disable-next-line @typescript-eslint/no-explicit-any // eslint-disable-next-line @typescript-eslint/no-explicit-any
return props.columns.map((curr, idx) => { return cloneColumns.map((curr, idx) => {
const { key, title, children, fixed, isResizable, ...args } = const { key, title, children, fixed, isResizable, ...args } =
curr as C curr as C
const isLeftFixedActivated = fixed === 'left' const isLeftFixedActivated = fixed === 'left'
@ -229,6 +241,10 @@ export default defineComponent({
? nodeSiblings.splice(nodeIndex, 0, dragNode) ? nodeSiblings.splice(nodeIndex, 0, dragNode)
: nodeSiblings.splice(nodeIndex + 1, 0, dragNode) : nodeSiblings.splice(nodeIndex + 1, 0, dragNode)
if (!isEmpty(tableColumnType)) {
nodeSiblings.unshift(tableColumnType as TreeOption)
}
// 触发事件,更新树形数据源 // 触发事件,更新树形数据源
event(nodeSiblings as C[]) event(nodeSiblings as C[])
} }

View File

@ -13,7 +13,7 @@ import { NPopselect } from 'naive-ui'
import { RIcon } from '@/components' import { RIcon } from '@/components'
import { call } from '@/utils' import { call } from '@/utils'
import { config } from '../shared' import { config, propsOptions } from '../shared'
import props from '../props' import props from '../props'
import type { MaybeArray } from '@/types' import type { MaybeArray } from '@/types'
@ -39,16 +39,6 @@ export default defineComponent({
setup(props) { setup(props) {
const popoverShow = ref(false) const popoverShow = ref(false)
const propsPopselectValue = ref<PropsComponentPopselectKeys[]>([]) const propsPopselectValue = ref<PropsComponentPopselectKeys[]>([])
const propsOptions = [
{
label: '斑马条纹',
value: 'striped',
},
{
label: '表格边框',
value: 'bordered',
},
]
const updatePopselectValue = (value: PropsComponentPopselectKeys[]) => { const updatePopselectValue = (value: PropsComponentPopselectKeys[]) => {
const { onPopselectChange } = props const { onPopselectChange } = props
@ -78,7 +68,6 @@ export default defineComponent({
return { return {
propsPopselectValue, propsPopselectValue,
propsOptions,
popoverShow, popoverShow,
updatePopselectValue, updatePopselectValue,
} }
@ -87,7 +76,7 @@ export default defineComponent({
return ( return (
<NPopselect <NPopselect
v-model:value={this.propsPopselectValue} v-model:value={this.propsPopselectValue}
options={this.propsOptions} options={propsOptions}
trigger="click" trigger="click"
multiple multiple
onUpdateValue={this.updatePopselectValue.bind(this)} onUpdateValue={this.updatePopselectValue.bind(this)}

View File

@ -0,0 +1,257 @@
/* eslint-disable @typescript-eslint/no-explicit-any */
import { effectDispose } from '@/utils'
import type { Recordable } from '@/types'
import type { MaybeRef } from '@vueuse/core'
export type RowKey = string | number
export type Action =
| 'check'
| 'uncheck'
| 'checkAll'
| 'uncheckAll'
| 'multipleCheck'
export interface UseCheckedRowKeysOptions<T = unknown> {
/**
*
* @description
* DataTable
* 使 key
*/
rowKey: string
/**
*
* @param keys keys
* @param rows rows
* @param meta
*
* @description
* keys
*/
onChange?: (
keys: RowKey[],
rows: T[],
meta: { row: T | undefined; action: Action },
) => void
}
/**
*
* @param tableData DataTable data
* @param bindRowKey DataTable rowKey
* @param key target key
*
* @description
* key
* DataTable data
*/
const findRow = (
tableData: Recordable[],
bindRowKey: string,
key: RowKey,
): Recordable | undefined => {
if (!tableData.length) {
return void 0
}
for (const curr of tableData) {
if (curr[bindRowKey] === key) {
return curr
}
if (curr?.children?.length) {
const foundRow = findRow(curr.children, bindRowKey, key)
if (foundRow) {
return foundRow
}
}
}
return void 0
}
/**
*
* @param data DataTable
* @param options
*
* @description
* 便 checkedRowKeys
*
* rowKey
*
* onUpdateCheckedRowKeys checkedRowKeysBind
* 使 v-model:checkedRowKeys
*
* @example
* <template>
* <RDataTable
* v-model:checkedRowKeys="checkedRowKeys"
* onUpdateCheckedRowKeys="checkedRowKeysBind"
* :data="data"
* :columns="columns"
* />
* </template>
* <script lang="ts" setup>
* import { useCheckedRowKeys } from '@/components'
*
* const data = ref([{ ...table data }])
* const columns = [{ ...table columns }]
* const [checkedRowKeys, { checkedRowKeysBind }] = useCheckedRowKeys(data, { rowKey: 'key' })
* </script>
*/
const useCheckedRowKeys = <T extends Recordable>(
data: MaybeRef<T[] | undefined>,
options?: UseCheckedRowKeysOptions<T>,
) => {
const keysRef = ref<RowKey[]>([])
const rowsRef = ref<any[]>([])
const { rowKey: bindRowKey = 'key', onChange } = options || {}
/**
*
* @param keys keys
* @param rows rows
* @param meta
*
* @description
* onUpdateCheckedRowKeys
*/
const bind = (
keys: RowKey[],
rows: Recordable[],
meta: {
row: Recordable | undefined
action: Action
},
) => {
keysRef.value = keys
rowsRef.value = rows
onChange?.(
keys,
rows as any,
meta as {
row: any | undefined
action: Action
},
)
}
/**
*
* @description
* key
* key DataTable rowKey
*/
const getKeys = () => keysRef.value
/**
*
* @description
*
*/
const getRows = () => rowsRef.value
/**
*
* @description
* key rows
*/
const clearAll = () => {
keysRef.value = []
rowsRef.value = []
onChange?.(keysRef.value, rowsRef.value, {
row: void 0,
action: 'uncheckAll',
})
}
/**
*
* @param key key
* @param rowKey key 'key' DataTable rowKey
*
* @description
* key
*
* naive ui key rows
* key rows
*/
const clearKey = (key: RowKey) => {
if (key === null || key === void 0) {
return
}
let _row!: Recordable<T>
keysRef.value = keysRef.value.filter((curr) => curr !== key)
rowsRef.value = rowsRef.value.filter((curr) => {
if (curr[bindRowKey] === key) {
_row = curr
return false
}
return true
})
onChange?.(keysRef.value, rowsRef.value, {
row: _row as any,
action: 'uncheck',
})
}
/**
*
* @param keys keys
*
* @description
* keys
*
* keys data rows rows rows
*/
const selectKey = (key: RowKey) => {
if (keysRef.value.includes(key)) {
return
}
keysRef.value.push(key)
const row = findRow(unref(data) || [], bindRowKey, key)
if (row) {
rowsRef.value.push(row)
onChange?.(keysRef.value, rowsRef.value, {
row: row as any,
action: 'check',
})
}
}
effectDispose(() => {
clearAll()
})
return [
keysRef,
{
checkedRows: rowsRef as Ref<T[]>,
checkedRowKeysBind: bind,
getKeys,
getRows,
clearAll,
clearKey,
selectKey,
},
] as const
}
export type UseCheckedRowKeysReturn = ReturnType<typeof useCheckedRowKeys>
export default useCheckedRowKeys

View File

@ -206,6 +206,6 @@ const props = {
>, >,
default: null, default: null,
}, },
} } as const
export default props export default props

View File

@ -13,3 +13,14 @@ export const config = {
tableIconSize: '18', tableIconSize: '18',
tableKey: Symbol('r-table'), tableKey: Symbol('r-table'),
} }
export const propsOptions = [
{
label: '斑马条纹',
value: 'striped',
},
{
label: '表格边框',
value: 'bordered',
},
]

View File

@ -4,6 +4,6 @@ const props: TransitionProps = {
transitionPropName: 'fade', transitionPropName: 'fade',
transitionMode: 'out-in', transitionMode: 'out-in',
transitionAppear: true, transitionAppear: true,
} } as const
export default props export default props

View File

@ -9,6 +9,7 @@ export * from './RQRCode'
export * from './RTable' export * from './RTable'
export * from './RTransitionComponent' export * from './RTransitionComponent'
export * from './RForm' export * from './RForm'
export * from './RSegment'
// 导出自定义组件类型 // 导出自定义组件类型
export type * from './RChart/src/types' export type * from './RChart/src/types'
@ -19,3 +20,4 @@ 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 './RForm/src/types'
export type * from './RModal/src/types' export type * from './RModal/src/types'
export type * from './RSegment/src/types'

View File

@ -13,6 +13,13 @@ import dayjs from 'dayjs'
import { DEFAULT_DAYJS_LOCAL } from '@/app-config' import { DEFAULT_DAYJS_LOCAL } from '@/app-config'
import 'dayjs/locale/zh-cn' import 'dayjs/locale/zh-cn'
/**
*
* @description
* vue dayjs
*
* dayjs
*/
export const setupDayjs = () => { export const setupDayjs = () => {
dayjs.locale(DEFAULT_DAYJS_LOCAL) dayjs.locale(DEFAULT_DAYJS_LOCAL)
} }

View File

@ -13,11 +13,11 @@
* *
* directive name: copy * directive name: copy
* *
* @description
* 使 value * 使 value
* *
* clipboard.js * clipboard.js
* *
* 使
* @example * @example
* <template> * <template>
* <button v-copy="copyText"></button> * <button v-copy="copyText"></button>

View File

@ -13,11 +13,12 @@
* *
* directive name: debounce * directive name: debounce
* *
* @description
*
* 使 func * 使 func
* *
* trigger wait trigger clickwait 500 * trigger wait trigger clickwait 500
* *
* 使
* @example * @example
* <template> * <template>
* <div v-debounce="{ func: () => console.log('debounce') }"></div> * <div v-debounce="{ func: () => console.log('debounce') }"></div>

View File

@ -13,12 +13,12 @@
* *
* directive name: disabled * directive name: disabled
* *
* @description
* 使 value * 使 value
* *
* ray-template__directive--disabled * ray-template__directive--disabled
* css * css
* *
* 使
* @example * @example
* <template> * <template>
* <button v-disabled="true"></button> * <button v-disabled="true"></button>

View File

@ -2,11 +2,11 @@
* *
* directive name: ellipsis * directive name: ellipsis
* *
* @description
* 使 width * 使 width
* *
* line type line 1type block * line type line 1type block
* *
* 使
* @example * @example
* <template> * <template>
* <div v-ellipsis="{ line: 2, width: 200 }"></div> * <div v-ellipsis="{ line: 2, width: 200 }"></div>

View File

@ -0,0 +1,47 @@
$ripple-animation-transition-in:
transform 0.4s cubic-bezier(0, 0, 0.2, 1),
opacity 0.2s cubic-bezier(0, 0, 0.2, 1) !default;
$ripple-animation-transition-out: opacity 0.5s cubic-bezier(0, 0, 0.2, 1) !default;
$ripple-animation-visible-opacity: 0.25 !default;
.v-ripple {
&__container {
position: absolute;
top: 0;
left: 0;
z-index: 0;
width: 100%;
height: 100%;
overflow: hidden;
pointer-events: none;
border-radius: inherit;
contain: strict;
}
&__animation {
position: absolute;
top: 0;
left: 0;
overflow: hidden;
pointer-events: none;
background: currentcolor;
border-radius: 50%;
opacity: 0;
will-change: transform, opacity;
&--enter {
opacity: 0;
transition: none;
}
&--in {
opacity: $ripple-animation-visible-opacity;
transition: $ripple-animation-transition-in;
}
&--out {
opacity: 0;
transition: $ripple-animation-transition-out;
}
}
}

View File

@ -0,0 +1,55 @@
/**
*
* directive name: ripple
*
* @description
*
*
* 使
* @example
* <template>
* <button v-ripple></button>
* </template>
* <template>
* <div
* style="height: 20px; line-height: 20px;text-align: center; border: 1px solid; padding: 6px;"
* v-ripple={[true, ['center']]}
* >
* modifiers center
* </div>
* </template>
*
* const DemoOne = () => <div v-ripple>hi</div>
* const DemoTwo = () => <div v-ripple={[true, ['circle']]}>hi</div>
*/
import './index.scss'
import { updateRipple, removeListeners, isRippleEnabled } from './utils'
import type { CustomDirectiveFC } from '@/directives/types'
import type { RippleValue, RippleElement } from './types'
const copyDirective: CustomDirectiveFC<RippleElement, RippleValue> = () => {
return {
mounted: (el, binding) => {
updateRipple(el, binding, false)
},
beforeUnmount: (el) => {
delete el.__d_ripple__
removeListeners(el)
},
updated: (el, binding) => {
if (binding.value === binding.oldValue) {
return
}
const wasEnabled = isRippleEnabled(binding.oldValue)
updateRipple(el, binding, wasEnabled)
},
}
}
export default copyDirective

View File

@ -0,0 +1,58 @@
export interface RippleOptions {
class?: string
center?: boolean
circle?: boolean
}
export type RippleValue =
| boolean
| {
class: string
}
export interface Ripple {
/**
*
* @description
*
*/
enabled?: boolean
/**
*
* @description
*
*/
centered?: boolean
/**
*
* @description
*
*/
circle?: boolean
/**
*
* @description
*
*/
class?: string
/**
*
* @description
*
*/
touched?: boolean
}
export interface RippleElement extends HTMLElement {
__d_ripple__?: Ripple
}
export interface RippleModifiers {
center?: boolean
circle?: boolean
}
export interface RippleBindingValue {
value?: RippleValue
modifiers: RippleModifiers
}

View File

@ -0,0 +1,283 @@
/**
*
* @description
* pure admin ripple directive
*
* ts
*
* @see https://github.dev/pure-admin/vue-pure-admin/blob/main/src/components/ReCol/index.ts
*/
import { isValueType } from '@/utils'
import type { RippleOptions, RippleElement, RippleBindingValue } from './types'
const animationClasses = {
vRippleAnimationEnter: 'v-ripple__animation--enter',
vRippleAnimationVisible: 'v-ripple__animation--visible',
vRippleAnimationIn: 'v-ripple__animation--in',
vRippleAnimationOut: 'v-ripple__animation--out',
vRippleAnimation: 'v-ripple__animation',
}
const vRippleContainerClass = 'v-ripple__container'
function transform(el: HTMLElement, value: string) {
el.style.transform = value
}
/**
*
* @param e current click event
*
* @description
*
*/
function rippleShow(e: PointerEvent) {
const value: RippleOptions = {}
const element = e.currentTarget as RippleElement | undefined
if (!element?.__d_ripple__ || element.__d_ripple__.touched) {
return
}
value.center = element.__d_ripple__.centered
if (element.__d_ripple__.class) {
value.class = element.__d_ripple__.class
}
ripples.show(e, element, value)
}
/**
*
* @param e current event
*
* @description
*
*/
function rippleHide(e: Event) {
const element = e.currentTarget as RippleElement | null
if (!element?.__d_ripple__) {
return
}
// 调用清理方法,隐藏水波纹
setTimeout(() => {
if (element.__d_ripple__) {
element.__d_ripple__.touched = false
}
})
ripples.hide(element)
}
/**
*
* @param e current click event
* @param el current binding element
* @param value current bind value
*
* @description
*
*/
const calculate = (
e: PointerEvent,
el: RippleElement,
value: RippleOptions = {},
) => {
const offset = el.getBoundingClientRect()
const localX = e.clientX - offset.left // 点击位置距离 el 水平距离
const localY = e.clientY - offset.top // 点击位置距离 el 垂直距离
let radius = 0 // 圆半径
let scale = 0.3 // 缩放比例
// 计算点击位置到 el 顶点最远距离,即为圆的最大半径(勾股定理)
if (el.__d_ripple__?.circle) {
scale = 0.15
radius = el.clientWidth / 2
radius = value.center
? radius
: radius + Math.sqrt((localX - radius) ** 2 + (localY - radius) ** 2) / 4
} else {
radius = Math.sqrt(el.clientWidth ** 2 + el.clientHeight ** 2) / 2
}
const centerX = `${(el.clientWidth - radius * 2) / 2}px` // 水平中心点
const centerY = `${(el.clientHeight - radius * 2) / 2}px` // 垂直中心点
const x = value.center ? centerX : `${localX - radius}px` // 点击位置水平坐标
const y = value.center ? centerY : `${localY - radius}px` // 点击位置垂直坐标
return {
radius,
scale,
x,
y,
centerX,
centerY,
}
}
const ripples = {
show(e: PointerEvent, el: RippleElement, value: RippleOptions = {}) {
if (!el?.__d_ripple__?.enabled) {
return
}
// 创建 ripple 元素和 ripple 父元素
const container = document.createElement('span')
const animation = document.createElement('span')
container.appendChild(animation)
container.className = vRippleContainerClass
if (value.class) {
container.className += ` ${value.class}`
}
const { radius, scale, x, y, centerX, centerY } = calculate(e, el, value)
// ripple 圆大小
const size = `${radius * 2}px`
animation.className = animationClasses['vRippleAnimation']
animation.style.width = size
animation.style.height = size
el.appendChild(container)
// 获取目标元素样式表
const computed = window.getComputedStyle(el)
// 防止 position 被覆盖导致 ripple 位置有问题
if (computed && computed.position === 'static') {
el.style.position = 'relative'
el.dataset.previousPosition = 'static'
}
animation.classList.add(animationClasses['vRippleAnimationEnter'])
animation.classList.add(animationClasses['vRippleAnimationVisible'])
transform(
animation,
`translate(${x}, ${y}) scale3d(${scale},${scale},${scale})`,
)
animation.dataset.activated = String(performance.now())
setTimeout(() => {
animation.classList.remove(animationClasses['vRippleAnimationEnter'])
animation.classList.add(animationClasses['vRippleAnimationIn'])
transform(animation, `translate(${centerX}, ${centerY}) scale3d(1,1,1)`)
}, 0)
},
hide(el: RippleElement | null) {
if (!el?.__d_ripple__?.enabled) {
return
}
const ripples = el.getElementsByClassName(
animationClasses['vRippleAnimation'],
)
if (ripples.length === 0) {
return
}
const animation = ripples[ripples.length - 1] as HTMLElement
if (animation.dataset.isHiding) {
return
}
animation.dataset.isHiding = 'true'
const diff = performance.now() - Number(animation.dataset.activated)
const delay = Math.max(250 - diff, 0)
setTimeout(() => {
animation.classList.remove(animationClasses['vRippleAnimationIn'])
animation.classList.add(animationClasses['vRippleAnimationOut'])
setTimeout(() => {
const ripples = el.getElementsByClassName(
animationClasses['vRippleAnimation'],
)
if (ripples.length === 1 && el.dataset.previousPosition) {
el.style.position = el.dataset.previousPosition
delete el.dataset.previousPosition
}
if (animation.parentNode?.parentNode === el) {
el.removeChild(animation.parentNode)
}
}, 300)
}, delay)
},
}
/**
*
* @param value valid value
*
* @description
*
*/
export function isRippleEnabled(value: unknown): value is true {
return typeof value === 'undefined' || !!value
}
/**
*
* @param el current binging element
*
* @description
*
*/
export function removeListeners(el: HTMLElement) {
el.removeEventListener('pointerdown', rippleShow)
el.removeEventListener('pointerup', rippleHide)
}
/**
*
* @param el current binding element
* @param binding current value
* @param isEnabled is enabled
*
* @description
*
*/
export function updateRipple(
el: RippleElement,
binding: RippleBindingValue,
isEnabled: boolean,
) {
const { value, modifiers } = binding
const enabled = isRippleEnabled(value)
if (!enabled) {
ripples.hide(el)
}
el.__d_ripple__ = el.__d_ripple__ ?? {}
el.__d_ripple__.enabled = enabled
el.__d_ripple__.centered = modifiers.center
el.__d_ripple__.circle = modifiers.circle
if (isValueType<object>(value, 'Object') && value.class) {
el.__d_ripple__.class = value.class
}
if (enabled && !isEnabled) {
el.addEventListener('pointerdown', rippleShow)
el.addEventListener('pointerup', rippleHide)
} else if (!enabled && isEnabled) {
removeListeners(el)
}
}

View File

@ -13,11 +13,11 @@
* *
* directive name: throttle * directive name: throttle
* *
* @description
* 使 func * 使 func
* *
* trigger wait trigger clickwait 500 * trigger wait trigger clickwait 500
* *
* 使
* @example * @example
* <template> * <template>
* <div v-throttle="{ func: () => console.log('throttle') }"></div> * <div v-throttle="{ func: () => console.log('throttle') }"></div>

View File

@ -45,9 +45,12 @@ const variableState = reactive({
}) })
export type VariableState = typeof variableState export type VariableState = typeof variableState
export type VariableStateKey = keyof VariableState export type VariableStateKey = keyof VariableState
type ReadonlyVariableState<T> = {
[K in keyof T & string]: Readonly<Ref<T[K]>>
}
/** /**
* *
* @param key variable key * @param key variable key
@ -84,3 +87,19 @@ export function setVariable<T extends VariableStateKey, FC extends AnyFC>(
export function getVariableToRefs<K extends VariableStateKey>(key: K) { export function getVariableToRefs<K extends VariableStateKey>(key: K) {
return readonly(toRef<VariableState, K>(variableState, key)) return readonly(toRef<VariableState, K>(variableState, key))
} }
/**
*
* @description
* variable ref
*
* getVariableToRefs variable 使
*
* @example
* const { globalMainLayoutLoad } = getVariable()
*/
export function getVariable(): ReadonlyVariableState<typeof variableState> {
return {
...toRefs(readonly(variableState)),
}
}

View File

@ -11,7 +11,7 @@
import { useMenuGetters, useMenuActions } from '@/store' import { useMenuGetters, useMenuActions } from '@/store'
import { useVueRouter, useAppRoot } from '@/hooks' import { useVueRouter, useAppRoot } from '@/hooks'
import { pick } from '@/utils' import { pick } from 'lodash-es'
import type { MenuTagOptions, Key, AppMenuOption } from '@/types' import type { MenuTagOptions, Key, AppMenuOption } from '@/types'

View File

@ -50,6 +50,13 @@ export interface UseElementFullscreenOptions {
* @default null * @default null
*/ */
backgroundColor?: string backgroundColor?: string
/**
*
* transition
*
* @default 'width 0.3s var(--r-bezier), height 0.3s var(--r-bezier)'
*/
transition?: string
} }
let currentZIndex = 999 let currentZIndex = 999
@ -92,8 +99,8 @@ export const useElementFullscreen = (
exit: _exit, exit: _exit,
backgroundColor, backgroundColor,
zIndex, zIndex,
transition = 'all 0.3s var(--r-bezier)',
} = options ?? {} } = options ?? {}
const cacheStyle: Partial<CSSProperties> = {} // 缓存一些需要被覆盖的样式,例如: transition
let isSetup = false let isSetup = false
const updateStyle = () => { const updateStyle = () => {
@ -110,7 +117,7 @@ export const useElementFullscreen = (
width: ${width.value}px !important; width: ${width.value}px !important;
height: ${height.value}px !important; height: ${height.value}px !important;
transform: translate(-${left}px, -${top}px) !important; transform: translate(-${left}px, -${top}px) !important;
transition: all 0.3s var(--r-bezier); transition: ${transition};
z-index: ${ z-index: ${
isValueType<null>(zIndex, 'Null') || isValueType<null>(zIndex, 'Null') ||
isValueType<undefined>(zIndex, 'Undefined') isValueType<undefined>(zIndex, 'Undefined')
@ -150,8 +157,7 @@ export const useElementFullscreen = (
isAppend = true isAppend = true
} }
cacheStyle.transition = element.style.transition element.style.transition = transition
element.style.transition = 'all 0.3s var(--r-bezier)'
_enter?.() _enter?.()
} }
@ -163,8 +169,6 @@ export const useElementFullscreen = (
const element = unrefElement(target) const element = unrefElement(target)
if (element) { if (element) {
;(element as HTMLElement).style.transition = cacheStyle.transition ?? ''
element.removeAttribute(ID_TAG) element.removeAttribute(ID_TAG)
} }
@ -189,8 +193,6 @@ export const useElementFullscreen = (
const element = unrefElement(target) as HTMLElement | null const element = unrefElement(target) as HTMLElement | null
if (element) { if (element) {
element.style.transition = cacheStyle.transition ?? ''
element.removeAttribute(ID_TAG) element.removeAttribute(ID_TAG)
} }

View File

@ -1,7 +1,8 @@
import { omit } from '@/utils' import { omit } from 'lodash-es'
import type { AnyFC } from '@/types' import type { AnyFC } from '@/types'
import type { PaginationProps } from 'naive-ui' import type { PaginationProps } from 'naive-ui'
import type { Ref } from 'vue'
type OmitKeys = type OmitKeys =
| 'themeOverrides' | 'themeOverrides'
@ -29,9 +30,6 @@ const defaultOptions: UsePaginationOptions = {
* *
* @description * @description
* 便 hook * 便 hook
*
* @warning
* callback
*/ */
export const usePagination = <T extends AnyFC>( export const usePagination = <T extends AnyFC>(
callback: T, callback: T,
@ -54,23 +52,23 @@ export const usePagination = <T extends AnyFC>(
]) ])
const methodsOptions = { const methodsOptions = {
onUpdatePage: (page: number) => { onUpdatePage: (page: number) => {
paginationRef.page = page paginationRef.value.page = page
callback() callback()
}, },
onUpdatePageSize: (pageSize: number) => { onUpdatePageSize: (pageSize: number) => {
paginationRef.pageSize = pageSize paginationRef.value.pageSize = pageSize
paginationRef.page = 1 paginationRef.value.page = 1
callback() callback()
}, },
} }
const paginationRef = reactive<PaginationProps>( const paginationRef = ref<PaginationProps>(
Object.assign({}, defaultOptions, omitOptions, methodsOptions), Object.assign({}, defaultOptions, omitOptions, methodsOptions),
) )
const updatePage = paginationRef.onUpdatePage as (page: number) => void const updatePage = paginationRef.value.onUpdatePage as (page: number) => void
const updatePageSize = paginationRef.onUpdatePageSize as ( const updatePageSize = paginationRef.value.onUpdatePageSize as (
pageSize: number, pageSize: number,
) => void ) => void
@ -79,7 +77,7 @@ export const usePagination = <T extends AnyFC>(
* @description * @description
* *
*/ */
const getItemCount = () => paginationRef.itemCount const getItemCount = () => paginationRef.value.itemCount
/** /**
* *
@ -89,7 +87,7 @@ export const usePagination = <T extends AnyFC>(
* *
*/ */
const setItemCount = (itemCount: number) => { const setItemCount = (itemCount: number) => {
paginationRef.itemCount = itemCount paginationRef.value.itemCount = itemCount
} }
/** /**
@ -97,7 +95,7 @@ export const usePagination = <T extends AnyFC>(
* @description * @description
* *
*/ */
const getPage = () => paginationRef.page const getPage = () => paginationRef.value.page
/** /**
* *
@ -117,7 +115,7 @@ export const usePagination = <T extends AnyFC>(
* @description * @description
* *
*/ */
const getPageSize = () => paginationRef.pageSize const getPageSize = () => paginationRef.value.pageSize
/** /**
* *
@ -137,7 +135,7 @@ export const usePagination = <T extends AnyFC>(
* @description * @description
* RTable * RTable
*/ */
const getPagination = () => paginationRef as UsePaginationOptions const getPagination = () => paginationRef.value as UsePaginationOptions
/** /**
* *
@ -147,7 +145,7 @@ export const usePagination = <T extends AnyFC>(
const getCallback = callback const getCallback = callback
return [ return [
paginationRef as PaginationProps, paginationRef as Ref<UsePaginationOptions>,
{ {
updatePage, updatePage,
updatePageSize, updatePageSize,

View File

@ -1,5 +1,6 @@
import print from 'print-js' import print from 'print-js'
import { unrefElement, omit } from '@/utils' import { unrefElement } from '@/utils'
import { omit } from 'lodash-es'
import type { BasicTarget } from '@/types' import type { BasicTarget } from '@/types'

View File

@ -15,3 +15,9 @@
- 导入 `svg` 图标 - 导入 `svg` 图标
- 命名(`命名必须全局唯一,并且尽量避免使用特殊符号` - 命名(`命名必须全局唯一,并且尽量避免使用特殊符号`
- 导入 `RIcon` 组件,配置 `name` 属性即可将 `svg` 作为图标使用 - 导入 `RIcon` 组件,配置 `name` 属性即可将 `svg` 作为图标使用
## 分包
`4.8.0` 版本起,支持了 `svg icon` 分包,可以按照业务模块进行分包管理。
仅需要在 `src/icons` 目录下新建一个文件夹,然后将对应的 `svg` 图标放入其中即可。

View File

Before

Width:  |  Height:  |  Size: 475 B

After

Width:  |  Height:  |  Size: 475 B

View File

Before

Width:  |  Height:  |  Size: 598 B

After

Width:  |  Height:  |  Size: 598 B

View File

Before

Width:  |  Height:  |  Size: 342 B

After

Width:  |  Height:  |  Size: 342 B

View File

Before

Width:  |  Height:  |  Size: 356 B

After

Width:  |  Height:  |  Size: 356 B

View File

Before

Width:  |  Height:  |  Size: 322 B

After

Width:  |  Height:  |  Size: 322 B

View File

Before

Width:  |  Height:  |  Size: 784 B

After

Width:  |  Height:  |  Size: 784 B

View File

Before

Width:  |  Height:  |  Size: 2.1 KiB

After

Width:  |  Height:  |  Size: 2.1 KiB

View File

Before

Width:  |  Height:  |  Size: 1018 B

After

Width:  |  Height:  |  Size: 1018 B

View File

Before

Width:  |  Height:  |  Size: 819 B

After

Width:  |  Height:  |  Size: 819 B

View File

Before

Width:  |  Height:  |  Size: 916 B

After

Width:  |  Height:  |  Size: 916 B

View File

Before

Width:  |  Height:  |  Size: 281 B

After

Width:  |  Height:  |  Size: 281 B

View File

Before

Width:  |  Height:  |  Size: 480 B

After

Width:  |  Height:  |  Size: 480 B

View File

Before

Width:  |  Height:  |  Size: 1000 B

After

Width:  |  Height:  |  Size: 1000 B

View File

Before

Width:  |  Height:  |  Size: 942 B

After

Width:  |  Height:  |  Size: 942 B

View File

Before

Width:  |  Height:  |  Size: 453 B

After

Width:  |  Height:  |  Size: 453 B

View File

Before

Width:  |  Height:  |  Size: 891 B

After

Width:  |  Height:  |  Size: 891 B

View File

Before

Width:  |  Height:  |  Size: 897 B

After

Width:  |  Height:  |  Size: 897 B

View File

@ -1,13 +0,0 @@
<svg t="1672296474975" class="icon" viewBox="0 0 1024 1024" version="1.1"
xmlns="http://www.w3.org/2000/svg" p-id="5926" width="64" height="64">
<path
fill="currentColor"
d="M832.1 185.1H609.4l-17.1-62c-9.6-34.6-40.5-58.8-75.3-58.8H196c-43.2 0-78.3 36.4-78.3 81.1V897c0 35.3 28.7 64 64 64H832c35.3 0 64-28.7 64-64V249c0.1-35.2-28.6-63.9-63.9-63.9z m-644.4-39.7c0-6.6 4.4-11.1 8.3-11.1h321c3.4 0 6.6 3.1 7.8 7.4l12 43.4H187.7v-39.7z m638.4 745.8H187.7V255.1h638.4v636.1z"
p-id="5927"></path>
<path fill="currentColor" d="M346.1 415.1m-35 0a35 35 0 1 0 70 0 35 35 0 1 0-70 0Z" p-id="5928"></path>
<path fill="currentColor" d="M462.3 380.1h257.8v70H462.3z" p-id="5929"></path>
<path fill="currentColor" d="M346.1 582.3m-35 0a35 35 0 1 0 70 0 35 35 0 1 0-70 0Z" p-id="5930"></path>
<path fill="currentColor" d="M462.3 547.3h257.8v70H462.3z" p-id="5931"></path>
<path fill="currentColor" d="M346.1 749.5m-35 0a35 35 0 1 0 70 0 35 35 0 1 0-70 0Z" p-id="5932"></path>
<path fill="currentColor" d="M462.3 714.5h257.8v70H462.3z" p-id="5933"></path>
</svg>

Before

Width:  |  Height:  |  Size: 1.0 KiB

View File

@ -1,6 +0,0 @@
<svg t="1698916917467" class="icon" viewBox="0 0 1024 1024" version="1.1"
xmlns="http://www.w3.org/2000/svg" p-id="10617" width="64" height="64">
<path
d="M776.145455 81.454545l214.10909 296.727273 10.472728 13.963637v552.727272H23.272727V392.145455l10.472728-13.963637L247.854545 81.454545h528.29091zM335.127273 406.109091H51.2v510.836364h921.6V406.109091H684.218182c-5.818182 36.072727-22.109091 69.818182-48.872727 96.581818-33.745455 33.745455-79.127273 52.363636-125.672728 52.363636-47.709091 0-91.927273-18.618182-125.672727-52.363636-25.6-26.763636-43.054545-60.509091-48.872727-96.581818z m621.381818-27.927273L762.181818 109.381818h-500.363636L67.490909 378.181818h293.271273c0 39.563636 16.256 76.8 44.183273 104.727273 27.927273 27.927273 66.327273 44.218182 104.727272 44.218182s76.8-16.290909 104.727273-44.218182c27.927273-27.927273 44.322909-65.163636 44.322909-104.727273H956.509091z"
fill="currentColor" p-id="10618"></path>
</svg>

Before

Width:  |  Height:  |  Size: 961 B

View File

Before

Width:  |  Height:  |  Size: 1.6 KiB

After

Width:  |  Height:  |  Size: 1.6 KiB

View File

Before

Width:  |  Height:  |  Size: 790 B

After

Width:  |  Height:  |  Size: 790 B

View File

Before

Width:  |  Height:  |  Size: 60 KiB

After

Width:  |  Height:  |  Size: 60 KiB

View File

Before

Width:  |  Height:  |  Size: 1.2 KiB

After

Width:  |  Height:  |  Size: 1.2 KiB

View File

Before

Width:  |  Height:  |  Size: 5.4 KiB

After

Width:  |  Height:  |  Size: 5.4 KiB

View File

Before

Width:  |  Height:  |  Size: 1.4 KiB

After

Width:  |  Height:  |  Size: 1.4 KiB

View File

@ -1,6 +0,0 @@
<svg t="1680413645364" class="icon" viewBox="0 0 1024 1024" version="1.1"
xmlns="http://www.w3.org/2000/svg" p-id="5387" width="64" height="64">
<path
d="M224 416.096V224l192-0.096 0.096 192.096L224 416.096zM416.096 160H223.904A64 64 0 0 0 160 223.904v192.192A64 64 0 0 0 223.904 480h192.192A64 64 0 0 0 480 416.096V223.904A64 64 0 0 0 416.096 160zM224 800.096V608l192-0.096 0.096 192.096L224 800.096zM416.096 544H223.904A64 64 0 0 0 160 607.904v192.192A64 64 0 0 0 223.904 864h192.192A64 64 0 0 0 480 800.096v-192.192A64 64 0 0 0 416.096 544zM608 416.096V224l192-0.096 0.096 192.096-192.096 0.096zM800.096 160h-192.192A64 64 0 0 0 544 223.904v192.192A64 64 0 0 0 607.904 480h192.192A64 64 0 0 0 864 416.096V223.904A64 64 0 0 0 800.096 160zM704 608a32 32 0 0 0-32 32v192a32 32 0 0 0 64 0v-192a32 32 0 0 0-32-32M576 608a32 32 0 0 0-32 32v192a32 32 0 0 0 64 0v-192a32 32 0 0 0-32-32M832 544a32 32 0 0 0-32 32v256a32 32 0 0 0 64 0v-256a32 32 0 0 0-32-32"
fill="currentColor" p-id="5388"></path>
</svg>

Before

Width:  |  Height:  |  Size: 1010 B

View File

Before

Width:  |  Height:  |  Size: 428 B

After

Width:  |  Height:  |  Size: 428 B

View File

Before

Width:  |  Height:  |  Size: 1.6 KiB

After

Width:  |  Height:  |  Size: 1.6 KiB

View File

Before

Width:  |  Height:  |  Size: 1.4 KiB

After

Width:  |  Height:  |  Size: 1.4 KiB

View File

Before

Width:  |  Height:  |  Size: 597 B

After

Width:  |  Height:  |  Size: 597 B

View File

Before

Width:  |  Height:  |  Size: 1.5 KiB

After

Width:  |  Height:  |  Size: 1.5 KiB

View File

Before

Width:  |  Height:  |  Size: 871 B

After

Width:  |  Height:  |  Size: 871 B

View File

Before

Width:  |  Height:  |  Size: 254 B

After

Width:  |  Height:  |  Size: 254 B

View File

Before

Width:  |  Height:  |  Size: 726 B

After

Width:  |  Height:  |  Size: 726 B

View File

Before

Width:  |  Height:  |  Size: 11 KiB

After

Width:  |  Height:  |  Size: 11 KiB

View File

Before

Width:  |  Height:  |  Size: 477 B

After

Width:  |  Height:  |  Size: 477 B

View File

Before

Width:  |  Height:  |  Size: 1.9 KiB

After

Width:  |  Height:  |  Size: 1.9 KiB

View File

Before

Width:  |  Height:  |  Size: 935 B

After

Width:  |  Height:  |  Size: 935 B

View File

Before

Width:  |  Height:  |  Size: 930 B

After

Width:  |  Height:  |  Size: 930 B

View File

Before

Width:  |  Height:  |  Size: 2.4 KiB

After

Width:  |  Height:  |  Size: 2.4 KiB

Some files were not shown because too many files have changed in this diff Show More