Animations
Keyframe animations let you make an element take a roundabout path from location to location, or you might want the element to return to its previous location.
Keyframes
A keyframe is a specific point in an animation:
- you define keyframes, and the browser fills in the points in between so the changes look smooth
- transforms only let you define the starting and ending point. Keyframes let you define multiple points
- If you repeat an animation, make sure the ending value matches the beginning value so changes are smooth
- Rules in a keyframe are a very high-priority origin in the cascade, so they take precedence over other declarations. If a box is green before an animation is applied that makes the box red, then the box is red for the duration of the animation and goes back to green at the end.
@keyframes
There are two main properties:
@keyframe
at-rule that defines the animationanimation
property that applies the animation to an element
The @keyframes
at-rule needs a name, and then you define different steps (keyframes) with percentages. Each keyframe describes what the step should look like when it completed that percentage of its motion.
@keyframes <keyframe-name> {
0% { /* ruleset */ }
25% { /* ruleset */ }
50% { /* ruleset */ }
100% { /* ruleset */ }
}
This keyframe animation moves a box 50px from left to right in 3 keyframes. The color also changes, but because you don’t define the color in the second step, the browser fills in the blanks to smoothly transition the colors defined at the first and last step.
Finally, you apply the animation to an element with the animation
property, which says “apply the over-and-back
keyframe animation to this element, make it last 1.5 seconds, the movement should be steady (linear), and repeat it three times”:
@keyframes over-and-back {
0% {
background-color: hsl(0, 50%, 50%);
transform: translate(0);
}
50% {
transform: translate(50px);
}
100% {
background-color: hsl(270, 50%, 90%);
transform: translate(0);
}
}
.box {
width: 100px;
height: 100px;
background-color: green;
animation: over-and-back 1.5s linear 3;
}
animation
The animation
property is shorthand for multiple properties:
.element {
animation: name duration timing-function delay iteration-count direction fill-mode;
}
name
: Name of the function defined in the@keyframe
at-ruleduration
: How long the animation laststiming-function
: How the animation accelerates/decelerates. Use a bezier curve or transition timing function (ease-in
, etc)iteration-count
: Number of times the animation repeats. By default, is1
.direction
: Whether the animation plays forward, backward, or alternate. MDN docsfill-mode
: HOw the animation applies styles to its target before and after execution. MDN docs.
Animating 3D transforms
WKND107SPP622
First, put your design and layout in place.
- Puts
perspective
on the grid container so all elements appear to come in from the same location - Applies an animation to each item
- Step 1:
- The negative
translateZ()
value makes it come from a distance, instead of from the viewport rotateY()
makes the card come in completely sideways, with any angle coming from the shared perspective added to the container
- The negative
- Step 2:
- brings the cards almost completely to the screen (
-160px
) - rotates them slightly
- brings the cards almost completely to the screen (
- Step 3:
- places the card elements completely flat
.flyin-grid__item {
animation: fly-in 600ms ease-in;
}
@keyframes fly-in {
0% {
transform: translateZ(-800px) rotateY(90deg);
opacity: 0;
}
56% {
transform: translateZ(-160px) rotateY(87deg);
opacity: 1;
}
100% {
transform: translateZ(0) rotateY(0);
}
}
animation-delay
Apply animation-delay
to an item to delay the animation. This lets you create staggered animations that display in a wave:
.flyin-grid__item {
animation: fly-in 600ms ease-in;
}
.flyin-grid__item:nth-child(2) {
animation-delay: 0.15s;
}
.flyin-grid__item:nth-child(3) {
animation-delay: 0.3s;
}
.flyin-grid__item:nth-child(4) {
animation-delay: 0.45s;
}
animation-fill-mode
animation-fill-mode
applies styles before, after, or during a keyframe. Accepts these values:
none
: defaultforwards
backwards
both
For definitions, see the MDN docs.
Styles defined in an animation are only applied during the animation. You might need to apply these styles before the animation. For example, if an element is zooming in on the screen, it will appear in its final location before the animation, then zoom in, then display normally. It makes more sense to apply the first keyframe’s styles before the animation begins so the element does not begin in its final location.
This example uses the backwards
value to apply keyframe properties before the animation begins. Apply animation-fill-mode
to the static element under the animation
property:
.flyin-grid__item {
animation: fly-in 600ms ease-in;
animation-fill-mode: backwards;
}
This essentially makes the animation pause on the first frame until it begins.
Practical usage
Refer to the Animation MDN docs for details about how to interact with animations with JS.
Waiting on server response
When a user action sends a request to a server, you want to communicate that the user is waiting on the server response. You can achieve this with an is-loading
class that displays a spinner on the screen.
Here, a button displays a spinner when waiting on a server response:
- make the button text transparent so you can replace it with the spinner
- make the button a containing element
- create a spinner out of a pseudo-element and position it absolutely
- create keyframes that rotate the pseudo-element
- it applies the animation until the class is removed (
infinite
)
- it applies the animation until the class is removed (
button.is-loading {
position: relative;
color: transparent;
}
button.is-loading::after {
position: absolute;
content: "";
display: block;
width: 1.4em;
height: 1.4em;
top: 50%; /* 1/2 the buttons width */
left: 50%; /* 1/2 the buttons height */
margin-left: -0.7em;
margin-top: -0.7em;
border-top: 2px solid white; /* top border makes it look like a crescent */
border-radius: 50%;
animation: spin 0.5s linear infinite;
}
@keyframes spin {
from {
transform: rotate(0deg);
}
to {
transform: rotate(360deg);
}
}
Here is the javascript to add the is-loading
class:
let input = document.querySelector('#trip');
let button = document.querySelector('#submit-button');
button.addEventListener('click', (e) => {
e.preventDefault();
button.classList.add('is-loading');
button.disabled = true;
input.disabled = true;
// submit form data to server
});
Shake element to get user attention
If a user needs to enter text into an element or save their work, you can animate an element to shake and draw their attention.
Here, the keyframe shakes the button left and right by 0.4em
. There are multiple steps defined within some rulesets because the actions are the same. Notice that the odd percentages shift to the left (negative values), while the positive shift the element to the right (positive values). The values at 80% and 90% shorten the movement so it is less intense at the end, which makes it slow down more smoothly.
.shake {
animation: shake 0.7s linear;
}
@keyframes shake {
0%,
100% {
transform: translateX(0);
}
10%,
30%,
50%,
70% {
transform: translateX(-0.4em);
}
20%,
40%,
60% {
transform: translateX(0.4em);
}
80% {
transform: translateX(0.3em);
}
90% {
transform: translateX(-0.3em);
}
}
Add Javascript to addd the shake
class. This script causes the button to shake after the user stops typing for 1 second:
- use a timeout to add the
shake
class - clear the timeout (reset it) when the button is selected or the user resumes typing
- remove the class with the
animationend
event type, so when the animation completes, the class is removed and it stops shaking. This means that you can add the class again
let input = document.querySelector('#trip');
let button = document.querySelector('#submit-button');
let timeout = null;
button.addEventListener('click', (e) => {
e.preventDefault();
clearTimeout(timeout); // clear timeout when button pressed
button.classList.add('is-loading');
button.disabled = true;
input.disabled = true;
});
input.addEventListener('keyup', function () {
clearTimeout(timeout); // reset timeout if user types again
timeout = setTimeout(function () {
button.classList.add('shake'); // add .shake when user hasn't typed for 1s
}, 1000);
});
button.addEventListener('animationend', function () {
button.classList.remove('shake'); // remove .shake when animation completes
});
Start animations based on scrolling
The animation-timeline: scroll();
property is not widely supported, so you need to do this with JS until it is.
You might want an animation to fire when you scroll past an element in either direction. This example spins a box positioned at the top-right of the viewport when you scroll.
scroll()
without an argument binds the animation to the closest ancestor element in the DOM that has a scrollbar. It also accepts these values:
nearest
: default behaviorroot
: animates using the document rootself
: animates relative to the scroll position of the current element
You can combine the previous values with these values to control the scroll direction:
block
(default direction)inline
y
x
For example, scroll(inline root)
scrolls relative to the inline scrolling of th root element
body {
margin: unset;
height: 200lvh;
}
.indicator {
position: fixed;
top: 15px;
right: 15px;
height: 50px;
width: 50px;
background-color: oklch(74% 0.08 260deg);
animation-name: spin;
animation-timeline: scroll();
}
@keyframes spin {
from {
rotate: 0deg;
}
to {
rotate: 360deg;
}
}