function drawMap(
geo,
{
dataAccessor,
areaAccessor,
valueAccessor,
titleAccessor = (d) => {},
width = 640,
height = 400,
margin = 10,
projection = d3.geoTransverseMercator().rotate([-80, 0, 0]),
scale = d3.scaleQuantize,
range = colorScheme[5],
domain = [0, 1],
nice = true,
maxRadius = 25,
unknown = "#f4f4f4",
stroke = "#fff",
strokeWidth = 0.5,
strokeOpacity,
strokeLinecap = "round",
strokeLinejoin = "round",
drawUnderlay = false,
underlayFill = "#fff",
underlayStroke = "#999",
colorLegendTitle,
tickFormat = ".2r",
tickValues,
legendWidth = 200,
areaTickUnit,
backgroundFill = "white",
debug
} = {}
) {
let features = geo.features.slice();
features = features.map((f) => ({ ...f, geoCentroid: d3.geoCentroid(f) }));
const D = features.map(dataAccessor);
const S = D.map(areaAccessor);
const V = D.map(valueAccessor);
const T = D.map(titleAccessor);
// Compute default domains.
domain = domain == null ? d3.extent(V) : domain;
// Construct scales
let color = scale(domain, range);
if (nice && typeof color.nice === "function") color.nice();
if (color.unknown && unknown !== undefined) color.unknown(unknown);
const radiusDomain = [0, d3.max(S)];
const radius = d3.scaleSqrt().domain(radiusDomain).range([0, maxRadius]);
projection = projection.scale === undefined ? projection() : projection;
// Adding margin to chart
// https://github.com/d3/d3-geo/blob/main/README.md#projection_fitSize
projection.fitExtent(
[
[margin, margin],
[width - margin, height - margin]
],
geo
);
const path = d3.geoPath(projection);
// Set intial positions for applying force
features = features.map((f, i) => {
const centroid = projection(f.geoCentroid);
const [x, y] = centroid;
const r = S[i] == null ? 0 : radius(S[i]);
const dx = 15;
const transitionRange = [
CSMath.inverseLerp(0, width, Math.max(0, x - dx)),
CSMath.inverseLerp(0, width, Math.min(width, x + dx))
];
return {
...f,
centroid,
x,
y,
radius: r,
transitionRange
};
});
features = applyForce(features, { width, height, debug });
// Set path interpolator
features = features.map((f, i) => {
const basePath = path(f);
const interpolater = flubber.toCircle(basePath, f.x, f.y, f.radius);
return {
...f,
interpolater
};
});
const svg = DOM.svg(width, height);
d3.select(svg).style("background", backgroundFill);
const plot = d3.select(svg).append("g").attr("class", "plot");
if (drawUnderlay) {
plot
.append("path")
.datum(geo)
.attr("class", "underlay")
.attr("stroke", underlayStroke)
.attr("stroke-linecap", strokeLinecap)
.attr("stroke-linejoin", strokeLinejoin)
.attr("stroke-width", strokeWidth)
.attr("stroke-opacity", strokeOpacity)
.attr("fill", underlayFill)
.attr("d", path);
}
const shapes = plot
.selectAll(".boundary")
.data(features)
.join("path")
.attr("class", "boundary")
.attr("stroke", stroke)
.attr("stroke-linecap", strokeLinecap)
.attr("stroke-linejoin", strokeLinejoin)
.attr("stroke-width", strokeWidth)
.attr("stroke-opacity", strokeOpacity)
.attr("fill", (d, i) => color(V[i]));
shapes.append("title").text((d, i) => T[i]);
// Add legends
const legend = d3
.select(svg)
.append("g")
.attr("class", "legend")
.attr("transform", `translate(${width - legendWidth - margin},${margin})`);
const colorKey = Legend(color, {
title: colorLegendTitle,
tickFormat,
tickValues,
width: legendWidth
});
const radiusKey = circleLegend({
scale: radius,
marginTop: 0,
marginBottom: 0,
stroke: "#666",
strokeWidth: 0.75,
// tickFormat: radius.tickFormat(4, "s"),
tickFormat: (d, i, g) => {
const str = radius.tickFormat(4, "s")(d);
return i === g.length - 1 && areaTickUnit
? `${str} ${areaTickUnit}`
: str;
},
tickFont: "10px sans-serif",
tickStrokeWidth: 0.75,
tickStroke: "#666"
});
legend.append("g").node().appendChild(colorKey);
legend
.append("g")
.attr("transform", `translate(${legendWidth / 4},70)`)
.node()
.appendChild(radiusKey);
if (debug) {
const debugCanvas = d3.select(svg).append("g").attr("class", "debug");
debugCanvas
.selectAll(".centroid")
.data(features)
.join("circle")
.attr("class", "centroid")
.attr("r", 2)
.attr("fill", "#f0f")
.attr("cx", (d) => d.centroid[0])
.attr("cy", (d) => d.centroid[1]);
}
function draw(t = 0) {
const T = eases.sineInOut(t);
shapes.each(function (d, i) {
const shape = d3.select(this);
const { interpolater, transitionRange } = d;
const shapeT = CSMath.clamp01(CSMath.inverseLerp(...transitionRange, T));
shape.attr("d", () => interpolater(eases.sineInOut(shapeT)));
});
}
return Object.assign(svg, { features, draw });
}