agb/book/games/pong/src/main.rs

75 lines
2.8 KiB
Rust
Raw Normal View History

// Games made using `agb` are no_std which means you don't have access to the standard
// rust library. This is because the game boy advance doesn't really have an operating
// system, so most of the content of the standard library doesn't apply.
//
// Provided you haven't disabled it, agb does provide an allocator, so it is possible
// to use both the `core` and the `alloc` built in crates.
2022-01-02 09:37:39 +11:00
#![no_std]
// `agb` defines its own `main` function, so you must declare your game's main function
// using the #[agb::entry] proc macro. Failing to do so will cause failure in linking
// which won't be a particularly clear error message.
2022-01-02 09:37:39 +11:00
#![no_main]
2022-08-02 07:26:32 +10:00
use agb::{
2022-08-02 07:50:37 +10:00
display::object::{Graphics, Tag},
2022-08-02 07:26:32 +10:00
include_aseprite,
};
2022-08-02 07:26:32 +10:00
// Import the sprites in to this constant. This holds the sprite
// and palette data in a way that is manageable by agb.
const GRAPHICS: &Graphics = include_aseprite!("gfx/sprites.aseprite");
// We define some easy ways of referencing the sprites
2022-08-02 07:50:37 +10:00
#[allow(dead_code)]
2022-08-02 07:26:32 +10:00
const PADDLE_END: &Tag = GRAPHICS.tags().get("Paddle End");
2022-08-02 07:50:37 +10:00
#[allow(dead_code)]
2022-08-02 07:26:32 +10:00
const PADDLE_MID: &Tag = GRAPHICS.tags().get("Paddle Mid");
const BALL: &Tag = GRAPHICS.tags().get("Ball");
2022-01-02 11:21:59 +11:00
// The main function must take 0 arguments and never return. The agb::entry decorator
// ensures that everything is in order. `agb` will call this after setting up the stack
// and interrupt handlers correctly.
2022-01-02 09:37:39 +11:00
#[agb::entry]
2022-08-02 07:26:32 +10:00
fn main(mut gba: agb::Gba) -> ! {
// Get the OAM manager
let object = gba.display.object.get();
2022-01-02 11:40:34 +11:00
2022-08-02 07:26:32 +10:00
// Create an object with the ball sprite
let mut ball = object.object_sprite(BALL.sprite(0));
2022-01-02 11:40:34 +11:00
2022-08-02 07:26:32 +10:00
// Place this at some point on the screen, (50, 50) for example
ball.set_x(50).set_y(50).show();
2022-01-02 22:27:26 +11:00
2022-08-02 07:26:32 +10:00
// Now commit the object controller so this change is reflected on the screen,
// this should normally be done in vblank but it'll work just fine here for now
object.commit();
let mut ball_x = 50;
let mut ball_y = 50;
let mut x_velocity = 1;
let mut y_velocity = 1;
2022-01-02 11:21:59 +11:00
loop {
2022-08-02 07:26:32 +10:00
// This will calculate the new position and enforce the position
// of the ball remains within the screen
ball_x = (ball_x + x_velocity).clamp(0, agb::display::WIDTH - 16);
ball_y = (ball_y + y_velocity).clamp(0, agb::display::HEIGHT - 16);
2022-08-02 07:26:32 +10:00
// We check if the ball reaches the edge of the screen and reverse it's direction
if ball_x == 0 || ball_x == agb::display::WIDTH - 16 {
x_velocity = -x_velocity;
}
if ball_y == 0 || ball_y == agb::display::HEIGHT - 16 {
y_velocity = -y_velocity;
}
2022-08-02 07:26:32 +10:00
// Set the position of the ball to match our new calculated position
ball.set_x(ball_x as u16).set_y(ball_y as u16);
2022-08-02 07:26:32 +10:00
// Wait for vblank, then commit the objects to the screen
agb::display::busy_wait_for_vblank();
2022-04-06 06:30:26 +10:00
object.commit();
}
2022-01-02 09:37:39 +11:00
}