const { PI, cos, sin, abs, sqrt, pow, floor, round, random } = Math;
const HALF_PI = 0.5 * PI;
const rand = n => n * random();
const randRange = n => n - rand(2 * n);
const fadeIn = (t, m) => t / m;
const fadeOut = (t, m) => (m - t) / m;
const fadeInOut = (t, m) => {
return abs((t + hm) % m - hm) / hm;
a: document.createElement('canvas'),
b: document.createElement('canvas') };
a: canvas.a.getContext('2d'),
b: canvas.b.getContext('2d') };
document.body.appendChild(canvas.b);
canvas.a.width = canvas.b.width = window.innerWidth;
canvas.a.height = canvas.b.height = window.innerHeight;
function mousehandler(e) {
hover = e.type === 'mousemove';
function getParticle(x, y) {
direction: floor(rand(6)) * 60 * TO_RAD,
turnDirection: randRange(1) * 0.1,
directionChangeRate: 20 + round(rand(10)),
this.destroy = this.life++ > this.ttl;
this.direction += this.life % this.directionChangeRate === 0 && round(randRange(1)) * 60 * TO_RAD;
this.velocity = fadeInOut(this.life, this.ttl) * this.speed;
this.position.x += cos(this.direction) * this.velocity;
this.position.y += sin(this.direction) * this.velocity;
ctx.a.strokeStyle = `hsla(${this.hue},100%,50%,${fadeInOut(this.life, this.ttl)})`;
ctx.a.strokeRect(this.position.x - 0.5 * this.size, this.position.y - 0.5 * this.size, this.size, this.size);
ctx.a.clearRect(0, 0, canvas.a.width, canvas.a.height);
mouse.x = window.innerWidth * 0.5 + cos(tick * 0.05) * 200;
mouse.y = window.innerHeight * 0.5 + sin(tick * 0.05) * 200;
tick % 2 === 0 && particles.push(getParticle(mouse.x, mouse.y));
for (let i = particles.length - 1; i >= 0; i--) {
if (particles[i].destroy) particles.splice(i, 1);
ctx.b.fillStyle = 'rgba(0,0,0,0.05)';
ctx.b.fillRect(0, 0, canvas.b.width, canvas.b.height);
ctx.b.globalCompositeOperation = "lighter";
ctx.b.filter = "blur(8px)";
ctx.b.drawImage(canvas.a, 0, 0, canvas.b.width, canvas.b.height);
ctx.b.globalCompositeOperation = "lighter";
ctx.b.drawImage(canvas.a, 0, 0, canvas.b.width, canvas.b.height);
window.requestAnimationFrame(draw);
window.addEventListener("load", setup);
window.addEventListener("resize", resize);
window.addEventListener("mousemove", mousehandler);
window.addEventListener("mouseout", mousehandler);