I am working on mastering some D3 graphs and charts and am currently working on a collapsible tree. I have the tree visualized on the webpage but no matter what resources I go to I cannot get the tree to become interactive and have the nodes open and close the branches.
I have slowly been adding code based on documentation on the D3 Collapsible Tree example as well as other forums, youtube, and chatgbt, but everything I have looked at kind of gives me the same issue, where when I click the nodes, they do register that something is happening (they change colour and get a boarder around them) but they do not expand or collapse anything on the tree.
Any guidance would be greatly appreciated. I have a feeling I'm mixing up some things or missing a line or two of valuable code, but I'm so new to this I don't know how to go about problem solving it.
repo: https://github.com/nikkivno/d3.jsSamples
Here is the code that should be working to open and close the branches:
function toggle(d) {
if (d.children) {
d._children = d.children;
d.children = null;
} else {
d.children = d._children;
d._children = null;
}
treeLayout(rootNode);
update();
}
function update() {
const nodes = svg.selectAll(".node")
.data(rootNode.descendants(), function(d) { return d.data.id; });
const links = svg.selectAll(".link")
.data(rootNode.links());
const linkEnter = links.enter().insert("path", ".node")
.attr("class", "link")
.attr("d", d3.linkHorizontal()
.x(function(d) { return d.y; })
.y(function(d) { return d.x; })
)
.style("stroke", "#ccc")
.style("stroke-width", 1);
const nodeEnter = nodes.enter().append("g")
.attr("class", "node")
.attr("transform", function(d) { return "translate(" + d.y + "," + d.x + ")"; })
.on("click", toggle);
nodeEnter.append("circle")
.attr("r", 5)
.attr("fill", d => d._children ? "#555" : "#999")
.attr("stroke-width", 10)
.style('fill', 'white');
nodeEnter.append("text")
.attr("dy", ".31em")
.attr("x", function(d) { return d.children ? -8 : 8; })
.style("text-anchor", function(d) { return d.children ? "end" : "start"; })
.style("fill", "white")
.text(function(d) { return d.data.name; })
.attr("stroke-linejoin", "round");
nodes.merge(nodeEnter).transition().duration(750)
.attr("transform", function(d) { return "translate(" + d.y + "," + d.x + ")"; });
links.merge(linkEnter).transition().duration(750)
.attr("d", d3.linkHorizontal()
.x(function(d) { return d.y; })
.y(function(d) { return d.x; })
);
nodes.exit().remove();
links.exit().remove();
}
Your issue is in the
toggle
function. In later versions ofd3
the event callback passes the event as first argument and data as second. Try:Running: