$ npm install unanime --save
import { animate } from 'unanime';
const animate = require('unanime');
import { animate } from 'unanime';
const myAnimation = animate(
'.targets',
[
{transform: 'translateX(100px)'}
],
{
duration: 1500,
direction: 'alternate',
iterations: 3,
autoplay: true
}
)
cont var = animate(
// Targets,
[
// Keyframes
],
{
// Options
}
)
String, Object or Array accepted.
// Targets
'.myTargets' // Under the hood, document.querySelectorAll is used
Array with one or multiple objects. When one object is specified, it's like 'TO' direction. Use all CSS(Js) property you want.
// Keyframes
[
{transform: 'translate3d(0, 0, 0)', backgroundColor: 'red'},
{transform: 'translate3d(0, 100px, 0)', backgroundColor: 'green'},
]
Add 'offset' property in a single frame to change the effect duration. (endpoint)
Example:
// Keyframes
[
{opacity: 0,},
{opacity: 0.2, offset: 0.8}, // opacity takes 0.8 of the animation duration to become 0.2
{opacity: 1}
]
Set your options, they all have a default value specified. You can use an anonymous function on every property.
// Options
{
duration: () => Math.floor(Math.random() * 10000); // 5683
}
const anotherTarget = document.querySelector('#anotherTarget');
// Inject in var for timeline usage
const myAnimation = animate(
// Targets (string, object or array)
['.targets', anotherTarget],
// Keyframes (array)
[
{transform: 'translate3d(0, 0, 0)', backgroundColor: 'red'},
{transform: 'translate3d(0, 100px, 0)', backgroundColor: 'green'},
],
// Options (object)
{
duration: 1000,
// animation's duration in ms (number, object) | default: 1000
direction: 'normal',
// 'normal', 'reverse', 'alternate' (string) | default: 'normal'
easing: 'easeOutBack',
// easings.net accepted, basic css or cubic-bezier (string) | default: 'linear'
delay: 200,
// delay before animation start (number, object) | default: 0
endDelay: 0,
// delay after animation finished (number, object) | default: 0
iterations: 3,
// times the animation is executed (positive number or Infinity) | default: 1
iterationStart: 1.2,
// select at which iteration animation start (number) | default: 0
fill: 'both',
// apply the style to your targets 'none', 'auto', 'backwards', 'forwards', 'both' (string) | default: 'both'
composite: 'replace',
// choose how each keyframe impact the style, 'replace', 'add', 'accumulate' | default: 'replace'
playbackRate: 1,
// choose the speed of your animation without changing duration (+ / - number) | default: 1
autoplay: false,
// start animation when ready (boolean) | default: false
initStyles: true,
// option that apply css style (first keyframes) to the target | default: false
commitStyles: false,
// option that apply css style to targets after animation finished | default: false
autocancel: false,
// remove animation from browser at the end | default: false
willChange: false,
// add and remove automatically a 'will-change: transform' to your target during animation | default: false
mergeTargets: false,
// merge targets in a single array, you can use this option when you want use stagger on different targets for example | default: false
}
)
myAnimation.play();
You can use normal css easing like 'ease-in' or cubic-bezier() property. You can also use presets from Easing.net as string.
A spring effect is include, simply use 'spring()' as string in your easing option. Read this page to learn more about spring effect and use the simulator.
// Options
{
easing: 'spring()' // will be init with default config spring(50, 1, 10)
}
You can also change spring settings:
easing: 'spring(stiffness, mass, damping, offset)'
// stiffness: min 50 || default 50
// mass: min 1 || default 1
// damping: min 1 || default 10
// offset: min 0.1 || default 1
I recommand you to change only the first param "stiffness" only at beginning, more you change "mass" more you need to calculate animations steps positions. Damping don't really need to be change. Don't forget to change the animation duration to have a good feelings.
You can check the number of computed steps by adding 'log' in your string.
easing: 'spring(250) log'
// 82
Sometimes the end of animation is a little too long, because of very small movements computed. You can decide to cut the end by adding the last param (offset): 1 = full animation , 0.5 = half.
Only two keyframes can be set within the array. Be sure to specify properties in exact same order.
// Keyframes
[
{transform: 'translateX(100%) scale(0.3)', opacity: 0},
{transform: 'translateX(0) scale(1)', opacity: 1}
]
Warning: some settings can cause infinite loop, in this case try another params.
Animation is not running by default, you have three options:
// Options
{ autoplay: true }
or
// Method
myAnimation.play();
or
// Timeline
const tl = timeline(
[
{animate: myAnimation}
]
)
tl.play();
Start the animation.
Reverse the current playbackRate and play animation.
Pause the animation.
Go back to first frame and pause animation.
Go back to first frame and play animation.
Go to the last frame animation.
Stop and remove animation from browser.
Go immediatly to a specific moment of animation (number from 0 to 1).
Go smoothely to a specific moment, new direction is calculed from the previous position, pass pinOptions as second argument ({smoothness: 0.05, delay: 0} // default)
You can target a specific animation inside your animate targets's array, by adding the index as last argument (start at 0).
Example:
myAnimation.play();
setTimeout(() => {
myAnimation.pause(2);
// pause the 3th target's animation after 2500ms
}, 2500);
Return the total duration with duration, delays, iterations ...
Return the current iteration when executed.
Return the animation's status: 'idle', 'running', 'finished'.
Return the current playbackRate.
Return the current progression (from 0 to 1).
Example:
myAnimation.getPlayState();
// 'running'
Set a new direction ('normal', 'reverse', 'alternate').
Set a new playbackRate ( + / - number). Change the speed of animation.
Set the option willChange to true or false.
Set the option initStyles to true or false.
Set the option commitStyles to true or false.
Example:
myAnimation.setDirection('alternate');
Execute when animation is ready.
Execute when animation is playing.
Execute when animation is paused.
Execute when animation is finished.
Execute when animation is removed (everytime the animation is paused/play or changed);
Execute when animation is canceled.
Example:
myAnimation.onReady(() => {
console.log('Ready to play !');
});
Staggering allows you to animate multiple elements with follow through and overlapping action.
Example:
// Options
{delay: {stagger: 100, from: 'center', direction: 'normal'}}
// Options
{ yourOption: {
default: 0, // initial value of your option added to stagger | default: option's default
stagger: 100, // amount that will be multiplied by array index (number) | default: 0
from: 3, // choose the starting point of your stagger propagation (string, number, array) | default: 'start'
direction: 'reverse' // select the way stagger in computed (string) | default: 'normal'
}
}
There are different ways to set 'from' option in stagger object.
'start', 'center', 'end' // select one mode
or
10 // select array index(+1)
or
[2, 7, 13] // select multiple starting points
Select the propagation's direction.
'normal' // default
or
'reverse'
Stagger can be used on every options property who takes a number. Example:
// Options
{
delay: {stagger: 200},
duration: {default: 2500, stagger: 100},
iterations: {default: 4, stagger: 0.05},
iterationStart: {default: 1, stagger: 0.02},
playbackRate: {default: 1, stagger: 0.05},
...
}
Easing can accept an array to set different easing on each elements of your targets. You can directly use this list of easing as string.
// Options
{
easing: ['easeInCirc', null, 'ease-out', 'cubic-bezier(.17,.67,.83,.67)']
// if null, get previous easing
}
import { animate, timeline } from 'unanime';
const tl = timeline(
// Animations list
[
{ animate: myAnimation1},
{ animate: myAnimation2, start: '<+800'},
],
// Timeline options
{autoplay: false, iterations: Infinity, direction: 'alternate'}
)
tl.play();
You can change the start point of each animations within the timeline by adding 'start'.
Use STRING to change the relative position of your animation.
{animate: myAnimation1, start: '+200'}, // +200ms from the relative start
{animate: myAnimation2, start: '-600'},
{animate: myAnimation3, start: '<'}, // start with the previous animation
{animate: myAnimation4, start: '<<<'}, // previous * 3 (myAnimation1)
{animate: myAnimation5, start: '<<+800'},
{animate: myAnimation6, start: '<-1200'}
Use NUMBER to change the absolute position of your animation.
{animate: myAnimation2, start: 6500} // start at 6500ms after the timeline started
Observer is a module to trigger your animation by scrolling (build with Intersection Observer API). Please be sure to undestand how works the API to use it (root, target, threshold ...) Intersection Observer API Doc
// Options
{
observer: {
root: null,
// set the root element, by default null automatically select the viewport (most of time, don't change this property)
rootMargin: '0px 0px 20% 0px',
// change root size by adding + / - margins, define in string as classic margins (px and % only)
target: '#sectionId',
// by default target automatically select the animation's target (auto detect array create only one observer on direct parent), change it to trigger the animation from another area. Under the hood a layout is created at the end of the body with class=${targetId}-intersection-observer
targetMargin: '20% 0px',
// change target size by adding + / - margins, use a string (px, %, vw ... accepted) (NEED overlay: true)
threshold: 0.3, // default: 0 / if pin is true default: 100
// set only one threshold if between 0 - 1 (threshold: 0.5)
// or an array of threshold (threshold: [0.2, 0.5, 0.8])
// or automatically calculate an array if you define the number you want (threshold: 36)
refreshInterval: 1000,
// set an interval to refresh target's placement when page's height grow up or during a window resize, default: -1 (disabled)
overlay: true,
// creating an overlay of the target, you'll can use the option "targetMargin", usefull if you want a static target position (with no overlay, the target position may change with an animation)
markers: true,
// set markers to true to see helpers on your screen
once: false,
// set to true if you want disable the observer once the animation is finished
split: false,
// set to true to split each target and control his own animation
toggleActions: 'play pause reverse reset',
// set quick actions depending of scroll action (enter / leave / enterback / leaveback)
// actions list: none, play, pause, resume, reverse, complete, restart, reset
pin: false,
// set to true to link your animation with your scroll
pinOptions: {
smoothness: 0.05, // default 0.05, go under to give more smoothness
delay: 80 // default 0, to add delay between scroll and animation
},
onEnter: (entry) => {
console.log('ENTERING' + entry.target);
},
// triggered action when root scroll down and enter in target area
// can access entry
onLeave: () => {
console.log('LEAVE');
},
// triggered action when root scroll down and leave the target area
onEnterBack: () => {
console.log('ENTER BACK');
},
// triggered action when root scroll up and enter back in the target area
onLeaveBack: () => {
console.log('LEAVE BACK');
},
// triggered action when root scroll up and leave back from the target area
onRefresh: () => {
console.log('REFRESH OBSERVER');
},
// triggered when observer is refreshing
}
}