function compactBeeswarm(data, { diameter = 1, x = d => d } = {}) {
const diameter2 = diameter ** 2;
const circles = data
.map((d, i, data) => ({ x: +x(d, i, data), placed: false, y: 0, data: d }))
.sort((a, b) => a.x - b.x);
function next() {
let best;
for (let c of circles) {
if (c.placed) continue;
if (!best || c.y < best.y) best = c;
}
return best;
}
function updateYValues(circle) {
for (let c of circles) {
if (c.placed) continue;
let xDiff = Math.abs(circle.x - c.x);
if (xDiff >= diameter) continue;
let yDiff = Math.sqrt(diameter2 - xDiff ** 2);
c.y = Math.max(c.y, circle.y + yDiff);
}
}
let time = 0;
for (let i = 0; i < circles.length; i++) {
let c = next();
c.placed = true;
c.insertionTime = ++time;
updateYValues(c);
}
return circles;
}