Search code examples
javascriptperformanceanimationcanvasrequestanimationframe

Controlling fps with requestAnimationFrame?


It seems like requestAnimationFrame is the de facto way to animate things now. It worked pretty well for me for the most part, but right now I'm trying to do some canvas animations and I was wondering: Is there any way to make sure it runs at a certain fps? I understand that the purpose of rAF is for consistently smooth animations, and I might run the risk of making my animation choppy, but right now it seems to run at drastically different speeds pretty arbitrarily, and I'm wondering if there's a way to combat that somehow.

I'd use setInterval but I want the optimizations that rAF offers (especially automatically stopping when the tab is in focus).

In case someone wants to look at my code, it's pretty much:

animateFlash: function() {
    ctx_fg.clearRect(0,0,canvasWidth,canvasHeight);
    ctx_fg.fillStyle = 'rgba(177,39,116,1)';
    ctx_fg.strokeStyle = 'none';
    ctx_fg.beginPath();
    for(var i in nodes) {
        nodes[i].drawFlash();
    }
    ctx_fg.fill();
    ctx_fg.closePath();
    var instance = this;
    var rafID = requestAnimationFrame(function(){
        instance.animateFlash();
    })

    var unfinishedNodes = nodes.filter(function(elem){
        return elem.timer < timerMax;
    });

    if(unfinishedNodes.length === 0) {
        console.log("done");
        cancelAnimationFrame(rafID);
        instance.animate();
    }
}

Where Node.drawFlash() is just some code that determines radius based off a counter variable and then draws a circle.


Solution

  • How to throttle requestAnimationFrame to a specific frame rate

    Demo throttling at 5 FPS: http://jsfiddle.net/m1erickson/CtsY3/

    This method works by testing the elapsed time since executing the last frame loop.

    Your drawing code executes only when your specified FPS interval has elapsed.

    The first part of the code sets some variables used to calculate elapsed time.

    var stop = false;
    var frameCount = 0;
    var $results = $("#results");
    var fps, fpsInterval, startTime, now, then, elapsed;
    
    
    // initialize the timer variables and start the animation
    
    function startAnimating(fps) {
        fpsInterval = 1000 / fps;
        then = Date.now();
        startTime = then;
        animate();
    }
    

    And this code is the actual requestAnimationFrame loop which draws at your specified FPS.

    // the animation loop calculates time elapsed since the last loop
    // and only draws if your specified fps interval is achieved
    
    function animate() {
    
        // request another frame
    
        requestAnimationFrame(animate);
    
        // calc elapsed time since last loop
    
        now = Date.now();
        elapsed = now - then;
    
        // if enough time has elapsed, draw the next frame
    
        if (elapsed > fpsInterval) {
    
            // Get ready for next frame by setting then=now, but also adjust for your
            // specified fpsInterval not being a multiple of RAF's interval (16.7ms)
            then = now - (elapsed % fpsInterval);
    
            // Put your drawing code here
    
        }
    }