version: 4.4.7

This commit is contained in:
XiaoDaiGua-Ray 2023-12-22 21:50:25 +08:00
parent f5ae01f33a
commit d98a42380d
15 changed files with 463 additions and 396 deletions

View File

@ -1,5 +1,27 @@
# CHANGE LOG # CHANGE LOG
## 4.4.7
## Feats
- 更新 `vue` 版本至 `3.3.13` 版本
- 更新 `naive-ui` 版本至 `3.36.0`。新增了几个新组件
- 更新 `vite` 版本至 `5.0.10`
- `appConfig` 相关
- 移除 `APP_WATERMARK_CONFIG` 配置项,现在水印配置项会被 `watermarkConfig` 替代
- 移除 `ROOT_ROUTE` 配置项,现在根路由配置项会被 `appRootRoute` 替代
- `variable` 相关
- 移除 `variable` 管理 `ROO_ROUTE` 配置项
- `commit-message` 新增 `plugin` 更新相关前缀配置
- `RQRCode` 组件
- 调整 `loading` 透明度
- `downloadQRCode` 方法将会返回一个 `Promise` 对象
- `basic`
- `downloadAnyFile` 方法将会返回一个 `Promise` 对象
- `types`
- 新增 `ReturnPromiseType` 工具类型,用于获取函数返回值的 `Promise` 类型
- 新增 `ConditionalExclude` 工具类型,用于条件排除指定类型
## 4.4.6 ## 4.4.6
## Feats ## Feats

View File

@ -8,6 +8,7 @@
// style: 代码格式(不影响功能,例如空格、分号等格式修正) | Code format (no functional impact, such as space, semicolon, etc.) // style: 代码格式(不影响功能,例如空格、分号等格式修正) | Code format (no functional impact, such as space, semicolon, etc.)
// version: 更新迭代 package.json 版本号 | Update the package.json version number // version: 更新迭代 package.json 版本号 | Update the package.json version number
// build: 构建 | Build // build: 构建 | Build
// plugin: 更新插件版本 | Update plugin version
module.exports = { module.exports = {
ignores: [(commit) => commit.includes('init')], ignores: [(commit) => commit.includes('init')],
@ -33,6 +34,7 @@ module.exports = {
'style', 'style',
'version', 'version',
'build', 'build',
'plugin',
], ],
], ],
}, },

View File

