This chapter will get you started with JavaScript for animations! We'll see how to kick off animations that should run repeatedly or should stop at a certain point.
-
The
setInterval()
function kicks off a repeated action and is called at regular intervals. TheclearInterval()
function stops a repeated action that was launched withsetInterval()
. -
The
setTimeout()
function executes an action once after a certain delay. -
The
requestAnimationFrame()
function asks the browser to execute a function that updates the animation as soon as possible. This works well for real-time animations. ThecancelAnimationFrame()
function stops an in-progress animation that was launched withrequestAnimationFrame()
. -
You can also create web animations via CSS.
Let's get started with animations by learning how to repeatedly modify an element's content. Here is the associated HTML code.
<h1 id="title">This page will self-destruct in <span id="counter">10</span> second(s)...</h1>
And now for the corresponding JavaScript code.
// Count down the counter
const decreaseCounter = () => {
// Convert counter text to a number
const counter = Number(counterElement.textContent);
// Decrease counter by one
counterElement.textContent = counter - 1;
};
const counterElement = document.getElementById("counter");
// Call the decreaseCounter function every second (1000 milliseconds)
setInterval(decreaseCounter, 1000);
Click here to see it in action. It works as expected... Kind of: the countdown never stops. We'll fix this a little later.
How did the previous example work? The JavaScript code defines a function called decreaseCounter()
that accesses and then decreases one by one the value of the HTML element named counter
.
Calling
Number()
in the function code is mandatory: it converts the counter string into a number, which endows it with subtraction functionality.
The call to setInterval()
triggers a repeated action. This function lets you call a function at regular intervals. Its parameters are the function to call and the time in milliseconds between each call. The returned value is an ID for the repeated action, which can be used to further modify it.
// Set up a repeated action
const intervalId = setInterval(callbackFunction, timeBetweenEachCall);
Let's try to stop the counter once the countdown is complete. We'll also modify the text of the page. Here's the JavaScript code for our example, updated to produce our desired result:
// Count down the counter until 0
const decreaseCounter = () => {
// Convert counter text to a number
const counter = Number(counterElement.textContent);
if (counter > 1) {
// Decrease counter by one
counterElement.textContent = counter - 1;
}
else {
// Cancel the repeated execution
clearInterval(intervalId);
// Modify the page title
const title = document.getElementById("title");
title.textContent = "BOOM!!";
}
};
const counterElement = document.getElementById("counter");
// Call the decreaseCounter function every second (1000 milliseconds)
const intervalId = setInterval(decreaseCounter, 1000);
Click here to see it in action.
In the decreaseCounter()
function, we only decrease the counter if the current value is higher than 1. If not, we call the function clearInterval()
and then modify the title of the page.
The clearInterval()
function lets you cut off repeated code execution. It takes as a parameter the ID of the action set by the call to setInterval()
.
// Cancel a repeated action set up with setInterval()
clearInterval(intervalId);
Imagine that you want to modify the page text after its "explosion" in the previous example. You'd modify our example as follows:
// Count down the counter until 0
const decreaseCounter = () => {
// Convert counter text to a number
const counter = Number(counterElement.textContent);
if (counter > 1) {
// Decrease counter by one
counterElement.textContent = counter - 1;
}
else {
// Cancel the repeated execution
clearInterval(intervalId);
// Modify the page title
const titleElement = document.getElementById("title");
titleElement.textContent = "BOOM!!";
// Modify the title after 2 seconds
setTimeout(() => {
titleElement.textContent = "Everything's broken now :(";
}, 2000);
}
};
const counterElement = document.getElementById("counter");
// Call the decreaseCounter function every second (1000 milliseconds)
const intervalId = setInterval(decreaseCounter, 1000);
Click here to see it in action.
Once the countdown has finished, we call the setTimeout()
function to set a new page title after a 2 second (2000 millisecond) delay.
The setTimeout()
function lets you execute a function once after a particular delay, expressed in milliseconds.
// Execute an action once, after a delay
setTimeout(callbackFunction, timeBeforeCall);
The previous solutions were convenient for making our pages a bit more dynamic, but weren't enough for adding real-time animation. Let's look at a better-performing solution.
Take, for example, the movement of a <div>
type element from left to right on the page. We start with the following HTML and CSS code that display a red block on the page.
<div id="frame">
<div id="block"></div>
</div>
#frame {
border: 1px solid red;
}
#block {
width: 20px;
height: 40px;
background: red;
position: relative;
}
And here is the JavaScript code that lets you move the red block.
// Move the block to the left
const moveBlock = () => {
// Convert the left position of the block (value of the form "XXpx") to a number
const xBlock = parseFloat(getComputedStyle(blockElement).left);
// Move the block to the right
blockElement.style.left = (xBlock + movement) + "px";
// Have the browser call moveBlock as soon as possible
requestAnimationFrame(moveBlock);
};
const blockElement = document.getElementById("block");
// Movement value in pixels
const movement = 7;
// Start the animation
requestAnimationFrame(moveBlock);
Click here to see it in action.
Upon page load, the red block moves (indefinitely) from left to right.
The example code defines a function called moveBlock()
which moves the block horizontally to the right. It grabs the current position of the block's left border than adds the value contained in the movement
variable. Next, the code calls the requestAnimationFrame()
method to keep the animation going.
Position values are written in pixels. These are the strings you saw that resemble "XXpx," which requires the use of the JavaScript parseFloat()
function to convert numeric values before making calculations.
Don't use
Number()
to convert a string with"px"
into a numerical value. This won't work, and you'll get aNaN
value (Not a Number) as a result!
The requestAnimationFrame()
function lets you ask the browser to execute a function as soon as possible, which updates the animation. It's the browser's job to make the animation as smooth as possible. The returned value of requestAnimationFrame()
is an ID for the animation, which can be used to further modify it.
Here is how requestAnimationFrame()
is used in combination with an animation function.
const animate = () => {
// Animation code
// ...
// At end of animation, request another one
animationId = requestAnimationFrame(animate);
};
// Animation start
let animationId = requestAnimationFrame(animate);
Let's now see how to stop the block before it reaches the border of the frame that contains it. We'll have to verify that the left border position is less than the width of the frame, bearing in mind the thickness of the block itself.
Here's the updated JavaScript code.
// Move the block to the right, all the way to the end of the frame
const moveBlock = () => {
// Convert the left position of the block (value of the form "XXpx") to a number
const xBlock = parseFloat(getComputedStyle(blockElement).left);
// Convert the width of the frame (value of the form "XXpx") to a number
const xMax = parseFloat(getComputedStyle(frame).width);
// If the block isn't already to the end of the frame
if (xBlock + blockWidth <= xMax) {
// Block movement
blockElement.style.left = (xBlock + movement) + "px";
animationId = requestAnimationFrame(moveBlock);
}
else {
// Cancel the animation
cancelAnimationFrame(animationId);
}
};
const blockElement = document.getElementById("block");
// Convert the block width (value of the form "XXpx") to a number
const blockWidth = parseFloat(getComputedStyle(block).width);
// Movement value in pixels
const movement = 7;
// Start the animation
let animationId = requestAnimationFrame(moveBlock);
Click here to see it in action.
The new moveBlock()
function checks that the block has arrived at the end of the frame before moving. If that's the case, the animation stops via a call to cancelAnimationFrame()
.
The cancelAnimationFrame()
functions stops the animation and takes the ID of the animation set by a prior call to requestAnimationFrame()
.
// Stop an animation
cancelAnimationFrame(animationID);
You just learned about the different possibilities that JavaScript offers for animating web pages. Just bear in mind there's another alternative: CSS.
This paragraph barely scratches the surface of CSS animations.
Let's check out how to get a similar effect as the previous example by using CSS instead of JavaScript. Remove any JavaScript code from your example and modify your CSS code as follows.
#frame {
border: 1px solid red;
}
#block {
width: 20px;
height: 40px;
background: red;
position: relative;
margin-left: -20px; /* Negative margin to simplify position calculations */
animation-name: moveBlock; /* Name of animation */
animation-duration: 6s; /* Length of animation */
animation-fill-mode: forwards; /* Let the block in its final position */
}
@keyframes moveBlock {
from {
/* Initial position: to the left of the frame (taking negative margin into account) */
left: 20px;
}
to {
/* Final position: within the right side of the frame (taking negative margin into account) */
left: 100%;
}
}
Click here to see it in action.
This code defines a CSS animation named moveBlock()
, which moves the block from the left to the right side of its containing frame. The result is virtually identical to the JavaScript version.
Now, decision time. How should you choose between setInterval()
, requestAnimationFrame()
, or CSS to animate your page? The answer depends on how complex your animation is. In theory, CSS animations are more efficient performance-wise, but you can't do everything with them.
Here's how you might want to approach your decision:
- Use
setInterval()
if the animation isn't in real-time and should just happen at regular intervals. - Favor CSS if the animation happens in real-time and can be managed with it.
- Use
requestAnimationFrame()
for any other case.
Write an interactive web page with a button to start and stop a chronometer counting the number of elapsed seconds.
The goal of this exercise is to make a basketball bounce across the screen. You can download the ball image here.
Start with the following HTML and CSS content.
<p>
<button id="start">Start</button>
<button id="stop" disabled>Stop</button>
</p>
<div id="frame">
<!-- Update the "src" attribute if you downloaded the image locally -->
<img id="ball" src="https://raw.githubusercontent.com/bpesquet/thejsway/master/resources/basketball.jpg">
</div>
#ball {
position: relative;
left: 0px;
}
Write the JavaScript code that makes the ball bounce horizontally.
With your solution, create a variable with values 1 or -1 that dictates the direction in which the ball should move.