React 之 过渡动画

news2025/1/23 7:19:07

一、React的过渡动画

  • 在开发中,我们想要给一个组件的显示和消失添加某种过渡动画,可以很好的增加用户体验

  • 可以通过原生的CSS来实现这些过渡动画,但是React社区为我们提供了react-transition-group用来完成过渡动画

  • React曾为开发者提供过动画插件 react-addons-css-transition-group,后由社区维护,形成了现在的 react-transition-group

这个库可以帮助我们方便的实现组件的 入场 和 离场 动画,使用时需要进行额外的安装

1. 安装

npm install react-transition-group --save
react-transition-group本身非常小,不会为应用程序增加过多的负担

2. 主要组件

  • Transition

  • 该组件是一个和平台无关的组件(不一定要结合CSS);

  • 在前端开发中,一般是结合CSS来完成样式,所以比较常用的是CSSTransition

  • CSSTransition

  • 在前端开发中,通常使用CSSTransition来完成过渡动画效果

  • 如果只有一个组件,直接使用这个即可

  • SwitchTransition

  • 两个组件显示和隐藏切换时,使用该组件

  • TransitionGroup

  • 将多个动画组件包裹在其中,一般用于列表中元素的动画

二、CSSTransition

1. 概念

CSSTransition是基于Transition组件构建的
CSSTransition执行过程中,有三个状态:appear、enter、exit;

它们有三种状态,需要定义对应的CSS样式:

  • 第一类,开始状态:

  • 对于的类是-appear、-enter、-exit

  • 第二类:执行动画:

  • 对应的类是-appear-active、-enter-active、-exit-active

  • 第三类:执行结束:

  • 对应的类是-appear-done、-enter-done、-exit-done;

2. 常见属性

in:触发进入或者退出状态

  • 如果添加了unmountOnExit={true},那么该组件会在执行退出动画结束后被移除掉

  • 当in为true时,触发进入状态,会添加-enter、-enter-acitve的class开始执行动画

  • 当动画执行结束后,会移除两个class,并且添加-enter-done的class

  • 当in为false时,触发退出状态,会添加-exit、-exit-active的class开始执行动画

  • 当动画执行结束后,会移除两个class,并且添加-enter-done的class;

classNames:动画class的名称

决定了在编写css时,对应的class名称:比如card-enter、card-enter-active、card-enter-done

timeout:过渡动画的时间

appear:是否在初次进入添加动画(需要和in同时为true)

unmountOnExit:退出后卸载组件

其他属性

3. 对应的钩子函数

主要为了检测动画的执行过程,来完成一些JavaScript的操作
  • onEnter:在进入动画之前被触发

  • onEntering:在应用进入动画时被触发

  • onEntered:在应用进入动画结束后被触发

  • onExit:离开动画之前被触发

  • onExiting:离开动画时被触发

  • onExited:离开动画之后被触发

4. 简单使用

组件

import React, { PureComponent } from 'react';
// 1. 导入动画组件
import { CSSTransition } from 'react-transition-group';
// 2. 导入动画样式
import './style.css';

export class App extends PureComponent {
  constructor() {
    super();
    this.state = {
      isShow: true
    };
  }
  render() {
    const { isShow } = this.state;
    return (
      <>
        <button onClick={(e) => this.setState({ isShow: !isShow })}>切换</button>
        {/* 
          in: 控制动画的状态
          timeout: 动画执行的时间
          classNames: 动画的类名前缀
          unmountOnExit: 动画退场时,移除dom
          appear: 初次加载时,是否执行动画
          onEnter: 开始进入动画
          onEntering: 执行进入动画
          onEntered: 进入动画结束
          onExit: 开始离开动画
          onExiting: 正在离开动画
          onExited: 离开动画结束
        */}
        <CSSTransition
          in={isShow}
          unmountOnExit={true}
          classNames='coder'
          timeout={2000}
          appear
          onEnter={(e) => console.log('开始进入动画')}
          onEntering={(e) => console.log('执行进入动画')}
          onEntered={(e) => console.log('进入动画结束')}
          onExit={(e) => console.log('开始离开动画')}
          onExiting={(e) => console.log('正在离开动画')}
          onExited={(e) => console.log('离开动画结束')}>
          <h2>APP</h2>
        </CSSTransition>
      </>
    );
  }
}

export default App;

样式

/* 第一次加载时显示 */
.coder-appear{
  transform: translateX(-200px);
}
.coder-appear-active{
  transform: translateX(0);
  transition: all 2s ease-in;
}

/* 进入动画 */
.coder-enter{
  opacity: 0;
}
.coder-enter-active{
    opacity: 1;
    transition: all 2s ease-in;
}

/* 退出动画 */
.coder-exit{
  opacity: 1;
}
.coder-exit-active{
  opacity: 0;
  transition: all 2s ease-in;
}

