gpt4 book ai didi

javascript - 如何将两个父节点连接到一个子节点以及如何务实地为树中的每个节点制作工具提示?在 D3 js (SVG) 中

转载 作者:行者123 更新时间:2023-12-03 07:04:50 24 4
gpt4 key购买 nike

我制作了一个树形图结构,它工作得非常好。但现在我想做一个小改变,但无法做出改变。我想将两个父节点连接到一个子节点......我也试图务实地在每个节点中添加一个工具提示。
例如——如果你运行代码,你会看得更清楚。我想制作一个 Hanna 的子节点并标记它将连接到一个名为“Eric”的子节点。
知道如何实现这一目标吗?

var svg = d3
.select("body")
.append("svg")
.attr("width", 600)
.attr("height", 600)
.append("g")
.attr("transform", "translate(50,50)");

//tree data
var data = [
{ child: "John", parent: "" },
{ child: "Aron", parent: "Kevin" },
{ child: "Kevin", parent: "John" },
{ child: "Hannah", parent: "Anna" },
{ child: "Rose", parent: "Sarah" },
{ child: "Anna", parent: "John" },
{ child: "Sarah", parent: "Kevin" },
{ child: "Mark", parent: "Anna" },
{ child: "Angle", parent: "Sarah" },
];

//to construct
var dataStructure = d3
.stratify()
.id(function (d) {
return d.child;
})
.parentId(function (d) {
return d.parent;
})(data);

//to define the size of the structure tree
var treeStructure = d3.tree().size([500, 300]);
var information = treeStructure(dataStructure);

//to make the connections curves
var connections = svg.append("g").selectAll("path").data(information.links());
connections
.enter()
.append("path")
.attr("d", function (d) {
return (
"M" +
d.source.x +
"," +
d.source.y +
" C " +
d.source.x +
"," +
(d.source.y + d.target.y) / 2 +
" " +
d.target.x +
"," +
(d.source.y + d.target.y) / 2 +
" " +
d.target.x +
"," +
d.target.y
);
});

//creating the circles with data info
var circles = svg
.append("g")
.selectAll("circle")
.data(information.descendants());

//placing the circles
circles
.enter()
.append("circle")
.attr("cx", function (d) {
return d.x;
})
.attr("cy", function (d) {
return d.y;
})
.attr("r", 7)
.append("text");

//names
var names = svg.append("g").selectAll("text").data(information.descendants());
names
.enter()
.append("text")
.text(function (d) {
return d.data.child;
})
.attr("x", function (d) {
return d.x + 7;
})
.attr("y", function (d) {
return d.y + 4;
});
circle {
fill: rgb(88, 147, 0);
}

path {
fill: none;
stroke: black;
}
<script src="https://d3js.org/d3.v6.min.js"></script>

最佳答案

您可以做的最简单的事情就是随机分配一个父节点,然后使用该父节点定位节点,并自己绘制链接。

var svg = d3
.select("body")
.append("svg")
.attr("width", 600)
.attr("height", 600)
.append("g")
.attr("transform", "translate(50,50)");

//tree data
var data = [
{ child: "Alice", parents: [] },
{ child: "Bob", parents: [] },
{ child: "Carol", parents: [] },
{ child: "Dave", parents: ["Alice", "Bob"] },
{ child: "Eve", parents: ["Alice", "Bob"] },
{ child: "Francis", parents: ["Bob", "Carol"] },
{ child: "Graham", parents: ["Carol"] },
{ child: "Hugh", parents: ["Eve", "Graham"] },
];

// Process the nodes, add a pseudo root node so we don't have
// multiple roots
data.forEach(function(d) {
d.parentId = d.parents.length > 0 ? d.parents[0] : "root";
});
data.unshift({ child: "root", parentId: "" });

//to construct
var dataStructure = d3
.stratify()
.id(function (d) {
return d.child;
})
.parentId(function (d) {
return d.parentId;
})(data);

//to define the size of the structure tree
var treeStructure = d3.tree().size([500, 300]);
var root = treeStructure(dataStructure);

var nodes = root.descendants()
.filter(function(d) { return d.id !== "root"; });

// Custom way to get all links we need to draw
var links = [];
nodes.forEach(function(node) {
node.data.parents.forEach(function(parentId) {
var parentNode = nodes.find(function(d) { return d.id === parentId; });
links.push({
source: parentNode,
target: node,
});
});
});

//to make the connections curves
var connections = svg.append("g").selectAll("path").data(links);
connections
.enter()
.append("path")
.attr("d", function (d) {
return (
"M" +
d.source.x +
"," +
d.source.y +
" C " +
d.source.x +
"," +
(d.source.y + d.target.y) / 2 +
" " +
d.target.x +
"," +
(d.source.y + d.target.y) / 2 +
" " +
d.target.x +
"," +
d.target.y
);
});

