Skip to content

Ultra-smooth sprite animation for Pixi using true delta-time interpolation

License

Notifications You must be signed in to change notification settings

kittykatattack/smoothie

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

17 Commits
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Smoothie - A Game Loop for Pixi

Smoothie is a super-fast and lightweight utility that gives you ultra-smooth sprite animation for the Pixi renderer using true delta-time interpolation. It also lets you specify the fps (frames-per-second) at which your game or application runs, and completely separates your sprite rendering loop from your application logic loop. Why is this cool? Because it means you can run your application logic at a crazy-low fps, like 30 or 12, but your game sprites will still animate at the highest rate at which your system is capable of. That gives you a huge boost in processing overhead without sacrificing smooth animation. And, Smoothie will seamlessly smoothe out all the frames in-between by interpolating sprite positions.

(Important! This library targets Pixi v3.0.11, which is the most stable version of Pixi, and is the only version I can recommend using. This library will eventually be upgraded for Pixi v4 when the v4 branch matures.)

Table of contents

Setting up
Animation
More about Smoothie's options
Smoothie's properties
Advanced features
License

Setting up

Link the smoothie.js file in your HTML document with a <script> tag.

<script src="smoothie.js"><script>

But make sure you've also loaded Pixi. (If you don't know how to install and use Pixi you can find out here.

In your JavaScript file, first create a Pixi renderer and stage, like this:

var renderer = PIXI.autoDetectRenderer(512, 512);
document.body.appendChild(renderer.view);
var stage = new PIXI.Container();

Next, create a new instance of Smoothie. Initialize it with an options object, which supplies Smoothie with the basic information it needs to run. Here's how to initialize it with the minimum number of options.

var smoothie = new Smoothie({
  engine: PIXI, 
  renderer: renderer,
  root: stage,
  fps: 30,
  update: update.bind(this)
});

You can see that the first three options are the PIXI global object, and the renderer and stage that you created in the first steps. (root refers to the "root container" object in your display list sprite hierarchy.)

The fourth option, fps defines the frames-per-second at which your game or application logic will run. If you leave this option out, Smoothie will default to 60 fps. If you set it to undefined Smoothie will remove the fps ceiling completely and run everything at the maximum frame rate your system is capable of (usually 60, but some device screens refresh at 120.) You will lose the advantage of smooth sprite animation with interpolation if you do this, but it's important to keep this feature in mind just in case you need it.

The last option, update is extremely important! It's the name of a function that will contain all of your game or application logic. Smoothie will run the function that you supply in a loop, at whatever frame rate you've specified. This is your game or application logic loop. Make sure to add bind(this) to the function name to ensure that it runs in the correct scope. In this example, the name of the function is called update. Here's a really simple example of the most basic update function you could write.

function update() {
  console.log("It loops!");  
}