效果

5. 问题

在严格模式下,使用可能会报错,因为CSSTransition还在使用之前过期的API => findDOMNode

解决方式一 : 关闭严格模式

解决方式二 : 改变源码

解决方式三 : 设置ref

import React, { Fragment, PureComponent, createRef } from 'react';
import { CSSTransition } from 'react-transition-group';
import './style.css';

export class App extends PureComponent {
  constructor() {
    super();
    this.state = {
      isShow: true
    };
    // 1. 创建ref对象
    this.titleRef = createRef();
  }
  render() {
    const { isShow } = this.state;
    return (
      <Fragment>
        <button onClick={(e) => this.setState({ isShow: !isShow })}>切换</button>
        {/* 
          nodeRef: 用于指定动画的DOM节点
        */}
        <CSSTransition
          // 3. 绑定这个ref对象
          nodeRef={this.titleRef}
          in={isShow}
          unmountOnExit={true}
          classNames='coder'
          timeout={2000}
          appear
          onEnter={(e) => console.log('开始进入动画')}
          onEntering={(e) => console.log('执行进入动画')}
          onEntered={(e) => console.log('进入动画结束')}
          onExit={(e) => console.log('开始离开动画')}
          onExiting={(e) => console.log('正在离开动画')}
          onExited={(e) => console.log('离开动画结束')}>
          {/* 2. 绑定这个ref对象 */}
          <h2 className='title' ref={this.titleRef}>
            APP
          </h2>
        </CSSTransition>
      </Fragment>
    );
  }
}

export default App;

三、SwitchTransition

SwitchTransition可以完成两个组件之间切换的炫酷动画:
  • 比如有一个按钮需要在on和off之间切换,希望看到on先从左侧退出,off再从右侧进入

  • 这个动画在vue中被称之为 vue transition modes

  • react-transition-group中使用SwitchTransition来实现该动画

SwitchTransition中主要有一个属性:mode,有两个值
  • in-out:表示新组件先进入,旧组件再移除

  • out-in:表示就组件先移除,新组件再进入

SwitchTransition的使用
  • SwitchTransition组件里面要有CSSTransition或者Transition组件,不能直接包裹想要切换的组件

  • SwitchTransition里面的CSSTransition或Transition组件不再像以前那样接受in属性来判断元素是何种状态,取而代之的是key属性

  • key必须要指定两个不同的值,用来判定是否切换

代码

import React, { Fragment, PureComponent, createRef } from 'react';
import { SwitchTransition, CSSTransition } from 'react-transition-group';
import './style.css';

export class App extends PureComponent {
  constructor() {
    super();
    this.state = {
      isShow: true
    };
    this.titleRef = createRef();
  }
  render() {
    const { isShow } = this.state;
    return (
      <Fragment>
        <button onClick={(e) => this.setState({ isShow: !isShow })}>切换</button>
        {/* 
          out-in: 新元素先执行进入动画,动画执行完毕后,旧元素执行离开动画
          in-out: 旧元素先执行离开动画,动画执行完毕后,新元素执行进入动画
        */}
        <SwitchTransition mode='out-in'>
          <CSSTransition
            // 必须有可以更改的key值
            key={isShow ? 'coder' : 'star'}
            nodeRef={this.titleRef}
            unmountOnExit={true}
            classNames='coder'
            timeout={2000}
            onEnter={(e) => console.log('开始进入动画')}
            onEntering={(e) => console.log('执行进入动画')}
            onEntered={(e) => console.log('进入动画结束')}
            onExit={(e) => console.log('开始离开动画')}
            onExiting={(e) => console.log('正在离开动画')}
            onExited={(e) => console.log('离开动画结束')}>
            <h2 className='title' ref={this.titleRef}>
              <span>{isShow ? 'coder' : 'star'}</span>
            </h2>
          </CSSTransition>
        </SwitchTransition>
      </Fragment>
    );
  }
}

export default App;

效果

四、TransitionGroup

当有一组动画时,需要将这些CSSTransition放入到一个TransitionGroup中来完成动画

样式

/* 第一次加载时显示 */
.book-appear{
  transform: translateX(-200px);
}
.book-appear-active{
  transform: translateX(0);
  transition: all 1s ease-in;
}

/* 进入动画 */
.book-enter{
  transform: translateX(100px);
  opacity: 0;
}
.book-enter-active{
  transform: translateX(0);
  opacity: 1;
  transition: all 1s ease-in;
}

/* 退出动画 */
.book-exit{
  opacity: 1;
  transform: translateX(0);

}
.book-exit-active{
  opacity: 0;
  transform: translateX(-100px);
  transition: all 1s ease-in;
}

组件

