This commit is contained in:
ray_wuhao 2023-07-12 14:45:24 +08:00
parent ca9bbf7673
commit 5502cd690d
25 changed files with 507 additions and 213 deletions

View File

@ -30,6 +30,7 @@ module.exports = {
defineEmits: 'readonly', defineEmits: 'readonly',
defineExpose: 'readonly', defineExpose: 'readonly',
withDefaults: 'readonly', withDefaults: 'readonly',
defineOptions: 'readonly',
}, },
rules: { rules: {
'@typescript-eslint/no-explicit-any': [ '@typescript-eslint/no-explicit-any': [
@ -173,13 +174,12 @@ module.exports = {
'error', 'error',
'PascalCase', 'PascalCase',
{ {
registeredComponentsOnly: true, registeredComponentsOnly: false,
globals: ['RouterView'],
}, },
], ],
'vue/no-unused-refs': ['error'], 'vue/no-unused-refs': ['error'],
'vue/prop-name-casing': ['error', 'camelCase'], 'vue/prop-name-casing': ['error', 'camelCase'],
'vue/component-options-name-casing': ['error', 'camelCase'], 'vue/component-options-name-casing': ['error', 'PascalCase'],
'vue/attribute-hyphenation': [ 'vue/attribute-hyphenation': [
'error', 'error',
'never', 'never',

7
.npmrc Normal file
View File

@ -0,0 +1,7 @@
package-lock=false
prefer-offline=true
save-exact=true
engine-strict=true
engines={
"pnpm": ">=8.6.6"
}

View File

@ -7,5 +7,6 @@
"i18n-ally.enabledParsers": ["json"], "i18n-ally.enabledParsers": ["json"],
"i18n-ally.sourceLanguage": "zh-CN", "i18n-ally.sourceLanguage": "zh-CN",
"i18n-ally.displayLanguage": "zh-CN", "i18n-ally.displayLanguage": "zh-CN",
"i18n-ally.enabledFrameworks": ["vue", "react"] "i18n-ally.enabledFrameworks": ["vue", "react"],
"typescript.tsdk": "node_modules/typescript/lib"
} }

View File

@ -1,5 +1,51 @@
# CHANGE LOG # CHANGE LOG
## 4.1.0
### Feats
- 升级 vue 版本为 v3.3.4。并且配套升级了模板的一些插件
- RayTransitionComponent 组件加入 Suspense 组件的支持(试验性加入,可能会移除)
- 更新部分组件的事件触发方式,类似 onUpdateValue、onupdate:value 方法改为 props 定义而非 emit受控、非受控
- 更新路由切换动画的透明度,视觉效果更友好
- App.tsx 组件内部逻辑抽离为 AppStyleProvider。将一些组件存放位置放在 AppComponents 文件包中
- 新增 useRequest useHookPlusRequest 两个请求 hook具体使用方法看示例基于 vue-hook-plus useRequest 实现)
- useRequest 支持直接配置请求与配置请求相关的配置(缓存、节流、防抖等)
- useHookPlusRequest 支持接收一个 Promise 返回值的方法,可以用来包裹 axios 方法然后进行请求配置
```ts
import axiosInstance from '@/axios/instance'
import { useRequest, useHookPlusRequest } from '@/axios/index'
// 使用 useRequest
const { data, loading, run } = useRequest<{
title: string
}>(
{
url: 'https://jsonplaceholder.typicode.com/todos/1',
method: 'get',
},
{
manual: true,
},
)
// 使用 useHookPlusRequest
export const getWeather = (city: string) => {
return axiosInstance<AxiosTestResponse>({
url: `https://www.tianqiapi.com/api?version=v9&appid=23035354&appsecret=8YvlPNrz&city=${city}`,
method: 'get',
})
}
const { data, loading, run } = useHookPlusRequest(getWeather, {
throttleWait: 1000,
})
// 手动更新请求参数
run('some value')
```
## 4.0.3 ## 4.0.3
### Feats ### Feats

View File

@ -29,15 +29,16 @@
"echarts": "^5.4.0", "echarts": "^5.4.0",
"lodash-es": "^4.17.21", "lodash-es": "^4.17.21",
"naive-ui": "^2.34.4", "naive-ui": "^2.34.4",
"pinia": "^2.0.17", "pinia": "^2.1.4",
"pinia-plugin-persistedstate": "^2.4.0", "pinia-plugin-persistedstate": "^3.1.0",
"print-js": "^1.6.0", "print-js": "^1.6.0",
"qrcode.vue": "^3.3.4", "qrcode.vue": "^3.3.4",
"sass": "^1.54.3", "sass": "^1.54.3",
"screenfull": "^6.0.2", "screenfull": "^6.0.2",
"vue": "^3.2.47", "vue": "^3.3.4",
"vue-hooks-plus": "1.7.6",
"vue-i18n": "^9.2.2", "vue-i18n": "^9.2.2",
"vue-router": "^4.1.3", "vue-router": "^4.2.4",
"vuedraggable": "^4.1.0", "vuedraggable": "^4.1.0",
"xlsx": "^0.18.5" "xlsx": "^0.18.5"
}, },
@ -46,14 +47,15 @@
"@babel/eslint-parser": "^7.19.1", "@babel/eslint-parser": "^7.19.1",
"@commitlint/cli": "^17.4.2", "@commitlint/cli": "^17.4.2",
"@commitlint/config-conventional": "^17.4.2", "@commitlint/config-conventional": "^17.4.2",
"@intlify/unplugin-vue-i18n": "^0.5.0", "@intlify/unplugin-vue-i18n": "^0.12.1",
"@types/crypto-js": "^4.1.1", "@types/crypto-js": "^4.1.1",
"@types/lodash-es": "^4.17.7", "@types/lodash-es": "^4.17.7",
"@types/scrollreveal": "^0.0.8", "@types/scrollreveal": "^0.0.8",
"@typescript-eslint/eslint-plugin": "^5.61.0", "@typescript-eslint/eslint-plugin": "^5.61.0",
"@typescript-eslint/parser": "^5.61.0", "@typescript-eslint/parser": "^5.61.0",
"@vitejs/plugin-vue": "^4.1.0", "@vitejs/plugin-vue": "^4.2.3",
"@vitejs/plugin-vue-jsx": "^3.0.1", "@vitejs/plugin-vue-jsx": "^3.0.1",
"@vue-hooks-plus/resolvers": "1.2.4",
"@vue/eslint-config-prettier": "^7.1.0", "@vue/eslint-config-prettier": "^7.1.0",
"@vue/eslint-config-typescript": "^11.0.3", "@vue/eslint-config-typescript": "^11.0.3",
"autoprefixer": "^10.4.8", "autoprefixer": "^10.4.8",
@ -75,8 +77,8 @@
"rollup-plugin-visualizer": "^5.8.3", "rollup-plugin-visualizer": "^5.8.3",
"svg-sprite-loader": "^6.0.11", "svg-sprite-loader": "^6.0.11",
"typescript": "^5.0.2", "typescript": "^5.0.2",
"unplugin-auto-import": "^0.11.0", "unplugin-auto-import": "^0.15.0",
"unplugin-vue-components": "^0.22.0", "unplugin-vue-components": "^0.25.1",
"vite": "^4.3.9", "vite": "^4.3.9",
"vite-plugin-compression": "^0.5.1", "vite-plugin-compression": "^0.5.1",
"vite-plugin-ejs": "^1.6.4", "vite-plugin-ejs": "^1.6.4",
@ -85,7 +87,7 @@
"vite-plugin-inspect": "^0.7.26", "vite-plugin-inspect": "^0.7.26",
"vite-plugin-svg-icons": "^2.0.1", "vite-plugin-svg-icons": "^2.0.1",
"vite-svg-loader": "^3.4.0", "vite-svg-loader": "^3.4.0",
"vue-tsc": "^1.4.2" "vue-tsc": "^1.8.4"
}, },
"description": "<!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section -->", "description": "<!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section -->",
"main": "index.ts", "main": "index.ts",

View File

@ -11,18 +11,15 @@
/** /**
* *
* 使 axios *
* *
* : *
* * 1. demo
* ``` * 2.
* const demoRequest = () => { * 3. setup 使使 useHookPlusRequest 便使 setup 使使
* return {} as AxiosResponseBody<AxiosTestResponse>
* }
* ```
*/ */
import useRequest from '@/axios/instance' import { request } from '@/axios/index'
import type { AxiosResponseBody } from '@/types/modules/axios' import type { AxiosResponseBody } from '@/types/modules/axios'
@ -31,14 +28,29 @@ interface AxiosTestResponse extends UnknownObjectKey {
city?: string city?: string
} }
interface JSONPlaceholder {
completed: boolean
id: number
title: string
userId: number
}
/** /**
* *
* @returns * @returns
* *
* @medthod get * @medthod get
*/ */
export const onAxiosTest = async (city: string) => { export const getWeather = (city: string) => {
return useRequest<AxiosTestResponse>({ return request<AxiosTestResponse>({
url: `https://www.tianqiapi.com/api?version=v9&appid=23035354&appsecret=8YvlPNrz&city=${city}`, url: `https://www.tianqiapi.com/api?version=v9&appid=23035354&appsecret=8YvlPNrz&city=${city}`,
method: 'get',
})
}
export const getTypicode = () => {
return request<JSONPlaceholder>({
url: 'https://jsonplaceholder.typicode.com/todos/1',
method: 'get',
}) })
} }

