diff options
author | Griffin Smith <root@gws.fyi> | 2019-07-14T20·20-0400 |
---|---|---|
committer | Griffin Smith <root@gws.fyi> | 2019-07-14T20·20-0400 |
commit | 575a051e6efcd8fd3b0a146f49040e543ae8e5b0 (patch) | |
tree | 3507592582cdffdd73ba7ca9ae893117682581b7 /src/game.rs | |
parent | e7ad87c7301f266dece36e7558c0f212e370aac6 (diff) |
Implement extremely basic combat
There's a gormlak, you can kill it. That's it.
Diffstat (limited to 'src/game.rs')
-rw-r--r-- | src/game.rs | 72 |
1 files changed, 59 insertions, 13 deletions
diff --git a/src/game.rs b/src/game.rs index f86d32d0463c..57c04cfb883f 100644 --- a/src/game.rs +++ b/src/game.rs @@ -1,15 +1,12 @@ use crate::display::{self, Viewport}; -use crate::entities::Character; -use crate::entities::{Creature, Entity}; +use crate::entities::{Character, Creature, Entity, EntityID, Identified}; use crate::messages::message; use crate::settings::Settings; use crate::types::command::Command; -use crate::types::entity_map::EntityID; use crate::types::entity_map::EntityMap; -use crate::types::pos; -use crate::types::Ticks; use crate::types::{ - BoundingBox, Collision, Dimensions, Position, Positioned, PositionedMut, + pos, BoundingBox, Collision, Dimensions, Position, Positioned, + PositionedMut, Ticks, }; use rand::rngs::SmallRng; use rand::SeedableRng; @@ -100,21 +97,29 @@ impl<'a> Game<'a> { } } + /// Returns a list of all creature entities at the given position + fn creatures_at<'b>(&'b self, pos: Position) -> Vec<&'b Creature> { + self.entities + .at(pos) + .iter() + .filter_map(|e| e.downcast_ref()) + .collect() + } + /// Returns a collision, if any, at the given Position in the game fn collision_at(&self, pos: Position) -> Option<Collision> { if !pos.within(self.viewport.inner) { Some(Collision::Stop) } else { - None + if self.creatures_at(pos).len() > 0 { + Some(Collision::Combat) + } else { + None + } } } fn character(&self) -> &Character { - debug!( - "ents: {:?} cid: {:?}", - self.entities.ids().map(|id| *id).collect::<Vec<u32>>(), - self.character_entity_id - ); (*self.entities.get(self.character_entity_id).unwrap()) .downcast_ref() .unwrap() @@ -128,6 +133,14 @@ impl<'a> Game<'a> { Ok(()) } + /// Remove the given entity from the game, drawing over it if it's visible + fn remove_entity(&mut self, entity_id: EntityID) -> io::Result<()> { + if let Some(entity) = self.entities.remove(entity_id) { + self.viewport.clear(entity.position())?; + } + Ok(()) + } + /// Step the game forward the given number of ticks fn tick(&mut self, ticks: Ticks) {} @@ -153,6 +166,37 @@ impl<'a> Game<'a> { self.viewport.write_message(message) } + fn attack(&mut self, creature_id: EntityID) -> io::Result<()> { + info!("Attacking creature {:?}", creature_id); + self.say("combat.attack")?; + let damage = self.character().damage(); + let creature = self + .entities + .get_mut(creature_id) + .and_then(|e| e.downcast_mut::<Creature>()) + .expect( + format!("Creature ID went away: {:?}", creature_id).as_str(), + ); + creature.damage(damage); + if creature.dead() { + self.say("combat.killed")?; + info!("Killed creature {:?}", creature_id); + self.remove_entity(creature_id)?; + } + Ok(()) + } + + fn attack_at(&mut self, pos: Position) -> io::Result<()> { + let creatures = self.creatures_at(pos); + if creatures.len() == 1 { + let creature = creatures.get(0).unwrap(); + self.attack(creature.id()) + } else { + // TODO prompt with a menu of creatures to combat + unimplemented!() + } + } + /// Run the game pub fn run(mut self) -> io::Result<()> { info!("Running game"); @@ -180,7 +224,9 @@ impl<'a> Game<'a> { new_pos, ); } - Some(Combat) => unimplemented!(), + Some(Combat) => { + self.attack_at(new_pos)?; + } Some(Stop) => (), } } |