@ -1,7 +1,7 @@
{ {
"name": "ray-template", "name": "ray-template",
"private": false, "private": false,
"version": "4.4.6", "version": "4.4.7",
"type": "module", "type": "module",
"engines": { "engines": {
"node": "^18.0.0 || >=20.0.0", "node": "^18.0.0 || >=20.0.0",
@ -44,11 +44,11 @@
"interactjs": "1.10.26", "interactjs": "1.10.26",
"lodash-es": "^4.17.21", "lodash-es": "^4.17.21",
"mockjs": "1.1.0", "mockjs": "1.1.0",
"naive-ui": "^2.35.0", "naive-ui": "^2.36.0",
"pinia": "^2.1.7", "pinia": "^2.1.7",
"pinia-plugin-persistedstate": "^3.2.0", "pinia-plugin-persistedstate": "^3.2.0",
"print-js": "^1.6.0", "print-js": "^1.6.0",
"vue": "^3.3.11", "vue": "^3.3.13",
"vue-hooks-plus": "1.8.5", "vue-hooks-plus": "1.8.5",
"vue-i18n": "^9.8.0", "vue-i18n": "^9.8.0",
"vue-router": "^4.2.5", "vue-router": "^4.2.5",
@ -93,7 +93,7 @@
"typescript": "^5.2.2", "typescript": "^5.2.2",
"unplugin-auto-import": "^0.16.6", "unplugin-auto-import": "^0.16.6",
"unplugin-vue-components": "^0.25.2", "unplugin-vue-components": "^0.25.2",
"vite": "^5.0.8", "vite": "^5.0.10",
"vite-plugin-cdn2": "0.15.2", "vite-plugin-cdn2": "0.15.2",
"vite-plugin-compression": "^0.5.1", "vite-plugin-compression": "^0.5.1",
"vite-plugin-ejs": "^1.7.0", "vite-plugin-ejs": "^1.7.0",

655
pnpm-lock.yaml generated

File diff suppressed because it is too large Load Diff

View File

@ -11,11 +11,7 @@
/** 系统配置 */ /** 系统配置 */
import type { import type { LayoutSideBarLogo, PreloadingConfig } from '@/types/modules/cfg'
LayoutSideBarLogo,
PreloadingConfig,
RootRoute,
} from '@/types/modules/cfg'
import type { AppMenuConfig, AppKeepAlive } from '@/types/modules/appConfig' import type { AppMenuConfig, AppKeepAlive } from '@/types/modules/appConfig'
/** /**
@ -44,21 +40,6 @@ export const PRE_LOADING_CONFIG: PreloadingConfig = {
titleColor: '#2d8cf0', titleColor: '#2d8cf0',
} }
/**
*
*
* ROOT_ROUTE Layout Root Path path
*
* globalRootRoute
*
*
* Root Route 使 useAppRoot
*/
export const ROOT_ROUTE: RootRoute = {
name: 'Dashboard',
path: '/dashboard',
}
/** /**
* *
* icon: LOGO , `RIcon` () * icon: LOGO , `RIcon` ()
@ -111,19 +92,3 @@ export const APP_CATCH_KEY = {
localeLanguage: 'localeLanguage', localeLanguage: 'localeLanguage',
token: 'token', token: 'token',
} as const } as const
/**
*
*
* 具体配置信息查看官网: https://www.naiveui.com/zh-CN/dark/components/watermark#API
*/
export const APP_WATERMARK_CONFIG = {
content: 'Trying be better~',
fontSize: 16,
lineHeight: 16,
width: 384,
height: 384,
xOffset: 12,
yOffset: 60,
rotate: -15,
} as const

View File

@ -60,7 +60,7 @@ export default defineComponent({
const qrcodeURL = ref<QRCodeRenderResponse>() const qrcodeURL = ref<QRCodeRenderResponse>()
const spinOverrides = { const spinOverrides = {
opacitySpinning: '0.1', opacitySpinning: '0.01',
} }
let gifBuffer: GIFBuffer let gifBuffer: GIFBuffer
let watchCallback!: WatchStopHandle let watchCallback!: WatchStopHandle
@ -119,10 +119,12 @@ export default defineComponent({
const downloadQRCode = (fileName?: DownloadFilenameType) => { const downloadQRCode = (fileName?: DownloadFilenameType) => {
if (qrcodeURL.value && isValueType<string>(qrcodeURL.value, 'String')) { if (qrcodeURL.value && isValueType<string>(qrcodeURL.value, 'String')) {
downloadAnyFile( return downloadAnyFile(
qrcodeURL.value, qrcodeURL.value,
fileName || new Date().getTime() + '.png', fileName || new Date().getTime() + '.png',
) )
} else {
return Promise.reject()
} }
} }

View File

@ -27,11 +27,7 @@
* createVariableState({ your state }) * createVariableState({ your state })
*/ */
import { ROOT_ROUTE, APP_WATERMARK_CONFIG } from '@/app-config/appConfig'
import { cloneDeep } from 'lodash-es'
import type { AnyFC } from '@/types/modules/utils' import type { AnyFC } from '@/types/modules/utils'
import type { Mutable } from '@/types/modules/helper'
/** /**
* *
@ -47,7 +43,6 @@ const variableState = reactive({
globalDrawerValue: false, // 全局抽屉控制器(小尺寸设备可用) globalDrawerValue: false, // 全局抽屉控制器(小尺寸设备可用)
globalMainLayoutLoad: true, // LayoutContent 区域加载控制器,会触发强制刷新 globalMainLayoutLoad: true, // LayoutContent 区域加载控制器,会触发强制刷新
layoutContentMaximize: false, // LayoutContent 区域全屏控制器 layoutContentMaximize: false, // LayoutContent 区域全屏控制器
globalRootRoute: cloneDeep(ROOT_ROUTE), // 全局根路由配置,同步至 ROOT_ROUTE
layoutContentSpinning: false, // LayoutContent 区域加载控制器,不会触发强制刷新 layoutContentSpinning: false, // LayoutContent 区域加载控制器,不会触发强制刷新
}) })

View File

@ -9,29 +9,29 @@
* @remark * @remark
*/ */
import { setVariable, getVariableToRefs } from '@/global-variable' import { useSettingGetters, useSettingActions } from '@/store'
import { cloneDeep } from 'lodash-es'
import type { DeepMutable } from '@/types/modules/helper' import type { AppRootRoute } from '@/store/modules/setting/type'
export function useAppRoot() { export function useAppRoot() {
const globalRootRoute = getVariableToRefs('globalRootRoute') const { getAppRootRoute } = useSettingGetters()
const { updateSettingState } = useSettingActions()
/** /**
* *
* @remark * @remark
*/ */
const getRootRoute = computed(() => globalRootRoute.value) const getRootRoute = getAppRootRoute
/** /**
* *
* @remark path * @remark path
*/ */
const getRootPath = computed(() => globalRootRoute.value.path) const getRootPath = computed(() => getAppRootRoute.value.path)
/** /**
* *
* @remark name * @remark name
*/ */
const getRootName = computed(() => globalRootRoute.value.name) const getRootName = computed(() => getAppRootRoute.value.name)
/** /**
* *
@ -42,11 +42,11 @@ export function useAppRoot() {
* @example * @example
* setRootRoute({ path: '/your root path', name: 'your root name' }) * setRootRoute({ path: '/your root path', name: 'your root name' })
*/ */
const setRootRoute = (route: DeepMutable<typeof globalRootRoute.value>) => { const setRootRoute = (route: AppRootRoute) => {
const routeRef = getVariableToRefs('globalRootRoute') updateSettingState(
const assignRoute = Object.assign(cloneDeep(routeRef.value), route) 'appRootRoute',
Object.assign({}, getAppRootRoute.value, route),
setVariable('globalRootRoute', assignRoute) )
} }
return { return {

View File

@ -17,6 +17,10 @@
import { useWindowSize } from '@vueuse/core' import { useWindowSize } from '@vueuse/core'
import { watchEffectWithTarget } from '@/utils' import { watchEffectWithTarget } from '@/utils'
import type { UseWindowSizeOptions } from '@vueuse/core'
export interface UseDeviceOptions extends UseWindowSizeOptions {}
/** /**
* *
* *
@ -28,8 +32,8 @@ import { watchEffectWithTarget } from '@/utils'
* isTabletOrSmaller.value => true // 当前尺寸为平板或者更小 * isTabletOrSmaller.value => true // 当前尺寸为平板或者更小
* isTabletOrSmaller.value => false // 当前尺寸为桌面或者更大 * isTabletOrSmaller.value => false // 当前尺寸为桌面或者更大
*/ */
export function useDevice() { export function useDevice(options?: UseDeviceOptions) {
const { width, height } = useWindowSize() const { width, height } = useWindowSize(options)
const isTabletOrSmaller = ref(false) const isTabletOrSmaller = ref(false)
const update = () => { const update = () => {

View File

@ -102,10 +102,11 @@ export const useDomToImage = <T extends HTMLElement>(
created, created,
createdError, createdError,
finally: _finally, finally: _finally,
imageType: _imageType,
} = options ?? {} } = options ?? {}
const run = ( const run = (
imageType: UseDomToImageOptions['imageType'] = 'jpeg', imageType?: UseDomToImageOptions['imageType'],
): Promise<DomToImageResult> => { ): Promise<DomToImageResult> => {
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
const element = unrefElement(target) const element = unrefElement(target)
@ -118,10 +119,7 @@ export const useDomToImage = <T extends HTMLElement>(
return reject('useDomToImage: element is undefined.') return reject('useDomToImage: element is undefined.')
} }
const type = imageType ?? options?.imageType domToImageMethods[imageType ?? _imageType ?? 'jpeg']?.(element, options)
const matchFc = domToImageMethods[type] || domToImageMethods['jpeg']
matchFc(element, options)
.then((res) => { .then((res) => {
created?.(res, element) created?.(res, element)

View File

@ -71,6 +71,12 @@ export const useSettingGetters = () => {
*/ */
const getWatermarkConfig = computed(() => variable.watermarkConfig) const getWatermarkConfig = computed(() => variable.watermarkConfig)
/**
*
* @remark app
*/
const getAppRootRoute = computed(() => variable.appRootRoute)
return { return {
getDrawerPlacement, getDrawerPlacement,
getPrimaryColorOverride, getPrimaryColorOverride,
@ -83,6 +89,7 @@ export const useSettingGetters = () => {
getContentTransition, getContentTransition,
getWatermarkSwitch, getWatermarkSwitch,
getWatermarkConfig, getWatermarkConfig,
getAppRootRoute,
} }
} }

View File

@ -4,8 +4,6 @@ import { colorToRgba, setStorage } from '@/utils'
import { useI18n } from '@/hooks/web' import { useI18n } from '@/hooks/web'
import { APP_THEME } from '@/app-config/designConfig' import { APP_THEME } from '@/app-config/designConfig'
import { useDayjs } from '@/hooks/web' import { useDayjs } from '@/hooks/web'
import { APP_WATERMARK_CONFIG } from '@/app-config/appConfig'
import { cloneDeep } from 'lodash-es'
import type { SettingState } from '@/store/modules/setting/type' import type { SettingState } from '@/store/modules/setting/type'
import type { LocalKey } from '@/hooks/web' import type { LocalKey } from '@/hooks/web'
@ -37,7 +35,20 @@ export const piniaSettingStore = defineStore(
copyrightSwitch: true, // 底部区域开关 copyrightSwitch: true, // 底部区域开关
contentTransition: 'scale', // 切换过渡效果 contentTransition: 'scale', // 切换过渡效果
watermarkSwitch: false, // 水印开关, watermarkSwitch: false, // 水印开关,
watermarkConfig: cloneDeep(APP_WATERMARK_CONFIG), watermarkConfig: {
content: 'Trying be better~',
fontSize: 16,
lineHeight: 16,
width: 384,
height: 384,
xOffset: 12,
yOffset: 60,
rotate: -15,
},
appRootRoute: {
name: 'Dashboard',
path: '/dashboard',
},
}) })
/** 修改当前语言 */ /** 修改当前语言 */

View File

@ -1,6 +1,21 @@
import type { GlobalThemeOverrides } from 'naive-ui' import type { GlobalThemeOverrides } from 'naive-ui'
import type { Placement } from '@/types/modules/component' import type { Placement } from '@/types/modules/component'
import type { APP_WATERMARK_CONFIG } from '@/app-config/appConfig'
export interface WatermarkConfig {
content: string
fontSize: number
lineHeight: number
width: number
height: number
xOffset: number
yOffset: number
rotate: number
}
export interface AppRootRoute {
name: string
path: string
}
export interface SettingState { export interface SettingState {
drawerPlacement: Placement drawerPlacement: Placement
@ -13,5 +28,6 @@ export interface SettingState {
watermarkSwitch: boolean watermarkSwitch: boolean
copyrightSwitch: boolean copyrightSwitch: boolean
contentTransition: string contentTransition: string
watermarkConfig: typeof APP_WATERMARK_CONFIG watermarkConfig: WatermarkConfig
appRootRoute: AppRootRoute
} }

View File

@ -13,6 +13,18 @@ export type ConditionalKeys<Base, Condition> = NonNullable<
}[keyof Base] }[keyof Base]
> >
/**
*
*
*
* @example
* ConditionalExclude<{ a: string, b: number }, string> // { b: number }
*/
export type ConditionalExclude<Base, Condition> = Omit<
Base,
ConditionalKeys<Base, Condition>
>
/** /**
* *
* *
@ -68,3 +80,17 @@ export type DeepMutable<T> = {
? DeepMutable<T[P]> ? DeepMutable<T[P]>
: T[P] : T[P]
} }
/**
*
* Promise
*
* @example
* ReturnPromiseType<Promise<string>> // string
* ReturnPromiseType<Promise<string> | Promise<number>> // string | number
*/
export type ReturnPromiseType<T extends Promise<any>> = T extends Promise<
infer U
>
? U
: never

View File

@ -150,9 +150,7 @@ export const downloadAnyFile = (
try { try {
if (typeof data === 'string') { if (typeof data === 'string') {
downloadBase64File(data, fileName) downloadBase64File(data, fileName)
resolve() return resolve()
return
} }
if (data instanceof ArrayBuffer) { if (data instanceof ArrayBuffer) {
@ -162,9 +160,7 @@ export const downloadAnyFile = (
} else if (data instanceof File || data instanceof Blob) { } else if (data instanceof File || data instanceof Blob) {
blobData = data blobData = data
} else { } else {
reject(new Error('downloadAnyFile: Unsupported data type.')) return reject(new Error('downloadAnyFile: Unsupported data type.'))
return
} }
const url = URL.createObjectURL(blobData) const url = URL.createObjectURL(blobData)
@ -181,18 +177,18 @@ export const downloadAnyFile = (
link.addEventListener('load', () => { link.addEventListener('load', () => {
remove() remove()
resolve() return resolve()
}) })
link.addEventListener('error', (error) => { link.addEventListener('error', (error) => {
remove() remove()
reject(error) return reject(error)
}) })
document.body.appendChild(link) document.body.appendChild(link)
link.click() link.click()
} catch (error) { } catch (error) {
reject(error) return reject(error)
} }
}) })
} }