D3:将数据值标签添加到多线图

时间:2015-02-18 17:02:35

标签: javascript d3.js

我正在使用这个多线图,但到目前为止还没有在每个滴答(每天)生成数据值标签。

<script>

var margin = {top: 30, right: 40, bottom: 30, left: 50},
    width = 600 - margin.left - margin.right,
    height = 270 - margin.top - margin.bottom;

var parseDate = d3.time.format("%d-%m-%y").parse;

var x = d3.time.scale().range([0, width]);
var y = d3.scale.linear().range([height, 0]);

var xAxis = d3.svg.axis().scale(x)
    .orient("bottom").ticks(7);

var yAxis = d3.svg.axis().scale(y)
    .orient("left").ticks(7);

var valueline = d3.svg.line()
    .x(function(d) { return x(d.date); })
    .y(function(d) { return y(d.sev3); });

var valueline2 = d3.svg.line()
    .x(function(d) { return x(d.date); })
    .y(function(d) { return y(d.sev4); });

var svg = d3.select("body")
    .append("svg")
        .attr("width", width + margin.left + margin.right)
        .attr("height", height + margin.top + margin.bottom)
    .append("g")
        .attr("transform", "translate(" + margin.left + "," + margin.top + ")");

// Get the data
d3.csv("data.tsv", function(error, data) {
    data.forEach(function(d) {
        d.date = parseDate(d.date);
        d.sev3 = +d.sev3;
        d.sev4 = +d.sev4;
    });

    // Scale the range of the data
    x.domain(d3.extent(data, function(d) { return d.date; }));
    y.domain([0, d3.max(data, function(d) { return Math.max(d.sev3, d.sev4); })]);

    svg.append("path")      // Add the valueline path.
        .attr("class", "line")
        .attr("d", valueline(data));

    svg.append("path")      // Add the valueline2 path.
        .attr("class", "line")
        .style("stroke", "red")
        .attr("d", valueline2(data));



    svg.append("g")         // Add the X Axis
        .attr("class", "x axis")
        .attr("transform", "translate(0," + height + ")")
        .call(xAxis);

    svg.append("g")         // Add the Y Axis
        .attr("class", "y axis")
        .call(yAxis);

    svg.append("text")
        .attr("transform", "translate(" + (width+3) + "," + y(data[0].sev4) + ")")
        .attr("dy", ".35em")
        .attr("text-anchor", "start")
        .style("fill", "red")
        .text("Sev4");

    svg.append("text")
        .attr("transform", "translate(" + (width+3) + "," + y(data[0].sev3) + ")")
        .attr("dy", ".35em")
        .attr("text-anchor", "start")
        .style("fill", "steelblue")
        .text("Sev3");



});

</script>

Data.tsv

date,sev3,sev4
20-02-15,0,0
19-02-15,0,0
18-02-15,0,0
17-02-15,481,200
16-02-15,691,200
15-02-15,296,200
14-02-15,307,200

上面的代码给出了这样的: with current code

这就是我想要完成的事情What i want

我知道我必须使用.append(&#34; text&#34;)并将文本定位在与数据点相同的x,y坐标上,然后从数据中提取值以输入&# 34;文本&#34;但我在整合这个概念时遇到了困难。

我怀疑使用valueline.append进行选择?我已经看了一些HEAP的例子,我没有一个带有数据值标签的linegraph存在,如果确实如此,请指出我:)

有什么想法吗?

1 个答案:

答案 0 :(得分:4)

您的文字将不可见,因为它位于您的svg边界之外:您添加了一个由margin.left翻译的组,并将您的x放在宽度+3,这意味着位于宽度+3你的svg左边框的+ margin.left。

尝试使用以下内容替换您的附加文字:

svg.append("text")
    .attr("transform", "translate(" + (width/2) + "," + y(data[0].sev4) + ")")
    .attr("dy", ".35em")
    .attr("text-anchor", "start")
    .style("fill", "red")
    .text("Sev4");

svg.append("text")
    .attr("transform", "translate(" + (width/2) + "," + y(data[0].sev3) + ")")
    .attr("dy", ".35em")
    .attr("text-anchor", "start")
    .style("fill", "steelblue")
    .text("Sev3");

我没有测试它,所以我无法保证,但你的代码似乎很好,这是我唯一看到的。

此添加的结果:Your suggestion only moves the line labels to center but does not add value labels to the data points

修改

在您的澄清之后,这是一个插件:http://plnkr.co/edit/lDlseqUQQXgoFwTK5Aop?p=preview

您感兴趣的部分是:

svg.append('g')
.classed('labels-group', true)
.selectAll('text')
.data(data)
.enter()
.append('text')
.classed('label', true)
.attr({
  'x': function(d, i) {
    return x(d.date);
  },
  'y': function(d, i) {
    return y(d.sev3);
  }
})
.text(function(d, i) {
  return d.sev3;
});

这将绘制您的标签。这是你试图实现的结果吗?