animation = (options = {}) => {
const {
width: w = 600,
height: h = w,
pixelRatio: dpr = window.devicePixelRatio || 1,
count: n = 70,
lineWidth: lw = 1.5,
bgColor: bg = '#fff',
fillColor: fill = '#fff',
strokeColor: stroke = '#000',
t = (t) => 1 - t,
x = (t, ti) => w/2,
y = (t, ti) => mix(-.1, 1.5, ti) * h,
r = (t, ti) => mix(.1, .3, ease((ti-t+1) % 1, 2)) * h,
s = (t, ti) => mix(.55, .4, 1 - ease((ti+t) % 1, 4)),
a = (t, ti) => mix(-.02, .07, ease((ti+t) % 1, 3)),
} = options;
const c = DOM.context2d(w, h, dpr);
c.lineWidth = lw;
c.strokeStyle = stroke;
c.canvas.style.maxWidth = '100%';
return _t => {
c.fillStyle = bg;
c.fillRect(0, 0, w, h);
c.fillStyle = fill;
const o = t(_t);
for(let i = 0; i < n; i++) {
const ti = i/n, _r = r(o, ti),
ellipse = (l = 0) => c.ellipse(
x(o, ti),
y(o, ti),
l + _r,
l + _r * s(o, ti),
a(o, ti) * Math.PI,
0,
Math.PI * 2
);
if(lw > 0) c.fillStyle = stroke; c.beginPath(); ellipse(lw*2); c.fill();
c.fillStyle = fill; c.beginPath(); ellipse(lw); c.fill();
}
return c.canvas;
};
}