createmap = () => {
const div = d3.create("div").style("width", width);
const details = div
.append("div")
.style("position", "absolute")
.style("top", 0)
.style("right", 0)
.style("width", "350px")
.style("height", "400px");
const svg = div.append("svg");
svg
.style("width", width)
.attr("viewBox", [0, 0, width, Math.min(height, 750)])
.style("font", "12px sans-serif");
const path = d3.geoPath(projection).pointRadius(2);
const hStyle = svg.append("defs").append("style");
const g = svg.append("g").style("vector-effect", "non-scaling-stroke");
svg.style("background", "white");
const bg = g
.append("g")
.on(
"click",
() => (
details.html(``),
dots
.selectAll(".dot")
.classed("selected", false)
.classed("highlight", false)
)
);
bg.append("path")
.datum(topojson.merge(deptsTopo, deptsTopo.objects.departements.geometries))
.attr("d", path)
.style("fill", "rgba(0,0,0,0.03)")
.style("stroke", "black")
.style("stroke-width", ".75");
bg.append("path")
.datum(topojson.mesh(deptsTopo))
.attr("d", path)
.style("fill", "none")
.style("stroke-width", 0.25)
.style("stroke", "black");
const phantoms = g
.append("path")
.attr("id", "phantoms")
.style("fill", "#ddd")
.datum({
type: "MultiPoint",
coordinates: aires2.map((d) => [d.lon, d.lat])
})
.attr("d", path);
const dots = g.append("g").attr("id", "dots").style("cursor", "pointer");
const deptLabelsBg = g
.append("g")
.style("text-anchor", "middle")
.selectAll("text")
.data(topojson.feature(deptsTopo, deptsTopo.objects.departements).features)
.join("text")
.text((d) => `${d.properties.code}`)
.attr("transform", (d) => `translate(${labelPos(d)})`)
.attr("pointer-events", "none");
const deptLabels = deptLabelsBg.clone(true);
deptLabelsBg.style("stroke", "white").style("stroke-width", 2.5);
let r = 1;
const zoom = d3
.zoom()
.scaleExtent([0.8, 10])
.on("zoom", ({ transform }) => {
g.attr("transform", transform);
r = 1 / Math.sqrt(transform.k);
deptLabels.style("font", `${12 * Math.pow(r, 1.5)}px sans-serif`);
deptLabelsBg
.style("font", `${12 * Math.pow(r, 1.5)}px sans-serif`)
.style("stroke-width", 2.5 * r * r);
if (transform.k > 8) {
deptLabelsBg.text((d) => `${d.properties.code}. ${d.properties.nom}`);
deptLabels.text((d) => `${d.properties.code}. ${d.properties.nom}`);
} else {
deptLabelsBg.text((d) => `${d.properties.code}`);
deptLabels.text((d) => `${d.properties.code}`);
}
resize();
});
svg.call(zoom).call(zoom.transform, d3.zoomIdentity);
return Object.assign(div.node(), { update });
function update(data) {
const pts = dots
.selectAll(".dot")
.data(data)
.join("g")
.html("")
.attr("transform", (d) => `translate(${d.x0},${d.y0}`)
.style("stroke-width", 0.75)
.style("stroke", "#333")
.attr("class", "dot")
.style("fill", (d) => color(d.Note));
pts.filter((d) => d.Note != 3).append("circle");
pts.filter((d) => d.Note == 3).append("rect");
pts.append("title").text((d) => d.Ville);
pts.on("mouseover", function () {
const that = this;
d3.select(that).raise();
pts.classed("highlight", function () {
return this === that;
});
});
pts.on("click", (event, d) => {
if (d === details.datum()) d = undefined;
details.datum(d);
details.html(tipcontent);
pts.classed("selected", (x) => x === d);
event.preventDefault();
});
pts.attr("cx", (d) => d.x).attr("cy", (d) => d.y);
resize();
}
function resize() {
phantoms.attr("d", path.pointRadius(2 * r));
const a = Math.min(1, Math.pow(r, 1.8));
dots
.selectAll(".dot circle")
.attr("r", 3 * r)
.attr("cx", (d) => a * d.x + (1 - a) * d.x0)
.attr("cy", (d) => a * d.y + (1 - a) * d.y0)
.style("vector-effect", "non-scaling-stroke");
dots
.selectAll(".dot rect")
.attr("width", 5.5 * r)
.attr("height", 5.5 * r)
.attr("x", (d) => a * d.x + (1 - a) * d.x0 - (5.5 * r) / 2)
.attr("y", (d) => a * d.y + (1 - a) * d.y0 - (5.5 * r) / 2)
.style("vector-effect", "non-scaling-stroke");
hStyle.text(`
circle.highlight {stroke-width: 2!important}
circle.selected {stroke-width: 3!important}
`);
}
}