How to create sprite sheets for Phaser 3 with TexturePacker

Joachim Grill, Andreas Löw
GitHub
How to create sprite sheets for Phaser 3 with TexturePacker

What you are going to learn

  • Creating sprite sheets with TexturePacker
  • Loading sprite sheets in Phaser 3
  • Setting Pivot Points with TexturePacker
  • Playing Animations from the sprite sheet
  • Optimizing start up time and reducing download size

Read this tutorial if you are still using Phaser 2: Creating Sprite Sheets for Phaser 2

Here's a preview of the game scene you are going to create during this tutorial - all assets included. The complete tutorial code is available on GitHub.

Loading Phaser Demo

Why should I use a sprite sheet?

The first step is to create a sprite sheet. Using sprite sheets with Phaser serves two primary purposes:

Speed up loading of the game:

By loading all graphics at once instead of numerous single images from a web server, the loading time of your game is significantly reduced.

Improve frame rate:

Using a sprite atlas enhances the game's performance. With WebGL, textures only need to be set once for rendering, resulting in improved frame rates.

TexturePacker is an essential part of our daily workflow. Every bit of GPU memory helps when dealing with mobile html5 games, so intelligent packing of assets is a must. And Texture Packer has all the features we need to effortlessly create atlases for our games.

Richard Davey (@photonstorm)
Creator of Phaser

Set up a new Phaser project

Creating a new phaser project is quite simple: just clone the Phaser 3 Webpack Project Template from GitHub, fetch the node modules, and run the start script:

git clone git@github.com:photonstorm/phaser3-project-template.git MyProject
cd MyProject
npm install
npm start

The start script bundles your project sources using webpack and serves your app on localhost:8080.

The complete demo project we're going to create in this tutorial is also available on GitHub.

Creating sprite sheets - the easy way

The easiest way to create your sprite sheets is using TexturePacker. Please download TexturePacker from here:

When starting the application choose Try TexturePacker Pro. In the main window use the Choose Data Format button and select Phaser 3 from the list. You can use the filter to find it faster.

Selecting Phaser 3 exporter for TexturePacker

Be careful to select the Phaser 3 format, only this one supports pivot point editing, multi-pack with one single json file and normal map packing. The other two Phaser data file formats can be used with older Phaser versions. In this case please have a look on the previous version of this tutorial

The demo project of this tutorial already contains some artwork in the /sprites-folder. Simply drag and drop the cityscene folder into TexturePacker.

Adding folders has two main advantages over adding single sprites

  • Adding or removing sprites in the folder also adds or removes them from the sprite sheet.
  • Sub folder names become part of the sprite names — e.g. capguy/walk/0001.png and not just 0001.png

After that, use the file selection button to enter a name for the JSON data file. Name it cityscene.json and place it in the public/assets/spritesheets folder of your project. By default, TexturePacker will save the texture image as cityscene.png in the same folder.

Finally, press Publish sprite sheet to create and save the sprite sheet. We are now finished with TexturePacker. That's all it takes to create a sprite sheet.

Creating sprite sheets for Phaser 3

Loading the sprite sheet in Phaser

The project template contains a src/index.js file which displays a bouncing phaser logo. For our demo app we remove the content of the preload() and create() functions and reuse the configuration as starting point:

src/index.js
import Phaser from 'phaser';

class MyGame extends Phaser.Scene
{
    constructor()
    {
        super();
    }

    preload()
    {
    }

    create()
    {
    }
}

const config = {
    type: Phaser.AUTO,
    parent: 'phaser-example',
    width: 800,
    height: 600,
    scene: MyGame
};

const game = new Phaser.Game(config);

The preload() function is used to load assets. Let's add our own one, which loads the sprite sheet:

preload()
{
    this.load.multiatlas('cityscene', 'assets/spritesheets/cityscene.json', 'assets/spritesheets');
}

The first parameter 'cityscene' specifies the key that can be used to access the atlas after it has been loaded. The second parameter 'assets/spritesheets/cityscene.json' is the atlas definition to load, the last parameter 'assets/spritesheets' is the name of the folder in which the image files are stored.

The create() function is used to set up our game scene. Let's add the background sprite to the scene. Within the sheet the sprite is referenced by its filename (enable "Trim sprite names" in TexturePacker if you prefer sprite names without .png extension):

create()
{
    this.add.sprite(0, 0, 'cityscene', 'background.png');
}

After saving the index.js file the npm start script will automatically rebuild the game and refresh the browser window. You will see only a quarter of our background image in the top-left corner:

Bad pivot point in Phaser 3

By default, the pivot point of a sprite is in the center of the image, so the call this.add.sprite(0, 0, ...) places the center of our background sprite at position (0,0). In the next section we will learn how to fix this.

Setting pivot points with TexturePacker

Pivot points can easily set for each sprite in TexturePacker. Select background.png on the left side of the TexturePacker window, and then click on the Sprite Settings button in the toolbar. On the left side you can now enable and configure the pivot point of the selected sprite:

Editing Phaser 3 anchor points

Select Top left in the Predefined input field, and re-publish the sprite sheet. Now reload the browser window displaying your phaser game (changed asset files are not detected automatically). The background image should be perfectly placed now:

Aligning sprites with Phaser 3

