React@16.x(42)路由v5.x(7)常见应用场景(4)- 路由切换动画

目录

通过一个例子来说明如何实现。

1,实现路由切换

基础样式

App.jsx

js 复制代码
import { BrowserRouter as Router, Route, NavLink } from "react-router-dom";
import { Home, News, Goods } from "./page";
import "./App.css";

export default function App() {
    return (
        <div className="container">
            <Router>
                <div className="nav-box">
                    <NavLink to="/" exact>
                        首页
                    </NavLink>
                    <NavLink to="/news">新闻页</NavLink>
                    <NavLink to="/goods">商品页</NavLink>
                </div>
                <div className="page-box">
                    <Route path="/" exact component={Home}></Route>
                    <Route path="/news" component={News}></Route>
                    <Route path="/goods" component={Goods}></Route>
                </div>
            </Router>
        </div>
    );
}

App.css

css 复制代码
.container {
    margin: 0 auto;
    width: 400px;
}

.nav-box {
    background-color: lightblue;
    line-height: 3;
    text-align: center;
}

.nav-box a {
    display: inline-block;
    margin: 10px;
}

.nav-box a.active {
    color: salmon;
}

.page-box {
    height: 200px;
    outline: 1px solid salmon;
}

page.jsx

js 复制代码
import React from "react";
import "./page.css";

export function Home() {
    return <div className="page home">Home</div>;
}

export function News() {
    return <div className="page news">News</div>;
}

export function Goods() {
    return <div className="page goods">Goods</div>;
}

page.css

css 复制代码
.page {
    height: 100%;
    text-align: center;
}

.home {
    background-color: lightblue;
}

.news {
    background-color: lightgreen;
}

.goods {
    background-color: lightyellow;
}

显示效果:

2,使用 CSSTransition 添加动画

CSSTransition 的使用参考 React@16.x(33)动画(上)

先看最终效果

  1. 可以看到路由页面(一开始)是重叠在一起的,所以需要使用绝对定位来实现。
  2. 动画过程中,对应的路由页面在动画开始前,应该是未加载的状态。同理,退出后应该是卸载状态。所以在 CSSTransition 组件上需要添加 mountOnEnterunmountOnExit

完整代码:

1,自定义动画组件 TransitionRoute.jsx

match 属性未匹配时为 null

js 复制代码
import React from "react";
import { Route } from "react-router-dom";
import { CSSTransition } from "react-transition-group";
import "animate.css";

export default function TransitionRoute(props) {
    const { component: Comp, ...rest } = props;
    return (
        <Route {...rest}>
            {({ match }) => {
                return (
                    <CSSTransition
                        in={!!match}
                        timeout={500}
                        mountOnEnter
                        unmountOnExit
                        classNames={{
                            enterActive: "animate__fadeInRight",
                            exitActive: "animate__fadeOutLeft",
                        }}
                    >
                        <Comp></Comp>
                    </CSSTransition>
                );
            }}
        </Route>
    );
}

2,App.jsx

只需要将 Route 替换为 TransitionRoute 即可。

3,样式改动

App.css,父级元素添加相对定位。

css 复制代码
.page-box {
    position: relative;
}

page.jsx 。因为想使用 animate.css ,所以对进行动画的元素添加公共样式 animate__animated

js 复制代码
import React from "react";
import "./page.css";

export function Home() {
    return <div className="page animate__animated home">Home</div>;
}

export function News() {
    return <div className="page animate__animated news">News</div>;
}

export function Goods() {
    return <div className="page  animate__animated goods">Goods</div>;
}

page.css,添加绝对定位。

css 复制代码
.page {
    position: absolute;
    left: 0;
    top: 0;
    height: 100%;
    width: 100%;
}

3,注意点

  • 当不使用 Swtich 组件时,每个Route 组件都会加载,只是不显示而已。所以可实现切换路由的动画。
  • 使用 Swtich 组件后,因为它只会加载匹配到的 Route 组件,其他Route 组件并没有加载,所以无法完成退出进入动画

在 React 的插件中可以检查下:

不使用 Switch 组件时,3个 Route 组件都会渲染,只是未匹配的不展示:

使用 Switch 后,只会渲染一个 Route 组件:


以上。

相关推荐
abc800211703436 分钟前
前端Bug 修复手册
前端·bug
Best_Liu~39 分钟前
el-table实现固定列,及解决固定列导致部分滚动条无法拖动的问题
前端·javascript·vue.js
_斯洛伐克2 小时前
下降npm版本
前端·vue.js
苏十八3 小时前
前端进阶:Vue.js
前端·javascript·vue.js·前端框架·npm·node.js·ecmascript
st紫月3 小时前
用MySQL+node+vue做一个学生信息管理系统(四):制作增加、删除、修改的组件和对应的路由
前端·vue.js·mysql
乐容4 小时前
vue3使用pinia中的actions,需要调用接口的话
前端·javascript·vue.js
似水明俊德4 小时前
ASP.NET Core Blazor 5:Blazor表单和数据
java·前端·javascript·html·asp.net
至天5 小时前
UniApp 中 Web/H5 正确使用反向代理解决跨域问题
前端·uni-app·vue3·vue2·vite·反向代理
与墨学长5 小时前
Rust破界:前端革新与Vite重构的深度透视(中)
开发语言·前端·rust·前端框架·wasm
H-J-L6 小时前
Web基础与HTTP协议
前端·http·php