d3.js - tween numbers on multiple elements

1.2k Views Asked by At

I've built a chart with a series of "small multiples", each representing a percentage of completion, arranged in rows and columns. Each row has a different number that represents "complete".

You can view the chart in this fiddle: http://jsfiddle.net/rolfsf/Lhnm9a9m/

I'm using transitions to grow the bars from 0 to x% width.

I also want to use a transition to show the actual count of each measure (e.g. 10 out of 17) incrementing from 0.

I've got all the numbers incrementing from 0 using a text tween function, but all of the measures stop at the same count, rather than their correct count.

I must be either using the wrong data in the tween, or placing the tween in the wrong part of the script... but I can't figure out where the problem is.

How do I get the numbers to increment properly??

My data looks like this:

var sets = [

        {"title": "Set-1", "count": 17, "measures": [10, 13, 16, 14]},
        {"title": "Set-2", "count": 23, "measures": [12, 18, 19, 23]},
        {"title": "Set-3", "count": 25, "measures": [19, 22, 23, 20]},
        {"title": "Set-4", "count": 4, "measures": [4, 4, 4, 4]},
        {"title": "Set-5", "count": 8, "measures": [5, 7, 8, 6]}
];

The chart is called like this:

d3  .select('#overview-graph')
    .datum(sets)
    .call(relativeCompletionChart()
    //options
    );

and here's the reusable chart script:

