function* render(options = {}) {
const {
size: s = 600,
pad = -.1,
seed = 'b',
count = 500,
duration = 4000,
gif = false,
invalidation: invalidate = invalidation,
} = options;
const p = s * pad, sp = s - (p * 2);
const rng = seedrandom(seed);
const coords = [p,p, s-p,p, p,s-p, s-p,s-p];
for(let n = 0; n < count; n++) coords.push(p+sp*rng(), p+sp*rng());
const delaunay = new Delaunator(new Float64Array(coords));
const offsets = new Float64Array(delaunay.triangles.length * 2);
for(let i = 0; i < delaunay.triangles.length; i++) {
offsets[i*2 ] = delaunay.coords[2*delaunay.triangles[i]];
offsets[i*2+1] = delaunay.coords[2*delaunay.triangles[i]+1];
}
const ctx = DOM.context2d(s, s, 1);
ctx.canvas.style.maxWidth = '100%';
if(gif) return yield* renderGif(invalidate, drawFrame, duration, {
preview: drawFrame(0),
filename: `delaunator-shards-${Date.now()/1000|0}`,
fps: 40,
});
const to = Date.now();
while(true) {
yield drawFrame(((Date.now()-to)/duration) % 1);
}
function drawFrame(t) {
ctx.fillStyle = '#000';
ctx.fillRect(0, 0, s, s);
t = 1-t;
const o = offsets;
for(let i = 0; i < o.length; i += 6) {
const
ti = (((i / o.length) ** .1) * 2 + t) % 1,
_t = clamp(0, 1, 1.5 * easeCos(ti ** 4 + .4)),
x0 = o[i ], y0 = o[i+1],
x1 = o[i+2], y1 = o[i+3],
x2 = o[i+4], y2 = o[i+5],
origins = [
[x0, y0, .3],
[x1, y1, .5],
[x2, y2, 1],
];
for(let [xc, yc, l] of origins) {
ctx.fillStyle = `hsl(0,0%,${_t * l * 100}%)`;
ctx.beginPath();
ctx.moveTo(mix(xc, x0, _t), mix(yc, y0, _t));
ctx.lineTo(mix(xc, x1, _t), mix(yc, y1, _t));
ctx.lineTo(mix(xc, x2, _t), mix(yc, y2, _t));
ctx.closePath();
ctx.fill();
}
}
return ctx.canvas;
}
}