d3:对象的子数组

时间:2013-12-10 22:36:49

标签: javascript d3.js

我有以下结构:

[
    { 'length': 10, attributes: [1,2,3] },
    { 'length': 7, attributes: [1,3,4,5] },
    { 'length': 12, attributes: [3,5,7,9,10] },
]

 and I am doing the following:


x = d3.scale.linear().domain([0, maxHeight]).range([50, w]),
y = d3.scale.linear().domain([0, maxHeight]).range([h, 20]);
z = d3.scale.linear().domain([0, maxHeight]).range([0, h - 20]);

var chart = svg.selectAll("g.chart")
    .data(items)
    .enter()
    .append("svg:g")
    .attr("class", "chart");

chart.append("svg:rect")
    .attr("fill", 'darkblue')
    .attr("class", 'data')
    .attr("x", function(d, i) { return x(i+1); })
    .attr("y", function(d, i) { return bottom - z(d['length']) + 15 })
    .attr("width", 4)
    .attr("height", function(d, i)  { return z(d['length']) - z(d['min']); })

我想要做的是在每个矩形上添加圆圈,这些圆圈对应于我的结构中的属性。基本上,(对于一个'item'},我应该看到这样的东西:

<g class="chart">
    <rect fill="darkblue" class="data" x="626.1538461538462" y="15" width="6" height="530"></rect>
    <circle cx="626.1538461538462" cy="(y1)" r="5" style="fill: #ffff00; stroke: #808080;"></circle>
    <circle cx="626.1538461538462" cy="(y2)" r="5" style="fill: #ffff00; stroke: #808080;"></circle>
    <circle cx="626.1538461538462" cy="(y3)" r="5" style="fill: #ffff00; stroke: #808080;"></circle>
</g>

我唯一能想到的是循环遍历属性并逐个元素地添加它们:

for (z=0; z< 3; ++z)
{
    chart.append("svg:circle")
    .data(items[z]['attributes'])
    .style("fill", 'yellow')
    .style("stroke", "gray")
    .attr("cx", function(d, i) { return x(i+1); })
    .attr("cy", function(d, i) 
    { 
        console.log(d); 
        return bottom - 15;
    })
    .attr("r", 5);
}

有更好的方法吗?

2 个答案:

答案 0 :(得分:6)

您可以创建嵌套选择而不是循环:

chart.selectAll("svg:circle")
    .data(function(item) { return item.attributes; })
  .enter()
    .append("svg:circle")
    .style("fill", 'yellow')
    .style("stroke", "gray")
    .attr("cx", function(d, i) { return x(i+1); })
    .attr("cy", function(d, i) 
    { 
        console.log(d); 
        return bottom - 15;
    })
    .attr("r", 5);

示例

要使每个cx parent的{​​{1}}保持一致,您可以通过rect通过

parent_idx

答案 1 :(得分:3)

您可以使用嵌套选择。主要选择将创建组,每个组将有一个绑定到它的数据项。

var data = [
    {name: 'A', items: [1, 2]},
    {name: 'B', items: [2, 3, 4]}
];

var cScale = d3.scale.category10()
    .domain(d3.range(10));

var grp = svg.selectAll('g.main')
    .data(data)
    .enter()
    .append('g')
    .attr('class', 'main')
    .attr('transform', function(d, i) {
        return 'translate(0,' + i * 20 + ')';
    });

然后,您可以创建嵌套选择,将访问器函数传递给数据方法。我有一个带有rect元素的例子,但是圆圈是相同的:

grp.selectAll('rect')
    .data(function(d) { return d.items; })
    .enter()
    .append('rect')
    .attr('x', function(d) { return 10 * d; })
    .attr('width', 8)
    .attr('height', 10)
    .attr('fill', function(d) { return cScale(d); });

您可能会发现文章Nested Selections很有用。我也写了一个小jsfiddle:http://jsfiddle.net/pnavarrc/h2YVd/