#include "factory.h" #include "masks.h" #include "components/animator.h" #include "components/collider.h" #include "components/mover.h" #include "components/player.h" #include "components/hurtable.h" #include "components/timer.h" #include "components/enemy.h" #include "components/ghost_frog.h" #include "components/orb.h" using namespace TL; Entity* Factory::player(World* world, Point position) { auto en = world->add_entity(position); auto anim = en->add(Animator("player")); anim->play("idle"); anim->depth = -10; auto hitbox = en->add(Collider::make_rect(Recti(-4, -12, 8, 12))); hitbox->mask = Mask::player; auto mover = en->add(Mover()); mover->collider = hitbox; en->add(Player()); return en; } Entity* Factory::bramble(World* world, Point position) { auto en = world->add_entity(position); auto anim = en->add(Animator("bramble")); anim->play("idle"); anim->depth = -5; auto hitbox = en->add(Collider::make_rect(Recti(-4, -8, 8, 8))); hitbox->mask = Mask::enemy; auto hurtable = en->add(Hurtable()); hurtable->hurt_by = Mask::player_attack; hurtable->collider = hitbox; hurtable->on_hurt = [](Hurtable* self) { pop(self->world(), self->entity()->position + Point(0, -4)); self->entity()->destroy(); }; return en; } Entity* Factory::pop(World* world, Point position) { auto en = world->add_entity(position); auto anim = en->add(Animator("pop")); anim->play("pop"); anim->depth = -20; auto timer = en->add(Timer(anim->animation()->duration(), [](Timer* self) { self->entity()->destroy(); })); return en; } Entity* Factory::spitter(World* world, Point position) { auto en = world->add_entity(position); en->add(Enemy()); auto anim = en->add(Animator("spitter")); anim->play("idle"); anim->depth = -5; auto hitbox = en->add(Collider::make_rect(Recti(-6, -12, 12, 12))); hitbox->mask = Mask::enemy; auto hurtable = en->add(Hurtable()); hurtable->hurt_by = Mask::player_attack; hurtable->collider = hitbox; hurtable->on_hurt = [](Hurtable* self) mutable { Time::pause_for(0.1f); pop(self->world(), self->entity()->position + Point(0, -4)); self->entity()->destroy(); }; auto timer = en->add(Timer(1.0f, [](Timer* self) { bullet(self->world(), self->entity()->position + Point(-8, -8), -1); self->get()->play("shoot"); self->entity()->add(Timer(0.4f, [](Timer* self) { self->get()->play("idle"); })); self->start(3.0f); })); return en; } Entity* Factory::bullet(World* world, Point position, int direction) { auto en = world->add_entity(position); auto anim = en->add(Animator("bullet")); anim->play("idle"); anim->depth = -5; auto hitbox = en->add(Collider::make_rect(Recti(-4, -4, 8, 8))); hitbox->mask = Mask::enemy; auto mover = en->add(Mover()); mover->collider = hitbox; mover->speed = Vec2f(direction * 40, 0); mover->gravity = 130; mover->on_hit_x = [](Mover* self) { self->entity()->destroy(); }; mover->on_hit_y = [](Mover* self) { self->speed.y = -60; }; auto hurtable = en->add(Hurtable()); hurtable->hurt_by = Mask::player_attack; hurtable->collider = hitbox; hurtable->on_hurt = [](Hurtable* self) mutable { Time::pause_for(0.1f); pop(self->world(), self->entity()->position + Point(0, -4)); self->entity()->destroy(); }; en->add(Timer(2.5f, [](Timer* self) { self->get()->flicker_timer = 100; })); en->add(Timer(3.0f, [](Timer* self) { self->entity()->destroy(); })); return en; } namespace { class MosquitoBehavior : public Component { public: int health = 2; float timer = 0; void update() override { auto mover = get(); auto player = world()->first(); if (player) { auto diff = player->entity()->position.x - entity()->position.x; auto dist = Calc::abs(diff); if (dist < 100) mover->speed.x += Calc::sign(diff) * 100 * Time::delta; else mover->speed.x = Calc::approach(mover->speed.x, 0, 100 * Time::delta); if (Calc::abs(mover->speed.x) > 50) mover->speed.x = Calc::approach(mover->speed.x, Calc::sign(mover->speed.x) * 50, 800 * Time::delta); mover->speed.y = Calc::sin(timer) * 10; } timer += Time::delta * 4; } void hurt() { health--; if (health <= 0) { Factory::pop(world(), entity()->position); entity()->destroy(); } else { auto mover = get(); auto player = world()->first(); auto sign = Calc::sign(player->entity()->position.x - entity()->position.x); mover->speed.x = -sign * 140; } } }; } Entity* Factory::mosquito(World* world, Point position) { auto en = world->add_entity(position); auto mosquito = en->add(MosquitoBehavior()); en->add(Enemy()); auto anim = en->add(Animator("mosquito")); anim->play("fly"); anim->depth = -5; auto hitbox = en->add(Collider::make_rect(Recti(-4, -4, 8, 8))); hitbox->mask = Mask::enemy; auto mover = en->add(Mover()); auto hurtable = en->add(Hurtable()); hurtable->hurt_by = Mask::player_attack; hurtable->collider = hitbox; hurtable->on_hurt = [](Hurtable* self) { self->get()->hurt(); }; return en; } namespace { void make_door_contents(Entity* en) { auto anim = en->add(Animator("door")); anim->play("idle"); anim->depth = -1; auto hitbox = en->add(Collider::make_rect(Recti(-6, -16, 12, 16))); hitbox->mask = Mask::solid; } } Entity* Factory::door(World* world, Point position, bool wait_for_player) { auto en = world->add_entity(position); if (!wait_for_player) make_door_contents(en); // check if all enemies are dead en->add(Timer(0.25f, [waiting = wait_for_player](Timer* self) mutable { self->start(0.25f); if (waiting) { auto player = self->world()->first(); if (player->entity()->position.x > self->entity()->position.x + 12) { make_door_contents(self->entity()); Factory::pop(self->world(), self->entity()->position + Point(0, -8)); waiting = false; } else return; } if (!self->world()->first()) { Factory::pop(self->world(), self->entity()->position + Point(0, -8)); self->entity()->destroy(); } })); return en; } Entity* Factory::blob(World* world, Point position) { auto en = world->add_entity(position); en->add(Enemy()); auto anim = en->add(Animator("blob")); anim->play("idle"); anim->depth = -5; auto hitbox = en->add(Collider::make_rect(Recti(-4, -8, 8, 8))); hitbox->mask = Mask::enemy; auto mover = en->add(Mover()); mover->collider = hitbox; mover->gravity = 300; mover->friction = 400; mover->on_hit_y = [](Mover* self) { self->get()->play("idle"); self->stop_y(); }; en->add(Timer(2.0f, [](Timer* self) { auto mover = self->get(); if (!mover->on_ground()) { self->start(0.05f); } else { self->get()->play("jump"); self->start(2.0f); mover->speed.y = -90; auto player = self->world()->first(); if (player) { auto dir = Calc::sign(player->entity()->position.x - self->entity()->position.x); if (dir == 0) dir = 1; self->get()->scale = Vec2f(dir, 1); mover->speed.x = dir * 40; } } })); auto hurtable = en->add(Hurtable()); hurtable->hurt_by = Mask::player_attack; hurtable->collider = hitbox; hurtable->on_hurt = [health = 3](Hurtable* self) mutable { auto player = self->world()->first(); if (player) { auto mover = self->get(); auto sign = Calc::sign(self->entity()->position.x - player->entity()->position.x); mover->speed.x = sign * 120; } health--; if (health <= 0) { pop(self->world(), self->entity()->position + Point(0, -4)); self->entity()->destroy(); } }; return en; } Entity* Factory::ghost_frog(World* world, Point position) { auto en = world->add_entity(position); en->add(GhostFrog()); en->add(Enemy()); auto anim = en->add(Animator("ghostfrog")); anim->play("sword"); anim->depth = -5; auto hitbox = en->add(Collider::make_rect(Recti(-4, -12, 8, 12))); hitbox->mask = Mask::enemy; auto mover = en->add(Mover()); mover->collider = hitbox; mover->gravity = 0; mover->friction = 100; mover->on_hit_x = [](Mover* self) { self->get()->on_hit_x(self); }; mover->on_hit_y = [](Mover* self) { self->get()->on_hit_y(self); }; auto hurtable = en->add(Hurtable()); hurtable->hurt_by = Mask::player_attack; hurtable->collider = hitbox; hurtable->on_hurt = [](Hurtable* self) { self->get()->on_hurt(self); }; return en; } Entity* Factory::orb(World* world, Point position) { auto en = world->add_entity(position); en->add(Orb()); auto anim = en->add(Animator("bullet")); anim->play("idle"); anim->depth = -5; auto hitbox = en->add(Collider::make_rect(Recti(-4, -4, 8, 8))); hitbox->mask = Mask::enemy; auto mover = en->add(Mover()); mover->collider = hitbox; mover->on_hit_x = [](Mover* self) { Factory::pop(self->world(), self->entity()->position); self->entity()->destroy(); }; mover->on_hit_y = [](Mover* self) { Factory::pop(self->world(), self->entity()->position); self->entity()->destroy(); }; auto hurtable = en->add(Hurtable()); hurtable->hurt_by = Mask::player_attack; hurtable->collider = en->add(Collider::make_rect(Recti(-8, -8, 16, 16))); hurtable->on_hurt = [](Hurtable* self) { self->get()->on_hit(); }; return en; }