Skip to content

过渡动画

react-transition-group

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

  • 这个库可以帮助我们方便的实现组件的 入场离场 动画,使用时需要进行额外的安装
bash
npm install react-transition-group --save

react-transition-group 主要包含四个组件:

  • Transition
  • CSSTransition
  • SwitchTransition

react-transition-group主要包含四个组件:

  • Transition
    • 该组件是一个和平台无关的组件(不一定要结合 CSS)
    • 在前端开发中,我们一般是结合 CSS 来完成样式,所以比较常用的是 CSSTransition
  • CSSTransition
    • 在前端开发中,通常使用 CSSTransition 来完成过渡动画效果
  • SwitchTransition
    • 两个组件显示和隐藏切换时,使用该组件
  • TransitionGroup
    • 将多个动画组件包裹在其中,一般用于列表中元素的动画

CSSTransition

CSSTransition 是基于 Transition 组件构建的:

  • CSSTransition 执行过程中,有三个状态:appear、enter、exit
  • 它们有三种状态,需要定义对应的 CSS 样式:
    • 第一类,开始状态:对于的类是 -appear-enter-exit
    • 第二类:执行动画:对应的类是 -appear-active-enter-active-exit-active
    • 第三类:执行结束:对应的类是 -appear-done-enter-done-exit-done

CSSTransition常见对应的属性:

  • 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 名称:比如 xxx-enterxxx-enter-activexxx-enter-done
  • timeout
    • 是否在初次进入添加动画(需要和 in 同时为 true)
  • unmountOnExit
    • 退出后卸载组件
  • http://reactcommunity.org/react-transition-group/transition
  • CSSTransition 对应的钩子函数:主要为了检测动画的执行过程,来完成一些 JavaScript 的操作
    • onEnter:在进入动画之前被触发
    • onEntering:在应用进入动画时被触发
    • onEntered:在应用进入动画结束后被触发
jsx
import React, { createRef, PureComponent } from 'react'
import { CSSTransition } from 'react-transition-group'

export class App extends PureComponent {
  constructor(props) {
    super(props)
    this.state = {
      isShow: true
    }
    this.sectionRef = createRef()
  }

  render() {
    const { isShow } = this.state

    return (
      <div>
        <button onClick={e => this.setState({ isShow: !isShow })}>切换</button>
        <CSSTransition
          nodeRef={this.sectionRef}
          in={isShow}
          unmountOnExit={true}
          classNames='why'
          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('执行离开结束')}
        >
          <div className='section' ref={this.sectionRef}>
            <h2>哈哈哈</h2>
          </div>
        </CSSTransition>
      </div>
    )
  }
}
css
/* 进入动画 */
.why-appear, .why-enter {
  opacity: 0;
}
.why-appear-active, .why-enter-active {
  opacity: 1;
  transition: opacity 2s ease;
}

/* 离开动画 */
.why-exit {
  opacity: 1;
}
.why-exit-active {
  opacity: 0;
  transition: opacity 2s ease;
}

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 属性
jsx
import React, { PureComponent } from 'react'
import { SwitchTransition, CSSTransition } from 'react-transition-group'

export class App extends PureComponent {
  constructor() {
    super()

    this.state = {
      isLogin: true
    }
  }

  render() {
    const { isLogin } = this.state

    return (
      <div>
        <SwitchTransition mode='out-in'>
          <CSSTransition
            key={isLogin ? 'exit' : 'login'}
            classNames='login'
            timeout={1000}
          >
            <button onClick={e => this.setState({ isLogin: !isLogin })}>
              {isLogin ? '退出' : '登录'}
            </button>
          </CSSTransition>
        </SwitchTransition>
      </div>
    )
  }
}
css
.login-enter {
  transform: translateX(100px);
  opacity: 0;
}
.login-enter-active {
  transform: translateX(0);
  opacity: 1;
  transition: all 1s ease;
}

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

TransitionGroup

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

jsx
import React, { PureComponent } from 'react'
import { TransitionGroup, CSSTransition } from 'react-transition-group'

export class App extends PureComponent {
  constructor() {
    super()

    this.state = {
      books: [
        { id: 111, name: '你不知道JS', price: 99 },
        { id: 222, name: 'JS高级程序设计', price: 88 },
        { id: 333, name: 'Vuejs高级设计', price: 77 }
      ]
    }
  }

  addNewBook() {
    const books = [...this.state.books]
    books.push({ id: new Date().getTime(), name: 'React高级程序设计', price: 99 })
    this.setState({ books })
  }

  removeBook(index) {
    const books = [...this.state.books]
    books.splice(index, 1)
    this.setState({ books })
  }

  render() {
    const { books } = this.state

    return (
      <div>
        <h2>书籍列表:</h2>
        <TransitionGroup component='ul'>
          {books.map((item, index) => {
            return (
              <CSSTransition key={item.id} classNames='book' timeout={1000}>
                <li>
                  <span>
                    {item.name}-{item.price}
                  </span>
                  <button onClick={e => this.removeBook(index)}>删除</button>
                </li>
              </CSSTransition>
            )
          })}
        </TransitionGroup>
        <button onClick={e => this.addNewBook()}>添加新书籍</button>
      </div>
    )
  }
}
css
.book-enter {
  transform: translateX(150px);
  opacity: 0;
}
.book-enter-active {
  transform: translateX(0);
  opacity: 1;
  transition: all 1s ease;
}

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

常备不懈,才能有备无患