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

View File

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

View File

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

View File

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

View File

@ -38,188 +38,34 @@ import {
import { LabelLayout, UniversalTransition } from 'echarts/features' // 标签自动布局, 全局过渡动画等特性
import { CanvasRenderer } from 'echarts/renderers' // `echarts` 渲染器
import props from './props'
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 { setupChartTheme } from './helper'
import { APP_THEME } from '@/app-config/designConfig'
import { useResizeObserver } from '@vueuse/core'
import type { PropType, WatchStopHandle } from 'vue'
import type { AnyFC, MaybeArray } from '@/types/modules/utils'
import type { WatchStopHandle } from 'vue'
import type { AnyFC } from '@/types/modules/utils'
import type { DebouncedFunc } from 'lodash-es'
import type {
LoadingOptions,
AutoResize,
ChartTheme,
} from '@/components/RChart/type'
import type {
UseResizeObserverReturn,
MaybeComputedElementRef,
MaybeElement,
} from '@vueuse/core'
import type { ChartTheme } from '@/components/RChart/type'
import type { UseResizeObserverReturn } from '@vueuse/core'
import type { ECharts, EChartsCoreOption, SetOptionOpts } from 'echarts/core'
export type EChartsExtensionInstallRegisters = typeof CanvasRenderer
export type { RayChartInst } from './type'
export type { RayChartInst, EChartsExtensionInstallRegisters } from './type'
const defaultChartOptions = {
notMerge: false,
lazyUpdate: true,
silent: false,
replaceMerge: [],
}
export default defineComponent({
name: 'RChart',
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: () => ({}),
},
},
props,
setup(props, { expose }) {
const settingStore = useSetting()
const { themeValue } = storeToRefs(settingStore)
@ -229,7 +75,7 @@ export default defineComponent({
let resizeThrottleReturn: DebouncedFunc<AnyFC> | null // resize 防抖方法实例
let resizeOvserverReturn: UseResizeObserverReturn | null
const { echartTheme } = APP_THEME
let watchOptionsReturn: WatchStopHandle | null
let watchCallback: WatchStopHandle | null
const cssVarsRef = computed(() => {
const cssVars = {
@ -266,7 +112,7 @@ export default defineComponent({
CandlestickChart,
ScatterChart,
PictorialBarChart,
]) // 注册类型
]) // 注册 chart series type
echarts.use([LabelLayout, UniversalTransition]) // 注册布局, 过度效果
@ -487,17 +333,15 @@ export default defineComponent({
watchEffect(() => {
/** 监听 options 变化 */
if (props.watchOptions) {
watchOptionsReturn = watch(
watchCallback = watch(
() => props.options,
(noptions) => {
/** 重新组合 options */
const options = combineChartOptions(noptions)
const setOpt = Object.assign({}, props.setChartOptions, {
notMerge: false,
lazyUpdate: true,
silent: false,
replaceMerge: [],
})
const setOpt = Object.assign(
props.setChartOptions,
defaultChartOptions,
)
/** 如果 options 发生变动更新 echarts */
echartInstanceRef.value?.setOption(options, setOpt)
},
@ -505,6 +349,8 @@ export default defineComponent({
deep: true,
},
)
} else {
watchCallback?.()
}
props.loading
@ -531,7 +377,7 @@ export default defineComponent({
onBeforeUnmount(() => {
unmount()
watchOptionsReturn?.()
watchCallback?.()
})
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 { CanvasRenderer } from 'echarts/renderers' // `echarts` 渲染器
export interface ChartThemeRawModules {
default: Record<string, UnknownObjectKey>
@ -66,3 +67,5 @@ export interface RayChartInst {
*/
render: () => void
}
export type EChartsExtensionInstallRegisters = typeof CanvasRenderer

View File

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

View File

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

View File

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

View File

@ -21,8 +21,13 @@ import props from './props'
import { call } from '@/utils/vue/index'
import { uuid } from '@use-utils/hook'
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 { C as CType } from './type'
@ -69,8 +74,8 @@ export default defineComponent({
* RTable rowProps
*
*/
const combineRowProps = (arr: Record<string, unknown>, idx: number) => {
const interceptRowProps = props.rowProps?.(arr, idx)
const combineRowProps = (row: Record<string, unknown>, idx: number) => {
const interceptRowProps = props.rowProps?.(row, idx)
return {
...interceptRowProps,

View File

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

View File

@ -26,7 +26,7 @@
import { useKeepAlive } from '@/store'
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({
name: 'RTransitionComponent',
})
defineProps({
transitionPropName: {
type: String,
default: 'fade',
},
transitionMode: {
type: String as PropType<'default' | 'out-in' | 'in-out' | undefined>,
default: 'out-in',
},
transitionAppear: {
type: Boolean,
default: false,
},
withDefaults(defineProps<TransitionProps>(), {
transitionPropName: 'fade',
transitionMode: 'out-in',
transitionAppear: true,
})
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
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) {

View File

@ -1,7 +1,7 @@
import { getAppDefaultLanguage } from '@/locales/helper'
import { setStorage } from '@use-utils/cache'
import { set } from 'lodash-es'
import { addClass, removeClass, colorToRgba } from '@/utils/element'
import { colorToRgba } from '@/utils/element'
import { useI18n } from '@/hooks/web/index'
import { APP_THEME } from '@/app-config/designConfig'
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 */
import type CryptoJS from 'crypto-js'
import type { BasicTarget } from './vue'
export type StorageLike = 'sessionStorage' | 'localStorage'
@ -56,3 +57,7 @@ export type ElementSelector = string | `attr:${string}`
export type MaybeArray<T> = T | T[]
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 { MaybeElement } from './element'
import type { ComponentPublicInstance } from 'vue'
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 MaybeComputedElementRef<T extends MaybeElement = MaybeElement> =
MaybeRefOrGetter<T>
export type UnRefElementReturn<T extends MaybeElement = MaybeElement> =
T extends VueInstance ? Exclude<MaybeElement, VueInstance> : T | undefined
export type BasicTarget<T extends TargetType = Element> =
| (() => TargetValue<T>)
| TargetValue<T>
| Ref<TargetValue<T>>

View File

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

View File

@ -9,17 +9,27 @@
* @remark
*/
import type {
MaybeComputedElementRef,
UnRefElementReturn,
VueInstance,
} from '@/types/modules/vue'
import type { MaybeElement } from '@/types/modules/element'
import type { BasicTarget, TargetType, TargetValue } from '@/types/modules/vue'
import type { ComponentPublicInstance } from 'vue'
export function unrefElement<T extends MaybeElement>(
elRef: MaybeComputedElementRef<T>,
): UnRefElementReturn<T> {
const plain = toValue(elRef)
export function unrefElement<T extends TargetType>(
target: BasicTarget<T>,
defaultTarget?: T,
) {
if (!target) {
return defaultTarget
}
return (plain as VueInstance)?.$el ?? plain
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 viteEslint from 'vite-plugin-eslint'
import mockDevServerPlugin from 'vite-plugin-mock-dev-server'
import vueI18nPlugin from '@intlify/unplugin-vue-i18n/vite'
import { createSvgIconsPlugin } from 'vite-plugin-svg-icons'
import unpluginViteComponents from 'unplugin-vue-components/vite'
import { cdn as viteCDNPlugin } from 'vite-plugin-cdn2'
@ -41,7 +40,13 @@ export default function (mode: string): PluginOption[] {
vue(),
viteVueJSX(),
title,
viteVeI18nPlugin({}),
viteVeI18nPlugin({
runtimeOnly: true,
compositionOnly: true,
forceStringify: true,
defaultSFCLang: 'json',
include: [path.resolve(__dirname, '../locales/**')],
}),
viteAutoImport({
eslintrc: {
enabled: true,
@ -149,13 +154,6 @@ export default function (mode: string): PluginOption[] {
reload: true,
build: true,
}),
vueI18nPlugin({
runtimeOnly: true,
compositionOnly: true,
forceStringify: true,
defaultSFCLang: 'json',
include: [path.resolve(__dirname, '../locales/**')],
}),
createSvgIconsPlugin({
iconDirs: [path.resolve(process.cwd(), 'src/icons')],
symbolId: 'icon-[dir]-[name]',
@ -165,7 +163,6 @@ export default function (mode: string): PluginOption[] {
viteCDNPlugin({
modules: [
'vue',
'vue-demi',
'pinia',
'naive-ui',
'vue-router',