html 如何在D3.JS中更改树图中节点之间的线宽

3lxsmp7m  于 2023-05-05  发布在  其他
关注(0)|答案(1)|浏览(176)

我试图通过D3.JS的树图(特别是Mike Bostock的可折叠树图链接here的版本)Map调查数据二进制问题。
这里的想法是,我希望节点之间的线条宽度根据回答该问题答案的调查受访者的数量进行缩放(例如:父节点=问题1,并且“是”响应等于子节点1,“否”响应等于子节点2,并且父节点和子节点之间的线的宽度等于选择该响应的人口的比例)。
有什么建议吗?让我知道这是否有意义--请参阅下面的示例数据结构和基本树。
谢谢!
初始数据结构如下所示:
{ "name": "Question", "children": [ { "name": "Yes", }, { "name": "No" } ] };
Basic tree looks like this

kcugc4gi

kcugc4gi1#

下面演示了如何调整树图示例的笔划宽度,应该会让你朝着正确的方向开始。
根据代码示例的'styling version',您可以向数据添加笔划宽度(这里称为厚度)的质量;

  1. var treeData =
  2. {
  3. "name": "Top Level",
  4. "value": 10,
  5. "type": "black",
  6. "level": "red",
  7. "thickness": 2,
  8. "children": [
  9. {
  10. "name": "Level 2: A",
  11. "value": 15,
  12. "type": "grey",
  13. "level": "red",
  14. "thickness": 4,
  15. "children": [
  16. {
  17. "name": "Son of A",
  18. "value": 5,
  19. "type": "steelblue",
  20. "thickness": 6,
  21. "level": "orange"
  22. },
  23. {
  24. "name": "Daughter of A",
  25. "value": 8,
  26. "type": "steelblue",
  27. "thickness": 1,
  28. "level": "red"
  29. }
  30. ]
  31. },
  32. {
  33. "name": "Level 2: B",
  34. "value": 10,
  35. "type": "grey",
  36. "thickness": 5,
  37. "level": "green"
  38. }
  39. ]
  40. };

然后在绘制链接时调用它;

  1. var link = g.selectAll(".link")
  2. .data( nodes.descendants().slice(1))
  3. .enter().append("path")
  4. .attr("class", "link")
  5. .style("stroke", function(d) { return d.data.level; })
  6. .style("stroke-width", function(d) { return d.data.thickness; })
  7. .attr("d", function(d) {
  8. return "M" + d.y + "," + d.x
  9. + "C" + (d.y + d.parent.y) / 2 + "," + d.x
  10. + " " + (d.y + d.parent.y) / 2 + "," + d.parent.x
  11. + " " + d.parent.y + "," + d.parent.x;
  12. });

请记住,我已经重复了d3.js v4的例子,你可能会考虑迁移到v7(updated example of the styling code)。

