agb/examples/hyperspace-roll/src/graphics.rs

300 lines
9.2 KiB
Rust
Raw Normal View History

2022-07-25 06:33:51 +10:00
use agb::{
2023-04-06 03:35:45 +10:00
display::object::{OamManaged, Object, Sprite, Tag},
2022-07-25 06:33:51 +10:00
fixnum::Vector2D,
};
use alloc::vec::Vec;
use crate::{EnemyAttackType, Face, Ship};
2023-12-11 03:35:15 +11:00
static SPRITES: &agb::display::object::Graphics = agb::include_aseprite!(
2022-07-25 06:33:51 +10:00
"gfx/dice-faces.aseprite",
"gfx/ships.aseprite",
"gfx/small-sprites.aseprite"
);
2023-12-11 03:35:15 +11:00
pub static FACE_SPRITES: &FaceSprites = {
static S_SHOOT: &Sprite = SPRITES.tags().get("shoot").sprite(0);
static S_SHIELD: &Sprite = SPRITES.tags().get("shield").sprite(0);
static S_MALFUNCTION: &Sprite = SPRITES.tags().get("malfunction").sprite(0);
static S_HEAL: &Sprite = SPRITES.tags().get("player_heal").sprite(0);
static S_BYPASS: &Sprite = SPRITES.tags().get("shield bypass").sprite(0);
static S_DOUBLE_SHOT: &Sprite = SPRITES.tags().get("double shoot").sprite(0);
static S_TRIPLE_SHOT: &Sprite = SPRITES.tags().get("triple shoot").sprite(0);
static S_BLANK: &Sprite = SPRITES.tags().get("blank").sprite(0);
static S_DISRUPT: &Sprite = SPRITES.tags().get("disruption").sprite(0);
static S_MALFUNCTION_SHOOT: &Sprite = SPRITES.tags().get("malfunction shot").sprite(0);
static S_DOUBLE_SHIELD: &Sprite = SPRITES.tags().get("double shield").sprite(0);
static S_TRIPLE_SHIELD: &Sprite = SPRITES.tags().get("triple shield").sprite(0);
static S_DOUBLE_SHIELD_VALUE: &Sprite = SPRITES.tags().get("double shield value").sprite(0);
static S_DOUBLE_SHOT_VALUE: &Sprite = SPRITES.tags().get("double shoot power").sprite(0);
static S_TRIPLE_SHOT_VALUE: &Sprite = SPRITES.tags().get("triple shoot power").sprite(0);
static S_BURST_SHIELD: &Sprite = SPRITES.tags().get("burst shield").sprite(0);
static S_INVERT: &Sprite = SPRITES.tags().get("swap shield and shoot").sprite(0);
&FaceSprites {
sprites: [
S_SHOOT,
S_SHIELD,
S_MALFUNCTION,
S_HEAL,
S_BYPASS,
S_DOUBLE_SHOT,
S_TRIPLE_SHOT,
S_BLANK,
S_DISRUPT,
S_MALFUNCTION_SHOOT,
S_DOUBLE_SHIELD,
S_TRIPLE_SHIELD,
S_DOUBLE_SHIELD_VALUE,
S_DOUBLE_SHOT_VALUE,
S_TRIPLE_SHOT_VALUE,
S_BURST_SHIELD,
S_INVERT,
],
}
};
pub static ENEMY_ATTACK_SPRITES: &EnemyAttackSprites = {
static S_SHOOT: &Sprite = SPRITES.tags().get("enemy shoot").sprite(0);
static S_SHIELD: &Sprite = SPRITES.tags().get("enemy shield").sprite(0);
static S_HEAL: &Sprite = SPRITES.tags().get("enemy heal").sprite(0);
2022-07-25 06:33:51 +10:00
2023-12-11 03:35:15 +11:00
&EnemyAttackSprites {
sprites: [S_SHOOT, S_SHIELD, S_HEAL],
}
};
pub static SELECT_BOX: &Tag = SPRITES.tags().get("selection");
pub static SELECTED_BOX: &Sprite = SPRITES.tags().get("selected").sprite(0);
pub static MODIFIED_BOX: &Sprite = SPRITES.tags().get("modified").sprite(0);
pub static BULLET_SPRITE: &Sprite = SPRITES.tags().get("bullet").sprite(0);
pub static DISRUPT_BULLET: &Sprite = SPRITES.tags().get("disrupt bullet").sprite(0);
pub static BURST_BULLET: &Sprite = SPRITES.tags().get("burst shield bullet").sprite(0);
pub static SHIELD: &Tag = SPRITES.tags().get("ship shield");
pub static SHIP_SPRITES: &ShipSprites = {
static S_PLAYER: &Sprite = SPRITES.tags().get("player").sprite(0);
static S_DRONE: &Sprite = SPRITES.tags().get("drone").sprite(0);
static S_PILOTED_SHIP: &Sprite = SPRITES.tags().get("piloted ship").sprite(0);
static S_SHIELD: &Sprite = SPRITES.tags().get("ship shield").sprite(0);
&ShipSprites {
sprites: [S_PLAYER, S_DRONE, S_PILOTED_SHIP, S_SHIELD],
}
};
2022-07-25 06:33:51 +10:00
2023-12-11 03:35:15 +11:00
pub static SMALL_SPRITES: &SmallSprites = &SmallSprites {};
2022-07-25 06:33:51 +10:00
pub struct FaceSprites {
sprites: [&'static Sprite; 17],
}
impl FaceSprites {
pub fn sprite_for_face(&self, face: Face) -> &'static Sprite {
self.sprites[face as usize]
}
}
pub struct ShipSprites {
sprites: [&'static Sprite; 4],
}
impl ShipSprites {
pub fn sprite_for_ship(&self, ship: Ship) -> &'static Sprite {
self.sprites[ship as usize]
}
}
pub struct SmallSprites;
2023-12-11 03:35:15 +11:00
static NUMBERS: &Tag = SPRITES.tags().get("numbers");
static RED_BAR: &Tag = SPRITES.tags().get("red bar");
2022-07-25 06:33:51 +10:00
impl SmallSprites {
2023-12-11 03:35:15 +11:00
pub fn number(&self, i: u32) -> &'static Sprite {
NUMBERS.sprite(i as usize)
2022-07-25 06:33:51 +10:00
}
2023-12-11 03:35:15 +11:00
pub fn slash(&self) -> &'static Sprite {
NUMBERS.sprite(10)
2022-07-25 06:33:51 +10:00
}
2023-12-11 03:35:15 +11:00
pub fn red_bar(&self, i: usize) -> &'static Sprite {
RED_BAR.sprite(i)
2022-07-25 06:33:51 +10:00
}
}
pub struct EnemyAttackSprites {
sprites: [&'static Sprite; 3],
}
impl EnemyAttackSprites {
pub fn sprite_for_attack(&self, attack: EnemyAttackType) -> &'static Sprite {
self.sprites[attack as usize]
}
}
pub struct HealthBar<'a> {
max: usize,
sprites: Vec<Object<'a>>,
}
impl<'a> HealthBar<'a> {
2023-04-06 03:35:45 +10:00
pub fn new(pos: Vector2D<i32>, max: usize, obj: &'a OamManaged) -> Self {
2022-07-25 06:33:51 +10:00
assert_eq!(max % 8, 0);
let sprites = (0..(max / 8))
.map(|i| {
2023-04-06 03:35:45 +10:00
let mut health_object = obj.object_sprite(SMALL_SPRITES.red_bar(0));
2022-07-25 06:33:51 +10:00
health_object
.set_position(pos + (i as i32 * 8, 0).into())
.show();
health_object
})
.collect();
Self { max, sprites }
}
2023-04-06 03:35:45 +10:00
pub fn set_value(&mut self, new_value: usize, obj: &'a OamManaged) {
2022-07-25 06:33:51 +10:00
assert!(new_value <= self.max);
for (i, sprite) in self.sprites.iter_mut().enumerate() {
if (i + 1) * 8 < new_value {
2023-04-24 03:55:23 +10:00
sprite.set_sprite(obj.sprite(SMALL_SPRITES.red_bar(0)));
2022-07-25 06:33:51 +10:00
} else if i * 8 < new_value {
2023-04-24 03:55:23 +10:00
sprite.set_sprite(obj.sprite(SMALL_SPRITES.red_bar(8 - (new_value - i * 8))));
2022-07-25 06:33:51 +10:00
} else {
2023-04-24 03:55:23 +10:00
sprite.set_sprite(obj.sprite(SMALL_SPRITES.red_bar(8)));
2022-07-25 06:33:51 +10:00
}
}
}
pub fn show(&mut self) {
for obj in self.sprites.iter_mut() {
obj.show();
}
}
pub fn hide(&mut self) {
for obj in self.sprites.iter_mut() {
obj.hide();
}
}
}
pub struct FractionDisplay<'a> {
sprites: Vec<Object<'a>>,
digits: usize,
current_current: usize,
current_max: usize,
}
impl<'a> FractionDisplay<'a> {
2023-04-06 03:35:45 +10:00
pub fn new(pos: Vector2D<i32>, digits: usize, obj: &'a OamManaged) -> Self {
2022-07-25 06:33:51 +10:00
let mut sprites = Vec::with_capacity(digits * 2 + 1);
for i in 0..digits {
2023-04-06 03:35:45 +10:00
let mut left_digit = obj.object_sprite(SMALL_SPRITES.number(0));
2023-06-04 08:29:54 +10:00
left_digit
.set_position(pos + (i as i32 * 4, 0).into())
.show();
2022-07-25 06:33:51 +10:00
sprites.push(left_digit);
2023-04-06 03:35:45 +10:00
let mut right_digit = obj.object_sprite(SMALL_SPRITES.number(0));
2023-06-04 08:29:54 +10:00
right_digit
.set_position(pos + (i as i32 * 4 + digits as i32 * 4 + 7, 0).into())
.show();
2022-07-25 06:33:51 +10:00
sprites.push(right_digit);
}
2023-04-06 03:35:45 +10:00
let mut slash = obj.object_sprite(SMALL_SPRITES.slash());
2023-06-04 08:29:54 +10:00
slash
.set_position(pos + (digits as i32 * 4 + 1, 0).into())
.show();
2022-07-25 06:33:51 +10:00
sprites.push(slash);
Self {
sprites,
digits,
current_current: 0,
current_max: 0,
}
}
2023-04-06 03:35:45 +10:00
pub fn set_value(&mut self, current: usize, max: usize, obj: &'a OamManaged) {
2022-07-25 06:33:51 +10:00
if self.current_current == current && self.current_max == max {
return;
}
let mut current = current;
let mut max = max;
for i in 0..self.digits {
let current_value_digit = current % 10;
current /= 10;
let current_value_sprite = &mut self.sprites[(self.digits - i) * 2 - 2];
current_value_sprite
2023-04-24 03:55:23 +10:00
.set_sprite(obj.sprite(SMALL_SPRITES.number(current_value_digit as u32)));
2022-07-25 06:33:51 +10:00
let max_value_digit = max % 10;
max /= 10;
let max_value_sprite = &mut self.sprites[(self.digits - i) * 2 - 1];
2023-04-24 03:55:23 +10:00
max_value_sprite.set_sprite(obj.sprite(SMALL_SPRITES.number(max_value_digit as u32)));
2022-07-25 06:33:51 +10:00
}
}
}
pub struct NumberDisplay<'a> {
objects: Vec<Object<'a>>,
value: Option<u32>,
position: Vector2D<i32>,
}
impl<'a> NumberDisplay<'a> {
pub fn new(position: Vector2D<i32>) -> Self {
Self {
objects: Vec::new(),
value: None,
position,
}
}
2023-04-06 03:35:45 +10:00
pub fn set_value(&mut self, new_value: Option<u32>, obj: &'a OamManaged) {
2022-07-25 06:33:51 +10:00
if self.value == new_value {
return;
}
self.value = new_value;
self.objects.clear();
if let Some(mut new_value) = new_value {
if new_value == 0 {
2023-04-06 03:35:45 +10:00
let mut zero_object = obj.object_sprite(SMALL_SPRITES.number(0));
2022-07-25 06:33:51 +10:00
zero_object.show().set_position(self.position);
self.objects.push(zero_object);
return;
}
let mut digit = 0;
while new_value != 0 {
let current_value_digit = new_value % 10;
new_value /= 10;
let mut current_value_obj =
2023-04-06 03:35:45 +10:00
obj.object_sprite(SMALL_SPRITES.number(current_value_digit));
2022-07-25 06:33:51 +10:00
current_value_obj
.show()
.set_position(self.position - (digit * 4, 0).into());
digit += 1;
self.objects.push(current_value_obj);
}
}
}
}