强制布局 - 标记并固定节点

时间:2016-01-12 12:57:30

标签: javascript d3.js force-layout

我正在基于d3js构建自己的力量拖拽。我关注的是很好的例子site,它几​​乎包含了我所需要的一切:

我遇到了这个问题,我恐怕无法解决,因为我对d3j没有足够的了解。

我正在尝试结合两个功能: - 拖动节点,它们会停留在释放它们的位置 - 所有节点都有一个名称,所以当我拖动它们时,名称也跟随节点。

我的问题是,当图表加载时,它会正确显示节点及其各自的名称。但是当我拖动它们时,文本仍保留在屏幕上。

这是标签的示例代码,完整的jsfiddle是here

进行标记的代码示例:

var node = svg.selectAll(".node")
.data(graph.nodes)
.enter().append("g")
.attr("class", "node")
.call(force.drag);
node.append("circle")
.attr("r", 8)
.style("fill", function (d) {
return color(d.group);
})
node.append("text")
  .attr("dx", 10)
  .attr("dy", ".35em")
  .text(function(d) { return d.name })
  .style("stroke", "gray");
force.on("tick", function () {
link.attr("x1", function (d) {
    return d.source.x;
})
    .attr("y1", function (d) {
    return d.source.y;
})
    .attr("x2", function (d) {
    return d.target.x;
})
    .attr("y2", function (d) {
    return d.target.y;
});   

d3.selectAll("circle").attr("cx", function (d) {
    return d.x;
})
    .attr("cy", function (d) {
    return d.y;
});

d3.selectAll("text").attr("x", function (d) {
    return d.x;
})
    .attr("y", function (d) {
    return d.y;
});

这是执行固定的示例代码,完整的jsfiddle是here

var node = svg.selectAll(".node")
.data(graph.nodes)
.enter().append("circle")
.attr("class", "node")
.attr("r", 8)
.style("fill", function (d) {
return color(d.group);
})
.on('dblclick', releasenode)
.call(node_drag);
 var node_drag = d3.behavior.drag()
    .on("dragstart", dragstart)
    .on("drag", dragmove)
    .on("dragend", dragend);

function dragstart(d, i) {
   force.stop() //stop the force auto positioning before you start dragging
}

function dragmove(d, i) {
    d.px += d3.event.dx;
    d.py += d3.event.dy;
    d.x += d3.event.dx;
    d.y += d3.event.dy; 
}

function dragend(d, i) {
    d.fixed = true; // 
    force.resume();
}

function releasenode(d) {
    d.fixed = false; // 
    //force.resume();
}

我做了什么我结合了两个代码示例。节点最初是正确渲染的,但是当我尝试拖动它们以便它们被固定时,实验室会保持原始位置并且不会更新(不遵循节点)。 这是我初始化这两个的地方:

var node = svg.selectAll(".node")
.data(graph.nodes)
.enter().append("g")
.attr("class", "node")    
.style("fill", function (d) {
return color(d.group);
 })
.call(force.drag) //-- HERE -- Take care of labels  
.on('dblclick', connectedNodes) //Added code 
.on('mouseover', tip.show) //Tool tip show
.on('mouseout', tip.hide) //Tool tip remove 
//.on('click', navigateToPage)//Add event listener to open another Form
//.call(node_drag); //-- HERE --Drag Nodes to position

node.append("circle")
.attr("r", 8)
.style("fill", function (d) {

return color(d.group);
})

代码工作正常,如果我使用其中之一。但结合起来时,表现不正常。我怎样才能激活它们?如果htere是另一种固定节点的方法,那就太好了。

2 个答案:

答案 0 :(得分:2)

我认为这是一个更好的解决方案:http://jsfiddle.net/bzas8j5z/

var dragstart = function(d) {
    d.fixed = true;
};

var drag = force.drag().
    on("dragstart", dragstart);

var node = svg.selectAll(".node")
    .data(graph.nodes)
    .enter().append("g")
    .attr("class", "node")
    .call(drag);

参考此处:https://gist.github.com/mbostock/3750558

答案 1 :(得分:0)

这不能成为!!!

我用这段代码解决了这个问题:

node.on("mousedown", function(d) { d.fixed = true; });

希望其他人也能从这个解决方案中受益。