Need To Connect Two Nodes Of Different Circle Packed Layout In D3 And Pack Layout Collapse/expand
I want to connect node inside one big circle to node inside another big circle or sometimes to another bigger circle itself. Is there a way to achieve the same ? I am able to conne
Solution 1:
Here is an example with collapsing/expandind nodes. The sizes and margins should be adjusted according to your requirements. Suggest to see the snippet in a full-page mode:
const data = {
name: "root",
children: [
{
name: "A",
children: [
{name: "A1", value: 7}, {name: "A2", value: 8}, {name: "A3", value: 9}, {name: "A4", value: 10}, {name: "A5", value: 10}
]
},
{
name: "B",
children: [
{name: "B1", value: 11}, {name: "B2", value: 7}, {name: "B3", value: 8},
]
},
{
name: "C",
value: 10
},
{
name: "D",
value: 10
},
{
name: "E",
value: 10
}
],
links: [{from: "A3", to: "C"}, {from: "A2", to: "E"}, {from: "B1", to: "D"}, {from: "B2", to: "B3"}, {from: "B1", to: "C"}]
};
constfindNode = (parent, name) => {
if (parent.name === name)
return parent;
if (parent.children) {
for (let child of parent.children) {
const found = findNode(child, name);
if (found) {
return found;
}
}
}
returnnull;
}
const svg = d3.select("svg");
const container = svg.append('g')
.attr('transform', 'translate(0,0)')
constonClickNode = (e, d) => {
e.stopPropagation();
e.preventDefault();
const node = findNode(data, d.data.name);
if(node.children && !node._children) {
node._children = node.children;
node.children = undefined;
node.value = 20;
updateGraph(data);
} else {
if (node._children && !node.children) {
node.children = node._children;
node._children = undefined;
node.value = undefined;
updateGraph(data);
}
}
}
constupdateGraph = graphData => {
constpack = data => d3.pack()
.size([600, 600])
.padding(0)
(d3.hierarchy(data)
.sum(d => d.value * 3.5)
.sort((a, b) => b.value - a.value));
const root = pack(graphData);
const nodes = root.descendants().slice(1);
console.log('NODES: ', nodes);
const nodeElements = container
.selectAll("g.node")
.data(nodes, d => d.data.name);
const addedNodes = nodeElements.enter()
.append("g")
.classed('node', true)
.style('cursor', 'pointer')
.on('click', (e, d) =>onClickNode(e, d));
addedNodes.append('circle')
.attr('stroke', 'black')
addedNodes.append("text")
.text(d => d.data.name)
.attr('text-anchor', 'middle')
.attr('alignment-baseline', 'middle')
.style('visibility', 'hidden')
.style('fill', 'black');
const mergedNodes = addedNodes.merge(nodeElements);
mergedNodes
.transition()
.duration(500)
.attr('transform', d =>`translate(${d.x},${d.y})`);
mergedNodes.select('circle')
.attr("fill", d => d.children ? "#ffe0e0" : "#ffefef")
.transition()
.duration(1000)
.attr('r', d => d.value)
mergedNodes.select('text')
.attr('dy', d => d.children ? d.value + 10 : 0)
.transition()
.delay(1000)
.style('visibility', 'visible')
const exitedNodes = nodeElements.exit()
exitedNodes.select('circle')
.transition()
.duration(500)
.attr('r', 1);
exitedNodes.select('text')
.remove();
exitedNodes
.transition()
.duration(750)
.remove();
constlinkPath = d => {
constfrom = nodes.find(n => n.data.name === d.from);
const to = nodes.find(n => n.data.name === d.to);
if (!from || !to)
returnnull;
const length = Math.hypot(from.x - to.x, from.y - to.y);
const fd = from.value / length;
const fx = from.x + (to.x - from.x) * fd;
const fy = from.y + (to.y - from.y) * fd;
const td = to.value / length;
const tx = to.x + (from.x - to.x) * td;
const ty = to.y + (from.y - to.y) * td;
return`M ${fx},${fy} L ${tx},${ty}`;
};
const linkElements = container.selectAll('path.link')
.data(data.links.filter(linkPath));
const addedLinks = linkElements.enter()
.append('path')
.classed('link', true)
.attr('marker-end', 'url(#arrowhead-to)')
.attr('marker-start', 'url(#arrowhead-from)');
addedLinks.merge(linkElements)
.transition()
.delay(750)
.attr('d', linkPath)
linkElements.exit().remove();
}
updateGraph(data);
text {
font-family: "Ubuntu";
font-size: 12px;
}
.link {
stroke: blue;
fill: none;
}
<scriptsrc="https://cdnjs.cloudflare.com/ajax/libs/d3/6.7.0/d3.min.js"></script><svgwidth="600"height="600"><defs><markerid="arrowhead-to"markerWidth="10"markerHeight="7"refX="10"refY="3.5"orient="auto"><polygonfill="blue"points="0 0, 10 3.5, 0 7" /></marker><markerid="arrowhead-from"markerWidth="10"markerHeight="7"refX="0"refY="3.5"orient="auto"><polygonfill="blue"points="10 0, 0 3.5, 10 7" /></marker></defs></svg>
Post a Comment for "Need To Connect Two Nodes Of Different Circle Packed Layout In D3 And Pack Layout Collapse/expand"