50
src/axios/index.ts Normal file
View File

@ -0,0 +1,50 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-07-11
*
* @workspace ray-template
*
* @remark
*/
/**
*
* vue-hook-plus axios
*
* (vue-hook-plus )[https://inhiblab-core.gitee.io/docs/hooks/useRequest/basic/]
*
* vue-hook-plus useRequest
* axios hook
*
* 使
*/
import inst from './instance'
import useHookPlusRequest from 'vue-hooks-plus/es/useRequest'
import request from '@/axios/instance'
import type { UseRequestOptions } from 'vue-hooks-plus/es/useRequest/types'
import type { AxiosRequestConfig } from 'axios'
function useRequest<
Response,
HookPlusParams extends unknown[] = unknown[],
HookPlusPlugin = unknown,
>(
fetchOption: AxiosRequestConfig<Response>,
option?: UseRequestOptions<Response, HookPlusParams, HookPlusPlugin>,
) {
const fc = () => {
const cb = inst<Response>(fetchOption)
return cb
}
const hooks = useHookPlusRequest(fc, Object.assign({}, option))
return hooks
}
export { useRequest, useHookPlusRequest, request }

View File

@ -41,10 +41,11 @@ import { CanvasRenderer } from 'echarts/renderers' // `echarts` 渲染器
import { useSetting } from '@/store' import { useSetting } from '@/store'
import { cloneDeep, throttle } from 'lodash-es' import { cloneDeep, throttle } from 'lodash-es'
import { on, off, addStyle, completeSize } from '@/utils/element' import { on, off, addStyle, completeSize } from '@/utils/element'
import { call } from '@/utils/vue/index'
import type { PropType } from 'vue' import type { PropType } from 'vue'
import type { EChartsInstance } from '@/types/modules/component' import type { EChartsInstance } from '@/types/modules/component'
import type { AnyFunc } from '@/types/modules/utils' import type { AnyFunc, MaybeArray } from '@/types/modules/utils'
import type { DebouncedFunc } from 'lodash-es' import type { DebouncedFunc } from 'lodash-es'
export type AutoResize = export type AutoResize =
@ -71,6 +72,8 @@ export interface LoadingOptions {
export type ChartTheme = 'dark' | '' | object export type ChartTheme = 'dark' | '' | object
export type EChartsExtensionInstallRegisters = typeof CanvasRenderer
/** /**
* *
* @returns LoadingOptions * @returns LoadingOptions
@ -164,8 +167,10 @@ const RayChart = defineComponent({
* *
* () => EChartsInstance * () => EChartsInstance
*/ */
type: Function, type: [Function, Array] as PropType<
default: () => ({}), MaybeArray<(e: EChartsInstance) => void>
>,
default: null,
}, },
error: { error: {
/** /**
@ -174,8 +179,8 @@ const RayChart = defineComponent({
* *
* () => void * () => void
*/ */
type: Function, type: [Function, Array] as PropType<MaybeArray<() => void>>,
default: () => ({}), default: null,
}, },
theme: { theme: {
type: [String, Object] as PropType<ChartTheme>, type: [String, Object] as PropType<ChartTheme>,
@ -199,7 +204,7 @@ const RayChart = defineComponent({
* `echarts` * `echarts`
* *
*/ */
type: Array as PropType<(typeof CanvasRenderer)[]>, type: Array as PropType<EChartsExtensionInstallRegisters[]>,
default: () => [], default: () => [],
}, },
watchOptions: { watchOptions: {
@ -323,6 +328,7 @@ const RayChart = defineComponent({
const options = useMergeOptions() const options = useMergeOptions()
/** 获取 dom 容器实际宽高 */ /** 获取 dom 容器实际宽高 */
const { height, width } = element.getBoundingClientRect() const { height, width } = element.getBoundingClientRect()
const { success, error } = props
/** 如果高度为 0, 则以 200px 填充 */ /** 如果高度为 0, 则以 200px 填充 */
if (height === 0) { if (height === 0) {
@ -347,12 +353,15 @@ const RayChart = defineComponent({
options && echartInstance.setOption(options) options && echartInstance.setOption(options)
/** 渲染成功回调 */ /** 渲染成功回调 */
props.success?.(echartInstance) if (success) {
call(success, echartInstance)
}
} catch (e) { } catch (e) {
/** 渲染失败回调 */ /** 渲染失败回调 */
props.error?.() if (error) {
call(error)
console.error(e) }
console.error('RayChart render error: ', e)
} }
} }

View File

@ -16,17 +16,26 @@ import { collapseGridProps } from './props'
import { NCard, NGrid, NGridItem, NSpace } from 'naive-ui' import { NCard, NGrid, NGridItem, NSpace } from 'naive-ui'
import RayIcon from '@/components/RayIcon' import RayIcon from '@/components/RayIcon'
import { call } from '@/utils/vue/index'
const RayCollapseGrid = defineComponent({ const RayCollapseGrid = defineComponent({
name: 'RayCollapseGrid', name: 'RayCollapseGrid',
props: collapseGridProps, props: collapseGridProps,
emits: ['updateValue'], setup(props) {
setup(props, { emit }) {
const modelCollapsed = ref(props.value) const modelCollapsed = ref(props.value)
const handleCollapse = () => { const handleCollapse = () => {
modelCollapsed.value = !modelCollapsed.value modelCollapsed.value = !modelCollapsed.value
emit('updateValue', modelCollapsed.value) const { onUpdateValue, 'onUpdate:value': _onUpdateValue } = props
if (onUpdateValue) {
call(onUpdateValue, modelCollapsed.value)
}
if (_onUpdateValue) {
call(_onUpdateValue, modelCollapsed.value)
}
} }
const CollapseIcon = () => ( const CollapseIcon = () => (

View File

@ -2,6 +2,7 @@ import { gridProps } from 'naive-ui'
import type { PropType } from 'vue' import type { PropType } from 'vue'
import type { CollapseToggleText } from './type' import type { CollapseToggleText } from './type'
import type { AnyFunc, MaybeArray } from '@/types/modules/utils'
export const collapseGridProps = { export const collapseGridProps = {
value: { value: {
@ -38,6 +39,14 @@ export const collapseGridProps = {
type: Boolean, type: Boolean,
default: false, default: false,
}, },
onUpdateValue: {
type: [Function, Array] as PropType<MaybeArray<(bool: boolean) => void>>,
default: null,
},
'onUpdate:value': {
type: [Function, Array] as PropType<MaybeArray<(bool: boolean) => void>>,
default: null,
},
...gridProps, ...gridProps,
} as const } as const

View File

@ -16,5 +16,18 @@
& svg[RayIconAttribute="ray-icon"] { & svg[RayIconAttribute="ray-icon"] {
width: var(--ray-icon-width); width: var(--ray-icon-width);
height: var(--ray-icon-height); height: var(--ray-icon-height);
fill: currentColor;
}
}
.ray-icon-path__animate {
stroke-dasharray: var(--ray-icon-path-length);
stroke-dashoffset: var(--ray-icon-path-length);
animation: rayIconPathAnimate 2s forwards;
}
@keyframes rayIconPathAnimate {
to {
stroke-dashoffset: 0;
} }
} }

View File

@ -11,6 +11,11 @@
import './index.scss' import './index.scss'
import { call } from '@/utils/vue/index'
import type { PropType } from 'vue'
import type { MaybeArray } from '@/types/modules/utils'
const RayIcon = defineComponent({ const RayIcon = defineComponent({
name: 'RayIcon', name: 'RayIcon',
props: { props: {
@ -53,11 +58,12 @@ const RayIcon = defineComponent({
type: String, type: String,
default: 'default', default: 'default',
}, },
onClick: {
type: [Function, Array] as PropType<MaybeArray<(e: MouseEvent) => void>>,
default: null,
}, },
emits: ['click'], },
setup(props, ctx) { setup(props) {
const emit = ctx.emit
const modelColor = computed(() => props.color) 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(() => {
@ -75,10 +81,13 @@ const RayIcon = defineComponent({
return cssVar return cssVar
}) })
const handleClick = () => { const handleClick = (e: MouseEvent) => {
emit('click') const { onClick } = props
}
if (onClick) {
call(onClick, e)
}
}
return { return {
modelColor, modelColor,
symbolId, symbolId,

View File

@ -14,8 +14,10 @@ import './index.scss'
import { NSpin } from 'naive-ui' import { NSpin } from 'naive-ui'
import { completeSize, on, off } from '@use-utils/element' import { completeSize, on, off } from '@use-utils/element'
import { call } from '@/utils/vue/index'
import type { PropType } from 'vue' import type { PropType } from 'vue'
import type { MaybeArray } from '@/types/modules/utils'
import type { SpinProps } from 'naive-ui' import type { SpinProps } from 'naive-ui'
const RayIframe = defineComponent({ const RayIframe = defineComponent({
@ -77,7 +79,9 @@ const RayIframe = defineComponent({
* iframe * iframe
* 返回值: iframe , Event * 返回值: iframe , Event
*/ */
type: Function, type: [Function, Array] as PropType<
MaybeArray<(el: HTMLIFrameElement, e: Event) => void>
>,
default: null, default: null,
}, },
error: { error: {
@ -86,7 +90,7 @@ const RayIframe = defineComponent({
* iframe * iframe
* 返回值: iframe , Event * 返回值: iframe , Event
*/ */
type: Function, type: [Function, Array] as PropType<MaybeArray<(e: Event) => void>>,
default: null, default: null,
}, },
customSpinProps: { customSpinProps: {
@ -115,13 +119,21 @@ const RayIframe = defineComponent({
const iframeLoadSuccess = (e: Event) => { const iframeLoadSuccess = (e: Event) => {
spinShow.value = false spinShow.value = false
props.success?.(iframeRef.value, e) const { success } = props
if (success) {
call(success, iframeRef.value as HTMLIFrameElement, e)
}
} }
const iframeLoadError = (e: Event) => { const iframeLoadError = (e: Event) => {
spinShow.value = false spinShow.value = false
props.error?.(iframeRef.value, e) const { error } = props
if (error) {
call(error, e)
}
} }
const getIframeRef = () => { const getIframeRef = () => {

View File

@ -1,46 +0,0 @@
<template>
<RouterView>
<template #default="{ Component, route }">
<transition
:name="transitionPropName"
:mode="transitionMode"
:appear="transitionAppear"
>
<keep-alive
v-if="setupKeepAlive"
:max="maxKeepAliveLength"
:include="keepAliveInclude"
:exclude="keepAliveExclude"
>
<component :is="Component" :key="route.fullPath" />
</keep-alive>
<component :is="Component" v-else :key="route.fullPath" />
</transition>
</template>
</RouterView>
</template>
<script lang="ts" setup>
import { useKeepAlive } from '@/store'
import { APP_KEEP_ALIVE } from '@/appConfig/appConfig'
import type { PropType } from 'vue'
defineProps({
transitionPropName: {
type: String,
default: 'fade',
},
transitionMode: {
type: String as PropType<'default' | 'out-in' | 'in-out' | undefined>,
default: 'out-in',
},
transitionAppear: {
type: Boolean,
default: true,
},
})
const keepAliveStore = useKeepAlive()
const { keepAliveInclude } = storeToRefs(keepAliveStore)
const { setupKeepAlive, maxKeepAliveLength, keepAliveExclude } = APP_KEEP_ALIVE
</script>

View File

@ -0,0 +1,58 @@
<template>
<!-- 这是一个魔法注释删不的如果删了会出现一个异常提示不信你试试 -->
<RouterView v-slot="{ Component, route }">
<template v-if="Component">
<Transition
:name="transitionPropName"
:mode="transitionMode"
:appear="transitionAppear"
>
<Suspense>
<KeepAlive
v-if="setupKeepAlive"
:max="maxKeepAliveLength"
:include="keepAliveInclude"
:exclude="keepAliveExclude"
>
<Component :is="Component" :key="route.fullPath" />
</KeepAlive>
<Component :is="Component" v-else :key="route.fullPath" />
</Suspense>
</Transition>
</template>
</RouterView>
</template>
<script lang="ts" setup>
import { useKeepAlive } from '@/store'
import { APP_KEEP_ALIVE } from '@/appConfig/appConfig'
import type { PropType } from 'vue'
/**
*
* 使用宏编译模式时可以使用 defineOptions 声明组件选项
* 常用方法即是声明该组件的 name inheritAttrs 等属性
*/
defineOptions({
name: 'TransitionComponent',
})
defineProps({
transitionPropName: {
type: String,
default: 'fade',
},
transitionMode: {
type: String as PropType<'default' | 'out-in' | 'in-out' | undefined>,
default: 'out-in',
},
transitionAppear: {
type: Boolean,
default: true,
},
})
const keepAliveStore = useKeepAlive()
const { keepAliveInclude } = storeToRefs(keepAliveStore)
const { setupKeepAlive, maxKeepAliveLength, keepAliveExclude } = APP_KEEP_ALIVE
</script>

View File

@ -11,11 +11,13 @@
import './index.scss' import './index.scss'
import RayTransitionComponent from '@/components/RayTransitionComponent/TransitionComponent.vue' import RayTransitionComponent from '@/components/RayTransitionComponent/index.vue'
import { NSpin } from 'naive-ui' import { NSpin } from 'naive-ui'
import { useSetting } from '@/store' import { useSetting } from '@/store'
import type { GlobalThemeOverrides } from 'naive-ui'
const ContentWrapper = defineComponent({ const ContentWrapper = defineComponent({
name: 'ContentWrapper', name: 'ContentWrapper',
setup() { setup() {
@ -24,6 +26,9 @@ const ContentWrapper = defineComponent({
const { reloadRouteSwitch } = storeToRefs(settingStore) const { reloadRouteSwitch } = storeToRefs(settingStore)
const spinning = ref(false) const spinning = ref(false)
const thmeOverridesSpin: GlobalThemeOverrides['Spin'] = {
opacitySpinning: '0',
}
const setupLayoutContentSpin = () => { const setupLayoutContentSpin = () => {
router.beforeEach(() => { router.beforeEach(() => {
@ -42,11 +47,17 @@ const ContentWrapper = defineComponent({
return { return {
reloadRouteSwitch, reloadRouteSwitch,
spinning, spinning,
thmeOverridesSpin,
} }
}, },
render() { render() {
return this.reloadRouteSwitch ? ( return this.reloadRouteSwitch ? (
<NSpin show={this.spinning} description="loading..." size="large"> <NSpin
show={this.spinning}
description="loading..."
size="large"
themeOverrides={this.thmeOverridesSpin}
>
<RayTransitionComponent class="content-wrapper" /> <RayTransitionComponent class="content-wrapper" />
</NSpin> </NSpin>
) : ( ) : (

View File

@ -48,3 +48,5 @@ export type PartialCSSStyleDeclaration = Partial<
> >
export type ElementSelector = string | `attr:${string}` export type ElementSelector = string | `attr:${string}`
export type MaybeArray<T> = T | T[]

View File

@ -18,7 +18,7 @@ import type { CacheType } from '@/types/modules/utils'
* @param key key * @param key key
* @param value * @param value
*/ */
export function setStorage<T = unknown>( function setStorage<T = unknown>(
key: string, key: string,
value: T, value: T,
type: CacheType = 'sessionStorage', type: CacheType = 'sessionStorage',
@ -41,14 +41,10 @@ export function setStorage<T = unknown>(
} }
/** 重载函数 getStorage */ /** 重载函数 getStorage */
export function getStorage<T>( function getStorage<T>(key: string, storageType: CacheType, defaultValue: T): T
key: string,
storageType: CacheType,
defaultValue: T,
): T
/** 重载函数 getStorage */ /** 重载函数 getStorage */
export function getStorage<T>( function getStorage<T>(
key: string, key: string,
storageType?: CacheType, storageType?: CacheType,
defaultValue?: T, defaultValue?: T,
@ -59,7 +55,7 @@ export function getStorage<T>(
* @param key key * @param key key
* @returns * @returns
*/ */
export function getStorage<T>( function getStorage<T>(
key: string, key: string,
storageType: CacheType = 'sessionStorage', storageType: CacheType = 'sessionStorage',
defaultValue?: T, defaultValue?: T,
@ -91,7 +87,7 @@ export function getStorage<T>(
* - all-sessionStorage: 删除所有 sessionStorage * - all-sessionStorage: 删除所有 sessionStorage
* - all-localStorage: 删除所有 localStorage * - all-localStorage: 删除所有 localStorage
*/ */
export function removeStorage( function removeStorage(
key: string | 'all' | 'all-sessionStorage' | 'all-localStorage', key: string | 'all' | 'all-sessionStorage' | 'all-localStorage',
type: CacheType = 'sessionStorage', type: CacheType = 'sessionStorage',
) { ) {
@ -124,3 +120,5 @@ export function removeStorage(
: window.sessionStorage.removeItem(key) : window.sessionStorage.removeItem(key)
} }
} }
export { setStorage, getStorage, removeStorage }

View File

@ -237,8 +237,7 @@ export const colorToRgba = (color: string, alpha = 1) => {
* @remark 使 querySelectorAll * @remark 使 querySelectorAll
* @remark attribute , 'attr:xxx' * @remark attribute , 'attr:xxx'
* *
* : * @example
*
* class: * class:
* const el = queryElements('.demo') * const el = queryElements('.demo')
* id: * id:

37
src/utils/vue/call.ts Normal file
View File

@ -0,0 +1,37 @@
/* eslint-disable @typescript-eslint/no-explicit-any */
import type { AnyFunc, MaybeArray } from '@/types/modules/utils'
function call(funcs: MaybeArray<() => void>): void
function call<A1>(funcs: MaybeArray<(a1: A1) => void>, a1: A1): void
function call<A1, A2>(
funcs: MaybeArray<(a1: A1, a2: A2) => void>,
a1: A1,
a2: A2,
): void
function call<A1, A2, A3>(
funcs: MaybeArray<(a1: A1, a2: A2, a3: A3) => void>,
a1: A1,
a2: A2,
a3: A3,
): void
function call<A1, A2, A3, A4>(
funcs: MaybeArray<(a1: A1, a2: A2, a3: A3, a4: A4) => void>,
a1: A1,
a2: A2,
a3: A3,
a4: A4,
): void
function call<A extends any[]>(funcs: AnyFunc[] | AnyFunc, ...args: A) {
if (Array.isArray(funcs)) {
funcs.forEach((func) => (call as any)(func, ...args))
} else {
return funcs(...args)
}
}
export { call }

1
src/utils/vue/index.ts Normal file
View File

@ -0,0 +1 @@
export { call } from './call'

View File

@ -1,96 +1,154 @@
import './index.scss' import './index.scss'
import {
NCard, import { NCard, NLayout, NSpace, NInput, NButton } from 'naive-ui'
NLayout, import { getWeather, getTypicode } from '@use-api/test'
NDataTable, import { useRequest, useHookPlusRequest } from '@/axios/index'
NLayoutContent,
NLayoutHeader,
NSpace,
NInput,
NButton,
} from 'naive-ui'
import { onAxiosTest } from '@use-api/test'
import { isArray } from 'lodash-es'
const Axios = defineComponent({ const Axios = defineComponent({
name: 'RAxios', name: 'RAxios',
setup() { setup() {
const state = reactive({ const state = reactive({
weatherData: [] as UnknownObjectKey[], weatherData: [],
inputCityValue: '', inputCityValue: null,
throttleDemoInputValue: null,
debounceDemoInputValue: null,
weatherDemoInputValue: null,
}) })
const columns = [
{
title: '空气指数',
key: 'air',
},
{
title: '风速',
key: 'win_meter',
},
{
title: '能见度',
key: 'visibility',
},
{
title: '天气情况',
key: 'wea_day',
},
{
title: '提示',
key: 'air_tips',
},
]
const handleInputCityValue = async (value: string) => { const {
try { data: throttleDemoValue,
const cb = await onAxiosTest(value) loading: throttleDemoLoading,
run: throttleDemoRun,
state.weatherData = cb.data } = useHookPlusRequest(getTypicode, {
} catch (e) { throttleWait: 1000,
window.$message.error('请求已被取消')
}
}
onBeforeMount(async () => {
const cb = await onAxiosTest('成都')
state.weatherData = cb.data
}) })
const {
data: debounceDemoValue,
loading: debounceDemoLoading,
run: debounceDemoRun,
} = useHookPlusRequest(getTypicode, {
debounceWait: 1000,
})
const {
data: weatherDemoValue,
loading: weatherDemoLoading,
run: weatherDemoRun,
} = useHookPlusRequest(getWeather, {
throttleWait: 1000,
})
const {
data: demoData,
loading: demoLoading,
run: demoRun,
} = useRequest<{
title: string
}>(
{
url: 'https://jsonplaceholder.typicode.com/todos/1',
method: 'get',
},
{
manual: true,
},
)
return { return {
...toRefs(state), ...toRefs(state),
columns, throttleDemoValue,
handleInputCityValue, throttleDemoLoading,
throttleDemoRun,
debounceDemoValue,
debounceDemoLoading,
debounceDemoRun,
weatherDemoValue,
weatherDemoLoading,
weatherDemoRun,
demoData,
demoLoading,
demoRun,
} }
}, },
render() { render() {
return ( return (
<NLayout> <NLayout>
<NLayoutHeader bordered> <NSpace vertical>
<NCard title="请求函数"> <h1></h1>
axios <NCard>
<h2>useRequest</h2>
<p></p>
<h2>useHookPlusRequest</h2>
<p> <p>
=&gt; =&gt; 使3g网络 =&gt; promise axios
</p> </p>
</NCard> </NCard>
</NLayoutHeader> <h1>使 useRequest </h1>
<NLayoutHeader bordered> <NCard title="请求函数">
<NSpace class="axios-header__btn" align="center"> <h3>
<NInput 1. axios
v-model:value={this.inputCityValue} </h3>
onInput={this.handleInputCityValue.bind(this)} <h3>
placeholder="请输入城市" 2. =&gt; =&gt; 使3g网络 =&gt;
/>
<NButton onClick={this.handleInputCityValue.bind(this, '')}> </h3>
<h3>3.</h3>
</NButton> </NCard>
<NCard title="useRequest示例手动触发">
<NSpace vertical>
<NButton onClick={this.demoRun.bind(this)}></NButton>
<h3>
:&nbsp;
{this.demoLoading ? '获取中...' : this.demoData?.title}
</h3>
</NSpace>
</NCard>
<h1>使 useHookPlusRequest </h1>
<NCard title="节流">
<NSpace vertical>
<NInput
v-model:value={this.throttleDemoInputValue}
onUpdateValue={() => {
this.throttleDemoRun()
}}
/>
<h3></h3>
<h3>
:&nbsp;
{this.throttleDemoLoading ? '获取中...' : '获取成功!!!'}
</h3>
</NSpace>
</NCard>
<NCard title="防抖">
<NSpace vertical>
<NInput
v-model:value={this.debounceDemoInputValue}
onUpdateValue={() => {
this.debounceDemoRun()
}}
/>
<h3></h3>
<h3>
:&nbsp;
{this.debounceDemoLoading ? '获取中...' : '获取成功!!!'}
</h3>
</NSpace>
</NCard>
<NCard title="获取气候">
<NSpace vertical>
<NInput
v-model:value={this.weatherDemoInputValue}
onUpdateValue={(val) => {
this.weatherDemoRun(val)
}}
/>
<h3></h3>
<h3>
:&nbsp;
{this.weatherDemoLoading ? '获取中...' : '获取成功!!!'}
</h3>
</NSpace>
</NCard>
</NSpace> </NSpace>
</NLayoutHeader>
<NLayoutContent>
<NDataTable data={this.weatherData} columns={this.columns} />
</NLayoutContent>
</NLayout> </NLayout>
) )
}, },

View File

@ -26,6 +26,7 @@
"@use-micro/*": ["src/micro/*"] "@use-micro/*": ["src/micro/*"]
}, },
"suppressImplicitAnyIndexErrors": true, "suppressImplicitAnyIndexErrors": true,
"typeRoots": ["./src/types/app.d.ts", "./src/types/global.d.ts"],
"types": [ "types": [
"@intlify/unplugin-vue-i18n/messages", "@intlify/unplugin-vue-i18n/messages",
"naive-ui/volar", "naive-ui/volar",

View File

@ -1,6 +1,5 @@
import path from 'node:path' import path from 'node:path'
import autoImport from 'unplugin-auto-import/vite' // 自动导入
import unpluginViteComponents from 'unplugin-vue-components/vite' // 自动按需导入 import unpluginViteComponents from 'unplugin-vue-components/vite' // 自动按需导入
import vueI18nPlugin from '@intlify/unplugin-vue-i18n/vite' // i18n import vueI18nPlugin from '@intlify/unplugin-vue-i18n/vite' // i18n
import { createSvgIconsPlugin } from 'vite-plugin-svg-icons' // `svg icon` import { createSvgIconsPlugin } from 'vite-plugin-svg-icons' // `svg icon`
@ -27,25 +26,6 @@ export const viteSVGIcon = (options?: ViteSvgIconsPlugin) => {
return createSvgIconsPlugin(Object.assign({}, defaultOptions, options)) return createSvgIconsPlugin(Object.assign({}, defaultOptions, options))
} }
/**
*
* @param imp
* @returns auto import plugin
*
*
*/
export const viteAutoImport = async (imp: (ImportsMap | PresetName)[] = []) =>
autoImport({
include: [
/\.[tj]sx?$/, // .ts, .tsx, .js, .jsx
/\.vue$/,
/\.vue\?vue/, // .vue
/\.md$/, // .md
],
dts: true,
imports: ['vue', 'vue-router', 'pinia', '@vueuse/core', 'vue-i18n', ...imp],
})
/** /**
* *
* @param resolvers * @param resolvers

View File

@ -2,7 +2,6 @@ import { defineConfig } from 'vite'
import vue from '@vitejs/plugin-vue' import vue from '@vitejs/plugin-vue'
import { import {
viteAutoImport,
viteComponents, viteComponents,
viteVueI18nPlugin, viteVueI18nPlugin,
viteSVGIcon, viteSVGIcon,
@ -16,8 +15,10 @@ import vitePluginImp from 'vite-plugin-imp' // 按需打包工具
import { visualizer } from 'rollup-plugin-visualizer' // 打包体积分析工具 import { visualizer } from 'rollup-plugin-visualizer' // 打包体积分析工具
import viteCompression from 'vite-plugin-compression' // 压缩打包 import viteCompression from 'vite-plugin-compression' // 压缩打包
import { ViteEjsPlugin as viteEjsPlugin } from 'vite-plugin-ejs' import { ViteEjsPlugin as viteEjsPlugin } from 'vite-plugin-ejs'
import viteAutoImport from 'unplugin-auto-import/vite'
import { NaiveUiResolver } from 'unplugin-vue-components/resolvers' // 模板自动导入组件并且按需打包 import { NaiveUiResolver } from 'unplugin-vue-components/resolvers' // 模板自动导入组件并且按需打包
import { VueHooksPlusResolver } from '@vue-hooks-plus/resolvers'
import config from './cfg' import config from './cfg'
import pkg from './package.json' import pkg from './package.json'
@ -70,8 +71,21 @@ export default defineConfig(async ({ mode }) => {
viteVueJSX(), viteVueJSX(),
title, title,
viteInspect(), // 仅适用于开发模式(检查 `Vite` 插件的中间状态) viteInspect(), // 仅适用于开发模式(检查 `Vite` 插件的中间状态)
viteVeI18nPlugin(), viteVeI18nPlugin({}),
await viteAutoImport([ viteAutoImport({
include: [
/\.[tj]sx?$/, // .ts, .tsx, .js, .jsx
/\.vue$/,
/\.vue\?vue/, // .vue
/\.md$/, // .md
],
dts: true,
imports: [
'vue',
'vue-router',
'pinia',
'@vueuse/core',
'vue-i18n',
{ {
'naive-ui': [ 'naive-ui': [
'useDialog', 'useDialog',
@ -80,8 +94,10 @@ export default defineConfig(async ({ mode }) => {
'useLoadingBar', 'useLoadingBar',
], ],
}, },
]), ],
await viteComponents([NaiveUiResolver()]), resolvers: [NaiveUiResolver(), VueHooksPlusResolver()],
}),
await viteComponents([NaiveUiResolver(), VueHooksPlusResolver()]),
viteCompression(), viteCompression(),
viteVueI18nPlugin(), viteVueI18nPlugin(),
viteSvgLoader({ viteSvgLoader({