function relativeCompletionChart() {
    var width = 1200,
        margin = {top: 16, right: 16, bottom: 16, left: 16},
        onSetMouseOver = null,
        onSetClick = null;

    function chart(selection) {
        selection.each(function(data) {
            var titleColWidth = 0.3*width,
                setHeight = 24,
                barHeight = 22,
                setCount = data.length,
                colCount = data[0].measures.length,
                colWidth = (width - titleColWidth)/colCount,
                rangeWidth = colWidth - 4,
                height = ((setHeight * setCount) + margin.top + margin.bottom);


            var svg = d3.select(this)
                        .append("svg")
                        .attr("width", width)
                        .attr("height", height)
                        .attr("viewBox", "0 0 " + width + " " + height )
                        .attr("preserveAspectRatio", "xMidYMin meet");

            svg         .append('rect')
                        .attr("x", 0)
                        .attr('y', 0)
                        .attr('height', height)
                        .attr('width', width)
                        .attr('class', 'chart-bg');

            /**
            * Tween functions
            */
            function tweenText( newValue ) {
                return function() {
                  // get current value as starting point for tween animation
                  var currentValue = +this.textContent;
                  // create interpolator and do not show nasty floating numbers
                  var i = d3.interpolateRound( currentValue, newValue );

                  return function(t) {
                    this.textContent = i(t);
                  };
                }
            }

            function update(data) {

                var set = svg.selectAll("g.set")
                    .data(data);

                set.exit().remove();

                var setEnter = set
                        .enter().append("g")
                    .attr("class", "set")
                    .attr('transform', function (d, i) {
                        return 'translate(0, ' + (margin.top + i*setHeight) + ')';
                    });

                set.append("text")
                        .attr("class", "title")
                        .attr("x", (titleColWidth - 80))
                        .attr("y", 16)
                        .attr("text-anchor", "end")
                        .text(function (d){return d.title;});

                set.append("text")
                        .attr("class", "count")
                        .attr("x", (titleColWidth - 32))
                        .attr("y", 16)
                        .attr("text-anchor", "end")
                        .text(function (d){return d.count;});


                var ranges = set.selectAll("rect.range")
                        .data(function(d, i){return d.measures})
                            .enter().append('rect')
                        .attr("class", "range")
                        .attr("x",2)
                        .attr("y",0)
                        .attr('rx', 2)
                        .attr('ry', 2)
                        .attr("width", rangeWidth)
                        .attr("height",barHeight)
                        .attr("fill", "#CCCCCC")
                        .attr('transform', function (d, i) {
                            return 'translate(' + (titleColWidth + i*colWidth) + ', 0)';
                        });

                var measures = set.selectAll("rect.measure")
                        .data(function(d, i){return d.measures})
                            .enter().append('rect')
                        .attr("class", "measure")
                        .attr('rx', 2)
                        .attr('ry', 2)
                        .attr("x",2)
                        .attr("y",0)
                        .attr("width", 1)
                        .attr("height",barHeight)
                        .attr('transform', function (d, i) {
                            return 'translate(' + (titleColWidth + i*colWidth) + ', 0)';
                        });

                var markers = set.selectAll("line.marker")
                        .data(function(d, i){return d.measures})
                            .enter().append('line')
                        .attr("class", "marker")
                        .attr('x1', 2)
                        .attr('y1', 0)
                        .attr("x2",2)
                        .attr("y2",barHeight)
                        .attr('transform', function (d, i) {
                            return 'translate(' + (titleColWidth + i*colWidth) + ', 0)';
                        });

                var values = set.selectAll("text.value")     
                        .data(function(d, i){return d.measures})
                            .enter().append('text')
                        .text('0')
                        .attr("class", "value")
                        .attr("x", 8)
                        .attr("y", 16)
                        .attr("text-anchor", "start")
                        .attr('transform', function (d, i) {
                            return 'translate(' + (titleColWidth + i*colWidth) + ', 0)';
                        });


                //update widths
                set.selectAll("rect.measure")
                    .transition()
                    .duration(1000)
                    .delay(function(d, i) { return i * 20; })
                    .attr("width", function(d, i) { return d3.round((d/d3.select(this.parentNode).datum().count)*rangeWidth);});

                set.selectAll("line.marker")
                    .transition()
                    .duration(1000)
                    .delay(function(d, i) { return i * 20; })
                    .attr("x1", function(d, i) { return d3.round((d/d3.select(this.parentNode).datum().count)*rangeWidth + 1);})
                    .attr("x2", function(d, i) { return d3.round((d/d3.select(this.parentNode).datum().count)*rangeWidth + 1);});

                set.selectAll('text.value')
                    .transition()
                    .duration(1000)
                    .tween( 'text', function() {
                        // get current value as starting point for tween animation
                        var currentValue = +this.textContent;
                        // create interpolator and do not show nasty floating numbers
                        var interpolator = d3.interpolateRound( currentValue, 10 );

                        // this returned function will be called a couple
                        // of times to animate anything you want inside
                        // of your custom tween
                        return function( t ) {
                            // set new value to current text element
                            this.textContent = interpolator(t) + '/' + d3.select(this.parentNode).datum().count;
                        };
                    });                                
            }

            update(data);
        });

    }


    chart.width = function(_) {
        if (!arguments.length) return width;
        width = _;
        return chart;
    };

    chart.onSetClick = function(_) {
        if (!arguments.length) return onSetClick;
        onSetClick = _;
        return chart;
    };

    chart.onSetMouseOver = function(_) {
        if (!arguments.length) return onSetMouseOver;
        onSetMouseOver = _;
        return chart;
    };


    return chart;
 }

The relevant code for the tweening is pulled out here:

                    set.selectAll('text.value')
                    .transition()
                    .duration(1000)
                    .tween( 'text', function() {
                        // get current value as starting point for tween animation
                        var currentValue = +this.textContent;
                        // create interpolator and do not show nasty floating numbers
                        var interpolator = d3.interpolateRound( currentValue, 10 );

                        // this returned function will be called a couple
                        // of times to animate anything you want inside
                        // of your custom tween
                        return function( t ) {
                            // set new value to current text element
                            this.textContent = interpolator(t) + '/' + d3.select(this.parentNode).datum().count;
                        };
                    });       

Though I've also got an unused helper function in the script that I couldn't get to work:

           /**
            * Tween functions
            */
            function tweenText( newValue ) {
                return function() {
                  // get current value as starting point for tween animation
                  var currentValue = +this.textContent;
                  // create interpolator and do not show nasty floating numbers
                  var i = d3.interpolateRound( currentValue, newValue );

                  return function(t) {
                    this.textContent = i(t);
                  };
                }
            }                       
0

There are 0 best solutions below