phaser4-gamedev

star 132

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.

chongdashu By chongdashu schedule Updated 4/12/2026

name: phaser4-gamedev 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:

  1. WebGL-first, not Canvas-first: Phaser 4 is designed around WebGL. Treat Canvas as legacy compatibility, not the default target.
  2. Measure assets before loader config: Sprite and tile bugs often start as incorrect frame metadata, not rendering bugs.
  3. Prefer the simplest rendering path: Use standard game objects until scale or effect requirements justify filters, GPU layers, or shader work.
  4. Treat rendering features as architectural choices: Filters, lighting, shaders, and render textures affect coordinate systems, batching, and debugging.
  5. 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-gamedev/references/arcade-physics.md
Tilemaps, object layers, collision setup ../phaser-gamedev/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 }); // Stop current, start new
this.scene.launch('UIScene');                // Run in parallel
this.scene.pause('GameScene');               // Pause
this.scene.stop('UIScene');                  // Stop

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) {} // Receive data from previous scene
  preload() {}           // Load assets before create
  create() {}            // Set up game objects, physics, input
  update(time: number, delta: number) {} // Use delta for frame-rate independence
}

Frame-Rate Independent Movement

// Correct: scales with frame rate
this.player.x += this.speed * (delta / 1000);

// Wrong: varies with frame rate
this.player.x += this.speed;

Phaser 4 Migration Replacements

// Phaser 3
sprite.setTintFill(0xff0000);
// Phaser 4
sprite.setTint(0xff0000).setTintMode(Phaser.TintModes.FILL);
// Phaser 3
sprite.setPipeline('Light2D');
// Phaser 4
sprite.setLighting(true);
// Phaser 3
const mask = new Phaser.Display.Masks.BitmapMask(scene, maskObject);
sprite.setMask(mask);
// Phaser 4
sprite.filters.internal.addMask(maskObject);
// Phaser 3
colorMatrix.sepia();
// Phaser 4
colorMatrix.colorMatrix.sepia();
// Phaser 3
Math.TAU  // was PI/2
Math.PI2  // was PI*2
// Phaser 4
Math.PI_OVER_2  // PI/2
Math.TAU        // PI*2 (correct 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(); // required — without this, nothing lands on the texture

Use preserve() or render modes only when they solve a concrete problem. Extra indirection complicates debugging quickly.

TilemapGPULayer

// Add 'gpu' flag to get a TilemapGPULayer instead of TilemapLayer
const layer = map.createLayer('Ground', tileset, 0, 0, { gpu: true });

// After editing tile data, regenerate the GPU texture
layer.generateLayerDataTexture();

Constraints: orthographic only, one tileset, max 4096×4096 tiles.

SpriteGPULayer

const layer = this.add.spriteGPULayer(texture, frameCount);

// Reuse one config object — don't create millions of new objects
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'; // per-object: off | safe | safeAuto | full | fullAuto

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.

Install via CLI
npx skills add https://github.com/chongdashu/vibejam-starter-pack --skill phaser4-gamedev
Repository Details
star Stars 132
call_split Forks 39
navigation Branch main
article Path SKILL.md
More from Creator