(You can name this function anything you like, of course, it doesn't have to be called update.) Finally, the last thing you need to do is call Smoothie's start method to get it running.

smoothie.start();

Now you would see "It loops!" displayed in the browser's console at 30 frames per second.

This is the minimum setup that you need, but let's look at a more typical example that animates a sprite.

Animation

In this repository's examples folder you'll find a helloWorld.js file that contains all the working code for this next example. It very simply animates a cat sprite from the top left corner of the screen to the bottom right.

Example file

The JavaScript file first creates the Pixi renderer and stage, and uses them to initialize Smoothie.

//First, Create a Pixi renderer and stage
var renderer = PIXI.autoDetectRenderer(512, 512);
document.body.appendChild(renderer.view);
var stage = new PIXI.Container();

//Next, create a new instance of Smoothie
var smoothie = new Smoothie({
  engine: PIXI, 
  renderer: renderer,
  root: stage,
  update: update.bind(this),
  fps: 30
});

Pixi's loader is then used to load an image of the cat from the images folder.

PIXI.loader
  .add("images/cat.png")
  .load(setup);

As soon as the image is loaded, the loader calls the setup function. The setup function creates the cat sprite, and, very importantly, starts Smoothie.

var cat;

//This code will run when the loader has finished loading the image
function setup() {

  cat = new PIXI.Sprite(PIXI.loader.resources["images/cat.png"].texture);
  stage.addChild(cat);

  //After everything is set up, 
  //start Smoothie by calling its `start` method
  smoothie.start();
}

We assigned the update function to Smoothie's update property, so we need to include it in our program. Our update function will contain all the code needed to animate the sprite, and will run at the same fps that we assigned to Smoothie.

function update() {

  //Use any physics or game logic code here
  cat.x += 1;
  cat.y += 1;
}

What's so cool about Smoothie is that even though your game logic might be running at a very low frame rate, like 12 fps, your animations will render at the highest frame rate that your system is capable of. That gives you a vast amount of extra processing power to play with - about 500% more than if you were running your logic at 60fps. That's just what you need for math or physics-heavy games. And, Smoothie smoothes out any rendering variations between frames caused by hiccups in the browser, like garbage collection spikes. That means you get ultra-smooth animation at any frame rate. And, your render loop is completely separated from your game logic loop.

You can add any game logic, animation or physics code to your update function, or use it as the basis for creating a game state manager, as described here. Smoothie does all the hard work for you - you'll never have to worry about rendering again.

More about Smoothie's options

In addition to the 5 options you can supply to Smoothie's constructor when you initialize it, there are few more you can - optionally - supply.

Smoothie interpolates each sprite's position (its x and y values) and rotation by default. However, you can interpolate more properties if you need to: size (width and height), scale (scale.x and scale.y) and alpha. You can switch on interpolation for all these properties by adding the properties option in Smoothie's constructor. properties is an object that defines 5 Boolean (true/false) values for the sprite properties you want to interpolate. Here's how to switch all of them on:

properties: {
  position: true, 
  rotation: true, 
  alpha: true, 
  scale: true, 
  size: true,
  tile: true
}

Selectively add whichever properties you want. Setting the last property, tile to true switched on smooth interpolation for Pixi tiling sprite properties: tilePosition.x, tilePosition.y, tileScale.x and tileScale.y.

You can also add a Boolean interpolate property that defines whether Smoothie should use interpolation (true) or not (false).

interpolate: true

If you set it to false, Smoothie will render the sprites and the update function logic at the same frame rate. That means jittery animation at low frame rates - if that's what you want, you've got it!

Just for your reference, here's an example of how you might initialize Smoothie with all of its possible options (only the first four are required).

var smoothie = new Smoothie({
  engine: PIXI, 
  renderer: renderer,
  root: stage,
  update: update.bind(this),
  fps: 30,
  interpolate: true,
  properties: {
    position: true, 
    rotation: true, 
    alpha: true, 
    scale: true, 
    size: true,
    tile: true
  }
});

And don't forget to call Smoothie's start method when you're ready to start animating!

Smoothie's properties

Smoothie has a few properties that you can change at run-time.

  • fps: Smoothie's frame rate. If you set fps to undefined, Smoothie removes the fps ceiling. It then runs the user-defined update function at the maximum frame rate your system is capable of.
  • interpolate: A Boolean (true or false) value to switch sprite interpolation on or off.
  • properties: an object containing the Boolean values position, rotation, scale, size, and alpha which determine which sprite properties are interpolated.
  • pause(): A method to pause Smoothie. It stops the game loop from running.
  • resume(): Resume Smoothie if you've paused it.

And now you know how to use Smoothie!

Advanced features

Smoothie has a few extra features that you'll probably never need to use. But, just in case, here they are.

Set a rendering frame rate clamp

To help you fine tune your render loop, Smoothie give you a renderFps property that lets you set the maximum frame rate at which sprites should render. You can set it either in the constructor options or, like this at runtime:

smoothie.renderFps = 30;

This will clamp the upper-limit rendering fps to 30 (You can set it to any fps you want). What that means is that, if your device screen is synced to 60Hz, a renderFps of 30, will make your sprite render only every second frame. That can save you an additional 200% processing overhead.

Should you do this? Probably not! Or, only as a last resort. That's because browsers optimize rendering in all kinds of complex and inscrutable ways. So it's safer just to trust that they're doing the best job they can, and not try and second guess or preemptively optimize them. You might inadvertently cause more jankiness than you're trying to fix. ...Or not! The only way to know for sure is to test with your particular application and see.

One other warning: if your game or application is running so badly that a renderFps clamp is the only thing that can save it, you've gone to far. Take a good a look at your application code and try and optimize it in other ways.

Find the Delta Time

The amount of difference between the Smoothie's frame rate and the rate at which the browser is rendering sprites is accessible in a read-only property called dt (for "delta time"). You can access it like this:

smoothie.dt

It's a normalized value (a decimal number between 0 and 1) which you might think of some use for.

Prevent texture bleed

If you ever notice any texture bleed in your sprites, set the sprite's texture scale mode to NEAREST. Here's how:

anySprite.texture.baseTexture.scaleMode = PIXI.SCALE_MODES.NEAREST;

This forces Pixi to display sprite images using integer (whole number) pixel values, not floating point (decmial numbers) which it does by default. However, it will also make slow animations appear jankier because sprite positions will appear to change in minimum units of 1 pixel. A better solution is to use a tool like Texture Packer that packages sprites with 2 pixels of padding around them to prevent texture bleed.

License

Smoothie is licenced under the MIT license. So if you want to use Smoothie for anything, include to reverse engineer Smoothie so that Smoothie itself no longer exists, go for it!

About

Ultra-smooth sprite animation for Pixi using true delta-time interpolation

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Contributors 3

  •  
  •  
  •