{
const container = d3.select(DOM.element("div"))
.style("position", "relative")
.style("width", width + "px")
.style("height", "720px");
const dpr = window.devicePixelRatio || 1;
const height = Math.min(width, 680);
const canvas = d3.create("canvas")
.attr("width", dpr * width)
.attr("height", dpr * height)
.style("width", width + "px")
.style("height", height + "px");
container.node().appendChild(canvas.node());
const context = canvas.node().getContext("2d");
context.scale(dpr, dpr);
let currentYear = 2022;
const slider = container.append("input")
.attr("type", "range")
.attr("min", 1960)
.attr("max", 2022)
.attr("step", 1)
.property("value", currentYear)
.style("position", "absolute")
.style("bottom", "10px")
.style("left", "50%")
.style("transform", "translateX(-50%)");
let autoPlay = false;
const autoPlayButton = container.append("button")
.text("Auto-play: OFF")
.style("position", "absolute")
.style("bottom", "10px")
.style("right", "10px")
.on("click", function() {
autoPlay = !autoPlay;
d3.select(this).text(`Auto-play: ${autoPlay ? "ON" : "OFF"}`);
});
const legendWidth = 300, legendHeight = 60;
const legendSVG = container.append("svg")
.attr("width", 300)
.attr("height", 300)
.style("position", "absolute")
.style("bottom", "60px")
.style("left", "50%")
.style("transform", "translateX(-50%)");
const defs = legendSVG.append("defs");
const gradient = defs.append("linearGradient")
.attr("id", "legend-gradient")
.attr("x1", "0%")
.attr("x2", "100%")
.attr("y1", "0%")
.attr("y2", "0%");
let legendScale;
let dataYear, co2ByCountry, maxCo2, colorScale;
function updateData(year) {
dataYear = data
.map(d => ({ ...d, year: +d.year, co2: +d.co2 }))
.filter(d =>
d.year === +year &&
!isNaN(d.co2) &&
d.country && d.country !== "" &&
d.co2 > 0
);
co2ByCountry = new Map(dataYear.map(d => [d.country.toLowerCase().trim(), d.co2]));
maxCo2 = d3.max(dataYear, d => d.co2);
colorScale = d3.scaleSequentialLog(d3.interpolateInferno)
.domain([1, maxCo2]);
const stops = d3.range(0, 1.01, 0.1);
gradient.selectAll("stop")
.data(stops)
.join("stop")
.attr("offset", d => (d * 100) + "%")
.attr("stop-color", d => {
const value = Math.exp(Math.log(1) + d * (Math.log(maxCo2) - Math.log(1)));
return colorScale(value);
});
legendScale = d3.scaleLog()
.domain([1, maxCo2])
.range([0, legendWidth]);
}
updateData(currentYear);
const projection = d3.geoOrthographic()
.fitExtent([[10, 80], [width - 10, height - 10]], { type: "Sphere" });
const path = d3.geoPath(projection, context);
let rotationAngle = 0;
let autoRotate = true;
let initialRotation, dragStartPos;
d3.select(canvas.node())
.call(d3.drag()
.on("start", function(event) {
autoRotate = false;
initialRotation = projection.rotate();
dragStartPos = [event.x, event.y];
})
.on("drag", function(event) {
const dx = event.x - dragStartPos[0],
dy = event.y - dragStartPos[1];
const sensitivity = 0.5;
let newLambda = initialRotation[0] + dx * sensitivity,
newPhi = initialRotation[1] - dy * sensitivity;
newPhi = Math.max(-90, Math.min(90, newPhi));
projection.rotate([newLambda, newPhi]);
rotationAngle = newLambda;
draw();
})
.on("end", function() {
setTimeout(() => { autoRotate = true; }, 100);
})
);
function draw() {
context.clearRect(0, 0, width, height);
if (autoRotate) {
rotationAngle = (performance.now() / 50) % 360;
projection.rotate([rotationAngle, -15]);
}
context.beginPath();
path(land);
context.fillStyle = "#ccc";
context.fill();
for (const feature of countries) {
context.beginPath();
path(feature);
const countryName = feature.properties.name?.toLowerCase().trim();
const co2Value = co2ByCountry.get(countryName);
const fillColor = (co2Value != null)
? colorScale(co2Value)
: "#eee";
context.fillStyle = fillColor;
context.fill();
}
context.beginPath();
path(borders);
context.strokeStyle = "#fff";
context.lineWidth = 0.5;
context.stroke();
context.beginPath();
path({ type: "Sphere" });
context.strokeStyle = "#000";
context.lineWidth = 1.5;
context.stroke();
context.save();
context.font = "bold 48px sans-serif";
context.fillStyle = "black";
context.fillText(String(currentYear), 30, 60);
context.restore();
context.save();
context.font = "bold 24px sans-serif";
context.textAlign = "center";
context.fillStyle = "black";
context.fillText("Country by CO₂ " + currentYear, width / 2, 30);
context.restore();
}
slider.on("input", function() {
currentYear = +this.value;
updateData(currentYear);
draw();
});
setInterval(() => {
if (autoPlay) {
currentYear = +currentYear + 1;
if (currentYear > +slider.attr("max")) {
currentYear = +slider.attr("min");
}
slider.property("value", currentYear);
updateData(currentYear);
draw();
}
}, 1000);
d3.timer(elapsed => {
if (autoRotate) {
rotationAngle = (elapsed / 50) % 360;
projection.rotate([rotationAngle, -15]);
}
draw();
});
return container.node();
}