v4.2.1发布

This commit is contained in:
XiaoDaiGua-Ray 2023-09-22 21:50:24 +08:00
parent 4d4d5451dc
commit 4960eb8175
19 changed files with 72 additions and 30 deletions

View File

@ -1,5 +1,18 @@
# CHANGE LOG
## 4.2.1
经过综合考虑,还是给模板增加 `cdn` 的配置。基于 `vite-plugin-cdn2` 插件实现。
### Feats
- 指令相关
- `v-copy` 指令将使用 `String` 强制转换传入的值
- 统一暴露节流、防抖指令的配置项类型 `import type { DebounceBindingOptions, ThrottleBindingOptions } from '@/directives/type'`
- 现在 `v-disabled` 指令生效时会降低一点元素的亮度
- `changeMenuModelValue` 方法添加节流锁,避免重复刷新 url 导致的一些问题
- 新增 `cdn`,缩减构建体积。如果不需要该配置,搜索 `viteCDNPlugin` 注释即可
## 4.2.0
针对分包做了全局的重新设计、调整。让包的名称更加语义化最重要的是重新抽离了一些全局可能常用的方法例如useI18n、useDayjs 等,在以前这些方法存放于对应的包中,其实这样很不合理,所以现在统一存放于 `src/hooks` 包中。并且该包以后统一存放 `hooks` 方法,并不是 `utils` 方法,做了一个本质的区分,所以 `xxxCopilot.ts` 文件中的方法并不会移动,维持存放在原有的模块下。

2
cfg.ts
View File

