Animations in React

Intro to

by Keziyah Lewis

reactjsexample.com/tic-tac-toe-game-on-react-js/

hackernoon.com

react-native-loader (github)

react-motion demo

reaction-page-transition demo

react-screen

lottie-react-native

How do you animate a React component?

Today, we will use

  • CSS transitions + react-transition-group
  • a 3rd party package (react-screen)
  • Higher Order Components
  • TweenMax

to animate a UI for a fake e-commerce site.

Because the sharing economy has officially gone too far.

Task 1

Animate the appearance of a search box upon clicking a 

search icon. 

Tools needed:

  • react-transition-group (fork and download from Github)

Approach

CSS transitions

source: codeproject.com

Text

from Facebook's React docs

Starting Point

Import CSSTransitionGroup, and wrap it around Search

import React, { Component } from 'react'
import {Link} from 'react-router'
import CSSTransitionGroup from './react-transition-group/src/CSSTransitionGroup'
import Search from './Search'


class Hero extends Component {
    render() {
        return (
            <div id="heroimg">
                <div id="hero-text" className="container">
                    <div className="col-md-8">
                        <h1>New hair don't care.</h1>
                        <h4>Boot the bundles. Skip the beauty shop. Buy hair for your next look from people right in your neighborhood.</h4>
                    </div> 
                </div>
                <CSSTransitionGroup>
                    <Search/>
                </CSSTransitionGroup>
            </div>
        )
    }
}

export default Hero 

Hero.jsx

Add CSS

/*Keep consistency with class names
**Use "appear" for initial loading of the page
**Use "enter" for components that load after the page load
    such as those that load in response to an event. 
**The transition seconds in CSS should match the milliseconds passed into
    CSSTransitionGroup
*/

/*Conditions for when the search form first appears.*/
 .search-enter {
    width: 0; 
 }

/*Specify length of time and end result of transition, before
the search component renders as normal*/

  .search-enter.search-enter-active {
    width: 90%; 
    transition: width 0.4s ease-in; 
  }

style.css

Add transition props

<CSSTransitionGroup 
transitionName="search"
 transitionAppear={false}
 transitionEnterTimeout={400}
 transitionEnter={true}
 transitionLeave={false}>
                     <Search/>
 </CSSTransitionGroup>

Hero.jsx

Make sure the transition name and timeout values reflect what is in style.css.

Add click handler + logic

class Hero extends Component {
    constructor() {
        super()

        this.state = { shouldShowSearch: false }
        this.showSearch = this.showSearch.bind(this)
  }

    showSearch() {
        this.setState({ shouldShowSearch: true })
    }

