mirror of
https://github.com/italicsjenga/agb.git
synced 2025-01-26 00:56:38 +11:00
update to use new sprite system
This commit is contained in:
parent
4c57f85acd
commit
9dca54c9bb
3 changed files with 428 additions and 205 deletions
170
examples/the-purple-night/Cargo.lock
generated
170
examples/the-purple-night/Cargo.lock
generated
|
@ -24,6 +24,9 @@ dependencies = [
|
|||
"agb_sound_converter",
|
||||
"bare-metal",
|
||||
"bitflags",
|
||||
"hashbrown",
|
||||
"modular-bitfield",
|
||||
"rustc-hash",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
|
@ -37,6 +40,7 @@ dependencies = [
|
|||
name = "agb_image_converter"
|
||||
version = "0.6.0"
|
||||
dependencies = [
|
||||
"asefile",
|
||||
"image",
|
||||
"proc-macro2",
|
||||
"quote",
|
||||
|
@ -64,6 +68,31 @@ dependencies = [
|
|||
"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]]
|
||||
name = "autocfg"
|
||||
version = "1.1.0"
|
||||
|
@ -132,11 +161,24 @@ dependencies = [
|
|||
|
||||
[[package]]
|
||||
name = "deflate"
|
||||
version = "1.0.0"
|
||||
version = "0.8.6"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "c86f7e25f518f4b81808a2cf1c50996a61f5c2eb394b2393bd87f2a4780a432f"
|
||||
checksum = "73770f8e1fe7d64df17ca66ad28994a0a623ea497fa69486e14984e715c5d174"
|
||||
dependencies = [
|
||||
"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]]
|
||||
|
@ -148,6 +190,26 @@ dependencies = [
|
|||
"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]]
|
||||
name = "hound"
|
||||
version = "3.4.0"
|
||||
|
@ -156,9 +218,9 @@ checksum = "8a164bb2ceaeff4f42542bdb847c41517c78a60f5649671b2a07312b6e117549"
|
|||
|
||||
[[package]]
|
||||
name = "image"
|
||||
version = "0.24.1"
|
||||
version = "0.23.14"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "db207d030ae38f1eb6f240d5a1c1c88ff422aa005d10f8c6c6fc5e75286ab30e"
|
||||
checksum = "24ffcb7e7244a9bf19d35bf2883b9c080c4ced3c07a9895572178cdb8f13f6a1"
|
||||
dependencies = [
|
||||
"bytemuck",
|
||||
"byteorder",
|
||||
|
@ -169,6 +231,12 @@ dependencies = [
|
|||
"png",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "libc"
|
||||
version = "0.2.119"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "1bf2e165bb3457c8e098ea76f3e3bc9db55f87aa90d52d0e6be741470916aaa4"
|
||||
|
||||
[[package]]
|
||||
name = "libflate"
|
||||
version = "0.1.27"
|
||||
|
@ -182,14 +250,60 @@ dependencies = [
|
|||
]
|
||||
|
||||
[[package]]
|
||||
name = "miniz_oxide"
|
||||
version = "0.5.1"
|
||||
name = "log"
|
||||
version = "0.4.14"
|
||||
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 = [
|
||||
"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]]
|
||||
name = "num-integer"
|
||||
version = "0.1.44"
|
||||
|
@ -213,9 +327,9 @@ dependencies = [
|
|||
|
||||
[[package]]
|
||||
name = "num-rational"
|
||||
version = "0.4.0"
|
||||
version = "0.3.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "d41702bd167c2df5520b384281bc111a4b5efcf7fbc4c9c222c815b07e0a6a6a"
|
||||
checksum = "12ac428b1cb17fce6f731001d307d351ec70a6d202fc2e60f7d4c5e42d8f4f07"
|
||||
dependencies = [
|
||||
"autocfg",
|
||||
"num-integer",
|
||||
|
@ -232,15 +346,21 @@ dependencies = [
|
|||
]
|
||||
|
||||
[[package]]
|
||||
name = "png"
|
||||
version = "0.17.5"
|
||||
name = "once_cell"
|
||||
version = "1.10.0"
|
||||
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 = [
|
||||
"bitflags",
|
||||
"crc32fast",
|
||||
"deflate",
|
||||
"miniz_oxide",
|
||||
"miniz_oxide 0.3.7",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
|
@ -267,6 +387,12 @@ version = "1.0.3"
|
|||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "3582f63211428f83597b51b2ddb88e2a91a9d52d12831f9d08f5e624e8977422"
|
||||
|
||||
[[package]]
|
||||
name = "rustc-hash"
|
||||
version = "1.1.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "08d43f7aa6b08d49f382cde6a7982047c3426db949b1424bc4b7ec9ae12c6ce2"
|
||||
|
||||
[[package]]
|
||||
name = "serde"
|
||||
version = "1.0.136"
|
||||
|
@ -287,6 +413,12 @@ dependencies = [
|
|||
"syn",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "static_assertions"
|
||||
version = "1.1.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "a2eb9349b6444b326872e140eb1cf5e7c522154d69e7a0ffb0fb81c06b37543f"
|
||||
|
||||
[[package]]
|
||||
name = "syn"
|
||||
version = "1.0.86"
|
||||
|
@ -340,6 +472,18 @@ version = "0.2.2"
|
|||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
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]]
|
||||
name = "xml-rs"
|
||||
version = "0.8.4"
|
||||
|
|
Binary file not shown.
|
@ -15,7 +15,7 @@ use rng::get_random;
|
|||
use agb::{
|
||||
display::{
|
||||
background::{BackgroundDistributor, BackgroundRegular},
|
||||
object::{ObjectControl, ObjectStandard},
|
||||
object::{Object, ObjectController, Sprite, Tag, TagMap},
|
||||
Priority, HEIGHT, WIDTH,
|
||||
},
|
||||
fixnum::{FixedNum, Rect, Vector2D},
|
||||
|
@ -23,7 +23,34 @@ use agb::{
|
|||
};
|
||||
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");
|
||||
|
||||
type Number = FixedNum<8>;
|
||||
|
@ -127,7 +154,7 @@ impl Level {
|
|||
}
|
||||
|
||||
struct Entity<'a> {
|
||||
sprite: ObjectStandard<'a>,
|
||||
sprite: Object<'a, 'a>,
|
||||
position: Vector2D<Number>,
|
||||
velocity: Vector2D<Number>,
|
||||
collision_mask: Rect<u16>,
|
||||
|
@ -135,8 +162,11 @@ struct Entity<'a> {
|
|||
}
|
||||
|
||||
impl<'a> Entity<'a> {
|
||||
fn new(object_controller: &'a ObjectControl, collision_mask: Rect<u16>) -> Self {
|
||||
let mut sprite = object_controller.get_object_standard();
|
||||
fn new(object_controller: &'a ObjectController, collision_mask: Rect<u16>) -> Self {
|
||||
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);
|
||||
Entity {
|
||||
sprite,
|
||||
|
@ -321,34 +351,30 @@ impl SwordState {
|
|||
SwordState::Swordless => Number::new(6) / 256,
|
||||
}
|
||||
}
|
||||
fn idle_animation(self, counter: &mut u16) -> u16 {
|
||||
if *counter >= 4 * 8 {
|
||||
*counter = 0;
|
||||
}
|
||||
fn idle_animation(self, counter: u16) -> &'static Sprite {
|
||||
let counter = counter as usize;
|
||||
match self {
|
||||
SwordState::LongSword => (*counter / 8) * 4,
|
||||
SwordState::ShortSword => (41 + *counter / 8) * 4,
|
||||
SwordState::Dagger => (96 + *counter / 8) * 4,
|
||||
SwordState::Swordless => (154 + *counter / 8) * 4,
|
||||
SwordState::LongSword => LONGSWORD_IDLE.get_animation_sprite(counter / 8),
|
||||
SwordState::ShortSword => SHORTSWORD_IDLE.get_animation_sprite(counter / 8),
|
||||
SwordState::Dagger => KNIFE_IDLE.get_animation_sprite(counter / 8),
|
||||
SwordState::Swordless => SWORDLESS_IDLE.get_animation_sprite(counter / 8),
|
||||
}
|
||||
}
|
||||
fn jump_offset(self) -> u16 {
|
||||
fn jump_tag(self) -> &'static Tag {
|
||||
match self {
|
||||
SwordState::LongSword => 10,
|
||||
SwordState::ShortSword => 51,
|
||||
SwordState::Dagger => 106,
|
||||
SwordState::Swordless => 164,
|
||||
SwordState::LongSword => LONGSWORD_JUMP,
|
||||
SwordState::ShortSword => SHORTSWORD_JUMP,
|
||||
SwordState::Dagger => KNIFE_JUMP,
|
||||
SwordState::Swordless => SWORDLESS_JUMP,
|
||||
}
|
||||
}
|
||||
fn walk_animation(self, counter: &mut u16) -> u16 {
|
||||
if *counter >= 6 * 4 {
|
||||
*counter = 0;
|
||||
}
|
||||
fn walk_animation(self, counter: u16) -> &'static Sprite {
|
||||
let counter = counter as usize;
|
||||
match self {
|
||||
SwordState::LongSword => (4 + *counter / 4) * 4,
|
||||
SwordState::ShortSword => (45 + *counter / 4) * 4,
|
||||
SwordState::Dagger => (100 + *counter / 4) * 4,
|
||||
SwordState::Swordless => (158 + *counter / 4) * 4,
|
||||
SwordState::LongSword => LONGSWORD_WALK.get_animation_sprite(counter / 4),
|
||||
SwordState::ShortSword => SHORTSWORD_WALK.get_animation_sprite(counter / 4),
|
||||
SwordState::Dagger => KNIFE_WALK.get_animation_sprite(counter / 4),
|
||||
SwordState::Swordless => SWORDLESS_WALK.get_animation_sprite(counter / 4),
|
||||
}
|
||||
}
|
||||
fn attack_duration(self) -> u16 {
|
||||
|
@ -375,20 +401,20 @@ impl SwordState {
|
|||
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 {
|
||||
(self.jump_attack_duration() - timer) / 8
|
||||
}
|
||||
fn hold_frame(self) -> u16 {
|
||||
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 {
|
||||
match self {
|
||||
|
@ -398,26 +424,15 @@ impl SwordState {
|
|||
SwordState::Swordless => 0,
|
||||
}
|
||||
}
|
||||
fn to_sprite_id(self, frame: u16) -> u16 {
|
||||
fn attack_tag(self) -> &'static Tag {
|
||||
match self {
|
||||
SwordState::LongSword => (16 + frame) * 4,
|
||||
SwordState::ShortSword => (57 + frame) * 4,
|
||||
SwordState::Dagger => (112 + frame) * 4,
|
||||
SwordState::Swordless => 0,
|
||||
}
|
||||
}
|
||||
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,
|
||||
}
|
||||
SwordState::LongSword => LONGSWORD_ATTACK,
|
||||
SwordState::ShortSword => SHORTSWORD_ATTACK,
|
||||
SwordState::Dagger => KNIFE_ATTACK,
|
||||
SwordState::Swordless => SWORDLESS_ATTACK,
|
||||
}
|
||||
}
|
||||
|
||||
fn fudge(self, frame: u16) -> i32 {
|
||||
match self {
|
||||
SwordState::LongSword => long_sword_fudge(frame),
|
||||
|
@ -520,15 +535,15 @@ struct 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(
|
||||
object_controller,
|
||||
Rect::new((0_u16, 0_u16).into(), (4_u16, 12_u16).into()),
|
||||
);
|
||||
entity
|
||||
.sprite
|
||||
.set_sprite_size(agb::display::object::Size::S16x16);
|
||||
entity.sprite.set_tile_id(0);
|
||||
let s = object_controller
|
||||
.get_sprite(LONGSWORD_IDLE.get_sprite(0))
|
||||
.unwrap();
|
||||
entity.sprite.set_sprite(s);
|
||||
entity.sprite.show();
|
||||
entity.position = (144, 0).into();
|
||||
entity.sprite.commit();
|
||||
|
@ -549,6 +564,7 @@ impl<'a> Player<'a> {
|
|||
|
||||
fn update(
|
||||
&mut self,
|
||||
controller: &'a ObjectController,
|
||||
buttons: &ButtonController,
|
||||
level: &Level,
|
||||
sfx: &mut sfx::Sfx,
|
||||
|
@ -580,13 +596,15 @@ impl<'a> Player<'a> {
|
|||
self.entity.sprite.set_hflip(self.facing == Tri::Negative);
|
||||
self.entity.velocity.x += self.sword.ground_walk_force() * x as i32;
|
||||
if self.entity.velocity.x.abs() > Number::new(1) / 10 {
|
||||
self.entity
|
||||
.sprite
|
||||
.set_tile_id(self.sword.walk_animation(&mut self.sprite_offset));
|
||||
let sprite = controller
|
||||
.get_sprite(self.sword.walk_animation(self.sprite_offset))
|
||||
.unwrap();
|
||||
self.entity.sprite.set_sprite(sprite);
|
||||
} else {
|
||||
self.entity
|
||||
.sprite
|
||||
.set_tile_id(self.sword.idle_animation(&mut self.sprite_offset));
|
||||
let sprite = controller
|
||||
.get_sprite(self.sword.idle_animation(self.sprite_offset))
|
||||
.unwrap();
|
||||
self.entity.sprite.set_sprite(sprite);
|
||||
}
|
||||
|
||||
if b_press && self.sword != SwordState::Swordless {
|
||||
|
@ -604,9 +622,11 @@ impl<'a> Player<'a> {
|
|||
*a -= 1;
|
||||
let frame = self.sword.attack_frame(*a);
|
||||
self.fudge_factor.x = self.sword.fudge(frame) * self.facing as i32;
|
||||
self.entity
|
||||
.sprite
|
||||
.set_tile_id(self.sword.to_sprite_id(frame));
|
||||
let tag = self.sword.attack_tag();
|
||||
let sprite = controller
|
||||
.get_sprite(tag.get_animation_sprite(frame as usize))
|
||||
.unwrap();
|
||||
self.entity.sprite.set_sprite(sprite);
|
||||
|
||||
hurtbox = self.sword.ground_attack_hurtbox(frame);
|
||||
|
||||
|
@ -618,9 +638,11 @@ impl<'a> Player<'a> {
|
|||
*a -= 1;
|
||||
let frame = self.sword.hold_frame();
|
||||
self.fudge_factor.x = self.sword.fudge(frame) * self.facing as i32;
|
||||
self.entity
|
||||
.sprite
|
||||
.set_tile_id(self.sword.to_sprite_id(frame));
|
||||
let tag = self.sword.attack_tag();
|
||||
let sprite = controller
|
||||
.get_sprite(tag.get_animation_sprite(frame as usize))
|
||||
.unwrap();
|
||||
self.entity.sprite.set_sprite(sprite);
|
||||
if *a == 0 {
|
||||
self.attack_timer = AttackTimer::Idle;
|
||||
}
|
||||
|
@ -632,7 +654,7 @@ impl<'a> Player<'a> {
|
|||
|
||||
match &mut self.attack_timer {
|
||||
AttackTimer::Idle => {
|
||||
let sprite = if self.sprite_offset < 3 * 4 {
|
||||
let frame = if self.sprite_offset < 3 * 4 {
|
||||
self.sprite_offset / 4
|
||||
} else if self.entity.velocity.y.abs() < Number::new(1) / 5 {
|
||||
3
|
||||
|
@ -643,9 +665,11 @@ impl<'a> Player<'a> {
|
|||
} else {
|
||||
2
|
||||
};
|
||||
self.entity
|
||||
.sprite
|
||||
.set_tile_id((sprite + self.sword.jump_offset()) * 4);
|
||||
let tag = self.sword.jump_tag();
|
||||
let sprite = controller
|
||||
.get_sprite(tag.get_animation_sprite(frame as usize))
|
||||
.unwrap();
|
||||
self.entity.sprite.set_sprite(sprite);
|
||||
|
||||
if x != Tri::Zero {
|
||||
self.facing = x;
|
||||
|
@ -665,9 +689,11 @@ impl<'a> Player<'a> {
|
|||
AttackTimer::Attack(a) => {
|
||||
*a -= 1;
|
||||
let frame = self.sword.jump_attack_frame(*a);
|
||||
self.entity
|
||||
.sprite
|
||||
.set_tile_id(self.sword.to_jump_sprite_id(frame));
|
||||
let tag = self.sword.jump_attack_tag();
|
||||
let sprite = controller
|
||||
.get_sprite(tag.get_animation_sprite(frame as usize))
|
||||
.unwrap();
|
||||
self.entity.sprite.set_sprite(sprite);
|
||||
|
||||
hurtbox = self.sword.air_attack_hurtbox(frame);
|
||||
|
||||
|
@ -799,9 +825,10 @@ impl BatData {
|
|||
}
|
||||
}
|
||||
|
||||
fn update(
|
||||
fn update<'a>(
|
||||
&mut self,
|
||||
entity: &mut Entity,
|
||||
controller: &'a ObjectController,
|
||||
entity: &mut Entity<'a>,
|
||||
player: &Player,
|
||||
level: &Level,
|
||||
sfx: &mut sfx::Sfx,
|
||||
|
@ -814,6 +841,8 @@ impl BatData {
|
|||
.unwrap_or(false);
|
||||
let should_damage = entity.collider().touches(player.entity.collider());
|
||||
|
||||
const BAT_IDLE: &Tag = TAG_MAP.get("bat");
|
||||
|
||||
match &mut self.bat_state {
|
||||
BatState::Idle => {
|
||||
self.sprite_offset += 1;
|
||||
|
@ -825,7 +854,10 @@ impl BatData {
|
|||
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() {
|
||||
self.bat_state = BatState::Chasing(300);
|
||||
|
@ -856,7 +888,11 @@ impl BatData {
|
|||
if self.sprite_offset >= 9 * 2 {
|
||||
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 {
|
||||
sfx.bat_flap();
|
||||
|
@ -879,7 +915,12 @@ impl BatData {
|
|||
}
|
||||
}
|
||||
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 = gravity / 16;
|
||||
entity.velocity.x = 0.into();
|
||||
|
@ -917,9 +958,10 @@ impl SlimeData {
|
|||
}
|
||||
}
|
||||
|
||||
fn update(
|
||||
fn update<'a>(
|
||||
&mut self,
|
||||
entity: &mut Entity,
|
||||
controller: &'a ObjectController,
|
||||
entity: &mut Entity<'a>,
|
||||
player: &Player,
|
||||
level: &Level,
|
||||
sfx: &mut sfx::Sfx,
|
||||
|
@ -940,9 +982,12 @@ impl SlimeData {
|
|||
self.sprite_offset = 0;
|
||||
}
|
||||
|
||||
entity
|
||||
.sprite
|
||||
.set_tile_id((29 + self.sprite_offset / 16) * 4);
|
||||
const IDLE: &Tag = TAG_MAP.get("slime idle");
|
||||
|
||||
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() {
|
||||
let direction = match player.entity.position.x.cmp(&entity.position.x) {
|
||||
|
@ -977,7 +1022,12 @@ impl SlimeData {
|
|||
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 {
|
||||
2 | 3 | 4 => (Number::new(1) / 5) * Number::new(*direction as i32),
|
||||
|
@ -1003,7 +1053,11 @@ impl SlimeData {
|
|||
}
|
||||
SlimeState::Dead(count) => {
|
||||
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;
|
||||
} else {
|
||||
return UpdateInstruction::Remove;
|
||||
|
@ -1033,9 +1087,10 @@ impl MiniFlameData {
|
|||
}
|
||||
}
|
||||
|
||||
fn update(
|
||||
fn update<'a>(
|
||||
&mut self,
|
||||
entity: &mut Entity,
|
||||
controller: &'a ObjectController,
|
||||
entity: &mut Entity<'a>,
|
||||
player: &Player,
|
||||
_level: &Level,
|
||||
sfx: &mut sfx::Sfx,
|
||||
|
@ -1051,6 +1106,8 @@ impl MiniFlameData {
|
|||
|
||||
self.sprite_offset += 1;
|
||||
|
||||
const ANGRY: &Tag = TAG_MAP.get("angry boss");
|
||||
|
||||
match &mut self.state {
|
||||
MiniFlameState::Idle(frames) => {
|
||||
*frames -= 1;
|
||||
|
@ -1065,13 +1122,9 @@ impl MiniFlameData {
|
|||
entity.velocity = resulting_direction.normalise() * Number::new(2);
|
||||
}
|
||||
} else {
|
||||
if self.sprite_offset >= 12 * 8 {
|
||||
self.sprite_offset = 0;
|
||||
}
|
||||
|
||||
entity
|
||||
.sprite
|
||||
.set_tile_id((137 + self.sprite_offset / 8) * 4);
|
||||
let sprite = ANGRY.get_animation_sprite(self.sprite_offset as usize / 8);
|
||||
let sprite = controller.get_sprite(sprite).unwrap();
|
||||
entity.sprite.set_sprite(sprite);
|
||||
|
||||
entity.velocity = (0.into(), Number::new(-1) / Number::new(4)).into();
|
||||
}
|
||||
|
@ -1113,17 +1166,13 @@ impl MiniFlameData {
|
|||
instruction = UpdateInstruction::DamagePlayer;
|
||||
}
|
||||
|
||||
if self.sprite_offset >= 12 * 2 {
|
||||
self.sprite_offset = 0;
|
||||
}
|
||||
|
||||
if entity.velocity.manhattan_distance() < Number::new(1) / Number::new(4) {
|
||||
self.state = MiniFlameState::Idle(90);
|
||||
}
|
||||
|
||||
entity
|
||||
.sprite
|
||||
.set_tile_id((137 + self.sprite_offset / 2) * 4);
|
||||
let sprite = ANGRY.get_animation_sprite(self.sprite_offset as usize / 2);
|
||||
let sprite = controller.get_sprite(sprite).unwrap();
|
||||
entity.sprite.set_sprite(sprite);
|
||||
}
|
||||
MiniFlameState::Dead => {
|
||||
entity.velocity = (0, 0).into();
|
||||
|
@ -1131,9 +1180,11 @@ impl MiniFlameData {
|
|||
instruction = UpdateInstruction::Remove;
|
||||
}
|
||||
|
||||
entity
|
||||
.sprite
|
||||
.set_tile_id((148 + self.sprite_offset / 12) * 4);
|
||||
const DEATH: &Tag = TAG_MAP.get("angry boss dead");
|
||||
|
||||
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;
|
||||
}
|
||||
|
@ -1165,9 +1216,10 @@ impl EmuData {
|
|||
}
|
||||
}
|
||||
|
||||
fn update(
|
||||
fn update<'a>(
|
||||
&mut self,
|
||||
entity: &mut Entity,
|
||||
controller: &'a ObjectController,
|
||||
entity: &mut Entity<'a>,
|
||||
player: &Player,
|
||||
level: &Level,
|
||||
sfx: &mut sfx::Sfx,
|
||||
|
@ -1189,9 +1241,11 @@ impl EmuData {
|
|||
self.sprite_offset = 0;
|
||||
}
|
||||
|
||||
entity
|
||||
.sprite
|
||||
.set_tile_id((170 + self.sprite_offset / 16) * 4);
|
||||
const IDLE: &Tag = TAG_MAP.get("emu - idle");
|
||||
|
||||
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() {
|
||||
let velocity = Number::new(1)
|
||||
|
@ -1234,9 +1288,11 @@ impl EmuData {
|
|||
sfx.emu_step();
|
||||
}
|
||||
|
||||
entity
|
||||
.sprite
|
||||
.set_tile_id((173 + self.sprite_offset / 2) * 4);
|
||||
const WALK: &Tag = TAG_MAP.get("emu-walk");
|
||||
|
||||
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 = gravity / 16;
|
||||
|
@ -1287,9 +1343,12 @@ impl EmuData {
|
|||
instruction = UpdateInstruction::Remove;
|
||||
}
|
||||
|
||||
entity
|
||||
.sprite
|
||||
.set_tile_id((177 + self.sprite_offset / 4) * 4);
|
||||
const DEATH: &Tag = TAG_MAP.get("emu - die");
|
||||
|
||||
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;
|
||||
}
|
||||
}
|
||||
|
@ -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 {
|
||||
EnemyData::Slime(_) => 29,
|
||||
EnemyData::Bat(_) => 78,
|
||||
EnemyData::MiniFlame(_) => 137,
|
||||
EnemyData::Emu(_) => 170,
|
||||
EnemyData::Slime(_) => SLIME.get_sprite(0),
|
||||
EnemyData::Bat(_) => BAT.get_sprite(0),
|
||||
EnemyData::MiniFlame(_) => MINI_FLAME.get_sprite(0),
|
||||
EnemyData::Emu(_) => EMU.get_sprite(0),
|
||||
}
|
||||
}
|
||||
|
||||
fn update(
|
||||
fn update<'a>(
|
||||
&mut self,
|
||||
entity: &mut Entity,
|
||||
controller: &'a ObjectController,
|
||||
entity: &mut Entity<'a>,
|
||||
player: &Player,
|
||||
level: &Level,
|
||||
sfx: &mut sfx::Sfx,
|
||||
) -> UpdateInstruction {
|
||||
match self {
|
||||
EnemyData::Slime(data) => data.update(entity, player, level, sfx),
|
||||
EnemyData::Bat(data) => data.update(entity, player, level, sfx),
|
||||
EnemyData::MiniFlame(data) => data.update(entity, player, level, sfx),
|
||||
EnemyData::Emu(data) => data.update(entity, player, level, sfx),
|
||||
EnemyData::Slime(data) => data.update(controller, entity, player, level, sfx),
|
||||
EnemyData::Bat(data) => data.update(controller, entity, player, level, sfx),
|
||||
EnemyData::MiniFlame(data) => data.update(controller, 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> {
|
||||
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());
|
||||
|
||||
entity
|
||||
.sprite
|
||||
.set_sprite_size(agb::display::object::Size::S16x16);
|
||||
entity.sprite.set_tile_id(enemy_data.tile_id());
|
||||
let sprite = enemy_data.sprite();
|
||||
let sprite = object_controller.get_sprite(sprite).unwrap();
|
||||
|
||||
entity.sprite.set_sprite(sprite);
|
||||
entity.sprite.show();
|
||||
|
||||
entity.sprite.commit();
|
||||
|
@ -1362,8 +1426,15 @@ impl<'a> Enemy<'a> {
|
|||
Self { entity, enemy_data }
|
||||
}
|
||||
|
||||
fn update(&mut self, player: &Player, level: &Level, sfx: &mut sfx::Sfx) -> UpdateInstruction {
|
||||
self.enemy_data.update(&mut self.entity, player, level, sfx)
|
||||
fn update(
|
||||
&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,
|
||||
entity: &mut Entity,
|
||||
controller: &'a ObjectController,
|
||||
entity: &mut Entity<'a>,
|
||||
player: &Player,
|
||||
_level: &Level,
|
||||
) -> UpdateInstruction {
|
||||
|
@ -1406,7 +1478,11 @@ impl ParticleData {
|
|||
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;
|
||||
UpdateInstruction::None
|
||||
|
@ -1416,7 +1492,11 @@ impl ParticleData {
|
|||
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 {
|
||||
entity.velocity.y = Number::new(-1) / 2;
|
||||
|
@ -1438,7 +1518,11 @@ impl ParticleData {
|
|||
UpdateInstruction::None
|
||||
}
|
||||
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 {
|
||||
entity.velocity.y = Number::new(-1) / 2;
|
||||
|
@ -1471,7 +1555,7 @@ struct Particle<'a> {
|
|||
|
||||
impl<'a> Particle<'a> {
|
||||
fn new(
|
||||
object_controller: &'a ObjectControl,
|
||||
object_controller: &'a ObjectController,
|
||||
particle_data: ParticleData,
|
||||
position: Vector2D<Number>,
|
||||
) -> Self {
|
||||
|
@ -1480,11 +1564,6 @@ impl<'a> Particle<'a> {
|
|||
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;
|
||||
|
||||
Self {
|
||||
|
@ -1493,8 +1572,15 @@ impl<'a> Particle<'a> {
|
|||
}
|
||||
}
|
||||
|
||||
fn update(&mut self, player: &Player, level: &Level) -> UpdateInstruction {
|
||||
self.particle_data.update(&mut self.entity, player, level)
|
||||
fn update(
|
||||
&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(
|
||||
&mut self,
|
||||
enemies: &mut Arena<Enemy<'a>>,
|
||||
object_controller: &'a ObjectControl,
|
||||
object_controller: &'a ObjectController,
|
||||
player: &Player,
|
||||
sfx: &mut sfx::Sfx,
|
||||
) -> BossInstruction {
|
||||
match self {
|
||||
BossState::Active(boss) => boss.update(enemies, object_controller, player, sfx),
|
||||
BossState::Following(boss) => {
|
||||
boss.update(player);
|
||||
boss.update(object_controller, player);
|
||||
BossInstruction::None
|
||||
}
|
||||
BossState::NotSpawned => BossInstruction::None,
|
||||
|
@ -1550,15 +1636,13 @@ struct 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(
|
||||
object_controller,
|
||||
Rect::new((0_u16, 0_u16).into(), (0_u16, 0_u16).into()),
|
||||
);
|
||||
entity.position = position;
|
||||
entity
|
||||
.sprite
|
||||
.set_sprite_size(agb::display::object::Size::S16x16);
|
||||
|
||||
Self {
|
||||
entity,
|
||||
following: true,
|
||||
|
@ -1567,11 +1651,11 @@ impl<'a> FollowingBoss<'a> {
|
|||
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;
|
||||
self.timer += 1;
|
||||
|
||||
if self.to_hole {
|
||||
let frame = if self.to_hole {
|
||||
let target: Vector2D<Number> = (17 * 8, -3 * 8).into();
|
||||
let difference = target - self.entity.position;
|
||||
if difference.manhattan_distance() < 1.into() {
|
||||
|
@ -1580,26 +1664,30 @@ impl<'a> FollowingBoss<'a> {
|
|||
self.entity.velocity = difference.normalise() * 2;
|
||||
}
|
||||
|
||||
let frame = (self.timer / 8) % 12;
|
||||
self.entity.sprite.set_tile_id((125 + frame as u16) * 4);
|
||||
self.timer / 8
|
||||
} else if self.timer < 120 {
|
||||
let frame = (self.timer / 20) % 12;
|
||||
self.entity.sprite.set_tile_id((125 + frame as u16) * 4);
|
||||
self.timer / 20
|
||||
} else if self.following {
|
||||
self.entity.velocity = difference / 16;
|
||||
if difference.manhattan_distance() < 20.into() {
|
||||
self.following = false;
|
||||
}
|
||||
let frame = (self.timer / 8) % 12;
|
||||
self.entity.sprite.set_tile_id((125 + frame as u16) * 4);
|
||||
self.timer / 8
|
||||
} else {
|
||||
self.entity.velocity = (0, 0).into();
|
||||
if difference.manhattan_distance() > 60.into() {
|
||||
self.following = true;
|
||||
}
|
||||
let frame = (self.timer / 16) % 12;
|
||||
self.entity.sprite.set_tile_id((125 + frame as u16) * 4);
|
||||
}
|
||||
self.timer / 16
|
||||
};
|
||||
|
||||
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();
|
||||
}
|
||||
|
||||
|
@ -1632,15 +1720,11 @@ enum BossInstruction {
|
|||
}
|
||||
|
||||
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(
|
||||
object_controller,
|
||||
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();
|
||||
Self {
|
||||
entity,
|
||||
|
@ -1655,7 +1739,7 @@ impl<'a> Boss<'a> {
|
|||
fn update(
|
||||
&mut self,
|
||||
enemies: &mut Arena<Enemy<'a>>,
|
||||
object_controller: &'a ObjectControl,
|
||||
object_controller: &'a ObjectController,
|
||||
player: &Player,
|
||||
sfx: &mut sfx::Sfx,
|
||||
) -> BossInstruction {
|
||||
|
@ -1731,8 +1815,14 @@ impl<'a> Boss<'a> {
|
|||
BossActiveState::WaitUntilKilled => 3.into(),
|
||||
};
|
||||
self.timer += 1;
|
||||
let frame = (self.timer / animation_rate) % 12;
|
||||
self.entity.sprite.set_tile_id(784 + (frame as u16) * 16);
|
||||
let frame = self.timer / animation_rate;
|
||||
|
||||
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();
|
||||
instruction
|
||||
|
@ -1753,7 +1843,7 @@ impl<'a> Boss<'a> {
|
|||
self.entity
|
||||
.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) {
|
||||
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;
|
||||
|
@ -1825,7 +1915,7 @@ impl<'a> Game<'a> {
|
|||
|
||||
fn advance_frame(
|
||||
&mut self,
|
||||
object_controller: &'a ObjectControl,
|
||||
object_controller: &'a ObjectController,
|
||||
sfx: &mut sfx::Sfx,
|
||||
) -> GameStatus {
|
||||
let mut state = GameStatus::Continue;
|
||||
|
@ -1921,7 +2011,7 @@ impl<'a> Game<'a> {
|
|||
|
||||
self.input.update();
|
||||
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);
|
||||
|
||||
|
@ -1935,7 +2025,7 @@ impl<'a> Game<'a> {
|
|||
continue;
|
||||
}
|
||||
|
||||
match enemy.update(&self.player, &self.level, sfx) {
|
||||
match enemy.update(object_controller, &self.player, &self.level, sfx) {
|
||||
UpdateInstruction::Remove => {
|
||||
remove.push(idx);
|
||||
}
|
||||
|
@ -1989,7 +2079,7 @@ impl<'a> Game<'a> {
|
|||
let mut remove = Vec::with_capacity(10);
|
||||
|
||||
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::HealBossAndRemove => {
|
||||
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() {
|
||||
for (idx, slime_spawn) in self
|
||||
.level
|
||||
|
@ -2111,7 +2201,7 @@ impl<'a> Game<'a> {
|
|||
}
|
||||
|
||||
fn new(
|
||||
object: &'a ObjectControl,
|
||||
object: &'a ObjectController,
|
||||
level: Level,
|
||||
background_distributor: &'a mut BackgroundDistributor,
|
||||
start_at_boss: bool,
|
||||
|
@ -2146,16 +2236,6 @@ impl<'a> Game<'a> {
|
|||
}
|
||||
|
||||
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();
|
||||
vblank.wait_for_vblank();
|
||||
|
||||
|
@ -2172,8 +2252,7 @@ fn game_with_level(gba: &mut agb::Gba) {
|
|||
let mut background = gba.display.video.tiled0();
|
||||
background.set_background_palettes(background::background.palettes);
|
||||
background.set_background_tilemap(0, background::background.tiles);
|
||||
let mut object = gba.display.object.get();
|
||||
object.enable();
|
||||
let object = gba.display.object.get();
|
||||
|
||||
let mut game = Game::new(
|
||||
&object,
|
||||
|
|
Loading…
Add table
Reference in a new issue