d3.js强制有向图:如何使节点大小取决于链接的值?

时间:2017-11-16 20:02:18

标签: javascript d3.js d3-force-directed

我有这样的数据:

var nodes = [
{
    "id": "A",
    "label": "Data A",
    "group": "0",
    "level": "0"
}, {
    "id": "B",
    "label": "Data B",
    "group": "1",
    "level": "1"
},
// etc //
]

var links = [
{
    "source": "A",
    "target": "B",
"strength": "0.5",
    "value": "276"
}, {
    "source": "A",
    "target": "C",
"strength": "0.5",
"value": "866"
},
// etc //
]

我一直在尝试根据value从所有节点作为其target的链接设置节点半径大小。

因此,例如,节点B的大小应该根据276的值(以节点B作为目标的链接)。

我使用的是:

var nodeElements = g.append("g") 
    .attr("class", "nodes")
    .selectAll("circle")
    .data(nodes)
    .enter().append("circle")
    .attr("r", function(node,link){
          var radius = 12;
          if (node.level == 0) radius = radius * 2; // Setting up the main node as bigger than others

          node.weight = link.filter(function(link) {
             if (link.target.index == node.index){
               var linkValue = link.value;
               if (linkValue > 500) ? (radius = 12) : (radius = 6);
             }
          });

          return radius;
      })
    .attr("fill", getNodeColor)
    .attr("stroke", "#fff")
    .attr('stroke-width', 2)
    .on('mouseover', selectNode)

但这似乎并不奏效。说它不识别link.filterI've taken from here。我正在使用d3.js v4。

试图寻找线索,但仍然没有提示。有什么想法吗?

1 个答案:

答案 0 :(得分:2)

有几件事需要更改,但首先您应该了解,您不会自动将links添加到节点的数据中。

所以filter语句的目的似乎是根据索引在该数组中找到相应的链接。 没有link变量传递到外部函数,而是你应该搜索上面定义的links数组,找到一个目标与节点的id的链接。

如果您只需要一个链接,而不是所有链接,请使用Array.prototype.find。但filterfind的工作方式相同 - 定义的匿名函数在数组中的每个项目上调用。 find在第一个找到的对象处停止,而filter则返回所有匹配。

.attr("r", function(dat, index, n) {
    var linkItem = links.find(function(link) {
        return link.target == dat.id;
    });
    var radius = 12;
    // linkItem would be undefined if the item was not found in the links
    if (linkItem && linkItem.value > 500) {
        radius = 12;
    } else {
        radius = 6;
    }

    // you should double the radius towards the end,
    // so that radius doesn't get overwritten
    if (dat.level === 0) {
        radius = 2 * radius;
    }

    return radius;
})