legend = () => {
const ticksize = 4;
const margin = {left: 0, right: 0, top: 18, bottom: ticksize + 30};
const width = 320 - margin.left - margin.right;
const height = 12;
const textpad = 12;
const x = d3.scaleBand().domain(d3.range(5)).range([0, width / 2]);
const epsilon = 1;
const wrapper = d3.create("div")
.style("width", `${width + margin.left + margin.right}px`)
.style("margin", "0 auto")
.style("margin-bottom", "-40px")
const titles = wrapper.append("div")
.style("font-family", franklinLight)
.style("margin-bottom", "12px")
.style("text-align", "center");
titles.append("div")
.style("font-size", "18px")
.style("font-weight", "bold")
.text("Change in length of mosquito season");
titles.append("div")
.text("1980-2009 vs. 2019-2023");
const svg = wrapper.append("svg")
.attr("font-family", franklinLight)
.attr("width", width + margin.left + margin.right)
.attr("height", height + margin.top + margin.bottom);
const g = svg.append("g")
.attr("transform", `translate(${[margin.left, margin.top]})`)
g.selectAll(".label")
.data(["← Shorter", "Longer →"])
.join("text")
.attr("font-size", 16)
.attr("x", (d, i) => width / 2 + (16 * (i === 0 ? -1 : 1)))
.attr("y", -6)
.attr("text-anchor", (d, i) => i === 0 ? "end" : "start")
.text(d => d);
const decrease = g.append("g");
const decreaseData = x.domain()
.map((x, i, e) => {
const weeks = e.length - x - 1;
return {
x,
color: color(1 - 7 + 7 * x - 28),
html: i === 0 ? `<tspan>${e.length - 1}</tspan><tspan x=0 dy=12>weeks</tspan>` : weeks === 0 ? "" : `${weeks}`
}});
const rect = [[0, 0], [x.bandwidth(), 0], [x.bandwidth(), height], [0, height]];
const decreaseRect = decrease.selectAll("polygon")
.data(decreaseData)
.join("polygon")
.attr("transform", d => `translate(${x(d.x)})`)
.attr("points", (d, i, e) => {
if (i === 0) {
return [
[0, height / 2],
[x.bandwidth() / 2, 0],
[x.bandwidth(), 0],
[x.bandwidth(), height],
[x.bandwidth() / 2, height],
[0, height / 2]
]
}
else {
return rect;
}
})
.attr("fill", d => d.color);
const decreaseTick = decrease.selectAll(".tick")
.data(decreaseData)
.join("g")
.attr("class", "tick")
.attr("transform", d => `translate(${x(d.x) + x.bandwidth()})`);
decreaseTick.append("line")
.attr("y2", (d, i, e) => height + ticksize * (i === e.length - 1 ? 0 : 1))
.attr("stroke", "black")
decreaseTick.append("text").filter(d => d)
.attr("font-size", 14)
.attr("text-anchor", "middle")
.attr("y", height + ticksize + textpad)
.html(d => d.html);
const increase = g.append("g")
.attr("transform", `translate(${width / 2})`);
const increaseData = x.domain()
.map((x, i, e) => {
const weeks = x;
return {
x,
color: color(epsilon + 7 * x),
html: weeks === 0 ? "" : `<tspan>${weeks}</tspan><tspan x=0 dy=12>${i === e.length - 1 ? " weeks</tspan>" : ""}`
}});
const increaseRect = increase.selectAll("polygon")
.data(increaseData)
.join("polygon")
.attr("transform", d => `translate(${x(d.x)})`)
.attr("points", (d, i, e) => {
if (i === e.length - 1) {
return [
[0, 0],
[x.bandwidth() / 2, 0],
[x.bandwidth(), height / 2],
[x.bandwidth() / 2, height],
[0, height],
[0, 0]
]
}
else {
return rect;
}
})
.attr("fill", d => d.color);
const increaseTick = increase.selectAll(".tick")
.data(increaseData)
.join("g")
.attr("class", "tick")
.attr("transform", d => `translate(${x(d.x)})`);
increaseTick.append("line")
.attr("y2", (d, i) => height + ticksize * Math.min(i, 1))
.attr("stroke", "black")
increaseTick.append("text").filter(d => d)
.attr("font-size", 14)
.attr("text-anchor", "middle")
.attr("y", height + ticksize + textpad)
.html(d => d.html);
return wrapper.node();
}