diff --git a/src/components/PageTransition/index.less b/src/components/PageTransition/index.less new file mode 100644 index 0000000..9286314 --- /dev/null +++ b/src/components/PageTransition/index.less @@ -0,0 +1,58 @@ +// 简化版页面过渡动画 +.simple-page-transition { + animation: pageEnter 0.5s cubic-bezier(0.4, 0, 0.2, 1); + will-change: opacity, transform; +} + +@keyframes pageEnter { + from { + opacity: 0; + transform: translateY(30px) scale(0.96); + filter: blur(4px); + } + to { + opacity: 1; + transform: translateY(0) scale(1); + filter: blur(0); + } +} + +// 复杂版页面过渡动画 +.page-transition { + position: relative; + height: 100%; + + .transition-content { + height: 100%; + + &.fadeIn { + animation: fadeIn 0.3s ease-in-out forwards; + } + + &.fadeOut { + animation: fadeOut 0.3s ease-in-out forwards; + } + } +} + +@keyframes fadeIn { + from { + opacity: 0; + transform: translateY(20px); + } + to { + opacity: 1; + transform: translateY(0); + } +} + +@keyframes fadeOut { + from { + opacity: 1; + transform: translateY(0); + } + to { + opacity: 0; + transform: translateY(-10px); + } +} \ No newline at end of file diff --git a/src/components/PageTransition/index.tsx b/src/components/PageTransition/index.tsx new file mode 100644 index 0000000..cb177f8 --- /dev/null +++ b/src/components/PageTransition/index.tsx @@ -0,0 +1,56 @@ +import React, { useEffect, useState } from 'react'; +import { CSSTransition, TransitionGroup } from 'react-transition-group'; +import { useLocation } from 'umi'; +import './index.less'; + +export interface PageTransitionProps { + children: React.ReactNode; + type?: 'fade' | 'slide' | 'scale'; + duration?: number; + className?: string; +} + +const PageTransition: React.FC = ({ + children, + type = 'fade', + duration = 300, + className = '', +}) => { + const location = useLocation(); + const [displayLocation, setDisplayLocation] = useState(location); + const [transitionStage, setTransitionStage] = useState('fadeIn'); + + useEffect(() => { + if (location !== displayLocation) setTransitionStage('fadeOut'); + }, [location, displayLocation]); + + return ( +
+
{ + if (transitionStage === 'fadeOut') { + setDisplayLocation(location); + setTransitionStage('fadeIn'); + } + }} + > + {children} +
+
+ ); +}; + +// 简化版本 - 仅使用CSS动画 +export const SimplePageTransition: React.FC<{ children: React.ReactNode; className?: string }> = ({ + children, + className = '', +}) => { + return ( +
+ {children} +
+ ); +}; + +export default PageTransition; \ No newline at end of file diff --git a/src/global.less b/src/global.less index 1a9e7c3..09099a3 100644 --- a/src/global.less +++ b/src/global.less @@ -33,12 +33,15 @@ ol { .ant-table { width: 100%; overflow-x: auto; - &-thead > tr, - &-tbody > tr { - > th, - > td { + + &-thead>tr, + &-tbody>tr { + + >th, + >td { white-space: pre; - > span { + + >span { display: block; } } @@ -47,8 +50,23 @@ ol { } // Compatible with IE11 -@media screen and(-ms-high-contrast: active), (-ms-high-contrast: none) { - body .ant-design-pro > .ant-layout { +@media screen and(-ms-high-contrast: active), +(-ms-high-contrast: none) { + body .ant-design-pro>.ant-layout { min-height: 100vh; } } + +// 页面过渡优化 +* { + box-sizing: border-box; +} + +// 增强页面切换体验 +.ant-tabs-content-holder { + overflow: hidden; +} + +.ant-tabs-tabpane { + outline: none; +} \ No newline at end of file diff --git a/src/layouts/BasicLayout.less b/src/layouts/BasicLayout.less index 3262332..d9491bd 100644 --- a/src/layouts/BasicLayout.less +++ b/src/layouts/BasicLayout.less @@ -1,45 +1,56 @@ @import '~antd/es/style/themes/default.less'; +@import '../components/PageTransition/index.less'; .ant-pro-global-header { box-shadow: none !important; } + .ant-pro-basicLayout-content { margin: 0 !important; } -.ant-menu-inline-collapsed .ant-pro-menu-item .ant-pro-menu-item-title { + +.ant-menu-inline-collapsed .ant-pro-menu-item .ant-pro-menu-item-title { opacity: 0; } + .ant-pro-sider-logo img { height: 21px !important; } -.main-tab >.ant-tabs-nav { + +.main-tab>.ant-tabs-nav { margin-bottom: 0 !important; padding: 2px 15px 0 16px; background-color: #fff; } -.main-tab > .ant-tabs-nav .ant-tabs-tab { + +.main-tab>.ant-tabs-nav .ant-tabs-tab { background-color: #fafafa; border-color: #f0f0f0; border-bottom: 0; } -.main-tab > .ant-tabs-nav .ant-tabs-tab button svg { + +.main-tab>.ant-tabs-nav .ant-tabs-tab button svg { // color: #fff; transition: all 0.3s; } -.main-tab > .ant-tabs-nav .ant-tabs-tab:hover { - background-color:#fafafa; + +.main-tab>.ant-tabs-nav .ant-tabs-tab:hover { + background-color: #fafafa; border-color: #f0f0f0; } -.main-tab > .ant-tabs-nav .ant-tabs-tab:hover button svg { + +.main-tab>.ant-tabs-nav .ant-tabs-tab:hover button svg { color: @link-color; } -.main-tab.ant-tabs-top > .ant-tabs-nav .ant-tabs-tab-active { + +.main-tab.ant-tabs-top>.ant-tabs-nav .ant-tabs-tab-active { background-color: #f6f9fb; border-color: #f0f0f0; box-shadow: 2px 0 6px 0 #e7e7e7; border-top-color: #3591fe; } -.main-tab.ant-tabs-top > .ant-tabs-nav .ant-tabs-tab-active::before { + +.main-tab.ant-tabs-top>.ant-tabs-nav .ant-tabs-tab-active::before { content: ''; position: absolute; display: block; @@ -49,17 +60,21 @@ width: 100%; height: 1px; } -.main-tab > .ant-tabs-nav .ant-tabs-tab-active button svg { + +.main-tab>.ant-tabs-nav .ant-tabs-tab-active button svg { color: @icon-color; } -.main-tab > .ant-tabs-content-holder { + +.main-tab>.ant-tabs-content-holder { height: calc(100vh - 86px); padding-top: 16px; overflow-y: auto; } -.main-tab > .ant-tabs-nav .ant-tabs-nav-wrap { + +.main-tab>.ant-tabs-nav .ant-tabs-nav-wrap { bottom: 1px; } + .tab-extra { width: 13px; height: 15px; @@ -68,10 +83,12 @@ // background-repeat: no-repeat; } + .ant-dropdown-open svg { color: @primary-color; // background-image: url(''); } + .tab-extra:hover { // background-image: url(''); } @@ -83,9 +100,11 @@ body *::-webkit-scrollbar { /* 滚动条整体样式 */ - width : 8px; /* 高宽分别对应横竖滚动条的尺寸 */ + width: 8px; + /* 高宽分别对应横竖滚动条的尺寸 */ height: 8px; } + body *::-webkit-scrollbar-thumb { /* 滚动条里面小方块 */ background-color: #d1cfcf; @@ -101,6 +120,7 @@ body *::-webkit-scrollbar-thumb { // ); border-radius: 10px; } + body *::-webkit-scrollbar-track { /* 滚动条里面轨道 */ background: #ededed; @@ -109,59 +129,71 @@ body *::-webkit-scrollbar-track { } // 列表页左侧多选树 -.pageTable-leftnav .ant-tree-list-holder{ +.pageTable-leftnav .ant-tree-list-holder { height: calc(100vh - 257px); padding-right: 0; padding-bottom: 0; overflow-y: auto; } -.leftHeight .ant-tree-list-holder{ - height: 100%!important; + +.leftHeight .ant-tree-list-holder { + height: 100% !important; } -.leftHeight .ant-tree-list{ - height: 560px + +.leftHeight .ant-tree-list { + height: 560px } -.pageTable-leftnavs .ant-tree-list-holder{ - height: 580px; - padding-right: 0; - padding-bottom: 0; - overflow-y: auto; + +.pageTable-leftnavs .ant-tree-list-holder { + height: 580px; + padding-right: 0; + padding-bottom: 0; + overflow-y: auto; } + .pageTable-leftnav .ant-pro-card-header.ant-pro-card-header-border { padding-top: 29px; } + .pageTable-leftnav .ant-pro-card-title { font-size: 14px; } -.pageTable-leftnav .ant-pro-card-split-vertical{ +.pageTable-leftnav .ant-pro-card-split-vertical { transition: width 0.3s; } + .pageTable-leftnav .ant-tree .ant-tree-treenode { padding-bottom: 16px; } + .pageTable-leftnavs .ant-tree .ant-tree-treenode { padding-bottom: 16px; } -.ant-table-tbody > tr.tablerow-no-border > td{ + +.ant-table-tbody>tr.tablerow-no-border>td { border-bottom: 0; } + // prolist 卡片类型 -.ant-pro-list .ant-pro-card-header { +.ant-pro-list .ant-pro-card-header { padding: 0 !important; } -.ant-pro-list .ant-pro-card.ant-pro-card-border.ant-pro-card-hoverable>.ant-pro-card-body { + +.ant-pro-list .ant-pro-card.ant-pro-card-border.ant-pro-card-hoverable>.ant-pro-card-body { margin: 0 !important; padding: 20px 0 0 0 !important; } -.ant-pro-list .ant-pro-card.ant-pro-card-border> ul.ant-pro-card-actions { +.ant-pro-list .ant-pro-card.ant-pro-card-border>ul.ant-pro-card-actions { margin-top: 20px; } -.ant-pro-list .ant-pro-card.ant-pro-card-border> ul.ant-pro-card-actions >li{ - margin:0; +.ant-pro-list .ant-pro-card.ant-pro-card-border>ul.ant-pro-card-actions>li { + margin: 0; } -.ant-pro-list .ant-pro-card.ant-pro-card-border.ant-pro-card-hoverable> ul.ant-pro-card-actions > li, .ant-pro-list .ant-pro-card .ant-pro-card-actions .ant-space-item { + +.ant-pro-list .ant-pro-card.ant-pro-card-border.ant-pro-card-hoverable>ul.ant-pro-card-actions>li, +.ant-pro-list .ant-pro-card .ant-pro-card-actions .ant-space-item { margin: 10px 0 0 0 !important; -} +} \ No newline at end of file diff --git a/src/layouts/BasicLayout.tsx b/src/layouts/BasicLayout.tsx index 3489719..343a9d5 100644 --- a/src/layouts/BasicLayout.tsx +++ b/src/layouts/BasicLayout.tsx @@ -28,6 +28,7 @@ import * as Icon from '@ant-design/icons' import IconFont from '@/components/IconFont'; import type { CurrentUser } from '@/models/user' import session from '@/utils/session'; +import { SimplePageTransition } from '@/components/PageTransition'; import upMenu from '../assets/tab/upMenu.png' import { getFieldEnum, getFieldEnumTravel, getFieldEnumTree, getFieldGetFieEnumList, getTravelFieldEnumTree, handleGetFieldEnumTreeTravel, handleGetNestingFIELDENUMList } from "@/services/options"; import { handleGetServerpartTree } from '@/pages/basicManage/serverpartAssets/service'; @@ -1039,11 +1040,13 @@ const BasicLayout: React.FC = (props) => { - Loading...}> - - {item.children} - - + + Loading...}> + + {item.children} + + + ) }