This commit is contained in:
XiaoDaiGua-Ray 2023-08-09 16:43:59 +08:00
parent be406cc026
commit 5eb201b25b
29 changed files with 320 additions and 100 deletions

View File

@ -8,5 +8,6 @@
"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" "typescript.tsdk": "node_modules/typescript/lib",
"synthwave84.disableGlow": true
} }

View File

@ -1,5 +1,18 @@
# CHANGE LOG # CHANGE LOG
## 4.1.6
### Feats
- 现在支持切换内容区域的过渡动画效果
- 优化了一些布局的样式细节
- 将过渡动画与 Spin 动画结合
- 拆分了布局组件,使得它看起来更合理
### Fixes
- 修复 RayChart 组件不能根据内容区域尺寸变化更新 chart 图
## 4.1.5 ## 4.1.5
### Fixes ### Fixes

View File

@ -76,7 +76,7 @@
"husky": "^8.0.3", "husky": "^8.0.3",
"lint-staged": "^13.1.0", "lint-staged": "^13.1.0",
"postcss": "^8.1.0", "postcss": "^8.1.0",
"postcss-px-to-viewport-update": "1.2.0", "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",
"svg-sprite-loader": "^6.0.11", "svg-sprite-loader": "^6.0.11",

View File

@ -11,25 +11,26 @@ module.exports = {
], ],
grid: true, grid: true,
}, },
// 由于该库的作者很久没更新了,导致 exclude include 并没有生效,所以使用的是其一个 fork 版本 // 为了适配 postcss8.x 版本的转换库
// 'postcss-px-to-viewport-update': { 'postcss-px-to-viewport-8-plugin': {
// inlinePxToViewport: true, inlinePxToViewport: true,
// /** 视窗的宽度(设计稿的宽度) */ /** 视窗的宽度(设计稿的宽度) */
// viewportWidth: 1920, viewportWidth: 1920,
// /** 视窗的高度(设计稿高度, 一般无需指定) */ /** 视窗的高度(设计稿高度, 一般无需指定) */
// viewportHeight: 1080, viewportHeight: 1080,
// /** 指定 px 转换为视窗单位值的小数位数 */ /** 指定 px 转换为视窗单位值的小数位数 */
// unitPrecision: 3, unitPrecision: 3,
// /** 指定需要转换成的视窗单位 */ /** 指定需要转换成的视窗单位 */
// viewportUnit: 'vw', viewportUnit: 'rem',
// /** 指定不转换为视窗单位的类 */ /** 指定不转换为视窗单位的类 */
// selectorBlackList: ['.ignore'], selectorBlackList: ['.ignore'],
// /** 小于或等于 1px 不转换为视窗单位 */ /** 小于或等于 1px 不转换为视窗单位 */
// minPixelValue: 1, minPixelValue: 1,
// /** 允许在媒体查询中转换 px */ /** 允许在媒体查询中转换 px */
// mediaQuery: false, mediaQuery: false,
// exclude: /(\/|\\)(node_modules)(\/|\\)/, // 忽略某些文件夹下的文件或特定文件,例如 'node_modules' 下的文件 // exclude: /(\/|\\)(node_modules)(\/|\\)/, // 忽略某些文件夹下的文件或特定文件,例如 'node_modules' 下的文件
// include: [/^src[/\\].*\.(vue|tsx|jsx|ts(?!d))$/], include: [/^src[/\\].*\.(vue|tsx|jsx|ts(?!d))$/],
// }, preserve: true,
},
}, },
} }

View File

@ -14,7 +14,7 @@
* *
* *
* *
* beforeRouteUpdate -> cancelAllRequest -> routerUpdate * beforeRouteUpdate -> cancelAllRequest -> routeUpdate
*/ */
import { axiosCanceler } from '@/axios/helper/interceptor' import { axiosCanceler } from '@/axios/helper/interceptor'

View File

