我正在创建一个很长的div,其中包含通过以下方法创建的数百条svg行:
function visualizeit(ORFdata,max) {
var browser = d3.select("#viewer")
.append("svg")
.attr("width", max/10)
.attr("height",'50%');
//Add svg to the svg container
for (orf in ORFdata) {
var line = browser.append("svg:line");
var object = ORFdata[orf]
line.datum(object)
line.attr("id", 'mygroup'+orf)
line.attr("x1", function(d){ return ORFdata[orf]["start"]/10})
line.attr("x2", function(d){ return ORFdata[orf]["stop"]/10})
line.attr("y1", function(d){ if (ORFdata[orf]["strand"] == "+1") {return 50} else {return 10}})
line.attr("y2", function(d){ if (ORFdata[orf]["strand"] == "+1") {return 50} else {return 10}})
line.style("stroke", "rgb(6,120,155)")
line.style("stroke-width", orf)
line.on('mouseover', function(d){console.log(d3.select("#mygroup"+orf).datum())})
}
}
但是,当我在任何行上进行鼠标悬停时,我只会从最后一个元素取回数据。起初我以为这是由于“ mygroup”造成的,所以我在它的+ orf上添加了一个计数器,但是它仍然以某种方式擦除了我以前存储的数据。
当我查看创建的html代码时,至少通过ID看svg是正确的。
<line id="mygroup50" x1="103356.7" x2="103231.1" y1="10" y2="10" style="stroke: #06789b; stroke-width: 50px;"></line>
但是在某处链接完全出错了...
我到目前为止如何解决...
var svgContainer = d3.select("body").append("svg")
.attr("width", max/10)
.attr("height", '50%');
//Add svg to the svg container
var lines = svgContainer.selectAll("line")
.data(ORFdata)
.enter()
.append("line")
.attr("x1", function(d){ return d.start/10})
.attr("y1", function(d){ if (d.strand == "+1") {return 65} else {return 10}})
.attr("x2", function(d){ return d.stop/10})
.attr("y2", function(d){ if (d.strand == "+1") {return 65} else {return 10}})
.attr("stroke-width","25")
.attr("stroke",function(d) {if (d.strand == "+1") {return 'green'} else {return 'red'}})
.on('mouseover', function(d) {console.log(d.start)})
}
最佳答案
您正在循环中创建一堆闭包。您创建的每个函数在其关闭范围内都具有变量orf
,但是您的循环正在更改orf
的值。当鼠标悬停事件触发时该函数运行时,orf
具有其最终值,因此,您选择的#mygroup + orf
将始终拾取最后一个元素。
这是一个关于闭包的好页面,其中有一段详细介绍了循环中闭包的陷阱:http://conceptf1.blogspot.ca/2013/11/javascript-closures.html。
在D3中,您可以通过使用数据联接而不是外部循环来解决此问题。这是一个很好的教程,应该有助于理解其工作原理:
http://bost.ocks.org/mike/join/