This commit is contained in:
XiaoDaiGua-Ray 2023-10-21 14:37:33 +08:00
parent 1f2ae05ca4
commit e81bb3f9a5
22 changed files with 369 additions and 287 deletions

View File

@ -1,5 +1,20 @@
# CHANGE LOG # CHANGE LOG
## 4.2.4
### Feats
- 优化 utils/element 包下的方法
- 更新 vue-hooks-plus 版本至 1.8.5
- 移除不必要的 cdn
- 优化了类型包的一些基础类型,剔除了一些无意义的类型
- 提取 RChart props 单独维护
- RTable 组件新增 onContextmenu props 属性。用于在启用右键菜单时被组件强行代理右键点击事件的回调
### Fixes
- 修复 RChart 组件不能被正常取消 watchOptions 问题
## 4.2.3 ## 4.2.3
### Fixes ### Fixes

View File

@ -1,7 +1,7 @@
{ {
"name": "ray-template", "name": "ray-template",
"private": false, "private": false,
"version": "4.2.3", "version": "4.2.4",
"type": "module", "type": "module",
"engines": { "engines": {
"node": ">=16.0.0", "node": ">=16.0.0",
@ -39,7 +39,7 @@
"pinia-plugin-persistedstate": "^3.1.0", "pinia-plugin-persistedstate": "^3.1.0",
"print-js": "^1.6.0", "print-js": "^1.6.0",
"vue": "^3.3.4", "vue": "^3.3.4",
"vue-hooks-plus": "1.8.2", "vue-hooks-plus": "1.8.5",
"vue-i18n": "^9.2.2", "vue-i18n": "^9.2.2",
"vue-router": "^4.2.4", "vue-router": "^4.2.4",
"vuedraggable": "^4.1.0", "vuedraggable": "^4.1.0",

View File

@ -77,6 +77,11 @@ const LockScreen = defineComponent({
clearable clearable
minlength={6} minlength={6}
maxlength={12} maxlength={12}
onKeydown={(e: KeyboardEvent) => {
if (e.code === 'Enter') {
this.lockScreen()
}
}}
/> />
</NFormItem> </NFormItem>
<NButton type="primary" onClick={this.lockScreen.bind(this)}> <NButton type="primary" onClick={this.lockScreen.bind(this)}>

View File

@ -115,6 +115,11 @@ const UnlockScreen = defineComponent({
clearable clearable
minlength={6} minlength={6}
maxlength={12} maxlength={12}
onKeydown={(e: KeyboardEvent) => {
if (e.code === 'Enter') {
this.unlockScreen()
}
}}
/> />
</NFormItem> </NFormItem>
<NSpace justify="space-between"> <NSpace justify="space-between">

View File

@ -38,188 +38,34 @@ import {
import { LabelLayout, UniversalTransition } from 'echarts/features' // 标签自动布局, 全局过渡动画等特性 import { LabelLayout, UniversalTransition } from 'echarts/features' // 标签自动布局, 全局过渡动画等特性
import { CanvasRenderer } from 'echarts/renderers' // `echarts` 渲染器 import { CanvasRenderer } from 'echarts/renderers' // `echarts` 渲染器
import props from './props'
import { useSetting } from '@/store' import { useSetting } from '@/store'
import { cloneDeep, throttle } from 'lodash-es' import { cloneDeep, throttle } from 'lodash-es'
import { on, off, completeSize } from '@/utils/element' import { on, off, completeSize } from '@/utils/element'
import { call } from '@/utils/vue/index' import { call } from '@/utils/vue/index'
import { setupChartTheme, loadingOptions } from './helper' import { setupChartTheme } from './helper'
import { APP_THEME } from '@/app-config/designConfig' import { APP_THEME } from '@/app-config/designConfig'
import { useResizeObserver } from '@vueuse/core' import { useResizeObserver } from '@vueuse/core'
import type { PropType, WatchStopHandle } from 'vue' import type { WatchStopHandle } from 'vue'
import type { AnyFC, MaybeArray } from '@/types/modules/utils' import type { AnyFC } from '@/types/modules/utils'
import type { DebouncedFunc } from 'lodash-es' import type { DebouncedFunc } from 'lodash-es'
import type { import type { ChartTheme } from '@/components/RChart/type'
LoadingOptions, import type { UseResizeObserverReturn } from '@vueuse/core'
AutoResize,
ChartTheme,
} from '@/components/RChart/type'
import type {
UseResizeObserverReturn,
MaybeComputedElementRef,
MaybeElement,
} from '@vueuse/core'
import type { ECharts, EChartsCoreOption, SetOptionOpts } from 'echarts/core' import type { ECharts, EChartsCoreOption, SetOptionOpts } from 'echarts/core'
export type EChartsExtensionInstallRegisters = typeof CanvasRenderer export type { RayChartInst, EChartsExtensionInstallRegisters } from './type'
export type { RayChartInst } from './type'
const defaultChartOptions = {
notMerge: false,
lazyUpdate: true,
silent: false,
replaceMerge: [],
}
export default defineComponent({ export default defineComponent({
name: 'RChart', name: 'RChart',
props: { props,
width: {
/**
*
* chart
*
* , 200px
*/
type: String,
default: '100%',
},
height: {
/**
*
* chart
*
* , 200px
*/
type: String,
default: '100%',
},
autoResize: {
/**
*
* `chart`
*
* , ,
*
*/
type: [Boolean, Object] as PropType<AutoResize>,
default: true,
},
canvasRender: {
/**
*
* `chart` , 使 `canvas`
*
* , `SVGRenderer`
*/
type: Boolean,
default: true,
},
showAria: {
/**
*
* `chart` 访
*
* `options` `aria`
*/
type: Boolean,
default: false,
},
options: {
type: Object as PropType<echarts.EChartsCoreOption>,
default: () => ({}),
},
onSuccess: {
/**
*
* chart
*
*
*
* () => ECharts
*/
type: [Function, Array] as PropType<MaybeArray<(e: ECharts) => void>>,
default: null,
},
onError: {
/**
*
*
*
* () => void
*/
type: [Function, Array] as PropType<MaybeArray<() => void>>,
default: null,
},
theme: {
type: [String, Object] as PropType<ChartTheme>,
default: '',
},
autoChangeTheme: {
/**
*
*
* , `theme`
*
* 注意: 这个属性重度依赖此模板
*/
type: Boolean,
default: true,
},
use: {
/**
*
* `echarts`
*
*/
type: Array as PropType<EChartsExtensionInstallRegisters[]>,
default: () => [],
},
watchOptions: {
/** 主动监听 options 变化 */
type: Boolean,
default: true,
},
loading: {
/** 加载动画 */
type: Boolean,
default: false,
},
loadingOptions: {
/** 配置加载动画样式 */
type: Object as PropType<LoadingOptions>,
default: () => loadingOptions(),
},
observer: {
/**
*
*
* autoResize
*
*/
type: Object as PropType<MaybeComputedElementRef<MaybeElement>>,
default: null,
},
throttleWait: {
/** 节流等待时间 */
type: Number,
default: 500,
},
animation: {
/** 是否强制启用渲染动画 */
type: Boolean,
default: true,
},
setChartOptions: {
/**
*
* options setOptions
*
*
* notMerge: false,
* lazyUpdate: true,
* silent: false,
* replaceMerge: [],
*
*
*/
type: Object as PropType<SetOptionOpts>,
default: () => ({}),
},
},
setup(props, { expose }) { setup(props, { expose }) {
const settingStore = useSetting() const settingStore = useSetting()
const { themeValue } = storeToRefs(settingStore) const { themeValue } = storeToRefs(settingStore)
@ -229,7 +75,7 @@ export default defineComponent({
let resizeThrottleReturn: DebouncedFunc<AnyFC> | null // resize 防抖方法实例 let resizeThrottleReturn: DebouncedFunc<AnyFC> | null // resize 防抖方法实例
let resizeOvserverReturn: UseResizeObserverReturn | null let resizeOvserverReturn: UseResizeObserverReturn | null
const { echartTheme } = APP_THEME const { echartTheme } = APP_THEME
let watchOptionsReturn: WatchStopHandle | null let watchCallback: WatchStopHandle | null
const cssVarsRef = computed(() => { const cssVarsRef = computed(() => {
const cssVars = { const cssVars = {
@ -266,7 +112,7 @@ export default defineComponent({
CandlestickChart, CandlestickChart,
ScatterChart, ScatterChart,
PictorialBarChart, PictorialBarChart,
]) // 注册类型 ]) // 注册 chart series type
echarts.use([LabelLayout, UniversalTransition]) // 注册布局, 过度效果 echarts.use([LabelLayout, UniversalTransition]) // 注册布局, 过度效果
@ -487,17 +333,15 @@ export default defineComponent({
watchEffect(() => { watchEffect(() => {
/** 监听 options 变化 */ /** 监听 options 变化 */
if (props.watchOptions) { if (props.watchOptions) {
watchOptionsReturn = watch( watchCallback = watch(
() => props.options, () => props.options,
(noptions) => { (noptions) => {
/** 重新组合 options */ /** 重新组合 options */
const options = combineChartOptions(noptions) const options = combineChartOptions(noptions)
const setOpt = Object.assign({}, props.setChartOptions, { const setOpt = Object.assign(
notMerge: false, props.setChartOptions,
lazyUpdate: true, defaultChartOptions,
silent: false, )
replaceMerge: [],
})
/** 如果 options 发生变动更新 echarts */ /** 如果 options 发生变动更新 echarts */
echartInstanceRef.value?.setOption(options, setOpt) echartInstanceRef.value?.setOption(options, setOpt)
}, },
@ -505,6 +349,8 @@ export default defineComponent({
deep: true, deep: true,
}, },
) )
} else {
watchCallback?.()
} }
props.loading props.loading
@ -531,7 +377,7 @@ export default defineComponent({
onBeforeUnmount(() => { onBeforeUnmount(() => {
unmount() unmount()
watchOptionsReturn?.() watchCallback?.()
}) })
return { return {

View File

@ -0,0 +1,170 @@
import type * as echarts from 'echarts/core' // `echarts` 核心模块
import type { PropType } from 'vue'
import type { MaybeArray } from '@/types/modules/utils'
import type {
LoadingOptions,
AutoResize,
ChartTheme,
} from '@/components/RChart/type'
import type { ECharts, SetOptionOpts } from 'echarts/core'
import type { MaybeComputedElementRef, MaybeElement } from '@vueuse/core'
import type { EChartsExtensionInstallRegisters } from './type'
import { loadingOptions } from './helper'
const props = {
width: {
/**
*
* chart
*
* , 200px
*/
type: String,
default: '100%',
},
height: {
/**
*
* chart
*
* , 200px
*/
type: String,
default: '100%',
},
autoResize: {
/**
*
* `chart`
*
* , ,
*
*/
type: [Boolean, Object] as PropType<AutoResize>,
default: true,
},
canvasRender: {
/**
*
* `chart` , 使 `canvas`
*
* , `SVGRenderer`
*/
type: Boolean,
default: true,
},
showAria: {
/**
*
* `chart` 访
*
* `options` `aria`
*/
type: Boolean,
default: false,
},
options: {
type: Object as PropType<echarts.EChartsCoreOption>,
default: () => ({}),
},
onSuccess: {
/**
*
* chart
*
*
*
* () => ECharts
*/
type: [Function, Array] as PropType<MaybeArray<(e: ECharts) => void>>,
default: null,
},
onError: {
/**
*
*
*
* () => void
*/
type: [Function, Array] as PropType<MaybeArray<() => void>>,
default: null,
},
theme: {
type: [String, Object] as PropType<ChartTheme>,
default: '',
},
autoChangeTheme: {
/**
*
*
* , `theme`
*
* 注意: 这个属性重度依赖此模板
*/
type: Boolean,
default: true,
},
use: {
/**
*
* `echarts`
*
*/
type: Array as PropType<EChartsExtensionInstallRegisters[]>,
default: () => [],
},
watchOptions: {
/** 主动监听 options 变化 */
type: Boolean,
default: true,
},
loading: {
/** 加载动画 */
type: Boolean,
default: false,
},
loadingOptions: {
/** 配置加载动画样式 */
type: Object as PropType<LoadingOptions>,
default: () => loadingOptions(),
},
observer: {
/**
*
*
* autoResize
*
*/
type: Object as PropType<MaybeComputedElementRef<MaybeElement>>,
default: null,
},
throttleWait: {
/** 节流等待时间 */
type: Number,
default: 500,
},
animation: {
/** 是否强制启用渲染动画 */
type: Boolean,
default: true,
},
setChartOptions: {
/**
*
* options setOptions
*
*
* notMerge: false,
* lazyUpdate: true,
* silent: false,
* replaceMerge: [],
*
*
*/
type: Object as PropType<SetOptionOpts>,
default: () => ({}),
},
}
export default props

View File

@ -10,6 +10,7 @@
*/ */
import type { ECharts, EChartsCoreOption } from 'echarts/core' import type { ECharts, EChartsCoreOption } from 'echarts/core'
import type { CanvasRenderer } from 'echarts/renderers' // `echarts` 渲染器
export interface ChartThemeRawModules { export interface ChartThemeRawModules {
default: Record<string, UnknownObjectKey> default: Record<string, UnknownObjectKey>
@ -66,3 +67,5 @@ export interface RayChartInst {
*/ */
render: () => void render: () => void
} }
export type EChartsExtensionInstallRegisters = typeof CanvasRenderer

View File

@ -65,7 +65,6 @@ const RIcon = defineComponent({
}, },
}, },
setup(props) { setup(props) {
const modelColor = computed(() => props.color)
const symbolId = computed(() => `#${props.prefix}-${props.name}`) const symbolId = computed(() => `#${props.prefix}-${props.name}`)
const cssVars = computed(() => { const cssVars = computed(() => {
const cssVar = { const cssVar = {
@ -82,7 +81,7 @@ const RIcon = defineComponent({
return cssVar return cssVar
}) })
const handleClick = (e: MouseEvent) => { const iconClick = (e: MouseEvent) => {
const { onClick } = props const { onClick } = props
if (onClick) { if (onClick) {
@ -91,10 +90,9 @@ const RIcon = defineComponent({
} }
return { return {
modelColor,
symbolId, symbolId,
cssVars, cssVars,
handleClick, iconClick,
} }
}, },
render() { render() {
@ -102,12 +100,12 @@ const RIcon = defineComponent({
<span <span
class={['ray-icon', this.customClassName]} class={['ray-icon', this.customClassName]}
style={[this.cssVars]} style={[this.cssVars]}
onClick={this.handleClick.bind(this)} onClick={this.iconClick.bind(this)}
> >
<svg <svg
{...({ RayIconAttribute: 'ray-icon', ariaHidden: true } as object)} {...({ RayIconAttribute: 'ray-icon', ariaHidden: true } as object)}
> >
<use {...{ 'xlink:href': this.symbolId }} fill={this.modelColor} /> <use {...{ 'xlink:href': this.symbolId }} fill={this.color} />
</svg> </svg>
</span> </span>
) )

View File

@ -53,24 +53,18 @@ const RIframe = defineComponent({
} }
} }
const getIframeRef = () => {
const iframeEl = iframeRef.value as HTMLElement
return iframeEl
}
expose({ expose({
iframeInst: iframeRef, iframeInst: iframeRef,
}) })
onMounted(() => { onMounted(() => {
on(getIframeRef(), 'load', iframeLoadSuccess.bind(this)) on(iframeRef.value, 'load', iframeLoadSuccess.bind(this))
on(getIframeRef(), 'error', iframeLoadError) on(iframeRef.value, 'error', iframeLoadError)
}) })
onBeforeUnmount(() => { onBeforeUnmount(() => {
off(getIframeRef(), 'load', iframeLoadSuccess) off(iframeRef.value, 'load', iframeLoadSuccess)
off(getIframeRef(), 'error', iframeLoadError) off(iframeRef.value, 'error', iframeLoadError)
}) })
return { return {

View File

@ -129,6 +129,8 @@ export default defineComponent({
() => props.text, () => props.text,
() => renderQRCode(), () => renderQRCode(),
) )
} else {
watchCallback?.()
} }
}) })
@ -141,7 +143,7 @@ export default defineComponent({
renderQRCode() renderQRCode()
}) })
onBeforeUnmount(() => { onBeforeUnmount(() => {
watchCallback && watchCallback() watchCallback?.()
}) })
return { return {
@ -175,7 +177,7 @@ export default defineComponent({
this.$slots.errorAction() this.$slots.errorAction()
) : ( ) : (
<> <>
<NButton text color="#ffffff"> <NButton text>
{{ {{
default: () => this.errorActionDescription, default: () => this.errorActionDescription,
icon: () => ( icon: () => (

View File

@ -21,8 +21,13 @@ import props from './props'
import { call } from '@/utils/vue/index' import { call } from '@/utils/vue/index'
import { uuid } from '@use-utils/hook' import { uuid } from '@use-utils/hook'
import config from './config' import config from './config'
import { throttle } from 'lodash-es'
import type { DropdownOption, DataTableInst } from 'naive-ui' import type {
DropdownOption,
DataTableInst,
DataTableCreateRowProps,
} from 'naive-ui'
import type { ComponentSize } from '@/types/modules/component' import type { ComponentSize } from '@/types/modules/component'
import type { C as CType } from './type' import type { C as CType } from './type'
@ -69,8 +74,8 @@ export default defineComponent({
* RTable rowProps * RTable rowProps
* *
*/ */
const combineRowProps = (arr: Record<string, unknown>, idx: number) => { const combineRowProps = (row: Record<string, unknown>, idx: number) => {
const interceptRowProps = props.rowProps?.(arr, idx) const interceptRowProps = props.rowProps?.(row, idx)
return { return {
...interceptRowProps, ...interceptRowProps,

View File

@ -15,6 +15,7 @@ import type { PropType, VNode, VNodeChild } from 'vue'
import type { MaybeArray } from '@/types/modules/utils' import type { MaybeArray } from '@/types/modules/utils'
import type { DropdownOption, DataTableColumn } from 'naive-ui' import type { DropdownOption, DataTableColumn } from 'naive-ui'
import type { DownloadTableOptions, PrintTableOptions } from './type' import type { DownloadTableOptions, PrintTableOptions } from './type'
import type { Recordable } from '@/types/modules/helper'
const props = { const props = {
...dataTableProps, ...dataTableProps,
@ -107,6 +108,17 @@ const props = {
>, >,
default: null, default: null,
}, },
onContextmenu: {
/**
*
* Table
* 使 rowProps
*/
type: [Function, Array] as PropType<
MaybeArray<(row: Recordable, index: number, e: MouseEvent) => void>
>,
default: null,
},
} }
export default props export default props

View File

@ -26,7 +26,7 @@
import { useKeepAlive } from '@/store' import { useKeepAlive } from '@/store'
import { APP_KEEP_ALIVE } from '@/app-config/appConfig' import { APP_KEEP_ALIVE } from '@/app-config/appConfig'
import type { PropType } from 'vue' import type { TransitionProps } from './type'
/** /**
* *
@ -36,20 +36,10 @@ import type { PropType } from 'vue'
defineOptions({ defineOptions({
name: 'RTransitionComponent', name: 'RTransitionComponent',
}) })
withDefaults(defineProps<TransitionProps>(), {
defineProps({ transitionPropName: 'fade',
transitionPropName: { transitionMode: 'out-in',
type: String, transitionAppear: true,
default: 'fade',
},
transitionMode: {
type: String as PropType<'default' | 'out-in' | 'in-out' | undefined>,
default: 'out-in',
},
transitionAppear: {
type: Boolean,
default: false,
},
}) })
const keepAliveStore = useKeepAlive() const keepAliveStore = useKeepAlive()

View File

@ -0,0 +1,7 @@
import type { BaseTransitionProps } from 'vue'
export interface TransitionProps {
transitionPropName?: string
transitionMode?: BaseTransitionProps['mode']
transitionAppear?: boolean
}

View File

@ -75,7 +75,7 @@ export const orderRoutes = (routes: AppRouteRecordRaw[]) => {
const nextOrder = next.meta?.order ?? 0 const nextOrder = next.meta?.order ?? 0
if (typeof currOrder !== 'number' || typeof nextOrder !== 'number') { if (typeof currOrder !== 'number' || typeof nextOrder !== 'number') {
throw new Error('orderRoutes error: order must be a number!') throw new TypeError('orderRoutes error: order must be a number!')
} }
if (currOrder === nextOrder) { if (currOrder === nextOrder) {

View File

@ -1,7 +1,7 @@
import { getAppDefaultLanguage } from '@/locales/helper' import { getAppDefaultLanguage } from '@/locales/helper'
import { setStorage } from '@use-utils/cache' import { setStorage } from '@use-utils/cache'
import { set } from 'lodash-es' import { set } from 'lodash-es'
import { addClass, removeClass, colorToRgba } from '@/utils/element' import { colorToRgba } from '@/utils/element'
import { useI18n } from '@/hooks/web/index' import { useI18n } from '@/hooks/web/index'
import { APP_THEME } from '@/app-config/designConfig' import { APP_THEME } from '@/app-config/designConfig'
import { useDayjs } from '@/hooks/web/index' import { useDayjs } from '@/hooks/web/index'

View File

@ -1,8 +0,0 @@
import type { VueInstance } from './vue'
export type MaybeElement =
| HTMLElement
| SVGElement
| VueInstance
| undefined
| null

View File

@ -1,5 +1,6 @@
/* eslint-disable @typescript-eslint/no-explicit-any */ /* eslint-disable @typescript-eslint/no-explicit-any */
import type CryptoJS from 'crypto-js' import type CryptoJS from 'crypto-js'
import type { BasicTarget } from './vue'
export type StorageLike = 'sessionStorage' | 'localStorage' export type StorageLike = 'sessionStorage' | 'localStorage'
@ -56,3 +57,7 @@ export type ElementSelector = string | `attr:${string}`
export type MaybeArray<T> = T | T[] export type MaybeArray<T> = T | T[]
export type DownloadAnyFileDataType = Blob | File | string | ArrayBuffer export type DownloadAnyFileDataType = Blob | File | string | ArrayBuffer
export type EventListenerTarget = BasicTarget<
HTMLElement | Element | Window | Document
>

View File

@ -1,14 +1,16 @@
import type { ComponentPublicInstance, MaybeRef } from 'vue' import type { ComponentPublicInstance } from 'vue'
import type { MaybeElement } from './element'
export type MaybeElementRef<T extends MaybeElement = MaybeElement> = MaybeRef<T> export type TargetValue<T> = T | undefined | null
export type VueInstance = ComponentPublicInstance export type TargetType =
| HTMLElement
| Element
| SVGElement
| Window
| Document
| ComponentPublicInstance
export type MaybeRefOrGetter<T> = MaybeRef<T> | (() => T) export type BasicTarget<T extends TargetType = Element> =
| (() => TargetValue<T>)
export type MaybeComputedElementRef<T extends MaybeElement = MaybeElement> = | TargetValue<T>
MaybeRefOrGetter<T> | Ref<TargetValue<T>>
export type UnRefElementReturn<T extends MaybeElement = MaybeElement> =
T extends VueInstance ? Exclude<MaybeElement, VueInstance> : T | undefined

View File

@ -1,15 +1,18 @@
import { isValueType } from '@use-utils/hook' import { isValueType } from '@use-utils/hook'
import { APP_REGEX } from '@/app-config/regexConfig' import { APP_REGEX } from '@/app-config/regexConfig'
import { unrefElement } from '@/utils/vue/index'
import type { import type {
EventListenerOrEventListenerObject, EventListenerOrEventListenerObject,
PartialCSSStyleDeclaration, PartialCSSStyleDeclaration,
ElementSelector, ElementSelector,
} from '@/types/modules/utils' } from '@/types/modules/utils'
import type { EventListenerTarget } from '@/types/modules/utils'
import type { BasicTarget } from '@/types/modules/vue'
/** /**
* *
* @param element Target element dom * @param target Target element dom
* @param event * @param event
* @param handler * @param handler
* @param useCapture * @param useCapture
@ -17,19 +20,21 @@ import type {
* @remark * @remark
*/ */
export const on = ( export const on = (
element: HTMLElement | Document | Window, target: EventListenerTarget,
event: string, event: string,
handler: EventListenerOrEventListenerObject, handler: EventListenerOrEventListenerObject,
useCapture: boolean | AddEventListenerOptions = false, useCapture: boolean | AddEventListenerOptions = false,
) => { ) => {
if (element && event && handler) { const targetElement = unrefElement(target, window)
element.addEventListener(event, handler, useCapture)
if (targetElement && event && handler) {
targetElement.addEventListener(event, handler, useCapture)
} }
} }
/** /**
* *
* @param element Target element dom * @param target Target element dom
* @param event * @param event
* @param handler * @param handler
* @param useCapture * @param useCapture
@ -37,30 +42,37 @@ export const on = (
* @remark * @remark
*/ */
export const off = ( export const off = (
element: HTMLElement | Document | Window, target: EventListenerTarget,
event: string, event: string,
handler: EventListenerOrEventListenerObject, handler: EventListenerOrEventListenerObject,
useCapture: boolean | AddEventListenerOptions = false, useCapture: boolean | AddEventListenerOptions = false,
) => { ) => {
if (element && event && handler) { const targetElement = unrefElement(target, window)
element.removeEventListener(event, handler, useCapture)
if (targetElement && event && handler) {
targetElement.removeEventListener(event, handler, useCapture)
} }
} }
/** /**
* *
* @param element Target element dom * @param target Target element dom
* @param className className: 'xxx xxx' | 'xxx' ( css ) * @param className className: 'xxx xxx' | 'xxx' ( css )
* *
* @remark className(: 'xxx xxx' | 'xxx') * @remark className(: 'xxx xxx' | 'xxx')
*/ */
export const addClass = (element: HTMLElement, className: string) => { export const addClass = (
if (element) { target: BasicTarget<Element | HTMLElement | SVGAElement>,
className: string,
) => {
const targetElement = unrefElement(target)
if (targetElement) {
const classes = className.trim().split(' ') const classes = className.trim().split(' ')
classes.forEach((item) => { classes.forEach((item) => {
if (item) { if (item) {
element.classList.add(item) targetElement.classList.add(item)
} }
}) })
} }
@ -68,19 +80,21 @@ export const addClass = (element: HTMLElement, className: string) => {
/** /**
* *
* @param element Target element dom * @param target Target element dom
* @param className className: 'xxx xxx' | 'xxx' ( css ) * @param className className: 'xxx xxx' | 'xxx' ( css )
* *
* @remark className(: 'xxx xxx' | 'xxx') * @remark className(: 'xxx xxx' | 'xxx')
* @remark removeAllClass class name * @remark removeAllClass class name
*/ */
export const removeClass = ( export const removeClass = (
element: HTMLElement, target: BasicTarget<Element | HTMLElement | SVGAElement>,
className: string | 'removeAllClass', className: string | 'removeAllClass',
) => { ) => {
if (element) { const targetElement = unrefElement(target)
if (targetElement) {
if (className === 'removeAllClass') { if (className === 'removeAllClass') {
const classList = element.classList const classList = targetElement.classList
classList.forEach((curr) => classList.remove(curr)) classList.forEach((curr) => classList.remove(curr))
} else { } else {
@ -88,7 +102,7 @@ export const removeClass = (
classes.forEach((item) => { classes.forEach((item) => {
if (item) { if (item) {
element.classList.remove(item) targetElement.classList.remove(item)
} }
}) })
} }
@ -97,15 +111,21 @@ export const removeClass = (
/** /**
* *
* @param element Target element dom * @param target Target element dom
* @param className className: 'xxx xxx' | 'xxx' ( css ) * @param className className: 'xxx xxx' | 'xxx' ( css )
* *
* @returns boolean * @returns boolean
* *
* @remark className(: 'xxx xxx' | 'xxx' ) * @remark className(: 'xxx xxx' | 'xxx' )
*/ */
export const hasClass = (element: HTMLElement, className: string) => { export const hasClass = (target: BasicTarget, className: string) => {
const elementClassName = element.className const targetElement = unrefElement(target)
if (!targetElement) {
return false
}
const elementClassName = targetElement.className
const classes = className const classes = className
.trim() .trim()
@ -117,7 +137,7 @@ export const hasClass = (element: HTMLElement, className: string) => {
/** /**
* *
* @param el Target element dom * @param target Target element dom
* @param styles (, ) * @param styles (, )
* *
* *
@ -139,10 +159,12 @@ export const hasClass = (element: HTMLElement, className: string) => {
* ``` * ```
*/ */
export const addStyle = ( export const addStyle = (
el: HTMLElement, target: BasicTarget<HTMLElement | SVGAElement>,
styles: PartialCSSStyleDeclaration | string, styles: PartialCSSStyleDeclaration | string,
) => { ) => {
if (!el) { const targetElement = unrefElement(target)
if (!targetElement) {
return return
} }
@ -165,8 +187,8 @@ export const addStyle = (
Object.keys(styleObj).forEach((key) => { Object.keys(styleObj).forEach((key) => {
const value = styleObj[key] const value = styleObj[key]
if (key in el.style) { if (key in targetElement.style) {
el.style[key] = value targetElement.style[key] = value
} }
}) })
} }
@ -177,15 +199,17 @@ export const addStyle = (
* @param styles * @param styles
*/ */
export const removeStyle = ( export const removeStyle = (
el: HTMLElement, target: BasicTarget<HTMLElement | SVGAElement>,
styles: (keyof CSSStyleDeclaration & string)[], styles: (keyof CSSStyleDeclaration & string)[],
) => { ) => {
if (!el) { const targetElement = unrefElement(target)
if (!targetElement) {
return return
} }
styles.forEach((curr) => { styles.forEach((curr) => {
el.style.removeProperty(curr) targetElement.style.removeProperty(curr)
}) })
} }

View File

@ -9,17 +9,27 @@
* @remark * @remark
*/ */
import type { import type { BasicTarget, TargetType, TargetValue } from '@/types/modules/vue'
MaybeComputedElementRef, import type { ComponentPublicInstance } from 'vue'
UnRefElementReturn,
VueInstance,
} from '@/types/modules/vue'
import type { MaybeElement } from '@/types/modules/element'
export function unrefElement<T extends MaybeElement>( export function unrefElement<T extends TargetType>(
elRef: MaybeComputedElementRef<T>, target: BasicTarget<T>,
): UnRefElementReturn<T> { defaultTarget?: T,
const plain = toValue(elRef) ) {
if (!target) {
return (plain as VueInstance)?.$el ?? plain return defaultTarget
}
let targetElement: TargetValue<T>
if (typeof target === 'function') {
targetElement = target()
} else if (isRef(target)) {
targetElement =
(target.value as ComponentPublicInstance)?.$el ?? target.value
} else {
targetElement = target
}
return targetElement
} }

View File

@ -23,7 +23,6 @@ import { ViteEjsPlugin as viteEjsPlugin } from 'vite-plugin-ejs'
import viteAutoImport from 'unplugin-auto-import/vite' import viteAutoImport from 'unplugin-auto-import/vite'
import viteEslint from 'vite-plugin-eslint' import viteEslint from 'vite-plugin-eslint'
import mockDevServerPlugin from 'vite-plugin-mock-dev-server' import mockDevServerPlugin from 'vite-plugin-mock-dev-server'
import vueI18nPlugin from '@intlify/unplugin-vue-i18n/vite'
import { createSvgIconsPlugin } from 'vite-plugin-svg-icons' import { createSvgIconsPlugin } from 'vite-plugin-svg-icons'
import unpluginViteComponents from 'unplugin-vue-components/vite' import unpluginViteComponents from 'unplugin-vue-components/vite'
import { cdn as viteCDNPlugin } from 'vite-plugin-cdn2' import { cdn as viteCDNPlugin } from 'vite-plugin-cdn2'
@ -41,7 +40,13 @@ export default function (mode: string): PluginOption[] {
vue(), vue(),
viteVueJSX(), viteVueJSX(),
title, title,
viteVeI18nPlugin({}), viteVeI18nPlugin({
runtimeOnly: true,
compositionOnly: true,
forceStringify: true,
defaultSFCLang: 'json',
include: [path.resolve(__dirname, '../locales/**')],
}),
viteAutoImport({ viteAutoImport({
eslintrc: { eslintrc: {
enabled: true, enabled: true,
@ -149,13 +154,6 @@ export default function (mode: string): PluginOption[] {
reload: true, reload: true,
build: true, build: true,
}), }),
vueI18nPlugin({
runtimeOnly: true,
compositionOnly: true,
forceStringify: true,
defaultSFCLang: 'json',
include: [path.resolve(__dirname, '../locales/**')],
}),
createSvgIconsPlugin({ createSvgIconsPlugin({
iconDirs: [path.resolve(process.cwd(), 'src/icons')], iconDirs: [path.resolve(process.cwd(), 'src/icons')],
symbolId: 'icon-[dir]-[name]', symbolId: 'icon-[dir]-[name]',
@ -165,7 +163,6 @@ export default function (mode: string): PluginOption[] {
viteCDNPlugin({ viteCDNPlugin({
modules: [ modules: [
'vue', 'vue',
'vue-demi',
'pinia', 'pinia',
'naive-ui', 'naive-ui',
'vue-router', 'vue-router',