如何使用更新的数据集更新d3.js饼图

时间:2016-01-28 06:34:46

标签: javascript d3.js

我混合并匹配了来自其他来源的大量代码。

目前,饼图显示正常,但是当我在下拉菜单中选择其他服务器时,它无法更新图片。控制台日志显示数据集确实正在更改。我在网上查了一下,好像我的元素没有被正确删除?但是,每当我尝试使用path.exit()。remove()时,它都无法显示任何内容。

这是我的代码和jsfiddle:http://jsfiddle.net/36q95k1c/2/

var ds1 = [1,1,1,1,1,1,1,3,0,0];
var ds2 = [0,0,0,1,1,1,1,1,1,1];
var ds3 = [0,0,1,1,0,0,0,0,0,0];
var ds4 = [0,0,2,0,5,3,0,0,0,0];
var ds5 = [0,0,0,0,0,0,0,0,0,0];
var ds6 = [0,0,0,0,0,0,0,0,0,0];
var ds7 = [0,0,0,0,0,0,0,0,0,0];
var ds8 = [0,0,0,0,0,0,0,0,0,0];
var ds9 = [0,0,0,0,0,0,0,0,0,0];
var ds10 = [0,0,0,0,0,0,0,0,0,0];
var ds11 = [0,0,0,0,0,0,0,0,0,0];
var ds12 = [0,0,0,0,0,0,0,0,0,0];
var ds13 = [0,0,0,0,0,0,0,0,0,0];
var ds14 = [0,0,0,0,0,0,0,0,0,0];



var dataset = {inner: [4,1,31,28,13,65,6,6,4,3],
    middle: ds1,
    outer: [1175,1802,8126,11926,37264,4267,2961,2909,850,12432]};

var victim_total = 4+1+31+28+13+65+6+6+4+3;
var killer_total = 22+4+37+72+2+20+2+11+3+3;
var general_total = 1175+1802+8126+11926+37264+4267+2961+2909+850+12432;
var legendRectSize = 25;
var legendSpacing = 6;



//Width and height
var w = 750;
var h = 750;
var r = 100;
var donutWidth = 225


var outerRadius = w / 2;
var innerRadius = donutWidth;
var arc = d3.svg.arc()
    .innerRadius(innerRadius)
    .outerRadius(outerRadius);

var pie = d3.layout.pie()
    .sort(null);

// Easy colors accessible via a 10-step ordinal scale
var color = d3.scale.category20();

// Create SVG element
var svg = d3.select("body")
    .append("svg")
    .attr("width", 2*w)
    .attr("height", h)
    .append('g')
    .attr('transform', 'translate(' + (w / 2) + 
    ',' + (h / 2) + ')');

// Define the div for the tooltip
var div = d3.select("body").append("div")   
    .attr("class", "tooltip")               
    .style("opacity", 0);


