javascript - 过滤掉节点并将其置于非事件状态

标签 javascript d3.js nodes force-layout

我正在使用 d3.js 和强制布局。现在,根据这个问题的答案: Change the collision behavior of many nodes stored in an array

通过点击“点击我”按钮,可以过滤掉红色节点并推开其他节点(增加碰撞属性)。它工作得很好,但现在我希望蓝色节点不受红色节点推开 Action 的影响。这意味着,绿色和黄色节点被推开,但蓝色节点保留在原始位置。所以我正在寻找一种方法来使这些蓝色节点在力布局中处于非事件状态。也许你们中的一些人可以提供帮助。谢谢大家!

var svg = d3.select("svg");

var colours = ["blue", "red", "green", "yellow"];

var data = d3.range(30).map(d => ({
    r: 6
}));

var simulation = d3.forceSimulation(data)
    .force("x", d3.forceX(150).strength(0.05))
    .force("y", d3.forceY(75).strength(0.05))
    .force("collide", d3.forceCollide(function(d) {
        return d.r + 1;
    }));

var node = svg.selectAll(".circles")
    .data(data)
    .enter()
    .append("circle")
    .attr("r", d => d.r)
    .attr("fill", (d, i) => colours[i%4]);

d3.select("button").on("click", function(d) {
		node.filter(function(){
		 return d3.select(this).attr("fill") === "red"
		}).each(d=>d.r = 40);
    simulation.nodes(data);
    simulation.alpha(0.8).restart();
})

simulation.nodes(data)
    .on("tick", d => {
        node.attr("cx", d => d.x).attr("cy", d => d.y);
    });
<script src="https://d3js.org/d3.v4.min.js"></script>
<button>Click me</button>
<br>
<svg></svg>

最佳答案

执行此操作的一种方法是检查模拟是否是第一次运行,如果不是,则过滤掉蓝色圆圈:

simulation.nodes(data)
    .on("tick", d => {
        if (firstTime) {
            node.attr("cx", d => d.x).attr("cy", d => d.y);
        } else {
            node.filter(function(e) {
                return d3.select(this).attr("fill") != "blue"
            }).attr("cx", d => d.x).attr("cy", d => d.y);
        }
    });

这是演示:

var svg = d3.select("svg");

var firstTime = true;

var colours = ["blue", "red", "green", "yellow"];

var data = d3.range(30).map(d => ({
  r: 6
}));

var simulation = d3.forceSimulation(data)
  .force("x", d3.forceX(150).strength(0.05))
  .force("y", d3.forceY(75).strength(0.05))
  .force("collide", d3.forceCollide(function(d) {
    return d.r + 1;
  }));

var node = svg.selectAll(".circles")
  .data(data)
  .enter()
  .append("circle")
  .attr("r", d => d.r)
  .attr("fill", (d, i) => colours[i % 4]);

d3.select("button").on("click", function(d) {
  firstTime = false;
  node.filter(function() {
    return d3.select(this).attr("fill") === "red"
  }).each(d => d.r = 40);
  simulation.nodes(data);
  simulation.alpha(0.8).restart();
})

simulation.nodes(data)
  .on("tick", d => {
    if (firstTime) {
      node.attr("cx", d => d.x).attr("cy", d => d.y);
    } else {
      node.filter(function(e) {
        return d3.select(this).attr("fill") != "blue"
      }).attr("cx", d => d.x).attr("cy", d => d.y);
    }
  });
<script src="https://d3js.org/d3.v4.min.js"></script>
<button>Click me</button>
<br>
<svg></svg>

顺便说一句,您想要的结果(“绿色和黄色节点被推开,但蓝色节点保持在原始位置”)在视觉上令人不快。

关于javascript - 过滤掉节点并将其置于非事件状态,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44731185/

相关文章:

javascript - D3.js Focus+Context via Brushing Barchart - 重新计算刷牙时的 BarWidth

javascript - React/Redux,重新加载路由中的组件

javascript - 为什么这段代码可以计算列表中的一个项目而不是其他项目?

javascript - 两个ES6类之间的实例交换

javascript - dc.js 行图默认 View 已排序并且单击时不会重新排序

java - 我应该如何使用自定义链表实现排序?

javascript - Promise 保持挂起状态, "then"或 "catch"未触发。响应式JS

swift - 如何从 scenekit 场景中删除特定节点?

javascript - 单击事件中的确认事件返回错误

php - 添加斜杠可以安全地防止数组中的 xss?