import React, { PureComponent } from 'react';
import { TransitionGroup, CSSTransition } from 'react-transition-group';
import './style.css';
export class App extends PureComponent {
  constructor() {
    super();
    this.state = {
      books: [
        { name: 'React', id: 1 },
        { name: 'Vue', id: 2 },
        { name: 'Angular', id: 3 }
      ]
    };
  }
  // 增加数据
  addBooks() {
    const books = [...this.state.books];
    books.push({
      name: 'JavaScript高级程序设计',
      id: new Date().getTime()
    });
    this.setState({ books });
  }
  // 删除数据
  deleteBooks(index) {
    const books = [...this.state.books];
    books.splice(index, 1);
    this.setState({ books });
  }
  render() {
    const { books } = this.state;
    /**
     * component 属性指定过渡组件的类型,默认是div
     * key 属性是必须唯一,如果使用index作为key,会出现删除数据时,动画效果不对的问题
     */
    return (
      <div>
        <TransitionGroup component='ul'>
          {books.map((item, index) => {
            return (
              <CSSTransition key={item.id} appear classNames='book' timeout={1000}>
                <li>
                  <span>{item.name}</span>
                  <button onClick={(e) => this.deleteBooks(index)}>删除</button>
                </li>
              </CSSTransition>
            );
          })}
        </TransitionGroup>
        <button onClick={(e) => this.addBooks()}>增加数据</button>
      </div>
    );
  }
}

export default App;

效果

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/727849.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

MySQL表/用户权限等基本操作

MySQL表操作练习题&#xff1a; 第一题&#xff1a; 具体要求如下所示&#xff1a; 1. 创建数据库Market&#xff1a; CREATE DATABASE Market;2. 创建customers表&#xff1a; 表结构如图所示&#xff1a; CREATE TABLE customers( c_num INT(11) PRIMARY KEY, c_name…

TiDB(8):技术内幕之计算

1 关系模型到 Key-Value 模型的映射 在这我们将关系模型简单理解为 Table 和 SQL 语句&#xff0c;那么问题变为如何在 KV 结构上保存 Table 以及如何在 KV 结构上运行 SQL 语句。 假设我们有这样一个表的定义&#xff1a; CREATE TABLE User {ID int,Name varchar(20),Role …

基于springboot+vue的文超市进销存管理系统(源代码+数据库+12000字论文)083

基于springbootvue的文超市进销存管理系统(源代码数据库12000字论文)083 一、系统介绍 (本项目有ssmvue版本) 本系统分为管理员、用户、员工三种角色 用户角色包含以下功能&#xff1a; 登录、注册、购物车、订单提交、商品评论、收藏、充值、收货地址管理、收藏管理、订单…

NXP-无感BLDC代码MCSPTE1AK116_BLDC_6Step代码详解

目录 开发平台 工程目录 Generated_Code Sources Config 电机的参数 BLDC参数 无感模式下的一些参数 Peripherals FTM/PDB/ADC配置参数 actuate_s32k meas_s32k motor_structure state_machine main main()主函数 PORT_IRQHandler() PDB0_IRQHandler() FTM1…

最大正方形 · Maximal Square

链接&#xff1a; 题解&#xff1a;九章算法 - 帮助更多程序员找到好工作&#xff0c;硅谷顶尖IT企业工程师实时在线授课为你传授面试技巧 1.暴力的方法&#xff1a;遍历每一个&#xff08;i&#xff0c;j&#xff09;位置&#xff0c;如果当前点为1&#xff0c;则以当前节点为…

如何编写PlantUml文本绘图时序图

效果如图 代码示例 startumlparticipant "上游" as BEGIN participant "SFTP" as SFTP control "文件系统" as FILE participant "业务系统" as BUSactivate BEGIN BEGIN ->SFTP: 上传文件 activate SFTP autonumber 1.0 FILE -&g…

常用的网址

画图网页&#xff1a; https://www.processon.com/diagrams 二进制转换网页&#xff1a; https://tool.oschina.net/hexconvert/ 在线网络计算器 https://www.sojson.com/convert/subnetmask.html 学习网站掘金&#xff1a; https://juejin.cn 注册外网账号网页&#xff1a…

使用Lambda表达式对List<Map<String,Object>>中key值相同的Map进行分组合并

现有两张表A表和B表&#xff0c;A表存放的是各省市的认证次数&#xff0c;B表存放的是各省市的申领次数&#xff0c;重点关注dq,cs这两个字段&#xff0c;其他的字段可忽略 A表&#xff08;省市认证次数表&#xff09; B表&#xff08;省市申领次数表&#xff09; 项目中有以下…

辅助性能优化——长安链性能分析工具原理及用法

如何提升区块链系统性能是很多开发者都会关注的事&#xff0c;但是有些对区块链并非十分熟悉的开发者可能会感到没有头绪。长安链提供了性能分析工具帮助开发者梳理系统耗时&#xff0c;优化系统性能。下面对长安链性能分析工具原理及使用进行介绍。 一、 概述 time_counter.s…

