I’m trying to make an animation inside a canvas: here, a numbered circle must be drawn and move from left to right one single time, disappearing as soon as it reaches the end of animation.
For now I managed to animate it in loop, but I need to animate at the same time (or with a set delay) multiple numbered circles, strating in different rows (changing y position) so they wont overlap.
Any idea how can I manage this? my JS code is the following:
JavaScript
x
57
57
1
// Single Animated Circle - Get Canvas element by Id
2
var canvas = document.getElementById("canvas");
3
4
// Set Canvas dimensions
5
canvas.width = 300;
6
canvas.height = 900;
7
8
// Get drawing context
9
var ctx = canvas.getContext("2d");
10
11
// Radius
12
var radius = 13;
13
// Starting Position
14
var x = radius;
15
var y = radius;
16
17
// Speed in x and y direction
18
var dx = 1;
19
var dy = 0;
20
21
// Generate random number
22
var randomNumber = Math.floor(Math.random() * 60) + 1;
23
24
if (randomNumber > 0 && randomNumber <= 10) {
25
ctx.strokeStyle = "#0b0bf1";
26
} else if (randomNumber > 10 && randomNumber <= 20) {
27
ctx.strokeStyle = "#f10b0b";
28
} else if (randomNumber > 20 && randomNumber <= 30) {
29
ctx.strokeStyle = "#0bf163";
30
} else if (randomNumber > 30 && randomNumber <= 40) {
31
ctx.strokeStyle = "#f1da0b";
32
} else if (randomNumber > 40 && randomNumber <= 50) {
33
ctx.strokeStyle = "#950bf1";
34
} else if (randomNumber > 50 && randomNumber <= 60) {
35
ctx.strokeStyle = "#0bf1e5";
36
}
37
38
function animate3() {
39
requestAnimationFrame(animate3);
40
41
ctx.clearRect(0, 0, 300, 900);
42
43
if (x + radius > 300 || x - radius < 0) {
44
x = radius;
45
}
46
47
x += dx;
48
49
ctx.beginPath();
50
ctx.arc(x, y, 12, 0, Math.PI * 2, false);
51
ctx.stroke();
52
ctx.fillText(randomNumber, x - 5, y + 3);
53
}
54
55
// Animate the Circle
56
57
animate3();
JavaScript
1
1
1
<canvas id="canvas"></canvas>
Advertisement
Answer
Here is a solution which doesn’t use classes as such and separates the animation logic from the updating – which can be useful if you want more precise control over timing.
JavaScript
1
64
64
1
// Some helper functions
2
const clamp = (number, min, max) => Math.min(Math.max(number, min), max);
3
4
// Choose and remove random member of arr with equal probability
5
const takeRandom = arr => arr.splice(parseInt(Math.random() * arr.length), 1)[0]
6
7
// Call a function at an interval, passing the amount of time that has passed since the last call
8
function update(callBack, interval) {
9
let now = performance.now();
10
let last;
11
setInterval(function() {
12
last = now;
13
now = performance.now();
14
callBack((now - last) / 1000);
15
})
16
}
17
18
const settings = {
19
width: 300,
20
height: 150,
21
radius: 13,
22
gap: 5,
23
circles: 5,
24
maxSpeed: 100,
25
colors: ["#0b0bf1", "#f10b0b", "#0bf163", "#f1da0b", "#950bf1", "#0bf1e5"]
26
};
27
const canvas = document.getElementById("canvas");
28
canvas.width = settings.width;
29
canvas.height = settings.height;
30
const ctx = canvas.getContext("2d");
31
32
// Set circle properties
33
const circles = [Array(settings.circles).keys()].map(i => ({
34
number: i + 1,
35
x: settings.radius,
36
y: settings.radius + (settings.radius * 2 + settings.gap) * i,
37
radius: settings.radius,
38
dx: settings.maxSpeed * Math.random(), // This is the speed in pixels per second
39
dy: 0,
40
color: takeRandom(settings.colors)
41
}));
42
43
function drawCircle(circle) {
44
ctx.strokeStyle = circle.color;
45
ctx.beginPath();
46
ctx.arc(circle.x, circle.y, circle.radius, 0, Math.PI * 2, false);
47
ctx.stroke();
48
ctx.fillText(circle.number, circle.x - 5, circle.y + 3);
49
}
50
51
function updateCircle(circle, dt) {
52
// Update a circle's position after dt seconds
53
circle.x = clamp(circle.x + circle.dx * dt, circle.radius + 1, settings.width - circle.radius - 1);
54
circle.y = clamp(circle.y + circle.dy * dt, circle.radius + 1, settings.height - circle.radius - 1);
55
}
56
57
function animate() {
58
ctx.clearRect(0, 0, settings.width, settings.height);
59
circles.forEach(drawCircle);
60
requestAnimationFrame(animate);
61
}
62
63
update(dt => circles.forEach(circle => updateCircle(circle, dt)), 50);
64
animate();
JavaScript
1
1
1
<canvas id="canvas" style="border: solid 1px black"></canvas>