The autowalker now uses the GOAP AI system to walk the map and do its thing. The code needs a big cleanup, so I might just do a full rewrite based on what I know now.
This commit is contained in:
		
							parent
							
								
									fc66d221d4
								
							
						
					
					
						commit
						ff81c78d13
					
				
					 3 changed files with 162 additions and 48 deletions
				
			
		
							
								
								
									
										125
									
								
								autowalker.cpp
									
										
									
									
									
								
							
							
						
						
									
										125
									
								
								autowalker.cpp
									
										
									
									
									
								
							|  | @ -1,20 +1,33 @@ | |||
| #include "autowalker.hpp" | ||||
| #include "inventory.hpp" | ||||
| #include "ai.hpp" | ||||
| 
 | ||||
| template<typename Comp> | ||||
| Pathing compute_paths(gui::FSM& fsm, int& count_out) { | ||||
| int number_left(gui::FSM& fsm) { | ||||
|   int count = 0; | ||||
| 
 | ||||
|   fsm.$level.world->query<components::Position, Comp>( | ||||
|     [&](const auto ent, auto&, auto&) { | ||||
|         if(ent != fsm.$level.player) { | ||||
|           count++; | ||||
|         } | ||||
|     }); | ||||
| 
 | ||||
|   return count; | ||||
| } | ||||
| 
 | ||||
| template<typename Comp> | ||||
| Pathing compute_paths(gui::FSM& fsm) { | ||||
|   auto& walls_original = fsm.$level.map->$walls; | ||||
|   auto walls_copy = walls_original; | ||||
| 
 | ||||
|   Pathing paths{matrix::width(walls_copy), matrix::height(walls_copy)}; | ||||
|   count_out = 0; | ||||
| 
 | ||||
|   fsm.$level.world->query<components::Position>( | ||||
|   [&](const auto ent, auto& position) { | ||||
|       if(ent != fsm.$level.player) { | ||||
|         if(fsm.$level.world->has<Comp>(ent)) { | ||||
|           paths.set_target(position.location); | ||||
|           count_out = count_out + 1; | ||||
|         } else { | ||||
|           // this will mark that spot as a wall so we don't path there temporarily
 | ||||
|           walls_copy[position.location.y][position.location.x] = WALL_PATH_LIMIT; | ||||
|  | @ -28,17 +41,18 @@ Pathing compute_paths(gui::FSM& fsm, int& count_out) { | |||
| } | ||||
| 
 | ||||
| Pathing Autowalker::path_to_enemies() { | ||||
|   return compute_paths<components::Combat>(fsm, enemy_count); | ||||
|   return compute_paths<components::Combat>(fsm); | ||||
| } | ||||
| 
 | ||||
| Pathing Autowalker::path_to_items() { | ||||
|   return compute_paths<components::InventoryItem>(fsm, item_count); | ||||
|   return compute_paths<components::InventoryItem>(fsm); | ||||
| } | ||||
| 
 | ||||
| Pathing Autowalker::path_to_devices() { | ||||
|   return compute_paths<components::Device>(fsm, device_count); | ||||
|   return compute_paths<components::Device>(fsm); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void Autowalker::window_events() { | ||||
|   fsm.$window.handleEvents( | ||||
|       [&](const sf::Event::KeyPressed &) { | ||||
|  | @ -57,9 +71,11 @@ void Autowalker::process_combat() { | |||
|       || fsm.in_state(gui::State::ATTACKING)) | ||||
|   { | ||||
|     if(fsm.in_state(gui::State::ATTACKING)) { | ||||
|       fmt::println("In attacking state, sending a TICK"); | ||||
|       send_event(gui::Event::TICK); | ||||
|     } else { | ||||
|       send_event(gui::Event::ATTACK); | ||||
|       fmt::println("Not in ATTACK, sending an ATTACK to continue combat."); | ||||
|       send_event(gui::Event::ATTACK);; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | @ -73,12 +89,15 @@ bool Autowalker::path_player(Pathing& paths, Point& target_out) { | |||
|   bool found = paths.random_walk(target_out, false, PATHING_TOWARD); | ||||
| 
 | ||||
|   if(!found) { | ||||
|     dbc::log("no neighbor found, aborting autowalk"); | ||||
|     dbc::log("no neighbor found in any direction, aborting autowalk"); | ||||
|     matrix::dump("NO TOWARD", paths.$paths, target_out.x, target_out.y); | ||||
|     return false; | ||||
|   } | ||||
| 
 | ||||
|   if(!fsm.$level.map->can_move(target_out)) { | ||||
|     dbc::log("neighbors is telling me to go to a bad spot."); | ||||
|     matrix::dump("BAD TARGET PATHS", paths.$paths, target_out.x, target_out.y); | ||||
|     matrix::dump("BAD TARGET MAP", fsm.$level.map->walls(), target_out.x, target_out.y); | ||||
|     return false; | ||||
|   } | ||||
| 
 | ||||
|  | @ -156,34 +175,64 @@ void Autowalker::autowalk() { | |||
|   window_events(); | ||||
|   if(!fsm.autowalking) return; | ||||
| 
 | ||||
|   process_combat(); | ||||
|   auto paths = path_to_enemies(); | ||||
|   int move_attempts = 0; | ||||
| 
 | ||||
|   if(enemy_count == 0) { | ||||
|     dbc::log("Killed everything, now finding items."); | ||||
|     paths = path_to_items(); | ||||
|   } | ||||
|   auto start = ai::load_state("Walker::initial_state"); | ||||
|   auto goal = ai::load_state("Walker::final_state"); | ||||
| 
 | ||||
|   if(enemy_count == 0 && item_count == 0) { | ||||
|     dbc::log("No more items, find the exit."); | ||||
|     paths = path_to_devices(); | ||||
|   } | ||||
|   do { | ||||
|     int enemy_count = number_left<components::Combat>(fsm); | ||||
|     int item_count = number_left<components::InventoryItem>(fsm); | ||||
| 
 | ||||
|   if(enemy_count == 0 && | ||||
|       item_count == 0 && | ||||
|       device_count == 0) | ||||
|   { | ||||
|     fsm.autowalking = false; | ||||
|     dbc::log("no more enemies, items, or devices."); | ||||
|     return; | ||||
|   } | ||||
|     fmt::println("ENEMY COUNT: {}, ITEM COUNT: {}", enemy_count, item_count); | ||||
| 
 | ||||
|     window_events(); | ||||
|     ai::set(start, "no_more_enemies", enemy_count == 0); | ||||
|     ai::set(start, "no_more_items", item_count == 0); | ||||
|     ai::set(start, "enemy_found", | ||||
|         fsm.in_state(gui::State::IN_COMBAT) || | ||||
|         fsm.in_state(gui::State::ATTACKING)); | ||||
| 
 | ||||
|     auto a_plan = ai::plan("Walker::actions", start, goal); | ||||
| 
 | ||||
|     // need a test for plan complete and only action is END
 | ||||
|     for(auto action : a_plan.script) { | ||||
|       if(action.$name == "find_enemy") { | ||||
|         // this is where to test if enemy found and update state
 | ||||
|         fmt::println("FINDING AN ENEMY"); | ||||
|         auto paths = path_to_enemies(); | ||||
|         auto pos = get_current_position(); | ||||
|         matrix::dump("FINDING", paths.$paths, pos.x, pos.y); | ||||
|         process_move(paths); | ||||
|       } else if(action.$name == "kill_enemy") { | ||||
|         fmt::println("KILLING ENEMY"); | ||||
|         process_combat(); | ||||
|       } else if(action.$name == "find_healing") { | ||||
|         fmt::println("FINDING HEALING"); | ||||
|         auto paths = path_to_items(); | ||||
|         process_move(paths); | ||||
|         // do the path to healing thing
 | ||||
|       } else if(action.$name == "collect_items") { | ||||
|         fmt::println("COLLECTING ITEMS"); | ||||
|         auto paths = path_to_items(); | ||||
|         process_move(paths); | ||||
|         // path to the items and get them all
 | ||||
|       } else if(action.$name == "END") { | ||||
|         fmt::println("END STATE, complete? {}", a_plan.complete); | ||||
|         fsm.autowalking = false; | ||||
|       } else { | ||||
|         fmt::println("Unknown action: {}", action.$name); | ||||
|       } | ||||
| 
 | ||||
|       move_attempts++; | ||||
|     } | ||||
|   } while(move_attempts < 100 && fsm.autowalking); | ||||
| } | ||||
| 
 | ||||
| void Autowalker::process_move(Pathing& paths) { | ||||
|   Point current = get_current_position(); | ||||
|   Point target = current; | ||||
| 
 | ||||
|   show_map_overlay(paths.$paths, current); | ||||
| 
 | ||||
| 
 | ||||
|   if(!path_player(paths, target)) { | ||||
|     dbc::log("no paths found, aborting autowalk"); | ||||
|     fsm.autowalking = false; | ||||
|  | @ -191,28 +240,12 @@ void Autowalker::autowalk() { | |||
|   } | ||||
| 
 | ||||
|   rotate_player(current, target); | ||||
|   while(fsm.in_state(gui::State::ROTATING)) send_event(gui::Event::TICK); | ||||
| 
 | ||||
|   int move_attempts = 0; | ||||
|   do { | ||||
|     process_combat(); | ||||
|     process_move(); | ||||
|     // BUG: sometimes in idle but there's an enemy near but combat hasn't started
 | ||||
|     // for now just toss out an ATTACK and it'll be ignored or cause combat
 | ||||
|     send_event(gui::Event::ATTACK); | ||||
|     move_attempts++; | ||||
|   } while(move_attempts < 100 && !player_has_moved(target)); | ||||
| } | ||||
| 
 | ||||
| void Autowalker::process_move() { | ||||
|   send_event(gui::Event::MOVE_FORWARD); | ||||
|   while(fsm.in_state(gui::State::MOVING)) send_event(gui::Event::TICK); | ||||
| } | ||||
| 
 | ||||
| bool Autowalker::player_has_moved(Point target) { | ||||
|   Point current = get_current_position(); | ||||
|   return current.x == target.x && current.y == target.y; | ||||
| } | ||||
| 
 | ||||
| void Autowalker::send_event(gui::Event ev) { | ||||
|   fsm.event(ev); | ||||
|   fsm.render(); | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Zed A. Shaw
						Zed A. Shaw