Intro
HTML canvas is a way to draw 2d graphics on web pages. It's exactly what it sounds like - a blank canvas to create drawings programatically. I mostly use it for making games and animations. In this post, I want to show some cool things to do with canvas. This isn't a tutorial. The Mozilla docs will do a better job of explaining how to use canvas. I will include code samples though and I'll embed some canvas elements directly on this page.
Here are a few of my canvas projects:
canvas collection - a collection of animations
the game nook - a site where I publish mini games
ball drop - new year's eve countdown
bouncing ball
This is like the "hello world" of canvas. The goal is to have a ball bounce around the screen. This requires drawing a circle, creating an animation loop and detecting collisions with the edges of the screen.
let canvas = document.getElementById("bouncing-ball");
let ctx = canvas.getContext("2d");
let ball = {
x: canvas.width / 2,
y: canvas.height / 2,
radius: 30,
xVel: 5,
yVel: 5
};
function animate() {
ctx.clearRect(0, 0, canvas.width, canvas.height);
ctx.beginPath();
ctx.arc(ball.x, ball.y, ball.radius, 0, 2 * Math.PI);
ctx.fillStyle = "DodgerBlue";
ctx.fill();
ball.x += ball.xVel;
ball.y += ball.yVel;
// collide with edges of screen
if(ball.x - ball.radius <= 0 || ball.x + ball.radius >= canvas.width) {
ball.xVel *= -1;
}
if(ball.y - ball.radius <= 0 || ball.y + ball.radius >= canvas.height) {
ball.yVel *= -1;
}
requestAnimationFrame(animate);
}
animate();
paint
A simple painting program. You can click and drag your mouse to draw on the canvas. The "clear" button below the canvas will clear the drawing. This program below could be extended to add different colors, line thickness, shapes, etc. I have a more full featured version of a paint program on this website - https://markyman4.github.io/whiteboard.
const canvas = document.getElementById("paint");
const ctx = canvas.getContext("2d");
const clearBtn = document.getElementById("paint-clear-btn");
let mouseIsDown = false;
let prevMousePos = null;
let curMousePos = null;
// clear the canvas when button clicked
clearBtn.onclick = () => {
ctx.clearRect(0, 0, canvas.width, canvas.height);
}
canvas.addEventListener("mousedown", () => mouseIsDown = true);
canvas.addEventListener("mouseup", () => mouseIsDown = false);
canvas.addEventListener("mousemove", ev => {
// keep track of current and last mouse position
// this is how we know start and end points for each line segment
if(prevMousePos === null) {
prevMousePos = curMousePos = {x: ev.offsetX, y: ev.offsetY};
} else {
prevMousePos = curMousePos;
curMousePos = {x: ev.offsetX, y: ev.offsetY};
}
// only draw if mouse is clicked
if(!mouseIsDown) {
return;
}
ctx.beginPath();
ctx.moveTo(prevMousePos.x, prevMousePos.y);
ctx.lineTo(curMousePos.x, curMousePos.y);
ctx.strokeStyle = "black";
ctx.lineWidth = 3;
ctx.stroke();
});
particles
I use particle effects a lot because they look cool and are easy to implement. The implementation of the particles is usually similar to the code sample below. This example looks like fireworks. You can also click and drag your mouse around the canvas to "draw" particles.
class Particle {
constructor(x, y, radius, color) {
this.x = x;
this.y = y;
this.radius = radius;
this.color = color;
// random movement
this.xVel = (Math.random() * 5) * (Math.random() < 0.5 ? -1 : 1);
this.yVel = (Math.random() * 5) * (Math.random() < 0.5 ? -1 : 1);
// console.log(this.xVel)
}
// draw a circle of the specified color
draw(ctx) {
ctx.beginPath();
ctx.arc(this.x, this.y, this.radius, 0, 2 * Math.PI);
ctx.fillStyle = this.color;
ctx.fill();
}
// decrease particle size and move in direction
update() {
this.x += this.xVel;
this.y += this.yVel;
if(this.radius > 0.3) {
this.radius -= 0.3;
}
}
}
/*****************************************************/
const canvas = document.getElementById("particles");
const ctx = canvas.getContext("2d");
let particles = [];
let colors = ["#0377fc", "#e32424", "#f4fc03", "#3ec73c", "#9433cc", "#e34fdc"];
let mouseIsDown = false;
function spawnFirework(x, y) {
let radius = Math.random() * 10 + 5;
for(let i = 0; i < 20; i++) {
let color = colors[Math.floor(Math.random() * colors.length - 1)];
particles.push(new Particle(x, y, radius, color));
}
}
// randomly spawn a "firework" (group of 20 particles) every 0.5 seconds
window.setInterval(() => {
let x = Math.random() * canvas.width;
let y = Math.random() * canvas.height;
spawnFirework(x, y);
}, 500);
// spawn particles when clicking
canvas.addEventListener("mousedown", () => {
mouseIsDown = true;
});
canvas.addEventListener("mouseup", () => {
mouseIsDown = false;
});
canvas.addEventListener("mousemove", (ev) => {
if(mouseIsDown) {
spawnFirework(ev.offsetX, ev.offsetY);
}
});
function animate() {
ctx.clearRect(0, 0, canvas.width, canvas.height);
// update the particles, if size gets too small, remove them
let smallParticleIndices = [];
particles.forEach((p, idx) => {
p.draw(ctx);
p.update();
if(p.radius <= 0.2) {
smallParticleIndices.push(idx);
}
});
// remove the small particles, iterate in reverse so we don't delete the wrong indexes
for(let i = smallParticleIndices.length - 1; i >= 0; i--) {
particles.splice(smallParticleIndices[i], 1);
}
requestAnimationFrame(animate);
}
animate();
Conclusion
The great thing about the canvas is that it's built into your standard HTML and JavaScript tools, so no third-party libraries are required. All you need to build canvas creations is a computer with a web browser and a text editor (i.e. pretty much any computer!). As far as 2D graphics APIs go, canvas is simple and feature-rich which makes it a great choice for a lot of 2D graphics projects.
Notice that in all the examples I showed, there's really not that much code involved. You can make a lot of really cool things without writing all that much code. I love doing "mini projects" with canvas becuase you can spend an hour or two and create something really fun that you can easily publish as a website and share it with anyone. All these mini projects are what lead me to create my canvas collections website and the game nook. I got sick of creating repositories for each little project and wanted to put them all in one place!
Canvas is widely used, but a lot of times I wonder why we don't hear more talk about it. It seems like such an awesome tool that's readily available. I think a lot of people assume that for making games or animations, they are going to need to install a library. I would love to see more creations done with just canvas. My goal with the game nook website is to eventually get other people contributing to it. It would be great to have a website to showcase all the cool things people can make with just the basic tools.