chart = {
const svg = d3.create("svg")
.attr("viewBox", [0, 0, 975, 610])
.attr("stroke-linecap", "round")
.attr("stroke-linejoin", "round");
fnGeoProjection.fitExtent(
[[0, 0],[width, height]],
topojson.feature(europe, europe.objects.nutsrg)
)
svg.selectAll("path")
.data(topojson.feature(europe, europe.objects.nutsrg).features)
.enter()
.append("path")
.attr("fill", d =>
d.properties.id in mapping && mapping[d.properties.id].change
? fnColor(mapping[d.properties.id].change)
: '#aaa'
)
.attr("d", fnGeoPath)
.on("mouseover", fnOver)
.on("mouseout", fnOut)
function capitalizeFirstLetter(string) {
return string.charAt(0).toUpperCase() + string.slice(1);
}
const text = svg.append("text")
.attr("fill", "black")
.attr("text-anchor", "middle")
.attr("font-family", "sans-serif")
.attr("font-size", "18px")
.style("pointer-events", "none")
.attr("opacity", 0)
function fnOver(d) {
d3.select(this).attr('fill-opacity', 0.4)
const v = mapping[d.properties.id].change
text.text(`${capitalizeFirstLetter(d.properties.na.toLowerCase())}: ${v ? d3.format("+.1%")(v) : '?%'}`)
.transition()
.attr("opacity", 1)
}
function fnOut() {
d3.select(this).attr('fill-opacity', null)
text.transition().attr("opacity", 0)
}
function fnMove() {
const p = d3.mouse(this)
text.attr('x', p[0])
.attr('y', p[1] - 20)
}
svg.on("mousemove", fnMove)
svg.append("path")
.attr("fill", "none")
.attr("stroke", "#222")
.attr("stroke-width", 0.5)
.style("pointer-events", "none")
.attr("d", fnGeoPath(topojson.mesh(europe, europe.objects.nutsrg)))
return svg.node();
}