Created a combat system to start with and also added a 'HEARING' mechanic where enemies can hear you from a certain distance before moving to you.
This commit is contained in:
		
							parent
							
								
									753bc70b77
								
							
						
					
					
						commit
						9102bdc8ad
					
				
					 9 changed files with 70 additions and 24 deletions
				
			
		
							
								
								
									
										14
									
								
								combat.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								combat.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,14 @@ | ||||||
|  | #include "combat.hpp" | ||||||
|  | #include "rand.hpp" | ||||||
|  | 
 | ||||||
|  | int Combat::fight(Combat &target) { | ||||||
|  |   int attack = Random::uniform<int>(0,1); | ||||||
|  |   int my_dmg = 0; | ||||||
|  | 
 | ||||||
|  |   if(attack) { | ||||||
|  |     my_dmg = Random::uniform<int>(1, damage); | ||||||
|  |     target.hp -= my_dmg; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return my_dmg; | ||||||
|  | } | ||||||
							
								
								
									
										11
									
								
								combat.hpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								combat.hpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,11 @@ | ||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include "components.hpp" | ||||||
|  | 
 | ||||||
|  | struct Combat { | ||||||
|  |   int hp; | ||||||
|  |   int damage; | ||||||
|  |   bool dead; | ||||||
|  | 
 | ||||||
|  |   int fight(Combat &target); | ||||||
|  | }; | ||||||
|  | @ -1,6 +1,7 @@ | ||||||
| #pragma once | #pragma once | ||||||
| #include "dinkyecs.hpp" | #include "dinkyecs.hpp" | ||||||
| #include "map.hpp" | #include "map.hpp" | ||||||
|  | #include "combat.hpp" | ||||||
| #include <deque> | #include <deque> | ||||||
| 
 | 
 | ||||||
| namespace Components { | namespace Components { | ||||||
|  | @ -17,11 +18,6 @@ namespace Components { | ||||||
|     int dy; |     int dy; | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|   struct Combat { |  | ||||||
|     int hp; |  | ||||||
|     int damage; |  | ||||||
|   }; |  | ||||||
| 
 |  | ||||||
|   struct Treasure { |   struct Treasure { | ||||||
|     int amount; |     int amount; | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
							
								
								
									
										1
									
								
								gui.cpp
									
										
									
									
									
								
							
							
						
						
									
										1
									
								
								gui.cpp
									
										
									
									
									
								
							|  | @ -164,6 +164,7 @@ void GUI::run_systems() { | ||||||
|   System::enemy_pathing($world, $game_map, player); |   System::enemy_pathing($world, $game_map, player); | ||||||
|   System::motion($world, $game_map); |   System::motion($world, $game_map); | ||||||
|   System::combat($world, player); |   System::combat($world, player); | ||||||
|  |   System::death($world); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void GUI::resize_map(int new_size) { | void GUI::resize_map(int new_size) { | ||||||
|  |  | ||||||
							
								
								
									
										1
									
								
								map.hpp
									
										
									
									
									
								
							
							
						
						
									
										1
									
								
								map.hpp
									
										
									
									
									
								
							|  | @ -57,6 +57,7 @@ public: | ||||||
|   int limit() { return $limit; } |   int limit() { return $limit; } | ||||||
|   size_t width() { return $walls[0].size(); } |   size_t width() { return $walls[0].size(); } | ||||||
|   size_t height() { return $walls.size(); } |   size_t height() { return $walls.size(); } | ||||||
|  |   int distance(Point to) { return $paths[to.y][to.x]; } | ||||||
|   Room &room(size_t at) { |   Room &room(size_t at) { | ||||||
|     return $rooms[at]; |     return $rooms[at]; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  | @ -32,6 +32,7 @@ roguish = executable('roguish', [ | ||||||
|   'gui.cpp', |   'gui.cpp', | ||||||
|   'rand.cpp', |   'rand.cpp', | ||||||
|   'collider.cpp', |   'collider.cpp', | ||||||
|  |   'combat.cpp', | ||||||
|   'systems.cpp', |   'systems.cpp', | ||||||
|   ], |   ], | ||||||
|   dependencies: dependencies) |   dependencies: dependencies) | ||||||
|  |  | ||||||
|  | @ -10,7 +10,9 @@ TODO: | ||||||
| * Write a test that generates a ton of maps then confirms there's a path from one room to every other room? | * Write a test that generates a ton of maps then confirms there's a path from one room to every other room? | ||||||
| * Lua integration? | * Lua integration? | ||||||
| 
 | 
 | ||||||
| * Combat system and simple loot system. | * BUG: If map is < 90 zome out crashes. | ||||||
|  | 
 | ||||||
|  | * Simple loot system. | ||||||
| * Actually render FTXUI ansi output instead of the gui.cpp hack. | * Actually render FTXUI ansi output instead of the gui.cpp hack. | ||||||
| * Remove entity from world, _or_ mark them dead, switch their icon, and make them an entity the player walks over? | * Remove entity from world, _or_ mark them dead, switch their icon, and make them an entity the player walks over? | ||||||
| * Bring back sounds, check out SoLoud. | * Bring back sounds, check out SoLoud. | ||||||
|  |  | ||||||
							
								
								
									
										55
									
								
								systems.cpp
									
										
									
									
									
								
							
							
						
						
									
										55
									
								
								systems.cpp
									
										
									
									
									
								
							|  | @ -10,6 +10,8 @@ using namespace fmt; | ||||||
| 
 | 
 | ||||||
| using namespace Components; | using namespace Components; | ||||||
| 
 | 
 | ||||||
|  | #define HEARING_DISTANCE 8 | ||||||
|  | 
 | ||||||
| void System::enemy_pathing(DinkyECS::World &world, Map &game_map, Player &player) { | void System::enemy_pathing(DinkyECS::World &world, Map &game_map, Player &player) { | ||||||
|   const auto &player_position = world.component<Position>(player.entity); |   const auto &player_position = world.component<Position>(player.entity); | ||||||
|   game_map.set_target(player_position.location); |   game_map.set_target(player_position.location); | ||||||
|  | @ -17,9 +19,11 @@ void System::enemy_pathing(DinkyECS::World &world, Map &game_map, Player &player | ||||||
| 
 | 
 | ||||||
|   world.system<Position, Motion>([&](const auto &ent, auto &position, auto &motion) { |   world.system<Position, Motion>([&](const auto &ent, auto &position, auto &motion) { | ||||||
|     if(ent != player.entity) { |     if(ent != player.entity) { | ||||||
|       Point out = position.location; |       Point out = position.location; // copy
 | ||||||
|       game_map.neighbors(out, false); |       if(game_map.distance(out) < HEARING_DISTANCE) { | ||||||
|       motion = { int(out.x - position.location.x), int(out.y - position.location.y)}; |         game_map.neighbors(out, false); | ||||||
|  |         motion = { int(out.x - position.location.x), int(out.y - position.location.y)}; | ||||||
|  |       } | ||||||
|     } |     } | ||||||
|   }); |   }); | ||||||
|   game_map.clear_target(player_position.location); |   game_map.clear_target(player_position.location); | ||||||
|  | @ -64,6 +68,22 @@ void System::motion(DinkyECS::World &world, Map &game_map) { | ||||||
|   }); |   }); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void System::death(DinkyECS::World &world) { | ||||||
|  |   auto &collider = world.get<spatial_map>(); | ||||||
|  | 
 | ||||||
|  |   world.system<Position, Combat>([&](const auto &ent, auto &position, auto &combat) { | ||||||
|  |     // bring out yer dead
 | ||||||
|  |     if(combat.hp <= 0 && !combat.dead) { | ||||||
|  |       combat.dead = true; | ||||||
|  |       // take them out of collision map
 | ||||||
|  |       collider.remove(position.location); | ||||||
|  | 
 | ||||||
|  |       // remove their motion so they're dead
 | ||||||
|  |       world.remove<Motion>(ent); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| void System::combat(DinkyECS::World &world, Player &player) { | void System::combat(DinkyECS::World &world, Player &player) { | ||||||
|   auto& collider = world.get<spatial_map>(); |   auto& collider = world.get<spatial_map>(); | ||||||
|  | @ -77,25 +97,24 @@ void System::combat(DinkyECS::World &world, Player &player) { | ||||||
|   if(found) { |   if(found) { | ||||||
|     for(auto entity : nearby) { |     for(auto entity : nearby) { | ||||||
|       auto& enemy_combat = world.component<Combat>(entity); |       auto& enemy_combat = world.component<Combat>(entity); | ||||||
|       int player_dmg = Random::uniform<int>(1, enemy_combat.damage); |       int player_dmg = player_combat.fight(enemy_combat); | ||||||
|       enemy_combat.hp -= player_dmg; |  | ||||||
|       log.log(format("YOU HIT {} damage! Enemy has {} HP left.", |  | ||||||
|             player_dmg, enemy_combat.hp)); |  | ||||||
| 
 | 
 | ||||||
|       if(enemy_combat.hp <= 0) { |       if(player_dmg > 0) { | ||||||
|         log.log("--- ENEMY DEAD!---"); |         log.log(format("You HIT for {} damage, HP left {}.", player_dmg, enemy_combat.hp)); | ||||||
|         auto enemy_position = world.component<Position>(entity); |  | ||||||
|         collider.remove(enemy_position.location); |  | ||||||
|         world.remove<Motion>(entity); |  | ||||||
|       } else { |       } else { | ||||||
|         int attack = Random::uniform<int>(0,1); |         log.log("You missed! They're quick!"); | ||||||
|         if(attack) { |       } | ||||||
|           int dmg = Random::uniform<int>(1, enemy_combat.damage); | 
 | ||||||
|           player_combat.hp -= dmg; |       if(enemy_combat.hp > 0) { | ||||||
|           log.log(format("HIT! You took {} damage.", dmg)); |         int enemy_dmg = enemy_combat.fight(player_combat); | ||||||
|  | 
 | ||||||
|  |         if(enemy_dmg > 0) { | ||||||
|  |           log.log(format("Enemy HIT YOU for {} damage.", enemy_dmg)); | ||||||
|         } else { |         } else { | ||||||
|           log.log("You dodged! Run!"); |           log.log("Enemy MISSED, you dodged it."); | ||||||
|         } |         } | ||||||
|  |       } else { | ||||||
|  |         log.log("ENEMY DEAD! YOU WIN!"); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  | @ -9,6 +9,7 @@ using namespace Components; | ||||||
| namespace System { | namespace System { | ||||||
|   void motion(DinkyECS::World &world, Map &game_map); |   void motion(DinkyECS::World &world, Map &game_map); | ||||||
|   void combat(DinkyECS::World &world, Player &player); |   void combat(DinkyECS::World &world, Player &player); | ||||||
|  |   void death(DinkyECS::World &world); | ||||||
|   void enemy_pathing(DinkyECS::World &world, Map &game_map, Player &player); |   void enemy_pathing(DinkyECS::World &world, Map &game_map, Player &player); | ||||||
|   void draw_map(DinkyECS::World &world, Map &game_map, ftxui::Canvas &canvas, size_t view_x, size_t view_y); |   void draw_map(DinkyECS::World &world, Map &game_map, ftxui::Canvas &canvas, size_t view_x, size_t view_y); | ||||||
|   void draw_entities(DinkyECS::World &world, Map &game_map, ftxui::Canvas &canvas, const Point &cam_orig, size_t view_x, size_t view_y); |   void draw_entities(DinkyECS::World &world, Map &game_map, ftxui::Canvas &canvas, const Point &cam_orig, size_t view_x, size_t view_y); | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Zed A. Shaw
						Zed A. Shaw