Windows兼容性设置图文教程,Windows兼容模式怎么设置?服务器兼容是什么意思?服务器兼容性怎么改?

兼容性&#xff08;compatibility&#xff09;是指硬件之间、软件之间或是软硬件组合系统之间的相互协调工作的程度。兼容的概念比较广&#xff0c;相对于硬件来说&#xff0c;几种不同的电脑部件&#xff0c;如CPU、主板、显示卡等&#xff0c;如果在工作时能够相互配合、稳定…

备战秋招004(20230706)

文章目录 前言一、今天学习了什么&#xff1f;二、关于问题的答案1.SE 总结 前言 提示&#xff1a;这里为每天自己的学习内容心情总结&#xff1b; Learn By Doing&#xff0c;Now or Never&#xff0c;Writing is organized thinking. 目前的想法是&#xff0c;根据 Java G…

三种方法将视频转换为AVI格式,与大家分享!

将视频转换为AVI格式是常见的需求&#xff0c;因为AVI格式具有广泛的兼容性和可编辑性。本文将介绍三种常用的方法&#xff0c;包括记灵在线工具、剪映和格式工厂。这些方法简单易行&#xff0c;帮助您将视频文件快速转换为AVI格式&#xff0c;满足不同的需求。 方法一&#x…

EasyCVR接入大量设备级联后出现分组加载异常是什么原因?

EasyCVR可拓展性强、视频能力灵活、部署轻快&#xff0c;可支持的主流标准协议有GB28181、RTSP/Onvif、RTMP等&#xff0c;以及厂家私有协议与SDK接入&#xff0c;包括海康Ehome、海大宇等设备的SDK等&#xff0c;能对外分发RTSP、RTMP、FLV、HLS、WebRTC等格式的视频流。 有用…

接口自动化测试实战之pytest框架+allure讲解

一、前言 本文章主要会讲解Python中pytest框架的讲解&#xff0c;介绍什么是pytest、为何要测试、为何使用以及参考和扩展等等&#xff0c;话不多说&#xff0c;咱们直接进入主题哟。 二、pytest讲解 2.1 什么是pytest&#xff1f; pytest是一款单元测试框架&#xff0c;在…

尚硅谷Docker实战教程-笔记10【高级篇,Docker微服务实战】

尚硅谷大数据技术-教程-学习路线-笔记汇总表【课程资料下载】视频地址&#xff1a;尚硅谷Docker实战教程&#xff08;docker教程天花板&#xff09;_哔哩哔哩_bilibili 尚硅谷Docker实战教程-笔记01【基础篇&#xff0c;Docker理念简介、官网介绍、平台入门图解、平台架构图解】…

浅析住宅小区电动车充电桩的电气设计与平台管理系统

安科瑞电气股份有限公司 上海嘉定201801 摘要&#xff1a;根据目前对于新能源汽车发展规划及政策&#xff0c;以及国内外充电设施的主要类型和技术参数。论述地下车库电动汽车充电桩的供配电系统的设计及设计过程中需要注意的一些问题。 关键词&#xff1a;充电桩&#xff1b…

力扣题库刷题笔记36--有效的数独

1、题目如下&#xff1a; 2、个人Python代码实现如下&#xff1a; 3、个人Python代码思路&#xff1a; 先放一个AI解释的思路&#xff1a; 个人理解&#xff0c;本题思路其实很简单&#xff0c;判断每一行、每一列、每一个3*3的子数独是否存在重复数字&#xff0c;如果存在则返…

不用转化器把pdf转化成Excel,分享两个实用方法!

将PDF文件转换为Excel格式通常是进行数据提取和分析的重要步骤。尽管市面上有许多PDF转Excel的工具&#xff0c;但本文将介绍两种无需使用转换器的实用方法&#xff0c;分别是复制粘贴法和使用记灵在线工具。这些方法简单易行&#xff0c;帮助您快速将PDF中的数据提取到Excel表…

第21章:索引优化与查询优化

一、索引优化与查询优化 1.什么情况下要进行数据库调优 ①索引失效&#xff0c;没有充分利用到索引---索引建立 ②关联查询太多join---SQL优化 ③服务器调优和各个参数的设置---调整my.cnf ④数据过多---分库分表 2.SQL优化的技术 ①物理查询优化&#xff1a;通过索引和…

图论算法:DFS求有向图或无向图两点间所有路径及Dijkstra算法求最短路径

1、目的 1)根据有向图获取指定起终点的所有路径; 2)直接求解两点间最短路径。 2、示例效果 2.1 原始数据 路线起终点整理如下: // 共计12个顶点,19条边。 (起点,终点,1)最后的1代表起点终点是连通的。 起点,终点,1:2 4 1 起点,终点,1:9 10 1 起点,终点,1:…