From 811bd0ca1cb1c58c5d8b63a5037aeec23c205f27 Mon Sep 17 00:00:00 2001 From: XiaoDaiGua-Ray <443547225@qq.com> Date: Fri, 11 Aug 2023 22:50:38 +0800 Subject: [PATCH 01/19] 4.1.7 --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index a9c15207..0f4ce76c 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "ray-template", "private": false, - "version": "4.1.6", + "version": "4.1.7", "type": "module", "engines": { "node": ">=16.0.0", From 6f98e8fb0db8554dbb18254c3a799450059c083a Mon Sep 17 00:00:00 2001 From: XiaoDaiGua-Ray <443547225@qq.com> Date: Mon, 14 Aug 2023 17:42:05 +0800 Subject: [PATCH 02/19] =?UTF-8?q?v4.1.7=E7=BB=86=E8=8A=82=E6=9B=B4?= =?UTF-8?q?=E6=96=B0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- cfg.ts | 4 ++++ mock/demo/person.mock.ts | 15 +++------------ mock/shared/database.ts | 20 ++++++++++++++++++-- mock/shared/utils.ts | 4 ++++ src/components/RayTable/src/index.tsx | 1 - src/views/demo/mock-demo/index.tsx | 3 +++ tsconfig.json | 7 +++++-- 7 files changed, 37 insertions(+), 17 deletions(-) diff --git a/cfg.ts b/cfg.ts index c3367645..9ac1939b 100644 --- a/cfg.ts +++ b/cfg.ts @@ -135,6 +135,10 @@ const config: AppConfigExport = { find: '@use-images', replacement: path.resolve(__dirname, './src/assets/images'), }, + { + find: '@mock', + replacement: path.resolve(__dirname, './mock'), + }, ], } diff --git a/mock/demo/person.mock.ts b/mock/demo/person.mock.ts index 8cfefbbf..f4232eeb 100644 --- a/mock/demo/person.mock.ts +++ b/mock/demo/person.mock.ts @@ -1,26 +1,17 @@ import { defineMock } from 'vite-plugin-mock-dev-server' +import { pagination, stringify, response, array } from '@mock/shared/utils' +import { tableMock } from '@mock/shared/database' import Mock from 'mockjs' -import { pagination, stringify, response } from '../shared/utils' -import { array } from '../shared/database' export const getPersonList = defineMock({ url: '/api/list', method: 'GET', delay: 500, response: (req, res) => { - const person = () => ({ - id: Mock.Random.guid(), - address: Mock.Random.county(true), - email: Mock.Random.email(), - name: Mock.Random.cname(), - age: Mock.Random.integer(18, 60), - createDate: Mock.Random.date(), - }) - const { query: { page, pageSize, email }, } = req - let list = array(100).map(() => person()) + let list = array(100).map(() => tableMock()) let length = list.length if (!page || !pageSize) { diff --git a/mock/shared/database.ts b/mock/shared/database.ts index 8652e13d..91b817c5 100644 --- a/mock/shared/database.ts +++ b/mock/shared/database.ts @@ -9,6 +9,22 @@ * @remark 今天也是元气满满撸代码的一天 */ -export function array(length: number) { - return new Array(length).fill(0) +import Mock from 'mockjs' + +/** + * + * @param option 自定义配置 + * + * 基础表格数据 + */ +export function tableMock(option?: object) { + return { + ...option, + id: Mock.Random.guid(), + address: Mock.Random.county(true), + email: Mock.Random.email(), + name: Mock.Random.cname(), + age: Mock.Random.integer(18, 60), + createDate: Mock.Random.date(), + } } diff --git a/mock/shared/utils.ts b/mock/shared/utils.ts index bfcb991b..94ccd564 100644 --- a/mock/shared/utils.ts +++ b/mock/shared/utils.ts @@ -11,6 +11,10 @@ /* eslint-disable @typescript-eslint/no-explicit-any */ +export function array(length: number) { + return new Array(length).fill(0) +} + /** * * @param pageCurrent 当前页码 diff --git a/src/components/RayTable/src/index.tsx b/src/components/RayTable/src/index.tsx index 7a2ca908..41213fd8 100644 --- a/src/components/RayTable/src/index.tsx +++ b/src/components/RayTable/src/index.tsx @@ -255,7 +255,6 @@ const RayTable = defineComponent({ } }, render() { - console.log(this.action) return ( { + console.log(data) + }, }) const columns = [ { diff --git a/tsconfig.json b/tsconfig.json index 6d680ace..322df7b6 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -23,7 +23,9 @@ "@use-api/*": ["src/axios/api/*"], "@use-images": ["src/assets/images"], "@use-images/*": ["src/assets/images"], - "@use-micro/*": ["src/micro/*"] + "@use-micro/*": ["src/micro/*"], + "@mock/*": ["mock/*"], + "@mock": ["mock/*"] }, "suppressImplicitAnyIndexErrors": true, "typeRoots": ["./src/types/app.d.ts", "./src/types/global.d.ts"], @@ -38,6 +40,7 @@ "package.json", "vite-env.d.ts", "./unplugin/**/*", - "src/**/*" + "src/**/*", + "mock/**/*" ] } From 17c5ca7e5029d573084ca5535a430023186b362e Mon Sep 17 00:00:00 2001 From: XiaoDaiGua-Ray <443547225@qq.com> Date: Sun, 20 Aug 2023 14:56:51 +0800 Subject: [PATCH 03/19] =?UTF-8?q?v4.1.8=E7=89=88=E6=9C=AC=E5=8F=91?= =?UTF-8?q?=E5=B8=83?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- CHANGELOG.md | 16 ++ package.json | 4 +- src/appConfig/designConfig.ts | 6 + src/components/RayChart/index.scss | 6 + src/components/RayChart/index.tsx | 226 +++++++++++------- src/components/RayChart/type.ts | 25 ++ src/components/RayIframe/index.ts | 3 + src/components/RayIframe/src/index.tsx | 20 +- src/components/RayTable/src/index.scss | 15 +- src/components/RayTable/src/index.tsx | 83 +++---- src/components/RayTable/src/props.ts | 43 ++-- .../components/SettingDrawer/index.tsx | 10 - src/router/routes.ts | 4 +- src/store/modules/setting/index.ts | 12 - src/store/modules/setting/type.ts | 1 - src/styles/base.scss | 19 +- src/styles/mixins.scss | 14 ++ src/types/modules/cfg.ts | 1 + src/types/modules/component.ts | 3 - src/views/demo/echart/index.tsx | 68 ++++-- 20 files changed, 355 insertions(+), 224 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index cfe443df..efc011ad 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,21 @@ # CHANGE LOG +## 4.1.8 + +### Feats + +- 更新 `vite` 版本至 `v4.4.9` +- 更新 `vue-hooks-plus` 版本至 `v1.8.1` +- 更新了 RayTable 的一些事件的命名 +- `RayChart` 组件做了一些调整 + - 支持指定 observer 监听对象,默认为 chart 组件本身 + - 默认开启 autoChangeTheme 功能 + - 支持配置 throttleWait 节流等待时间,默认 500ms + - 支持通过配置 `desginConfig.echartTheme` 属性指定 `echart theme`。并且只需按照约定方式注册的主题,只需要指定主题名称,即可完成 `light` `dark` 两种主题指定 + - RayChartInst 新增 dispose render 方法,允许手动渲染与卸载 chart 图 + - 新增 animation 属性,如果为 true 则会强制触发渲染过渡动画。该配置受 `options.animation` 属性影响,如果该配置为 false 则不会启用过渡动画 +- 移除反转色功能 + ## 4.1.7 ### Feats diff --git a/package.json b/package.json index 0f4ce76c..d24f928b 100644 --- a/package.json +++ b/package.json @@ -41,7 +41,7 @@ "sass": "^1.54.3", "screenfull": "^6.0.2", "vue": "^3.3.4", - "vue-hooks-plus": "1.7.6", + "vue-hooks-plus": "1.8.1", "vue-i18n": "^9.2.2", "vue-router": "^4.2.4", "vuedraggable": "^4.1.0", @@ -85,7 +85,7 @@ "typescript": "^5.0.2", "unplugin-auto-import": "^0.15.0", "unplugin-vue-components": "^0.25.1", - "vite": "^4.3.9", + "vite": "^4.4.9", "vite-plugin-compression": "^0.5.1", "vite-plugin-ejs": "^1.6.4", "vite-plugin-eslint": "1.8.1", diff --git a/src/appConfig/designConfig.ts b/src/appConfig/designConfig.ts index 013d2d67..f07b04bb 100644 --- a/src/appConfig/designConfig.ts +++ b/src/appConfig/designConfig.ts @@ -60,4 +60,10 @@ export const APP_THEME: AppTheme = { * 地址: */ APP_NAIVE_UI_THEME_OVERRIDES: {}, + /** + * + * 配置 echart 主题颜色 + * 约定配置时以:主题名称为文件名,其文件夹下两个主题风格的 json 文件。并且暗色主题必须为 xxx-dark.json + */ + echartTheme: 'macarons', } diff --git a/src/components/RayChart/index.scss b/src/components/RayChart/index.scss index 5ea9021d..4e574163 100644 --- a/src/components/RayChart/index.scss +++ b/src/components/RayChart/index.scss @@ -5,4 +5,10 @@ outline: none; box-sizing: border-box; transition: width 0.35s var(--r-bezier); + + & .ray-chart__container { + width: 100%; + height: 100%; + box-sizing: border-box; + } } diff --git a/src/components/RayChart/index.tsx b/src/components/RayChart/index.tsx index fcce942c..7439beec 100644 --- a/src/components/RayChart/index.tsx +++ b/src/components/RayChart/index.tsx @@ -36,17 +36,19 @@ import { PictorialBarChart, } from 'echarts/charts' // 系列类型(后缀都为 `SeriesOption`) import { LabelLayout, UniversalTransition } from 'echarts/features' // 标签自动布局, 全局过渡动画等特性 -import { CanvasRenderer } from 'echarts/renderers' // `echarts` 渲染器 +import { + CanvasRenderer, + // SVGRenderer, +} from 'echarts/renderers' // `echarts` 渲染器 import { useSetting } from '@/store' import { cloneDeep, throttle } from 'lodash-es' import { on, off, completeSize } from '@/utils/element' import { call } from '@/utils/vue/index' import { setupChartTheme, loadingOptions } from './helper' -import { LAYOUT_CONTENT_REF } from '@/appConfig/routerConfig' +import { APP_THEME } from '@/appConfig/designConfig' import type { PropType } from 'vue' -import type { EChartsInstance } from '@/types/modules/component' import type { AnyFC, MaybeArray } from '@/types/modules/utils' import type { DebouncedFunc } from 'lodash-es' import type { @@ -54,9 +56,15 @@ import type { AutoResize, ChartTheme, } from '@/components/RayChart/type' -import type { UseResizeObserverReturn } from '@vueuse/core' +import type { + UseResizeObserverReturn, + MaybeComputedElementRef, + MaybeElement, +} from '@vueuse/core' +import type { ECharts, EChartsCoreOption } from 'echarts/core' export type EChartsExtensionInstallRegisters = typeof CanvasRenderer +export type { RayChartInst } from './type' const RayChart = defineComponent({ name: 'RayChart', @@ -116,21 +124,19 @@ const RayChart = defineComponent({ type: Object as PropType, default: () => ({}), }, - success: { + onSuccess: { /** * * 返回 chart 实例 * * 渲染成功回调函数 * - * () => EChartsInstance + * () => ECharts */ - type: [Function, Array] as PropType< - MaybeArray<(e: EChartsInstance) => void> - >, + type: [Function, Array] as PropType void>>, default: null, }, - error: { + onError: { /** * * 渲染失败回调函数 @@ -148,13 +154,12 @@ const RayChart = defineComponent({ /** * * 是否自动跟随模板主题切换 - * * 如果开启此属性, 则会覆盖 `theme` 属性 * - * 注意: 这个属性重度依赖此模板, 所以默认不开启. 并且动态切换主题有一定的性能问题 + * 注意: 这个属性重度依赖此模板 */ type: Boolean, - default: false, + default: true, }, use: { /** @@ -180,15 +185,37 @@ const RayChart = defineComponent({ type: Object as PropType, default: () => loadingOptions(), }, + observer: { + /** + * + * 需要被监听尺寸的元素 + * 需要开启 autoResize 才能生效 + * 默认以父元素作为监听对象 + */ + type: Object as PropType>, + default: null, + }, + throttleWait: { + /** 节流等待时间 */ + type: Number, + default: 500, + }, + animation: { + /** 是否强制启用渲染动画 */ + type: Boolean, + default: true, + }, }, setup(props, { expose }) { const settingStore = useSetting() const { themeValue } = storeToRefs(settingStore) const rayChartRef = ref() // `echart` 容器实例 - const echartInstanceRef = ref() // `echart` 拷贝实例, 解决直接使用响应式实例带来的问题 - let echartInstance: EChartsInstance // `echart` 实例 - let resizeThrottle: DebouncedFunc // resize 防抖方法实例 - let resizeOvserverReturn: UseResizeObserverReturn | undefined + const rayChartWrapperRef = ref() + const echartInstanceRef = ref() // `echart` 实例 + let echartInstance: ECharts | null // `echart` 拷贝实例, 解决直接使用响应式实例带来的问题 + let resizeThrottleReturn: DebouncedFunc | null // resize 防抖方法实例 + let resizeOvserverReturn: UseResizeObserverReturn | null + const { echartTheme } = APP_THEME const cssVarsRef = computed(() => { const cssVars = { @@ -198,9 +225,6 @@ const RayChart = defineComponent({ return cssVars }) - const modelLoadingOptions = computed(() => - loadingOptions(props.loadingOptions), - ) /** * @@ -237,7 +261,7 @@ const RayChart = defineComponent({ echarts.use([CanvasRenderer]) // 注册渲染器 try { - echarts.use(props.use) + echarts.use(props.use?.filter(Boolean)) } catch (e) { console.error( 'Error: wrong property and method passed in extend attribute', @@ -253,10 +277,17 @@ const RayChart = defineComponent({ * * 如果有需要特殊全局配置的可以在此继续写... */ - const combineChartOptions = () => { - let options = cloneDeep(props.options) + const combineChartOptions = (ops: EChartsCoreOption) => { + let options = cloneDeep(ops) - const assign = (opts: object) => Object.assign({}, options, opts) + const assign = (opts: object) => + Object.assign( + { + animation: true, + }, + options, + opts, + ) if (props.showAria) { options = assign({ @@ -277,17 +308,16 @@ const RayChart = defineComponent({ * 渲染 `echart` * * 缓存两个实例 - * * 直接使用响应式代理实例会出现诡异的问题, 例如 `legend` 点击时报错 */ - const renderChart = (theme: ChartTheme = 'macarons') => { + const renderChart = (theme: ChartTheme = echartTheme) => { /** 获取 dom 容器 */ const element = rayChartRef.value as HTMLElement /** 获取配置项 */ - const options = combineChartOptions() + const options = combineChartOptions(props.options) /** 获取 dom 容器实际宽高 */ const { height, width } = element.getBoundingClientRect() - const { success, error } = props + const { onSuccess, onError } = props try { /** 注册主题 */ @@ -305,16 +335,22 @@ const RayChart = defineComponent({ echartInstanceRef.value = echartInstance /** 设置 options 配置项 */ - options && echartInstance.setOption(options) + options && echartInstance.setOption({}) + + if (props.animation) { + setTimeout(() => { + options && echartInstance?.setOption(options) + }) + } /** 渲染成功回调 */ - if (success) { - call(success, echartInstance) + if (onSuccess) { + call(onSuccess, echartInstance) } } catch (e) { /** 渲染失败回调 */ - if (error) { - call(error) + if (onError) { + call(onError) } console.error('RayChart render error: ', e) @@ -329,9 +365,9 @@ const RayChart = defineComponent({ */ const renderThemeChart = (bool?: boolean) => { if (props.autoChangeTheme) { - bool ? renderChart('macarons-dark') : renderChart() + bool ? renderChart(`${echartTheme}-dark`) : renderChart() - return void 0 + return } if (!props.theme) { @@ -357,10 +393,51 @@ const RayChart = defineComponent({ } } + const mount = () => { + // 避免重复渲染 + if (echartInstance?.getDom()) { + console.warn( + 'RayChart mount: There is a chart instance already initialized on the dom. Execution was interrupted', + ) + + return + } + + if (props.autoChangeTheme) { + /** 注册 echarts */ + renderThemeChart(themeValue.value) + } else { + props.theme ? renderChart(`${echartTheme}-dark`) : renderChart() + } + + /** 注册事件 */ + if (props.autoResize) { + resizeThrottleReturn = throttle(resizeChart, props.throttleWait) + /** 监听内容区域尺寸变化更新 chart */ + resizeOvserverReturn = useResizeObserver( + props.observer || rayChartWrapperRef, + resizeThrottleReturn, + ) + + on(window, 'resize', resizeThrottleReturn) + } + } + + const unmount = () => { + /** 卸载 echarts */ + destroyChart() + /** 卸载事件柄 */ + resizeThrottleReturn && off(window, 'resize', resizeThrottleReturn) + /** 注销防抖 */ + resizeThrottleReturn?.cancel() + /** 注销 observer 监听 */ + resizeOvserverReturn?.stop?.() + } + /** 监听全局主题变化, 然后重新渲染对应主题 echarts */ watch( - () => [themeValue.value], - ([theme]) => { + () => themeValue.value, + (theme) => { /** * * Q: 为什么需要重新卸载再渲染 @@ -375,19 +452,19 @@ const RayChart = defineComponent({ }, ) + /** + * + * 贴花跟随主题渲染 + * + * 自动跟随模板主题或者指定主题皆可 + */ watch( () => props.showAria, () => { destroyChart() - /** - * - * 贴花跟随主题渲染 - * - * 自动跟随模板主题或者指定主题皆可 - */ if (props.autoChangeTheme || props.theme) { - themeValue.value ? renderChart('macarons-dark') : renderChart() + themeValue.value ? renderChart(`${echartTheme}-dark`) : renderChart() } else { renderChart() } @@ -397,27 +474,36 @@ const RayChart = defineComponent({ /** 显示/隐藏加载动画 */ watch( () => props.loading, - (newData) => { - newData - ? echartInstance?.showLoading(modelLoadingOptions.value) + (ndata) => { + ndata + ? echartInstance?.showLoading(props.loadingOptions) : echartInstance?.hideLoading() }, ) - /** 监听 options 变化 */ if (props.watchOptions) { + /** 监听 options 变化 */ watch( - () => props.watchOptions, - () => { + () => props.options, + (noptions) => { /** 重新组合 options */ - const options = combineChartOptions() + const options = combineChartOptions(noptions) /** 如果 options 发生变动更新 echarts */ echartInstance?.setOption(options) }, + { + deep: true, + }, ) } + expose({ + echart: echartInstanceRef, + dispose: unmount, + render: mount, + }) + onBeforeMount(async () => { /** 注册 echarts 组件与渲染器 */ await registerChartCore() @@ -425,51 +511,25 @@ const RayChart = defineComponent({ onMounted(() => { nextTick(() => { - /** 注册 echarts */ - if (props.autoChangeTheme) { - renderThemeChart(themeValue.value) - } else { - props.theme ? renderChart('macarons-dark') : renderChart() - } - - /** 注册事件 */ - if (props.autoResize) { - resizeThrottle = throttle(resizeChart, 500) - - on(window, 'resize', resizeThrottle) - } - - /** 监听内容区域尺寸变化更新 chart */ - resizeOvserverReturn = useResizeObserver( - LAYOUT_CONTENT_REF.value as unknown as Ref, - resizeThrottle, - ) + mount() }) }) onBeforeUnmount(() => { - /** 卸载 echarts */ - destroyChart() - /** 卸载事件柄 */ - off(window, 'resize', resizeThrottle) - /** 注销防抖 */ - resizeThrottle.cancel() - resizeOvserverReturn?.stop?.() - }) - - expose({ - echart: echartInstanceRef, + unmount() }) return { rayChartRef, cssVarsRef, - echartInstance: echartInstanceRef, + rayChartWrapperRef, } }, render() { return ( -
+
+
+
) }, }) diff --git a/src/components/RayChart/type.ts b/src/components/RayChart/type.ts index d2810740..0750be61 100644 --- a/src/components/RayChart/type.ts +++ b/src/components/RayChart/type.ts @@ -9,6 +9,8 @@ * @remark 今天也是元气满满撸代码的一天 */ +import type { ECharts, EChartsCoreOption } from 'echarts/core' + export interface ChartThemeRawModules { default: Record } @@ -41,3 +43,26 @@ export type AutoResize = } export type ChartTheme = 'macarons-dark' | string | object | 'macarons' + +export interface RayChartInst { + /** + * + * echart 实例 + * 访问当前 chart 图所有方法与属性 + * + * @default undefined + */ + echart: Ref + /** + * + * 手动卸载当前 chart 图 + * 注意:不会卸载当前组件,仅仅是卸载 chart + */ + dispose: () => void + /** + * + * 手动渲染 chart 图 + * 注意:会根据当前的 options 配置项与 props 配置项重新渲染 chart + */ + render: () => void +} diff --git a/src/components/RayIframe/index.ts b/src/components/RayIframe/index.ts index ecea3ef2..e0123a4c 100644 --- a/src/components/RayIframe/index.ts +++ b/src/components/RayIframe/index.ts @@ -1,3 +1,6 @@ import RayIframe from './src/index' +import type { RayIframeInst } from './src/index' + export default RayIframe +export type { RayIframeInst } diff --git a/src/components/RayIframe/src/index.tsx b/src/components/RayIframe/src/index.tsx index f7834848..2ec2871e 100644 --- a/src/components/RayIframe/src/index.tsx +++ b/src/components/RayIframe/src/index.tsx @@ -20,6 +20,10 @@ import type { PropType } from 'vue' import type { MaybeArray } from '@/types/modules/utils' import type { SpinProps } from 'naive-ui' +export interface RayIframeInst { + iframe: Ref +} + const RayIframe = defineComponent({ name: 'RayIframe', props: { @@ -73,7 +77,7 @@ const RayIframe = defineComponent({ type: String, default: null, }, - success: { + onSuccess: { /** * * iframe 加载成功回调 @@ -84,7 +88,7 @@ const RayIframe = defineComponent({ >, default: null, }, - error: { + onError: { /** * * iframe 加载失败回调 @@ -119,20 +123,20 @@ const RayIframe = defineComponent({ const iframeLoadSuccess = (e: Event) => { spinShow.value = false - const { success } = props + const { onSuccess } = props - if (success) { - call(success, iframeRef.value as HTMLIFrameElement, e) + if (onSuccess) { + call(onSuccess, iframeRef.value as HTMLIFrameElement, e) } } const iframeLoadError = (e: Event) => { spinShow.value = false - const { error } = props + const { onError } = props - if (error) { - call(error, e) + if (onError) { + call(onError, e) } } diff --git a/src/components/RayTable/src/index.scss b/src/components/RayTable/src/index.scss index 71ef89d8..06a30254 100644 --- a/src/components/RayTable/src/index.scss +++ b/src/components/RayTable/src/index.scss @@ -1,21 +1,10 @@ -@keyframes scaleScreenfull { - 0% { - transform: scale(1); - } - 50% { - transform: scale(1.3); - } - 100% { - transform: scale(1); - } -} - .ray-table { & .ray-table-icon { transition: transform 0.3s var(--r-bezier); &:hover { - animation: scaleScreenfull 0.3s linear; + @include scaleAnimate(); + animation: elementScale 0.3s linear; animation-direction: alternate; } } diff --git a/src/components/RayTable/src/index.tsx b/src/components/RayTable/src/index.tsx index 41213fd8..bc3de22c 100644 --- a/src/components/RayTable/src/index.tsx +++ b/src/components/RayTable/src/index.tsx @@ -58,8 +58,7 @@ import type { ComponentSize } from '@/types/modules/component' const RayTable = defineComponent({ name: 'RayTable', props: props, - emits: ['update:columns', 'exportSuccess', 'exportError'], - setup(props, { emit, expose }) { + setup(props, { expose }) { const rayTableInstance = ref() const tableUUID = uuid(16) // 表格 id, 用于打印表格 @@ -68,7 +67,15 @@ const RayTable = defineComponent({ const modelColumns = computed({ get: () => props.columns, set: (arr) => { - emit('update:columns', arr) + const { onUpdateColumns, 'onUpdate:columns': _onUpdateColumns } = props + + if (onUpdateColumns) { + call(onUpdateColumns, arr) + } + + if (_onUpdateColumns) { + call(_onUpdateColumns, arr) + } }, }) as unknown as WritableComputedRef const menuConfig = reactive({ @@ -76,7 +83,6 @@ const RayTable = defineComponent({ y: 0, showMenu: false, }) - let prevRightClickIndex = -1 // 缓存上次点击索引位置 const cssVars = computed(() => { const cssVar = { '--ray-table-header-space': props.tableHeaderSpace, @@ -86,6 +92,7 @@ const RayTable = defineComponent({ }) const tableSize = ref(props.size) const tableMethods = ref>() + let prevRightClickIndex = -1 // 缓存上次点击索引位置 /** 注入相关属性 */ provide('tableSettingProvider', { @@ -111,17 +118,12 @@ const RayTable = defineComponent({ key: string | number, option: DropdownOption, ) => { - const { onRightMenuClick, 'onUpdate:rightMenuClick': _onRightMenuClick } = - props + const { onRightMenuClick } = props if (onRightMenuClick) { call(onRightMenuClick, key, prevRightClickIndex, option) } - if (_onRightMenuClick) { - call(_onRightMenuClick, key, prevRightClickIndex, option) - } - menuConfig.showMenu = false } @@ -136,22 +138,24 @@ const RayTable = defineComponent({ const handleRowProps = (arr: ActionOptions, idx: number) => { const interceptRowProps = props.rowProps?.(arr, idx) + const contextmenu = modelRightClickMenu.value.length + ? (e: MouseEvent) => { + e.preventDefault() + + prevRightClickIndex = idx + menuConfig.showMenu = false + + nextTick().then(() => { + menuConfig.showMenu = true + menuConfig.x = e.clientX + menuConfig.y = e.clientY + }) + } + : void 0 + return { ...interceptRowProps, - onContextmenu: (e: MouseEvent) => { - e.preventDefault() - - prevRightClickIndex = idx - - menuConfig.showMenu = false - - nextTick().then(() => { - menuConfig.showMenu = true - - menuConfig.x = e.clientX - menuConfig.y = e.clientY - }) - }, + onContextmenu: contextmenu, } } @@ -164,6 +168,8 @@ const RayTable = defineComponent({ * 按需导入 `xlsx` 减少体积, 不依赖传统 `file save` 插件导出方式 */ const handleExportPositive = async () => { + const { onExportSuccess, onExportError } = props + if (props.data.length && props.columns.length) { try { await exportFileToXLSX( @@ -174,9 +180,9 @@ const RayTable = defineComponent({ }, ) - emit('exportSuccess') + onExportSuccess && call(onExportSuccess) } catch (e) { - emit('exportError') + onExportError && call(onExportError) } } } @@ -276,21 +282,16 @@ const RayTable = defineComponent({ ...this.$slots, }} - {this.showMenu ? ( - // 右键菜单 - (this.showMenu = false)} - onSelect={this.handleRightMenuSelect.bind(this)} - /> - ) : ( - '' - )} + (this.showMenu = false)} + onSelect={this.handleRightMenuSelect.bind(this)} + /> ), header: () => this.title ||
, diff --git a/src/components/RayTable/src/props.ts b/src/components/RayTable/src/props.ts index 468f4675..4e41f180 100644 --- a/src/components/RayTable/src/props.ts +++ b/src/components/RayTable/src/props.ts @@ -15,7 +15,7 @@ import type { PropType, VNode, VNodeChild } from 'vue' import type { DropdownMixedOption } from './type' import type PrintConfiguration from 'print-js' import type { MaybeArray } from '@/types/modules/utils' -import type { DropdownOption } from 'naive-ui' +import type { DropdownOption, DataTableColumn } from 'naive-ui' const rayTableProps = { ...dataTableProps, // 继承 `data table props` @@ -39,14 +39,6 @@ const rayTableProps = { >, default: null, }, - 'onUpdate:rightMenuClick': { - type: [Function, Array] as PropType< - MaybeArray< - (key: string | number, index: number, option: DropdownOption) => void - > - >, - default: null, - }, title: { /** * @@ -77,16 +69,6 @@ const rayTableProps = { type: Object as PropType, default: () => ({}), }, - showMenu: { - /** - * - * 是否展示右键菜单 - * - * 默认启用 - */ - type: Boolean, - default: true, - }, exportTooltip: { /** * @@ -225,7 +207,30 @@ const rayTableProps = { type: Boolean, default: false, }, + /** 导出成功 */ + onExportSuccess: { + type: [Function, Array] as PropType void>>, + default: null, + }, + /** 导出失败 */ + onExportError: { + type: [Function, Array] as PropType void>>, + default: null, + }, + onUpdateColumns: { + type: [Function, Array] as PropType< + MaybeArray<(arr: DataTableColumn[]) => void> + >, + default: null, + }, + 'onUpdate:columns': { + type: [Function, Array] as PropType< + MaybeArray<(arr: DataTableColumn[]) => void> + >, + default: null, + }, } as const + export default rayTableProps /** diff --git a/src/layout/components/SiderBar/components/SettingDrawer/index.tsx b/src/layout/components/SiderBar/components/SettingDrawer/index.tsx index 44f196ac..e03dcd5a 100644 --- a/src/layout/components/SiderBar/components/SettingDrawer/index.tsx +++ b/src/layout/components/SiderBar/components/SettingDrawer/index.tsx @@ -48,7 +48,6 @@ const SettingDrawer = defineComponent({ primaryColorOverride, menuTagSwitch, breadcrumbSwitch, - invertSwitch, footerSwitch, contentTransition, } = storeToRefs(settingStore) @@ -87,7 +86,6 @@ const SettingDrawer = defineComponent({ menuTagSwitch, changeSwitcher, breadcrumbSwitch, - invertSwitch, footerSwitch, contentTransitionOptions, contentTransition, @@ -155,14 +153,6 @@ const SettingDrawer = defineComponent({ } /> - - - this.changeSwitcher(bool, 'invertSwitch') - } - /> - diff --git a/src/router/routes.ts b/src/router/routes.ts index f707f8a2..3486b88f 100644 --- a/src/router/routes.ts +++ b/src/router/routes.ts @@ -3,8 +3,6 @@ import { getAppRawRoutes } from './routeModules' import { ROOT_ROUTE } from '@/appConfig/appConfig' import { expandRoutes } from '@/router/helper/expandRoutes' -const { path } = ROOT_ROUTE - export default async () => [ { path: '/', @@ -14,7 +12,7 @@ export default async () => [ { path: '/', name: 'layout', - redirect: path, + redirect: ROOT_ROUTE.path, component: Layout, children: expandRoutes(getAppRawRoutes()), }, diff --git a/src/store/modules/setting/index.ts b/src/store/modules/setting/index.ts index 9c8ac87a..9de80aca 100644 --- a/src/store/modules/setting/index.ts +++ b/src/store/modules/setting/index.ts @@ -32,7 +32,6 @@ export const useSetting = defineStore( reloadRouteSwitch: true, // 刷新路由开关 menuTagSwitch: true, // 多标签页开关 spinSwitch: false, // 全屏加载 - invertSwitch: false, // 反转色模式 breadcrumbSwitch: true, // 面包屑开关 localeLanguage: getAppDefaultLanguage(), lockScreenSwitch: false, // 锁屏开关 @@ -93,17 +92,6 @@ export const useSetting = defineStore( } } - /** 动态添加反转色 class name */ - watch( - () => settingState.invertSwitch, - (newData) => { - const body = document.body - const className = 'ray-template--invert' - - newData ? addClass(body, className) : removeClass(body, className) - }, - ) - return { ...toRefs(settingState), updateLocale, diff --git a/src/store/modules/setting/type.ts b/src/store/modules/setting/type.ts index ead87cf2..0ec6b67a 100644 --- a/src/store/modules/setting/type.ts +++ b/src/store/modules/setting/type.ts @@ -10,7 +10,6 @@ export interface SettingState { spinSwitch: boolean breadcrumbSwitch: boolean localeLanguage: string - invertSwitch: boolean lockScreenSwitch: boolean lockScreenInputSwitch: boolean footerSwitch: boolean diff --git a/src/styles/base.scss b/src/styles/base.scss index 4c135db3..2c3bcc9b 100644 --- a/src/styles/base.scss +++ b/src/styles/base.scss @@ -44,17 +44,24 @@ img { } body { - font-family: Inter, -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Oxygen, Ubuntu, Cantarell, "Fira Sans", - "Droid Sans", "Helvetica Neue", sans-serif; + font-family: + Inter, + -apple-system, + BlinkMacSystemFont, + "Segoe UI", + Roboto, + Oxygen, + Ubuntu, + Cantarell, + "Fira Sans", + "Droid Sans", + "Helvetica Neue", + sans-serif; text-rendering: optimizeLegibility; -webkit-font-smoothing: antialiased; -moz-osx-font-smoothing: grayscale; } -body.ray-template--invert { - filter: invert(1); -} - body .ray-template__directive--disabled { opacity: 0.3 !important; pointer-events: none !important; diff --git a/src/styles/mixins.scss b/src/styles/mixins.scss index 28aeffd1..16d4ca7d 100644 --- a/src/styles/mixins.scss +++ b/src/styles/mixins.scss @@ -47,3 +47,17 @@ @content; } } + +@mixin scaleAnimate($scale-from: 1, $scale-to: 1.3) { + @keyframes elementScale { + 0% { + transform: scale($scale-from); + } + 50% { + transform: scale($scale-to); + } + 100% { + transform: scale($scale-from); + } + } +} diff --git a/src/types/modules/cfg.ts b/src/types/modules/cfg.ts index aadb2165..f1384207 100644 --- a/src/types/modules/cfg.ts +++ b/src/types/modules/cfg.ts @@ -79,4 +79,5 @@ export interface AppTheme { APP_THEME_COLOR: string[] APP_PRIMARY_COLOR: AppPrimaryColor APP_NAIVE_UI_THEME_OVERRIDES: GlobalThemeOverrides + echartTheme: string } diff --git a/src/types/modules/component.ts b/src/types/modules/component.ts index e9fc9b12..33048ada 100644 --- a/src/types/modules/component.ts +++ b/src/types/modules/component.ts @@ -1,10 +1,7 @@ -import type { ECharts } from 'echarts/core' import type { MenuOption, MenuDividerOption, MenuGroupOption } from 'naive-ui' export type ComponentSize = 'small' | 'medium' | 'large' -export type EChartsInstance = ECharts - export type Placement = 'top' | 'right' | 'bottom' | 'left' export type NaiveMenuOptions = MenuOption | MenuDividerOption | MenuGroupOption diff --git a/src/views/demo/echart/index.tsx b/src/views/demo/echart/index.tsx index 0853d308..c478bcfe 100644 --- a/src/views/demo/echart/index.tsx +++ b/src/views/demo/echart/index.tsx @@ -1,14 +1,15 @@ import './index.scss' -import { NCard, NSwitch, NSpace, NP, NH2 } from 'naive-ui' +import { NCard, NSwitch, NSpace, NP, NH2, NButton } from 'naive-ui' import RayChart from '@/components/RayChart/index' -import type { EChartsInstance } from '@/types/modules/component' +import type { ECharts } from 'echarts/core' +import type { RayChartInst } from '@/components/RayChart/index' const Echart = defineComponent({ name: 'REchart', setup() { - const baseChartRef = ref() + const baseChartRef = ref() const chartLoading = ref(false) const chartAria = ref(false) const state = reactive({ @@ -186,7 +187,7 @@ const Echart = defineComponent({ chartAria.value = bool } - const handleChartRenderSuccess = (chart: EChartsInstance) => { + const handleChartRenderSuccess = (chart: ECharts) => { window.$notification.info({ title: '可视化图渲染成功回调函数', content: '可视化图渲染成功, 并且返回了当前可视化图实例', @@ -196,6 +197,14 @@ const Echart = defineComponent({ console.log(baseChartRef.value, chart) } + const mountChart = () => { + baseChartRef.value?.render() + } + + const unmountChart = () => { + baseChartRef.value?.dispose() + } + return { baseOptions, baseChartRef, @@ -207,39 +216,52 @@ const Echart = defineComponent({ basePieOptions, baseLineOptions, ...toRefs(state), + mountChart, + unmountChart, } }, render() { return (
- RayChart 组件使用 - - 该组件会默认以 200*200 - 宽高进行填充。预设了常用的图、方法组件,如果不满足需求,需要用 use - 方法进行手动拓展。该组件实现了自动跟随模板主题切换功能,但是动态切换损耗较大,所以默认不启用。 - 该组件可以让你只需要关注 options 的配置,无需关心 chart - 图的资源管理。并且该组件可以自动监听 options - 的变化,所以天生支持响应式,可以让你放心的加载异步数据。 - - 能跟随主题切换的可视化图 + +
    +
  • +

    当未获取到宽高时,组件会默认以 200*200 尺寸填充。

    +
  • +
  • +

    + 默认启用 autoChangeTheme,自动监听模板主题变化(RayTemplate + 独有) +

    +
  • +
  • +

    默认启用 watchOptions,自动监听配置项变化

    +
  • +
  • +

    默认启用 animation,强制启用渲染过渡动画

    +
  • +
+
+ 强制渲染过渡动画(animation) + + 渲染 + 卸载 +
- 渲染成功后运行回调函数 -
- -
不跟随主题切换的暗色主题可视化图
- +
加载动画 Date: Sun, 20 Aug 2023 14:57:22 +0800 Subject: [PATCH 04/19] 4.1.8 --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index d24f928b..226ba8b9 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "ray-template", "private": false, - "version": "4.1.7", + "version": "4.1.8", "type": "module", "engines": { "node": ">=16.0.0", From 2690e71713a4915ba3b6db70e09d8ca3f6f71667 Mon Sep 17 00:00:00 2001 From: XiaoDaiGua-Ray <443547225@qq.com> Date: Fri, 25 Aug 2023 16:49:16 +0800 Subject: [PATCH 05/19] =?UTF-8?q?v4.1.8=E4=B8=80=E4=BA=9B=E7=BB=86?= =?UTF-8?q?=E8=8A=82=E8=A1=A5=E5=85=85?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .eslintignore | 1 + .eslintrc.cjs | 12 +--- .vscode/settings.json | 10 ++- CHANGELOG.md | 3 + src/axios/index.ts | 6 +- src/components/RayChart/index.tsx | 50 ++++++++++---- src/directives/modules/copy/index.ts | 12 ++-- src/directives/modules/debounce/index.ts | 13 ++-- src/directives/modules/disabled/index.ts | 11 ++-- src/directives/modules/throttle/index.ts | 10 +-- src/icons/language.svg | 9 +-- src/icons/{scroll_reveal.svg => mouse.svg} | 0 src/router/helper/permission.ts | 2 +- src/router/modules/demo/router-demo.ts | 1 - src/router/modules/demo/svg-icons.ts | 17 +++++ src/store/modules/menu/index.ts | 2 - src/styles/base.scss | 29 +++++++++ src/views/demo/doc/index.tsx | 2 +- src/views/demo/echart/index.tsx | 28 +++++++- src/views/demo/iframe/index.tsx | 4 +- .../router-demo/router-demo-detail/index.tsx | 9 ++- .../router-demo/router-demo-home/index.tsx | 3 - src/views/demo/svg-icons/index.scss | 10 +++ src/views/demo/svg-icons/index.tsx | 65 +++++++++++++++++++ 24 files changed, 237 insertions(+), 72 deletions(-) rename src/icons/{scroll_reveal.svg => mouse.svg} (100%) create mode 100644 src/router/modules/demo/svg-icons.ts create mode 100644 src/views/demo/svg-icons/index.scss create mode 100644 src/views/demo/svg-icons/index.tsx diff --git a/.eslintignore b/.eslintignore index 108c8d65..989633d8 100644 --- a/.eslintignore +++ b/.eslintignore @@ -15,3 +15,4 @@ src/locales/lang .depcheckrc src/components/RayChart/theme *.md +src/icons/*.svg \ No newline at end of file diff --git a/.eslintrc.cjs b/.eslintrc.cjs index 7aeaa58d..f564c44a 100644 --- a/.eslintrc.cjs +++ b/.eslintrc.cjs @@ -106,7 +106,7 @@ module.exports = { 'no-lone-blocks': 2, // 禁止不必要的嵌套块 'no-multi-spaces': 1, // 禁止使用多余的空格 'no-multiple-empty-lines': [1, { max: 2 }], // 空行最多不能超过 `2` 行 - 'no-new-func': 1, // 禁止使用 `new Function` + 'no-new-func': 2, // 禁止使用 `new Function` 'no-new-object': 2, // 禁止使用 `new Object` 'no-new-require': 2, // 禁止使用 `new require` 'no-sparse-arrays': 2, // 禁止稀疏数组 @@ -124,7 +124,6 @@ module.exports = { 'no-useless-call': 2, // 禁止不必要的 `call` 和 `apply` 'no-var': 'error', // 禁用 `var` 'no-with': 2, // 禁用 `with` - 'no-undef': 0, 'use-isnan': 2, // 强制使用 isNaN 判断 NaN 'no-multi-assign': 2, // 禁止连续声明变量 'prefer-arrow-callback': 2, // 强制使用箭头函数作为回调 @@ -143,15 +142,6 @@ module.exports = { ], 'vue/require-v-for-key': ['error'], 'vue/require-valid-default-prop': ['error'], - 'no-use-before-define': [ - 'error', - { - functions: true, - classes: true, - variables: false, - allowNamedExports: false, - }, - ], 'vue/component-definition-name-casing': ['error', 'PascalCase'], 'vue/html-closing-bracket-newline': [ 'error', diff --git a/.vscode/settings.json b/.vscode/settings.json index 93251d2d..409e9964 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -9,5 +9,13 @@ "i18n-ally.displayLanguage": "zh-CN", "i18n-ally.enabledFrameworks": ["vue", "react"], "typescript.tsdk": "node_modules/typescript/lib", - "synthwave84.disableGlow": true + "alias-skip.mappings": { + "@": "/src", + "@use-utils": "/src/utils", + "@use-api": "/src/axios/api", + "@use-images": "/src/assets/images", + "@mock": "/mock" + }, + "alias-skip.allowedsuffix": ["ts", "tsx"], + "alias-skip.rootpath": "package.json" } diff --git a/CHANGELOG.md b/CHANGELOG.md index efc011ad..76374d13 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -15,6 +15,9 @@ - RayChartInst 新增 dispose render 方法,允许手动渲染与卸载 chart 图 - 新增 animation 属性,如果为 true 则会强制触发渲染过渡动画。该配置受 `options.animation` 属性影响,如果该配置为 false 则不会启用过渡动画 - 移除反转色功能 +- 新增图标页面 +- 修改国际化图标 +- 剔除无用代码,性能++++ ## 4.1.7 diff --git a/src/axios/index.ts b/src/axios/index.ts index 2dec3231..ec52083e 100644 --- a/src/axios/index.ts +++ b/src/axios/index.ts @@ -29,10 +29,10 @@ import type { AppRawRequestConfig } from '@/axios/type' /** * - * 该方法有一定的局限性,仅可在 setup 环境中使用 - * 如果在非 vue component 文件中使用,会抛出一些警告 + * 该方法有一定的局限性,仅可在 effect 作用域中使用 + * 如果在非 vue effect scope 中使用,会抛出一些警告 * - * 非 vue component 中使用 + * 非 vue effect 中使用 * @example * * const getUser = () => request({ url: 'http://localhost:3000/user' }) diff --git a/src/components/RayChart/index.tsx b/src/components/RayChart/index.tsx index 7439beec..bc6838ab 100644 --- a/src/components/RayChart/index.tsx +++ b/src/components/RayChart/index.tsx @@ -61,7 +61,7 @@ import type { MaybeComputedElementRef, MaybeElement, } from '@vueuse/core' -import type { ECharts, EChartsCoreOption } from 'echarts/core' +import type { ECharts, EChartsCoreOption, SetOptionOpts } from 'echarts/core' export type EChartsExtensionInstallRegisters = typeof CanvasRenderer export type { RayChartInst } from './type' @@ -205,6 +205,22 @@ const RayChart = defineComponent({ type: Boolean, default: true, }, + setChartOptions: { + /** + * + * 当 options 配置项更改时候,setOptions 方法配置项 + * + * 默认值 + * notMerge: false, + * lazyUpdate: true, + * silent: false, + * replaceMerge: [], + * + * 会自动进行合并配置项 + */ + type: Object as PropType, + default: () => ({}), + }, }, setup(props, { expose }) { const settingStore = useSetting() @@ -264,7 +280,7 @@ const RayChart = defineComponent({ echarts.use(props.use?.filter(Boolean)) } catch (e) { console.error( - 'Error: wrong property and method passed in extend attribute', + 'register chart Core error: wrong property and method passed in extend attribute', ) } } @@ -481,22 +497,28 @@ const RayChart = defineComponent({ }, ) - if (props.watchOptions) { - /** 监听 options 变化 */ - watch( - () => props.options, - (noptions) => { + /** 监听 options 变化 */ + watch( + () => props.options, + (noptions) => { + if (props.watchOptions) { /** 重新组合 options */ const options = combineChartOptions(noptions) + const setOpt = Object.assign({}, props.setChartOptions, { + notMerge: false, + lazyUpdate: true, + silent: false, + replaceMerge: [], + }) /** 如果 options 发生变动更新 echarts */ - echartInstance?.setOption(options) - }, - { - deep: true, - }, - ) - } + echartInstance?.setOption(options, setOpt) + } + }, + { + deep: true, + }, + ) expose({ echart: echartInstanceRef, diff --git a/src/directives/modules/copy/index.ts b/src/directives/modules/copy/index.ts index becc4554..b40cbe82 100644 --- a/src/directives/modules/copy/index.ts +++ b/src/directives/modules/copy/index.ts @@ -23,11 +23,9 @@ const copyDirective: CustomDirectiveFC = () => { let clipboard: ClipboardJS | null return { - mounted: (el, binding) => { - const value = binding.value - + mounted: (el, { value }) => { clipboard = new ClipboardJS(el, { - text: () => String(value), + text: () => value, }) clipboard?.on('success', () => { @@ -37,12 +35,10 @@ const copyDirective: CustomDirectiveFC = () => { window.$message.error('复制失败') }) }, - updated: (el, binding) => { + updated: (el, { value }) => { /** 其实这块代码写的挺蠢的, 但是我目前不知道怎么去优化, 阿巴阿巴阿巴 */ - const value = binding.value - clipboard = new ClipboardJS(el, { - text: () => String(value), + text: () => value, }) }, beforeUnmount: () => { diff --git a/src/directives/modules/debounce/index.ts b/src/directives/modules/debounce/index.ts index ec5cb83e..f7bd712a 100644 --- a/src/directives/modules/debounce/index.ts +++ b/src/directives/modules/debounce/index.ts @@ -30,16 +30,19 @@ const debounceDirective: CustomDirectiveFC< let debounceFunction: DebouncedFunc | null return { - beforeMount: (el, binding) => { - const { func, trigger = 'click', wait = 500, options } = binding.value + beforeMount: (el, { value }) => { + const { func, trigger = 'click', wait = 500, options } = value + if (typeof func !== 'function') { throw new Error('debounce directive value must be a function') } - debounceFunction = debounce(func, wait, Object.assign({}, {}, options)) + + debounceFunction = debounce(func, wait, Object.assign({}, options)) + on(el, trigger, debounceFunction) }, - beforeUnmount: (el, binding) => { - const { trigger = 'click' } = binding.value + beforeUnmount: (el, { value }) => { + const { trigger = 'click' } = value if (debounceFunction) { debounceFunction.cancel() diff --git a/src/directives/modules/disabled/index.ts b/src/directives/modules/disabled/index.ts index d49a6302..5bda0374 100644 --- a/src/directives/modules/disabled/index.ts +++ b/src/directives/modules/disabled/index.ts @@ -16,7 +16,6 @@ import { addClass, removeClass } from '@/utils/element' -import type { Directive } from 'vue' import type { CustomDirectiveFC } from '@/directives/type' const updateElementDisabledType = (el: HTMLElement, value: boolean) => { @@ -30,13 +29,13 @@ const updateElementDisabledType = (el: HTMLElement, value: boolean) => { const disabledDirective: CustomDirectiveFC = () => { return { - mounted: (el, binding) => { - const value = binding.value - + mounted: (el, { value }) => { updateElementDisabledType(el, value) }, - updated: (el, binding) => { - const value = binding.value + updated: (el, { value, oldValue }) => { + if (value === oldValue) { + return + } updateElementDisabledType(el, value) }, diff --git a/src/directives/modules/throttle/index.ts b/src/directives/modules/throttle/index.ts index 6a322aca..ccf91301 100644 --- a/src/directives/modules/throttle/index.ts +++ b/src/directives/modules/throttle/index.ts @@ -30,19 +30,19 @@ const throttleDirective: CustomDirectiveFC< let throttleFunction: DebouncedFunc | null return { - beforeMount: (el, binding) => { - const { func, trigger = 'click', wait = 500, options } = binding.value + beforeMount: (el, { value }) => { + const { func, trigger = 'click', wait = 500, options } = value if (typeof func !== 'function') { throw new Error('throttle directive value must be a function') } - throttleFunction = throttle(func, wait, Object.assign({}, {}, options)) + throttleFunction = throttle(func, wait, Object.assign({}, options)) on(el, trigger, throttleFunction) }, - beforeUnmount: (el, binding) => { - const { trigger = 'click' } = binding.value + beforeUnmount: (el, { value }) => { + const { trigger = 'click' } = value if (throttleFunction) { throttleFunction.cancel() diff --git a/src/icons/language.svg b/src/icons/language.svg index b399fdcb..96673e07 100644 --- a/src/icons/language.svg +++ b/src/icons/language.svg @@ -1,6 +1,7 @@ - + \ No newline at end of file diff --git a/src/icons/scroll_reveal.svg b/src/icons/mouse.svg similarity index 100% rename from src/icons/scroll_reveal.svg rename to src/icons/mouse.svg diff --git a/src/router/helper/permission.ts b/src/router/helper/permission.ts index a16469f0..3042d817 100644 --- a/src/router/helper/permission.ts +++ b/src/router/helper/permission.ts @@ -41,7 +41,7 @@ export const permissionRouter = (router: Router) => { beforeEach((to, from, next) => { const token = getStorage(APP_CATCH_KEY.token) - const catchRoutePath = getStorage( + const catchRoutePath = getStorage( 'menuKey', 'sessionStorage', ROOT_ROUTE.path, diff --git a/src/router/modules/demo/router-demo.ts b/src/router/modules/demo/router-demo.ts index 29f267cf..d7a41d81 100644 --- a/src/router/modules/demo/router-demo.ts +++ b/src/router/modules/demo/router-demo.ts @@ -29,7 +29,6 @@ const routerDemo: AppRouteRecordRaw = { import('@/views/demo/router-demo/router-demo-detail/index'), meta: { noLocalTitle: '信息详情', - hidden: true, sameLevel: true, }, }, diff --git a/src/router/modules/demo/svg-icons.ts b/src/router/modules/demo/svg-icons.ts new file mode 100644 index 00000000..c7aa971c --- /dev/null +++ b/src/router/modules/demo/svg-icons.ts @@ -0,0 +1,17 @@ +import { t } from '@/locales/useI18n' +import { LAYOUT } from '@/router/constant/index' + +import type { AppRouteRecordRaw } from '@/router/type' + +const previewSVGIcons: AppRouteRecordRaw = { + path: '/svg-icons', + name: 'PreviewSVGIcons', + component: () => import('@/views/demo/svg-icons/index'), + meta: { + noLocalTitle: 'SVG图标', + icon: 'other', + order: 3, + }, +} + +export default previewSVGIcons diff --git a/src/store/modules/menu/index.ts b/src/store/modules/menu/index.ts index 323f5b59..4b35bfca 100644 --- a/src/store/modules/menu/index.ts +++ b/src/store/modules/menu/index.ts @@ -34,11 +34,9 @@ import { } from './helper' import { useI18n } from '@/locales/useI18n' import { getAppRawRoutes } from '@/router/routeModules' -import { expandRoutes } from '@/router/helper/expandRoutes' import { useKeepAlive } from '@/store' import { useVueRouter } from '@/router/helper/useVueRouter' -import type { MenuOption } from 'naive-ui' import type { AppRouteMeta, AppRouteRecordRaw } from '@/router/type' import type { AppMenuOption, diff --git a/src/styles/base.scss b/src/styles/base.scss index 2c3bcc9b..623162c6 100644 --- a/src/styles/base.scss +++ b/src/styles/base.scss @@ -41,6 +41,35 @@ fieldset, img { border: 0; vertical-align: middle; + border-style: none; +} + +a { + background-color: transparent; +} + +sub { + bottom: -0.25em; +} + +sup { + top: -0.5em; +} + +button, +input, +optgroup, +select, +textarea { + font-family: inherit; /* 1 */ + font-size: 100%; /* 1 */ + line-height: 1.15; /* 1 */ + margin: 0; /* 2 */ +} + +html { + line-height: 1.15; /* 1 */ + -webkit-text-size-adjust: 100%; /* 2 */ } body { diff --git a/src/views/demo/doc/index.tsx b/src/views/demo/doc/index.tsx index 23dc6b56..027f4aeb 100644 --- a/src/views/demo/doc/index.tsx +++ b/src/views/demo/doc/index.tsx @@ -21,7 +21,7 @@ const RTemplateDoc = defineComponent({ ) diff --git a/src/views/demo/echart/index.tsx b/src/views/demo/echart/index.tsx index c478bcfe..12eb9b4c 100644 --- a/src/views/demo/echart/index.tsx +++ b/src/views/demo/echart/index.tsx @@ -3,6 +3,8 @@ import './index.scss' import { NCard, NSwitch, NSpace, NP, NH2, NButton } from 'naive-ui' import RayChart from '@/components/RayChart/index' +import dayjs from 'dayjs' + import type { ECharts } from 'echarts/core' import type { RayChartInst } from '@/components/RayChart/index' @@ -82,9 +84,9 @@ const Echart = defineComponent({ }, ], } - const baseLineOptions = { + const baseLineOptions = ref({ title: { - text: 'Stacked Area Chart', + text: dayjs().valueOf(), }, tooltip: { trigger: 'axis', @@ -177,7 +179,7 @@ const Echart = defineComponent({ data: [820, 932, 901, 934, 1290, 1330, 1320], }, ], - } + }) const handleLoadingShow = (bool: boolean) => { state.loading = bool @@ -205,6 +207,19 @@ const Echart = defineComponent({ baseChartRef.value?.dispose() } + const handleUpdateTitle = () => { + baseLineOptions.value.title.text = dayjs().valueOf() + + const createData = () => Math.floor((Math.random() + 1) * 100) + + baseLineOptions.value.series[0].data = new Array(7) + .fill(0) + .map(() => createData()) + baseLineOptions.value.series[1].data = new Array(7) + .fill(0) + .map(() => createData()) + } + return { baseOptions, baseChartRef, @@ -218,6 +233,7 @@ const Echart = defineComponent({ ...toRefs(state), mountChart, unmountChart, + handleUpdateTitle, } }, render() { @@ -240,12 +256,18 @@ const Echart = defineComponent({
  • 默认启用 animation,强制启用渲染过渡动画

  • +
  • +

    配置 setChartOptions 属性,可以定制化合并模式

    +
  • 强制渲染过渡动画(animation) 渲染 卸载 + + 更新配置项 +
    - + diff --git a/src/views/demo/router-demo/router-demo-detail/index.tsx b/src/views/demo/router-demo/router-demo-detail/index.tsx index 49e9005a..46cb6d23 100644 --- a/src/views/demo/router-demo/router-demo-detail/index.tsx +++ b/src/views/demo/router-demo/router-demo-detail/index.tsx @@ -13,12 +13,17 @@ import { NCard, NSpace } from 'naive-ui' const RouterDemoDetail = defineComponent({ name: 'RouterDemoDetail', - render() { return ( 我是平层路由详情页面 - 可以点击面包屑或者菜单返回到主页面 + +

    1. 可以点击面包屑或者菜单返回到主页面

    +

    + 2. 如果这个页面需要配置多个详情页面,只需将该路由所在的 children + 中,将所需页面配置为 sameLevel 即可。 +

    +
    ) }, diff --git a/src/views/demo/router-demo/router-demo-home/index.tsx b/src/views/demo/router-demo/router-demo-home/index.tsx index b72bf663..9319a692 100644 --- a/src/views/demo/router-demo/router-demo-home/index.tsx +++ b/src/views/demo/router-demo/router-demo-home/index.tsx @@ -52,9 +52,6 @@ const RouterDemoHome = defineComponent({ onClick={() => { router.push({ path: '/router-demo/router-demo-detail', - query: { - row: JSON.stringify(row), - }, }) }} > diff --git a/src/views/demo/svg-icons/index.scss b/src/views/demo/svg-icons/index.scss new file mode 100644 index 00000000..44170c60 --- /dev/null +++ b/src/views/demo/svg-icons/index.scss @@ -0,0 +1,10 @@ +.pre-view-icons__card { + width: 120px; + height: 120px; + padding: 18px; + display: flex; + justify-content: center; + align-items: center; + border: 1px solid var(--n-border-color); + border-radius: var(--n-border-radius); +} diff --git a/src/views/demo/svg-icons/index.tsx b/src/views/demo/svg-icons/index.tsx new file mode 100644 index 00000000..32f53652 --- /dev/null +++ b/src/views/demo/svg-icons/index.tsx @@ -0,0 +1,65 @@ +/** + * + * @author Ray + * + * @date 2023-08-25 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +import './index.scss' + +import { NSpace, NCard, NPopover } from 'naive-ui' +import RayIcon from '@/components/RayIcon/index' + +const PreviewSVGIcons = defineComponent({ + name: 'PreviewSVGIcons', + setup() { + const icons = ref([]) + const iconsModules = import.meta.glob('@/icons/**.svg') + + Object.keys(iconsModules).forEach((curr) => { + const iconName = curr.match(/\/(\w+)\.svg/)![1] + + if (iconName) { + icons.value.push(iconName) + } + }) + + return { + icons, + } + }, + render() { + return ( + + {{ + 'header-extra': () => '点击图标复制代码', + default: () => ( + + {this.icons.map((curr) => ( +
    `} + > + + {{ + trigger: () => ( + + ), + default: () => curr, + }} + +
    + ))} +
    + ), + }} +
    + ) + }, +}) + +export default PreviewSVGIcons From b1890df8f6464561ee3573f19ea095d80c960ab6 Mon Sep 17 00:00:00 2001 From: XiaoDaiGua-Ray <443547225@qq.com> Date: Fri, 25 Aug 2023 17:01:35 +0800 Subject: [PATCH 06/19] =?UTF-8?q?v4.1.8=E7=BB=86=E8=8A=82=E6=9B=B4?= =?UTF-8?q?=E6=96=B0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/icons/language.svg | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/icons/language.svg b/src/icons/language.svg index 96673e07..0b670b90 100644 --- a/src/icons/language.svg +++ b/src/icons/language.svg @@ -1,6 +1,6 @@
    Ray Template

    -

    Ray Template

    - +
    -[![All Contributors](https://img.shields.io/badge/all_contributors-1-orange.svg?style=flat-square)](#contributors-) +# Ray Template - +一个基于 vite4.x & ts(x) & pinia & vue3.x 的中后台模板 -## 前言 +
    -> 该项目模板采用 `vue3.x` `vite4.x` `pinia` `tsx` 进行开发。 -> 使用 `naive ui` 作为组件库。 -> 预设了最佳构建体验的配置与常用搬砖工具。意在提供一个简洁、快速上手的模板。 -> 该模板不支持移动端设备。 - -## 感谢 - -> 感谢 [`yun`](https://me.yka.moe/) 对于本人的支持。 - -## 预览地址 - -- [点击预览](https://xiaodaigua-ray.github.io/ray-template/#/) -- [点击预览(加速地址)](https://ray-template.yunkuangao.com/#/) - -## 文档地址 - -- [文档](https://xiaodaigua-ray.github.io/ray-template-doc/) -- [文档(加速地址)](https://ray-template.yunkuangao.com/ray-template-doc/) - -## 更新日志 - -- [日志](https://github.com/XiaoDaiGua-Ray/xiaodaigua-ray.github.io/blob/main/CHANGELOG.md) - -## 常见问题 - -- [常见问题](https://github.com/XiaoDaiGua-Ray/ray-template/blob/main/COMMONPROBLEM.md) - -## 特性 +## ✨ 特性 - **最新技术栈**:使用 Vue3.x/vite4.x 等前端前沿技术开发 - **TypeScript**:应用程序级 JavaScript 的语言 @@ -49,7 +21,21 @@ - **组件**:二次封装了多个常用的组件 - **Axios 请求**:二次封装 axios 库 -## 准备 +## 🪄 预览地址 + +- [点击预览](https://xiaodaigua-ray.github.io/ray-template/#/) +- [点击预览(加速地址)](https://ray-template.yunkuangao.com/#/) + +## 🦾 文档地址 + +- [文档](https://xiaodaigua-ray.github.io/ray-template-doc/) +- [文档(加速地址)](https://ray-template.yunkuangao.com/ray-template-doc/) + +## 🔋 更新日志 + +- [更新日志](https://github.com/XiaoDaiGua-Ray/xiaodaigua-ray.github.io/blob/main/CHANGELOG.md) + +## 🪴 准备 - [node](http://nodejs.org/) 和 [git](https://git-scm.com/) -项目开发环境 - [Vite](https://vitejs.dev/) - 熟悉 vite 特性 @@ -62,16 +48,9 @@ - [Pinia](https://pinia.vuejs.org/zh/introduction.html) - 状态管理器 pinia 使用 - [TSX](https://github.com/vuejs/babel-plugin-jsx/blob/main/packages/babel-plugin-jsx/README-zh_CN.md) - tsx 基本语法 -## 未来 +## 📦 起步 -> 根据个人时间空余情况,会不定时对该模板进行更新和迭代。希望将该工具的功能不断补全(虽然现在已经是足够日常开发和使用),将该模板打造为一个更加健全的中后台模板。如果你有好的想法和建议,可以直接联系我或者直接提 `issues` 即可。 - -## 提示 - -> 项目默认启用严格模式 `eslint`,但是由于 `vite-plugin-eslint` 插件优先级最高,所以如果出现自动导入类型错误提示,请优先解决其他问题。 -> 建议开启 `vscode` 保存自动修复功能。 - -## 项目安装 +### 获取项目 ```sh # github @@ -81,12 +60,12 @@ git clone https://github.com/XiaoDaiGua-Ray/ray-template.git git clone https://gh.yka.moe/https://github.com/XiaoDaiGua-Ray/ray-template.git ``` -## 拉取依赖 +### 拉取依赖 ```sh -# yarn +# pnpm -yarn +pnpm ``` ```sh @@ -95,12 +74,12 @@ yarn npm install ``` -## 启动项目 +### 启动项目 ```sh -# yarn +# pnpm -yarn dev +pnpm dev ``` ```sh @@ -109,12 +88,12 @@ yarn dev npm run dev ``` -## 项目打包 +### 项目打包 ```sh -# yarn +# pnpm -yarn build +pnpm build ``` ```sh @@ -123,12 +102,12 @@ yarn build npm run build ``` -## 预览项目 +### 预览项目 ```sh -# yarn +# pnpm -yarn preview +pnpm preview ``` ```sh @@ -137,12 +116,12 @@ yarn preview npm run preview ``` -## 体积分析 +### 体积分析 ```sh -# yarn +# pnpm -yarn report +pnpm report ``` ```sh @@ -151,48 +130,24 @@ yarn report npm run report ``` -## 浏览器支持 +## 🪴 项目活动 -> 仅支持现代浏览器,不支持 `IE` +![Alt](https://repobeats.axiom.co/api/embed/fab6071297ab281913a42f07a2779b488cfd62b8.svg 'Repobeats analytics image') + +### 贡献者 + +感谢他们的所做的一切贡献 🐝 ! + + + + + +## 浏览器支持 | [ Edge](http://godban.github.io/browsers-support-badges/)
    IE | [ Edge](http://godban.github.io/browsers-support-badges/)
    Edge | [Firefox](http://godban.github.io/browsers-support-badges/)
    Firefox | [Chrome](http://godban.github.io/browsers-support-badges/)
    Chrome | [Safari](http://godban.github.io/browsers-support-badges/)
    Safari | | :--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------: | :----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------: | :---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------: | :-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------: | :-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------: | | not support | last 2 versions | last 2 versions | last 2 versions | last 2 versions | -## 最后,希望大家搬砖愉快 +## 📄 证书 -## 贡献者 - - - - - - - - - - -
    Cloud
    Cloud

    🔧
    - - - - - - -## Contributors ✨ - -Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)): - - - - - - - - - -This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome! - -## License - -[MIT © Ray-2020](./LICENSE) +[MIT License](https://github.com/XiaoDaiGua-Ray/ray-template/blob/main/LICENSE) © 2022-PRESENT [Ray](https://github.com/XiaoDaiGua-Ray/ray-template) diff --git a/package.json b/package.json index 226ba8b9..68a8f3c1 100644 --- a/package.json +++ b/package.json @@ -25,6 +25,7 @@ }, "dependencies": { "@vueuse/core": "^9.1.0", + "awesome-qr": "2.1.5-rc.0", "axios": "^1.2.0", "clipboard": "^2.0.11", "crypto-js": "^4.1.1", @@ -37,7 +38,6 @@ "pinia": "^2.1.4", "pinia-plugin-persistedstate": "^3.1.0", "print-js": "^1.6.0", - "qrcode.vue": "^3.3.4", "sass": "^1.54.3", "screenfull": "^6.0.2", "vue": "^3.3.4", diff --git a/postcss.config.cjs b/postcss.config.cjs index b5f63577..be83e78f 100644 --- a/postcss.config.cjs +++ b/postcss.config.cjs @@ -22,6 +22,8 @@ module.exports = { unitPrecision: 3, /** 指定需要转换成的视窗单位 */ viewportUnit: 'rem', + /** 制定字体转换单位 */ + fontViewportUnit: 'rem', /** 指定不转换为视窗单位的类 */ selectorBlackList: ['.ignore'], /** 小于或等于 1px 不转换为视窗单位 */ diff --git a/src/components/RayQRCode/index.ts b/src/components/RayQRCode/index.ts new file mode 100644 index 00000000..6002f700 --- /dev/null +++ b/src/components/RayQRCode/index.ts @@ -0,0 +1,9 @@ +import RayQRcode from './src/index' + +export default RayQRcode +export type { + QRCodeStatus, + QRCodeLevel, + QRCodeRenderResponse, + QRCodeInst, +} from './src/type' diff --git a/src/components/RayQRCode/src/index.scss b/src/components/RayQRCode/src/index.scss new file mode 100644 index 00000000..e447cce8 --- /dev/null +++ b/src/components/RayQRCode/src/index.scss @@ -0,0 +1,26 @@ +.ray-qrcode { + position: relative; + display: inline-flex; + + & .ray-qrcode__error { + position: absolute; + width: 100%; + top: 50%; + left: 50%; + transform: translate(-50%, -50%); + z-index: 10; + background-color: rgba(0, 0, 0, 0.7); + width: 100%; + height: 100%; + @include flexCenter; + flex-direction: column; + gap: 18px 0; + + & .ray-qrcode__error-content { + text-align: center; + font-size: 18px; + font-weight: 500; + color: #ffffff; + } + } +} diff --git a/src/components/RayQRCode/src/index.tsx b/src/components/RayQRCode/src/index.tsx new file mode 100644 index 00000000..e90f62f7 --- /dev/null +++ b/src/components/RayQRCode/src/index.tsx @@ -0,0 +1,140 @@ +/** + * + * @author Ray + * + * @date 2023-08-29 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +import './index.scss' + +import { NButton, NSpin } from 'naive-ui' +import RayIcon from '@/components/RayIcon/index' + +import props from './props' +import { AwesomeQR } from 'awesome-qr' +import { isValueType, downloadBase64File } from '@use-utils/hook' +import { call } from '@/utils/vue/index' + +import type { QRCodeRenderResponse } from './type' + +const RayQRcode = defineComponent({ + name: 'RayQRcode', + props, + setup(props, ctx) { + const { expose } = ctx + + const qrcodeURL = ref() + const spinOverrides = { + opacitySpinning: '0.1', + } + + const renderQRCode = () => { + new AwesomeQR({ + ...props, + }) + .draw() + .then((res) => { + const { onSuccess } = props + + if (onSuccess) { + call(onSuccess, res) + } + + qrcodeURL.value = res + }) + .catch((err) => { + const { onError } = props + + if (onError) { + call(onError, err) + } + }) + } + + const errorActionClick = () => { + if (ctx.slots.errorAction) { + return + } + + const { onReload } = props + + if (onReload) { + call(onReload) + } + } + + const downloadQRCode = (fileName?: string) => { + if (qrcodeURL.value && isValueType(qrcodeURL.value, 'String')) { + downloadBase64File(qrcodeURL.value, fileName) + } + } + + watchEffect(() => { + if (props.watchText) { + nextTick().then(() => { + renderQRCode() + }) + } + }) + + expose({ + downloadQRCode, + }) + + onMounted(() => { + renderQRCode() + }) + + return { + qrcodeURL, + spinOverrides, + errorActionClick, + } + }, + render() { + return ( +
    + + + + {this.status === 'error' ? ( +
    +
    + {isValueType(this.errorDescription, 'String') + ? this.errorDescription + : () => this.errorDescription} +
    +
    + {this.$slots.errorAction ? ( + this.$slots.errorAction() + ) : ( + <> + + {{ + default: () => this.errorActionDescription, + icon: () => ( + + ), + }} + + + )} +
    +
    + ) : null} +
    + ) + }, +}) + +export default RayQRcode diff --git a/src/components/RayQRCode/src/props.ts b/src/components/RayQRCode/src/props.ts new file mode 100644 index 00000000..434b3d57 --- /dev/null +++ b/src/components/RayQRCode/src/props.ts @@ -0,0 +1,281 @@ +/** + * + * @author Ray + * + * @date 2023-08-29 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +import type { QRCodeStatus, QRCodeLevel } from './type' +import type { PropType, VNode } from 'vue' +import type { MaybeArray } from '@/types/modules/utils' +import type { Options } from 'awesome-qr' + +const props = { + watchText: { + /** + * + * Atuo watch QR code text + * If update text, then re-render QR code + * + * @default true + */ + type: Boolean, + default: true, + }, + status: { + /** + * + * QR code status + * + * @default undefined + */ + type: String as PropType, + }, + errorDescription: { + /** + * + * QR code error description label + * + * @default 二维码已过期 + */ + type: [String, Object] as PropType, + default: '二维码已过期', + }, + errorActionDescription: { + /** + * + * QR code error action description label + * + * @default 重新加载 + */ + type: String, + default: '重新加载', + }, + text: { + /** + * + * Text to be encoded in the QR code + */ + type: String, + required: true, + }, + size: { + /** + * + * Size of the QR code in pixel. + * + * @default 160 + */ + type: Number, + default: 160, + }, + margin: { + /** + * + * Size of margins around the QR code body in pixel. + * + * @default 12 + */ + type: Number, + default: 12, + }, + correctLevel: { + /** + * + * Error correction level of the QR code + * Accepts a value provided by _QRErrorCorrectLevel_ + * + * @default 1 + */ + type: Number as PropType, + default: 1, + validator: (value: unknown) => [0, 1, 2, 3].includes(value as number), + }, + maskPattern: { + /** + * + * Specify the mask pattern to be used in QR code encoding + * Accepts a value provided by _QRMaskPattern_ + */ + type: Number, + }, + version: { + /** + * + * Specify the version to be used in QR code encoding + * Accepts an integer in range [1, 40] + */ + type: Number, + }, + components: { + /** + * + * Options to control components in the QR code. + * + * @default {data:{scale...},...} + */ + type: Object as PropType, + default: () => ({ + data: { + scale: 1, + }, + timing: { + scale: 1, + protectors: false, + }, + alignment: { + scale: 1, + protectors: false, + }, + cornerAlignment: { + scale: 1, + protectors: true, + }, + }), + }, + colorDark: { + /** + * + * Color of the blocks on the QR code + * Accepts a CSS <color> + * + * @default #000000 + */ + type: String, + default: '#000000', + }, + colorLight: { + /** + * + * Color of the blocks on the QR code + * Accepts a CSS <color> + * + * @default #ffffff + */ + type: String, + default: '#ffffff', + }, + autoColor: { + /** + * + * Automatically calculate the _colorLight_ value from the QR code's background + * + * @default true + */ + type: Boolean, + default: true, + }, + backgroundImage: { + /** + * + * Background image to be used in the QR code + * Accepts a `data:` string in web browsers or a Buffer in Node.js + * + * @default undefined + */ + type: String, + }, + backgroundDimming: { + /** + * + * Color of the dimming mask above the background image + * Accepts a CSS <color> + * + * @default rgba(0, 0, 0, 0) + */ + type: String, + default: 'rgba(0, 0, 0, 0)', + }, + gifBackground: { + /** + * + * GIF background image to be used in the QR code + * + * @default undefined + */ + type: ArrayBuffer, + }, + whiteMargin: { + /** + * + * Use a white margin instead of a transparent one which reveals the background of the QR code on margins + * + * @default true + */ + type: Boolean, + default: true, + }, + logoImage: { + /** + * + * Logo image to be displayed at the center of the QR code + * Accepts a `data:` string in web browsers or a Buffer in Node.js + * When set to `undefined` or `null`, the logo is disabled + * + * @default undefined + */ + type: String, + }, + logoScale: { + /** + * + * Ratio of the logo size to the QR code size + * + * @default 0.4 + */ + type: Number, + default: 0.4, + }, + logoMargin: { + /** + * + * Size of margins around the logo image in pixels + * + * @default 6 + */ + type: Number, + default: 6, + }, + logoCornerRadius: { + /** + * Corner radius of the logo image in pixels. + * + * @default 8 + */ + type: Number, + default: 8, + }, + onSuccess: { + /** + * + * When the QR code is successfully generated, this callback is called + */ + type: [Function, Array] as PropType< + MaybeArray<(dataURL: ArrayBuffer | string | undefined) => void> + >, + default: null, + }, + onError: { + /** + * + * When the QR code generation fails, this callback is called + */ + type: [Function, Array] as PropType void>>, + default: null, + }, + onReload: { + /** + * + * When reload button is clicked, this callback is called + * This method will not execute if the errorAction slot is used + */ + type: [Function, Array] as PropType void>>, + default: null, + }, +} + +export default props diff --git a/src/components/RayQRCode/src/type.ts b/src/components/RayQRCode/src/type.ts new file mode 100644 index 00000000..994a0869 --- /dev/null +++ b/src/components/RayQRCode/src/type.ts @@ -0,0 +1,26 @@ +/** + * + * @author Ray + * + * @date 2023-08-29 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +export type QRCodeStatus = 'error' | 'success' | 'loading' + +export type QRCodeLevel = 0 | 1 | 2 | 3 + +export type QRCodeRenderResponse = string | ArrayBuffer | Buffer | undefined + +export type QRCodeInst = { + /** + * + * @param fileName file name + * + * 如果未设置名称,则默认以 时间戳.png 命名 + */ + downloadQRCode: (fileName?: string) => void +} diff --git a/src/router/modules/demo/qrcode.ts b/src/router/modules/demo/qrcode.ts new file mode 100644 index 00000000..27add67f --- /dev/null +++ b/src/router/modules/demo/qrcode.ts @@ -0,0 +1,17 @@ +import { t } from '@/locales/useI18n' +import { LAYOUT } from '@/router/constant/index' + +import type { AppRouteRecordRaw } from '@/router/type' + +const qrcode: AppRouteRecordRaw = { + path: '/qrcode', + name: 'RQRCode', + component: () => import('@/views/demo/qrcode/index'), + meta: { + noLocalTitle: '二维码', + icon: 'other', + order: 3, + }, +} + +export default qrcode diff --git a/src/utils/hook.ts b/src/utils/hook.ts index 1cecf4ef..00d3b11a 100644 --- a/src/utils/hook.ts +++ b/src/utils/hook.ts @@ -33,6 +33,22 @@ export const arrayBufferToBase64Image = (data: ArrayBuffer): string | null => { return base64 } +/** + * + * @param base64 base64 + * @param fileName file name + * + * @remark 下载 base64 文件 + */ +export const downloadBase64File = (base64: string, fileName?: string) => { + const link = document.createElement('a') + + link.href = base64 + link.download = fileName || new Date().getTime() + '.png' + + link.click() +} + /** * * @param value 目标值 diff --git a/src/views/demo/qrcode/index.tsx b/src/views/demo/qrcode/index.tsx new file mode 100644 index 00000000..ce049390 --- /dev/null +++ b/src/views/demo/qrcode/index.tsx @@ -0,0 +1,104 @@ +/** + * + * @author Ray + * + * @date 2023-08-30 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +import { NSpace, NCard, NButton } from 'naive-ui' +import RayQRcode from '@/components/RayQRCode/index' + +import LOGO from '@/assets/images/ray.svg' + +import type { QRCodeStatus, QRCodeInst } from '@/components/RayQRCode/index' + +const RQRCode = defineComponent({ + name: 'RQRCode', + setup() { + const qrcodeText = ref('ray template yes') + const qrcodeStatus = ref() + const rayQRCodeRef = ref() + + return { + qrcodeText, + qrcodeStatus, + rayQRCodeRef, + } + }, + render() { + return ( + + +

    + 基于 awesome-qr 进行封装,支持 LOGO、gif、backgroundImage 等属性。 +

    +

    该组件会自动监听文本内容变化,然后重新渲染(watchText)

    +

    具体使用请参考 props 配置项

    +
    + + + + + + + + + { + window.$message.error('relod props') + }} + /> + + + + + + + { + this.qrcodeStatus = 'loading' + + setTimeout(() => { + this.qrcodeText = 'text updated: ' + new Date().getTime() + this.qrcodeStatus = void 0 + }, 1000) + }} + > + 更新二维码内容 + + { + this.rayQRCodeRef?.downloadQRCode() + }} + > + 下载二维码 + + + + + 当前二维码内容:{this.qrcodeText} + + + +
    + ) + }, +}) + +export default RQRCode diff --git a/src/views/login/components/QRCodeSignin/index.tsx b/src/views/login/components/QRCodeSignin/index.tsx index fa5025ab..d4b042c4 100644 --- a/src/views/login/components/QRCodeSignin/index.tsx +++ b/src/views/login/components/QRCodeSignin/index.tsx @@ -11,7 +11,9 @@ import './index.scss' -import QRCode from 'qrcode.vue' +import RayQRcode from '@/components/RayQRCode/index' + +import LOGO from '@/assets/images/ray.svg' /** * @@ -34,7 +36,7 @@ const QRCodeSignin = defineComponent({ render() { return ( ) }, diff --git a/src/vite-env.d.ts b/src/vite-env.d.ts index bb2e02f1..0f7192c3 100644 --- a/src/vite-env.d.ts +++ b/src/vite-env.d.ts @@ -2,6 +2,7 @@ /// /// /// +/// declare module '*.vue' { import type { DefineComponent } from 'vue' diff --git a/unplugin/.eslintrc-auto-import.json b/unplugin/.eslintrc-auto-import.json index f2bd02bd..6aeb49b9 100644 --- a/unplugin/.eslintrc-auto-import.json +++ b/unplugin/.eslintrc-auto-import.json @@ -8,6 +8,7 @@ "PropType": true, "Ref": true, "VNode": true, + "WritableComputedRef": true, "acceptHMRUpdate": true, "asyncComputed": true, "autoResetRef": true, @@ -121,10 +122,12 @@ "useArrayFilter": true, "useArrayFind": true, "useArrayFindIndex": true, + "useArrayFindLast": true, "useArrayJoin": true, "useArrayMap": true, "useArrayReduce": true, "useArraySome": true, + "useArrayUnique": true, "useAsyncQueue": true, "useAsyncState": true, "useAttrs": true, @@ -136,6 +139,7 @@ "useBrowserLocation": true, "useCached": true, "useClipboard": true, + "useCloned": true, "useColorMode": true, "useConfirmDialog": true, "useCounter": true, @@ -212,11 +216,14 @@ "useParallax": true, "usePermission": true, "usePointer": true, + "usePointerLock": true, "usePointerSwipe": true, "usePreferredColorScheme": true, + "usePreferredContrast": true, "usePreferredDark": true, "usePreferredLanguages": true, "usePreferredReducedMotion": true, + "usePrevious": true, "useRafFn": true, "useRefHistory": true, "useResizeObserver": true, @@ -230,6 +237,7 @@ "useSessionStorage": true, "useShare": true, "useSlots": true, + "useSorted": true, "useSpeechRecognition": true, "useSpeechSynthesis": true, "useStepper": true, diff --git a/unplugin/auto-imports.d.ts b/unplugin/auto-imports.d.ts index faf7b218..5fecb52b 100644 --- a/unplugin/auto-imports.d.ts +++ b/unplugin/auto-imports.d.ts @@ -118,10 +118,12 @@ declare global { const useArrayFilter: typeof import('@vueuse/core')['useArrayFilter'] const useArrayFind: typeof import('@vueuse/core')['useArrayFind'] const useArrayFindIndex: typeof import('@vueuse/core')['useArrayFindIndex'] + const useArrayFindLast: typeof import('@vueuse/core')['useArrayFindLast'] const useArrayJoin: typeof import('@vueuse/core')['useArrayJoin'] const useArrayMap: typeof import('@vueuse/core')['useArrayMap'] const useArrayReduce: typeof import('@vueuse/core')['useArrayReduce'] const useArraySome: typeof import('@vueuse/core')['useArraySome'] + const useArrayUnique: typeof import('@vueuse/core')['useArrayUnique'] const useAsyncQueue: typeof import('@vueuse/core')['useAsyncQueue'] const useAsyncState: typeof import('@vueuse/core')['useAsyncState'] const useAttrs: typeof import('vue')['useAttrs'] @@ -133,6 +135,7 @@ declare global { const useBrowserLocation: typeof import('@vueuse/core')['useBrowserLocation'] const useCached: typeof import('@vueuse/core')['useCached'] const useClipboard: typeof import('@vueuse/core')['useClipboard'] + const useCloned: typeof import('@vueuse/core')['useCloned'] const useColorMode: typeof import('@vueuse/core')['useColorMode'] const useConfirmDialog: typeof import('@vueuse/core')['useConfirmDialog'] const useCounter: typeof import('@vueuse/core')['useCounter'] @@ -209,11 +212,14 @@ declare global { const useParallax: typeof import('@vueuse/core')['useParallax'] const usePermission: typeof import('@vueuse/core')['usePermission'] const usePointer: typeof import('@vueuse/core')['usePointer'] + const usePointerLock: typeof import('@vueuse/core')['usePointerLock'] const usePointerSwipe: typeof import('@vueuse/core')['usePointerSwipe'] const usePreferredColorScheme: typeof import('@vueuse/core')['usePreferredColorScheme'] + const usePreferredContrast: typeof import('@vueuse/core')['usePreferredContrast'] const usePreferredDark: typeof import('@vueuse/core')['usePreferredDark'] const usePreferredLanguages: typeof import('@vueuse/core')['usePreferredLanguages'] const usePreferredReducedMotion: typeof import('@vueuse/core')['usePreferredReducedMotion'] + const usePrevious: typeof import('@vueuse/core')['usePrevious'] const useRafFn: typeof import('@vueuse/core')['useRafFn'] const useRefHistory: typeof import('@vueuse/core')['useRefHistory'] const useResizeObserver: typeof import('@vueuse/core')['useResizeObserver'] @@ -227,6 +233,7 @@ declare global { const useSessionStorage: typeof import('@vueuse/core')['useSessionStorage'] const useShare: typeof import('@vueuse/core')['useShare'] const useSlots: typeof import('vue')['useSlots'] + const useSorted: typeof import('@vueuse/core')['useSorted'] const useSpeechRecognition: typeof import('@vueuse/core')['useSpeechRecognition'] const useSpeechSynthesis: typeof import('@vueuse/core')['useSpeechSynthesis'] const useStepper: typeof import('@vueuse/core')['useStepper'] @@ -284,5 +291,5 @@ declare global { // for type re-export declare global { // @ts-ignore - export type { Component, ComponentPublicInstance, ComputedRef, InjectionKey, PropType, Ref, VNode } from 'vue' + export type { Component, ComponentPublicInstance, ComputedRef, InjectionKey, PropType, Ref, VNode, WritableComputedRef } from 'vue' } diff --git a/vite.config.ts b/vite.config.ts index 543a39fa..ccdbd04e 100644 --- a/vite.config.ts +++ b/vite.config.ts @@ -105,7 +105,7 @@ export default defineConfig(async ({ mode }) => { viteCompression(), viteVueI18nPlugin(), viteSvgLoader({ - defaultImport: 'component', // 默认以 `componetn` 形式导入 `svg` + defaultImport: 'url', // 默认以 `componetn` 形式导入 `svg` }), viteSVGIcon(), viteEslint({ From ccc4d6c710903984b2d8d03ee19994b60ba4bc3b Mon Sep 17 00:00:00 2001 From: XiaoDaiGua-Ray <443547225@qq.com> Date: Thu, 31 Aug 2023 15:30:17 +0800 Subject: [PATCH 09/19] 4.1.9 --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index 68a8f3c1..121cddaf 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "ray-template", "private": false, - "version": "4.1.8", + "version": "4.1.9", "type": "module", "engines": { "node": ">=16.0.0", From 689813f1a066810249838e5a1865aed1f09dd40a Mon Sep 17 00:00:00 2001 From: XiaoDaiGua-Ray <443547225@qq.com> Date: Mon, 4 Sep 2023 15:01:17 +0800 Subject: [PATCH 10/19] =?UTF-8?q?feat:=20v4.1.9=E8=A1=A5=E5=85=85qrcode?= =?UTF-8?q?=E7=BB=84=E4=BB=B6?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- CHANGELOG.md | 4 +- COMMONPROBLEM.md | 21 -------- package.json | 2 +- src/components/RayQRCode/src/index.tsx | 52 ++++++++++++++++++- src/components/RayQRCode/src/props.ts | 20 +++++++ src/layout/components/MenuTag/index.tsx | 2 +- src/office/index.ts | 27 ---------- src/router/modules/demo/office.ts | 43 --------------- src/views/demo/office/index.tsx | 21 -------- .../demo/office/views/document/index.tsx | 47 ----------------- .../demo/office/views/presentation/index.tsx | 24 --------- .../demo/office/views/spreadsheet/index.tsx | 24 --------- src/views/demo/qrcode/index.tsx | 6 +++ vite.config.ts | 1 + 14 files changed, 82 insertions(+), 212 deletions(-) delete mode 100644 COMMONPROBLEM.md delete mode 100644 src/office/index.ts delete mode 100644 src/router/modules/demo/office.ts delete mode 100644 src/views/demo/office/index.tsx delete mode 100644 src/views/demo/office/views/document/index.tsx delete mode 100644 src/views/demo/office/views/presentation/index.tsx delete mode 100644 src/views/demo/office/views/spreadsheet/index.tsx diff --git a/CHANGELOG.md b/CHANGELOG.md index d25d2273..e60912c4 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -5,9 +5,11 @@ ### Feats - 新增 RayQRCode 组件(二维码) - - 基于 awesome-qr 封装,继承其所有特性。并且拓展了 状态、下载、自动更新等属性 + - 基于 awesome-qr 封装,继承其所有特性。并且拓展 状态、下载、自动更新等属性 - 自动卸载于释放内存,仅需要关注 text 内容填充 - 移除 qrcode.vue 依赖 +- 更新 vue-hooks-plus 版本至 v1.8.2 +- 移除 office 功能集成 ### Fixes diff --git a/COMMONPROBLEM.md b/COMMONPROBLEM.md deleted file mode 100644 index fa14ec84..00000000 --- a/COMMONPROBLEM.md +++ /dev/null @@ -1,21 +0,0 @@ -## 常见问题 - -### 路由 - -#### 缓存失效 - -> 如果出现缓存配置不生效的情况可以按照如下方法进行排查 - -- 查看 APP_KEEP_ALIVE setupKeepAlive 属性是否配置为 true -- 查看每个组件的 `name` 是否唯一,[`KeepAlive`](https://cn.vuejs.org/guide/built-ins/keep-alive.html) 组件重度依赖组件 `name` 作为唯一标识。详情可以查看官方文档 -- 查看该页面的路由配置是否正确,比如:`path` 是否按照模板约定方式进行配置 - -#### 自动导入失败 - -> 模板采用自动导入路由模块方式。如果发现路由导入有误、或者导入报错,请查看文件命名是否有误。 - -### 国际化 - -#### 国际化切换错误、警告 - -> 模板二次封装 [`useI18n`](https://github.com/XiaoDaiGua-Ray/ray-template/blob/main/src/locales/useI18n.ts) 方法,首选该方法作为国际化语言切换方法。 diff --git a/package.json b/package.json index 121cddaf..20f7cd91 100644 --- a/package.json +++ b/package.json @@ -41,7 +41,7 @@ "sass": "^1.54.3", "screenfull": "^6.0.2", "vue": "^3.3.4", - "vue-hooks-plus": "1.8.1", + "vue-hooks-plus": "1.8.2", "vue-i18n": "^9.2.2", "vue-router": "^4.2.4", "vuedraggable": "^4.1.0", diff --git a/src/components/RayQRCode/src/index.tsx b/src/components/RayQRCode/src/index.tsx index e90f62f7..a7a33832 100644 --- a/src/components/RayQRCode/src/index.tsx +++ b/src/components/RayQRCode/src/index.tsx @@ -21,6 +21,35 @@ import { call } from '@/utils/vue/index' import type { QRCodeRenderResponse } from './type' +const readGIFAsArrayBuffer = ( + url: string, +): Promise => { + return new Promise((resolve, reject) => { + const xhr = new XMLHttpRequest() + + xhr.responseType = 'blob' + + xhr.onload = () => { + const reader = new FileReader() + + reader.onloadend = () => { + resolve(reader.result) + } + reader.onerror = (e) => { + reject(e) + } + reader.onabort = (e) => { + reject(e) + } + + reader.readAsArrayBuffer(xhr.response) + } + + xhr.open('GET', url) + xhr.send() + }) +} + const RayQRcode = defineComponent({ name: 'RayQRcode', props, @@ -31,10 +60,28 @@ const RayQRcode = defineComponent({ const spinOverrides = { opacitySpinning: '0.1', } + let gifBuffer: string | ArrayBuffer | null + + const getGIFImageByURL = async () => { + const { gifBackgroundURL } = props + + if (!gifBackgroundURL) { + return + } + + try { + gifBuffer = await readGIFAsArrayBuffer(gifBackgroundURL) + } catch (e) { + console.error(e) + } + } const renderQRCode = () => { + const { gifBackgroundURL, gifBackground, ...ops } = props + new AwesomeQR({ - ...props, + ...ops, + gifBackground: (gifBuffer as ArrayBuffer) ?? undefined, }) .draw() .then((res) => { @@ -85,7 +132,8 @@ const RayQRcode = defineComponent({ downloadQRCode, }) - onMounted(() => { + onMounted(async () => { + await getGIFImageByURL() renderQRCode() }) diff --git a/src/components/RayQRCode/src/props.ts b/src/components/RayQRCode/src/props.ts index 434b3d57..84fc7741 100644 --- a/src/components/RayQRCode/src/props.ts +++ b/src/components/RayQRCode/src/props.ts @@ -190,6 +190,15 @@ const props = { type: String, default: 'rgba(0, 0, 0, 0)', }, + gifBackgroundURL: { + /** + * + * GIF background image to be used in the QR code + * + * @default undefined + */ + type: String, + }, gifBackground: { /** * @@ -249,6 +258,17 @@ const props = { type: Number, default: 8, }, + dotScale: { + /** + * + * Ratio of the real size to the full size of the blocks. + * This can be helpful when you want to make more parts of the background visible. + * + * @default 1 + */ + type: Number, + default: 1, + }, onSuccess: { /** * diff --git a/src/layout/components/MenuTag/index.tsx b/src/layout/components/MenuTag/index.tsx index a62d9526..b052d724 100644 --- a/src/layout/components/MenuTag/index.tsx +++ b/src/layout/components/MenuTag/index.tsx @@ -388,7 +388,7 @@ const MenuTag = defineComponent({ if (tags?.length) { const [menuTag] = tags - nextTick(() => { + nextTick().then(() => { menuTag.scrollIntoView?.() }) } diff --git a/src/office/index.ts b/src/office/index.ts deleted file mode 100644 index 2ddf5541..00000000 --- a/src/office/index.ts +++ /dev/null @@ -1,27 +0,0 @@ -/** - * - * @author Ray - * - * @date 2023-03-22 - * - * @workspace ray-template - * - * @remark 今天也是元气满满撸代码的一天 - */ - -/** - * - * onlyoffice - * - * 该功能暂未实现, 后续应该会补上 - * 由于该方法需要后端进行相关配合, 因为目前还在考虑是否接上私有 onlyoffice 服务器, 所以该功能暂未实现 - * 望多多理解, 理解万岁 - */ - -import { getAppEnvironment } from '@use-utils/hook' -import request from '@/axios/instance' - -export const getOfficeDocumentApi = async (uuid: string) => { - const { VITE_APP_OFFICE_PROXY_URL } = getAppEnvironment() - const { get } = request -} diff --git a/src/router/modules/demo/office.ts b/src/router/modules/demo/office.ts deleted file mode 100644 index cec96641..00000000 --- a/src/router/modules/demo/office.ts +++ /dev/null @@ -1,43 +0,0 @@ -import { t } from '@/locales/useI18n' -import { LAYOUT } from '@/router/constant/index' - -import type { AppRouteRecordRaw } from '@/router/type' - -const office: AppRouteRecordRaw = { - path: '/office', - name: 'ROffice', - component: () => import('@/views/demo/office/index'), - meta: { - i18nKey: t('menu.Office'), - icon: 'office', - hidden: true, - }, - children: [ - { - path: 'document', - name: 'Document', - component: () => import('@/views/demo/office/views/document/index'), - meta: { - i18nKey: 'Office_Document', - }, - }, - { - path: 'presentation', - name: 'Presentation', - component: () => import('@/views/demo/office/views/presentation/index'), - meta: { - i18nKey: 'Office_Presentation', - }, - }, - { - path: 'spreadsheet', - name: 'Spreadsheet', - component: () => import('@/views/demo/office/views/spreadsheet/index'), - meta: { - i18nKey: 'Office_Spreadsheet', - }, - }, - ], -} - -export default office diff --git a/src/views/demo/office/index.tsx b/src/views/demo/office/index.tsx deleted file mode 100644 index c7df6024..00000000 --- a/src/views/demo/office/index.tsx +++ /dev/null @@ -1,21 +0,0 @@ -/** - * - * @author Ray - * - * @date 2023-03-22 - * - * @workspace ray-template - * - * @remark 今天也是元气满满撸代码的一天 - */ - -import { RouterView } from 'vue-router' - -const Office = defineComponent({ - name: 'ROffice', - render() { - return - }, -}) - -export default Office diff --git a/src/views/demo/office/views/document/index.tsx b/src/views/demo/office/views/document/index.tsx deleted file mode 100644 index f2907266..00000000 --- a/src/views/demo/office/views/document/index.tsx +++ /dev/null @@ -1,47 +0,0 @@ -/** - * - * @author Ray - * - * @date 2023-03-22 - * - * @workspace ray-template - * - * @remark 今天也是元气满满撸代码的一天 - */ - -import { uuid } from '@/utils/hook' - -import type { PropType } from 'vue' - -const Document = defineComponent({ - name: 'RDocument', - setup() { - const editorUUID = uuid(16) - const state = reactive({}) - const config = { - document: { - fileType: 'docx', - key: editorUUID, - title: 'Example Document Title.docx', - url: 'https://example.com/url-to-example-document.docx', - }, - documentType: 'word', - authorization: 'a2122252', - token: 'a2122252', - Authorization: 'a2122252', - editorConfig: { - lang: 'zh-cn', - }, - } - - return { - ...toRefs(state), - editorUUID, - } - }, - render() { - return
    - }, -}) - -export default Document diff --git a/src/views/demo/office/views/presentation/index.tsx b/src/views/demo/office/views/presentation/index.tsx deleted file mode 100644 index 797a39dd..00000000 --- a/src/views/demo/office/views/presentation/index.tsx +++ /dev/null @@ -1,24 +0,0 @@ -/** - * - * @author Ray - * - * @date 2023-03-22 - * - * @workspace ray-template - * - * @remark 今天也是元气满满撸代码的一天 - */ - -import type { PropType } from 'vue' - -const Presentation = defineComponent({ - name: 'RPresentation', - setup() { - return {} - }, - render() { - return
    - }, -}) - -export default Presentation diff --git a/src/views/demo/office/views/spreadsheet/index.tsx b/src/views/demo/office/views/spreadsheet/index.tsx deleted file mode 100644 index 14bf9951..00000000 --- a/src/views/demo/office/views/spreadsheet/index.tsx +++ /dev/null @@ -1,24 +0,0 @@ -/** - * - * @author Ray - * - * @date 2023-03-22 - * - * @workspace ray-template - * - * @remark 今天也是元气满满撸代码的一天 - */ - -import type { PropType } from 'vue' - -const Spreadsheet = defineComponent({ - name: 'RSpreadsheet', - setup() { - return {} - }, - render() { - return
    - }, -}) - -export default Spreadsheet diff --git a/src/views/demo/qrcode/index.tsx b/src/views/demo/qrcode/index.tsx index ce049390..b91d296f 100644 --- a/src/views/demo/qrcode/index.tsx +++ b/src/views/demo/qrcode/index.tsx @@ -43,6 +43,12 @@ const RQRCode = defineComponent({ +
    diff --git a/vite.config.ts b/vite.config.ts index ccdbd04e..02c8cad3 100644 --- a/vite.config.ts +++ b/vite.config.ts @@ -115,6 +115,7 @@ export default defineConfig(async ({ mode }) => { fix: true, exclude: ['dist/**', '**/node_modules/**', 'vite-env.d.ts', '*.md'], include: ['src/**/*.{vue,js,jsx,ts,tsx}'], + cache: true, }), vitePluginImp({ libList: [ From 6acc80f18a943864f44f7684a79b387e0d674766 Mon Sep 17 00:00:00 2001 From: XiaoDaiGua-Ray <443547225@qq.com> Date: Fri, 8 Sep 2023 15:13:36 +0800 Subject: [PATCH 11/19] =?UTF-8?q?v4.1.9=E9=83=A8=E5=88=86=E7=BB=86?= =?UTF-8?q?=E8=8A=82=E8=A1=A5=E5=85=85?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- CHANGELOG.md | 2 + README.md | 50 +++---------------- cfg.ts | 4 +- src/app-components/app/AppAvatar/index.tsx | 2 +- .../provider/AppNaiveGlobalProvider/index.tsx | 9 ++-- .../index.tsx | 10 ++-- .../provider/AppStyleProvider/index.tsx | 4 +- src/{appConfig => app-config}/appConfig.ts | 0 src/{appConfig => app-config}/designConfig.ts | 0 src/{appConfig => app-config}/localConfig.ts | 0 src/{appConfig => app-config}/regexConfig.ts | 0 .../requestConfig.ts | 0 src/{appConfig => app-config}/routerConfig.ts | 0 src/axios/inject/request/provide.ts | 2 +- src/axios/instance.ts | 2 +- src/components/RayChart/index.tsx | 2 +- src/components/RayQRCode/src/index.tsx | 15 +++--- src/components/RayQRCode/src/props.ts | 9 ++++ src/components/RayQRCode/src/type.ts | 2 + .../RayTransitionComponent/index.vue | 2 +- src/dayjs/index.ts | 2 +- src/directives/modules/copy/index.ts | 35 +++++++------ src/directives/modules/copy/type.ts | 3 ++ src/layout/components/Menu/index.tsx | 2 +- src/layout/components/MenuTag/index.tsx | 2 +- .../components/SettingDrawer/index.tsx | 2 +- src/layout/components/SiderBar/index.tsx | 2 +- src/layout/default/ContentWrapper/index.tsx | 4 +- src/layout/index.tsx | 2 +- src/locales/helper.ts | 4 +- src/locales/index.ts | 2 +- src/locales/lang/zh-CN/views/login/index.json | 2 +- src/router/helper/helper.ts | 2 +- src/router/helper/permission.ts | 4 +- src/router/helper/routerCopilot.ts | 4 +- src/router/routes.ts | 2 +- src/store/index.ts | 7 ++- src/store/modules/keep-alive/index.ts | 2 +- src/store/modules/menu/helper.ts | 2 +- src/store/modules/setting/index.ts | 2 +- src/utils/element.ts | 2 +- src/views/login/components/Signin/index.tsx | 2 +- src/views/login/index.tsx | 2 +- 43 files changed, 98 insertions(+), 110 deletions(-) rename src/app-components/provider/{AppRequestCanceler => AppRequestCancelerProvider}/index.tsx (77%) rename src/{appConfig => app-config}/appConfig.ts (100%) rename src/{appConfig => app-config}/designConfig.ts (100%) rename src/{appConfig => app-config}/localConfig.ts (100%) rename src/{appConfig => app-config}/regexConfig.ts (100%) rename src/{appConfig => app-config}/requestConfig.ts (100%) rename src/{appConfig => app-config}/routerConfig.ts (100%) diff --git a/CHANGELOG.md b/CHANGELOG.md index e60912c4..dbd18e1c 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -10,6 +10,8 @@ - 移除 qrcode.vue 依赖 - 更新 vue-hooks-plus 版本至 v1.8.2 - 移除 office 功能集成 +- 统一包命名方式 +- 更改 v-copy 实现细节(使用方式不变) ### Fixes diff --git a/README.md b/README.md index 144c050c..ad22f5e9 100644 --- a/README.md +++ b/README.md @@ -12,14 +12,16 @@ ## ✨ 特性 -- **最新技术栈**:使用 Vue3.x/vite4.x 等前端前沿技术开发 +- **最新技术栈**:使用 vue3.x/vite4.x/pinia 等前端前沿技术开发 - **TypeScript**:应用程序级 JavaScript 的语言 - **主题**:可配置的主题 - **国际化**:内置完善的国际化方案 - **Mock 数据**:内置 Mock 数据方案 - **权限**:内置完善的动态路由权限生成方案 - **组件**:二次封装了多个常用的组件 -- **Axios 请求**:二次封装 axios 库 +- **Axios 请求**:二次封装 axios 库,支持:取消、防抖、自动重复取消等功能 +- **缓存**:任意深度页面缓存 +- **SVG**:内置 svg icon 解决方案 ## 🪄 预览地址 @@ -37,7 +39,7 @@ ## 🪴 准备 -- [node](http://nodejs.org/) 和 [git](https://git-scm.com/) -项目开发环境 +- [Node](http://nodejs.org/) 和 [git](https://git-scm.com/) - 项目开发环境 - [Vite](https://vitejs.dev/) - 熟悉 vite 特性 - [Vue3](https://v3.vuejs.org/) - 熟悉 Vue 基础语法 - [TypeScript](https://www.typescriptlang.org/) - 熟悉 TypeScript 基本语法 @@ -63,73 +65,33 @@ git clone https://gh.yka.moe/https://github.com/XiaoDaiGua-Ray/ray-template.git ### 拉取依赖 ```sh -# pnpm - pnpm ``` -```sh -# npm - -npm install -``` - ### 启动项目 ```sh -# pnpm - pnpm dev ``` -```sh -# npm - -npm run dev -``` - ### 项目打包 ```sh -# pnpm - pnpm build ``` -```sh -# npm - -npm run build -``` - ### 预览项目 ```sh -# pnpm - pnpm preview ``` -```sh -# npm - -npm run preview -``` - ### 体积分析 ```sh -# pnpm - pnpm report ``` -```sh -# npm - -npm run report -``` - ## 🪴 项目活动 ![Alt](https://repobeats.axiom.co/api/embed/fab6071297ab281913a42f07a2779b488cfd62b8.svg 'Repobeats analytics image') @@ -150,4 +112,4 @@ npm run report ## 📄 证书 -[MIT License](https://github.com/XiaoDaiGua-Ray/ray-template/blob/main/LICENSE) © 2022-PRESENT [Ray](https://github.com/XiaoDaiGua-Ray/ray-template) +[MIT License](https://github.com/XiaoDaiGua-Ray/ray-template/blob/main/LICENSE) © 2022-PRESENT [Ray](https://github.com/XiaoDaiGua-Ray) diff --git a/cfg.ts b/cfg.ts index 9ac1939b..9d803ba9 100644 --- a/cfg.ts +++ b/cfg.ts @@ -43,8 +43,8 @@ import { buildOptions, mixinCSSPlugin, } from './vite-plugin/index' -import { APP_THEME } from './src/appConfig/designConfig' -import { PRE_LOADING_CONFIG, SIDE_BAR_LOGO } from './src/appConfig/appConfig' +import { APP_THEME } from './src/app-config/designConfig' +import { PRE_LOADING_CONFIG, SIDE_BAR_LOGO } from './src/app-config/appConfig' import type { AppConfigExport } from '@/types/modules/cfg' diff --git a/src/app-components/app/AppAvatar/index.tsx b/src/app-components/app/AppAvatar/index.tsx index 777597a4..4e561648 100644 --- a/src/app-components/app/AppAvatar/index.tsx +++ b/src/app-components/app/AppAvatar/index.tsx @@ -22,7 +22,7 @@ import './index.scss' import { NAvatar, NSpace } from 'naive-ui' import { avatarProps, spaceProps } from 'naive-ui' -import { APP_CATCH_KEY } from '@/appConfig/appConfig' +import { APP_CATCH_KEY } from '@/app-config/appConfig' import { getStorage } from '@/utils/cache' import type { PropType } from 'vue' diff --git a/src/app-components/provider/AppNaiveGlobalProvider/index.tsx b/src/app-components/provider/AppNaiveGlobalProvider/index.tsx index cdbc4602..fe12f1d0 100644 --- a/src/app-components/provider/AppNaiveGlobalProvider/index.tsx +++ b/src/app-components/provider/AppNaiveGlobalProvider/index.tsx @@ -31,9 +31,9 @@ import { import { useSetting } from '@/store' import { naiveLocales } from '@/locales/helper' -const GlobalProvider = defineComponent({ +export default defineComponent({ name: 'GlobalProvider', - setup() { + setup(_, { expose }) { const settingStore = useSetting() const modelPrimaryColorOverride = computed( @@ -54,6 +54,7 @@ const GlobalProvider = defineComponent({ configProviderProps: computed(() => ({ theme: modelThemeValue.value, })), + notificationProviderProps: {}, }, ) @@ -62,6 +63,8 @@ const GlobalProvider = defineComponent({ window.$loadingBar = loadingBar // 注入 `loadingBar` window.$notification = notification // 注入 `notification` + expose() + return { modelPrimaryColorOverride, modelThemeValue, @@ -90,5 +93,3 @@ const GlobalProvider = defineComponent({ ) }, }) - -export default GlobalProvider diff --git a/src/app-components/provider/AppRequestCanceler/index.tsx b/src/app-components/provider/AppRequestCancelerProvider/index.tsx similarity index 77% rename from src/app-components/provider/AppRequestCanceler/index.tsx rename to src/app-components/provider/AppRequestCancelerProvider/index.tsx index 44f177d9..6aa15f36 100644 --- a/src/app-components/provider/AppRequestCanceler/index.tsx +++ b/src/app-components/provider/AppRequestCancelerProvider/index.tsx @@ -19,12 +19,14 @@ import { axiosCanceler } from '@/axios/helper/interceptor' -const AppRequestCanceler = defineComponent({ - name: 'AppRequestCanceler', - setup() { +const AppRequestCancelerProvider = defineComponent({ + name: 'AppRequestCancelerProvider', + setup(_, { expose }) { onBeforeRouteUpdate(() => { axiosCanceler.cancelAllRequest() }) + + expose() }, render() { return ( @@ -37,4 +39,4 @@ const AppRequestCanceler = defineComponent({ }, }) -export default AppRequestCanceler +export default AppRequestCancelerProvider diff --git a/src/app-components/provider/AppStyleProvider/index.tsx b/src/app-components/provider/AppStyleProvider/index.tsx index 7cd3c14d..d60a4edb 100644 --- a/src/app-components/provider/AppStyleProvider/index.tsx +++ b/src/app-components/provider/AppStyleProvider/index.tsx @@ -20,7 +20,7 @@ import type { SettingState } from '@/store/modules/setting/type' const AppStyleProvider = defineComponent({ name: 'AppStyleProvider', - setup() { + setup(_, { expose }) { const settingStore = useSetting() const { themeValue } = storeToRefs(settingStore) @@ -97,6 +97,8 @@ const AppStyleProvider = defineComponent({ immediate: true, }, ) + + expose() }, render() { return
    diff --git a/src/appConfig/appConfig.ts b/src/app-config/appConfig.ts similarity index 100% rename from src/appConfig/appConfig.ts rename to src/app-config/appConfig.ts diff --git a/src/appConfig/designConfig.ts b/src/app-config/designConfig.ts similarity index 100% rename from src/appConfig/designConfig.ts rename to src/app-config/designConfig.ts diff --git a/src/appConfig/localConfig.ts b/src/app-config/localConfig.ts similarity index 100% rename from src/appConfig/localConfig.ts rename to src/app-config/localConfig.ts diff --git a/src/appConfig/regexConfig.ts b/src/app-config/regexConfig.ts similarity index 100% rename from src/appConfig/regexConfig.ts rename to src/app-config/regexConfig.ts diff --git a/src/appConfig/requestConfig.ts b/src/app-config/requestConfig.ts similarity index 100% rename from src/appConfig/requestConfig.ts rename to src/app-config/requestConfig.ts diff --git a/src/appConfig/routerConfig.ts b/src/app-config/routerConfig.ts similarity index 100% rename from src/appConfig/routerConfig.ts rename to src/app-config/routerConfig.ts diff --git a/src/axios/inject/request/provide.ts b/src/axios/inject/request/provide.ts index f2becd29..1b06c3c9 100644 --- a/src/axios/inject/request/provide.ts +++ b/src/axios/inject/request/provide.ts @@ -20,7 +20,7 @@ import { useAxiosInterceptor, axiosCanceler } from '@/axios/helper/interceptor' import { appendRequestHeaders } from '@/axios/helper/axiosCopilot' -import { APP_CATCH_KEY } from '@/appConfig/appConfig' +import { APP_CATCH_KEY } from '@/app-config/appConfig' import { getStorage } from '@/utils/cache' import type { diff --git a/src/axios/instance.ts b/src/axios/instance.ts index 5f717cf1..e968087a 100644 --- a/src/axios/instance.ts +++ b/src/axios/instance.ts @@ -17,7 +17,7 @@ */ import axios from 'axios' -import { AXIOS_CONFIG } from '@/appConfig/requestConfig' +import { AXIOS_CONFIG } from '@/app-config/requestConfig' import { useAxiosInterceptor, axiosCanceler } from '@/axios/helper/interceptor' import { setupResponseInterceptor, diff --git a/src/components/RayChart/index.tsx b/src/components/RayChart/index.tsx index bc6838ab..2fd661b4 100644 --- a/src/components/RayChart/index.tsx +++ b/src/components/RayChart/index.tsx @@ -46,7 +46,7 @@ import { cloneDeep, throttle } from 'lodash-es' import { on, off, completeSize } from '@/utils/element' import { call } from '@/utils/vue/index' import { setupChartTheme, loadingOptions } from './helper' -import { APP_THEME } from '@/appConfig/designConfig' +import { APP_THEME } from '@/app-config/designConfig' import type { PropType } from 'vue' import type { AnyFC, MaybeArray } from '@/types/modules/utils' diff --git a/src/components/RayQRCode/src/index.tsx b/src/components/RayQRCode/src/index.tsx index a7a33832..f66085c4 100644 --- a/src/components/RayQRCode/src/index.tsx +++ b/src/components/RayQRCode/src/index.tsx @@ -19,11 +19,9 @@ import { AwesomeQR } from 'awesome-qr' import { isValueType, downloadBase64File } from '@use-utils/hook' import { call } from '@/utils/vue/index' -import type { QRCodeRenderResponse } from './type' +import type { QRCodeRenderResponse, GIFBuffer } from './type' -const readGIFAsArrayBuffer = ( - url: string, -): Promise => { +const readGIFAsArrayBuffer = (url: string): Promise => { return new Promise((resolve, reject) => { const xhr = new XMLHttpRequest() @@ -50,7 +48,7 @@ const readGIFAsArrayBuffer = ( }) } -const RayQRcode = defineComponent({ +export default defineComponent({ name: 'RayQRcode', props, setup(props, ctx) { @@ -60,7 +58,7 @@ const RayQRcode = defineComponent({ const spinOverrides = { opacitySpinning: '0.1', } - let gifBuffer: string | ArrayBuffer | null + let gifBuffer: GIFBuffer const getGIFImageByURL = async () => { const { gifBackgroundURL } = props @@ -77,7 +75,7 @@ const RayQRcode = defineComponent({ } const renderQRCode = () => { - const { gifBackgroundURL, gifBackground, ...ops } = props + const { gifBackground, ...ops } = props new AwesomeQR({ ...ops, @@ -149,6 +147,7 @@ const RayQRcode = defineComponent({ @@ -184,5 +183,3 @@ const RayQRcode = defineComponent({ ) }, }) - -export default RayQRcode diff --git a/src/components/RayQRCode/src/props.ts b/src/components/RayQRCode/src/props.ts index 84fc7741..7fc97324 100644 --- a/src/components/RayQRCode/src/props.ts +++ b/src/components/RayQRCode/src/props.ts @@ -15,6 +15,15 @@ import type { MaybeArray } from '@/types/modules/utils' import type { Options } from 'awesome-qr' const props = { + loadingDescription: { + /** + * + * Loading status description label + * + * @default undefined + */ + type: String, + }, watchText: { /** * diff --git a/src/components/RayQRCode/src/type.ts b/src/components/RayQRCode/src/type.ts index 994a0869..6cffb552 100644 --- a/src/components/RayQRCode/src/type.ts +++ b/src/components/RayQRCode/src/type.ts @@ -24,3 +24,5 @@ export type QRCodeInst = { */ downloadQRCode: (fileName?: string) => void } + +export type GIFBuffer = string | ArrayBuffer | null diff --git a/src/components/RayTransitionComponent/index.vue b/src/components/RayTransitionComponent/index.vue index 7970c719..93a7edc4 100644 --- a/src/components/RayTransitionComponent/index.vue +++ b/src/components/RayTransitionComponent/index.vue @@ -24,7 +24,7 @@