Golang Game Development: Building a Basic Platformer with Bappa Framework

Ever wanted to build your own platformer game in Go but didn’t know where to start? Today’s your lucky day. While documentation and examples are helpful, nothing beats a hands-on tutorial that takes you from zero to a functioning game. As the creator of Bappa, I‘ve designed this guide to walk you through creating a basic 2D platformer with jumping mechanics, collisions, one-way platforms, and smooth animations — all using Go and the Bappa Framework. This tutorial walks you through creating a 2D platformer from the ground up, covering: Setting up your project structure Implementing player movement and jumping Creating collision detection Building one-way platforms Adding smooth animations Complete beginner to game development or Go? You can still follow along! The code examples are designed to work out-of-the-box, and I explain the core concepts as we go. If you get stuck, the companion repository has the complete code at each stage. By the end of this guide, you’ll not only have a working game but also understand the fundamentals that make it tick. If you find this approach helpful, consider giving Bappa a star on GitHub to support its development. You can also find the original blog here Let’s start building! Project Setup Let's get started by setting up our project! Although bappacreate is typically recommended for new projects, we'll start from the most basic repository to better understand the framework's fundamentals. Initial Project Structure Download the base project from github The template includes only the essential assets and an initialized Go module named platformer Creating the Main File First, create a new file named main.go with the following content: // main.go package main import ( "embed" "log" "github.com/TheBitDrifter/coldbrew" ) //go:embed assets/* var assets embed.FS const ( RESOLUTION_X = 640 RESOLUTION_Y = 360 MAX_SPRITES_CACHED = 100 MAX_SOUNDS_CACHED = 100 MAX_SCENES_CACHED = 12 ) func main() { // Create the client client := coldbrew.NewClient( RESOLUTION_X, RESOLUTION_Y, MAX_SPRITES_CACHED, MAX_SOUNDS_CACHED, MAX_SCENES_CACHED, assets, ) // Configure client settings client.SetTitle("Platformer") client.SetResizable(true) client.SetMinimumLoadTime(30) // Run the client if err := client.Start(); err != nil { log.Fatal(err) } } Installing Dependencies Now that we have our main.go file, with the required imports, to install dependencies execute the following from your project directory: go get github.com/TheBitDrifter/coldbrew@latest go mod tidy Running the Game To run your game, execute the following command from your project directory: go run . At this point, you'll see a blank window with the specified resolution. While not very exciting yet, this provides the foundation for our platformer. In the next section, we'll start adding game elements and bringing our world to life. Creating Your First Scene Let's create our first scene. Start by creating a scenes/ directory in your project root. We'll need two files to set up our scene structure: Scene Structure Definition First, let's create the base scene structure that we'll use throughout our game: // scenes/scene.go package scenes import "github.com/TheBitDrifter/blueprint" type Scene struct { Name string Plan blueprint.Plan Width, Height int } Implementing Scene One Next, let's create our first scene with a parallax background: // scenes/scene_one.go package scenes import ( "github.com/TheBitDrifter/blueprint" "github.com/TheBitDrifter/warehouse" ) const SCENE_ONE_NAME = "scene one" var SceneOne = Scene{ Name: SCENE_ONE_NAME, Plan: sceneOnePlan, Width: 1600, Height: 500, } func sceneOnePlan(height, width int, sto warehouse.Storage) error { err := blueprint.NewParallaxBackgroundBuilder(sto). AddLayer("backgrounds/city/sky.png", 0.025, 0.025). AddLayer("backgrounds/city/far.png", 0.025, 0.05). AddLayer("backgrounds/city/mid.png", 0.1, 0.1). AddLayer("backgrounds/city/near.png", 0.2, 0.2). Build() if err != nil { return err } return nil } Understanding the Scene Structure Let's break down what's happening in these files: The Scene structure in scene.go defines our basic scene template with: A name identifier A plan function (type blueprint.Plan) Width and height dimensions In scene_one.go, we create our first scene implementation: We define a constant for the scene name Create an instance of our Scene structure called SceneOne Implement the scene's plan function The sceneOnePlan function is particularly interesting - it's what Bappa calls a blueprint.Plan. This special function sign

Mar 26, 2025 - 22:37
 0
Golang Game Development: Building a Basic Platformer with Bappa Framework

Ever wanted to build your own platformer game in Go but didn’t know where to start? Today’s your lucky day.

While documentation and examples are helpful, nothing beats a hands-on tutorial that takes you from zero to a functioning game. As the creator of Bappa, I‘ve designed this guide to walk you through creating a basic 2D platformer with jumping mechanics, collisions, one-way platforms, and smooth animations — all using Go and the Bappa Framework.

This tutorial walks you through creating a 2D platformer from the ground up, covering:

  • Setting up your project structure
  • Implementing player movement and jumping
  • Creating collision detection
  • Building one-way platforms
  • Adding smooth animations

Complete beginner to game development or Go? You can still follow along! The code examples are designed to work out-of-the-box, and I explain the core concepts as we go. If you get stuck, the companion repository has the complete code at each stage.

By the end of this guide, you’ll not only have a working game but also understand the fundamentals that make it tick. If you find this approach helpful, consider giving Bappa a star on GitHub to support its development.

You can also find the original blog here

Let’s start building!

Project Setup

Let's get started by setting up our project!

Although bappacreate is typically recommended for new projects, we'll start from the most basic repository to better understand the framework's fundamentals.

Initial Project Structure

  • Download the base project from github
  • The template includes only the essential assets and an initialized Go module named platformer

Creating the Main File

First, create a new file named main.go with the following content:

// main.go

package main

import (
    "embed"
    "log"
    "github.com/TheBitDrifter/coldbrew"
)

//go:embed assets/*
var assets embed.FS

const (
    RESOLUTION_X       = 640
    RESOLUTION_Y       = 360
    MAX_SPRITES_CACHED = 100
    MAX_SOUNDS_CACHED  = 100
    MAX_SCENES_CACHED  = 12
)

func main() {
    // Create the client
    client := coldbrew.NewClient(
        RESOLUTION_X,
        RESOLUTION_Y,
        MAX_SPRITES_CACHED,
        MAX_SOUNDS_CACHED,
        MAX_SCENES_CACHED,
        assets,
    )

    // Configure client settings
    client.SetTitle("Platformer")
    client.SetResizable(true)
    client.SetMinimumLoadTime(30)

    // Run the client
    if err := client.Start(); err != nil {
        log.Fatal(err)
    }
}

Installing Dependencies

Now that we have our main.go file, with the required imports, to install dependencies execute the following
from your project directory:

go get github.com/TheBitDrifter/coldbrew@latest
go mod tidy

Running the Game

To run your game, execute the following command from your project directory:

go run .

Image description

At this point, you'll see a blank window with the specified resolution. While not very exciting yet, this provides the foundation for our platformer. In the next section, we'll start adding game elements and bringing our world to life.

Creating Your First Scene

Let's create our first scene. Start by creating a scenes/ directory in your project root. We'll need two files to set up our scene structure:

Scene Structure Definition

First, let's create the base scene structure that we'll use throughout our game:


// scenes/scene.go

package scenes

import "github.com/TheBitDrifter/blueprint"

type Scene struct {
    Name          string
    Plan          blueprint.Plan
    Width, Height int
}

Implementing Scene One

Next, let's create our first scene with a parallax background:


// scenes/scene_one.go

package scenes

import (
    "github.com/TheBitDrifter/blueprint"
    "github.com/TheBitDrifter/warehouse"
)

const SCENE_ONE_NAME = "scene one"

var SceneOne = Scene{
    Name:   SCENE_ONE_NAME,
    Plan:   sceneOnePlan,
    Width:  1600,
    Height: 500,
}

func sceneOnePlan(height, width int, sto warehouse.Storage) error {
    err := blueprint.NewParallaxBackgroundBuilder(sto).
        AddLayer("backgrounds/city/sky.png", 0.025, 0.025).
        AddLayer("backgrounds/city/far.png", 0.025, 0.05).
        AddLayer("backgrounds/city/mid.png", 0.1, 0.1).
        AddLayer("backgrounds/city/near.png", 0.2, 0.2).
        Build()
    if err != nil {
        return err
    }
    return nil
}

Understanding the Scene Structure

Let's break down what's happening in these files:

  1. The Scene structure in scene.go defines our basic scene template with:
  • A name identifier
  • A plan function (type blueprint.Plan)
  • Width and height dimensions
  1. In scene_one.go, we create our first scene implementation:
    • We define a constant for the scene name
    • Create an instance of our Scene structure called SceneOne
    • Implement the scene's plan function

The sceneOnePlan function is particularly interesting - it's what Bappa calls a blueprint.Plan. This special function signature is expected by the client and gets called automatically when scenes become active. The plan function is where we define what should exist in our scene when it loads.

