| name | phaser4 |
| description | Build 2D browser games with Phaser 4: WebGL-first rendering, scenes, filters, lighting, shaders, DynamicTexture and RenderTexture, tilemaps, SpriteGPULayer, TilemapGPULayer, and Phaser 3 to 4 migration work. Trigger: phaser 4, phaser v4, migrate phaser 3 to 4, phaser webgl renderer, phaser filters, phaser render nodes, phaser SpriteGPULayer, phaser TilemapGPULayer. |
Phaser 4 Game Development
Build 2D browser games using Phaser 4's WebGL-first renderer, scene model, and updated rendering APIs.
Philosophy: Renderer-Aware, Asset-Exact
Phaser 4 is not Phaser 3 with a few renamed methods. The renderer, filter model, shader assumptions, texture orientation, and batching behavior changed. Good Phaser 4 work starts by choosing the right rendering path and measuring assets before code is written.
Before coding, ask:
- Is this a new Phaser 4 feature or a Phaser 3 migration?
- Does this feature stay on standard game object APIs, or does it depend on filters, shaders, lighting, or custom rendering?
- What is the asset source of truth: exact frame size, spacing, margin, atlas bounds, and texture orientation?
- Is the bottleneck CPU object churn, GPU fill rate, or batch breaking?
- Would
SpriteGPULayer, TilemapGPULayer, RenderTexture, or a plain Sprite solve this more cleanly?
Core principles:
- WebGL-first, not Canvas-first: Phaser 4 is designed around WebGL. Treat Canvas as legacy compatibility, not the default target.
- Measure assets before loader config: Sprite and tile bugs often start as incorrect frame metadata, not rendering bugs.
- Prefer the simplest rendering path: Use standard game objects until scale or effect requirements justify filters, GPU layers, or shader work.
- Treat rendering features as architectural choices: Filters, lighting, shaders, and render textures affect coordinate systems, batching, and debugging.
- Migration is selective redesign: Basic scene code may port cleanly, but masks, FX, custom pipelines, shaders, and texture workflows usually need real updates.
STOP: Before Loading Any Spritesheet or Atlas
Read references/spritesheets-and-textures.md first.
Spritesheet loading is still fragile. A few pixels off in frame size, spacing, or margin can create silent corruption that looks like animation or rendering bugs later.
NEVER guess frame dimensions.
DO NOT assume texture orientation details are irrelevant if compressed textures or custom shaders are involved.
STOP: Before Porting Phaser 3 Code
Read references/migration-hotspots.md first.
Search for the Phaser 3 APIs that changed meaning or disappeared. These are where most migration time goes:
setTintFill
BitmapMask
preFX / postFX
Phaser.Geom.Point
Math.TAU / Math.PI2
setPipeline('Light2D')
DynamicTexture / RenderTexture
- custom pipelines
- custom shader code
TileSprite cropping
Reference Files
Read these before working on the relevant feature:
| When working on... | Read first |
|---|
| Migrating Phaser 3 code | references/migration-hotspots.md |
| Loading spritesheets, atlases, compressed textures, or TileSprite | references/spritesheets-and-textures.md |
| Performance issues, GPU layers, filters, lighting, or batching | references/rendering-and-performance.md |
| Arcade physics, bodies, groups, pooling | ../phaser/references/arcade-physics.md |
| Tilemaps, object layers, collision setup | ../phaser/references/tilemaps.md |
Architecture Decisions (Make Early)
Rendering Path Choice
| Path | Use when |
|---|
| Standard game objects | Most gameplay, UI, and ordinary animation |
SpriteGPULayer | Very large numbers of mostly simple quads or particle-like members |
TilemapGPULayer | Very large orthographic tile layers using one tileset |
RenderTexture / DynamicTexture | You need capture, compositing, stamping, or texture reuse |
| Filters / Shader | The effect is genuinely image-space or shader-driven |
Physics System Choice
| System | Use when |
|---|
| Arcade | Platformers, shooters, most 2D action games |
| Matter | Physics puzzles, compound bodies, more realistic collisions |
| None | Menu scenes, card games, visual novels, strategy UIs |
Scene Structure
scenes/
āāā BootScene.ts # Asset loading, progress bar, shader/texture setup
āāā MenuScene.ts # Title screen and options
āāā GameScene.ts # Main gameplay
āāā UIScene.ts # HUD overlay (launched in parallel)
āāā GameOverScene.ts # End screen and restart flow
Scene Transitions
this.scene.start('GameScene', { level: 1 });
this.scene.launch('UIScene');
this.scene.pause('GameScene');
this.scene.stop('UIScene');
Core Patterns
Game Configuration
Prefer explicit WebGL unless there is a concrete reason not to.
const config: Phaser.Types.Core.GameConfig = {
type: Phaser.WEBGL,
width: 800,
height: 600,
roundPixels: false,
scale: {
mode: Phaser.Scale.FIT,
autoCenter: Phaser.Scale.CENTER_BOTH
},
physics: {
default: 'arcade',
arcade: { gravity: { y: 300 }, debug: false }
},
scene: [BootScene, MenuScene, GameScene]
};
Scene Lifecycle
class GameScene extends Phaser.Scene {
init(data: unknown) {}
preload() {}
create() {}
update(time: number, delta: number) {}
}
Frame-Rate Independent Movement
this.player.x += this.speed * (delta / 1000);
this.player.x += this.speed;
Phaser 4 Migration Replacements
sprite.setTintFill(0xff0000);
sprite.setTint(0xff0000).setTintMode(Phaser.TintModes.FILL);
sprite.setPipeline('Light2D');
sprite.setLighting(true);
const mask = new Phaser.Display.Masks.BitmapMask(scene, maskObject);
sprite.setMask(mask);
sprite.filters.internal.addMask(maskObject);
colorMatrix.sepia();
colorMatrix.colorMatrix.sepia();
Math.TAU
Math.PI2
Math.PI_OVER_2
Math.TAU
RenderTexture and DynamicTexture
Phaser 4 buffers drawing commands. Execute them deliberately.
const rt = this.add.renderTexture(0, 0, 256, 256);
rt.draw(sprite, 0, 0);
rt.render();
Use preserve() or render modes only when they solve a concrete problem. Extra indirection complicates debugging quickly.
TilemapGPULayer
const layer = map.createLayer('Ground', tileset, 0, 0, { gpu: true });
layer.generateLayerDataTexture();
Constraints: orthographic only, one tileset, max 4096Ć4096 tiles.
SpriteGPULayer
const layer = this.add.spriteGPULayer(texture, frameCount);
const memberConfig = {};
for (let i = 0; i < 100000; i++) {
memberConfig.x = Math.random() * 800;
memberConfig.y = Math.random() * 600;
memberConfig.scaleX = memberConfig.scaleY = 1;
memberConfig.alpha = 1;
layer.addMember(memberConfig);
}
Use it for starfields, particle-like swarms, animated backgrounds ā not for interactive gameplay entities that mutate frequently.
Pixel Rounding
Do not assume old roundPixels behavior. Phaser 4 defaults it to false.
sprite.vertexRoundMode = 'safe';
Use rounding intentionally for pixel art. Leave it off for rotated, scaled, or camera-heavy scenes.
Anti-Patterns to Avoid
| Anti-pattern | Why it hurts | Better |
|---|
| Treating Phaser 4 as a drop-in Phaser 3 upgrade | You miss renderer, filter, shader, and texture changes | Audit migration hotspots first, then port intentionally |
| Starting new work on Canvas-first assumptions | Many Phaser 4 features are WebGL-centric or unavailable in Canvas | Design for WebGL; treat Canvas as fallback only if required |
| Guessing spritesheet or atlas metadata | Visual corruption appears far from the actual mistake | Measure frames, spacing, margin, and bounds before loading |
| Using filters or shaders for every visual effect | More complexity, more batch breaks, harder debugging | Use plain sprites, textures, and tint where possible |
| Applying lighting or filters everywhere | Shader changes break batches and can tank performance | Reserve them for objects that benefit visually |
Forgetting render() on DynamicTexture or RenderTexture | Queued work never lands on the texture | Make render execution explicit in the workflow |
Using SpriteGPULayer for frequently mutated gameplay entities | Its strength is scale, not arbitrary object behavior | Keep complex interactive entities on normal game objects |
Assuming TilemapGPULayer is a universal tilemap replacement | It is orthographic-only and more constrained | Use it when the layer size and rendering profile justify it |
Making raw gl calls outside supported integration points | You can desync Phaser's renderer state | Use Extern or higher-level Phaser APIs |
| "The port compiles, so the migration is done" | Rendering, shader, and texture bugs survive the first compile | Re-test visuals explicitly after every render-touching change |
Variation Guidance
Avoid converging on a single Phaser 4 setup. Choose based on context:
- Rendering path: standard objects vs GPU layers vs textures vs shader/filter pipelines
- Physics: Arcade vs Matter vs none
- Content: tilemaps vs pure sprites vs hybrid
- Pixel art handling:
roundPixels off, safe per-object rounding, or deliberate full rounding
- Assets: spritesheets vs atlases vs single textures
- Scene layout: separate
UIScene vs in-scene HUD
- Tilemap:
TilemapLayer vs TilemapGPULayer (only when constraints fit)
What should vary is the architecture, not the rigor. Measure assets, check batching costs, and adapt the solution to the game's real constraints.
Remember
Phaser 4 gives you a more capable renderer and more explicit rendering tools, but it expects better architectural choices in return.
Before coding: what rendering path are you choosing, what assets define the truth, and what batch-breaking features are actually worth their cost?
Claude can do strong Phaser 4 work when the problem is framed precisely: scene boundaries, asset dimensions, rendering constraints, performance targets, and migration scope. These guidelines illuminate the path; they do not replace engineering judgment.