If the sprite isn't displayed as expected it's always a good idea to check the Javascript console in your browser. Maybe the sprite sheet wasn't found, or you're using a wrong sprite name? There shouldn't be any warning or error in the console!

Adding an animation

To add a walking character to the scene we have to create an animated sprite and move it across the screen. First we'll have to create the sprite and store it in a member variable of our scene:

create()
{
    this.add.sprite(0, 0, 'cityscene', 'background.png');

    this.capguy = this.add.sprite(0, 400, 'cityscene', 'capguy/walk/0001.png');
    this.capguy.setScale(0.5, 0.5);

This creates a sprite with the first frame of the animation: capguy/walk/0001.png and places it at position (0,400). The next line scales the sprite down by 50% because it would otherwise be a bit too big for the scene.

The function generateFrameNames() creates a bunch of frame names by creating zero-padded numbers between start and end, surrounded by prefix and suffix. 1 is the start index, 8 the end index, and the 4 is the number of digits to use:

    const frameNames = this.anims.generateFrameNames('cityscene', {
                           start: 1, end: 8, zeroPad: 4,
                           prefix: 'capguy/walk/', suffix: '.png'
                       });

The resulting names are:

  • { key:'cityscene', frame:'capguy/walk/0001.png' }
  • { key:'cityscene', frame:'capguy/walk/0002.png' }
  • ...
  • { key:'cityscene', frame:'capguy/walk/0008.png' }

Now we can create an animation called walk and add it to the capguy sprite:

    this.anims.create({ key: 'walk', frames: frameNames, frameRate: 10, repeat: -1 });
    this.capguy.anims.play('walk');
}

The result is Capguy walking on a spot at the left border:

Animating sprites with Phaser 3

Moving the sprite

There are several ways to move a sprite in Phaser. You are going to do a simple animation - just pushing the sprite along and resetting it after Capguy left the scene.

Add the following update method to the MyGame class, Phaser will call it periodically:

  update(time, delta)
  {
      this.capguy.x += delta/10;
      if (this.capguy.x > 800)
      {
          this.capguy.x = -50;
      }
  }

Not much to say about that: The time (in milliseconds) since the previous update call is passed as second parameter. To increase Capguy's position 100 pixels per second we divide delta by 10. After reaching the right border the sprite position is reset to the left border.

Using 9-slice game objects

9-Slice scaling is an efficient 2D graphic resizing technique that is particularly useful for scaling rectangular images and user interface elements without distorting their original aesthetics.

9-slice scaling

As you see, the image is split in 9 sectors. The corners stay identical and are not scaled at all. The borders at the top are scaled horizontally, the borders left and right are scaled vertically. The center is stretched in all directions.

Editing 9 slice sprites with TexturePacker
Editing 9-slice sprites in TexturePacker
  1. Open the sprite settings editor and select the sprite your want to edit on the left
  2. Check Enable 9-patch scaling
  3. Edit the sprite borders by dragging them with the mouse

To use such a game object in Phaser simply add

        // 9-slice objects
        this.add.nineslice(75, 50, 'cityscene', 'button.png', 100, 50);
        this.add.nineslice(250, 50, 'cityscene', 'button.png', 200, 50);

The result looks like this:

9-slice sprites in phaser

The arguments to this function are:

nineslice(x, y, texture, frame, width, height)
  • x, y: Position of the NiceSlice Object
  • texture: Name of the sprite sheet
  • frame: Name of the sprite
  • width and height of the object

There are some more parameters that you can use to specify the borders manually but these are not required if you use TexturePacker to define the NineSlice Object.

There are currently 2 restrictions in Phaser:

Phaser supports 9-slice game objects in the WebGL renderer only.

In Phaser 3.70.0 there's currently a bug that does not load the pivot points set in TexturePacker when using 9-slice sprites. See GitHub Issue #6655

Optimizing your game

The resulting cityscene.png has a size of around 400kb — Not much for a fast internet connection but for a mobile device it might be a good idea to optimize the loading time.

TexturePacker allows you to dramatically reduce the amount of memory used by your sprite sheets — while keeping the sprite quality almost like the original.

To enable this optimization select Texture format PNG-8 (indexed): Optimizing sprite sheets for Phaser 3

Here's the original sheet: 410kb High quality sprite sheet in Phaser 3

Here's the optimized sheet: 115kb - that is 72% less! Using png-8 (pngquant) sprite sheets with Phaser 3 to save memory

Multipack: The easy way to handle many sprites

So, what do you do if your game has more sprites than TexturePacker can fit on a single sprite sheet? Instead of generating multiple sprite sheets using separate TexturePacker projects, there's a much simpler solution:

Enable Auto Multipack in TexturePacker: This feature automatically generates as many sprite sheets as needed to hold all sprites. The great part is that all the necessary information is stored in the generated .json file, and you won't have to modify a single line of code!

For even better performance, consider grouping sprites that appear in the same scene of your game on the same sheet. With the Manual Multipack mode, you have the flexibility to manually add or remove sprite sheets, and you can easily move sprites between these sheets using drag-and-drop.

For more details see Multipack documentation.

Handling many sprites with Phaser 3

Finally

That's all for this tutorial. You should now have learned how to easily add animations to Phaser using TexturePacker and discovered how simple it is to optimize your game for improved startup time.