In this case, we're using the Blueprint API's ParallaxBackgroundBuilder to quickly create a multi-layered scrolling background. Each AddLayer call defines:

  • The image path relative to our assets directory
  • X and Y scroll speeds (smaller numbers = slower scrolling)

While there are more manual ways to create entities (which we'll explore later), the builder pattern used here provides a convenient shortcut for common setups.

Registering the Scene

Now that we have our scene defined, let's wire it up in our main file. We'll need to import our scenes package and the render systems from coldbrew:


// main.go

package main

import (
    "embed"
    "log"
    "github.com/TheBitDrifter/coldbrew"
    coldbrew_rendersystems "github.com/TheBitDrifter/coldbrew/rendersystems"
    "platformer/scenes"  // Import our scenes package
)

//go:embed assets/*
var assets embed.FS

const (
    RESOLUTION_X       = 640
    RESOLUTION_Y       = 360
    MAX_SPRITES_CACHED = 100
    MAX_SOUNDS_CACHED  = 100
    MAX_SCENES_CACHED  = 12
)

func main() {
    // Create the client
    client := coldbrew.NewClient(
        RESOLUTION_X,
        RESOLUTION_Y,
        MAX_SPRITES_CACHED,
        MAX_SOUNDS_CACHED,
        MAX_SCENES_CACHED,
        assets,
    )

    // Configure client settings
    client.SetTitle("Platformer")
    client.SetResizable(true)
    client.SetMinimumLoadTime(30)

    // Register scene One
    err := client.RegisterScene(
        scenes.SceneOne.Name,
        scenes.SceneOne.Width,
        scenes.SceneOne.Height,
        scenes.SceneOne.Plan,
        []coldbrew.RenderSystem{},
        []coldbrew.ClientSystem{},
        []blueprint.CoreSystem{},
    )
    if err != nil {
        log.Fatal(err)
    }

    // Register global systems
    client.RegisterGlobalRenderSystem(
        coldbrew_rendersystems.GlobalRenderer{},
    )

    // Activate the camera
    client.ActivateCamera()

    // Run the client
    if err := client.Start(); err != nil {
        log.Fatal(err)
    }
}

Note: For the coldbrew_rendersystems import you will likely need to run (again):

go mod tidy

When you run the game now, you should see your first scene with its background:

Basic Background

What's Happening Here?

We've made three crucial additions to get our scene running:

  1. Scene Registration: Using client.RegisterScene(), we register SceneOne with all its properties. For now, we're passing empty slices for our systems - we'll add those later as we build out game functionality.

  2. Global Renderer: The GlobalRenderer is a default rendering system provided by coldbrew that handles basic scene rendering. We register it using RegisterGlobalRenderSystem().

  3. Camera Activation: ActivateCamera() sets up our view into the game world. This is essential for seeing our parallax background in action.

With these pieces in place, our game now has its first visual elements!

Adding the Player

Now that we have a basic scene, let's add a playable character. First, we'll set up animations for our character's different states.

Setting Up Animations

Create an /animations directory in the project root and add the following file:


// animations/animations.go

package animations

import (
    blueprintclient "github.com/TheBitDrifter/blueprint/client"
    "github.com/TheBitDrifter/blueprint/vector"
)

var IdleAnimation = blueprintclient.AnimationData{
    Name:        "idle",
    RowIndex:    0,
    FrameCount:  6,
    FrameWidth:  144,
    FrameHeight: 116,
    Speed:       8,
}

var RunAnimation = blueprintclient.AnimationData{
    Name:        "run",
    RowIndex:    1,
    FrameCount:  8,
    FrameWidth:  144,
    FrameHeight: 116,
    Speed:       5,
}

var JumpAnimation = blueprintclient.AnimationData{
    Name:           "jump",
    RowIndex:       2,
    FrameCount:     3,
    FrameWidth:     144,
    FrameHeight:    116,
    Speed:          5,
    Freeze:         true,
    PositionOffset: vector.Two{X: 0, Y: 10},
}

var FallAnimation = blueprintclient.AnimationData{
    Name:           "fall",
    RowIndex:       3,
    FrameCount:     3,
    FrameWidth:     144,
    FrameHeight:    116,
    Speed:          5,
    Freeze:         true,
    PositionOffset: vector.Two{X: 0, Y: 10},
}

Each animation is defined by several key properties:

  • Name: Identifier for easier animation management
  • RowIndex: The row in the sprite sheet containing the animation frames
  • FrameCount: Number of frames in the animation
  • FrameWidth/Height: Dimensions of each frame
  • Speed: Ticks per animation frame
  • Freeze: When true, holds the last frame instead of looping
  • PositionOffset: Allows fine-tuning of the animation position

Creating the Player Entity

Now let's add a helper function to create our player. Add this to your scenes file:


// scenes/scenes.go

import (
 "platformer/animations"

 "github.com/TheBitDrifter/blueprint"
 "github.com/TheBitDrifter/blueprint/vector"
 "github.com/TheBitDrifter/warehouse"

 // New Imports:
 blueprintclient "github.com/TheBitDrifter/blueprint/client"
 blueprintinput "github.com/TheBitDrifter/blueprint/input"
 blueprintmotion "github.com/TheBitDrifter/blueprint/motion"
 blueprintspatial "github.com/TheBitDrifter/blueprint/spatial"
)
func NewPlayer(sto warehouse.Storage, x, y float64) error {
    // Get or create the archetype
    playerArchetype, err := sto.NewOrExistingArchetype(
        blueprintspatial.Components.Position,
        blueprintspatial.Components.Position,
        blueprintspatial.Components.Shape,
        blueprintspatial.Components.Direction,
        blueprintmotion.Components.Dynamics,
        blueprintinput.Components.InputBuffer,
        blueprintclient.Components.CameraIndex,
        blueprintclient.Components.SpriteBundle,
        blueprintclient.Components.SoundBundle,
    )

    // Position state
    playerPos := blueprintspatial.NewPosition(x, y)
    // Hitbox state
    playerHitbox := blueprintspatial.NewRectangle(18, 58)
    // Physics state
    playerDynamics := blueprintmotion.NewDynamics(10)
    // Basic Direction State
    playerDirection := blueprintspatial.NewDirectionRight()
    // Input state
    playerInputBuffer := blueprintinput.InputBuffer{ReceiverIndex: 0}
    // Camera Reference
    playerCameraIndex := blueprintclient.CameraIndex(0)
    // Sprite Reference
    playerSprites := blueprintclient.NewSpriteBundle().
        AddSprite("characters/box_man_sheet.png", true).
        WithAnimations(animations.IdleAnimation, animations.RunAnimation, animations.FallAnimation, animations.JumpAnimation).
        SetActiveAnimation(animations.IdleAnimation).
        WithOffset(vector.Two{X: -72, Y: -59}).
        WithPriority(20)

    // Generate the player
    err = playerArchetype.Generate(1,
        playerPos,
        playerHitbox,
        playerDynamics,
        playerDirection,
        playerInputBuffer,
        playerCameraIndex,
        playerSprites,
    )
    if err != nil {
        return err
    }
    return nil
}

Note: You may see a compiler warning about copying the atomic playerSprites field. This warning can be safely ignored as the copy only occurs during entity template instantiation. The template object is temporary and only used to initialize the entity's components. Once created, all runtime access to the entity's state is done through pointers, maintaining proper atomic field semantics.

Understanding the Player Components

Bappa uses an Archetypal ECS approach, where entities are created from archetypes (groups of components). Let's examine some key components:

  1. Input Buffer: The playerInputBuffer with ReceiverIndex: 0 connects to Coldbrew's first input receiver. For our single-player game, we'll use the first receiver.

  2. Camera Index: Similar to receivers, Coldbrew supports up to eight cameras. We use index 0 for our single camera setup.

  3. Sprite Bundle: The Blueprint API helps set up player sprites and animations. We:

    • Add the sprite sheet
    • Configure animations
    • Set the initial animation
    • Position the sprite relative to its hitbox
    • Set render priority (20 means it renders above lower-priority elements)

Adding the Player to Scene One

Finally, let's add the player to our scene:


// scenes/scene_one.go

func sceneOnePlan(height, width int, sto warehouse.Storage) error {
    // ...Existing background code...

    err = NewPlayer(sto, 100, 100)
    if err != nil {
        return err
    }
    return nil
}

Now when you run the game, you'll see our character idling in the corner:

Image description

Adding Basic Movement

Before implementing our full platformer physics, let's start with some basic movement functionality. We'll need to set up input actions and create some systems to handle movement and physics.

Setting Up Input Actions

First, let's create our action definitions:


// actions/actions.go

package actions

import (
    blueprintinput "github.com/TheBitDrifter/blueprint/input"
)

var (
    Left  = blueprintinput.NewInput()
    Right = blueprintinput.NewInput()
    Jump  = blueprintinput.NewInput()
    Down  = blueprintinput.NewInput()
)

Mapping Keys to Actions

Now we'll update our main file to map keyboard keys to our actions:


// main.go

package main

import (

    // New Import:
    coldbrew_clientsystems "github.com/TheBitDrifter/coldbrew/clientsystems"
)

func main() {
    // ... Existing code

    // Register receiver/actions
    receiver1, _ := client.ActivateReceiver()
    receiver1.RegisterKey(ebiten.KeySpace, actions.Jump)
    receiver1.RegisterKey(ebiten.KeyW, actions.Jump)
    receiver1.RegisterKey(ebiten.KeyA, actions.Left)
    receiver1.RegisterKey(ebiten.KeyD, actions.Right)
    receiver1.RegisterKey(ebiten.KeyS, actions.Down)

    // Default client systems for camera mapping and receiver mapping
    client.RegisterGlobalClientSystem(
        coldbrew_clientsystems.InputBufferSystem{},
        &coldbrew_clientsystems.CameraSceneAssignerSystem{},
    )

    // Run the client
    if err := client.Start(); err != nil {
        log.Fatal(err)
    }
}

Creating Core Systems

Let's create our first core systems. Create a coresystems/ directory with these files:


// coresystems/player_movement_system.go"

package coresystems

import (
    "platformer/actions"
    "github.com/TheBitDrifter/blueprint"
    blueprintinput "github.com/TheBitDrifter/blueprint/input"
    blueprintmotion "github.com/TheBitDrifter/blueprint/motion"
    blueprintspatial "github.com/TheBitDrifter/blueprint/spatial"
)

const (
    speed = 120.0
)

type PlayerMovementSystem struct{}

func (sys PlayerMovementSystem) Run(scene blueprint.Scene, dt float64) error {
    // Query all entities with input buffers (players)
    cursor := scene.NewCursor(blueprint.Queries.InputBuffer)

    for range cursor.Next() {
        dyn := blueprintmotion.Components.Dynamics.GetFromCursor(cursor)
        incomingInputs := blueprintinput.Components.InputBuffer.GetFromCursor(cursor)
        direction := blueprintspatial.Components.Direction.GetFromCursor(cursor)

        _, pressedLeft := incomingInputs.ConsumeInput(actions.Left)
        if pressedLeft {
            direction.SetLeft()
            dyn.Vel.X = -speed
        }

        _, pressedRight := incomingInputs.ConsumeInput(actions.Right)
        if pressedRight {
            direction.SetRight()
            dyn.Vel.X = speed
        }

        _, pressedUp := incomingInputs.ConsumeInput(actions.Jump)
        if pressedUp {
            dyn.Vel.Y = -speed
        }

        _, pressedDown := incomingInputs.ConsumeInput(actions.Down)
        if pressedDown {
            dyn.Vel.Y = speed
        }
    }
    return nil
}
// coresystems/friction_system.go

package coresystems

import (
    "github.com/TheBitDrifter/blueprint"
    blueprintmotion "github.com/TheBitDrifter/blueprint/motion"
    "github.com/TheBitDrifter/tteokbokki/motion"
)

const (
    DEFAULT_FRICTION = 0.5
    DEFAULT_DAMP     = 0.9
)

type FrictionSystem struct{}

func (FrictionSystem) Run(scene blueprint.Scene, dt float64) error {
    // Iterate through entities with dynamics components(physics)
    cursor := scene.NewCursor(blueprint.Queries.Dynamics)
    for range cursor.Next() {
        // Get the dynamics
        dyn := blueprintmotion.Components.Dynamics.GetFromCursor(cursor)
        friction := motion.Forces.Generator.NewHorizontalFrictionForce(dyn.Vel, DEFAULT_FRICTION)
        motion.Forces.AddForce(dyn, friction)

        motion.Forces.Generator.ApplyHorizontalDamping(dyn, DEFAULT_DAMP)
    }
    return nil
}

// coresystems/common.go

package coresystems

import (
    "github.com/TheBitDrifter/blueprint"
    tteo_coresystems "github.com/TheBitDrifter/tteokbokki/coresystems"
)

var DefaultCoreSystems = []blueprint.CoreSystem{
    FrictionSystem{},
    PlayerMovementSystem{},
    tteo_coresystems.IntegrationSystem{}, // Update velocities and positions
    tteo_coresystems.TransformSystem{},   // Update collision shapes
}

The systems work together to handle player movement:

  1. PlayerMovementSystem uses ConsumeInput() to check for our basic actions and updates player velocity and direction accordingly.

  2. FrictionSystem applies friction and damping forces so the player stops naturally. We're only applying horizontal friction for now, as gravity will handle vertical movement later.

  3. common.go bundles our systems with the default physics systems from Bappa for easy scene assignment.

Registering the Systems

Finally, let's update our scene registration to use these systems:


// main.go

import (
    "platformer/coresystems"
)

func main() {
    err := client.RegisterScene(
        scenes.SceneOne.Name,
        scenes.SceneOne.Width,
        scenes.SceneOne.Height,
        scenes.SceneOne.Plan,
        []coldbrew.RenderSystem{},
        []coldbrew.ClientSystem{},
        coresystems.DefaultCoreSystems,  // Register our core systems
    )
}

With these systems in place, you can now move the player using WASD keys! The player should move smoothly and come to a stop when you release the keys thanks to our friction system.

Image description

Note: In Bappa, you can also assign friction at the collision/surface level through the state in an Entity's Dynamics (physics) component. For this simple guide, however, we'll focus on using the force/global system approach instead.

Making the Camera Follow the Player

To make the camera follow our player, we'll need to create our first client system. Let's create a /clientsystems directory with two files:

Camera Follower System


// clientsystems/camera_follower_system.go

package clientsystems

import (
    "math"
    blueprintclient "github.com/TheBitDrifter/blueprint/client"
    blueprintinput "github.com/TheBitDrifter/blueprint/input"
    blueprintspatial "github.com/TheBitDrifter/blueprint/spatial"
    "github.com/TheBitDrifter/blueprint/vector"
    "github.com/TheBitDrifter/coldbrew"
    "github.com/TheBitDrifter/warehouse"
)

type CameraFollowerSystem struct{}

func (CameraFollowerSystem) Run(cli coldbrew.LocalClient, scene coldbrew.Scene) error {
    // Query players who have a camera (camera index component)
    playersWithCamera := warehouse.Factory.NewQuery()
    playersWithCamera.And(
        blueprintspatial.Components.Position,
        blueprintinput.Components.InputBuffer,
        blueprintclient.Components.CameraIndex,
    )
    playerCursor := scene.NewCursor(playersWithCamera)

    for range playerCursor.Next() {
        // Get the players position
        playerPos := blueprintspatial.Components.Position.GetFromCursor(playerCursor)
        // Get the players camera
        camIndex := int(*blueprintclient.Components.CameraIndex.GetFromCursor(playerCursor))
        cam := cli.Cameras()[camIndex]
        // Get the cameras local scene position
        _, cameraScenePosition := cam.Positions()

        // Center camera directly on player (offset by half camera size)
        cameraScenePosition.X = math.Round(playerPos.X - float64(cam.Surface().Bounds().Dx())/2)
        cameraScenePosition.Y = math.Round(playerPos.Y - float64(cam.Surface().Bounds().Dy())/2)

        // Lock the camera to the scene boundaries
        lockCameraToSceneBoundaries(cam, scene, cameraScenePosition)
    }
    return nil
}

// lockCameraToSceneBoundaries constrains camera position within scene boundaries
func lockCameraToSceneBoundaries(cam coldbrew.Camera, scene coldbrew.Scene, cameraPos *vector.Two) {
    sceneWidth := scene.Width()
    sceneHeight := scene.Height()
    camWidth, camHeight := cam.Dimensions()

    // Calculate maximum positions to keep camera within scene bounds
    maxX := sceneWidth - camWidth
    maxY := sceneHeight - camHeight

    // Constrain camera X position
    if cameraPos.X > float64(maxX) {
        cameraPos.X = float64(maxX)
    }
    if cameraPos.X < 0 {
        cameraPos.X = 0
    }

    // Constrain camera Y position
    if cameraPos.Y > float64(maxY) {
        cameraPos.Y = float64(maxY)
    }
    if cameraPos.Y < 0 {
        cameraPos.Y = 0
    }
}

Default Client Systems


// clientsystems/common.go

package clientsystems

import (
    "github.com/TheBitDrifter/coldbrew"
    coldbrew_clientsystems "github.com/TheBitDrifter/coldbrew/clientsystems"
)

var DefaultClientSystems = []coldbrew.ClientSystem{
    &CameraFollowerSystem{},
    &coldbrew_clientsystems.BackgroundScrollSystem{},
}

How the Camera System Works

The CameraFollowerSystem does several important things:

  1. Queries for players that have a camera index component
  2. Gets the player's position and associated camera
  3. Centers the camera on the player by offsetting it by half the camera's dimensions
  4. Uses lockCameraToSceneBoundaries to keep the camera within the level boundaries

In common.go, we bundle our camera system with the BackgroundScrollSystem. This default system works with the GlobalRenderer and our previously defined parallax speeds to create smooth background scrolling as the camera moves.

Updating the Scene Registration

Finally, let's update our scene registration in main.go to use these systems:


// main.go

func main() {
    // ...Existing code
    err := client.RegisterScene(
        scenes.SceneOne.Name,
        scenes.SceneOne.Width,
        scenes.SceneOne.Height,
        scenes.SceneOne.Plan,
        []coldbrew.RenderSystem{},
        clientsystems.DefaultClientSystems,  // Add our client systems
        coresystems.DefaultCoreSystems,
    )
}

Now when you run the game, the camera will smoothly follow the player while maintaining the parallax background effect! The camera will stay within the scene boundaries even if the player moves beyond them.

Image description

Adding Basic Collisions

Now that we can move our player around, let's implement collision detection. We'll start by creating different types of terrain that the player can collide with.

Creating Terrain Tags

First, let's create component tags to differentiate between terrain types. Create a /components directory:

// components/tags.go

package components

import "github.com/TheBitDrifter/warehouse"

var (
    BlockTerrainTag = warehouse.FactoryNewComponent[struct{}]()
    PlatformTag     = warehouse.FactoryNewComponent[struct{}]()
)

Adding Terrain Creation Helpers

Next, let's add helper functions to create different types of terrain:


// scenes/scene.go

func NewFloor(sto warehouse.Storage, y float64) error {
    terrainArchetype, err := sto.NewOrExistingArchetype(
        blueprintclient.Components.SpriteBundle,
        components.BlockTerrainTag,
        blueprintspatial.Components.Shape,
        blueprintspatial.Components.Position,
        blueprintmotion.Components.Dynamics,
    )
    if err != nil {
        return err
    }
    return terrainArchetype.Generate(1,
        blueprintspatial.NewPosition(1500, y),
        blueprintspatial.NewRectangle(4000, 50),
        blueprintclient.NewSpriteBundle().
            AddSprite("terrain/floor.png", true).
            WithOffset(vector.Two{X: -1500, Y: -25}),
    )
}

func NewInvisibleWalls(sto warehouse.Storage, width, height int) error {
    terrainArchetype, err := sto.NewOrExistingArchetype(
        blueprintclient.Components.SpriteBundle,
        components.BlockTerrainTag,
        blueprintspatial.Components.Shape,
        blueprintspatial.Components.Position,
        blueprintmotion.Components.Dynamics,
    )
    if err != nil {
        return err
    }

    // Wall left (invisible)
    err = terrainArchetype.Generate(1,
        blueprintspatial.NewRectangle(10, float64(height+300)),
        blueprintspatial.NewPosition(0, 0),
    )
    if err != nil {
        return err
    }

    // Wall right (invisible)
    return terrainArchetype.Generate(1,
        blueprintspatial.NewRectangle(10, float64(height+300)),
        blueprintspatial.NewPosition(float64(width), 0),
    )
}

func NewBlock(sto warehouse.Storage, x, y float64) error {
    terrainArchetype, err := sto.NewOrExistingArchetype(
        blueprintclient.Components.SpriteBundle,
        components.BlockTerrainTag,
        blueprintspatial.Components.Shape,
        blueprintspatial.Components.Position,
        blueprintmotion.Components.Dynamics,
    )
    if err != nil {
        return err
    }
    return terrainArchetype.Generate(1,
        blueprintspatial.NewPosition(x, y),
        blueprintspatial.NewRectangle(64, 75),
        blueprintclient.NewSpriteBundle().
            AddSprite("terrain/block.png", true).
            WithOffset(vector.Two{X: -33, Y: -38}),
    )
}

Updating the Scene

Now let's add terrain to our scene:


// scenes/scene_one.go

func sceneOnePlan(height, width int, sto warehouse.Storage) error {
    // ... Existing code ...

    err = NewInvisibleWalls(sto, width, height)
    if err != nil {
        return err
    }

    err = NewBlock(sto, 285, 390)
    if err != nil {
        return err
    }

    err = NewFloor(sto, 460)
    if err != nil {
        return err
    }
    return nil
}

Note: While creating levels programmatically works for this tutorial, it quickly becomes tedious for larger projects. For more efficient level design, consider using LDTK (Level Designer Toolkit) with Bappa's LDTK integration. This combination provides a visual editor and streamlined workflow for creating complex game levels. The bappa-create platformer template offers a ready-to-use example implementation.

Implementing Collision Detection

Create a new collision system:


// coresystems/player_block_collision_system.go

package coresystems

import (
    "platformer/components"
    "github.com/TheBitDrifter/blueprint"
    blueprintmotion "github.com/TheBitDrifter/blueprint/motion"
    blueprintspatial "github.com/TheBitDrifter/blueprint/spatial"
    "github.com/TheBitDrifter/tteokbokki/motion"
    "github.com/TheBitDrifter/tteokbokki/spatial"
    "github.com/TheBitDrifter/warehouse"
)

type PlayerBlockCollisionSystem struct{}

func (s PlayerBlockCollisionSystem) Run(scene blueprint.Scene, dt float64) error {
    // Create cursors
    blockTerrainQuery := warehouse.Factory.NewQuery().And(components.BlockTerrainTag)
    blockTerrainCursor := scene.NewCursor(blockTerrainQuery)
    playerCursor := scene.NewCursor(blueprint.Queries.InputBuffer)

    // Outer loop is blocks
    for range blockTerrainCursor.Next() {
        // Inner is players
        for range playerCursor.Next() {
            err := s.resolve(scene, blockTerrainCursor, playerCursor)
            if err != nil {
                return err
            }
        }
    }
    return nil
}

func (PlayerBlockCollisionSystem) resolve(scene blueprint.Scene, blockCursor, playerCursor *warehouse.Cursor) error {
    // Get the player pos, shape, and dynamics
    playerPosition := blueprintspatial.Components.Position.GetFromCursor(playerCursor)
    playerShape := blueprintspatial.Components.Shape.GetFromCursor(playerCursor)
    playerDynamics := blueprintmotion.Components.Dynamics.GetFromCursor(playerCursor)

    // Get the block pos, shape, and dynamics
    blockPosition := blueprintspatial.Components.Position.GetFromCursor(blockCursor)
    blockShape := blueprintspatial.Components.Shape.GetFromCursor(blockCursor)
    blockDynamics := blueprintmotion.Components.Dynamics.GetFromCursor(blockCursor)

    // Check for a collision
    if ok, collisionResult := spatial.Detector.Check(
        *playerShape, *blockShape, playerPosition.Two, blockPosition.Two,
    ); ok {
        // Resolve collision
        motion.Resolver.Resolve(
            &playerPosition.Two,
            &blockPosition.Two,
            playerDynamics,
            blockDynamics,
            collisionResult,
        )
    }
    return nil
}

Registering the Collision System

Update the default core systems:


// coresystems/common.go

var DefaultCoreSystems = []blueprint.CoreSystem{
    FrictionSystem{},
    PlayerMovementSystem{},
    tteo_coresystems.IntegrationSystem{}, // Update velocities and positions
    tteo_coresystems.TransformSystem{},   // Update collision shapes
    PlayerBlockCollisionSystem{},         // Add collision detection
}

Adding Debug Visualization

Finally, let's add the debug renderer to visualize hitboxes:


// main.go

func main() {
    // ... Existing code ...

    client.RegisterGlobalRenderSystem(
        coldbrew_rendersystems.GlobalRenderer{},
        &coldbrew_rendersystems.DebugRenderer{}, // Add debug visualization
    )
}

You can now toggle hitbox visualization by pressing the 0 key while the game is running. This is incredibly useful for debugging collision issues!

Basic Collisions

Deep Dive: Collision Architecture (optional section)

Let's explore how the collision system we just built works in detail.

Component Tags and ECS

The Bappa Framework uses an Archetypal ECS (Entity Component System) pattern. Our terrain tag system demonstrates this:

// components/tags.go

var (
    BlockTerrainTag = warehouse.FactoryNewComponent[struct{}]()
    PlatformTag     = warehouse.FactoryNewComponent[struct{}]()
)

These tags are empty struct components that act as markers. They allow us to:

  1. Differentiate between terrain types (blocks vs platforms)
  2. Query for specific entities efficiently
  3. Keep our collision logic separated by type

Terrain Archetypes

When we create terrain, we're defining a collection of components that make up that entity type:

terrainArchetype, err := sto.NewOrExistingArchetype(
    blueprintclient.Components.SpriteBundle,    // Visual representation
    components.BlockTerrainTag,                 // Type identifier
    blueprintspatial.Components.Shape,          // Collision shape
    blueprintspatial.Components.Position,       // World position
    blueprintmotion.Components.Dynamics,        // Physics properties
)

This archetype definition tells Bappa that our terrain entities need:

  • Visual representation (sprites)
  • Type identification (our custom tag)
  • Physical properties (shape, position, dynamics)

Collision Detection Flow

Our collision system works in multiple stages:

  1. Query Phase: We find relevant entities using component queries:
blockTerrainQuery := warehouse.Factory.NewQuery().And(components.BlockTerrainTag)
blockTerrainCursor := scene.NewCursor(blockTerrainQuery)
playerCursor := scene.NewCursor(blueprint.Queries.InputBuffer)
  1. Check Phase: For each potential collision pair, we check for intersection:
if ok, collisionResult := spatial.Detector.Check(
    *playerShape, *blockShape, playerPosition.Two, blockPosition.Two,
); ok {
    // Collision detected!
}
  1. Resolution Phase: When a collision is detected, we resolve it:
motion.Resolver.Resolve(
    &playerPosition.Two,
    &blockPosition.Two,
    playerDynamics,
    blockDynamics,
    collisionResult,
)

Debug Visualization

The debug renderer we added is particularly useful because it lets us see the collision shapes:

client.RegisterGlobalRenderSystem(
    coldbrew_rendersystems.GlobalRenderer{},
    &coldbrew_rendersystems.DebugRenderer{},
)

When enabled (by pressing 0), it shows:

  • Hitbox boundaries for all entities
  • Position markers
  • Collision shapes

This helps us:

  1. Verify collision shapes match their sprites
  2. Debug collision detection issues
  3. Understand how entities interact physically

System Ordering

Notice how we placed the collision system last in our core systems:

var DefaultCoreSystems = []blueprint.CoreSystem{
    FrictionSystem{},
    PlayerMovementSystem{},
    tteo_coresystems.IntegrationSystem{},
    tteo_coresystems.TransformSystem{},
    PlayerBlockCollisionSystem{},  // Last!
}

This order is important because:

  1. Player movement updates velocities
  2. Integration system applies those velocities
  3. Transform system updates collision shapes
  4. Finally, we detect and resolve any collisions

If we put collision detection earlier, we might miss collisions that occur due to movement in the current frame!

Proper Movement and Gravity

Now that we have working collisions, it's time to add gravity and create a 'proper' platforming movement system. Let's begin by adding a new core gravity system:

Implementing Gravity


// coresystems/gravity_system.go

package coresystems

import (
    "github.com/TheBitDrifter/blueprint"
    blueprintmotion "github.com/TheBitDrifter/blueprint/motion"
    "github.com/TheBitDrifter/tteokbokki/motion"
)

const (
    DEFAULT_GRAVITY  = 9.8
    PIXELS_PER_METER = 50.0
)

type GravitySystem struct{}

func (GravitySystem) Run(scene blueprint.Scene, dt float64) error {
    // Iterate through entities with dynamics components(physics)
    cursor := scene.NewCursor(blueprint.Queries.Dynamics)
    for range cursor.Next() {
        // Get the dynamics
        dyn := blueprintmotion.Components.Dynamics.GetFromCursor(cursor)

        // Get the mass
        mass := 1 / dyn.InverseMass

        // Use the motion package to calc the gravity force
        gravity := motion.Forces.Generator.NewGravityForce(mass, DEFAULT_GRAVITY, PIXELS_PER_METER)

        // Apply the force
        motion.Forces.AddForce(dyn, gravity)
    }
    return nil
}

Now update the default core systems in coresystems/common.go:

// coresystems/common.go

var DefaultCoreSystems = []blueprint.CoreSystem{
    GravitySystem{}, // Add gravity system
    FrictionSystem{},
    PlayerMovementSystem{},
    tteo_coresystems.IntegrationSystem{},
    tteo_coresystems.TransformSystem{},
    PlayerBlockCollisionSystem{},
}

Creating the Grounded State

Now that we have gravity enabled, it's time to update the movement system. While the horizontal movement is fine, we need to introduce the concept of jumping. People can't jump when they're not on top of things, right? So in order to add jumping mechanics, we need to start tracking if the player is grounded or not. Let's begin by introducing a custom OnGround component:


// components/components.go

package components

import "github.com/TheBitDrifter/warehouse"

type OnGround struct {
    Landed, LastTouch int
}

var OnGroundComponent = warehouse.FactoryNewComponent[OnGround]()

Detecting Ground Contact

Now we need to update the PlayerBlockCollisionSystem to use this component:

// coresystems/player_block_collision_system.go

// ...Existing code

func (PlayerBlockCollisionSystem) resolve(scene blueprint.Scene, blockCursor, playerCursor *warehouse.Cursor) error {
    // Get the player pos, shape, and dynamics
    playerPosition := blueprintspatial.Components.Position.GetFromCursor(playerCursor)
    playerShape := blueprintspatial.Components.Shape.GetFromCursor(playerCursor)
    playerDynamics := blueprintmotion.Components.Dynamics.GetFromCursor(playerCursor)

    // Get the block pos, shape, and dynamics
    blockPosition := blueprintspatial.Components.Position.GetFromCursor(blockCursor)
    blockShape := blueprintspatial.Components.Shape.GetFromCursor(blockCursor)
    blockDynamics := blueprintmotion.Components.Dynamics.GetFromCursor(blockCursor)

    // Check for a collision
    if ok, collisionResult := spatial.Detector.Check(
        *playerShape, *blockShape, playerPosition.Two, blockPosition.Two,
    ); ok {
        // Otherwise resolve as normal
        motion.Resolver.Resolve(
            &playerPosition.Two,
            &blockPosition.Two,
            playerDynamics,
            blockDynamics,
            collisionResult,
        )

        // Add ground handling here:
        currentTick := scene.CurrentTick()
        playerAlreadyGrounded, onGround := components.OnGroundComponent.GetFromCursorSafe(playerCursor)

        // Update onGround accordingly (create or update)
        if !playerAlreadyGrounded {
            playerEntity, err := playerCursor.CurrentEntity()
            if err != nil {
                return err
            }
            // We cannot mutate during a cursor iteration, so we use the enqueue API
            err = playerEntity.EnqueueAddComponentWithValue(
                components.OnGroundComponent,
                components.OnGround{LastTouch: currentTick, Landed: currentTick},
            )
            if err != nil {
                return err
            }
        } else {
            onGround.LastTouch = currentTick
        }
    }
    return nil
}

Some things to note here:

  1. We use the GetFromCursorSafe API to check and safely access the OnGround component.
  2. We cannot mutate an entity's composition while iterating, so we leverage the Enqueue API.

Next up we need a clearing system to remove the OnGround component. We could do it inside the collision system, but I prefer a dedicated approach:


// coresystems/onground_clearing_system.go

package coresystems

import (
    "platformer/components"

    "github.com/TheBitDrifter/blueprint"
    "github.com/TheBitDrifter/warehouse"
)

type OnGroundClearingSystem struct{}

func (OnGroundClearingSystem) Run(scene blueprint.Scene, dt float64) error {
    const expirationTicks = 15

    onGroundQuery := warehouse.Factory.NewQuery().And(components.OnGroundComponent)
    onGroundCursor := scene.NewCursor(onGroundQuery)

    // Iterate through matched entities
    for range onGroundCursor.Next() {
        onGround := components.OnGroundComponent.GetFromCursor(onGroundCursor)

        // If it's expired, remove it
        if scene.CurrentTick()-onGround.LastTouch > expirationTicks {
            groundedEntity, _ := onGroundCursor.CurrentEntity()

            // We can't mutate while iterating so we enqueue the changes instead
            err := groundedEntity.EnqueueRemoveComponent(components.OnGroundComponent)
            if err != nil {
                return err
            }
        }
    }
    return nil
}

Update the core systems again to include our new clearing system:


// coresystems/common.go

var DefaultCoreSystems = []blueprint.CoreSystem{
    GravitySystem{},
    FrictionSystem{},
    PlayerMovementSystem{},
    tteo_coresystems.IntegrationSystem{},
    tteo_coresystems.TransformSystem{},
    PlayerBlockCollisionSystem{},
    OnGroundClearingSystem{}, // Add OnGroundClearingSystem
}

Adding Jump Mechanics

Now we can finally update the PlayerMovementSystem with ground tracking and jumping mechanics:


// coresystems/player_movement_system.go

package coresystems

import (
    "platformer/actions"
    "platformer/components"

    "github.com/TheBitDrifter/blueprint"
    blueprintinput "github.com/TheBitDrifter/blueprint/input"
    blueprintmotion "github.com/TheBitDrifter/blueprint/motion"
    blueprintspatial "github.com/TheBitDrifter/blueprint/spatial"
)

const (
    speed      = 120.0
    jumpforce  = 220.0  // Add jump force constant
)

type PlayerMovementSystem struct{}

func (sys PlayerMovementSystem) Run(scene blueprint.Scene, dt float64) error {
    // Query all entities with input buffers (players)
    cursor := scene.NewCursor(blueprint.Queries.InputBuffer)

    for range cursor.Next() {
        dyn := blueprintmotion.Components.Dynamics.GetFromCursor(cursor)
        incomingInputs := blueprintinput.Components.InputBuffer.GetFromCursor(cursor)
        direction := blueprintspatial.Components.Direction.GetFromCursor(cursor)
        isGrounded := components.OnGroundComponent.CheckCursor(cursor)  // Check if player is on ground

        _, pressedLeft := incomingInputs.ConsumeInput(actions.Left)
        if pressedLeft {
            direction.SetLeft()
            dyn.Vel.X = -speed
        }

        _, pressedRight := incomingInputs.ConsumeInput(actions.Right)
        if pressedRight {
            direction.SetRight()
            dyn.Vel.X = speed
        }

        // Only allow jumping when grounded
        _, pressedUp := incomingInputs.ConsumeInput(actions.Jump)
        if pressedUp && isGrounded {
            dyn.Vel.Y = -jumpforce
        }

        // Handle down press (we'll implement platform dropping later)
        _, _ = incomingInputs.ConsumeInput(actions.Down)
    }
    return nil
}

Fixing Bugs — Corner Snapping and Side Wall Jumping

While our new movement system might seem solid at first glance, there are some issues that need to be worked out. There are two primary issues to fix:

  1. Corner Snapping — When hugging the corner and jumping, the player can trigger a collision with their bottom face and the top face of terrain despite having upwards velocity. This confuses the resolver and creates a sticky or snapping effect.

  2. Side Wall Jumping — Currently the collision system marks the player as grounded for ANY collision, not just collisions with the top of objects. While some games do provide wall jumping, with our current implementation, the player can accumulate massive amounts of Y velocity by repeatedly jumping against the sides of terrain.

Bug Jumping Example

Fortunately, these issues are easy to fix:


// coresystems/player_block_collision_system.go

// ...Existing code

func (PlayerBlockCollisionSystem) resolve(scene blueprint.Scene, blockCursor, playerCursor *warehouse.Cursor) error {
    // Get the player pos, shape, and dynamics
    playerPosition := blueprintspatial.Components.Position.GetFromCursor(playerCursor)
    playerShape := blueprintspatial.Components.Shape.GetFromCursor(playerCursor)
    playerDynamics := blueprintmotion.Components.Dynamics.GetFromCursor(playerCursor)

    // Get the block pos, shape, and dynamics
    blockPosition := blueprintspatial.Components.Position.GetFromCursor(blockCursor)
    blockShape := blueprintspatial.Components.Shape.GetFromCursor(blockCursor)
    blockDynamics := blueprintmotion.Components.Dynamics.GetFromCursor(blockCursor)

    // Check for a collision
    if ok, collisionResult := spatial.Detector.Check(
        *playerShape, *blockShape, playerPosition.Two, blockPosition.Two,
    ); ok {

        // Determine collision surfaces
        playerOnTopOfBlock := collisionResult.IsTopB()
        blockOnTopOfPlayer := collisionResult.IsTop()

        // Prevents snapping on AAB corner transitions/collisions
        if playerOnTopOfBlock && playerDynamics.Vel.Y < 0 {
            return nil
        }
        if blockOnTopOfPlayer && playerDynamics.Vel.Y > 0 {
            return nil
        }

        // Otherwise resolve as normal
        motion.Resolver.Resolve(
            &playerPosition.Two,
            &blockPosition.Two,
            playerDynamics,
            blockDynamics,
            collisionResult,
        )

        // Only set player as grounded when they're on top of a block
        if !playerOnTopOfBlock {
            return nil
        }

        currentTick := scene.CurrentTick()
        playerAlreadyGrounded, onGround := components.OnGroundComponent.GetFromCursorSafe(playerCursor)
        // Update onGround accordingly (create or update)
        if !playerAlreadyGrounded {
            playerEntity, err := playerCursor.CurrentEntity()
            if err != nil {
                return err
            }
            // We cannot mutate during a cursor iteration, so we use the enqueue API
            err = playerEntity.EnqueueAddComponentWithValue(
                components.OnGroundComponent,
                components.OnGround{LastTouch: currentTick, Landed: currentTick},
            )
            if err != nil {
                return err
            }
        } else {
            onGround.LastTouch = currentTick
        }
    }
    return nil
}

By checking the collision data and the player's velocity, we fix both issues:

  1. We prevent corner snapping in both directions by:
  • Skipping collision resolution when the player is moving upward (negative Y velocity) while colliding with the top of a block
  • Skipping collision resolution when the player is moving downward (positive Y velocity) while a block is on top of the player

This prevents the player from getting stuck on corners during both upward jumps and downward slides.

  1. We only set the player as grounded when they're actually on top of a block by checking playerOnTopOfBlock before updating the OnGround component.

With these changes, our platformer movement feels more natural and avoids common collision pitfalls.

Image description

Adding One Way Platforms

With block style terrain working, the next challenge is one-way platforms. What makes it especially tricky is that we're working in a discrete system, but the solution to the platform problem is inherently continuous. The crux of the issue is that you can actually enter a top-bottom/player-platform collision from a previous position of the side, bottom, or top. It's only a valid collision if the player was coming from the top, but if you only have the current frame data you're left unable to determine this.

So our solution is to introduce a tiny bit of state to the PlayerPlatformCollisionSystem to track the player's last n positions. We can then check if they cleared the platform recently enough for it to be considered valid. Some would argue that the system should not contain state in a proper ECS. I don't disagree, but sometimes when it's simple and easy enough, I don't mind breaking the rules to solve the problem in a more straightforward way.

Creating Platform Entities

To get started let's define a helper function to create the platforms:


// scenes/scene.go

// ...Existing code

func NewPlatform(sto warehouse.Storage, x, y float64) error {
    platformArche, err := sto.NewOrExistingArchetype(
        components.PlatformTag,
        blueprintclient.Components.SpriteBundle,
        blueprintspatial.Components.Shape,
        blueprintspatial.Components.Position,
        blueprintmotion.Components.Dynamics,
    )
    if err != nil {
        return err
    }
    return platformArche.Generate(1,
        blueprintspatial.NewPosition(x, y),
        blueprintspatial.NewTriangularPlatform(144, 16),
        blueprintclient.NewSpriteBundle().
            AddSprite("terrain/platform.png", true).
            WithOffset(vector.Two{X: -72, Y: -8}),
    )
}

Adding Platforms to the Scene

And now we can place some platforms:

// scenes/scene_one.go

func sceneOnePlan(height, width int, sto warehouse.Storage) error {
    // ...Existing code

    err = NewPlatform(sto, 130, 350)
    if err != nil {
        return err
    }
    err = NewPlatform(sto, 220, 270)
    if err != nil {
        return err
    }
    err = NewPlatform(sto, 320, 170)
    if err != nil {
        return err
    }
    err = NewPlatform(sto, 420, 300)
    if err != nil {
        return err
    }
    return nil
}

Implementing Platform Collision System

Finally let's write the system that handles collision with platforms:


// coresystems/player_platform_collision_system.go

package coresystems

import (
    "platformer/components"

    "github.com/TheBitDrifter/blueprint"
    blueprintmotion "github.com/TheBitDrifter/blueprint/motion"
    blueprintspatial "github.com/TheBitDrifter/blueprint/spatial"
    "github.com/TheBitDrifter/blueprint/vector"
    "github.com/TheBitDrifter/tteokbokki/motion"
    "github.com/TheBitDrifter/tteokbokki/spatial"
    "github.com/TheBitDrifter/warehouse"
)

type PlayerPlatformCollisionSystem struct {
    playerLastPositions []vector.Two
    maxPositionsToTrack int
}

func NewPlayerPlatformCollisionSystem() *PlayerPlatformCollisionSystem {
    trackCount := 15 // higher count == more tunneling protection == higher cost
    return &PlayerPlatformCollisionSystem{
        playerLastPositions: make([]vector.Two, 0, trackCount),
        maxPositionsToTrack: trackCount,
    }
}

func (s *PlayerPlatformCollisionSystem) Run(scene blueprint.Scene, dt float64) error {
    platformTerrainQuery := warehouse.Factory.NewQuery().And(components.PlatformTag)
    platformCursor := scene.NewCursor(platformTerrainQuery)
    playerCursor := scene.NewCursor(blueprint.Queries.InputBuffer)

    for range platformCursor.Next() {
        for range playerCursor.Next() {
            err := s.resolve(scene, platformCursor, playerCursor)
            if err != nil {
                return err
            }
            playerPos := blueprintspatial.Components.Position.GetFromCursor(playerCursor)
            s.trackPosition(playerPos.Two)
        }
    }
    return nil
}

func (s *PlayerPlatformCollisionSystem) resolve(scene blueprint.Scene, platformCursor, playerCursor *warehouse.Cursor) error {
    // Get the player state
    playerShape := blueprintspatial.Components.Shape.GetFromCursor(playerCursor)
    playerPosition := blueprintspatial.Components.Position.GetFromCursor(playerCursor)
    playerDynamics := blueprintmotion.Components.Dynamics.GetFromCursor(playerCursor)

    // Get the platform state
    platformShape := blueprintspatial.Components.Shape.GetFromCursor(platformCursor)
    platformPosition := blueprintspatial.Components.Position.GetFromCursor(platformCursor)
    platformDynamics := blueprintmotion.Components.Dynamics.GetFromCursor(platformCursor)

    // Check for collision
    if ok, collisionResult := spatial.Detector.Check(
        *playerShape, *platformShape, playerPosition.Two, platformPosition.Two,
    ); ok {

        // Check if any of the past player positions indicate the player was above the platform
        platformTop := platformShape.Polygon.WorldVertices[0].Y

        playerWasAbove := s.checkAnyPlayerPositionWasAbove(platformTop, playerShape.LocalAAB.Height)

        // We only want to resolve collisions when:
        // 1. The player is falling (vel.Y > 0)
        // 2. The collision is with the top of the platform
        // 3. The player was above the platform at some point (within n ticks)
        if playerDynamics.Vel.Y > 0 && collisionResult.IsTopB() && playerWasAbove {

            motion.Resolver.Resolve(
                &playerPosition.Two,
                &platformPosition.Two,
                playerDynamics,
                platformDynamics,
                collisionResult,
            )

            // Standard onGround handling
            currentTick := scene.CurrentTick()

            // If not grounded, enqueue onGround with values
            playerAlreadyGrounded, onGround := components.OnGroundComponent.GetFromCursorSafe(playerCursor)

            if !playerAlreadyGrounded {
                playerEntity, _ := playerCursor.CurrentEntity()
                err := playerEntity.EnqueueAddComponentWithValue(
                    components.OnGroundComponent,
                    components.OnGround{LastTouch: currentTick, Landed: currentTick},
                )
                if err != nil {
                    return err
                }
            } else {
                onGround.LastTouch = currentTick
            }
        }
    }
    return nil
}

// trackPosition adds a position to the history and ensures only the last N are kept
func (s *PlayerPlatformCollisionSystem) trackPosition(pos vector.Two) {
    // Add the new position
    s.playerLastPositions = append(s.playerLastPositions, pos)

    // If we've exceeded our max, remove the oldest position
    if len(s.playerLastPositions) > s.maxPositionsToTrack {
        s.playerLastPositions = s.playerLastPositions[1:]
    }
}

// checkAnyPlayerPositionWasAbove checks if the player was above a non-rotated platform in any historical position
func (s *PlayerPlatformCollisionSystem) checkAnyPlayerPositionWasAbove(platformTop float64, playerHeight float64) bool {
    if len(s.playerLastPositions) == 0 {
        return false
    }

    // Check all stored positions to see if the player was above in any of them
    for _, pos := range s.playerLastPositions {
        playerBottom := pos.Y + playerHeight/2
        if playerBottom <= platformTop {
            return true // Found at least one position where player was above
        }
    }

    return false
}

Registering the System

Now let's add our platform collision system to the main game loop:

// ...Existing code

var DefaultCoreSystems = []blueprint.CoreSystem{
    GravitySystem{},
    FrictionSystem{},
    PlayerMovementSystem{},
    tteo_coresystems.IntegrationSystem{},
    tteo_coresystems.TransformSystem{},
    PlayerBlockCollisionSystem{},

    // Added with function here:
    NewPlayerPlatformCollisionSystem(),
    OnGroundClearingSystem{},
}

How the Platform System Works

If we take a look at the PlayerPlatformCollision system, it's basically the same as the PlayerBlockCollisionSystem except it uses the helper
function (checkAnyPlayerPositionWasAbove()) to determine valid collisions. Furthermore, we define then use the NewPlayerPlatformCollisionSystem()
in coresystems/common.go. This is significant because this method returns a pointer system. This system must be passed by reference because it
now has internal state for historical player positions.

Image description

Descend Platforms

Okay for our last movement based feature, let's add the functionality to descend platforms. First we need to introduce a IgnorePlatform component:

Creating the Ignore Platform Component


// components/components.go

// ... Existing code

type IgnorePlatform struct {
    Items [5]struct {
        LastActive int
        EntityID   int
        Recycled   int
    }
}

var IgnorePlatformComponent = warehouse.FactoryNewComponent[IgnorePlatform]()

Updating Player Movement for Drop-Through

Now we can update the PlayerMovementSystem:


// coresystems/player_movement_system.go

// ...Existing code

func (sys PlayerMovementSystem) Run(scene blueprint.Scene, dt float64) error {
    // Query all entities with input buffers (players)
    cursor := scene.NewCursor(blueprint.Queries.InputBuffer)

    for range cursor.Next() {
        dyn := blueprintmotion.Components.Dynamics.GetFromCursor(cursor)
        incomingInputs := blueprintinput.Components.InputBuffer.GetFromCursor(cursor)
        direction := blueprintspatial.Components.Direction.GetFromCursor(cursor)
        isGrounded := components.OnGroundComponent.CheckCursor(cursor)

        _, pressedLeft := incomingInputs.ConsumeInput(actions.Left)
        if pressedLeft {
            direction.SetLeft()
            dyn.Vel.X = -speed
        }

        _, pressedRight := incomingInputs.ConsumeInput(actions.Right)
        if pressedRight {
            direction.SetRight()

            dyn.Vel.X = speed
        }
        _, pressedUp := incomingInputs.ConsumeInput(actions.Jump)
        if pressedUp && isGrounded {
            dyn.Vel.Y = -jumpforce
        }

        // Add down handling here:
        _, pressedDown := incomingInputs.ConsumeInput(actions.Down)
        if pressedDown && !pressedUp { // <- you cant drop and jump same tick
            playerEntity, _ := cursor.CurrentEntity()
            err := playerEntity.EnqueueAddComponent(components.IgnorePlatformComponent)
            if err != nil {
                return err
            }
        }
    }
    return nil
}

Creating the Ignore Platform Clearing System

And we're gonna need another clearing system for this new component:


// coresystems/ignore_platform_clearing_system.go

package coresystems

import (
    "platformer/components"

    "github.com/TheBitDrifter/blueprint"
    "github.com/TheBitDrifter/warehouse"
)

type IgnorePlatformClearingSystem struct{}

func (IgnorePlatformClearingSystem) Run(scene blueprint.Scene, dt float64) error {
    ignorePlatformQuery := warehouse.Factory.NewQuery().And(components.IgnorePlatformComponent)
    ignorePlatformCursor := scene.NewCursor(ignorePlatformQuery)

    const expirationTicks = 15

    for range ignorePlatformCursor.Next() {

        ignorePlatform := components.IgnorePlatformComponent.GetFromCursor(ignorePlatformCursor)
        currentTick := scene.CurrentTick()

        // Track if we have any active ignores left
        anyActive := false

        // Check each ignore entry
        for i := range ignorePlatform.Items {
            // Skip already cleared entries
            if ignorePlatform.Items[i].EntityID == 0 {
                continue
            }

            // Check if this entry has expired
            if currentTick-ignorePlatform.Items[i].LastActive > expirationTicks {
                // Clear this specific entry by setting its EntityID to 0
                ignorePlatform.Items[i].EntityID = 0
                ignorePlatform.Items[i].Recycled = 0
                ignorePlatform.Items[i].LastActive = 0

            } else {
                anyActive = true
            }
        }

        // If we don't have any active ignores left, remove the entire component
        if !anyActive {
            ignoringEntity, _ := ignorePlatformCursor.CurrentEntity()
            err := ignoringEntity.EnqueueRemoveComponent(components.IgnorePlatformComponent)
            if err != nil {
                return err
            }
        }
    }
    return nil
}

Registering the New System


// coresystems/common.go

// ...Existing code
var DefaultCoreSystems = []blueprint.CoreSystem{
    GravitySystem{},
    FrictionSystem{},
    PlayerMovementSystem{},
    tteo_coresystems.IntegrationSystem{},
    tteo_coresystems.TransformSystem{},
    PlayerBlockCollisionSystem{},
    NewPlayerPlatformCollisionSystem(),
    OnGroundClearingSystem{},

    // Added:
    IgnorePlatformClearingSystem{},
}

Updating Platform Collision to Support Drop-Through

Finally we can update the PlayerPlatformCollisionSystem:

// coresystems/player_platform_collision_system.go

// ...Existing code

func (s *PlayerPlatformCollisionSystem) resolve(scene blueprint.Scene, platformCursor, playerCursor *warehouse.Cursor) error {
    // Get the player state
    playerShape := blueprintspatial.Components.Shape.GetFromCursor(playerCursor)
    playerPosition := blueprintspatial.Components.Position.GetFromCursor(playerCursor)
    playerDynamics := blueprintmotion.Components.Dynamics.GetFromCursor(playerCursor)

    // Get the platform state
    platformShape := blueprintspatial.Components.Shape.GetFromCursor(platformCursor)
    platformPosition := blueprintspatial.Components.Position.GetFromCursor(platformCursor)
    platformDynamics := blueprintmotion.Components.Dynamics.GetFromCursor(platformCursor)

    // Check for collision
    if ok, collisionResult := spatial.Detector.Check(
        *playerShape, *platformShape, playerPosition.Two, platformPosition.Two,
    ); ok {

        // Adding IgnorePlatform Logic Part One --------------------
        ignoringPlatforms, ignorePlatform := components.IgnorePlatformComponent.GetFromCursorSafe(playerCursor)

        platformEntity, err := platformCursor.CurrentEntity()
        if err != nil {
            return err
        }
        if ignoringPlatforms {
            for _, ignored := range ignorePlatform.Items {
                if ignored.EntityID == int(platformEntity.ID()) && ignored.Recycled == platformEntity.Recycled() {
                    return nil
                }
            }
        }

        //  ---------------------------------

        // Check if any of the past player positions indicate the player was above the platform
        platformTop := platformShape.Polygon.WorldVertices[0].Y

        playerWasAbove := s.checkAnyPlayerPositionWasAbove(platformTop, playerShape.LocalAAB.Height)

        // We only want to resolve collisions when:
        // 1. The player is falling (vel.Y > 0)
        // 2. The collision is with the top of the platform
        // 3. The player was above the platform at some point (within n ticks)
        if playerDynamics.Vel.Y > 0 && collisionResult.IsTopB() && playerWasAbove {

            motion.Resolver.Resolve(
                &playerPosition.Two,
                &platformPosition.Two,
                playerDynamics,
                platformDynamics,
                collisionResult,
            )

            // Standard onGround handling
            currentTick := scene.CurrentTick()

            // If not grounded, enqueue onGround with values
            playerAlreadyGrounded, onGround := components.OnGroundComponent.GetFromCursorSafe(playerCursor)

            if !playerAlreadyGrounded {
                playerEntity, _ := playerCursor.CurrentEntity()
                err := playerEntity.EnqueueAddComponentWithValue(
                    components.OnGroundComponent,
                    components.OnGround{LastTouch: currentTick, Landed: currentTick},
                )
                if err != nil {
                    return err
                }
            } else {
                onGround.LastTouch = currentTick
            }

            // Adding IgnorePlatform Logic Part Two --------------------
            // Here is where we do the actual ignore platform tracking!
            if ignoringPlatforms {
                // Use the maximum possible int64 value as initial comparison point
                var oldestTick int64 = math.MaxInt64
                oldestIndex := -1

                // Iterate through all ignored platforms
                for i, ignored := range ignorePlatform.Items {
                    // Check if this platform entity is already in the ignore list
                    // by comparing both entity ID and recycled status
                    if ignored.EntityID == int(platformEntity.ID()) && ignored.Recycled == platformEntity.Recycled() {
                        // Platform is already being ignored, no need to add it again
                        return nil
                    }

                    // Track the item with the oldest "LastActive" timestamp
                    // This helps us identify which item to replace if the ignore list is full
                    if int64(ignored.LastActive) < oldestTick {
                        oldestTick = int64(ignored.LastActive)
                        oldestIndex = i
                    }
                }

                // If we found an item to replace (oldestIndex != -1),
                // update that slot with the current platform entity's information
                if oldestIndex != -1 {
                    // Replace the oldest ignored platform with the current one
                    ignorePlatform.Items[oldestIndex].EntityID = int(platformEntity.ID())
                    ignorePlatform.Items[oldestIndex].Recycled = platformEntity.Recycled()
                    ignorePlatform.Items[oldestIndex].LastActive = currentTick
                    return nil
                }
            }
            // ---------------------------------------
        }
    }
    return nil
}

Understanding the Implementation

In these snippets we introduce a new IgnorePlatformComponent. The choice to use an array over a slice isn't a huge deal but it's worth talking about briefly. Bappa is an archetypal ECS that likes to store components of a given archetype contiguously in memory via a table like structure. So if you can get away with pre-allocating the memory and use value semantics, avoiding types like slices/maps which introduce indirection due to their dynamic sizing, the result should be a more optimal cache friendly memory layout.

Inside the PlayerMovementSystem we enqueue the creation of the component when the player is pressing the down key.

We create a new IgnorePlatformClearingSystem and add it to the DefaultCoreSystems slice. This system queries the IgnorePlatformComponent entities and checks each array entry. If they're passed expiration they get cleared out. If they're all expired we enqueue the removal of the component.

Lastly, we update the PlayerPlatformCollisionSystem in the highlighted sections. First we add a guard clause that returns early if we're ignoring the current platform. At the end we add the IgnorePlatform tracking logic.

Image description

The Animation System

With the base movement covered it's time to show more than just an idle animation! Now we're going to create our second client system:

Creating the Animation System


// clientsystems/player_animation_system.go

package clientsystems

import (
    "math"
    "platformer/animations"
    "platformer/components"

    "github.com/TheBitDrifter/blueprint"
    blueprintclient "github.com/TheBitDrifter/blueprint/client"
    blueprintmotion "github.com/TheBitDrifter/blueprint/motion"
    "github.com/TheBitDrifter/coldbrew"
)

type PlayerAnimationSystem struct{}

func (PlayerAnimationSystem) Run(cli coldbrew.LocalClient, scene coldbrew.Scene) error {
    cursor := scene.NewCursor(blueprint.Queries.InputBuffer)

    for range cursor.Next() {
        // Get state
        bundle := blueprintclient.Components.SpriteBundle.GetFromCursor(cursor)
        spriteBlueprint := &bundle.Blueprints[0]
        dyn := blueprintmotion.Components.Dynamics.GetFromCursor(cursor)
        grounded, onGround := components.OnGroundComponent.GetFromCursorSafe(cursor)
        if grounded {
            grounded = scene.CurrentTick() == onGround.LastTouch
        }

        // Player is moving horizontal and grounded (running)
        if math.Abs(dyn.Vel.X) > 20 && grounded {
            spriteBlueprint.TryAnimation(animations.RunAnimation)

        // Player is moving down and not grounded (falling)
        } else if dyn.Vel.Y > 0 && !grounded {
            spriteBlueprint.TryAnimation(animations.FallAnimation)

        // Player is moving up and not grounded (jumping)
        } else if dyn.Vel.Y <= 0 && !grounded {
            spriteBlueprint.TryAnimation(animations.JumpAnimation)

        // Default: player is idle
        } else {
            spriteBlueprint.TryAnimation(animations.IdleAnimation)
        }
    }
    return nil
}

Registering the Animation System

// ...Existing code
var DefaultClientSystems = []coldbrew.ClientSystem{
    &CameraFollowerSystem{},
    &coldbrew_clientsystems.BackgroundScrollSystem{},

    // Added:
    PlayerAnimationSystem{},
}

How the Animation System Works

This system is pretty straightforward. We query the player entities and change their animation state based on various component states. Primarily we check the grounded and velocity state to determine whether to play the falling, running, jumping, or idle animation.

The system makes the following decisions:

  1. If the player is moving horizontally and is grounded, play the running animation
  2. If the player is moving downward and not grounded, play the falling animation
  3. If the player is moving upward and not grounded, play the jumping animation
  4. If none of the above conditions are met, play the idle animation

By linking animation states directly to physics properties, we create a responsive character that visually reflects its movement state without requiring additional code in the movement or collision systems.

Image description

Next Steps

Alright, at this point we're going to conclude the tutorial! There is still a lot more functionality that could be added and if you're interested in things such as sounds, multiple scenes, multiple cameras, multiple players, LDTK, slopes, etc, there are resources for you!

Resources for Further Learning

You can find examples, and docs that go over Bappa functionality in depth. Furthermore, Bappa-Create includes multiple platformer templates that build directly upon this tutorial with the aforementioned functionality.

For more efficient level design, consider using LDTK (Level Designer Toolkit) with Bappa's LDTK integration. This provides a visual editor for creating complex game levels without having to code everything manually.

What We've Accomplished

In this tutorial, we've built a solid foundation for a platformer game with the Bappa Framework, including:

  • A scrolling parallax background
  • A player character with physics-based movement
  • Solid terrain blocks with collision detection
  • One-way platforms with drop-through functionality
  • Animation that responds to player state

Conclusion

Thanks for following along, Happy coding!

Best,
TBD