viewof centroidal = {
const { width, height, data } = imgdata;
const n = quads.length;
const points = new Float64Array(n * 2);
quads.forEach(({ center }, i) => {
points[i * 2] = center[0];
points[i * 2 + 1] = center[1];
});
const del = new d3.Delaunay(points);
const vor = del.voronoi([0, 0, width, height]);
const c = new Float64Array(n * 2);
const s = new Float64Array(n);
const getDensity = (x, y) => lumGradient[y * width + x];
const getCentroids = () => {
c.fill(0);
s.fill(0);
for (let y = 0, i = 0; y < height; ++y) {
for (let x = 0; x < width; ++x) {
const w = getDensity(x, y);
i = del.find(x + 0.5, y + 0.5, i);
s[i] += w;
c[i * 2] += w * (x + 0.5);
c[i * 2 + 1] += w * (y + 0.5);
}
}
for (let i = 0; i < n; ++i) {
const x0 = points[i * 2],
y0 = points[i * 2 + 1];
const x1 = s[i] ? c[i * 2] / s[i] : x0,
y1 = s[i] ? c[i * 2 + 1] / s[i] : y0;
points[i * 2] = x1;
points[i * 2 + 1] = y1;
}
};
const jitter = () => {
for (let i = 0; i < n; ++i) {
const x = points[i * 2],
y = points[i * 2 + 1];
points[i * 2] = x + Math.random() * 0.5 - 0.25;
points[i * 2 + 1] = y + Math.random() * 0.5 - 0.25;
}
};
let ctx = DOM.context2d(width, height, 1);
ctx.canvas.value = null;
ctx.strokeStyle = "black";
for (let i = 0; i < lloydIterations; i++) {
getCentroids();
if (i == lloydIterations - 1) jitter();
vor.update();
ctx.clearRect(0, 0, width, height);
ctx.beginPath();
vor.render(ctx);
vor.renderBounds(ctx);
ctx.stroke();
for (let i = 0; i < n; i++) {
ctx.beginPath();
ctx.arc(points[i * 2], points[i * 2 + 1], 1, 0, Math.PI * 2);
ctx.fill();
}
yield ctx.canvas;
}
ctx.canvas.value = vor;
ctx.canvas.dispatchEvent(new CustomEvent("input"));
}