2017-07-19 67 views
0

我跟D3版本4力試驗向圖,並看着吉姆Vallandingham的教程和代碼爲出發點。力向圖中d3v4

http://vallandingham.me/bubble_chart_v4/

和我試圖從內森悠

https://flowingdata.com/2016/08/23/make-a-moving-bubbles-chart-to-show-clustering-and-distributions/

我剝去吉姆Vallandingham的代碼氣泡圖什麼,我想我需要在這裏產生類似的例子動畫並且可以通過改變指標值顯示各個州,但由於某些原因的代碼不想要的不同狀態之間進行動畫處理。我想重繪功能無法正常工作。這可能是一個明顯的錯誤,或者通過一個完全不知道做,但是如果你能幫助將是巨大的。

這裏是我的代碼:

function bubbleChart() { 
    var width = 940; 
    var height = 600; 
    var center = { x: width/2, y: height/3 }; 

    var years = ["0","2008", "2009", "2010"]; 

    var yearCenters = { 
    2008: { x: width/3, y: 2 * height/3 }, 
    2009: { x: width/2, y: 2 * height/3 }, 
    2010: { x: 2 * width/3, y: 2 * height/3 } 
    }; 

    // @v4 strength to apply to the position forces 
    var forceStrength = 0.03; 

    // These will be set in create_nodes and create_vis 
    var svg = null; 
    var bubbles = null; 
    var nodes = []; 
    var index= 0; 

    function charge(d) { 
    return -Math.pow(d.radius, 2.3) * forceStrength; 
    } 

    // Here we create a force layout 
    var simulation = d3.forceSimulation() 
    .velocityDecay(0.2) 
    .force('x', d3.forceX().strength(forceStrength).x(center.x)) 
    .force('y', d3.forceY().strength(forceStrength).y(center.y)) 
    .force('charge', d3.forceManyBody().strength(charge)) 
    .on('tick', ticked); 

    // @v4 Force starts up automatically, which we don't want as there aren't any nodes yet. 
    simulation.stop(); 

    // Nice looking colors 
    var fillColor = d3.scaleOrdinal() 
    .domain(['low', 'medium', 'high']) 
    .range(['#d84b2a', '#beccae', '#7aa25c']); 

    function createNodes(rawData) { 
    var myNodes = rawData.map(function (d) { 
     return { 
     id: d.id, 
     radius: 5, 
     value: +d.total_amount, 
     name: d.grant_title, 
     org: d.organization, 
     group: d.group, 
     year: d.start_year, 
     x: Math.random() * 900, 
     y: Math.random() * 800 
     }; 
    }); 

    // sort them to prevent occlusion of smaller nodes. 
    myNodes.sort(function (a, b) { return b.value - a.value; }); 

    return myNodes; 
    } 

    /* 
    * Main entry point to the bubble chart. 
    */ 
    var chart = function chart(selector, rawData) { 
    // convert raw data into nodes data 
    nodes = createNodes(rawData); 

    // Create a SVG element inside the provided selector 
    // with desired size. 
    svg = d3.select(selector) 
     .append('svg') 
     .attr('width', width) 
     .attr('height', height); 

    // Bind nodes data to what will become DOM elements to represent them. 
    bubbles = svg.selectAll('.bubble') 
     .data(nodes, function (d) { return d.id; }); 

    // Create new circle elements each with class `bubble`. 
    // There will be one circle.bubble for each object in the nodes array. 
    // Initially, their radius (r attribute) will be 0. 
    // @v4 Selections are immutable, so lets capture the 
    // enter selection to apply our transtition to below. 
    var bubblesE = bubbles.enter().append('circle') 
     .classed('bubble', true) 
     .attr('r', 0) 
     .attr('fill', function (d) { return fillColor(d.group); }) 
     .attr('stroke', function (d) { return d3.rgb(fillColor(d.group)).darker(); }) 
     .attr('stroke-width', 2) 

    // @v4 Merge the original empty selection and the enter selection 
    bubbles = bubbles.merge(bubblesE); 

    // Fancy transition to make bubbles appear, ending with the 
    // correct radius 
    bubbles.transition() 
     .duration(2000) 
     .attr('r', function (d) { return d.radius; }); 

    // Set the simulation's nodes to our newly created nodes array. 
    // @v4 Once we set the nodes, the simulation will start running automatically! 
    simulation.nodes(nodes); 

    chart.redraw(); 
    }; 

    // Callback function that is called after every tick of the force simulation. 
    // These x and y values are modified by the force simulation. 
    function ticked() { 
    bubbles 
     .attr('cx', function (d) { return d.x; }) 
     .attr('cy', function (d) { return d.y; }); 
    } 

    chart.redraw = function (index){ 
    simulation.force('x', d3.forceX().strength(forceStrength).x(nodePosX)); 
    simulation.force('y', d3.forceY().strength(forceStrength).y(nodePosY)); 
    simulation.alpha(1).restart(); 
    } 

    function nodePosX(d) { 
    if (+d.year <= +years[index]) { 
     return yearCenters[d.year].x; 
    } else { 
     return center.x; 
    } 
    } 
    function nodePosY(d) { 
    if (+d.year <= +years[index]) { 
     return yearCenters[d.year].y; 
    } else { 
     return center.y; 
    } 
    } 

    // return the chart function from closure. 
    return chart; 
} 

var myBubbleChart = bubbleChart(); 

myBubbleChart('#vis', data); 

for (i=0;i<4;i++){ 
    setInterval(function(){myBubbleChart.redraw(i);}, 100); 
} 
+1

你能創建一個可運行的計算器片段?這將會使你更容易看到發生了什麼事情? – Ian

回答

0

我誤解了如何用setInterval來重新繪製圖表,所以應該如下:

var i = 0; 
setInterval(function(){myBubbleChart.redraw(i++);}, 1000);