    render() {
        const searchIcon = <button className="mdl-button mdl-js-button" 
                            id="icon-search" onClick={this.showSearch}>
                            <span className="glyphicon glyphicon-search" 
                            aria-hidden="true"></span>
                            </button>

Hero.jsx

 <CSSTransitionGroup transitionName="search" transitionAppear={false}
     transitionEnterTimeout={400} transitionEnter={true} transitionLeave={false}>
                    {!this.state.shouldShowSearch && searchIcon}
                    {this.state.shouldShowSearch && <Search />}
  </CSSTransitionGroup>

There you go!

Task 2

Animate page transitions. 

Approach 

CSS Transitions

Tools Needed

Just CSS and keyframes or react-transition-group

Starting Point

'use strict'
import React from 'react'
import {Router, Route, IndexRoute, browserHistory} from 'react-router'
import {render} from 'react-dom'

import App from './components/App'
import Home from './components/Home'
import Results from './components/Results'

render(
    <Router history={browserHistory}>
      <Route path="/" component={App}>
          <IndexRoute component={Home} />
          <Route path="/results" component={Results} />
      </Route>
    </Router>,
  document.getElementById('main')
)

main.jsx

import React, { Component } from 'react'

class App extends Component {
    render() {
        return (
            <div>
               {this.props.children}
            </div>
        )
    }
}

export default App

App.jsx

Wrap the App's children in <CSSTransitionGroup /> and clone them.

import React, { Component } from 'react'
import CSSTransitionGroup from './react-transition-group/src/CSSTransitionGroup'

class App extends Component {
    render() {
        return (
            <div>
                //We will add transition properties later...
                <CSSTransitionGroup >
                //We need to clone the children because we are adding props
                    {React.cloneElement(this.props.children, 
                    {key: this.props.location.pathname})}
                //Won't work without the key
                //location.pathname -> /results
                </CSSTransitionGroup>
            </div>
        )
    }
}

export default App

App.jsx

/*Conditions for when the search form first appears.*/
.fade-enter {
    opacity: 0.01; 
}

/*Specify length of time and end result of transition, before
the search component renders as normal*/

.fade-enter.fade-enter-active {
    opacity: 1; 
    transition: opacity 0.8s ease-in;
}

/*Conditions for when the search icon first leaves*/
.fade-leave {
    opacity: 1
}
/*When it first appears you can't see anything*/
.fade-appear {
    opacity: 0;
  }

/*But while it's appearing...*/
.fade-appear.fade-appear-active {
    opacity: 1;
    transition: opacity 0.8s ease-in;
}

style.css

Add CSS

//Use keyframes without react-transition-group
//Just send class names to components as a prop

.anim {
    animation: anim 1s ease-out 0s 1; 
}


@keyframes anim {
    0% {
        transform: translateX(-100%);
    }
    100% {
         transform: translateX(0); 
    }
}

Alternative: Use keyframes

style.css

 <CSSTransitionGroup transitionName="fade" 
    transitionAppear={true} transitionEnter={true} 
    transitionLeave={true}
    transitionLeaveTimeout={200} 
    transitionEnterTimeout={800}
    transitionAppearTimeout={800}>
    {React.cloneElement(this.props.children, 
    {key: this.props.location.pathname})}
  </CSSTransitionGroup>

App.jsx

Add transition properties

Or if just using keyframes, add the className to the div that holds the component, and you don't need  <CSSTransitionGroup />

done & done

Task  3

Add a carousel for the how-to guide. 

Approach

Use a node package,

react-screen by Gao Sun. 

Tools needed

  • react-screen
  • link to CSS
  • see website 

source: react-view-pager on github

Starting Point

class HowTo extends Component {
    render() {
        return (
            <div className="container">
                <div id="tutorial-header">
                    <h1 id="hair-tutorial-header">Hairbnb is
                     revolutionizing the way you shop for hair.
                     Here's how. 
                    </h1>
                </div>
                <HowToCard image={image1} header={header1} paragraph={paragraph1}/>
                <HowToCard image={image2} header={header2} paragraph={paragraph2}/>
                <HowToCard image={image3} header={header3} paragraph={paragraph3}/>
                <HowToCard image={image4} header={header4} paragraph={paragraph4}/>
            </div>
        )
}
}

HowTo.jsx

<HowToCard />

That's it!

import React, { Component } from 'react'
import { Screen, ScreenSlice } from 'react-screen';
//Don't forget to link to the CSS file

class HowTo extends Component {
    render() {
        return (
            <div className="container">
                <div id="tutorial-header">
                    <h1 id="hair-tutorial-header">Hairbnb is
                     revolutionizing the way you shop for hair.
                     Here's how. 
                    </h1>
                </div>
                <Screen>
                    <ScreenSlice>
                        <HowToCard image={image1} header={header1} paragraph={paragraph1}/>
                    </ScreenSlice>
                    <ScreenSlice>
                         <HowToCard image={image2} header={header2} paragraph={paragraph2}/>
                    </ScreenSlice>
                    <ScreenSlice>
                        <HowToCard image={image3} header={header3} paragraph={paragraph3}/>
                    </ScreenSlice>
                    <ScreenSlice>
                        <HowToCard image={image4} header={header4} paragraph={paragraph4}/>
                    </ScreenSlice>
                </Screen>
            </div>
        )
}
}

HowTo.jsx

Task 4

Make each results card flip to reveal a "report user" form. 

Approach

Higher order components with TweenMax. 

Tools Needed

  • link to TweenMax in script tag in HTML
  • react-transition-group

source: codemyui.com

Starting Point

We have a ResultsCard component and a ReportCard component

<ResultsCard />

<ReportCard />

MakeFlip.jsx

Write HOC

import React from 'react'
import {findDOMNode, render} from 'react-dom'

function makeFlip (Component) {
    return class Flip extends React.Component {

    //This is called at the same time as componentDidMount
    componentWillEnter (callback) {
        const el = findDOMNode(this)
    }

    //This is called when the child has been removed from the ReactTransitionGroup
    componentWillLeave(callback) {
        const el = findDOMNode(this)
    }

    render() {
        return <Component { ...this.props } />
    }
    }
}

//Animation hooks that are provided a callback will prevent other 
//animations from running until the callback is called. 

This function takes a regular component and returns an enhanced component. 

Animate with TweenMax

 

The fromTo function lets you specify the start and end.

of an animation. 

TweenMax.fromTo(el, 
                1, 
                {rotation: 180, opacity: 0}, 
                {rotation: 0, opacity: 1, onComplete: callback}
              )

1.  The element that is being animated. 

2. Time in seconds

3. Starting point

4. Ending point.

function makeFlip (Component) {
    return class Flip extends React.Component {

    componentWillEnter (callback) {
        const el = findDOMNode(this)
        TweenMax.fromTo(el, 0.5, {rotationY: 180, opacity: 0}, 
        {rotationY: 0, opacity:1, onComplete: callback})
    }

//This hook is not needed for this animation
// (that's why time is set to 0.) 
//But if I were using it, I would have the card
// flip on the Y axis -180deg.
    componentWillLeave(callback) {
        const el = findDOMNode(this)
        TweenMax.fromTo(el, 0, {rotationY:0, opacity:0}, 
        {rotationY: -180, opacity: 0, onComplete: callback})
    }

    render() {
        return <Component { ...this.props } />
    }
    }
}

Add animation to lifecycle hooks.

MakeFlip.jsx

Export the enhanced functions.

import React from 'react'
import {findDOMNode, render} from 'react-dom'
import ResultCard from './ResultCard'
import ReportCard from './ReportCard'


function makeFlip (Component) {
    return class Flip extends React.Component 
        //...see previous slide
}

//Create the animated front card by passing it into the HOC above
export const Front = makeFlip(ResultCard)

//Create the animated back card by passing it into the HOC above
export const Back = makeFlip(ReportCard)

MakeFlip.jsx

import React, {Component} from 'react'
import TransitionGroup from './react-transition-group/src/TransitionGroup'
import {Front, Back} from './MakeFlip'

export default class Results extends Component {
  state = {
    shouldShowResult: true,
    shouldShowReport: false
  }

  toggle = () => {
    this.setState({
      shouldShowResult: !this.state.shouldShowResult,
      shouldShowReport: !this.state.shouldShowReport
    })
  }

  render () {
    return (
    <div>
  
//For as many cards as you have...
      <TransitionGroup>
        { this.state.shouldShowResult && <Front/>}
        { this.state.shouldShowReport && <Back/>}
      </TransitionGroup>

      <button onClick={this.toggle}>toggle</button>

    </div>
    )
  }
}

Results.jsx

Add toggling logic. Wrap components in a <TransitionGroup />

Render a toggle button or pass the toggle function as a prop to <Front /> and <Back />

...NOT <CSSTransitionGroup />

Take aways:

  • You don't need jQuery
  • All you really need is CSS
  • Use libraries or packages
  • HOCs are great for animation
  • Be as creative or as lazy as you want

Learn more.

Thank you

lottie-react

Link to presentation on               -  @KeziyahL

github.com/keziyah

Tech Talk: Intro to Animations in React

By Keziyah Lewis

Tech Talk: Intro to Animations in React

  • 2,173