function map(projection) {
const context = DOM.context2d(width, width / 2);
projection.fitExtent(
[
[2, 2],
[width - 2, width / 2 - 2]
],
{ type: "Sphere" }
);
var path = d3.geoPath(projection, context);
context.clearRect(0, 0, width, width / 2);
if (show_sphere)
context.beginPath(),
path({ type: "Sphere" }),
(context.fillStyle = "#fefef6"),
context.fill();
if (show_sphere)
context.beginPath(),
path(d3.geoGraticule()()),
(context.strokeStyle = "#ccc"),
context.stroke();
if (show_equator)
context.beginPath(),
path(
d3
.geoGraticule()
.step([0, 100])
.extent([
[-179.99, -25],
[179.99, 25]
])()
),
(context.strokeStyle = "brown"),
context.stroke();
if (show_land)
context.beginPath(),
path(land),
(context.fillStyle = "black"),
context.fill();
if (show_sphere || show_structure || !show_land)
context.beginPath(),
path({ type: "Sphere" }),
(context.strokeStyle = "#000"),
context.stroke();
// Polyhedral projections expose their structure as projection.tree()
// To draw them we need to cancel the rotate
if (show_structure && projection.tree) {
var rotate = projection.rotate();
projection.rotate([0, 0, 0]);
// run the tree of faces to get all sites and folds
var sites = [],
folds = [],
i = 0;
function recurse(face) {
var site = d3.geoCentroid({ type: "MultiPoint", coordinates: face.face });
site.id = face.id || i++;
sites.push(site);
if (face.children) {
face.children.forEach(function (child) {
folds.push({
type: "LineString",
coordinates: child.shared.map((e) =>
d3.geoInterpolate(e, face.centroid)(1e-5)
)
});
recurse(child);
});
}
}
recurse(projection.tree());
// sites & numbers
context.beginPath(),
path.pointRadius(10)({ type: "MultiPoint", coordinates: sites }),
(context.fillStyle = "white"),
(context.strokeStyle = "black"),
context.fill(),
context.stroke();
sites.forEach((site) => {
(context.textAlign = "center"),
(context.fillStyle = "black"),
(context.font = "16px Georgia"),
(context.textBaseline = "middle"),
context.fillText(site.id, projection(site)[0], projection(site)[1] - 1);
});
// folding lines
folds.forEach((fold) => {
context.beginPath(),
(context.lineWidth = 0.5),
context.setLineDash([3, 4]),
(context.strokeStyle = "#888"),
path(fold),
context.stroke(),
context.setLineDash([]);
});
// restore the projection’s rotate
projection.rotate(rotate);
}
d3.select(context.canvas).on("pointermove", function (event) {
const gni = projection.invert(d3.pointer(event));
context.beginPath(),
path.pointRadius(2)({ type: "Point", coordinates: gni }),
(context.fillStyle = "green"),
(context.strokeStyle = "black"),
context.fill(),
context.stroke();
});
return context.canvas;
}