@ -61,7 +61,7 @@ export const ROOT_ROUTE: Readonly<RootRoute> = {
* *
* , LOGO * , LOGO
*/ */
export const SIDE_BAR_LOGO: LayoutSideBarLogo = { export const SIDE_BAR_LOGO: LayoutSideBarLogo | undefined = {
icon: 'ray', icon: 'ray',
title: 'Ray Template', title: 'Ray Template',
url: '/dashboard', url: '/dashboard',

View File

@ -28,7 +28,7 @@ import type { LayoutInst } from 'naive-ui'
* }) * })
* ``` * ```
*/ */
export const LAYOUT_CONTENT_REF = ref<LayoutInst>() export const LAYOUT_CONTENT_REF = ref<LayoutInst | null>(null)
export const SETUP_ROUTER_ACTION = { export const SETUP_ROUTER_ACTION = {
/** 是否启用路由切换时顶部加载条 */ /** 是否启用路由切换时顶部加载条 */

View File

@ -4,4 +4,5 @@
border: none; border: none;
outline: none; outline: none;
box-sizing: border-box; box-sizing: border-box;
transition: width 0.35s var(--r-bezier);
} }

View File

@ -43,6 +43,7 @@ import { cloneDeep, throttle } from 'lodash-es'
import { on, off, completeSize } from '@/utils/element' import { on, off, completeSize } from '@/utils/element'
import { call } from '@/utils/vue/index' import { call } from '@/utils/vue/index'
import { setupChartTheme, loadingOptions } from './helper' import { setupChartTheme, loadingOptions } from './helper'
import { LAYOUT_CONTENT_REF } from '@/appConfig/routerConfig'
import type { PropType } from 'vue' import type { PropType } from 'vue'
import type { EChartsInstance } from '@/types/modules/component' import type { EChartsInstance } from '@/types/modules/component'
@ -53,6 +54,7 @@ import type {
AutoResize, AutoResize,
ChartTheme, ChartTheme,
} from '@/components/RayChart/type' } from '@/components/RayChart/type'
import type { UseResizeObserverReturn } from '@vueuse/core'
export type EChartsExtensionInstallRegisters = typeof CanvasRenderer export type EChartsExtensionInstallRegisters = typeof CanvasRenderer
@ -186,6 +188,7 @@ const RayChart = defineComponent({
const echartInstanceRef = ref<EChartsInstance>() // `echart` 拷贝实例, 解决直接使用响应式实例带来的问题 const echartInstanceRef = ref<EChartsInstance>() // `echart` 拷贝实例, 解决直接使用响应式实例带来的问题
let echartInstance: EChartsInstance // `echart` 实例 let echartInstance: EChartsInstance // `echart` 实例
let resizeThrottle: DebouncedFunc<AnyFC> // resize 防抖方法实例 let resizeThrottle: DebouncedFunc<AnyFC> // resize 防抖方法实例
let resizeOvserverReturn: UseResizeObserverReturn | undefined
const cssVarsRef = computed(() => { const cssVarsRef = computed(() => {
const cssVars = { const cssVars = {
@ -431,10 +434,16 @@ const RayChart = defineComponent({
/** 注册事件 */ /** 注册事件 */
if (props.autoResize) { if (props.autoResize) {
resizeThrottle = throttle(resizeChart, 1000) resizeThrottle = throttle(resizeChart, 500)
on(window, 'resize', resizeThrottle) on(window, 'resize', resizeThrottle)
} }
/** 监听内容区域尺寸变化更新 chart */
resizeOvserverReturn = useResizeObserver(
LAYOUT_CONTENT_REF.value as unknown as Ref<HTMLElement>,
resizeThrottle,
)
}) })
}) })
@ -445,6 +454,7 @@ const RayChart = defineComponent({
off(window, 'resize', resizeThrottle) off(window, 'resize', resizeThrottle)
/** 注销防抖 */ /** 注销防抖 */
resizeThrottle.cancel() resizeThrottle.cancel()
resizeOvserverReturn?.stop?.()
}) })
expose({ expose({

View File

@ -2,7 +2,7 @@
* *
* @author Ray <https://github.com/XiaoDaiGua-Ray> * @author Ray <https://github.com/XiaoDaiGua-Ray>
* *
* @date 2023-01-06 * @date 2023-08-08
* *
* @workspace ray-template * @workspace ray-template
* *
@ -10,7 +10,7 @@
*/ */
.ray-menu__logo { .ray-menu__logo {
height: 50px; height: $layoutHeaderHeight;
padding: 0 18px 0 24px; padding: 0 18px 0 24px;
display: flex; display: flex;
flex-wrap: nowrap; flex-wrap: nowrap;

View File

@ -0,0 +1,74 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-08-08
*
* @workspace ray-template
*
* @remark
*/
import './index.scss'
import { NEllipsis } from 'naive-ui'
import RayIcon from '@/components/RayIcon/index'
const SiderBarLogo = defineComponent({
name: 'SiderBarLogo',
props: {
collapsed: {
type: Boolean,
required: true,
},
},
setup() {
const router = useRouter()
const {
layout: { sideBarLogo },
} = __APP_CFG__
const handleSideBarLogoClick = () => {
if (sideBarLogo && sideBarLogo.url) {
sideBarLogo.jumpType === 'station'
? router.push(sideBarLogo.url)
: window.open(sideBarLogo.url)
}
}
return {
sideBarLogo,
handleSideBarLogoClick,
}
},
render() {
return this.sideBarLogo?.icon && this.sideBarLogo?.title ? (
<div
class={[
'ray-menu__logo',
this.sideBarLogo?.url ? 'ray-menu__logo-url' : '',
]}
onClick={this.handleSideBarLogoClick.bind(this)}
>
{this.sideBarLogo?.icon ? (
<RayIcon name={this.sideBarLogo.icon} size="30" />
) : (
''
)}
<h1
class={[
!this.collapsed ? 'ray-menu__logo-title--open' : '',
'ray-menu__logo-title',
]}
>
<NEllipsis>{this.sideBarLogo?.title}</NEllipsis>
</h1>
</div>
) : (
''
)
},
})
export default SiderBarLogo

View File

@ -1,11 +1,19 @@
import './index.scss' /**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2022-10-11
*
* @workspace ray-template
*
* @remark
*/
import { NMenu, NLayoutSider, NEllipsis } from 'naive-ui' import { NMenu, NLayoutSider } from 'naive-ui'
import RayIcon from '@/components/RayIcon/index' import SiderBarLogo from './components/SiderBarLogo/index'
import { useMenu } from '@/store' import { useMenu } from '@/store'
import { APP_MENU_CONFIG } from '@/appConfig/appConfig' import { APP_MENU_CONFIG } from '@/appConfig/appConfig'
import { useVueRouter } from '@/router/helper/useVueRouter'
import type { MenuInst } from 'naive-ui' import type { MenuInst } from 'naive-ui'
import type { NaiveMenuOptions } from '@/types/modules/component' import type { NaiveMenuOptions } from '@/types/modules/component'
@ -17,7 +25,6 @@ const LayoutMenu = defineComponent({
const menuRef = ref<MenuInst | null>(null) const menuRef = ref<MenuInst | null>(null)
const menuStore = useMenu() const menuStore = useMenu()
const { router } = useVueRouter()
const { changeMenuModelValue, collapsedMenu } = menuStore const { changeMenuModelValue, collapsedMenu } = menuStore
const modelMenuKey = computed({ const modelMenuKey = computed({
@ -33,17 +40,6 @@ const LayoutMenu = defineComponent({
}) })
const modelMenuOptions = computed(() => menuStore.options) const modelMenuOptions = computed(() => menuStore.options)
const modelCollapsed = computed(() => menuStore.collapsed) const modelCollapsed = computed(() => menuStore.collapsed)
const {
layout: { sideBarLogo },
} = __APP_CFG__
const handleSideBarLogoClick = () => {
if (sideBarLogo && sideBarLogo.url) {
sideBarLogo.jumpType === 'station'
? router.push(sideBarLogo.url)
: window.open(sideBarLogo.url)
}
}
const showMenuOption = () => { const showMenuOption = () => {
const key = modelMenuKey.value as string const key = modelMenuKey.value as string
@ -59,8 +55,6 @@ const LayoutMenu = defineComponent({
modelMenuOptions, modelMenuOptions,
modelCollapsed, modelCollapsed,
collapsedMenu, collapsedMenu,
sideBarLogo,
handleSideBarLogoClick,
menuRef, menuRef,
} }
}, },
@ -74,31 +68,7 @@ const LayoutMenu = defineComponent({
onUpdateCollapsed={this.collapsedMenu.bind(this)} onUpdateCollapsed={this.collapsedMenu.bind(this)}
nativeScrollbar={false} nativeScrollbar={false}
> >
{this.sideBarLogo ? ( <SiderBarLogo collapsed={this.modelCollapsed} />
<div
class={[
'ray-menu__logo',
this.sideBarLogo.url ? 'ray-menu__logo-url' : '',
]}
onClick={this.handleSideBarLogoClick.bind(this)}
>
{this.sideBarLogo.icon ? (
<RayIcon name={this.sideBarLogo.icon} size="30" />
) : (
''
)}
<h1
class={[
!this.modelCollapsed ? 'ray-menu__logo-title--open' : '',
'ray-menu__logo-title',
]}
>
<NEllipsis>{this.sideBarLogo.title}</NEllipsis>
</h1>
</div>
) : (
''
)}
<NMenu <NMenu
ref="menuRef" ref="menuRef"
v-model:value={this.modelMenuKey} v-model:value={this.modelMenuKey}

View File

@ -6,6 +6,7 @@ $menuTagWrapperWidth: 76px;
border-bottom: solid 1px var(--n-border-color); border-bottom: solid 1px var(--n-border-color);
display: flex; display: flex;
align-items: center; align-items: center;
padding: 4px 0;
& .menu-tag-sapce { & .menu-tag-sapce {
width: calc(100% - $space * 2); width: calc(100% - $space * 2);

View File

@ -388,7 +388,9 @@ const MenuTag = defineComponent({
if (tags?.length) { if (tags?.length) {
const [menuTag] = tags const [menuTag] = tags
menuTag.scrollIntoView?.() nextTick(() => {
menuTag.scrollIntoView?.()
})
} }
}) })
} }

View File

@ -22,7 +22,7 @@ import { NDropdown, NBreadcrumb, NBreadcrumbItem } from 'naive-ui'
import { useMenu } from '@/store' import { useMenu } from '@/store'
import type { DropdownOption, MenuOption } from 'naive-ui' import type { DropdownOption } from 'naive-ui'
import type { import type {
AppMenuOption, AppMenuOption,
MenuTagOptions, MenuTagOptions,

View File

@ -1,4 +1,5 @@
import './index.scss' import './index.scss'
import { import {
NDrawer, NDrawer,
NDrawerContent, NDrawerContent,
@ -8,6 +9,7 @@ import {
NColorPicker, NColorPicker,
NDescriptions, NDescriptions,
NDescriptionsItem, NDescriptionsItem,
NSelect,
} from 'naive-ui' } from 'naive-ui'
import ThemeSwitch from '@/layout/components/SiderBar/components/SettingDrawer/components/ThemeSwitch/index' import ThemeSwitch from '@/layout/components/SiderBar/components/SettingDrawer/components/ThemeSwitch/index'
@ -39,7 +41,8 @@ const SettingDrawer = defineComponent({
const { t } = useI18n() const { t } = useI18n()
const settingStore = useSetting() const settingStore = useSetting()
const { changePrimaryColor, changeSwitcher } = settingStore const { changePrimaryColor, changeSwitcher, updateContentTransition } =
settingStore
const { const {
themeValue, themeValue,
primaryColorOverride, primaryColorOverride,
@ -47,6 +50,7 @@ const SettingDrawer = defineComponent({
breadcrumbSwitch, breadcrumbSwitch,
invertSwitch, invertSwitch,
footerSwitch, footerSwitch,
contentTransition,
} = storeToRefs(settingStore) } = storeToRefs(settingStore)
const modelShow = computed({ const modelShow = computed({
@ -55,6 +59,24 @@ const SettingDrawer = defineComponent({
emit('update:show', bool) emit('update:show', bool)
}, },
}) })
const contentTransitionOptions = [
{
label: '无',
value: 'none',
},
{
label: '缩放效果',
value: 'scale',
},
{
label: '淡入淡出',
value: 'fade',
},
{
label: '闪入效果',
value: 'opacity',
},
]
return { return {
modelShow, modelShow,
@ -67,6 +89,9 @@ const SettingDrawer = defineComponent({
breadcrumbSwitch, breadcrumbSwitch,
invertSwitch, invertSwitch,
footerSwitch, footerSwitch,
contentTransitionOptions,
contentTransition,
updateContentTransition,
} }
}, },
render() { render() {
@ -92,6 +117,16 @@ const SettingDrawer = defineComponent({
v-model:value={this.primaryColorOverride.common!.primaryColor} v-model:value={this.primaryColorOverride.common!.primaryColor}
onUpdateValue={this.changePrimaryColor.bind(this)} onUpdateValue={this.changePrimaryColor.bind(this)}
/> />
<NDivider titlePlacement="center">
{t('headerSettingOptions.ContentTransition')}
</NDivider>
<NSelect
v-model:value={this.contentTransition}
options={this.contentTransitionOptions}
onUpdateValue={(value) => {
this.updateContentTransition(value)
}}
/>
<NDivider titlePlacement="center"> <NDivider titlePlacement="center">
{t('headerSettingOptions.InterfaceDisplay')} {t('headerSettingOptions.InterfaceDisplay')}
</NDivider> </NDivider>

View File

@ -17,8 +17,8 @@
import './index.scss' import './index.scss'
import RayTransitionComponent from '@/components/RayTransitionComponent/index.vue'
import { NSpin } from 'naive-ui' import { NSpin } from 'naive-ui'
import RayTransitionComponent from '@/components/RayTransitionComponent/index.vue'
import AppRequestCanceler from '@/app-components/provider/AppRequestCanceler/index' import AppRequestCanceler from '@/app-components/provider/AppRequestCanceler/index'
import { useSetting } from '@/store' import { useSetting } from '@/store'
@ -31,7 +31,7 @@ const ContentWrapper = defineComponent({
const settingStore = useSetting() const settingStore = useSetting()
const router = useRouter() const router = useRouter()
const { reloadRouteSwitch } = storeToRefs(settingStore) const { reloadRouteSwitch, contentTransition } = storeToRefs(settingStore)
const spinning = ref(false) const spinning = ref(false)
const thmeOverridesSpin: GlobalThemeOverrides['Spin'] = { const thmeOverridesSpin: GlobalThemeOverrides['Spin'] = {
opacitySpinning: '0', opacitySpinning: '0',
@ -43,9 +43,7 @@ const ContentWrapper = defineComponent({
}) })
router.afterEach(() => { router.afterEach(() => {
setTimeout(() => { spinning.value = false
spinning.value = false
}, 300)
}) })
} }
@ -55,6 +53,7 @@ const ContentWrapper = defineComponent({
reloadRouteSwitch, reloadRouteSwitch,
spinning, spinning,
thmeOverridesSpin, thmeOverridesSpin,
contentTransition,
} }
}, },
render() { render() {
@ -69,7 +68,7 @@ const ContentWrapper = defineComponent({
{this.reloadRouteSwitch ? ( {this.reloadRouteSwitch ? (
<RayTransitionComponent <RayTransitionComponent
class="content-wrapper" class="content-wrapper"
transitionPropName="layout-content" transitionPropName={this.contentTransition + '-transform'}
/> />
) : ( ) : (
'' ''

View File

@ -0,0 +1,24 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-08-09
*
* @workspace ray-template
*
* @remark
*/
import MenuTag from '@/layout/components/MenuTag/index'
const FeatureWrapper = defineComponent({
name: 'FeatureWrapper',
setup() {
return {}
},
render() {
return <MenuTag />
},
})
export default FeatureWrapper

View File

@ -1,4 +1,4 @@
.layout-footer-wrapper { .layout-footer-wrapper {
padding: 20px; padding: 0 20px 8px 20px;
text-align: center; text-align: center;
} }

View File

@ -0,0 +1,29 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-08-09
*
* @workspace ray-template
*
* @remark
*/
import { NSpace } from 'naive-ui'
import SiderBar from '@/layout/components/SiderBar/index'
const HeaderWrapper = defineComponent({
name: 'HeaderWrapper',
setup() {
return {}
},
render() {
return (
<NSpace wrapItem={false} size={[0, 0]}>
<SiderBar />
</NSpace>
)
},
})
export default HeaderWrapper

View File

@ -1,4 +1,4 @@
.r-layout-full.r-layout-full { .r-layout-full {
position: fixed; position: fixed;
inset: 0; inset: 0;

View File

@ -11,19 +11,19 @@
import './index.scss' import './index.scss'
import { NLayout, NLayoutContent, NScrollbar } from 'naive-ui' import { NLayout, NLayoutContent } from 'naive-ui'
import Menu from './components/Menu/index' import Menu from './components/Menu/index'
import SiderBar from './components/SiderBar/index'
import MenuTag from './components/MenuTag/index'
import ContentWrapper from '@/layout/default/ContentWrapper' import ContentWrapper from '@/layout/default/ContentWrapper'
import FooterWrapper from '@/layout/default/FooterWrapper' import FooterWrapper from '@/layout/default/FooterWrapper'
import HeaderWrapper from './default/HeaderWrapper'
import FeatureWrapper from './default/FeatureWrapper'
import { useSetting } from '@/store' import { useSetting } from '@/store'
import { LAYOUT_CONTENT_REF } from '@/appConfig/routerConfig' import { LAYOUT_CONTENT_REF } from '@/appConfig/routerConfig'
import { layoutHeaderCssVars } from '@/layout/layoutResize' import { layoutHeaderCssVars } from '@/layout/layoutResize'
import useAppLockScreen from '@/app-components/app/AppLockScreen/appLockVar' import useAppLockScreen from '@/app-components/app/AppLockScreen/appLockVar'
const Layout = defineComponent({ const RLayout = defineComponent({
name: 'RLayout', name: 'RLayout',
setup() { setup() {
const layoutSiderBarRef = ref<HTMLElement>() const layoutSiderBarRef = ref<HTMLElement>()
@ -57,8 +57,12 @@ const Layout = defineComponent({
<NLayout class="r-layout-full" style={[this.cssVarsRef]} hasSider> <NLayout class="r-layout-full" style={[this.cssVarsRef]} hasSider>
<Menu /> <Menu />
<NLayoutContent class="r-layout-full__viewer"> <NLayoutContent class="r-layout-full__viewer">
<SiderBar ref="layoutSiderBarRef" /> <HeaderWrapper ref="layoutSiderBarRef" />
{this.modelMenuTagSwitch ? <MenuTag ref="layoutMenuTagRef" /> : ''} {this.modelMenuTagSwitch ? (
<FeatureWrapper ref="layoutMenuTagRef" />
) : (
''
)}
<NLayoutContent <NLayoutContent
ref="LAYOUT_CONTENT_REF" ref="LAYOUT_CONTENT_REF"
class="r-layout-full__viewer-content" class="r-layout-full__viewer-content"
@ -75,4 +79,4 @@ const Layout = defineComponent({
}, },
}) })
export default Layout export default RLayout

View File

@ -6,5 +6,6 @@
"Light": "Light", "Light": "Light",
"PrimaryColorConfig": "Primary Color" "PrimaryColorConfig": "Primary Color"
}, },
"InterfaceDisplay": "Display" "InterfaceDisplay": "Display",
"ContentTransition": "Content Transition"
} }

View File

@ -6,5 +6,6 @@
"Light": "明亮", "Light": "明亮",
"PrimaryColorConfig": "主题色" "PrimaryColorConfig": "主题色"
}, },
"InterfaceDisplay": "界面显示" "InterfaceDisplay": "界面显示",
"ContentTransition": "动画效果"
} }

View File

@ -55,7 +55,6 @@ interface RouteMeta {
keepAlive?: boolean keepAlive?: boolean
sameLevel?: boolean sameLevel?: boolean
dev?: string | string[] dev?: string | string[]
needCancel?: boolean
} }
``` ```

View File

@ -38,8 +38,14 @@ export const useSetting = defineStore(
lockScreenSwitch: false, // 锁屏开关 lockScreenSwitch: false, // 锁屏开关
lockScreenInputSwitch: false, // 锁屏输入状态开关(预留该字段是为了方便拓展用, 但是舍弃了该字段, 改为使用 useAppLockScreen 方法) lockScreenInputSwitch: false, // 锁屏输入状态开关(预留该字段是为了方便拓展用, 但是舍弃了该字段, 改为使用 useAppLockScreen 方法)
footerSwitch: true, // 底部区域开关 footerSwitch: true, // 底部区域开关
contentTransition: 'scale', // 切换过渡效果
}) })
/** 更新过渡效果 */
const updateContentTransition = (value: string) => {
settingState.contentTransition = value
}
/** 修改当前语言 */ /** 修改当前语言 */
const updateLocale = (key: string) => { const updateLocale = (key: string) => {
locale(key) locale(key)
@ -103,6 +109,7 @@ export const useSetting = defineStore(
updateLocale, updateLocale,
changePrimaryColor, changePrimaryColor,
changeSwitcher, changeSwitcher,
updateContentTransition,
} }
}, },
{ {

View File

@ -14,4 +14,5 @@ export interface SettingState {
lockScreenSwitch: boolean lockScreenSwitch: boolean
lockScreenInputSwitch: boolean lockScreenInputSwitch: boolean
footerSwitch: boolean footerSwitch: boolean
contentTransition: string
} }

View File

@ -15,17 +15,64 @@
opacity: 0; opacity: 0;
} }
.fade-enter-active, /* fade-transform */
.fade-leave-active { .fade-transform-leave-active,
transition: all 0.35s ease; .fade-transform-enter-active {
transition: all 0.5s;
} }
.fade-enter-from { .fade-transform-enter-from {
opacity: 0; opacity: 0;
transform: translateX(-30px); transform: translateX(-50px);
} }
.fade-leave-to { .fade-transform-leave-to {
opacity: 0;
transform: translateX(50px);
}
/* down-transform */
.down-transform-leave-active,
.down-transform-enter-active {
transition: all 0.5s;
}
.down-transform-enter-from {
opacity: 0;
transform: translateY(-50px);
}
.down-transform-leave-to {
opacity: 0;
transform: translateY(50px);
}
/* scale-transform */
.scale-transform-leave-active,
.scale-transform-enter-active {
transition: all 0.5s;
}
.scale-transform-enter-from {
opacity: 0;
transform: scale(2);
}
.scale-transform-leave-to {
opacity: 0;
transform: scale(0.5);
}
/* opacity-transform */
.opacity-transform-leave-active,
.opacity-transform-enter-active {
transition: all 0.5s;
}
.opacity-transform-enter-from {
opacity: 0;
}
.opacity-transform-leave-to {
opacity: 0; opacity: 0;
transform: translateX(30px);
} }

View File

@ -67,7 +67,7 @@ export default defineConfig(async ({ mode }) => {
alias: alias, alias: alias,
}, },
plugins: [ plugins: [
vue({ reactivityTransform: true }), vue(),
viteVueJSX(), viteVueJSX(),
title, title,
viteVeI18nPlugin({}), viteVeI18nPlugin({}),