function updateLegend(newData) {
    //https://dl.dropboxusercontent.com/s/hfho50s0xd2dcpn/craftinggeneralstats1.csv?dl=1"
    //https://dl.dropboxusercontent.com/s/i152v8ccetr5gj0/craftinggeneralstats.csv?dl=1
    //Import CSV file
d3.csv("https://dl.dropboxusercontent.com/s/i152v8ccetr5gj0/craftinggeneralstats.csv?dl=1", function(data) {
  data.forEach(function(d) {
    d.outer = +d.count;
    d.middle = +d.countkiller;
    d.inner = +d.countvictim;
    d.label = d.label;
  });
 // function updateLegend(newData) {
  /*var step;
  for (step = 0; step < 10; step++) {
  // Runs 5 times, with values of step 0 through 4.
  data[step].countkiller = 1;
}*/
//data[i].countkiller = 0;
console.log(dataset.middle);
// Set up groups
var arcs = svg.selectAll("g.arc")
    .data(d3.values(dataset));
    arcs.enter()
    .append("g")    
    .attr("class", "arc")
    .attr("transform", "translate(" +0+ "," +0+ ")");



var path = arcs.selectAll("path")
    .data(function(d) { return pie(d); });
  path.enter().append("path")
  .on("mouseover", function(d, i, j) { 
 // console.log(d);
  //console.log(dataset.middle[3]);
     div.transition()       
                .duration(200)      
                .style("opacity", .9)
                .style("left", (d3.event.pageX) + "px")     
                .style("top", (d3.event.pageY - 28) + "px");
            if (j ==0)
                div.html("Victim" + "<br/>" + Math.round(1000 * d.value / victim_total) / 10 +"%")
            if (j ==1)
                div.html("Killer" + "<br/>" + Math.round(1000 * d.value / killer_total) / 10 +"%")
            if (j ==2)
            {
                div.html("Overall" + "<br/>" + Math.round(1000 * d.value / general_total) / 10 +"%")
                }

            })                  
        .on("mouseout", function(d) {       
            div.transition()        
                .duration(500)      
                .style("opacity", 0);   
        })
    .attr("fill", function(d, i) { return color(i); })
    .attr("d", function(d, i, j) { return arc.innerRadius(10+r*j).outerRadius(r*(j+1))(d); });


   // .attr("text-anchor", "middle")
   /* .text(function (d, i) {
    return data[i].label;
});*/


// Setup legend
 var legend = svg.selectAll('.legend')                     
          .data(color.domain())                                   
          .enter()                                                
          .append('g')                                            
          .attr('class', 'legend')                                
          .attr('transform', function(d, i) {                     
            var height = legendRectSize + legendSpacing;          
            var offset =  height * color.domain().length / 2;     
            var horz = -2 * legendRectSize;                       
            var vert = i * height - offset;                       
            return 'translate(' + (horz +(w/2)) +  ',' + vert + ')';        
          });                                                     

        legend.append('rect')                                     
          .attr('width', legendRectSize)                          
          .attr('height', legendRectSize)                         
          .style('fill', color)                                   
          .style('stroke', color)                                   
            .on('click', function(label) {                            
              var rect = d3.select(this);                             
              var enabled = true;                                     


              if (rect.attr('class') === 'disabled') {                
                rect.attr('class', '');                               
              } else {                                                
                if (totalEnabled < 2) return;                         
                rect.attr('class', 'disabled');                       
                enabled = false;                                      
              }                                                       

              pie.value(function(d) {                                 
                if (d.label === label) d.enabled = enabled;           
                return (d.enabled) ? d.count : 0;                     
              });                                                    
              arcs = arcs.data(d3.values(dataset))                                 
            });                                                       
        legend.append('text')                                     
          .attr('x', legendRectSize + legendSpacing)              
          .attr('y', legendRectSize - legendSpacing)              
          .text(function (d, i) {
      return data[i].label;
});     
    path.exit().remove(); 
d3.select('#opts')  
  .on('change', function(d) {
    var server = eval(d3.select(this).property('value'));
    dataset.middle = server;
    updateLegend(dataset);
});

});
}

     updateLegend(dataset);

2 个答案:

答案 0 :(得分:0)

在当前版本中,您只需添加enter选项的声明,该声明仅影响插入的弧线。您需要指导D3如何处理update选项,这会影响更新的弧线和exit选择,这会影响被删除的弧线。您可以在path.enter()语句之前添加这些代码行。

path.attr("d", function(d, i, j) {
  return arc.innerRadius(10 + r * j).outerRadius(r * (j + 1))(d);
});

path.exit().remove();

更新的小提琴:http://jsfiddle.net/36q95k1c/5/

答案 1 :(得分:0)

要添加Hieu Le的好答案,对于有关动画的查询,找出添加transition()调用的正确位置可能有点棘手。试着把它放在Hieu Le's jsfiddle的100号线上,看看它是否属于你之后:

path.transition().attr("d", function(d, i, j) {
  return arc.innerRadius(10 + r * j).outerRadius(r * (j + 1))(d);
});

一旦您获得动画效果,请查看the docs以获取有关不同插值和补间的信息。

干杯!