@ -47,7 +47,7 @@ import type { BuildOptions } from 'vite'
const config: AppConfigExport = {
/** 公共基础路径配置, 如果为空则会默认以 '/' 填充 */
base: '/ray-template/',
// base: '/ray-template/',
/** 配置首屏加载信息 */
preloadingConfig: PRE_LOADING_CONFIG,
/** 默认主题色(不可省略, 必填), 也用于 ejs 注入 */

View File

@ -38,7 +38,6 @@
"pinia": "^2.1.4",
"pinia-plugin-persistedstate": "^3.1.0",
"print-js": "^1.6.0",
"sass": "^1.54.3",
"screenfull": "^6.0.2",
"vue": "^3.3.4",
"vue-hooks-plus": "1.8.2",
@ -81,11 +80,13 @@
"postcss-px-to-viewport-8-plugin": "1.2.2",
"prettier": "^2.7.1",
"rollup-plugin-visualizer": "^5.8.3",
"sass": "1.54.3",
"svg-sprite-loader": "^6.0.11",
"typescript": "^5.0.2",
"unplugin-auto-import": "^0.15.0",
"unplugin-vue-components": "^0.25.1",
"vite": "^4.4.9",
"vite-plugin-cdn2": "0.12.4",
"vite-plugin-compression": "^0.5.1",
"vite-plugin-ejs": "^1.6.4",
"vite-plugin-eslint": "1.8.1",

View File

@ -27,25 +27,25 @@ const RayLink = defineComponent({
key: 'ray-js-note',
src: 'https://note.youdao.com/s/ObWEe2BB',
tooltip: 'Ray的前端学习笔记',
icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.jpeg',
icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.navigator.png',
},
{
key: 'ray-js-cover',
src: 'https://note.youdao.com/s/IC8xKPdB',
tooltip: 'Ray的面试题总结',
icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.jpeg',
icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.navigator.png',
},
{
key: 'ray-template-doc',
src: 'https://xiaodaigua-ray.github.io/ray-template-doc/',
tooltip: 'Ray Template Doc',
icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.jpeg',
icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.navigator.png',
},
{
key: 'ray-template-doc-out',
src: 'https://ray-template.yunkuangao.com/',
tooltip: 'Ray Template Doc (国内地址)',
icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.jpeg',
icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.navigator.png',
},
]

View File

@ -114,7 +114,10 @@ export default defineComponent({
const downloadQRCode = (fileName?: string) => {
if (qrcodeURL.value && isValueType<string>(qrcodeURL.value, 'String')) {
downloadBase64File(qrcodeURL.value, fileName)
downloadBase64File(
qrcodeURL.value,
fileName || new Date().getTime() + '.png',
)
}
}

View File

@ -11,7 +11,6 @@
import { combineDirective } from './helper/combine'
import { forIn } from 'lodash-es'
import { isValueType } from '@/utils/hook'
import type { App } from 'vue'
import type { DirectiveModules } from '@/directives/type'
@ -40,10 +39,7 @@ export const setupDirectives = (app: App<Element>) => {
forIn(directivesModules, (value, key) => {
const dname = key.match(regexExtractDirectiveName)?.[0]
if (
isValueType<string>(dname, 'String') &&
regexDirectiveName.test(dname)
) {
if (typeof dname === 'string' && regexDirectiveName.test(dname)) {
app.directive(dname, value?.())
} else {
console.error(`[setupDirectives] ${dname} is not a valid directive name`)

View File

@ -23,7 +23,7 @@ const copyDirective: CustomDirectiveFC<CopyElement, string> = () => {
return {
mounted: (el, { value }) => {
const clipboard = new ClipboardJS(el, {
text: () => value,
text: () => String(value),
})
clipboard.on('success', () => {
@ -40,7 +40,7 @@ const copyDirective: CustomDirectiveFC<CopyElement, string> = () => {
el.$$clipboard?.destroy()
el.$$clipboard = new ClipboardJS(el, {
text: () => value,
text: () => String(value),
})
}
},

View File

@ -33,7 +33,7 @@ const debounceDirective: CustomDirectiveFC<
const { func, trigger = 'click', wait = 500, options } = value
if (typeof func !== 'function') {
throw new Error('debounce directive value must be a function')
throw new TypeError('debounce directive value must be a function')
}
debounceFunction = debounce(func, wait, Object.assign({}, options))

View File

@ -3,7 +3,7 @@ import type { AnyFC } from '@/types/modules/utils'
export interface DebounceBindingOptions {
func: AnyFC
trigger: string
wait: number
options: DebounceSettings
trigger?: string
wait?: number
options?: DebounceSettings
}

View File

@ -33,7 +33,7 @@ const throttleDirective: CustomDirectiveFC<
const { func, trigger = 'click', wait = 500, options } = value
if (typeof func !== 'function') {
throw new Error('throttle directive value must be a function')
throw new TypeError('throttle directive value must be a function')
}
throttleFunction = throttle(func, wait, Object.assign({}, options))

View File

@ -3,7 +3,7 @@ import type { AnyFC } from '@/types/modules/utils'
export interface ThrottleBindingOptions {
func: AnyFC
trigger: string
wait: number
options: ThrottleSettings
trigger?: string
wait?: number
options?: ThrottleSettings
}

View File

@ -1,6 +1,9 @@
import type { Directive } from 'vue'
import type { App } from 'vue'
export type { DebounceBindingOptions } from './modules/debounce/type'
export type { ThrottleBindingOptions } from './modules/throttle/type'
export type CustomDirectiveFC<T, K> = () => Directive<T, K>
export interface DirectiveModules extends Object {

View File

@ -70,9 +70,7 @@ const ContentWrapper = defineComponent({
class="content-wrapper"
transitionPropName={this.contentTransition + '-transform'}
/>
) : (
''
)}
) : null}
</NSpin>
)
},

View File

@ -36,6 +36,7 @@ import { useI18n } from '@/hooks/web/index'
import { getAppRawRoutes } from '@/router/appRouteModules'
import { useKeepAlive } from '@/store'
import { useVueRouter } from '@/hooks/web/index'
import { throttle } from 'lodash-es'
import type { AppRouteMeta, AppRouteRecordRaw } from '@/router/type'
import type {
@ -354,7 +355,7 @@ export const useMenu = defineStore(
return {
...toRefs(menuState),
changeMenuModelValue,
changeMenuModelValue: throttle(changeMenuModelValue, 500),
setupAppMenu,
collapsedMenu,
spliceMenTagOptions,

View File

@ -54,7 +54,7 @@ export const useSignin = defineStore(
role: 'admin',
name: signinForm.name,
avatar:
'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.jpeg',
'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.navigator.png',
}
resolve({

View File

@ -95,4 +95,5 @@ body .ray-template__directive--disabled {
opacity: 0.3 !important;
pointer-events: none !important;
cursor: not-allowed !important;
filter: brightness(0.88);
}

View File

@ -40,11 +40,11 @@ export const arrayBufferToBase64Image = (data: ArrayBuffer): string | null => {
*
* @remark base64
*/
export const downloadBase64File = (base64: string, fileName?: string) => {
export const downloadBase64File = (base64: string, fileName: string) => {
const link = document.createElement('a')
link.href = base64
link.download = fileName || new Date().getTime() + '.png'
link.download = fileName
link.click()
}

View File

@ -21,6 +21,10 @@ import {
} from 'naive-ui'
import type { ConditionalPick } from '@/types/modules/helper'
import type {
DebounceBindingOptions,
ThrottleBindingOptions,
} from '@/directives/type'
const RDirective = defineComponent({
name: 'RDirective',

View File

@ -26,13 +26,16 @@ 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'
import { NaiveUiResolver } from 'unplugin-vue-components/resolvers'
import { VueHooksPlusResolver } from '@vue-hooks-plus/resolvers'
import config from './cfg'
export default function (mode: string) {
import type { PluginOption } from 'vite'
export default function (mode: string): PluginOption[] {
const { title, appPrimaryColor, preloadingConfig } = config
return [
@ -161,5 +164,24 @@ export default function (mode: string) {
inject: 'body-last',
customDomId: '__svg__icons__dom__',
}),
viteCDNPlugin({
modules: [
'vue',
'vue-demi',
'pinia',
'naive-ui',
'vue-router',
'vue-i18n',
'dayjs',
'echarts',
'vuedraggable',
'xlsx',
'axios',
'screenfull',
'print-js',
'clipboard',
'lodash-es',
],
}),
]
}