D3强制布局:折叠子节点子集(不透明度变化)

时间:2016-08-17 14:07:03

标签: javascript jquery d3.js force-layout

我正在使用D3力布局并尝试折叠子节点,同时保持图形固定,例如只需更改它们及其链接的不透明度即可。但是,我并不想一次性崩溃所有节点 - 实际上每个节点都有一个名为“group”的属性,可以是1,2或3.

当我点击某个节点时,会出现一个工具提示,其中每个组都有3个按钮 - 点击正确的按钮后,我希望该类型的子节点崩溃,但所有子节点(所有3个)小组也崩溃了。

这是fiddle

到目前为止,我尝试创建节点组ID和相应链接ID的数组,然后使用JQuery隐藏这些DOM对象,但这不起作用:

function collapseNodes(node, collapseType, isCollapsed) {
    var collapseData = minimise(node, collapseType);
    var cNodes = collapseData.nodes,
        cLinks = collapseData.links;
    console.log(collapseData);
    var newClass = isCollapsed ? "uncollapsed" : "collapsed";

    cNodes.forEach(function(n) {
        d3.select(n).style("opacity", 0);
    });

    cLinks.forEach(function(l) {
        d3.select(l).style("opacity", 0);
    });
}

function minimise(node, assetMinType = "") {
    // Function to minimise a node
    var minNodes = [];
    var minLinks = [];

    if (node.group == 'asset') {
        node.children.forEach(function(child) {
            if (child.group == assetMinType)
                minimiseRec(child, node.id);
        });
    }
    else {
        minimiseRec(node, "");

        // We want to keep the top node and link
        minNodes.shift();
        minLinks.shift();

    }

    function minimiseRec(node, parentID) {
        minNodes.push("#" + node.id);
        minLinks.push("#parent_" + parentID + "_child_" + node.address);

        node.children.map(function(child) {
            minimise(child, node.id);
        });
    }

    return { nodes: minNodes, links: minLinks };
}

有谁知道如何做到最好?

感谢。

1 个答案:

答案 0 :(得分:1)

如果只想改变它的不透明度,那么这很简单。我想如果你隐藏一个节点的孩子你想隐藏他们的孩子的孩子等等?

首先我设置一个变量d.hiddenNode。这样我可以切换不透明度。我把它设置为假。然后点击功能:

.on('click', function(d) {
    console.log(d);
    if(d.hiddenNode){
        hideChildren(d, false);
      d.hiddenNode = false;
    } else {
        hideChildren(d, true);
      d.hiddenNode = true;
    }

  })

现在因为你想隐藏孩子的孩子等,你需要一个像这样的递归函数。一个隐藏链接和节点(我已经评论解释):

function hideChildren(node, hide) {

  for (var i = 0; i < node.children.length; i++) {
    recurseChildren(node.children[i]); //loop through children to hide
  }

  function recurseChildren(node) {

    nodeEnter.each(function(d) { //go through all nodes to check for children
      if (d.index == node.index) { //if child is found
        d3.select(this).style('opacity', function(){ 
        return hide ? 0 : 1;        //toggle opacity depending on d.hiddenNode value
        }) //.remove(); 
      }
    })

    link.each(function(d) { //same with links
      if (d.source.index == node.index || d.target.index == node.index ) { //if source or target are hidden hide the link
        d3.select(this).style('opacity', function(){
        return hide ? 0 : 1;        
        }) //.remove(); 
      }
    })   

    if (node.children) { //if this node has children, call again but with their children
      for (var i = 0; i < node.children.length; i++) {
        recurseChildren(node.children[i]);
      }
    }
  }
}

这是更新的小提琴:http://jsfiddle.net/thatOneGuy/3g4fqfa8/2/

修改

对于弹出窗口,我刚刚创建了一个包含3组1,2,3:

的div
<div id='groupChoice'>
<div id='group1' class = 'group' onclick='hideGroup(1)'>1</div>
<div id='group2' class = 'group' onclick='hideGroup(2)'>2</div>
<div id='group3' class = 'group' onclick='hideGroup(3)'>3</div>
</div>

将其设置为隐藏在CSS中:

.group{
  width: 50px;
  height:50px;
  border : 1px solid black;
}

#groupChoice{
  visibility:hidden;
}

我稍微改变了逻辑,但这是评论的更新小提琴:http://jsfiddle.net/thatOneGuy/3g4fqfa8/3/

需要通过链接完成一些工作。我没有足够的时间来完成它,对不起。但这应该让你开始。

基本上,当您点击某个节点时,弹出窗口,您选择了一个要隐藏的组,它们会被隐藏。再次点击同一节点,选择要显示的节点。这是非常快速的,所以它有很多错误,但基础是它应该有帮助:)