function map(anchor) {
const context = DOM.context2d(width, height),
path = d3.geoPath().context(context);
context.beginPath();
for (let k = 0; k < graph.sources.length; k++) {
const i = graph.sources[k],
j = graph.targets[k];
if (i < j) {
context.moveTo(...pts[i]);
context.lineTo(...pts[j]);
}
}
context.strokeStyle = "#ccc";
context.lineWidth = 1;
context.stroke();
for (let i = 0; i < origins.length; i++) {
const c = d3
.tricontour()
.value((d, j) => (tree.origin[j] === origins[i] ? 1 : 0))
.contour(pts.concat(anchor.includes("use") ? ethereal : []), 0.5);
context.beginPath();
path(c);
context.globalAlpha = 0.2;
context.strokeStyle = context.fillStyle = color(origins[i]);
context.fill();
context.globalAlpha = 1;
}
if (anchor.includes("show")) {
context.beginPath();
path.pointRadius(1);
path({ type: "MultiPoint", coordinates: ethereal });
context.fillStyle = "#888";
context.fill();
}
context.lineWidth = 2;
for (const k of origins) {
context.beginPath();
context.strokeStyle = color(k);
for (let i = 0; i < pts.length; i++) {
const j = tree.predecessor[i];
if (j > -1 && tree.origin[j] === k) {
context.moveTo(...pts[i]);
context.lineTo(...pts[j]);
}
}
context.stroke();
}
path.pointRadius(6);
for (const k of origins.filter(i => i < pts.length)) {
context.beginPath();
path({
type: "Point",
coordinates: pts[k]
});
context.fillStyle = color(k);
context.strokeStyle = "white";
context.fill();
context.stroke();
}
return context.canvas;
}