//creating the circles with data info
var circles = svg
.append("g")
.selectAll("circle")
.data(nodes);

//placing the circles
circles
.enter()
.append("circle")
.attr("cx", function (d) {
return d.x;
})
.attr("cy", function (d) {
return d.y;
})
.attr("r", 7)
.append("text");

//names
var names = svg.append("g").selectAll("text").data(nodes);
names
.enter()
.append("text")
.text(function (d) {
return d.id;
})
.attr("x", function (d) {
return d.x + 7;
})
.attr("y", function (d) {
return d.y + 4;
});
circle {
fill: rgb(88, 147, 0);
}

path {
fill: none;
stroke: black;
}
<script src="https://d3js.org/d3.v6.min.js"></script>

但是,您可以看到这有一些问题。我们希望链接尽可能短,因此我们应该按其父节点对节点进行排序 - 具有相同父节点的子节点应该位于两个父节点之间,如果两个父节点之间不应该有任何节点合伙。
可能最简单的方法是使用节点链接图。通过施加正确的力,您可以将节点推向最佳布局。
我强制链接要短,树要居中在图表的中间,节点要处于正确的级别,每个节点要与任何其他节点相距 100 像素。
我运行了 100 次模拟并在渲染任何内容之前停止它,从而避免了您通常看到的运动和抽搐。就用户体验而言,它看起来像一个普通的树状结构。
我还使用自定义属性 d.dy在节点上作为“所需的 y 值”。这个值是节点在模拟中被吸引的值,但由于它可能会稍微偏离,所以看起来很乱。通过像这样捕捉 y 轴值,我避免了这种情况并使它看起来更有条理。

var levelRadius = 50;
var radius = 10;
var size = 500;
var margin = 50;

var svg = d3
.select("body")
.append("svg")
.attr("width", size + 2 * margin)
.attr("height", size + 2 * margin)
.append("g")
.attr("transform", "translate(" + [margin, margin] + ")");

//tree data
var data = [{
child: "Alice",
parents: []
},
{
child: "Bob",
parents: []
},
{
child: "Carol",
parents: []
},
{
child: "Dave",
parents: ["Alice", "Bob"]
},
{
child: "Eve",
parents: ["Alice", "Bob"]
},
{
child: "Francis",
parents: ["Bob", "Carol"]
},
{
child: "Graham",
parents: ["Carol"]
},
{
child: "Hugh",
parents: ["Eve", "Graham"]
},
];

data.forEach(function(d, i) {
d.level = d.parents.length ?
data.find(function(p) {
return p.child === d.parents[0];
}).level + 1 :
0;

// Desired y level, otherwise try for sensible defaults
d.dy = d.y = d.level * levelRadius * 2;
d.x = size / 2 - i * levelRadius;
});

var nodes = data;

// Custom way to get all links we need to draw
var links = [];
nodes.forEach(function(node) {
node.parents.forEach(function(parentId) {
var parentNode = nodes.find(function(d) {
return d.child === parentId;
});
links.push({
id: parentNode.child + " - " + node.child,
source: parentNode,
target: node,
});
});
});

var simulation = d3.forceSimulation(nodes)
.force("link", d3.forceLink(links).id(d => d.id))
.force("collide", d3.forceCollide(levelRadius))
.force("x", d3.forceX()
.x(function(d) {
return size / 2;
})
.strength(0.2)
)
.force("y", d3.forceY()
.y(function(d) {
return d.dy;
})
.strength(5)
);
// Run the simulation once, even before rendering anything
simulation.tick(100)
.stop();

//to make the connections curves
var link = svg.append("g").selectAll("path")
.data(links)
.enter()
.append("path")
.attr("d", function(d) {
return (
"M" +
d.source.x +
"," +
d.source.dy +
" C " +
d.source.x +
"," +
(d.source.y + d.target.dy) / 2 +
" " +
d.target.x +
"," +
(d.source.y + d.target.dy) / 2 +
" " +
d.target.x +
"," +
d.target.dy
);
});

var node = svg
.selectAll(".node")
.data(nodes)
.enter()
.append("g")
.attr("class", "node")
.attr("transform", function(d) {
// Use d.dy to snap the node to the level that we want it at
return "translate(" + [d.x, d.dy] + ")";
});

node.append("circle")
.attr("r", radius);
node.append("text")
.attr("dominant-baseline", "middle")
.attr("dx", radius + 3)
.text(function(d) {
return d.child;
});
circle {
fill: rgb(88, 147, 0);
}

path {
fill: none;
stroke: black;
}
<script src="https://d3js.org/d3.v6.min.js"></script>

关于javascript - 如何将两个父节点连接到一个子节点以及如何务实地为树中的每个节点制作工具提示?在 D3 js (SVG) 中,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/64587495/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com