function setup_hexmap(precincts) {
let W = 1000;
let w = width <= W ? width : W;
let h = 0.65 * w;
let svg = d3.create("svg:svg").attr("width", w).attr("height", h);
let r = (36 * w) / W;
svg
.append("g")
.selectAll("path.outer")
.data(precincts)
.join("path")
.attr("class", "outer")
.attr("d", (o) => hex(o.i, o.j, r))
.attr("stroke", "white")
.attr("stroke-width", 2)
.attr("fill", "#ddd");
svg
.append("g")
.selectAll("path.inner")
.data(precincts)
.join("path")
.attr("class", "inner")
.attr("d", (o) => hex(o.i, o.j, r, 0.5))
.attr("stroke", "black")
.attr("stroke-width", (o) => (o.minority ? 5 : 1))
.attr("fill", (o) => (o.color == "yellow" ? "#EBB35F" : "#B180B0"));
let groups = d3.group(precincts, (o) => o.district);
let good_partition = true;
d3.range(1, 10).forEach(function (i) {
if (groups.has(i)) {
if (groups.get(i).length != 15) {
good_partition = false;
}
} else {
good_partition = false;
}
});
if (good_partition) {
groups.forEach(function (g) {
let purples = g.filter((p) => p.color == "purple");
let yellows = g.filter((p) => p.color == "yellow");
let color = purples.length > yellows.length ? "purple" : "yellow";
let merged_precinct = polygonClipping.union(
g.map((o) => [vertices(o.i, o.j, 1)])
);
svg
.append("path")
.attr(
"d",
d3.line().curve(d3.curveLinearClosed)(merged_precinct[0][0].map(([x, y]) => [r * x, r * y]))
)
.attr("stroke", "black")
.attr("stroke-width", 5)
.attr("fill", color)
.attr("fill-opacity", 0.1)
.on("pointerenter", function () {
d3.select(this).attr("fill-opacity", 0.2);
})
.on("pointerleave", function () {
d3.select(this).attr("fill-opacity", 0.1);
});
});
}
return svg.node();
}