function display(contour, triangles) {
const svg = d3
.create("svg")
.style("background", "#333")
.attr("viewBox", [
scales.x(-1.5 / 20),
scales.y(-1.5 / 20),
scales.x(1 + 1.5 / 20) - scales.x(-1.5 / 20),
scales.y(1 + 1.5 / 20) - scales.y(-1.5 / 20)
]);
const g = svg.append("g");
const path = d3.geoPath().pointRadius(scales.x(2 / width) - scales.x(0));
const contours = contour(data);
g.append("g")
.selectAll("path")
.data(contours)
.join("path")
.attr("d", path)
.attr("fill", (d) => scales.color(d.value))
.attr("fill-opacity", 0.8)
.append("title")
.text((d) => d.value);
function vertices(a) {
return Array.from(triangles)
.filter((_, i) => i % 3 === a)
.map((d) => data[d]);
}
g.append("path")
.attr("vector-effect", "non-scaling-stroke")
.attr(
"d",
path({
type: "MultiLineString",
coordinates: d3.zip(vertices(0), vertices(1), vertices(2), vertices(0))
})
)
.attr("fill", "none")
.attr("stroke-width", 1)
.attr("stroke", "white");
const pts = g
.append("g")
.selectAll("circle")
.data(data)
.join("path")
.attr("d", (d) => path({ type: "Point", coordinates: d }))
.attr("fill", (d) => scales.color(d[2]));
path.pointRadius(2 * path.pointRadius());
const coordinates = Array.from(triangulated.hull).map((i) => data[i]);
coordinates.push(coordinates[0]);
g.append("path")
.attr("vector-effect", "non-scaling-stroke")
.attr(
"d",
path({
type: "LineString",
coordinates
})
)
.attr("fill", "none")
.attr("stroke-width", 2)
.attr("stroke", "red");
g.append("g")
.selectAll("circle")
.data(triangulated.hull)
.join("path")
.attr("d", (d) => path({ type: "Point", coordinates: data[d] }))
.attr("fill", "red");
g.append("g")
.selectAll("text")
.data(data)
.join("text")
.style("fill", "white")
.style("font-size", scales.x(14 / width) - scales.x(0))
.attr("transform", (d) => `translate(${d[0]}, ${d[1]})`)
.text((d, i) => i);
return svg.node();
}