Creating scroll-triggered animations in React
At the first company I worked on I was asked to do something that looked impossible at first. They wanted a website unlike any other, with complex animations that seamlessly unfold as the user scroll. I didn't refuse to do it, though. I decided to take the chance and that's how Animate Screen was created.
Animate Screen is an npm package that allows you to develop complex animations in React that elegantly unfold as you scroll. It is built upon the GSAP library, which is the most robust animation library for the web, and it simplifies the logic you would need to handle if you just used this library.
Dependencies
This package is built upon the GSAP library, but you don't need an in-depth knowledge of GSAP. A modest familiarity with the properties used in defining a tween is all that's necessary.
Installation
To install this package you have to run the following command.
npm install animate-screen
Screen animation
To begin using this package, you have to include the AnimateScreen
component. This component contains all the components that have to be animated and defines all the animation phases.
import { AnimateScreen } from 'animate-screen'
function App() {
return (
<AnimateScreen phases={null} config={null}>
{/* components */}
</AnimateScreen>
)
}
export default App
Animation phases
To define the animation phases you can create a phases.js file and create an instance of the ScreenAnimationPhases
class.
/* phases.js */
import { ScreenAnimationPhases } from 'animate-screen'
const phases = new ScreenAnimationPhases()
phases.add('boxes-right', 0, 1, true)
phases.add('boxes-hidden', 0, 1, true)
export default phases
The add()
method is used to add a phase and it receives the following arguments:
- name: Name of the phase.
- delay: How many screens have to be scrolled to start the phase.
- duration: How many screens have to be scrolled to finish the phase.
- snap: If it has to snap at the end of the phase or not.
After that, you have to pass this instance to the phases
prop.
// ...
import phases from 'phases'
function App() {
return (
<AnimateScreen phases={phases} config={null}>
{/* components */}
</AnimateScreen>
)
}
// ...
Configuration
You also need to define the configuration of the animation, and you define it by passing an object with these properties to the config
props.
// ...
const config = {
scrub: 2, // Takes 2 seconds to "catch up" to the scrollbar
snap: {
delay: 0.2, // Wait 0.2 seconds before snapping
duration: {
min: 1, // Snap takes at least 1 second
max: 2, // Snap takes at most 2 seconds
},
ease: 'power1.inOut', // Snap uses the "power1.inOut" easing function
},
}
function App() {
return (
<AnimateScreen phases={phases} config={config}>
{/* components */}
</AnimateScreen>
)
}
// ...
Component animation
You are now ready to create the components that have to be animated. You can create a boxes folder with the boxes.jsx and boxes.module.css files.
/* boxes/boxes.jsx */
import styles from './boxes.module.css'
function Boxes() {
return (
<>
<div className={styles.box1} />
<div className={styles.box2} />
</>
)
}
export default Boxes
/* boxes/boxes.module.css */
.box1,
.box2 {
position: fixed;
width: 100px;
height: 100px;
top: calc(50% - 50px);
background: red;
}
.box1 {
left: calc(25% - 50px);
}
.box2 {
left: calc(75% - 50px);
}
// ...
import Boxes from './boxes/boxes'
// ...
function App() {
return (
<AnimateScreen phases={phases} config={config}>
<Boxes />
</AnimateScreen>
)
}
// ...
There is a small adjustment you have to do to the component. To animate it you have to use the AnimateComponent
component.
// ...
import { AnimateComponent } from 'animate-screen'
function Boxes() {
return (
<AnimateComponent
animationPhases={null}
render={() => (
<>
<div className={styles.box1} />
<div className={styles.box2} />
</>
)}
/>
)
}
// ...
Element tags
To animate the elements within the component, you need to assign tags to these elements. To accomplish this, use the ref()
function.
// ...
function Boxes() {
return (
<AnimateComponent
animationPhases={null}
render={(ref) => (
<>
<div ref={ref('box')} className={styles.box1} />
<div ref={ref('box')} className={styles.box2} />
</>
)}
/>
)
}
// ...
You can also pass multiple tags to the same element if you want.
// ...
function Boxes() {
return (
<AnimateComponent
animationPhases={null}
render={(ref) => (
<>
<div ref={ref('box', 'box-1')} className={styles.box1} />
<div ref={ref('box', 'box-2')} className={styles.box2} />
</>
)}
/>
)
}
// ...
Animation
You can now define the animation of the component in the different phases. To do it, you can create a boxes.phases.js file and use the ComponentAnimationPhases
class.
/* boxes/boxes.phases.js */
import { ComponentAnimationPhases } from 'animate-screen'
const phases = new ComponentAnimationPhases()
phases.set('boxes-right', null)
phases.set('boxes-hidden', null)
export default phases
The set()
method is used to set an animation for a phase and it receives the following arguments:
- name: Name of the phase.
- animation: Animation of the phase.
After that, you have to pass this instance to the animationPhases
prop.
// ...
import animationPhases from './boxes.phases'
function Boxes() {
return (
<AnimateComponent
animationPhases={animationPhases}
render={(ref) => (
<>
<div ref={ref('box', 'box-1')} className={styles.box1} />
<div ref={ref('box', 'box-2')} className={styles.box2} />
</>
)}
/>
)
}
// ...
To create the animations you can create a phases folder with the boxes-right.js and boxes-hidden.js files and use the ComponentAnimation
class.
/* boxes/phases/boxes-right.js */
import { ComponentAnimation } from 'animate-screen'
const animation = new ComponentAnimation()
animation.ref('box').to({ x: '100%' }).ease('power1.inOut').start(0).end(1)
animation
.ref('box-1')
.to({ background: 'orange' })
.ease('none')
.start(0)
.end(0.5)
animation
.ref('box-2')
.to({ background: 'yellow' })
.ease('none')
.start(0.5)
.end(1)
export default animation
/* boxes/phases/boxes-hidden.js */
import { ComponentAnimation } from 'animate-screen'
const animation = new ComponentAnimation()
animation.ref('box').to({ autoAlpha: 0 }).ease('power1.inOut').start(0).end(1)
export default animation
The ref()
method receives a tag as argument and returns an object that is used to define the animation of the elements with this tag. The object returned has the following methods:
to()
: It is used to define the properties to animate. The properties are the same you would use ingsap.to()
.ease()
: It is used to define the easing function. If you don't call this method the easing function will be "none" by default.start()
: It is used to define the start time of the animation with a number between 0 and 1. If you don't call this method the start time will be 0.end()
: It is used to define the end time of the animation with a number between 0 and 1. if you don't call this method the end time will be 1.
Finally, you have to pass these instances to the set()
methods.
/* boxes/boxes.phases.js */
import { ComponentAnimationPhases } from 'animate-screen'
import boxesRight from './phases/boxes-right'
import boxesHidden from './phases/boxes-hidden'
const phases = new ComponentAnimationPhases()
phases.set('boxes-right', boxesRight)
phases.set('boxes-hidden', boxesHidden)
export default phases
That's everything you need to know to create animations. After doing all of that, the component has all its animations set up to be triggered while you scroll.
Callbacks
In addition to defining scroll-triggered animations, you can also specify callbacks to execute while you scroll. To do it you have to use the setCallbacksPhases
prop.
// ...
import { useCallback } from 'react'
// ...
import animationPhases from './boxes.phases'
function Boxes() {
return (
<AnimateComponent
animationPhases={animationPhases}
setCallbacksPhases={useCallback((callbacks) => {
callbacks.add('boxes-hidden', 0.5, () => alert('Hello world!'))
}, [])}
render={(ref) => (
<>
<div ref={ref('box', 'box-1')} className={styles.box1} />
<div ref={ref('box', 'box-2')} className={styles.box2} />
</>
)}
/>
)
}
// ...
This prop receives a function you can use to add callbacks. To add them you have to use the add()
method of the callbacks parameter passing the following arguments:
- name: Name of the phase you want to add the callback to.
- time: Number between 0 and 1 that defines when the callback will be executed within the phase.
- callback: Callback to execute.