Phaser 4 -- Sprite Animations
AnimationManager (global), AnimationState (per-sprite), creating animations from spritesheets and atlases, playing/pausing/chaining, animation events, frame callbacks.
Related skills: ../sprites-and-images/SKILL.md, ../loading-assets/SKILL.md
Quick Start
js
// In preload -- load a spritesheet
this.load.spritesheet('explosion', 'explosion.png', {
frameWidth: 64,
frameHeight: 64
});
// In create -- define a global animation
this.anims.create({
key: 'explode',
frames: this.anims.generateFrameNumbers('explosion', { start: 0, end: 11 }),
frameRate: 24,
repeat: 0
});
// Play it on a sprite
const sprite = this.add.sprite(400, 300, 'explosion');
sprite.play('explode');
Core Concepts
AnimationManager vs AnimationState
Phaser has two distinct animation objects:
| Aspect | AnimationManager | AnimationState |
|---|
| Access | (in a Scene) or | |
| Scope | Global -- shared across all scenes | Per-sprite instance |
| Purpose | Create/store animation definitions | Control playback on one Game Object |
| Class | Phaser.Animations.AnimationManager
| Phaser.Animations.AnimationState
|
The AnimationManager is a singleton owned by the Game. Animations registered there are available in every Scene. The AnimationState lives on each Sprite and handles playback for that specific object.
An
is a sequence of
objects plus timing data. Created via
this.anims.create(config)
(global) or
sprite.anims.create(config)
(local to one sprite).
Local vs Global Animations
When
is called, it first checks for a local animation with that key, then falls back to the global AnimationManager. Use local for sprite-specific animations; use global when shared across sprites.
js
// Global animation -- available to all sprites
this.anims.create({ key: 'walk', frames: 'player_walk', frameRate: 12, repeat: -1 });
// Local animation -- only on this sprite
sprite.anims.create({ key: 'walk', frames: 'npc_walk', frameRate: 10, repeat: -1 });
// This plays the LOCAL version because local takes priority
sprite.play('walk');
Common Patterns
Spritesheet Animation
Use
for spritesheets (numeric frame indices).
js
this.load.spritesheet('dude', 'dude.png', { frameWidth: 32, frameHeight: 48 });
// All frames
this.anims.create({
key: 'run',
frames: this.anims.generateFrameNumbers('dude', { start: 0, end: 7 }),
frameRate: 10,
repeat: -1
});
// Custom frame sequence
this.anims.create({
key: 'idle',
frames: this.anims.generateFrameNumbers('dude', { frames: [0, 1, 2, 1] }),
frameRate: 6,
repeat: -1
});
- (default ) -- first frame index
- (default , meaning last frame) -- final frame index
- -- a single frame to prepend before the range
- -- explicit array of frame indices (overrides start/end)
Atlas Animation
Use
for texture atlases (string-based frame names).
js
this.load.atlas('gems', 'gems.png', 'gems.json');
this.anims.create({
key: 'ruby_sparkle',
frames: this.anims.generateFrameNames('gems', {
prefix: 'ruby_',
start: 1,
end: 6,
zeroPad: 4 // produces ruby_0001 through ruby_0006
}),
frameRate: 12,
repeat: -1
});
- -- prepended to each frame number
- -- appended after each frame number
- , -- numeric range
- -- left-pad numbers to this length with zeros
- -- explicit array of frame numbers (overrides start/end)
If you call
with no config, it returns all frames from the atlas.
String as Frames
Pass a texture key string as
to use all frames from that texture, sorted numerically by default. Set
to disable sorting.
js
this.anims.create({ key: 'walk', frames: 'player_walk', frameRate: 12, repeat: -1 });
Yoyo and Repeat
js
this.anims.create({
key: 'pulse',
frames: this.anims.generateFrameNumbers('orb', { start: 0, end: 5 }),
frameRate: 10,
yoyo: true, // plays forward then backward
repeat: -1, // -1 = forever
repeatDelay: 500 // ms pause between each repeat cycle
});
When
is true, the animation plays forward then reverses. The full cycle counts as one play.
Chaining Animations
js
sprite.play('attack');
sprite.chain('idle'); // play idle after attack completes
sprite.chain(['fall', 'land', 'idle']); // chain multiple
sprite.anims.chain(); // clear the chain queue
Chaining is per-sprite. Chained animations start after
or
. An animation with
never completes -- call
to trigger the chain.
Playing in Reverse
js
// Play an animation from last frame to first
sprite.playReverse('walk');
// Reverse direction mid-playback
sprite.anims.reverse();
sets
and
. The
method toggles direction mid-playback.
Play Variants
js
sprite.play('walk', true); // ignoreIfPlaying = true
sprite.anims.playAfterDelay('walk', 1000); // play after 1s delay
sprite.anims.playAfterRepeat('walk', 2); // play after current anim repeats 2x
Animation Mixing
Adds a transition delay between two specific animations, set globally on the AnimationManager.
js
this.anims.addMix('idle', 'walk', 200);
this.anims.addMix('walk', 'idle', 300);
sprite.play('idle');
sprite.play('walk'); // 200ms mix delay applied automatically
this.anims.removeMix('idle', 'walk'); // remove specific pair
this.anims.removeMix('idle'); // remove all mixes for 'idle'
Mix delays only apply with
, not
or
.
Pause, Resume, and Stop
js
sprite.anims.pause(); // pause per-sprite
sprite.anims.resume(); // resume per-sprite
this.anims.pauseAll(); // global pause
this.anims.resumeAll(); // global resume
sprite.anims.stop(); // stop immediately
sprite.anims.stopAfterDelay(2000); // stop after 2 seconds
sprite.anims.stopAfterRepeat(1); // stop after 1 more repeat
sprite.anims.stopOnFrame(frame); // stop when a specific frame is reached
All stop methods fire
(not
). Chained animations trigger after stop.
Animation Events
js
sprite.on('animationcomplete', (anim, frame, gameObject, frameKey) => {
console.log('completed:', anim.key);
});
// Key-specific complete -- only fires for the named animation
sprite.on('animationcomplete-explode', (anim, frame, gameObject, frameKey) => {
gameObject.destroy();
});
Available events:
,
,
,
,
,
,
. All share the same callback signature:
(animation, frame, gameObject, frameKey)
.
Frame-Level Callbacks via animationupdate
js
sprite.on('animationupdate', (anim, frame, gameObject, frameKey) => {
if (anim.key === 'attack' && frame.index === 4) {
this.checkHit(gameObject);
}
});
Per-Frame Duration
Individual frames can have a
(ms) that is added to the base msPerFrame.
js
this.anims.create({
key: 'combo',
frames: [
{ key: 'fighter', frame: 'punch1', duration: 50 },
{ key: 'fighter', frame: 'kick', duration: 200 }, // hold longer
{ key: 'fighter', frame: 'recover', duration: 100 }
],
frameRate: 24
});
Visibility, Random Start, and TimeScale
js
// Visibility control
this.anims.create({
key: 'appear', frames: 'sparkle', frameRate: 12,
showOnStart: true, // sprite.visible = true when anim starts (after delay)
hideOnComplete: true, // sprite.visible = false when anim completes
showBeforeDelay: true // show first frame immediately even during delay
});
// Random start frame -- each sprite begins on a different frame
this.anims.create({
key: 'ambient', frames: 'fire', frameRate: 10, repeat: -1,
randomFrame: true
});
// TimeScale -- per-sprite or global speed control
sprite.anims.timeScale = 2; // 2x speed
sprite.play({ key: 'walk', timeScale: 0.5 }); // half speed via config
this.anims.globalTimeScale = 0.5; // affects ALL animations
Staggered Playback
js
const enemies = this.add.group({ key: 'enemy', repeat: 9 });
this.anims.staggerPlay('walk', enemies.getChildren(), 100);
// Each sprite starts 100ms after the previous. Pass staggerFirst: false to skip delay on first.
JSON Export and Import
js
// Export all global animations to JSON
const data = this.anims.toJSON();
// Import animations from JSON (pass true to clear existing animations first)
this.anims.fromJSON(data);
this.anims.fromJSON(data, true);
// Check before creating to avoid duplicate warning
if (!this.anims.exists('walk')) {
this.anims.create({ key: 'walk', frames: 'player_walk', frameRate: 12, repeat: -1 });
}
Modifying Animation Frames at Runtime
js
const anim = this.anims.get('walk');
// Add frames to the end
anim.addFrame(this.anims.generateFrameNumbers('player', { start: 8, end: 10 }));
// Insert frames at a specific index
anim.addFrameAt(this.anims.generateFrameNumbers('player', { frames: [5] }), 2);
// Remove a specific frame object
const frame = anim.frames[3];
anim.removeFrame(frame);
// Remove frame at index
anim.removeFrameAt(0);
Aseprite Support
js
this.load.aseprite('paladin', 'paladin.png', 'paladin.json');
// In create:
this.anims.createFromAseprite('paladin'); // all tags
this.anims.createFromAseprite('paladin', ['walk']); // specific tags only
sprite.play('walk'); // play by tag name
Configuration Reference
AnimationConfig (used with )
| Property | Type | Default | Description |
|---|
| string | -- | Unique identifier for the animation |
| string or AnimationFrame[] | | Texture key string (uses all frames) or array of frame config objects |
| boolean | | Numerically sort frames when using a string key |
| string | | Fallback texture key if not set per-frame |
| number | | Playback rate in frames per second (used if is null) |
| number | | Total animation length in ms (derives frameRate if set) |
| boolean | | Skip frames when lagging behind |
| number | | Delay before playback starts (ms) |
| number | | Times to repeat after first play (-1 = infinite) |
| number | | Delay before each repeat (ms) |
| boolean | | Reverse back to start before repeating |
| boolean | | Show first frame immediately during delay period |
| boolean | | Set visible=true when animation starts |
| boolean | | Set visible=false when animation completes |
| boolean | | Start from a random frame |
PlayAnimationConfig (used with )
All AnimationConfig timing properties are available, plus:
| Property | Type | Default | Description |
|---|
| string or Animation | -- | Animation key or instance to play |
| number | | Frame index to begin playback from |
| number | | Speed multiplier for this playback |
Values in PlayAnimationConfig override the animation definition for this specific playback instance.
Duration vs FrameRate Priority
- If both and are null: defaults to 24 fps.
- If only is set: frameRate is calculated as
totalFrames / (duration / 1000)
.
- If is set (even if duration is also set): frameRate wins, and duration is derived as
(totalFrames / frameRate) * 1000
.
Events
Sprite Event Flow
- -- after delay expires, before first update
- -- each frame change
- -- each repeat cycle
- -- natural end (finite repeat)
- -- same, with animation key appended
Stopped manually:
fires instead of complete. Restarted mid-play:
fires.
All callbacks:
(animation, frame, gameObject, frameKey)
.
AnimationManager Events
API Quick Reference
AnimationManager ()
| Method | Description |
|---|
| Create and register a global animation |
| Remove a global animation by key |
| Get an Animation instance by key |
| Check if a key is already registered |
generateFrameNumbers(key, config)
| Generate frame array from a spritesheet |
generateFrameNames(key, config)
| Generate frame array from an atlas |
| Play an animation on an array of Game Objects |
staggerPlay(key, children, stagger)
| Staggered play across multiple Game Objects |
| / | Pause/resume all animations globally |
addMix(animA, animB, delay)
| Set transition delay between two animations |
| Remove a mix pairing |
| Get the mix delay between two animations |
createFromAseprite(key, tags?, target?)
| Create animations from Aseprite JSON |
| Export all animations as JSON data |
| Load animations from JSON data (pass to clear existing first) |
AnimationState ()
| Method | Description |
|---|
play(key, ignoreIfPlaying?)
| Play an animation |
playReverse(key, ignoreIfPlaying?)
| Play an animation in reverse |
playAfterDelay(key, delay)
| Play after a delay in ms |
playAfterRepeat(key, repeatCount?)
| Play after current anim repeats N times |
| Queue animation(s) to play after current one |
| Stop immediately |
| Stop after a delay in ms |
stopAfterRepeat(repeatCount?)
| Stop after N more repeats |
| Stop when a specific frame is reached |
| Pause playback |
| Resume playback |
restart(includeDelay?, resetRepeats?)
| Restart from beginning |
| Reverse direction mid-playback |
| Get the current animation key |
| Get the current frame key |
| Get progress 0-1 |
| Set progress 0-1 |
| Change repeat count during playback |
| Get total frame count |
| Create a local animation on this sprite |
| Check if a local animation exists |
| Get a local animation by key |
Key properties:
,
,
,
,
,
,
.
Gotchas
- Animations are global by default. registers across all Scenes. Do not recreate in every Scene -- it logs a warning and returns the existing one.
- never fires . Use to end infinite animations. Listen for instead.
- beats . If both are set, frameRate wins. Set only (leave frameRate null) to control total length.
- Per-frame is additive. Added on top of base msPerFrame, not a replacement.
- stops the current animation (fires ). Use to skip if already playing.
- Mix delays only work with . / bypass mixes.
- Local animations override global. Same key on a sprite's local map takes priority.
- Sprite shorthand methods. , , , wrap .
- Chained anims fire after stop too. Clear the queue with before stopping if unwanted.
- end=-1 means last frame. The frame is excluded automatically.
Source File Map
| File | Purpose |
|---|
src/animations/AnimationManager.js
| Global singleton -- create, remove, get, generateFrame*, mix, staggerPlay |
src/animations/Animation.js
| Animation definition -- frames, timing, yoyo, repeat logic |
src/animations/AnimationState.js
| Per-sprite component -- play, stop, pause, chain, events |
src/animations/AnimationFrame.js
| Single frame data -- textureKey, textureFrame, duration, progress |
src/animations/events/index.js
| All animation event constants |
src/animations/typedefs/Animation.js
| AnimationConfig typedef |
src/animations/typedefs/PlayAnimationConfig.js
| PlayAnimationConfig typedef |
src/animations/typedefs/GenerateFrameNumbers.js
| Config for generateFrameNumbers |
src/animations/typedefs/GenerateFrameNames.js
| Config for generateFrameNames |