mirror of
https://github.com/XiaoDaiGua-Ray/ray-template.git
synced 2025-04-06 03:57:49 +08:00
v4.2.1发布
This commit is contained in:
parent
4d4d5451dc
commit
4960eb8175
13
CHANGELOG.md
13
CHANGELOG.md
@ -1,5 +1,18 @@
|
|||||||
# CHANGE LOG
|
# 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
|
## 4.2.0
|
||||||
|
|
||||||
针对分包,做了全局的重新设计、调整。让包的名称更加语义化;最重要的是,重新抽离了一些全局可能常用的方法,例如:useI18n、useDayjs 等,在以前这些方法存放于对应的包中,其实这样很不合理,所以现在统一存放于 `src/hooks` 包中。并且该包以后统一存放 `hooks` 方法,并不是 `utils` 方法,做了一个本质的区分,所以 `xxxCopilot.ts` 文件中的方法并不会移动,维持存放在原有的模块下。
|
针对分包,做了全局的重新设计、调整。让包的名称更加语义化;最重要的是,重新抽离了一些全局可能常用的方法,例如:useI18n、useDayjs 等,在以前这些方法存放于对应的包中,其实这样很不合理,所以现在统一存放于 `src/hooks` 包中。并且该包以后统一存放 `hooks` 方法,并不是 `utils` 方法,做了一个本质的区分,所以 `xxxCopilot.ts` 文件中的方法并不会移动,维持存放在原有的模块下。
|
||||||
|
2
cfg.ts
2
cfg.ts
@ -47,7 +47,7 @@ import type { BuildOptions } from 'vite'
|
|||||||
|
|
||||||
const config: AppConfigExport = {
|
const config: AppConfigExport = {
|
||||||
/** 公共基础路径配置, 如果为空则会默认以 '/' 填充 */
|
/** 公共基础路径配置, 如果为空则会默认以 '/' 填充 */
|
||||||
base: '/ray-template/',
|
// base: '/ray-template/',
|
||||||
/** 配置首屏加载信息 */
|
/** 配置首屏加载信息 */
|
||||||
preloadingConfig: PRE_LOADING_CONFIG,
|
preloadingConfig: PRE_LOADING_CONFIG,
|
||||||
/** 默认主题色(不可省略, 必填), 也用于 ejs 注入 */
|
/** 默认主题色(不可省略, 必填), 也用于 ejs 注入 */
|
||||||
|
@ -38,7 +38,6 @@
|
|||||||
"pinia": "^2.1.4",
|
"pinia": "^2.1.4",
|
||||||
"pinia-plugin-persistedstate": "^3.1.0",
|
"pinia-plugin-persistedstate": "^3.1.0",
|
||||||
"print-js": "^1.6.0",
|
"print-js": "^1.6.0",
|
||||||
"sass": "^1.54.3",
|
|
||||||
"screenfull": "^6.0.2",
|
"screenfull": "^6.0.2",
|
||||||
"vue": "^3.3.4",
|
"vue": "^3.3.4",
|
||||||
"vue-hooks-plus": "1.8.2",
|
"vue-hooks-plus": "1.8.2",
|
||||||
@ -81,11 +80,13 @@
|
|||||||
"postcss-px-to-viewport-8-plugin": "1.2.2",
|
"postcss-px-to-viewport-8-plugin": "1.2.2",
|
||||||
"prettier": "^2.7.1",
|
"prettier": "^2.7.1",
|
||||||
"rollup-plugin-visualizer": "^5.8.3",
|
"rollup-plugin-visualizer": "^5.8.3",
|
||||||
|
"sass": "1.54.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.15.0",
|
"unplugin-auto-import": "^0.15.0",
|
||||||
"unplugin-vue-components": "^0.25.1",
|
"unplugin-vue-components": "^0.25.1",
|
||||||
"vite": "^4.4.9",
|
"vite": "^4.4.9",
|
||||||
|
"vite-plugin-cdn2": "0.12.4",
|
||||||
"vite-plugin-compression": "^0.5.1",
|
"vite-plugin-compression": "^0.5.1",
|
||||||
"vite-plugin-ejs": "^1.6.4",
|
"vite-plugin-ejs": "^1.6.4",
|
||||||
"vite-plugin-eslint": "1.8.1",
|
"vite-plugin-eslint": "1.8.1",
|
||||||
|
@ -27,25 +27,25 @@ const RayLink = defineComponent({
|
|||||||
key: 'ray-js-note',
|
key: 'ray-js-note',
|
||||||
src: 'https://note.youdao.com/s/ObWEe2BB',
|
src: 'https://note.youdao.com/s/ObWEe2BB',
|
||||||
tooltip: 'Ray的前端学习笔记',
|
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',
|
key: 'ray-js-cover',
|
||||||
src: 'https://note.youdao.com/s/IC8xKPdB',
|
src: 'https://note.youdao.com/s/IC8xKPdB',
|
||||||
tooltip: 'Ray的面试题总结',
|
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',
|
key: 'ray-template-doc',
|
||||||
src: 'https://xiaodaigua-ray.github.io/ray-template-doc/',
|
src: 'https://xiaodaigua-ray.github.io/ray-template-doc/',
|
||||||
tooltip: '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',
|
key: 'ray-template-doc-out',
|
||||||
src: 'https://ray-template.yunkuangao.com/',
|
src: 'https://ray-template.yunkuangao.com/',
|
||||||
tooltip: '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',
|
||||||
},
|
},
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -114,7 +114,10 @@ export default defineComponent({
|
|||||||
|
|
||||||
const downloadQRCode = (fileName?: string) => {
|
const downloadQRCode = (fileName?: string) => {
|
||||||
if (qrcodeURL.value && isValueType<string>(qrcodeURL.value, 'String')) {
|
if (qrcodeURL.value && isValueType<string>(qrcodeURL.value, 'String')) {
|
||||||
downloadBase64File(qrcodeURL.value, fileName)
|
downloadBase64File(
|
||||||
|
qrcodeURL.value,
|
||||||
|
fileName || new Date().getTime() + '.png',
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -11,7 +11,6 @@
|
|||||||
|
|
||||||
import { combineDirective } from './helper/combine'
|
import { combineDirective } from './helper/combine'
|
||||||
import { forIn } from 'lodash-es'
|
import { forIn } from 'lodash-es'
|
||||||
import { isValueType } from '@/utils/hook'
|
|
||||||
|
|
||||||
import type { App } from 'vue'
|
import type { App } from 'vue'
|
||||||
import type { DirectiveModules } from '@/directives/type'
|
import type { DirectiveModules } from '@/directives/type'
|
||||||
@ -40,10 +39,7 @@ export const setupDirectives = (app: App<Element>) => {
|
|||||||
forIn(directivesModules, (value, key) => {
|
forIn(directivesModules, (value, key) => {
|
||||||
const dname = key.match(regexExtractDirectiveName)?.[0]
|
const dname = key.match(regexExtractDirectiveName)?.[0]
|
||||||
|
|
||||||
if (
|
if (typeof dname === 'string' && regexDirectiveName.test(dname)) {
|
||||||
isValueType<string>(dname, 'String') &&
|
|
||||||
regexDirectiveName.test(dname)
|
|
||||||
) {
|
|
||||||
app.directive(dname, value?.())
|
app.directive(dname, value?.())
|
||||||
} else {
|
} else {
|
||||||
console.error(`[setupDirectives] ${dname} is not a valid directive name`)
|
console.error(`[setupDirectives] ${dname} is not a valid directive name`)
|
||||||
|
@ -23,7 +23,7 @@ const copyDirective: CustomDirectiveFC<CopyElement, string> = () => {
|
|||||||
return {
|
return {
|
||||||
mounted: (el, { value }) => {
|
mounted: (el, { value }) => {
|
||||||
const clipboard = new ClipboardJS(el, {
|
const clipboard = new ClipboardJS(el, {
|
||||||
text: () => value,
|
text: () => String(value),
|
||||||
})
|
})
|
||||||
|
|
||||||
clipboard.on('success', () => {
|
clipboard.on('success', () => {
|
||||||
@ -40,7 +40,7 @@ const copyDirective: CustomDirectiveFC<CopyElement, string> = () => {
|
|||||||
el.$$clipboard?.destroy()
|
el.$$clipboard?.destroy()
|
||||||
|
|
||||||
el.$$clipboard = new ClipboardJS(el, {
|
el.$$clipboard = new ClipboardJS(el, {
|
||||||
text: () => value,
|
text: () => String(value),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -33,7 +33,7 @@ const debounceDirective: CustomDirectiveFC<
|
|||||||
const { func, trigger = 'click', wait = 500, options } = value
|
const { func, trigger = 'click', wait = 500, options } = value
|
||||||
|
|
||||||
if (typeof func !== 'function') {
|
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))
|
debounceFunction = debounce(func, wait, Object.assign({}, options))
|
||||||
|
@ -3,7 +3,7 @@ import type { AnyFC } from '@/types/modules/utils'
|
|||||||
|
|
||||||
export interface DebounceBindingOptions {
|
export interface DebounceBindingOptions {
|
||||||
func: AnyFC
|
func: AnyFC
|
||||||
trigger: string
|
trigger?: string
|
||||||
wait: number
|
wait?: number
|
||||||
options: DebounceSettings
|
options?: DebounceSettings
|
||||||
}
|
}
|
||||||
|
@ -33,7 +33,7 @@ const throttleDirective: CustomDirectiveFC<
|
|||||||
const { func, trigger = 'click', wait = 500, options } = value
|
const { func, trigger = 'click', wait = 500, options } = value
|
||||||
|
|
||||||
if (typeof func !== 'function') {
|
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))
|
throttleFunction = throttle(func, wait, Object.assign({}, options))
|
||||||
|
@ -3,7 +3,7 @@ import type { AnyFC } from '@/types/modules/utils'
|
|||||||
|
|
||||||
export interface ThrottleBindingOptions {
|
export interface ThrottleBindingOptions {
|
||||||
func: AnyFC
|
func: AnyFC
|
||||||
trigger: string
|
trigger?: string
|
||||||
wait: number
|
wait?: number
|
||||||
options: ThrottleSettings
|
options?: ThrottleSettings
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,9 @@
|
|||||||
import type { Directive } from 'vue'
|
import type { Directive } from 'vue'
|
||||||
import type { App } 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 type CustomDirectiveFC<T, K> = () => Directive<T, K>
|
||||||
|
|
||||||
export interface DirectiveModules extends Object {
|
export interface DirectiveModules extends Object {
|
||||||
|
@ -70,9 +70,7 @@ const ContentWrapper = defineComponent({
|
|||||||
class="content-wrapper"
|
class="content-wrapper"
|
||||||
transitionPropName={this.contentTransition + '-transform'}
|
transitionPropName={this.contentTransition + '-transform'}
|
||||||
/>
|
/>
|
||||||
) : (
|
) : null}
|
||||||
''
|
|
||||||
)}
|
|
||||||
</NSpin>
|
</NSpin>
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
|
@ -36,6 +36,7 @@ import { useI18n } from '@/hooks/web/index'
|
|||||||
import { getAppRawRoutes } from '@/router/appRouteModules'
|
import { getAppRawRoutes } from '@/router/appRouteModules'
|
||||||
import { useKeepAlive } from '@/store'
|
import { useKeepAlive } from '@/store'
|
||||||
import { useVueRouter } from '@/hooks/web/index'
|
import { useVueRouter } from '@/hooks/web/index'
|
||||||
|
import { throttle } from 'lodash-es'
|
||||||
|
|
||||||
import type { AppRouteMeta, AppRouteRecordRaw } from '@/router/type'
|
import type { AppRouteMeta, AppRouteRecordRaw } from '@/router/type'
|
||||||
import type {
|
import type {
|
||||||
@ -354,7 +355,7 @@ export const useMenu = defineStore(
|
|||||||
|
|
||||||
return {
|
return {
|
||||||
...toRefs(menuState),
|
...toRefs(menuState),
|
||||||
changeMenuModelValue,
|
changeMenuModelValue: throttle(changeMenuModelValue, 500),
|
||||||
setupAppMenu,
|
setupAppMenu,
|
||||||
collapsedMenu,
|
collapsedMenu,
|
||||||
spliceMenTagOptions,
|
spliceMenTagOptions,
|
||||||
|
@ -54,7 +54,7 @@ export const useSignin = defineStore(
|
|||||||
role: 'admin',
|
role: 'admin',
|
||||||
name: signinForm.name,
|
name: signinForm.name,
|
||||||
avatar:
|
avatar:
|
||||||
'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.jpeg',
|
'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.navigator.png',
|
||||||
}
|
}
|
||||||
|
|
||||||
resolve({
|
resolve({
|
||||||
|
@ -95,4 +95,5 @@ body .ray-template__directive--disabled {
|
|||||||
opacity: 0.3 !important;
|
opacity: 0.3 !important;
|
||||||
pointer-events: none !important;
|
pointer-events: none !important;
|
||||||
cursor: not-allowed !important;
|
cursor: not-allowed !important;
|
||||||
|
filter: brightness(0.88);
|
||||||
}
|
}
|
||||||
|
@ -40,11 +40,11 @@ export const arrayBufferToBase64Image = (data: ArrayBuffer): string | null => {
|
|||||||
*
|
*
|
||||||
* @remark 下载 base64 文件
|
* @remark 下载 base64 文件
|
||||||
*/
|
*/
|
||||||
export const downloadBase64File = (base64: string, fileName?: string) => {
|
export const downloadBase64File = (base64: string, fileName: string) => {
|
||||||
const link = document.createElement('a')
|
const link = document.createElement('a')
|
||||||
|
|
||||||
link.href = base64
|
link.href = base64
|
||||||
link.download = fileName || new Date().getTime() + '.png'
|
link.download = fileName
|
||||||
|
|
||||||
link.click()
|
link.click()
|
||||||
}
|
}
|
||||||
|
@ -21,6 +21,10 @@ import {
|
|||||||
} from 'naive-ui'
|
} from 'naive-ui'
|
||||||
|
|
||||||
import type { ConditionalPick } from '@/types/modules/helper'
|
import type { ConditionalPick } from '@/types/modules/helper'
|
||||||
|
import type {
|
||||||
|
DebounceBindingOptions,
|
||||||
|
ThrottleBindingOptions,
|
||||||
|
} from '@/directives/type'
|
||||||
|
|
||||||
const RDirective = defineComponent({
|
const RDirective = defineComponent({
|
||||||
name: 'RDirective',
|
name: 'RDirective',
|
||||||
|
@ -26,13 +26,16 @@ import mockDevServerPlugin from 'vite-plugin-mock-dev-server'
|
|||||||
import vueI18nPlugin from '@intlify/unplugin-vue-i18n/vite'
|
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 { NaiveUiResolver } from 'unplugin-vue-components/resolvers'
|
import { NaiveUiResolver } from 'unplugin-vue-components/resolvers'
|
||||||
import { VueHooksPlusResolver } from '@vue-hooks-plus/resolvers'
|
import { VueHooksPlusResolver } from '@vue-hooks-plus/resolvers'
|
||||||
|
|
||||||
import config from './cfg'
|
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
|
const { title, appPrimaryColor, preloadingConfig } = config
|
||||||
|
|
||||||
return [
|
return [
|
||||||
@ -161,5 +164,24 @@ export default function (mode: string) {
|
|||||||
inject: 'body-last',
|
inject: 'body-last',
|
||||||
customDomId: '__svg__icons__dom__',
|
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',
|
||||||
|
],
|
||||||
|
}),
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user