**更新:**这里是交互式示例的完整代码,具有不同的链接厚度

  1. <!DOCTYPE html>
  2. <meta charset="UTF-8">
  3. <style>
  4. .node circle {
  5. fill: #fff;
  6. stroke: steelblue;
  7. stroke-width: 3px;
  8. }
  9. .node text {
  10. font: 12px sans-serif;
  11. }
  12. .link {
  13. fill: none;
  14. stroke: #ccc;
  15. stroke-width: 2px;
  16. }
  17. </style>
  18. <body>
  19. <!-- load the d3.js library -->
  20. <script src="https://d3js.org/d3.v4.min.js"></script>
  21. <script>
  22. var treeData =
  23. {
  24. "name": "Top Level",
  25. "thickness": 4,
  26. "children": [
  27. {
  28. "name": "Level 2: A",
  29. "thickness": 3,
  30. "children": [
  31. {
  32. "name": "Son of A",
  33. "thickness": 2
  34. },
  35. {
  36. "name": "Daughter of A",
  37. "thickness": 1
  38. }
  39. ]
  40. },
  41. {
  42. "name": "Level 2: B",
  43. "thickness": 6
  44. }
  45. ]
  46. };
  47. // Set the dimensions and margins of the diagram
  48. var margin = {top: 20, right: 90, bottom: 30, left: 90},
  49. width = 960 - margin.left - margin.right,
  50. height = 500 - margin.top - margin.bottom;
  51. // append the svg object to the body of the page
  52. // appends a 'group' element to 'svg'
  53. // moves the 'group' element to the top left margin
  54. var svg = d3.select("body").append("svg")
  55. .attr("width", width + margin.right + margin.left)
  56. .attr("height", height + margin.top + margin.bottom)
  57. .append("g")
  58. .attr("transform", "translate("
  59. + margin.left + "," + margin.top + ")");
  60. var i = 0,
  61. duration = 750,
  62. root;
  63. // declares a tree layout and assigns the size
  64. var treemap = d3.tree().size([height, width]);
  65. // Assigns parent, children, height, depth
  66. root = d3.hierarchy(treeData, function(d) { return d.children; });
  67. root.x0 = height / 2;
  68. root.y0 = 0;
  69. // Collapse after the second level
  70. root.children.forEach(collapse);
  71. update(root);
  72. // Collapse the node and all it's children
  73. function collapse(d) {
  74. if(d.children) {
  75. d._children = d.children
  76. d._children.forEach(collapse)
  77. d.children = null
  78. }
  79. }
  80. function update(source) {
  81. // Assigns the x and y position for the nodes
  82. var treeData = treemap(root);
  83. // Compute the new tree layout.
  84. var nodes = treeData.descendants(),
  85. links = treeData.descendants().slice(1);
  86. // Normalize for fixed-depth.
  87. nodes.forEach(function(d){ d.y = d.depth * 180});
  88. // ****************** Nodes section ***************************
  89. // Update the nodes...
  90. var node = svg.selectAll('g.node')
  91. .data(nodes, function(d) {return d.id || (d.id = ++i); });
  92. // Enter any new modes at the parent's previous position.
  93. var nodeEnter = node.enter().append('g')
  94. .attr('class', 'node')
  95. .attr("transform", function(d) {
  96. return "translate(" + source.y0 + "," + source.x0 + ")";
  97. })
  98. .on('click', click);
  99. // Add Circle for the nodes
  100. nodeEnter.append('circle')
  101. .attr('class', 'node')
  102. .attr('r', 1e-6)
  103. .style("fill", function(d) {
  104. return d._children ? "lightsteelblue" : "#fff";
  105. });
  106. // Add labels for the nodes
  107. nodeEnter.append('text')
  108. .attr("dy", ".35em")
  109. .attr("x", function(d) {
  110. return d.children || d._children ? -13 : 13;
  111. })
  112. .attr("text-anchor", function(d) {
  113. return d.children || d._children ? "end" : "start";
  114. })
  115. .text(function(d) { return d.data.name; });
  116. // UPDATE
  117. var nodeUpdate = nodeEnter.merge(node);
  118. // Transition to the proper position for the node
  119. nodeUpdate.transition()
  120. .duration(duration)
  121. .attr("transform", function(d) {
  122. return "translate(" + d.y + "," + d.x + ")";
  123. });
  124. // Update the node attributes and style
  125. nodeUpdate.select('circle.node')
  126. .attr('r', 10)
  127. .style("fill", function(d) {
  128. return d._children ? "lightsteelblue" : "#fff";
  129. })
  130. .attr('cursor', 'pointer');
  131. // Remove any exiting nodes
  132. var nodeExit = node.exit().transition()
  133. .duration(duration)
  134. .attr("transform", function(d) {
  135. return "translate(" + source.y + "," + source.x + ")";
  136. })
  137. .remove();
  138. // On exit reduce the node circles size to 0
  139. nodeExit.select('circle')
  140. .attr('r', 1e-6);
  141. // On exit reduce the opacity of text labels
  142. nodeExit.select('text')
  143. .style('fill-opacity', 1e-6);
  144. // ****************** links section ***************************
  145. // Update the links...
  146. var link = svg.selectAll('path.link')
  147. .data(links, function(d) { return d.id; });
  148. // Enter any new links at the parent's previous position.
  149. var linkEnter = link.enter().insert('path', "g")
  150. .attr("class", "link")
  151. .style("stroke-width", function(d) { return d.data.thickness; })
  152. .attr('d', function(d){
  153. var o = {x: source.x0, y: source.y0}
  154. return diagonal(o, o)
  155. });
  156. // UPDATE
  157. var linkUpdate = linkEnter.merge(link);
  158. // Transition back to the parent element position
  159. linkUpdate.transition()
  160. .duration(duration)
  161. .attr('d', function(d){ return diagonal(d, d.parent) });
  162. // Remove any exiting links
  163. var linkExit = link.exit().transition()
  164. .duration(duration)
  165. .attr('d', function(d) {
  166. var o = {x: source.x, y: source.y}
  167. return diagonal(o, o)
  168. })
  169. .remove();
  170. // Store the old positions for transition.
  171. nodes.forEach(function(d){
  172. d.x0 = d.x;
  173. d.y0 = d.y;
  174. });
  175. // Creates a curved (diagonal) path from parent to the child nodes
  176. function diagonal(s, d) {
  177. path = `M ${s.y} ${s.x}
  178. C ${(s.y + d.y) / 2} ${s.x},
  179. ${(s.y + d.y) / 2} ${d.x},
  180. ${d.y} ${d.x}`
  181. return path
  182. }
  183. // Toggle children on click.
  184. function click(d) {
  185. if (d.children) {
  186. d._children = d.children;
  187. d.children = null;
  188. } else {
  189. d.children = d._children;
  190. d._children = null;
  191. }
  192. update(d);
  193. }
  194. }
  195. </script>
  196. </body>
展开查看全部

相关问题