mirror of
https://github.com/XiaoDaiGua-Ray/ray-template.git
synced 2025-04-06 03:57:49 +08:00
v4.1.6
This commit is contained in:
parent
be406cc026
commit
5eb201b25b
3
.vscode/settings.json
vendored
3
.vscode/settings.json
vendored
@ -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
|
||||||
}
|
}
|
||||||
|
13
CHANGELOG.md
13
CHANGELOG.md
@ -1,5 +1,18 @@
|
|||||||
# CHANGE LOG
|
# CHANGE LOG
|
||||||
|
|
||||||
|
## 4.1.6
|
||||||
|
|
||||||
|
### Feats
|
||||||
|
|
||||||
|
- 现在支持切换内容区域的过渡动画效果
|
||||||
|
- 优化了一些布局的样式细节
|
||||||
|
- 将过渡动画与 Spin 动画结合
|
||||||
|
- 拆分了布局组件,使得它看起来更合理
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
|
||||||
|
- 修复 RayChart 组件不能根据内容区域尺寸变化更新 chart 图
|
||||||
|
|
||||||
## 4.1.5
|
## 4.1.5
|
||||||
|
|
||||||
### Fixes
|
### Fixes
|
||||||
|
@ -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",
|
||||||
|
@ -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,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
@ -14,7 +14,7 @@
|
|||||||
* 路由更新前,取消上一路由所有请求
|
* 路由更新前,取消上一路由所有请求
|
||||||
*
|
*
|
||||||
* 生命周期示意图:
|
* 生命周期示意图:
|
||||||
* beforeRouteUpdate -> cancelAllRequest -> routerUpdate
|
* beforeRouteUpdate -> cancelAllRequest -> routeUpdate
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { axiosCanceler } from '@/axios/helper/interceptor'
|
import { axiosCanceler } from '@/axios/helper/interceptor'
|
||||||
|
@ -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',
|
||||||
|
@ -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 = {
|
||||||
/** 是否启用路由切换时顶部加载条 */
|
/** 是否启用路由切换时顶部加载条 */
|
||||||
|
@ -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);
|
||||||
}
|
}
|
||||||
|
@ -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({
|
||||||
|
@ -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;
|
74
src/layout/components/Menu/components/SiderBarLogo/index.tsx
Normal file
74
src/layout/components/Menu/components/SiderBarLogo/index.tsx
Normal 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
|
@ -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}
|
||||||
|
@ -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);
|
||||||
|
@ -388,7 +388,9 @@ const MenuTag = defineComponent({
|
|||||||
if (tags?.length) {
|
if (tags?.length) {
|
||||||
const [menuTag] = tags
|
const [menuTag] = tags
|
||||||
|
|
||||||
menuTag.scrollIntoView?.()
|
nextTick(() => {
|
||||||
|
menuTag.scrollIntoView?.()
|
||||||
|
})
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -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,
|
||||||
|
@ -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>
|
||||||
|
@ -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'}
|
||||||
/>
|
/>
|
||||||
) : (
|
) : (
|
||||||
''
|
''
|
||||||
|
24
src/layout/default/FeatureWrapper/index.tsx
Normal file
24
src/layout/default/FeatureWrapper/index.tsx
Normal 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
|
@ -1,4 +1,4 @@
|
|||||||
.layout-footer-wrapper {
|
.layout-footer-wrapper {
|
||||||
padding: 20px;
|
padding: 0 20px 8px 20px;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
29
src/layout/default/HeaderWrapper/index.tsx
Normal file
29
src/layout/default/HeaderWrapper/index.tsx
Normal 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
|
@ -1,4 +1,4 @@
|
|||||||
.r-layout-full.r-layout-full {
|
.r-layout-full {
|
||||||
position: fixed;
|
position: fixed;
|
||||||
inset: 0;
|
inset: 0;
|
||||||
|
|
||||||
|
@ -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
|
||||||
|
@ -6,5 +6,6 @@
|
|||||||
"Light": "Light",
|
"Light": "Light",
|
||||||
"PrimaryColorConfig": "Primary Color"
|
"PrimaryColorConfig": "Primary Color"
|
||||||
},
|
},
|
||||||
"InterfaceDisplay": "Display"
|
"InterfaceDisplay": "Display",
|
||||||
|
"ContentTransition": "Content Transition"
|
||||||
}
|
}
|
||||||
|
@ -6,5 +6,6 @@
|
|||||||
"Light": "明亮",
|
"Light": "明亮",
|
||||||
"PrimaryColorConfig": "主题色"
|
"PrimaryColorConfig": "主题色"
|
||||||
},
|
},
|
||||||
"InterfaceDisplay": "界面显示"
|
"InterfaceDisplay": "界面显示",
|
||||||
|
"ContentTransition": "动画效果"
|
||||||
}
|
}
|
||||||
|
@ -55,7 +55,6 @@ interface RouteMeta {
|
|||||||
keepAlive?: boolean
|
keepAlive?: boolean
|
||||||
sameLevel?: boolean
|
sameLevel?: boolean
|
||||||
dev?: string | string[]
|
dev?: string | string[]
|
||||||
needCancel?: boolean
|
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
@ -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,
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -14,4 +14,5 @@ export interface SettingState {
|
|||||||
lockScreenSwitch: boolean
|
lockScreenSwitch: boolean
|
||||||
lockScreenInputSwitch: boolean
|
lockScreenInputSwitch: boolean
|
||||||
footerSwitch: boolean
|
footerSwitch: boolean
|
||||||
|
contentTransition: string
|
||||||
}
|
}
|
||||||
|
61
src/styles/animate.scss
vendored
61
src/styles/animate.scss
vendored
@ -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);
|
|
||||||
}
|
}
|
||||||
|
@ -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({}),
|
||||||
|
Loading…
x
Reference in New Issue
Block a user