Camera

Show how to work with the optional camera systems.

There’s a separate configuration for the main game camera and the UI camera.

The threeforms.png sprite image for this example is:

Sprite

With the following threeforms.ron RON configuration file for positioning the center of the sprite:


(pivot_x: Center, pivot_y: Center)

The bunnymark.png sprite image for this example is:

Sprite

With the following bunnymark.ron RON configuration file for positioning the center of the sprite:


(pivot_x: Center, pivot_y: Center)

Source

use chuot::{Config, Context, Game, KeyCode, MouseButton};

/// How fast the "player" moves.
const PLAYER_SPEED: f32 = 90.0;

/// Define a game state for our example.
#[derive(Default)]
struct GameState {
    /// Simulate the horizontal player movement, the camera will follow this.
    player_x: f32,
    /// Simulate the vertical player movement, the camera will follow this.
    player_y: f32,
    /// Previous horizontal position, for smooth rendering.
    previous_player_x: f32,
    /// Previous vertical position, for smooth rendering.
    previous_player_y: f32,
}

impl Game for GameState {
    /// Move the "player" around with keyboard input.
    fn update(&mut self, ctx: Context) {
        // Set the previous position before updating the current one
        self.previous_player_x = self.player_x;
        self.previous_player_y = self.player_y;

        // Move left on left arrow or 'a' key
        if ctx.key_held(KeyCode::ArrowLeft) || ctx.key_held(KeyCode::KeyA) {
            self.player_x -= PLAYER_SPEED * ctx.delta_time();
        }
        // Move right on right arrow or 'd' key
        if ctx.key_held(KeyCode::ArrowRight) || ctx.key_held(KeyCode::KeyD) {
            self.player_x += PLAYER_SPEED * ctx.delta_time();
        }
        // Move up on up arrow or 'w' key
        if ctx.key_held(KeyCode::ArrowUp) || ctx.key_held(KeyCode::KeyW) {
            self.player_y -= PLAYER_SPEED * ctx.delta_time();
        }
        // Move down on down arrow or 's' key
        if ctx.key_held(KeyCode::ArrowDown) || ctx.key_held(KeyCode::KeyS) {
            self.player_y += PLAYER_SPEED * ctx.delta_time();
        }

        // Follow the player with the main camera
        ctx.main_camera().follow((self.player_x, self.player_y));

        // Shake the camera when clicking, using the mouse position to determine the intensity and duration
        if let Some((mouse_x, mouse_y)) = ctx.mouse() {
            if ctx.mouse_pressed(MouseButton::Left) {
                // Horizontal mouse is the duration, vertical mouse the force
                ctx.main_camera().shake(1.0, mouse_x / 10.0, mouse_y / 4.0);
            }
        }
    }

    /// Render the game.
    fn render(&mut self, ctx: Context) {
        // Draw some bunnies to show the camera movement
        for y in -10..10 {
            for x in -10..10 {
                // Load a sprite asset and draw it statically
                ctx.sprite("bunnymark")
                    // Draw it at the same position every frame
                    .translate_x(x as f32 * 150.0)
                    .translate_y(y as f32 * 150.0)
                    // Draw the sprite on the screen
                    .draw();

                // Draw the text on the same position as the sprite
                ctx.text("Beachball", &format!("{x},{y}"))
                    // Draw it at the same position every frame
                    .translate_x(x as f32 * 150.0)
                    .translate_y(y as f32 * 150.0)
                    // Text uses the UI camera by default, so use the main camera
                    .use_main_camera()
                    .draw();
            }
        }

        // Load a sprite asset and draw it
        ctx.sprite("threeforms")
            // Make the sprite move, the camera will follow it
            .translate_x(self.player_x)
            .translate_y(self.player_y)
            // Make the sprite render smoothly by interpolating based on the render blending factor,
            // this is applicable when there's multiple update tick calls in a single render tick,
            // if we don't do this the player will move in a jittery motion
            .translate_previous_x(self.previous_player_x)
            .translate_previous_y(self.previous_player_y)
            // Draw the sprite on the screen
            .draw();

        // Draw the instructions
        let instructions = if let Some((mouse_x, mouse_y)) = ctx.mouse() {
            format!(
                "Arrow keys to move\n\nClick to shake camera:\n- Duration:  1.00 s\n- Amplitude: {:.2} px\n- Frequency: {:.2} hz",
                mouse_x / 10.0,
                mouse_y / 4.0,
            )
        } else {
            "Arrow keys to move\n\nClick to shake camera".to_owned()
        };

        ctx.text("Beachball", &instructions).use_ui_camera().draw();
    }

    /// Setup the camera.
    fn init(&mut self, ctx: Context) {
        // Follow the camera slowly horizontally
        ctx.main_camera().set_lerp_x(0.1);
        // Follow the camera quickly vertically
        ctx.main_camera().set_lerp_y(0.5);
    }
}

/// Open an empty window.
fn main() {
    // Game configuration
    let config = Config {
        buffer_width: 360.0,
        buffer_height: 288.0,
        // Apply a minimum of 2 times scaling for the buffer
        // Will result in a minimum, and on web exact, window size of 720x576
        scaling: 2.0,
        ..Default::default()
    };

    // Spawn the window and run the 'game'
    GameState::default().run(chuot::load_assets!(), config);
}

Compatibility

Chuột VersionExample Works
0.3.0
0.3.1
Unreleased