chart = {
const svg = d3.create("svg")
.style("cursor", "default")
.attr("viewBox", [0, 0, width, height]);
const g = svg.selectAll(".waffle")
.data(waffles)
.join("g")
.attr("class", "waffle");
if (!whole) {
const numPerRow = Math.floor(width / (waffleSize + padding.x));
g.attr("transform", (d, i) => {
const r = Math.floor(i / numPerRow);
const c = i - r * numPerRow;
return `translate(${c * (waffleSize + padding.x)},${r * (waffleSize + padding.y)})`
});
}
const cellSize = scale.bandwidth();
const half = cellSize / 2;
const cells = g.append("g")
.selectAll(options.shape)
.data(d => d)
.join(options.shape)
.attr("fill", d => d.index === -1 ? "#ddd" : color(d.index));
if (isRect) {
cells.attr("x", d => scale(d.x))
.attr("y", d => whole ? 0 : scale(d.y))
.attr("rx", 3).attr("ry", 3)
.attr("width", cellSize).attr("height", cellSize)
}
else {
cells.attr("cx", d => scale(d.x) + half)
.attr("cy", d => whole ? 0 : scale(d.y) + half)
.attr("r", half);
}
if (whole) {
cells.append("title").text(d => {
const cd = chartData[d.index];
return `${cd.territory}\n${toCurrency(cd.profit)} (${cd.ratio.toFixed(1)}%)`;
});
cells.transition()
.duration(d => d.y * 100)
.ease(d3.easeBounce)
.attr(isRect ? "y" : "cy", d => scale(d.y) + (isRect ? 0 : half));
svg.transition().delay(550)
.on("end", () => drawLegend(svg, cells));
}
else {
g.append("text")
.style("font-size", 20)
.style("font-weight", "bold")
.attr("dy", "1.5em")
.attr("text-anchor", "middle")
.attr("fill", (d, i) => color(i))
.attr("transform", `translate(${waffleSize / 2},0)`)
.text((d, i) => `${chartData[i].ratio.toFixed(0)}%`);
g.append("g")
.attr("transform", `translate(0,${waffleSize + padding.y / 2.5})`)
.call(g => g.append("text").text((d, i) => chartData[i].territory))
.call(g => g.append("text").attr("dy", "1em").text((d, i) => toCurrency(chartData[i].profit)));
}
return svg.node();
}