viewof drawZoomableTernaryPlot = {
const node = DOM.svg(width, height);
const svg = d3.select(node);
const radius = zoomableTernaryPlot.radius();
const chart = svg
.append("g")
.attr("transform", `translate(${width / 2} ${height / 2 + yOffset})`)
.attr("font-family", "sans-serif")
.attr("id", "chart")
.on("mousemove", handleMouseMove);
const defs = chart.append("defs");
const clipPath = defs
.append("clipPath")
.attr("id", "trianglePath")
.append("path")
.attr("d", zoomableTernaryPlot.triangle());
const transformedTrianglePath = chart
.append("path")
.attr("d", zoomableTernaryPlot.triangle())
.attr("fill", "none")
.attr("stroke", "coral")
.attr("title", " Scaled & translated triangle")
.attr("stroke-width", 2);
const axisLabelsGroup = chart
.append("g")
.attr("class", "axis-labels")
.call(axisLabels, zoomableTernaryPlot.axisLabels());
const gridLinesPaths = zoomableTernaryPlot
.gridLines()
.map((axisGrid) => axisGrid.map(d3.line()).join(" "));
const gridGroup = chart
.append("g")
.attr("class", "grid")
.call(grid, gridLinesPaths);
const axisTicksGroups = chart
.append("g")
.attr("class", "ternary-ticks")
.attr("font-size", 10)
.selectAll("g")
.data(zoomableTernaryPlot.ticks())
.join("g")
.attr("class", "axis-ticks");
axisTicksGroups.call(ticks);
const trianglePath = chart
.append("path")
.attr("d", zoomableTernaryPlot.triangle())
.attr("fill", "transparent")
.attr("stroke", "black")
.attr("title", "Initial untransformed triangle")
.attr("stroke-width", 2);
const ternaryData = data.map((d) => {
const [x, y] = zoomableTernaryPlot(d);
return { x, y, ...d };
});
const dots = chart
.append("g")
.attr("class", "data")
.attr("clip-path", "url(#trianglePath)")
.selectAll("circle")
.data(ternaryData)
.join("circle")
.attr("r", 4)
.attr("cx", (d) => d.x)
.attr("cy", (d) => d.y)
.attr("fill", "#444")
.attr("stroke", "#ddd");
dots.append("title").text(
(d) => `${d.country}
Agriculture: ${d.agriculture}
Industry: ${d.industry}
Service: ${d.service}`
);
const zoom = d3.zoom().scaleExtent([1, 100]).on("zoom", zoomed);
const { x, y, k } = transformFromDomains(zoomBarycentric.domains());
const initialTransform = d3.zoomIdentity
.translate(x * radius, y * radius)
.scale(k);
chart.call(zoom).call(zoom.transform, initialTransform);
function zoomed({ transform }) {
const { x, y, k } = transform;
const tx = x / radius,
ty = y / radius;
const domains = domainsFromTransform({ k, x: tx, y: ty });
zoomBarycentric.domains(domains);
dots
.attr("cx", (d) => zoomableTernaryPlot(d)[0])
.attr("cy", (d) => zoomableTernaryPlot(d)[1]);
const gridLinesPaths = zoomableTernaryPlot
.gridLines()
.map((axisGrid) => axisGrid.map(d3.line()).join(" "));
gridGroup.call(grid, gridLinesPaths);
axisTicksGroups.data(zoomableTernaryPlot.ticks()).call(ticks, (d) => d);
}
function handleMouseMove(d) {
const xy = d3.pointer(d);
const inverse = zoomableTernaryPlot.invert(xy);
node.dispatchEvent(new CustomEvent("input"), { bubbles: true });
node.value = inverse;
}
return svg.node();
}