update to use new sprite system

This commit is contained in:
Corwin 2022-03-06 02:54:52 +00:00
parent 4c57f85acd
commit 9dca54c9bb
3 changed files with 428 additions and 205 deletions

View file

@ -24,6 +24,9 @@ dependencies = [
"agb_sound_converter", "agb_sound_converter",
"bare-metal", "bare-metal",
"bitflags", "bitflags",
"hashbrown",
"modular-bitfield",
"rustc-hash",
] ]
[[package]] [[package]]
@ -37,6 +40,7 @@ dependencies = [
name = "agb_image_converter" name = "agb_image_converter"
version = "0.6.0" version = "0.6.0"
dependencies = [ dependencies = [
"asefile",
"image", "image",
"proc-macro2", "proc-macro2",
"quote", "quote",
@ -64,6 +68,31 @@ dependencies = [
"syn", "syn",
] ]
[[package]]
name = "ahash"
version = "0.7.6"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "fcb51a0695d8f838b1ee009b3fbf66bda078cd64590202a864a8f3e8c4315c47"
dependencies = [
"getrandom",
"once_cell",
"version_check",
]
[[package]]
name = "asefile"
version = "0.3.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d0d5f7de918fd4cb18249819fc4bd27f6a5dbfbc9dcb271727f27dacf17ce880"
dependencies = [
"bitflags",
"byteorder",
"flate2",
"image",
"log",
"nohash",
]
[[package]] [[package]]
name = "autocfg" name = "autocfg"
version = "1.1.0" version = "1.1.0"
@ -132,11 +161,24 @@ dependencies = [
[[package]] [[package]]
name = "deflate" name = "deflate"
version = "1.0.0" version = "0.8.6"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "c86f7e25f518f4b81808a2cf1c50996a61f5c2eb394b2393bd87f2a4780a432f" checksum = "73770f8e1fe7d64df17ca66ad28994a0a623ea497fa69486e14984e715c5d174"
dependencies = [ dependencies = [
"adler32", "adler32",
"byteorder",
]
[[package]]
name = "flate2"
version = "1.0.22"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1e6988e897c1c9c485f43b47a529cef42fde0547f9d8d41a7062518f1d8fc53f"
dependencies = [
"cfg-if 1.0.0",
"crc32fast",
"libc",
"miniz_oxide 0.4.4",
] ]
[[package]] [[package]]
@ -148,6 +190,26 @@ dependencies = [
"cfg-if 0.1.10", "cfg-if 0.1.10",
] ]
[[package]]
name = "getrandom"
version = "0.2.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d39cd93900197114fa1fcb7ae84ca742095eed9442088988ae74fa744e930e77"
dependencies = [
"cfg-if 1.0.0",
"libc",
"wasi",
]
[[package]]
name = "hashbrown"
version = "0.12.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8c21d40587b92fa6a6c6e3c1bdbf87d75511db5672f9c93175574b3a00df1758"
dependencies = [
"ahash",
]
[[package]] [[package]]
name = "hound" name = "hound"
version = "3.4.0" version = "3.4.0"
@ -156,9 +218,9 @@ checksum = "8a164bb2ceaeff4f42542bdb847c41517c78a60f5649671b2a07312b6e117549"
[[package]] [[package]]
name = "image" name = "image"
version = "0.24.1" version = "0.23.14"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "db207d030ae38f1eb6f240d5a1c1c88ff422aa005d10f8c6c6fc5e75286ab30e" checksum = "24ffcb7e7244a9bf19d35bf2883b9c080c4ced3c07a9895572178cdb8f13f6a1"
dependencies = [ dependencies = [
"bytemuck", "bytemuck",
"byteorder", "byteorder",
@ -169,6 +231,12 @@ dependencies = [
"png", "png",
] ]
[[package]]
name = "libc"
version = "0.2.119"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1bf2e165bb3457c8e098ea76f3e3bc9db55f87aa90d52d0e6be741470916aaa4"
[[package]] [[package]]
name = "libflate" name = "libflate"
version = "0.1.27" version = "0.1.27"
@ -182,14 +250,60 @@ dependencies = [
] ]
[[package]] [[package]]
name = "miniz_oxide" name = "log"
version = "0.5.1" version = "0.4.14"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d2b29bd4bc3f33391105ebee3589c19197c4271e3e5a9ec9bfe8127eeff8f082" checksum = "51b9bbe6c47d51fc3e1a9b945965946b4c44142ab8792c50835a980d362c2710"
dependencies = [
"cfg-if 1.0.0",
]
[[package]]
name = "miniz_oxide"
version = "0.3.7"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "791daaae1ed6889560f8c4359194f56648355540573244a5448a83ba1ecc7435"
dependencies = [
"adler32",
]
[[package]]
name = "miniz_oxide"
version = "0.4.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a92518e98c078586bc6c934028adcca4c92a53d6a958196de835170a01d84e4b"
dependencies = [ dependencies = [
"adler", "adler",
"autocfg",
] ]
[[package]]
name = "modular-bitfield"
version = "0.11.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a53d79ba8304ac1c4f9eb3b9d281f21f7be9d4626f72ce7df4ad8fbde4f38a74"
dependencies = [
"modular-bitfield-impl",
"static_assertions",
]
[[package]]
name = "modular-bitfield-impl"
version = "0.11.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "5a7d5f7076603ebc68de2dc6a650ec331a062a13abaa346975be747bbfa4b789"
dependencies = [
"proc-macro2",
"quote",
"syn",
]
[[package]]
name = "nohash"
version = "0.2.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a0f889fb66f7acdf83442c35775764b51fed3c606ab9cee51500dbde2cf528ca"
[[package]] [[package]]
name = "num-integer" name = "num-integer"
version = "0.1.44" version = "0.1.44"
@ -213,9 +327,9 @@ dependencies = [
[[package]] [[package]]
name = "num-rational" name = "num-rational"
version = "0.4.0" version = "0.3.2"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d41702bd167c2df5520b384281bc111a4b5efcf7fbc4c9c222c815b07e0a6a6a" checksum = "12ac428b1cb17fce6f731001d307d351ec70a6d202fc2e60f7d4c5e42d8f4f07"
dependencies = [ dependencies = [
"autocfg", "autocfg",
"num-integer", "num-integer",
@ -232,15 +346,21 @@ dependencies = [
] ]
[[package]] [[package]]
name = "png" name = "once_cell"
version = "0.17.5" version = "1.10.0"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "dc38c0ad57efb786dd57b9864e5b18bae478c00c824dc55a38bbc9da95dde3ba" checksum = "87f3e037eac156d1775da914196f0f37741a274155e34a0b7e427c35d2a2ecb9"
[[package]]
name = "png"
version = "0.16.8"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3c3287920cb847dee3de33d301c463fba14dda99db24214ddf93f83d3021f4c6"
dependencies = [ dependencies = [
"bitflags", "bitflags",
"crc32fast", "crc32fast",
"deflate", "deflate",
"miniz_oxide", "miniz_oxide 0.3.7",
] ]
[[package]] [[package]]
@ -267,6 +387,12 @@ version = "1.0.3"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3582f63211428f83597b51b2ddb88e2a91a9d52d12831f9d08f5e624e8977422" checksum = "3582f63211428f83597b51b2ddb88e2a91a9d52d12831f9d08f5e624e8977422"
[[package]]
name = "rustc-hash"
version = "1.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "08d43f7aa6b08d49f382cde6a7982047c3426db949b1424bc4b7ec9ae12c6ce2"
[[package]] [[package]]
name = "serde" name = "serde"
version = "1.0.136" version = "1.0.136"
@ -287,6 +413,12 @@ dependencies = [
"syn", "syn",
] ]
[[package]]
name = "static_assertions"
version = "1.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a2eb9349b6444b326872e140eb1cf5e7c522154d69e7a0ffb0fb81c06b37543f"
[[package]] [[package]]
name = "syn" name = "syn"
version = "1.0.86" version = "1.0.86"
@ -340,6 +472,18 @@ version = "0.2.2"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8ccb82d61f80a663efe1f787a51b16b5a51e3314d6ac365b08639f52387b33f3" checksum = "8ccb82d61f80a663efe1f787a51b16b5a51e3314d6ac365b08639f52387b33f3"
[[package]]
name = "version_check"
version = "0.9.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "49874b5167b65d7193b8aba1567f5c7d93d001cafc34600cee003eda787e483f"
[[package]]
name = "wasi"
version = "0.10.2+wasi-snapshot-preview1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "fd6fbd9a79829dd1ad0cc20627bf1ed606756a7f77edff7b66b7064f9cb327c6"
[[package]] [[package]]
name = "xml-rs" name = "xml-rs"
version = "0.8.4" version = "0.8.4"

View file

@ -15,7 +15,7 @@ use rng::get_random;
use agb::{ use agb::{
display::{ display::{
background::{BackgroundDistributor, BackgroundRegular}, background::{BackgroundDistributor, BackgroundRegular},
object::{ObjectControl, ObjectStandard}, object::{Object, ObjectController, Sprite, Tag, TagMap},
Priority, HEIGHT, WIDTH, Priority, HEIGHT, WIDTH,
}, },
fixnum::{FixedNum, Rect, Vector2D}, fixnum::{FixedNum, Rect, Vector2D},
@ -23,7 +23,34 @@ use agb::{
}; };
use generational_arena::Arena; use generational_arena::Arena;
agb::include_gfx!("gfx/objects.toml"); const SPRITE_TAGS: (&[Sprite], &TagMap) =
agb::include_aseprite!("gfx/objects.aseprite", "gfx/boss.aseprite");
const TAG_MAP: &TagMap = SPRITE_TAGS.1;
const LONGSWORD_IDLE: &Tag = TAG_MAP.get("Idle - longsword");
const LONGSWORD_WALK: &Tag = TAG_MAP.get("Walk - longsword");
const LONGSWORD_JUMP: &Tag = TAG_MAP.get("Jump - longsword");
const LONGSWORD_ATTACK: &Tag = TAG_MAP.get("Attack - longsword");
const LONGSWORD_JUMP_ATTACK: &Tag = TAG_MAP.get("Jump attack - longsword");
const SHORTSWORD_IDLE: &Tag = TAG_MAP.get("Idle - shortsword");
const SHORTSWORD_WALK: &Tag = TAG_MAP.get("Walk - shortsword");
const SHORTSWORD_JUMP: &Tag = TAG_MAP.get("jump - shortsword");
const SHORTSWORD_ATTACK: &Tag = TAG_MAP.get("attack - shortsword");
const SHORTSWORD_JUMP_ATTACK: &Tag = TAG_MAP.get("jump attack - shortsword");
const KNIFE_IDLE: &Tag = TAG_MAP.get("idle - knife");
const KNIFE_WALK: &Tag = TAG_MAP.get("walk - knife");
const KNIFE_JUMP: &Tag = TAG_MAP.get("jump - knife");
const KNIFE_ATTACK: &Tag = TAG_MAP.get("attack - knife");
const KNIFE_JUMP_ATTACK: &Tag = TAG_MAP.get("jump attack - knife");
const SWORDLESS_IDLE: &Tag = TAG_MAP.get("idle swordless");
const SWORDLESS_WALK: &Tag = TAG_MAP.get("walk swordless");
const SWORDLESS_JUMP: &Tag = TAG_MAP.get("jump swordless");
const SWORDLESS_ATTACK: &Tag = KNIFE_ATTACK;
const SWORDLESS_JUMP_ATTACK: &Tag = KNIFE_JUMP_ATTACK;
agb::include_gfx!("gfx/background.toml"); agb::include_gfx!("gfx/background.toml");
type Number = FixedNum<8>; type Number = FixedNum<8>;
@ -127,7 +154,7 @@ impl Level {
} }
struct Entity<'a> { struct Entity<'a> {
sprite: ObjectStandard<'a>, sprite: Object<'a, 'a>,
position: Vector2D<Number>, position: Vector2D<Number>,
velocity: Vector2D<Number>, velocity: Vector2D<Number>,
collision_mask: Rect<u16>, collision_mask: Rect<u16>,
@ -135,8 +162,11 @@ struct Entity<'a> {
} }
impl<'a> Entity<'a> { impl<'a> Entity<'a> {
fn new(object_controller: &'a ObjectControl, collision_mask: Rect<u16>) -> Self { fn new(object_controller: &'a ObjectController, collision_mask: Rect<u16>) -> Self {
let mut sprite = object_controller.get_object_standard(); let s = object_controller
.get_sprite(LONGSWORD_IDLE.get_sprite(0))
.unwrap();
let mut sprite = object_controller.get_object(s).unwrap();
sprite.set_priority(Priority::P1); sprite.set_priority(Priority::P1);
Entity { Entity {
sprite, sprite,
@ -321,34 +351,30 @@ impl SwordState {
SwordState::Swordless => Number::new(6) / 256, SwordState::Swordless => Number::new(6) / 256,
} }
} }
fn idle_animation(self, counter: &mut u16) -> u16 { fn idle_animation(self, counter: u16) -> &'static Sprite {
if *counter >= 4 * 8 { let counter = counter as usize;
*counter = 0;
}
match self { match self {
SwordState::LongSword => (*counter / 8) * 4, SwordState::LongSword => LONGSWORD_IDLE.get_animation_sprite(counter / 8),
SwordState::ShortSword => (41 + *counter / 8) * 4, SwordState::ShortSword => SHORTSWORD_IDLE.get_animation_sprite(counter / 8),
SwordState::Dagger => (96 + *counter / 8) * 4, SwordState::Dagger => KNIFE_IDLE.get_animation_sprite(counter / 8),
SwordState::Swordless => (154 + *counter / 8) * 4, SwordState::Swordless => SWORDLESS_IDLE.get_animation_sprite(counter / 8),
} }
} }
fn jump_offset(self) -> u16 { fn jump_tag(self) -> &'static Tag {
match self { match self {
SwordState::LongSword => 10, SwordState::LongSword => LONGSWORD_JUMP,
SwordState::ShortSword => 51, SwordState::ShortSword => SHORTSWORD_JUMP,
SwordState::Dagger => 106, SwordState::Dagger => KNIFE_JUMP,
SwordState::Swordless => 164, SwordState::Swordless => SWORDLESS_JUMP,
} }
} }
fn walk_animation(self, counter: &mut u16) -> u16 { fn walk_animation(self, counter: u16) -> &'static Sprite {
if *counter >= 6 * 4 { let counter = counter as usize;
*counter = 0;
}
match self { match self {
SwordState::LongSword => (4 + *counter / 4) * 4, SwordState::LongSword => LONGSWORD_WALK.get_animation_sprite(counter / 4),
SwordState::ShortSword => (45 + *counter / 4) * 4, SwordState::ShortSword => SHORTSWORD_WALK.get_animation_sprite(counter / 4),
SwordState::Dagger => (100 + *counter / 4) * 4, SwordState::Dagger => KNIFE_WALK.get_animation_sprite(counter / 4),
SwordState::Swordless => (158 + *counter / 4) * 4, SwordState::Swordless => SWORDLESS_WALK.get_animation_sprite(counter / 4),
} }
} }
fn attack_duration(self) -> u16 { fn attack_duration(self) -> u16 {
@ -375,20 +401,20 @@ impl SwordState {
SwordState::Swordless => (self.attack_duration() - timer) / 8, SwordState::Swordless => (self.attack_duration() - timer) / 8,
} }
} }
fn jump_attack_tag(self) -> &'static Tag {
match self {
SwordState::LongSword => LONGSWORD_JUMP_ATTACK,
SwordState::ShortSword => SHORTSWORD_JUMP_ATTACK,
SwordState::Dagger => KNIFE_JUMP_ATTACK,
SwordState::Swordless => SWORDLESS_JUMP_ATTACK,
}
}
fn jump_attack_frame(self, timer: u16) -> u16 { fn jump_attack_frame(self, timer: u16) -> u16 {
(self.jump_attack_duration() - timer) / 8 (self.jump_attack_duration() - timer) / 8
} }
fn hold_frame(self) -> u16 { fn hold_frame(self) -> u16 {
7 7
} }
fn jump_attack_hold_frame(self) -> u16 {
match self {
SwordState::LongSword => 13,
SwordState::ShortSword => 54,
SwordState::Dagger => 109,
SwordState::Swordless => 0,
}
}
fn cooldown_time(self) -> u16 { fn cooldown_time(self) -> u16 {
match self { match self {
@ -398,26 +424,15 @@ impl SwordState {
SwordState::Swordless => 0, SwordState::Swordless => 0,
} }
} }
fn to_sprite_id(self, frame: u16) -> u16 { fn attack_tag(self) -> &'static Tag {
match self { match self {
SwordState::LongSword => (16 + frame) * 4, SwordState::LongSword => LONGSWORD_ATTACK,
SwordState::ShortSword => (57 + frame) * 4, SwordState::ShortSword => SHORTSWORD_ATTACK,
SwordState::Dagger => (112 + frame) * 4, SwordState::Dagger => KNIFE_ATTACK,
SwordState::Swordless => 0, SwordState::Swordless => SWORDLESS_ATTACK,
}
}
fn to_jump_sprite_id(self, frame: u16) -> u16 {
if frame == self.jump_attack_hold_frame() {
frame * 4
} else {
match self {
SwordState::LongSword => (24 + frame) * 4,
SwordState::ShortSword => (65 + frame) * 4,
SwordState::Dagger => (120 + frame) * 4,
SwordState::Swordless => 0,
}
} }
} }
fn fudge(self, frame: u16) -> i32 { fn fudge(self, frame: u16) -> i32 {
match self { match self {
SwordState::LongSword => long_sword_fudge(frame), SwordState::LongSword => long_sword_fudge(frame),
@ -520,15 +535,15 @@ struct Player<'a> {
} }
impl<'a> Player<'a> { impl<'a> Player<'a> {
fn new(object_controller: &'a ObjectControl) -> Player { fn new(object_controller: &'a ObjectController) -> Player {
let mut entity = Entity::new( let mut entity = Entity::new(
object_controller, object_controller,
Rect::new((0_u16, 0_u16).into(), (4_u16, 12_u16).into()), Rect::new((0_u16, 0_u16).into(), (4_u16, 12_u16).into()),
); );
entity let s = object_controller
.sprite .get_sprite(LONGSWORD_IDLE.get_sprite(0))
.set_sprite_size(agb::display::object::Size::S16x16); .unwrap();
entity.sprite.set_tile_id(0); entity.sprite.set_sprite(s);
entity.sprite.show(); entity.sprite.show();
entity.position = (144, 0).into(); entity.position = (144, 0).into();
entity.sprite.commit(); entity.sprite.commit();
@ -549,6 +564,7 @@ impl<'a> Player<'a> {
fn update( fn update(
&mut self, &mut self,
controller: &'a ObjectController,
buttons: &ButtonController, buttons: &ButtonController,
level: &Level, level: &Level,
sfx: &mut sfx::Sfx, sfx: &mut sfx::Sfx,
@ -580,13 +596,15 @@ impl<'a> Player<'a> {
self.entity.sprite.set_hflip(self.facing == Tri::Negative); self.entity.sprite.set_hflip(self.facing == Tri::Negative);
self.entity.velocity.x += self.sword.ground_walk_force() * x as i32; self.entity.velocity.x += self.sword.ground_walk_force() * x as i32;
if self.entity.velocity.x.abs() > Number::new(1) / 10 { if self.entity.velocity.x.abs() > Number::new(1) / 10 {
self.entity let sprite = controller
.sprite .get_sprite(self.sword.walk_animation(self.sprite_offset))
.set_tile_id(self.sword.walk_animation(&mut self.sprite_offset)); .unwrap();
self.entity.sprite.set_sprite(sprite);
} else { } else {
self.entity let sprite = controller
.sprite .get_sprite(self.sword.idle_animation(self.sprite_offset))
.set_tile_id(self.sword.idle_animation(&mut self.sprite_offset)); .unwrap();
self.entity.sprite.set_sprite(sprite);
} }
if b_press && self.sword != SwordState::Swordless { if b_press && self.sword != SwordState::Swordless {
@ -604,9 +622,11 @@ impl<'a> Player<'a> {
*a -= 1; *a -= 1;
let frame = self.sword.attack_frame(*a); let frame = self.sword.attack_frame(*a);
self.fudge_factor.x = self.sword.fudge(frame) * self.facing as i32; self.fudge_factor.x = self.sword.fudge(frame) * self.facing as i32;
self.entity let tag = self.sword.attack_tag();
.sprite let sprite = controller
.set_tile_id(self.sword.to_sprite_id(frame)); .get_sprite(tag.get_animation_sprite(frame as usize))
.unwrap();
self.entity.sprite.set_sprite(sprite);
hurtbox = self.sword.ground_attack_hurtbox(frame); hurtbox = self.sword.ground_attack_hurtbox(frame);
@ -618,9 +638,11 @@ impl<'a> Player<'a> {
*a -= 1; *a -= 1;
let frame = self.sword.hold_frame(); let frame = self.sword.hold_frame();
self.fudge_factor.x = self.sword.fudge(frame) * self.facing as i32; self.fudge_factor.x = self.sword.fudge(frame) * self.facing as i32;
self.entity let tag = self.sword.attack_tag();
.sprite let sprite = controller
.set_tile_id(self.sword.to_sprite_id(frame)); .get_sprite(tag.get_animation_sprite(frame as usize))
.unwrap();
self.entity.sprite.set_sprite(sprite);
if *a == 0 { if *a == 0 {
self.attack_timer = AttackTimer::Idle; self.attack_timer = AttackTimer::Idle;
} }
@ -632,7 +654,7 @@ impl<'a> Player<'a> {
match &mut self.attack_timer { match &mut self.attack_timer {
AttackTimer::Idle => { AttackTimer::Idle => {
let sprite = if self.sprite_offset < 3 * 4 { let frame = if self.sprite_offset < 3 * 4 {
self.sprite_offset / 4 self.sprite_offset / 4
} else if self.entity.velocity.y.abs() < Number::new(1) / 5 { } else if self.entity.velocity.y.abs() < Number::new(1) / 5 {
3 3
@ -643,9 +665,11 @@ impl<'a> Player<'a> {
} else { } else {
2 2
}; };
self.entity let tag = self.sword.jump_tag();
.sprite let sprite = controller
.set_tile_id((sprite + self.sword.jump_offset()) * 4); .get_sprite(tag.get_animation_sprite(frame as usize))
.unwrap();
self.entity.sprite.set_sprite(sprite);
if x != Tri::Zero { if x != Tri::Zero {
self.facing = x; self.facing = x;
@ -665,9 +689,11 @@ impl<'a> Player<'a> {
AttackTimer::Attack(a) => { AttackTimer::Attack(a) => {
*a -= 1; *a -= 1;
let frame = self.sword.jump_attack_frame(*a); let frame = self.sword.jump_attack_frame(*a);
self.entity let tag = self.sword.jump_attack_tag();
.sprite let sprite = controller
.set_tile_id(self.sword.to_jump_sprite_id(frame)); .get_sprite(tag.get_animation_sprite(frame as usize))
.unwrap();
self.entity.sprite.set_sprite(sprite);
hurtbox = self.sword.air_attack_hurtbox(frame); hurtbox = self.sword.air_attack_hurtbox(frame);
@ -799,9 +825,10 @@ impl BatData {
} }
} }
fn update( fn update<'a>(
&mut self, &mut self,
entity: &mut Entity, controller: &'a ObjectController,
entity: &mut Entity<'a>,
player: &Player, player: &Player,
level: &Level, level: &Level,
sfx: &mut sfx::Sfx, sfx: &mut sfx::Sfx,
@ -814,6 +841,8 @@ impl BatData {
.unwrap_or(false); .unwrap_or(false);
let should_damage = entity.collider().touches(player.entity.collider()); let should_damage = entity.collider().touches(player.entity.collider());
const BAT_IDLE: &Tag = TAG_MAP.get("bat");
match &mut self.bat_state { match &mut self.bat_state {
BatState::Idle => { BatState::Idle => {
self.sprite_offset += 1; self.sprite_offset += 1;
@ -825,7 +854,10 @@ impl BatData {
sfx.bat_flap(); sfx.bat_flap();
} }
entity.sprite.set_tile_id((78 + self.sprite_offset / 8) * 4); let sprite = BAT_IDLE.get_sprite(self.sprite_offset as usize / 8);
let sprite = controller.get_sprite(sprite).unwrap();
entity.sprite.set_sprite(sprite);
if (entity.position - player.entity.position).manhattan_distance() < 50.into() { if (entity.position - player.entity.position).manhattan_distance() < 50.into() {
self.bat_state = BatState::Chasing(300); self.bat_state = BatState::Chasing(300);
@ -856,7 +888,11 @@ impl BatData {
if self.sprite_offset >= 9 * 2 { if self.sprite_offset >= 9 * 2 {
self.sprite_offset = 0; self.sprite_offset = 0;
} }
entity.sprite.set_tile_id((78 + self.sprite_offset / 2) * 4);
let sprite = BAT_IDLE.get_sprite(self.sprite_offset as usize / 2);
let sprite = controller.get_sprite(sprite).unwrap();
entity.sprite.set_sprite(sprite);
if self.sprite_offset == 2 * 5 { if self.sprite_offset == 2 * 5 {
sfx.bat_flap(); sfx.bat_flap();
@ -879,7 +915,12 @@ impl BatData {
} }
} }
BatState::Dead => { BatState::Dead => {
entity.sprite.set_tile_id(87 * 4); const BAT_DEAD: &Tag = TAG_MAP.get("bat dead");
let sprite = BAT_DEAD.get_sprite(0);
let sprite = controller.get_sprite(sprite).unwrap();
entity.sprite.set_sprite(sprite);
let gravity: Number = 1.into(); let gravity: Number = 1.into();
let gravity = gravity / 16; let gravity = gravity / 16;
entity.velocity.x = 0.into(); entity.velocity.x = 0.into();
@ -917,9 +958,10 @@ impl SlimeData {
} }
} }
fn update( fn update<'a>(
&mut self, &mut self,
entity: &mut Entity, controller: &'a ObjectController,
entity: &mut Entity<'a>,
player: &Player, player: &Player,
level: &Level, level: &Level,
sfx: &mut sfx::Sfx, sfx: &mut sfx::Sfx,
@ -940,9 +982,12 @@ impl SlimeData {
self.sprite_offset = 0; self.sprite_offset = 0;
} }
entity const IDLE: &Tag = TAG_MAP.get("slime idle");
.sprite
.set_tile_id((29 + self.sprite_offset / 16) * 4); let sprite = IDLE.get_sprite(self.sprite_offset as usize / 16);
let sprite = controller.get_sprite(sprite).unwrap();
entity.sprite.set_sprite(sprite);
if (player.entity.position - entity.position).manhattan_distance() < 40.into() { if (player.entity.position - entity.position).manhattan_distance() < 40.into() {
let direction = match player.entity.position.x.cmp(&entity.position.x) { let direction = match player.entity.position.x.cmp(&entity.position.x) {
@ -977,7 +1022,12 @@ impl SlimeData {
sfx.slime_boing(); sfx.slime_boing();
} }
entity.sprite.set_tile_id((frame + 31) * 4); const CHASE: &Tag = TAG_MAP.get("Slime jump");
let sprite = CHASE.get_sprite(frame as usize);
let sprite = controller.get_sprite(sprite).unwrap();
entity.sprite.set_sprite(sprite);
entity.velocity.x = match frame { entity.velocity.x = match frame {
2 | 3 | 4 => (Number::new(1) / 5) * Number::new(*direction as i32), 2 | 3 | 4 => (Number::new(1) / 5) * Number::new(*direction as i32),
@ -1003,7 +1053,11 @@ impl SlimeData {
} }
SlimeState::Dead(count) => { SlimeState::Dead(count) => {
if *count < 5 * 4 { if *count < 5 * 4 {
entity.sprite.set_tile_id((36 + *count / 4) * 4); const DEATH: &Tag = TAG_MAP.get("Slime death");
let sprite = DEATH.get_sprite(*count as usize / 4);
let sprite = controller.get_sprite(sprite).unwrap();
entity.sprite.set_sprite(sprite);
*count += 1; *count += 1;
} else { } else {
return UpdateInstruction::Remove; return UpdateInstruction::Remove;
@ -1033,9 +1087,10 @@ impl MiniFlameData {
} }
} }
fn update( fn update<'a>(
&mut self, &mut self,
entity: &mut Entity, controller: &'a ObjectController,
entity: &mut Entity<'a>,
player: &Player, player: &Player,
_level: &Level, _level: &Level,
sfx: &mut sfx::Sfx, sfx: &mut sfx::Sfx,
@ -1051,6 +1106,8 @@ impl MiniFlameData {
self.sprite_offset += 1; self.sprite_offset += 1;
const ANGRY: &Tag = TAG_MAP.get("angry boss");
match &mut self.state { match &mut self.state {
MiniFlameState::Idle(frames) => { MiniFlameState::Idle(frames) => {
*frames -= 1; *frames -= 1;
@ -1065,13 +1122,9 @@ impl MiniFlameData {
entity.velocity = resulting_direction.normalise() * Number::new(2); entity.velocity = resulting_direction.normalise() * Number::new(2);
} }
} else { } else {
if self.sprite_offset >= 12 * 8 { let sprite = ANGRY.get_animation_sprite(self.sprite_offset as usize / 8);
self.sprite_offset = 0; let sprite = controller.get_sprite(sprite).unwrap();
} entity.sprite.set_sprite(sprite);
entity
.sprite
.set_tile_id((137 + self.sprite_offset / 8) * 4);
entity.velocity = (0.into(), Number::new(-1) / Number::new(4)).into(); entity.velocity = (0.into(), Number::new(-1) / Number::new(4)).into();
} }
@ -1113,17 +1166,13 @@ impl MiniFlameData {
instruction = UpdateInstruction::DamagePlayer; instruction = UpdateInstruction::DamagePlayer;
} }
if self.sprite_offset >= 12 * 2 {
self.sprite_offset = 0;
}
if entity.velocity.manhattan_distance() < Number::new(1) / Number::new(4) { if entity.velocity.manhattan_distance() < Number::new(1) / Number::new(4) {
self.state = MiniFlameState::Idle(90); self.state = MiniFlameState::Idle(90);
} }
entity let sprite = ANGRY.get_animation_sprite(self.sprite_offset as usize / 2);
.sprite let sprite = controller.get_sprite(sprite).unwrap();
.set_tile_id((137 + self.sprite_offset / 2) * 4); entity.sprite.set_sprite(sprite);
} }
MiniFlameState::Dead => { MiniFlameState::Dead => {
entity.velocity = (0, 0).into(); entity.velocity = (0, 0).into();
@ -1131,9 +1180,11 @@ impl MiniFlameData {
instruction = UpdateInstruction::Remove; instruction = UpdateInstruction::Remove;
} }
entity const DEATH: &Tag = TAG_MAP.get("angry boss dead");
.sprite
.set_tile_id((148 + self.sprite_offset / 12) * 4); let sprite = DEATH.get_animation_sprite(self.sprite_offset as usize / 12);
let sprite = controller.get_sprite(sprite).unwrap();
entity.sprite.set_sprite(sprite);
self.sprite_offset += 1; self.sprite_offset += 1;
} }
@ -1165,9 +1216,10 @@ impl EmuData {
} }
} }
fn update( fn update<'a>(
&mut self, &mut self,
entity: &mut Entity, controller: &'a ObjectController,
entity: &mut Entity<'a>,
player: &Player, player: &Player,
level: &Level, level: &Level,
sfx: &mut sfx::Sfx, sfx: &mut sfx::Sfx,
@ -1189,9 +1241,11 @@ impl EmuData {
self.sprite_offset = 0; self.sprite_offset = 0;
} }
entity const IDLE: &Tag = TAG_MAP.get("emu - idle");
.sprite
.set_tile_id((170 + self.sprite_offset / 16) * 4); let sprite = IDLE.get_sprite(self.sprite_offset as usize / 16);
let sprite = controller.get_sprite(sprite).unwrap();
entity.sprite.set_sprite(sprite);
if (entity.position.y - player.entity.position.y).abs() < 10.into() { if (entity.position.y - player.entity.position.y).abs() < 10.into() {
let velocity = Number::new(1) let velocity = Number::new(1)
@ -1234,9 +1288,11 @@ impl EmuData {
sfx.emu_step(); sfx.emu_step();
} }
entity const WALK: &Tag = TAG_MAP.get("emu-walk");
.sprite
.set_tile_id((173 + self.sprite_offset / 2) * 4); let sprite = WALK.get_sprite(self.sprite_offset as usize / 2);
let sprite = controller.get_sprite(sprite).unwrap();
entity.sprite.set_sprite(sprite);
let gravity: Number = 1.into(); let gravity: Number = 1.into();
let gravity = gravity / 16; let gravity = gravity / 16;
@ -1287,9 +1343,12 @@ impl EmuData {
instruction = UpdateInstruction::Remove; instruction = UpdateInstruction::Remove;
} }
entity const DEATH: &Tag = TAG_MAP.get("emu - die");
.sprite
.set_tile_id((177 + self.sprite_offset / 4) * 4); let sprite = DEATH.get_animation_sprite(self.sprite_offset as usize / 4);
let sprite = controller.get_sprite(sprite).unwrap();
entity.sprite.set_sprite(sprite);
self.sprite_offset += 1; self.sprite_offset += 1;
} }
} }
@ -1317,27 +1376,32 @@ impl EnemyData {
} }
} }
fn tile_id(&self) -> u16 { fn sprite(&self) -> &'static Sprite {
const SLIME: &Tag = TAG_MAP.get("slime idle");
const BAT: &Tag = TAG_MAP.get("bat");
const MINI_FLAME: &Tag = TAG_MAP.get("angry boss");
const EMU: &Tag = TAG_MAP.get("emu - idle");
match self { match self {
EnemyData::Slime(_) => 29, EnemyData::Slime(_) => SLIME.get_sprite(0),
EnemyData::Bat(_) => 78, EnemyData::Bat(_) => BAT.get_sprite(0),
EnemyData::MiniFlame(_) => 137, EnemyData::MiniFlame(_) => MINI_FLAME.get_sprite(0),
EnemyData::Emu(_) => 170, EnemyData::Emu(_) => EMU.get_sprite(0),
} }
} }
fn update( fn update<'a>(
&mut self, &mut self,
entity: &mut Entity, controller: &'a ObjectController,
entity: &mut Entity<'a>,
player: &Player, player: &Player,
level: &Level, level: &Level,
sfx: &mut sfx::Sfx, sfx: &mut sfx::Sfx,
) -> UpdateInstruction { ) -> UpdateInstruction {
match self { match self {
EnemyData::Slime(data) => data.update(entity, player, level, sfx), EnemyData::Slime(data) => data.update(controller, entity, player, level, sfx),
EnemyData::Bat(data) => data.update(entity, player, level, sfx), EnemyData::Bat(data) => data.update(controller, entity, player, level, sfx),
EnemyData::MiniFlame(data) => data.update(entity, player, level, sfx), EnemyData::MiniFlame(data) => data.update(controller, entity, player, level, sfx),
EnemyData::Emu(data) => data.update(entity, player, level, sfx), EnemyData::Emu(data) => data.update(controller, entity, player, level, sfx),
} }
} }
} }
@ -1348,13 +1412,13 @@ struct Enemy<'a> {
} }
impl<'a> Enemy<'a> { impl<'a> Enemy<'a> {
fn new(object_controller: &'a ObjectControl, enemy_data: EnemyData) -> Self { fn new(object_controller: &'a ObjectController, enemy_data: EnemyData) -> Self {
let mut entity = Entity::new(object_controller, enemy_data.collision_mask()); let mut entity = Entity::new(object_controller, enemy_data.collision_mask());
entity let sprite = enemy_data.sprite();
.sprite let sprite = object_controller.get_sprite(sprite).unwrap();
.set_sprite_size(agb::display::object::Size::S16x16);
entity.sprite.set_tile_id(enemy_data.tile_id()); entity.sprite.set_sprite(sprite);
entity.sprite.show(); entity.sprite.show();
entity.sprite.commit(); entity.sprite.commit();
@ -1362,8 +1426,15 @@ impl<'a> Enemy<'a> {
Self { entity, enemy_data } Self { entity, enemy_data }
} }
fn update(&mut self, player: &Player, level: &Level, sfx: &mut sfx::Sfx) -> UpdateInstruction { fn update(
self.enemy_data.update(&mut self.entity, player, level, sfx) &mut self,
controller: &'a ObjectController,
player: &Player,
level: &Level,
sfx: &mut sfx::Sfx,
) -> UpdateInstruction {
self.enemy_data
.update(controller, &mut self.entity, player, level, sfx)
} }
} }
@ -1394,9 +1465,10 @@ impl ParticleData {
} }
} }
fn update( fn update<'a>(
&mut self, &mut self,
entity: &mut Entity, controller: &'a ObjectController,
entity: &mut Entity<'a>,
player: &Player, player: &Player,
_level: &Level, _level: &Level,
) -> UpdateInstruction { ) -> UpdateInstruction {
@ -1406,7 +1478,11 @@ impl ParticleData {
return UpdateInstruction::Remove; return UpdateInstruction::Remove;
} }
entity.sprite.set_tile_id((70 + *frame / 3) * 4); const DUST: &Tag = TAG_MAP.get("dust");
let sprite = DUST.get_sprite(*frame as usize / 3);
let sprite = controller.get_sprite(sprite).unwrap();
entity.sprite.set_sprite(sprite);
*frame += 1; *frame += 1;
UpdateInstruction::None UpdateInstruction::None
@ -1416,7 +1492,11 @@ impl ParticleData {
return UpdateInstruction::Remove; // have played the animation 6 times return UpdateInstruction::Remove; // have played the animation 6 times
} }
entity.sprite.set_tile_id((88 + (*frame / 3) % 8) * 4); const HEALTH: &Tag = TAG_MAP.get("Heath");
let sprite = HEALTH.get_animation_sprite(*frame as usize / 3);
let sprite = controller.get_sprite(sprite).unwrap();
entity.sprite.set_sprite(sprite);
if *frame < 8 * 3 * 3 { if *frame < 8 * 3 * 3 {
entity.velocity.y = Number::new(-1) / 2; entity.velocity.y = Number::new(-1) / 2;
@ -1438,7 +1518,11 @@ impl ParticleData {
UpdateInstruction::None UpdateInstruction::None
} }
ParticleData::BossHealer(frame, target) => { ParticleData::BossHealer(frame, target) => {
entity.sprite.set_tile_id((88 + (*frame / 3) % 8) * 4); const HEALTH: &Tag = TAG_MAP.get("Heath");
let sprite = HEALTH.get_animation_sprite(*frame as usize / 3);
let sprite = controller.get_sprite(sprite).unwrap();
entity.sprite.set_sprite(sprite);
if *frame < 8 * 3 * 3 { if *frame < 8 * 3 * 3 {
entity.velocity.y = Number::new(-1) / 2; entity.velocity.y = Number::new(-1) / 2;
@ -1471,7 +1555,7 @@ struct Particle<'a> {
impl<'a> Particle<'a> { impl<'a> Particle<'a> {
fn new( fn new(
object_controller: &'a ObjectControl, object_controller: &'a ObjectController,
particle_data: ParticleData, particle_data: ParticleData,
position: Vector2D<Number>, position: Vector2D<Number>,
) -> Self { ) -> Self {
@ -1480,11 +1564,6 @@ impl<'a> Particle<'a> {
Rect::new((0u16, 0u16).into(), (0u16, 0u16).into()), Rect::new((0u16, 0u16).into(), (0u16, 0u16).into()),
); );
entity
.sprite
.set_sprite_size(agb::display::object::Size::S16x16);
entity.sprite.set_tile_id(particle_data.tile_id() * 4);
entity.sprite.show();
entity.position = position; entity.position = position;
Self { Self {
@ -1493,8 +1572,15 @@ impl<'a> Particle<'a> {
} }
} }
fn update(&mut self, player: &Player, level: &Level) -> UpdateInstruction { fn update(
self.particle_data.update(&mut self.entity, player, level) &mut self,
controller: &'a ObjectController,
player: &Player,
level: &Level,
) -> UpdateInstruction {
self.entity.sprite.show();
self.particle_data
.update(controller, &mut self.entity, player, level)
} }
} }
@ -1515,14 +1601,14 @@ impl<'a> BossState<'a> {
fn update( fn update(
&mut self, &mut self,
enemies: &mut Arena<Enemy<'a>>, enemies: &mut Arena<Enemy<'a>>,
object_controller: &'a ObjectControl, object_controller: &'a ObjectController,
player: &Player, player: &Player,
sfx: &mut sfx::Sfx, sfx: &mut sfx::Sfx,
) -> BossInstruction { ) -> BossInstruction {
match self { match self {
BossState::Active(boss) => boss.update(enemies, object_controller, player, sfx), BossState::Active(boss) => boss.update(enemies, object_controller, player, sfx),
BossState::Following(boss) => { BossState::Following(boss) => {
boss.update(player); boss.update(object_controller, player);
BossInstruction::None BossInstruction::None
} }
BossState::NotSpawned => BossInstruction::None, BossState::NotSpawned => BossInstruction::None,
@ -1550,15 +1636,13 @@ struct FollowingBoss<'a> {
} }
impl<'a> FollowingBoss<'a> { impl<'a> FollowingBoss<'a> {
fn new(object_controller: &'a ObjectControl, position: Vector2D<Number>) -> Self { fn new(object_controller: &'a ObjectController, position: Vector2D<Number>) -> Self {
let mut entity = Entity::new( let mut entity = Entity::new(
object_controller, object_controller,
Rect::new((0_u16, 0_u16).into(), (0_u16, 0_u16).into()), Rect::new((0_u16, 0_u16).into(), (0_u16, 0_u16).into()),
); );
entity.position = position; entity.position = position;
entity
.sprite
.set_sprite_size(agb::display::object::Size::S16x16);
Self { Self {
entity, entity,
following: true, following: true,
@ -1567,11 +1651,11 @@ impl<'a> FollowingBoss<'a> {
gone: false, gone: false,
} }
} }
fn update(&mut self, player: &Player) { fn update(&mut self, controller: &'a ObjectController, player: &Player) {
let difference = player.entity.position - self.entity.position; let difference = player.entity.position - self.entity.position;
self.timer += 1; self.timer += 1;
if self.to_hole { let frame = if self.to_hole {
let target: Vector2D<Number> = (17 * 8, -3 * 8).into(); let target: Vector2D<Number> = (17 * 8, -3 * 8).into();
let difference = target - self.entity.position; let difference = target - self.entity.position;
if difference.manhattan_distance() < 1.into() { if difference.manhattan_distance() < 1.into() {
@ -1580,26 +1664,30 @@ impl<'a> FollowingBoss<'a> {
self.entity.velocity = difference.normalise() * 2; self.entity.velocity = difference.normalise() * 2;
} }
let frame = (self.timer / 8) % 12; self.timer / 8
self.entity.sprite.set_tile_id((125 + frame as u16) * 4);
} else if self.timer < 120 { } else if self.timer < 120 {
let frame = (self.timer / 20) % 12; self.timer / 20
self.entity.sprite.set_tile_id((125 + frame as u16) * 4);
} else if self.following { } else if self.following {
self.entity.velocity = difference / 16; self.entity.velocity = difference / 16;
if difference.manhattan_distance() < 20.into() { if difference.manhattan_distance() < 20.into() {
self.following = false; self.following = false;
} }
let frame = (self.timer / 8) % 12; self.timer / 8
self.entity.sprite.set_tile_id((125 + frame as u16) * 4);
} else { } else {
self.entity.velocity = (0, 0).into(); self.entity.velocity = (0, 0).into();
if difference.manhattan_distance() > 60.into() { if difference.manhattan_distance() > 60.into() {
self.following = true; self.following = true;
} }
let frame = (self.timer / 16) % 12; self.timer / 16
self.entity.sprite.set_tile_id((125 + frame as u16) * 4); };
}
const BOSS: &Tag = TAG_MAP.get("happy boss");
let sprite = BOSS.get_animation_sprite(frame as usize);
let sprite = controller.get_sprite(sprite).unwrap();
self.entity.sprite.set_sprite(sprite);
self.entity.update_position_without_collision(); self.entity.update_position_without_collision();
} }
@ -1632,15 +1720,11 @@ enum BossInstruction {
} }
impl<'a> Boss<'a> { impl<'a> Boss<'a> {
fn new(object_controller: &'a ObjectControl, screen_coords: Vector2D<Number>) -> Self { fn new(object_controller: &'a ObjectController, screen_coords: Vector2D<Number>) -> Self {
let mut entity = Entity::new( let mut entity = Entity::new(
object_controller, object_controller,
Rect::new((0_u16, 0_u16).into(), (28_u16, 28_u16).into()), Rect::new((0_u16, 0_u16).into(), (28_u16, 28_u16).into()),
); );
entity
.sprite
.set_sprite_size(agb::display::object::Size::S32x32);
entity.sprite.set_palette(1);
entity.position = screen_coords + (144, 136).into(); entity.position = screen_coords + (144, 136).into();
Self { Self {
entity, entity,
@ -1655,7 +1739,7 @@ impl<'a> Boss<'a> {
fn update( fn update(
&mut self, &mut self,
enemies: &mut Arena<Enemy<'a>>, enemies: &mut Arena<Enemy<'a>>,
object_controller: &'a ObjectControl, object_controller: &'a ObjectController,
player: &Player, player: &Player,
sfx: &mut sfx::Sfx, sfx: &mut sfx::Sfx,
) -> BossInstruction { ) -> BossInstruction {
@ -1731,8 +1815,14 @@ impl<'a> Boss<'a> {
BossActiveState::WaitUntilKilled => 3.into(), BossActiveState::WaitUntilKilled => 3.into(),
}; };
self.timer += 1; self.timer += 1;
let frame = (self.timer / animation_rate) % 12; let frame = self.timer / animation_rate;
self.entity.sprite.set_tile_id(784 + (frame as u16) * 16);
const BOSS: &Tag = TAG_MAP.get("Boss");
let sprite = BOSS.get_animation_sprite(frame as usize);
let sprite = object_controller.get_sprite(sprite).unwrap();
self.entity.sprite.set_sprite(sprite);
self.entity.update_position_without_collision(); self.entity.update_position_without_collision();
instruction instruction
@ -1753,7 +1843,7 @@ impl<'a> Boss<'a> {
self.entity self.entity
.commit_with_size(offset + shake, (32, 32).into()); .commit_with_size(offset + shake, (32, 32).into());
} }
fn explode(&self, enemies: &mut Arena<Enemy<'a>>, object_controller: &'a ObjectControl) { fn explode(&self, enemies: &mut Arena<Enemy<'a>>, object_controller: &'a ObjectController) {
for _ in 0..(6 - self.health) { for _ in 0..(6 - self.health) {
let x_offset: Number = Number::from_raw(get_random()).rem_euclid(2.into()) - 1; let x_offset: Number = Number::from_raw(get_random()).rem_euclid(2.into()) - 1;
let y_offset: Number = Number::from_raw(get_random()).rem_euclid(2.into()) - 1; let y_offset: Number = Number::from_raw(get_random()).rem_euclid(2.into()) - 1;
@ -1825,7 +1915,7 @@ impl<'a> Game<'a> {
fn advance_frame( fn advance_frame(
&mut self, &mut self,
object_controller: &'a ObjectControl, object_controller: &'a ObjectController,
sfx: &mut sfx::Sfx, sfx: &mut sfx::Sfx,
) -> GameStatus { ) -> GameStatus {
let mut state = GameStatus::Continue; let mut state = GameStatus::Continue;
@ -1921,7 +2011,7 @@ impl<'a> Game<'a> {
self.input.update(); self.input.update();
if let UpdateInstruction::CreateParticle(data, position) = if let UpdateInstruction::CreateParticle(data, position) =
self.player.update(&self.input, &self.level, sfx) self.player.update(object_controller, &self.input, &self.level, sfx)
{ {
let new_particle = Particle::new(object_controller, data, position); let new_particle = Particle::new(object_controller, data, position);
@ -1935,7 +2025,7 @@ impl<'a> Game<'a> {
continue; continue;
} }
match enemy.update(&self.player, &self.level, sfx) { match enemy.update(object_controller, &self.player, &self.level, sfx) {
UpdateInstruction::Remove => { UpdateInstruction::Remove => {
remove.push(idx); remove.push(idx);
} }
@ -1989,7 +2079,7 @@ impl<'a> Game<'a> {
let mut remove = Vec::with_capacity(10); let mut remove = Vec::with_capacity(10);
for (idx, particle) in self.particles.iter_mut() { for (idx, particle) in self.particles.iter_mut() {
match particle.update(&self.player, &self.level) { match particle.update(object_controller, &self.player, &self.level) {
UpdateInstruction::Remove => remove.push(idx), UpdateInstruction::Remove => remove.push(idx),
UpdateInstruction::HealBossAndRemove => { UpdateInstruction::HealBossAndRemove => {
sfx.sunrise(); sfx.sunrise();
@ -2040,7 +2130,7 @@ impl<'a> Game<'a> {
} }
} }
fn load_enemies(&mut self, object_controller: &'a ObjectControl) { fn load_enemies(&mut self, object_controller: &'a ObjectController) {
if self.slime_load < self.level.slime_spawns.len() { if self.slime_load < self.level.slime_spawns.len() {
for (idx, slime_spawn) in self for (idx, slime_spawn) in self
.level .level
@ -2111,7 +2201,7 @@ impl<'a> Game<'a> {
} }
fn new( fn new(
object: &'a ObjectControl, object: &'a ObjectController,
level: Level, level: Level,
background_distributor: &'a mut BackgroundDistributor, background_distributor: &'a mut BackgroundDistributor,
start_at_boss: bool, start_at_boss: bool,
@ -2146,16 +2236,6 @@ impl<'a> Game<'a> {
} }
fn game_with_level(gba: &mut agb::Gba) { fn game_with_level(gba: &mut agb::Gba) {
{
let object = gba.display.object.get();
object.set_sprite_palettes(&[
objects::objects.palettes[0].clone(),
objects::boss.palettes[0].clone(),
]);
object.set_sprite_tilemap(objects::objects.tiles);
object.set_sprite_tilemap_at_idx(8192 - objects::boss.tiles.len(), objects::boss.tiles);
}
let vblank = agb::interrupt::VBlank::get(); let vblank = agb::interrupt::VBlank::get();
vblank.wait_for_vblank(); vblank.wait_for_vblank();
@ -2172,8 +2252,7 @@ fn game_with_level(gba: &mut agb::Gba) {
let mut background = gba.display.video.tiled0(); let mut background = gba.display.video.tiled0();
background.set_background_palettes(background::background.palettes); background.set_background_palettes(background::background.palettes);
background.set_background_tilemap(0, background::background.tiles); background.set_background_tilemap(0, background::background.tiles);
let mut object = gba.display.object.get(); let object = gba.display.object.get();
object.enable();
let mut game = Game::new( let mut game = Game::new(
&object, &object,