diff --git a/code/modules/mob/living/carbon/human/knpc.dm b/code/modules/mob/living/carbon/human/knpc.dm new file mode 100644 index 000000000000..c7bd37b0c98f --- /dev/null +++ b/code/modules/mob/living/carbon/human/knpc.dm @@ -0,0 +1,1865 @@ +#define AI_TRAIT_BRAWLER 1 //The "marines". These guys will attempt to stand and fight +#define AI_TRAIT_SUPPORT 2 //The "medics", civvies, etc. These lads will call for backup + +//KNPC flags +#define KNPC_IS_MARTIAL_ARTIST 1<<0 //Does the KNPC combo you right into the next round? +#define KNPC_IS_DODGER 1<<1 //Does the KNPC become like the wind in CQC? +#define KNPC_IS_MERCIFUL 1<<2 //Does the KNPC consider mobs in crit to be valid? +#define KNPC_IS_AREA_SPECIFIC 1<<3 //Does the KNPC scream out the area when calling for backup? +#define KNPC_IS_DOOR_BASHER 1<<4 //Does the KNPC kick the door off its hinges if it doesn't have a valid ID? +#define KNPC_IS_DOOR_HACKER 1<<5 //Does the KNPC hack the door open if it doesn't have a valid ID? +#define KNPC_STEAL_ID 1<<5 //Do we steal IDs? used to prevent a bug + +//Knpc pathfinding return values +#define KNPC_PATHFIND_SUCCESS 0 //Path successfully generated +#define KNPC_PATHFIND_SKIP 1 //Regenerating of path isn't needed, incapable of moving, etc +#define KNPC_PATHFIND_TIMEOUT 2 //Pathfinding is currently in timeout due to having failed previously. +#define KNPC_PATHFIND_FAIL 3 //No path found + +//Knpc pathfinding timeout defines +#define KNPC_TIMEOUT_BASE (3 SECONDS) //The base timeout applied if an knpc's pathfinding fails. +#define KNPC_TIMEOUT_STACK_CAP 9 //Every consecutive pathfinding fail adds a stacks; Timeout applied uses them as multiplier up to a cap of this value. + +//AI behaviour +#define AI_AGGRESSIVE 1 +#define AI_PASSIVE 2 +#define AI_RETALIATE 3 +#define AI_GUARD 4 + +DEFINE_BITFIELD(knpc_traits, list( + "KNPC_IS_MARTIAL_ARTIST" = KNPC_IS_MARTIAL_ARTIST, + "KNPC_IS_DODGER" = KNPC_IS_DODGER, + "KNPC_IS_MERCIFUL" = KNPC_IS_MERCIFUL, + "KNPC_IS_AREA_SPECIFIC" = KNPC_IS_AREA_SPECIFIC, + "KNPC_IS_DOOR_BASHER" = KNPC_IS_DOOR_BASHER, + "KNPC_IS_DOOR_HACKER" = KNPC_IS_DOOR_HACKER, + "KNPC_STEAL_ID" = KNPC_STEAL_ID, +)) + +//AI score defines +#define AI_SCORE_MAXIMUM 1000 //No goal combination should ever exceed this. +#define AI_SCORE_SUPERCRITICAL 500 +#define AI_SCORE_CRITICAL 100 +#define AI_SCORE_SUPERPRIORITY 75 +#define AI_SCORE_HIGH_PRIORITY 60 +#define AI_SCORE_PRIORITY 50 +#define AI_SCORE_DEFAULT 25 +#define AI_SCORE_LOW_PRIORITY 15 +#define AI_SCORE_VERY_LOW_PRIORITY 5 //Very low priority, acts as a failsafe to ensure that the AI always picks _something_ to do. + +#define KNPC_DIFFICULTY_EASY 1 //The "marines". These guys will attempt to stand and fight +#define KNPC_DIFFICULTY_HARD 2 //The "marines". These guys will attempt to stand and fight + +///This proc takes a dir and turns it into a list of its cardinal dirs. Useful for, say, handling things on multiple cardinals, and only cardinals, in case of diagonal positioning. +/proc/dir_to_cardinal_dirs(input_dir) + if(input_dir in list(NORTH, EAST, SOUTH, WEST)) + return list(input_dir) + switch(input_dir) + if(NORTHEAST) + return list(NORTH, EAST) + if(NORTHWEST) + return list(NORTH, WEST) + if(SOUTHEAST) + return list(SOUTH, EAST) + if(SOUTHWEST) + return list(SOUTH, WEST) + +//heap.dm +////////////////////// +//datum/heap object +////////////////////// + +/datum/heap + var/list/L + var/cmp + +/datum/heap/New(compare) + L = new() + cmp = compare + +/datum/heap/Destroy(force, ...) + for(var/i in L) // because this is before the list helpers are loaded + qdel(i) + L = null + return ..() + +/datum/heap/proc/is_empty() + return !length(L) + +//insert and place at its position a new node in the heap +/datum/heap/proc/insert(atom/A) + + L.Add(A) + swim(length(L)) + +//removes and returns the first element of the heap +//(i.e the max or the min dependant on the comparison function) +/datum/heap/proc/pop() + if(!length(L)) + return 0 + . = L[1] + + L[1] = L[length(L)] + L.Cut(length(L)) + if(length(L)) + sink(1) + +//Get a node up to its right position in the heap +/datum/heap/proc/swim(index) + var/parent = round(index * 0.5) + + while(parent > 0 && (call(cmp)(L[index],L[parent]) > 0)) + L.Swap(index,parent) + index = parent + parent = round(index * 0.5) + +//Get a node down to its right position in the heap +/datum/heap/proc/sink(index) + var/g_child = get_greater_child(index) + + while(g_child > 0 && (call(cmp)(L[index],L[g_child]) < 0)) + L.Swap(index,g_child) + index = g_child + g_child = get_greater_child(index) + +//Returns the greater (relative to the comparison proc) of a node children +//or 0 if there's no child +/datum/heap/proc/get_greater_child(index) + if(index * 2 > length(L)) + return 0 + + if(index * 2 + 1 > length(L)) + return index * 2 + + if(call(cmp)(L[index * 2],L[index * 2 + 1]) < 0) + return index * 2 + 1 + else + return index * 2 + +//Replaces a given node so it verify the heap condition +/datum/heap/proc/resort(atom/A) + var/index = L.Find(A) + + swim(index) + sink(index) + +/datum/heap/proc/List() + . = L.Copy() +//heap.dm end + +//path.dm +/** + * This file contains the stuff you need for using JPS (Jump Point Search) pathing, an alternative to A* that skips + * over large numbers of uninteresting tiles resulting in much quicker pathfinding solutions. Mind that diagonals + * cost the same as cardinal moves currently, so paths may look a bit strange, but should still be optimal. + */ + +/** + * This is the proc you use whenever you want to have pathfinding more complex than "try stepping towards the thing". + * If no path was found, returns an empty list, which is important for bots like medibots who expect an empty list rather than nothing. + * + * Arguments: + * * caller: The movable atom that's trying to find the path + * * end: What we're trying to path to. It doesn't matter if this is a turf or some other atom, we're gonna just path to the turf it's on anyway + * * max_distance: The maximum number of steps we can take in a given path to search (default: 30, 0 = infinite) + * * mintargetdistance: Minimum distance to the target before path returns, could be used to get near a target, but not right to it - for an AI mob with a gun, for example. + * * id: An ID card representing what access we have and what doors we can open. Its location relative to the pathing atom is irrelevant + * * simulated_only: Whether we consider turfs without atmos simulation (AKA do we want to ignore space) + * * exclude: If we want to avoid a specific turf, like if we're a mulebot who already got blocked by some turf + * * skip_first: Whether or not to delete the first item in the path. This would be done because the first item is the starting tile, which can break movement for some creatures. + */ +/proc/jps_get_path_to(caller, end, max_distance = 30, mintargetdist, id=null, simulated_only = TRUE, turf/exclude, skip_first = TRUE) + if(!caller || !get_turf(end)) + return + + var/l = SSpathfinder.mobs.getfree(caller) + while(!l) + stoplag(3) + l = SSpathfinder.mobs.getfree(caller) + + var/list/path + var/datum/pathfind/pathfind_datum = new(caller, end, id, max_distance, mintargetdist, simulated_only, exclude) + path = pathfind_datum.search() + qdel(pathfind_datum) + + SSpathfinder.mobs.found(l) + if(!path) + path = list() + if(length(path) > 0 && skip_first) + path.Cut(1,2) + return path + +/** + * A helper macro to see if it's possible to step from the first turf into the second one, minding things like door access and directional windows. + * Note that this can only be used inside the [datum/pathfind][pathfind datum] since it uses variables from said datum. + * If you really want to optimize things, optimize this, cuz this gets called a lot. + */ +#define CAN_STEP(cur_turf, next) (next && !next.density && cur_turf.Adjacent(next) && !(simulated_only && SSpathfinder.space_type_cache[next.type]) && !cur_turf.jps_LinkBlockedWithAccess(next,caller, id) && (next != avoid)) +/// Another helper macro for JPS, for telling when a node has forced neighbors that need expanding +#define STEP_NOT_HERE_BUT_THERE(cur_turf, dirA, dirB) ((!CAN_STEP(cur_turf, get_step(cur_turf, dirA)) && CAN_STEP(cur_turf, get_step(cur_turf, dirB)))) + +/// The JPS Node datum represents a turf that we find interesting enough to add to the open list and possibly search for new tiles from +/datum/jps_node + /// The turf associated with this node + var/turf/tile + /// The node we just came from + var/datum/jps_node/previous_node + /// The A* node weight (f_value = number_of_tiles + heuristic) + var/f_value + /// The A* node heuristic (a rough estimate of how far we are from the goal) + var/heuristic + /// How many steps it's taken to get here from the start (currently pulling double duty as steps taken & cost to get here, since all moves incl diagonals cost 1 rn) + var/number_tiles + /// How many steps it took to get here from the last node + var/jumps + /// Nodes store the endgoal so they can process their heuristic without a reference to the pathfind datum + var/turf/node_goal + +/datum/jps_node/New(turf/our_tile, datum/jps_node/incoming_previous_node, jumps_taken, turf/incoming_goal) + tile = our_tile + jumps = jumps_taken + if(incoming_goal) // if we have the goal argument, this must be the first/starting node + node_goal = incoming_goal + else if(incoming_previous_node) // if we have the parent, this is from a direct lateral/diagonal scan, we can fill it all out now + previous_node = incoming_previous_node + number_tiles = previous_node.number_tiles + jumps + node_goal = previous_node.node_goal + heuristic = get_dist(tile, node_goal) + f_value = number_tiles + heuristic + // otherwise, no parent node means this is from a subscan lateral scan, so we just need the tile for now until we call [datum/jps/proc/update_parent] on it + +/datum/jps_node/Destroy(force, ...) + previous_node = null + return ..() + +/datum/jps_node/proc/update_parent(datum/jps_node/new_parent) + previous_node = new_parent + node_goal = previous_node.node_goal + jumps = get_dist(tile, previous_node.tile) + number_tiles = previous_node.number_tiles + jumps + heuristic = get_dist(tile, node_goal) + f_value = number_tiles + heuristic + +/// TODO: Macro this to reduce proc overhead +/proc/jps_HeapPathWeightCompare(datum/jps_node/a, datum/jps_node/b) + return b.f_value - a.f_value + +/// The datum used to handle the JPS pathfinding, completely self-contained +/datum/pathfind + /// The thing that we're actually trying to path for + var/atom/movable/caller + /// The turf where we started at + var/turf/start + /// The turf we're trying to path to (note that this won't track a moving target) + var/turf/end + /// The open list/stack we pop nodes out from (TODO: make this a normal list and macro-ize the heap operations to reduce proc overhead) + var/datum/heap/open + ///An assoc list that serves as the closed list & tracks what turfs came from where. Key is the turf, and the value is what turf it came from + var/list/sources + /// The list we compile at the end if successful to pass back + var/list/path + + // general pathfinding vars/args + /// An ID card representing what access we have and what doors we can open. Its location relative to the pathing atom is irrelevant + var/obj/item/card/id/id + /// How far away we have to get to the end target before we can call it quits + var/mintargetdist = 0 + /// I don't know what this does vs , but they limit how far we can search before giving up on a path + var/max_distance = 30 + /// Space is big and empty, if this is TRUE then we ignore pathing through unsimulated tiles + var/simulated_only + /// A specific turf we're avoiding, like if a mulebot is being blocked by someone t-posing in a doorway we're trying to get through + var/turf/avoid + /// A specific turf we're avoiding, like if a mulebot is being blocked by someone t-posing in a doorway we're trying to get through + var/list/turfs_to_avoid = list(/turf/open/water/acid,/turf/open/lava) + +/datum/pathfind/New(atom/movable/caller, atom/goal, id, max_distance, mintargetdist, simulated_only, avoid, avoid_mobs) + src.caller = caller + end = get_turf(goal) + open = new /datum/heap(GLOBAL_PROC_REF(jps_HeapPathWeightCompare)) + sources = new() + src.id = id + src.max_distance = max_distance + src.mintargetdist = mintargetdist + src.simulated_only = simulated_only + src.avoid = avoid + +/** + * search() is the proc you call to kick off and handle the actual pathfinding, and kills the pathfind datum instance when it's done. + * + * If a valid path was found, it's returned as a list. If invalid or cross-z-level params are entered, or if there's no valid path found, we + * return null, which [/proc/jps_get_path_to] translates to an empty list (notable for simple bots, who need empty lists) + */ +/datum/pathfind/proc/search() + start = get_turf(caller) + if(!start || !end) + stack_trace("Invalid A* start or destination") + return + if(start.z != end.z || start == end ) //no pathfinding between z levels + return + if(max_distance && (max_distance < get_dist(start, end))) //if start turf is farther than max_distance from end turf, no need to do anything + return + + //initialization + var/datum/jps_node/current_processed_node = new (start, -1, 0, end) + open.insert(current_processed_node) + sources[start] = start // i'm sure this is fine + + //then run the main loop + while(!open.is_empty() && !path) + if(!caller) + return + current_processed_node = open.pop() //get the lower f_value turf in the open list + if(max_distance && (current_processed_node.number_tiles > max_distance))//if too many steps, don't process that path + continue + + var/turf/current_turf = current_processed_node.tile + for(var/scan_direction in list(EAST, WEST, NORTH, SOUTH)) + lateral_scan_spec(current_turf, scan_direction, current_processed_node) + + for(var/scan_direction in list(NORTHEAST, SOUTHEAST, NORTHWEST, SOUTHWEST)) + diag_scan_spec(current_turf, scan_direction, current_processed_node) + + CHECK_TICK + + //we're done! reverse the path to get it from start to finish + if(path) + for(var/i = 1 to round(0.5 * length(path))) + path.Swap(i, length(path) - i + 1) + + sources = null + qdel(open) + return path + +/// Called when we've hit the goal with the node that represents the last tile, then sets the path var to that path so it can be returned by [datum/pathfind/proc/search] +/datum/pathfind/proc/unwind_path(datum/jps_node/unwind_node) + path = new() + var/turf/iter_turf = unwind_node.tile + path.Add(iter_turf) + + while(unwind_node.previous_node) + var/dir_goal = get_dir(iter_turf, unwind_node.previous_node.tile) + for(var/i = 1 to unwind_node.jumps) + iter_turf = get_step(iter_turf,dir_goal) + path.Add(iter_turf) + unwind_node = unwind_node.previous_node + +/** + * For performing lateral scans from a given starting turf. + * + * These scans are called from both the main search loop, as well as subscans for diagonal scans, and they treat finding interesting turfs slightly differently. + * If we're doing a normal lateral scan, we already have a parent node supplied, so we just create the new node and immediately insert it into the heap, ezpz. + * If we're part of a subscan, we still need for the diagonal scan to generate a parent node, so we return a node datum with just the turf and let the diag scan + * proc handle transferring the values and inserting them into the heap. + * + * Arguments: + * * original_turf: What turf did we start this scan at? + * * heading: What direction are we going in? Obviously, should be cardinal + * * parent_node: Only given for normal lateral scans, if we don't have one, we're a diagonal subscan. +*/ +/datum/pathfind/proc/lateral_scan_spec(turf/original_turf, heading, datum/jps_node/parent_node) + var/steps_taken = 0 + + var/turf/current_turf = original_turf + var/turf/lag_turf = original_turf + + while(TRUE) + if(path) + return + lag_turf = current_turf + current_turf = get_step(current_turf, heading) + steps_taken++ + if(!CAN_STEP(lag_turf, current_turf)) + return + for(var/turf/checked_turf as anything in turfs_to_avoid) + if(istype(current_turf,checked_turf)) + return + + if(current_turf == end || (mintargetdist && (get_dist(current_turf, end) <= mintargetdist))) + var/datum/jps_node/final_node = new(current_turf, parent_node, steps_taken) + sources[current_turf] = original_turf + if(parent_node) // if this is a direct lateral scan we can wrap up, if it's a subscan from a diag, we need to let the diag make their node first, then finish + unwind_path(final_node) + return final_node + else if(sources[current_turf]) // already visited, essentially in the closed list + return + else + sources[current_turf] = original_turf + + if(parent_node && parent_node.number_tiles + steps_taken > max_distance) + return + + var/interesting = FALSE // have we found a forced neighbor that would make us add this turf to the open list? + + switch(heading) + if(NORTH) + if(STEP_NOT_HERE_BUT_THERE(current_turf, WEST, NORTHWEST) || STEP_NOT_HERE_BUT_THERE(current_turf, EAST, NORTHEAST)) + interesting = TRUE + if(SOUTH) + if(STEP_NOT_HERE_BUT_THERE(current_turf, WEST, SOUTHWEST) || STEP_NOT_HERE_BUT_THERE(current_turf, EAST, SOUTHEAST)) + interesting = TRUE + if(EAST) + if(STEP_NOT_HERE_BUT_THERE(current_turf, NORTH, NORTHEAST) || STEP_NOT_HERE_BUT_THERE(current_turf, SOUTH, SOUTHEAST)) + interesting = TRUE + if(WEST) + if(STEP_NOT_HERE_BUT_THERE(current_turf, NORTH, NORTHWEST) || STEP_NOT_HERE_BUT_THERE(current_turf, SOUTH, SOUTHWEST)) + interesting = TRUE + + if(interesting) + var/datum/jps_node/newnode = new(current_turf, parent_node, steps_taken) + if(parent_node) // if we're a diagonal subscan, we'll handle adding ourselves to the heap in the diag + open.insert(newnode) + return newnode + +/** + * For performing diagonal scans from a given starting turf. + * + * Unlike lateral scans, these only are called from the main search loop, so we don't need to worry about returning anything, + * though we do need to handle the return values of our lateral subscans of course. + * + * Arguments: + * * original_turf: What turf did we start this scan at? + * * heading: What direction are we going in? Obviously, should be diagonal + * * parent_node: We should always have a parent node for diagonals +*/ +/datum/pathfind/proc/diag_scan_spec(turf/original_turf, heading, datum/jps_node/parent_node) + var/steps_taken = 0 + var/turf/current_turf = original_turf + var/turf/lag_turf = original_turf + + while(TRUE) + if(path) + return + lag_turf = current_turf + current_turf = get_step(current_turf, heading) + steps_taken++ + if(!CAN_STEP(lag_turf, current_turf)) + return + for(var/turf/checked_turf as anything in turfs_to_avoid) + if(istype(current_turf,checked_turf)) + return + + if(current_turf == end || (mintargetdist && (get_dist(current_turf, end) <= mintargetdist))) + var/datum/jps_node/final_node = new(current_turf, parent_node, steps_taken) + sources[current_turf] = original_turf + unwind_path(final_node) + return + else if(sources[current_turf]) // already visited, essentially in the closed list + return + else + sources[current_turf] = original_turf + + if(parent_node.number_tiles + steps_taken > max_distance) + return + + var/interesting = FALSE // have we found a forced neighbor that would make us add this turf to the open list? + var/datum/jps_node/possible_child_node // otherwise, did one of our lateral subscans turn up something? + + switch(heading) + if(NORTHWEST) + if(STEP_NOT_HERE_BUT_THERE(current_turf, EAST, NORTHEAST) || STEP_NOT_HERE_BUT_THERE(current_turf, SOUTH, SOUTHWEST)) + interesting = TRUE + else + possible_child_node = (lateral_scan_spec(current_turf, WEST) || lateral_scan_spec(current_turf, NORTH)) + if(NORTHEAST) + if(STEP_NOT_HERE_BUT_THERE(current_turf, WEST, NORTHWEST) || STEP_NOT_HERE_BUT_THERE(current_turf, SOUTH, SOUTHEAST)) + interesting = TRUE + else + possible_child_node = (lateral_scan_spec(current_turf, EAST) || lateral_scan_spec(current_turf, NORTH)) + if(SOUTHWEST) + if(STEP_NOT_HERE_BUT_THERE(current_turf, EAST, SOUTHEAST) || STEP_NOT_HERE_BUT_THERE(current_turf, NORTH, NORTHWEST)) + interesting = TRUE + else + possible_child_node = (lateral_scan_spec(current_turf, SOUTH) || lateral_scan_spec(current_turf, WEST)) + if(SOUTHEAST) + if(STEP_NOT_HERE_BUT_THERE(current_turf, WEST, SOUTHWEST) || STEP_NOT_HERE_BUT_THERE(current_turf, NORTH, NORTHEAST)) + interesting = TRUE + else + possible_child_node = (lateral_scan_spec(current_turf, SOUTH) || lateral_scan_spec(current_turf, EAST)) + + if(interesting || possible_child_node) + var/datum/jps_node/newnode = new(current_turf, parent_node, steps_taken) + open.insert(newnode) + if(possible_child_node) + possible_child_node.update_parent(newnode) + open.insert(possible_child_node) + if(possible_child_node.tile == end || (mintargetdist && (get_dist(possible_child_node.tile, end) <= mintargetdist))) + unwind_path(possible_child_node) + return + +/** + * For seeing if we can actually move between 2 given turfs while accounting for our access and the caller's pass_flags + * + * Arguments: + * * caller: The movable, if one exists, being used for mobility checks to see what tiles it can reach + * * ID: An ID card that decides if we can gain access to doors that would otherwise block a turf + * * simulated_only: Do we only worry about turfs with simulated atmos, most notably things that aren't space? +*/ +/turf/proc/jps_LinkBlockedWithAccess(turf/destination_turf, caller, ID) + var/actual_dir = get_dir(src, destination_turf) + + for(var/obj/structure/window/iter_window in src) + if(!iter_window.CanAStarPass(ID, actual_dir)) + return TRUE + + for(var/obj/machinery/door/window/iter_windoor in src) + if(!iter_windoor.CanAStarPass(ID, actual_dir)) + return TRUE + + var/reverse_dir = get_dir(destination_turf, src) + for(var/obj/iter_object in destination_turf) + if(!iter_object.CanAStarPass(ID, reverse_dir, caller)) + return TRUE + + return FALSE + +#undef CAN_STEP +#undef STEP_NOT_HERE_BUT_THERE +//end path.dm + +GLOBAL_LIST_EMPTY(knpcs) + +/datum/component/knpc + var/ai_trait = AI_AGGRESSIVE + var/static/list/ai_goals = null + var/datum/ai_goal/human/current_goal = null + var/view_range = 8 //How good is this mob's "eyes"? + var/guess_range = 12 //How far away will we assume they're still there after seeing them? + var/list/last_aggressors = list() + var/next_backup_call = 0 //Delay for calling for backup to avoid spam. + var/list/path = list() + var/turf/dest = null + var/tries = 0 //How quickly do we give up on following a path? To avoid lag... + var/max_tries = 10 + var/next_action = 0 + var/next_move = 0 + var/obj/effect/landmark/patrol_node/last_node = null //What was the last patrol node we visited? + var/stealing_id = FALSE + var/next_internals_attempt = 0 + var/static/list/climbable = typecacheof(list(/obj/structure/table, /obj/structure/railing)) // climbable structures + var/pathfind_timeout = 0 //If pathfinding fails, it is püt in timeout for a while to avoid spamming the server with pathfinding calls. + var/timeout_stacks = 0 //Consecutive pathfind fails add additional delay stacks to further counteract the effects of knpcs in unreachable locations. + var/list/turfs_to_avoid = list(/turf/open/water/acid,/turf/open/lava) + +/mob/living/carbon/human/ai_boarder + faction = list("neutral") + var/move_delay = 4 //How quickly do the boys travel? + var/action_delay = 6 //How long we delay between actions + var/knpc_traits = KNPC_IS_DODGER | KNPC_IS_MERCIFUL | KNPC_IS_AREA_SPECIFIC + var/difficulty = KNPC_DIFFICULTY_HARD //Whether to ignore overmap difficulty or not + var/list/outfit = list ( + /datum/outfit/job/assistant + ) + var/list/taunts = list("Engaging the enemy!") + var/list/call_lines = list("Enemy spotted!") + var/list/response_lines = list("On my way!") + + var/shut_up = TRUE + +/mob/living/carbon/human/ai_boarder/Initialize(mapload) + . = ..() + randomize_human(src) + var/outfit_path = pick(outfit) + var/datum/outfit/O = new outfit_path + O.equip(src) + AddComponent(/datum/component/knpc) + +/mob/living/carbon/human/ai_boarder/frontier + faction = list("frontier") + outfit = list (/datum/outfit/frontier) + +/mob/living/carbon/human/ai_boarder/hermit + faction = list("hermit") + taunts = list("...") + call_lines = list("...!!!") + response_lines = list("!") + + knpc_traits = KNPC_IS_DODGER | KNPC_IS_MERCIFUL + + outfit = list (/datum/outfit/whitesands) + + var/survivor_type + +/mob/living/carbon/human/ai_boarder/hermit/survivor + survivor_type = "survivor" + +/mob/living/carbon/human/ai_boarder/hermit/hunter + survivor_type = "hunter" + +/mob/living/carbon/human/ai_boarder/hermit/gunslinger + survivor_type = "gunslinger" + +/mob/living/carbon/human/ai_boarder/hermit/commando + survivor_type = "commando" + +/mob/living/carbon/human/ai_boarder/hermit/Initialize(mapload) + if(!survivor_type) + survivor_type = pick("survivor","hunter","gunslinger") + ..() + for(var/obj/item/gun/searching as obj in contents) + if(istype(searching, /obj/item/gun)) + searching.safety = FALSE + + var/mob_species = pickweight(list( + /datum/species/human = 50, + /datum/species/lizard = 25, + /datum/species/elzuose = 10, + /datum/species/moth = 10, + /datum/species/spider = 3, + /datum/species/fly = 2 + ) + ) + INVOKE_ASYNC(src, PROC_REF(set_species), mob_species) + + var/obj/item/clothing/suit/hooded/survivor_hood = wear_suit + if(survivor_hood) + survivor_hood.ToggleHood() + +/datum/outfit/whitesands/pre_equip(mob/living/carbon/human/H, visualsOnly) + var/mob/living/carbon/human/ai_boarder/hermit/hermit = H + var/survivor_type = hermit.survivor_type + var/picked + //to-do: learn how to make mobsprites for other survivors + //w_uniforms are random to show varied backgrounds, but similar goal + if(survivor_type == "survivor") + picked = pickweight(list( + /obj/item/clothing/under/color/random = 65, + /obj/item/clothing/under/rank/cargo/miner/lavaland = 10, + /obj/item/clothing/under/rank/prisoner = 10, + /obj/item/clothing/under/rank/cargo/miner/lavaland/old = 5, + /obj/item/clothing/under/color/khaki/buster = 5, + /obj/item/clothing/under/rank/cargo/miner = 5 + ) + ) + else if (survivor_type == "hunter") + picked = pickweight(list( + /obj/item/clothing/under/color/random = 50, + /obj/item/clothing/under/rank/cargo/miner/lavaland = 25, + /obj/item/clothing/under/rank/cargo/miner/lavaland/old = 15, + /obj/item/clothing/under/rank/security/officer/camo = 5, + /obj/item/clothing/under/utility = 5 + ) + ) + else if (survivor_type == "gunslinger" || survivor_type == "commando") + picked = pickweight(list( + /obj/item/clothing/under/rank/cargo/miner/lavaland = 35, + /obj/item/clothing/under/color/random = 25, + /obj/item/clothing/under/rank/cargo/miner/lavaland/old = 15, + /obj/item/clothing/under/rank/security/officer/camo = 10, + /obj/item/clothing/under/syndicate/camo = 10, + /obj/item/clothing/under/syndicate/combat = 5 + ) + ) + else + picked = /obj/item/clothing/under/color/random + + uniform = picked + + //storage is semi-randomized, giving some variety + if(survivor_type == "survivor") + picked = pickweight(list( + /obj/item/storage/belt/fannypack = 40, + /obj/item/storage/belt/mining = 20, + /obj/item/storage/belt/mining/alt = 15, + /obj/item/storage/belt/utility = 10, + /obj/item/storage/belt/bandolier = 9, + /obj/item/storage/belt/utility/full = 5, + /obj/item/storage/belt/chameleon= 1, + ) + ) + else if(survivor_type == "hunter") + picked = pickweight(list( + /obj/item/storage/belt/mining = 30, + /obj/item/storage/belt/fannypack = 20, + /obj/item/storage/belt/mining/alt = 15, + /obj/item/storage/belt/mining/primitive = 15, + /obj/item/storage/belt/bandolier = 10, + /obj/item/storage/belt/military = 7, + /obj/item/storage/belt/mining/vendor = 3, + ) + ) + else if(survivor_type == "gunslinger" || survivor_type == "commando") + picked = pickweight(list( + /obj/item/storage/belt/mining = 30, + /obj/item/storage/belt/bandolier = 30, + /obj/item/storage/belt/military = 20, + /obj/item/storage/belt/fannypack = 15, + /obj/item/storage/belt/mining/alt = 5, + /obj/item/storage/belt/mining/primitive = 5 + ) + ) + else + picked = /obj/item/storage/belt/fannypack + + belt = picked + + //everyone wears the same suit + suit = /obj/item/clothing/suit/hooded/survivor + + if (survivor_type == "gunslinger" || survivor_type == "commando") + if(prob(30)) + picked = /obj/item/clothing/shoes/combat //but sometimes there are nicer shoes + else + picked = /obj/item/clothing/shoes/workboots/mining + else + picked = /obj/item/clothing/shoes/workboots/mining + + shoes = picked + + //gloves are a tossup + picked = pickweight(list( + /obj/item/clothing/gloves/color/black = 60, + /obj/item/clothing/gloves/explorer = 30, + /obj/item/clothing/gloves/explorer/old = 10 + ) + ) + + gloves = picked + + //bags are semi-random. + picked = pickweight(list( + /obj/item/storage/backpack = 20, + /obj/item/storage/backpack/explorer = 20, + /obj/item/storage/backpack/satchel = 20, + /obj/item/storage/backpack/satchel/explorer = 20, + /obj/item/storage/backpack/messenger = 20 + ) + ) + + back = picked + + //as are bag contents + + backpack_contents = list() + backpack_contents += pickweight(list( //fallback in case of no weapons + /obj/item/wrench = 10, + /obj/item/crowbar = 15, + /obj/item/screwdriver = 5, + /obj/item/wirecutters = 10, + /obj/item/scalpel = 5, + /obj/item/flashlight/seclite = 10, + )) + if(prob(70)) + backpack_contents += pickweight(list( //these could stand to be expanded, right now they're just mildly modified miner ones, and I don't know how to plus that up. + /obj/item/soap = 10, + /obj/item/stack/marker_beacon/ten = 15, + /obj/item/mining_scanner = 5, + /obj/item/extinguisher/mini = 10, + /obj/item/kitchen/knife/combat = 5, + /obj/item/flashlight/seclite = 10, + /obj/item/stack/sheet/sinew = 10, + /obj/item/stack/sheet/bone = 5, + /obj/item/stack/sheet/animalhide/goliath_hide = 10, + /obj/item/stack/sheet/bone = 8, + /obj/item/reagent_containers/food/drinks/waterbottle = 10, + /obj/item/reagent_containers/food/drinks/waterbottle/empty = 2, + ) + ) + if(prob(70)) + backpack_contents += pickweight(list( + /obj/item/stack/sheet/animalhide/goliath_hide = 20, + /obj/item/stack/marker_beacon/ten = 10, + /obj/item/mining_scanner = 20, + /obj/item/extinguisher/mini = 10, + /obj/item/kitchen/knife/combat/survival = 10, + /obj/item/flashlight/seclite = 10, + /obj/item/stack/sheet/sinew = 10, + /obj/item/stack/sheet/bone = 10 + ) + ) + if(prob(70)) + backpack_contents += pickweight(list( + /obj/item/stack/sheet/animalhide/goliath_hide = 5, + /obj/item/stack/marker_beacon/ten = 5, + /obj/item/mining_scanner = 5, + /obj/item/extinguisher/mini = 10, + /obj/item/kitchen/knife/combat/survival = 12, + /obj/item/flashlight/seclite = 10, + /obj/item/stack/sheet/sinew = 5, + /obj/item/stack/sheet/bone = 5, + /obj/item/kitchen/knife/combat = 3, + /obj/item/reagent_containers/food/snacks/rationpack = 30 + ) + ) + if (prob(25)) //mayhaps a medkit + backpack_contents += pickweight(list( + /obj/item/storage/firstaid/regular = 50, + /obj/item/storage/firstaid/brute = 15, + /obj/item/storage/firstaid/medical = 15, + /obj/item/storage/firstaid/fire = 10, + /obj/item/storage/firstaid/advanced = 5, + /obj/item/storage/firstaid/ancient = 5 + ) + ) + if(prob(30)) //some pens maybe? + backpack_contents += /obj/item/reagent_containers/hypospray/medipen/survival + if(prob(5)) + backpack_contents += /obj/item/reagent_containers/hypospray/medipen/survival + + //pockets + if(survivor_type == "survivor") //could also use fleshing out + if(prob(30)) + l_pocket = /obj/item/reagent_containers/food/snacks/meat/steak/goliath + else + l_pocket = /obj/item/tank/internals/emergency_oxygen/engi + + if(survivor_type == "hunter") + l_pocket = /obj/item/tank/internals/emergency_oxygen/double + if (prob(20)) + l_pocket = /obj/item/reagent_containers/food/snacks/meat/steak/goliath + + if(survivor_type == "gunslinger" || survivor_type == "commando") + if(prob(50)) + l_pocket = /obj/item/ammo_box/magazine/skm_545_39 + r_pocket = /obj/item/tank/internals/emergency_oxygen/double + + else + r_pocket = /obj/item/tank/internals/emergency_oxygen/engi + l_pocket = /obj/item/radio + + //masks + picked = pickweight(list( + /obj/item/clothing/mask/gas = 40, + /obj/item/clothing/mask/gas/explorer = 20, + /obj/item/clothing/mask/gas/explorer/old = 20, + /obj/item/clothing/mask/gas/syndicate = 20, + /obj/item/clothing/mask/breath = 5, + /obj/item/clothing/mask/breath/medical = 5, + /obj/item/clothing/mask/breath/suns = 5, + /obj/item/clothing/mask/gas/sechailer = 10, + /obj/item/clothing/mask/gas/sechailer/balaclava = 10, + /obj/item/clothing/mask/gas/sechailer/balaclava/inteq = 10, + /obj/item/clothing/mask/gas/sechailer/swat = 1, + /obj/item/clothing/mask/gas/sechailer/swat/spacepol = 1, + ) + ) + + mask = picked + + //the eyes are the window into the soul. I don't think these things have souls but whatever. + if(prob(70)) + picked = pickweight(list( + /obj/item/clothing/glasses/heat = 20, + /obj/item/clothing/glasses/cold = 20, + /obj/item/clothing/glasses/meson = 40, + /obj/item/clothing/glasses = 20 + ) + ) + glasses = picked + else + glasses = null + + //and of course, ears. + if(prob(1)) //oh my god they can't hear the sandstorm coming they've got airpods in + picked = /obj/item/instrument/piano_synth/headphones/spacepods + else + picked = pickweight(list( + /obj/item/radio/headset = 50, + /obj/item/radio/headset/alt = 50 + ) + ) + ears = picked + + //exosuit bits + suit_store = null + var/spare_ammo_count = rand(1,2) + + if (survivor_type == "hunter") + r_hand = /obj/item/gun/ballistic/rifle/polymer + spare_ammo_count = rand(2,5) + for(var/i in 1 to spare_ammo_count) + backpack_contents += /obj/item/ammo_box/a762_stripper + + if(survivor_type == "gunslinger" || survivor_type == "commando") + if(prob(7) || survivor_type == "commando") //cause fuck you, thats why + uniform = /obj/item/clothing/under/rank/security/officer/camo + + suit = pickweight(list( + /obj/item/clothing/suit/armor/vest/bulletproof = 35, + /obj/item/clothing/suit/armor/vest/syndie = 20, + /obj/item/clothing/suit/armor/gezena = 20, + /obj/item/clothing/suit/armor/vest/marine = 1, + /obj/item/clothing/suit/armor/vest/marine/heavy = 1, + /obj/item/clothing/suit/armor/vest/marine = 1, + )) + head = pickweight(list( + /obj/item/clothing/head/helmet/bulletproof/x11 = 40, + /obj/item/clothing/head/helmet/bulletproof/m10 = 40, + + /obj/item/clothing/head/helmet/swat = 20, + /obj/item/clothing/head/helmet/swat/nanotrasen = 20, + /obj/item/clothing/head/helmet/gezena = 20, + + /obj/item/clothing/head/helmet/marine = 1, + /obj/item/clothing/head/helmet/marine/security = 1, + )) + spare_ammo_count = rand(2,3) + if(prob(10)) + r_hand = /obj/item/gun/ballistic/automatic/hmg/skm_lmg/drum_mag //hell + else + r_hand = /obj/item/gun/ballistic/automatic/assault/skm/pirate + for(var/i in 1 to spare_ammo_count) + if(prob(5)) + backpack_contents += /obj/item/ammo_box/magazine/skm_762_40/drum //die. + else if(prob(20)) + backpack_contents += /obj/item/ammo_box/magazine/skm_762_40/extended + else + backpack_contents += /obj/item/ammo_box/magazine/skm_762_40 + else + picked = pickweight(list( + /obj/item/gun/ballistic/automatic/smg/skm_carbine = 70, + /obj/item/gun/ballistic/automatic/pistol/spitter = 20, + /obj/item/gun/ballistic/automatic/smg/pounder = 4, + /obj/item/gun/ballistic/automatic/smg/firestorm = 4, + /obj/item/gun/ballistic/automatic/smg/vector = 4, + /obj/item/gun/ballistic/automatic/smg/cm5 = 4, + /obj/item/gun/ballistic/automatic/smg/skm_carbine/inteq = 4, + /obj/item/gun/ballistic/automatic/smg/cobra = 4, + /obj/item/gun/ballistic/automatic/smg/skm_carbine/inteq/proto = 4, + /obj/item/gun/ballistic/automatic/pistol/rattlesnake = 4, + /obj/item/gun/ballistic/automatic/pistol/mauler = 4, + )) + r_hand = picked + var/obj/item/gun/ballistic/current_gun = picked + for(var/i in 1 to spare_ammo_count) + backpack_contents += current_gun.mag_type + + internals_slot = ITEM_SLOT_RPOCKET + +/datum/component/knpc/Initialize() + if(!iscarbon(parent)) + return COMPONENT_INCOMPATIBLE + if(!ai_goals) + for(var/gtype in subtypesof(/datum/ai_goal/human)) + LAZYADD(ai_goals, new gtype) + START_PROCESSING(SSfastprocess, src) + //They're alive! + GLOB.knpcs.Add(src) + RegisterSignal(parent, COMSIG_LIVING_REVIVE, PROC_REF(restart)) + RegisterSignal(parent, COMSIG_ATOM_BULLET_ACT, PROC_REF(register_bullet)) + +//Swiper! no swiping +/datum/component/knpc/proc/steal_id(obj/item/card/id/their_id) + //Time to teach them about IDs :))) + stealing_id = FALSE + var/mob/living/carbon/human/H = parent + if(get_dist(H, their_id.loc) > 1) + return FALSE + var/obj/item/card/id/ID = H.get_idcard() + if(ID) + if(istype(ID, /obj/item/card/id/)) + their_id.forceMove(get_turf(H)) + H.visible_message("[H] snatches [their_id]. ") + ID.access |= their_id.access + else + H.put_in_inactive_hand(their_id) + if(H.equip_to_appropriate_slot(their_id)) + H.update_inv_hands() + return TRUE + +/datum/component/knpc/Destroy(force, silent) + GLOB.knpcs -= src + return ..() + +/datum/component/knpc/proc/pathfind_to(atom/target, turf/avoid) + if(pathfind_timeout > 0) + return KNPC_PATHFIND_TIMEOUT + var/mob/living/carbon/human/ai_boarder/H = parent + if(target == null) + path = list() + dest = null + return KNPC_PATHFIND_SKIP + if((dest && dest == get_turf(target) && length(path)) || H.incapacitated()) + return KNPC_PATHFIND_SKIP //No need to recalculate this path. + path = list() + dest = null + var/obj/item/card/id/access_card = H.wear_id + if(target) + dest = get_turf(target) + path = jps_get_path_to(H, dest, 120, 0, access_card, !(H.wear_suit?.clothing_flags & STOPSPRESSUREDAMAGE && H.head?.clothing_flags & STOPSPRESSUREDAMAGE), avoid) + //There's no valid path, try run against the wall. + if(!length(path) && !H.incapacitated()) + pathfind_timeout += KNPC_TIMEOUT_BASE * (1 + timeout_stacks) + timeout_stacks = min(timeout_stacks+1, KNPC_TIMEOUT_STACK_CAP) + return KNPC_PATHFIND_FAIL + timeout_stacks = 0 + return KNPC_PATHFIND_SUCCESS + +/datum/component/knpc/proc/next_path_step() + if(world.time < next_move) + return FALSE + var/mob/living/carbon/human/ai_boarder/H = parent + next_move = world.time + H.move_delay + if(H.incapacitated() || H.stat == DEAD) + return FALSE + if(pathfind_timeout > 0) //Pathfinding in timeout, move around aimlessly + H.set_resting(FALSE, FALSE) + var/turf/new_pos = get_step(H,pick(GLOB.cardinals)) + for(var/turf/checked_turf as anything in turfs_to_avoid) + if(istype(new_pos,checked_turf)) + return FALSE + H.Move(new_pos) + return FALSE + if(!path) + return FALSE + if(tries > 5) + //Add a bit of randomness to their movement to reduce "traffic jams" + var/turf/new_pos = get_step(H,pick(GLOB.cardinals)) + for(var/turf/checked_turf as anything in turfs_to_avoid) + if(istype(new_pos,checked_turf)) + return FALSE + H.Move(new_pos) + if(prob(10)) + H.toggle_resting() + return FALSE + + if(tries >= max_tries) + tries = 0 + if(length(last_node?.next_nodes)) //Skip this one. + pathfind_to(pick(last_node.next_nodes)) + else + pathfind_to(null) + last_node = null //Reset pathfinding fully. + return FALSE + if(length(path) > 1) + var/turf/next_turf = get_step_towards(H, path[1]) + var/turf/this_turf = get_turf(H) + //Walk when you see a wet floor + if(next_turf.GetComponent(/datum/component/wet_floor)) + H.m_intent = MOVE_INTENT_WALK + else + H.m_intent = MOVE_INTENT_RUN + + for(var/obj/machinery/door/firedoor/blocking_firelock in next_turf) + if((blocking_firelock.flags_1 & ON_BORDER_1) && !(blocking_firelock.dir in dir_to_cardinal_dirs(get_dir(next_turf, this_turf)))) + continue + if(!blocking_firelock.density || blocking_firelock.operating) + continue + if(blocking_firelock.welded) + break //If at least one firedoor in our way is welded shut, welp! + blocking_firelock.open() //Open one firelock per tile per try. + break + for(var/obj/machinery/door/firedoor/blocking_firelock in this_turf) + if(!((blocking_firelock.flags_1 & ON_BORDER_1) && (blocking_firelock.dir in dir_to_cardinal_dirs(get_dir(this_turf, next_turf))))) //Here, only firelocks on the border matter since fulltile firelocks let you exit. + continue + if(!blocking_firelock.density || blocking_firelock.operating) + continue + if(blocking_firelock.welded) + break //If at least one firedoor in our way is welded shut, welp! + blocking_firelock.open() //Open one firelock per tile per try. + break + for(var/obj/structure/possible_barrier in next_turf) //If we're stuck + if(!climbable.Find(possible_barrier.type)) + continue + H.forceMove(next_turf) + H.visible_message("[H] climbs onto [possible_barrier]!") + H.Stun(2 SECONDS) //Table. + if(get_turf(H) == path[1]) + increment_path() + return TRUE + step_towards(H, path[1]) + if(get_turf(H) == path[1]) //Successful move + increment_path() + tries = 0 + if(H.resting) + //Gotta bypass the do-after here... + H.set_resting(FALSE, FALSE) + else + tries++ + return FALSE + else if(length(path) == 1) + step_to(H, dest) + pathfind_to(null) + return TRUE + +/datum/component/knpc/proc/increment_path() + path.Cut(1, 2) + +///Allows the AI humans to kite around +/datum/component/knpc/proc/kite(atom/movable/target) + if(world.time < next_move) + return + var/mob/living/carbon/human/ai_boarder/H = parent + next_move = world.time + (H.move_delay * 2) + if(!target || !isturf(target.loc) || !isturf(H.loc) || H.stat == DEAD) + return + var/target_dir = get_dir(H,target) + var/static/list/cardinal_sidestep_directions = list(-90,-45,0,45,90) + var/static/list/diagonal_sidestep_directions = list(-45,0,45) + var/chosen_dir = 0 + if (target_dir & (target_dir - 1)) + chosen_dir = pick(diagonal_sidestep_directions) + else + chosen_dir = pick(cardinal_sidestep_directions) + if(chosen_dir) + chosen_dir = turn(target_dir,chosen_dir) + var/turf/new_pos = get_step(H,chosen_dir) + for(var/turf/checked_turf as anything in turfs_to_avoid) + if(istype(new_pos,checked_turf)) + return FALSE + H.Move(new_pos) + H.face_atom(target) //Looks better if they keep looking at you when dodging + +///Allows the AI actor to be revived by a medic, and get straight back into the fight! +/datum/component/knpc/proc/restart() + SIGNAL_HANDLER + START_PROCESSING(SSfastprocess, src) + +///Pick a goal from the available goals! +/datum/component/knpc/proc/pick_goal() + var/best_score = -1000 + var/datum/ai_goal/chosen = null + for(var/datum/ai_goal/human/H in ai_goals) + var/this_score = H.check_score(src) + if(this_score > best_score) + chosen = H + best_score = this_score + if(chosen) + chosen.assume(src) + +///Add someone to our threat list when they shoot us. Shamelessly lifted from monkey AI code. +/datum/component/knpc/proc/register_bullet(datum/source, obj/projectile/Proj) + SIGNAL_HANDLER + if(istype(Proj , /obj/projectile/beam)||istype(Proj, /obj/projectile/bullet)) + if((Proj.damage_type == BURN) || (Proj.damage_type == BRUTE)) + if(!Proj.nodamage && isliving(Proj.firer)) + last_aggressors += Proj.firer + +///Handles actioning on the goal every tick. +/datum/component/knpc/process() + var/mob/living/carbon/human/ai_boarder/H = parent + if(H.stat == DEAD) + return PROCESS_KILL + if(!H.can_resist()) + if(H.incapacitated()) //In crit or something.... + return + if(world.time >= next_action) + next_action = world.time + H.action_delay + pick_goal() + current_goal?.action(src) + if(length(path)) + next_path_step() + else //They should always be pathing somewhere... + dest = null + tries = 0 + path = list() + last_node = null + current_goal?.get_next_patrol_node(src) + pathfind_timeout = max(0, pathfind_timeout - 1) + +/datum/ai_goal + var/name = "Placeholder goal" //Please keep these human readable for debugging! + var/score = 0 + var/required_ai_flags = NONE //Set this if you want this task to only be achievable by certain types of ship. This is a bitfield. + +//Method to get the score of a certain action. This can change the "base" score if the score of a specific action goes up, to encourage skynet to go for that one instead. +//@param OM - If you want this score to be affected by the stats of an overmap. +/datum/ai_goal/proc/check_score() + return FALSE + +//Delete the AI's last orders, tell the AI ship what to do. +/datum/ai_goal/proc/assume() + return FALSE + +/datum/ai_goal/proc/action() + return FALSE + +/datum/ai_goal/human + name = "Placeholder goal" //Please keep these human readable for debugging! + score = 0 + required_ai_flags = null //Set this if you want this task to only be achievable by certain types of ship. + +/** +Method to get the score of a certain action. This can change the "base" score if the score +of a specific action goes up, to encourage skynet to go for that one instead. + +@param OM - If you want this score to be affected by the stats of an overmap. +*/ +/datum/ai_goal/check_score(datum/component/knpc/HA) + if(!istype(HA)) // why is this here >:( + return ..() + if(required_ai_flags && !(HA.ai_trait & required_ai_flags)) + return 0 + var/mob/M = HA.parent + if(M.client) //AI disabled... + return 0 + return score //Children sometimes NEED this true value to run their own checks. We also cancel here if the mob has been overtaken by someone. + +///Delete the AI's last orders, tell the AI ship what to do. +/datum/ai_goal/human/assume(datum/component/knpc/HA) + //message_admins("Goal [src] chosen!") + HA.current_goal = src + +///Method that gets all the potential aggressors for this target. +/datum/ai_goal/proc/get_aggressors(datum/component/knpc/HA) + . = list() + var/mob/living/carbon/human/ai_boarder/H = HA.parent + var/list/detected_objects = view(HA.view_range, HA.parent) + var/list/guessed_objects = view(HA.guess_range, HA.parent) + for(var/mob/living/M in guessed_objects) + //Invis is a no go. Non-human, -cyborg or -hostile mobs are ignored. + if(M.invisibility >= INVISIBILITY_ABSTRACT || M.alpha <= 0 || (!ishuman(M) && !iscyborg(M) && !ishostile(M))) + continue + // Dead mobs are ignored. + if(H.knpc_traits & KNPC_IS_MERCIFUL && M.stat >= UNCONSCIOUS) + continue + else if(M.stat == DEAD) + continue + if(H.faction_check_mob(M)) + continue + if(!(M in detected_objects) && !(M in HA.last_aggressors)) + continue + . += M + //Check for nearby mechas.... + if(length(GLOB.mechas_list)) + for(var/obj/mecha/OM as() in GLOB.mechas_list) + if(OM.z != H.z) + continue + if(get_dist(H, OM) > HA.view_range || !can_see(H, OM, HA.view_range)) + continue + if(OM.occupant && !H.faction_check_mob(OM.occupant)) + . += OM.occupant + +///What happens when this action is selected? You'll override this and check_score mostly. +/datum/ai_goal/human/action(datum/component/knpc/HA) + var/mob/living/carbon/human/H = HA.parent + if(H.incapacitated() || H.client) //An admin overtook this mob or something, ignore. + return FALSE + +/datum/ai_goal/human/proc/can_action(datum/component/knpc/HA) + var/mob/living/carbon/human/H = HA.parent + return (!H.incapacitated() && !H.client) //An admin overtook this mob or something, ignore. + +/** +Goal #1, get a weapon! +If we don't have a weapon, we really ought to grab one... +This is to account for sec Ju-Jitsuing boarding commandos. +*/ +/datum/ai_goal/human/acquire_weapon + name = "Acquire Weapon" //Please keep these human readable for debugging! + score = AI_SCORE_PRIORITY //Fighting takes priority + required_ai_flags = null //Set this if you want this task to only be achievable by certain types of ship. + +/datum/ai_goal/human/acquire_weapon/check_score(datum/component/knpc/HA) + if(!..()) + return 0 + var/mob/living/carbon/human/H = HA.parent + var/obj/item/gun/G = H.get_active_held_item() + //We already have a gun + if(G && istype(G)) + return 0 + var/obj/item/gun/G_New = locate(/obj/item/gun) in oview(HA.view_range, H) + if(G_New && gun_suitable(H, G_New)) + G_New.safety = FALSE + return AI_SCORE_CRITICAL //There is a gun really obviously in the open.... + return score + +/datum/ai_goal/human/proc/CheckFriendlyFire(mob/living/us, mob/living/them) + for(var/turf/T as() in getline(us,them)) // Not 100% reliable but this is faster than simulating actual trajectory + for(var/mob/living/L in T) + if(L == us || L == them) + continue + if(us.faction_check_mob(L)) + return TRUE + +/datum/ai_goal/human/acquire_weapon/action(datum/component/knpc/HA) + if(!can_action(HA)) + return + var/mob/living/carbon/human/H = HA.parent + var/obj/item/storage/S = H.back + var/obj/item/gun/target_item = null + var/obj/item/A = H.get_active_held_item() + //Okay first off, is the gun already on our person? + if(S) + var/list/expanded_contents = S.contents + H.contents + target_item = locate(/obj/item/gun) in expanded_contents + + if(target_item) + H.visible_message("[H] rummages around in their backpack...") + target_item.forceMove(get_turf(H)) //Put it on the floor so they can grab it + if(A) + if(!S) + A.forceMove(get_turf(H)) + else + A.forceMove(S) + if(H.put_in_hands(target_item)) + return TRUE //We're done! + //Now we run the more expensive check to find a gun laying on the ground. + var/best_distance = world.maxx + for(var/obj/O in oview(HA.view_range, H)) + var/dist = get_dist(O, H) + if(istype(O, /obj/structure/closet) && dist <= best_distance) + var/obj/structure/closet/C = O + var/obj/item/gun/G = locate(/obj/item/gun) in C.contents + if(G && C.allowed(H) && gun_suitable(H, G)) + target_item = G + best_distance = dist + if(istype(O, /obj/item/gun) && dist <= best_distance) + var/obj/item/gun/G = O + if(G && gun_suitable(H, G)) + target_item = O + best_distance = dist + if(target_item) + var/dist = get_dist(H, target_item) + if(dist > 1) + HA.pathfind_to(target_item) + else + if(istype(target_item.loc, /obj/structure/closet)) + var/obj/structure/closet/C = target_item.loc + if(C.open(H)) + H.visible_message("[H] pops open [C]...") + if(A) + if(!S) + A.forceMove(get_turf(H)) + else + A.forceMove(S) + if(istype(target_item, /obj/item/gun/ballistic)) + var/obj/item/gun/ballistic/B = target_item + var/obj/item/ammo_box/magazine/M = locate(B.mag_type) in oview(3, target_item) + if(M && S) //If they have a backpack, put the ammo in the backpack. + H.put_in_hands(M) + M.forceMove(S) + + if(H.put_in_hands(target_item)) + H.visible_message("[H] grabs [target_item]!") + +///Checks that G exists, has ammo and that H can fire it. Returns G if yes, FALSE otherwise. +/datum/ai_goal/human/acquire_weapon/proc/gun_suitable(mob/living/carbon/human/H, obj/item/gun/G) + G.safety = FALSE + return G.can_shoot() && G.can_trigger_gun(H) + + +/datum/ai_goal/human/engage_targets + name = "Engage targets" + score = AI_SCORE_SUPERPRIORITY //If we find a target, we want to engage! + required_ai_flags = null + +/datum/ai_goal/human/engage_targets/check_score(datum/component/knpc/HA) + if(!..()) + return 0 + var/list/enemies = get_aggressors(HA) + HA.last_aggressors = enemies + //We have people to fight + if(length(enemies) >= 1) + return score + return 0 //You can still fight with your bare hands... + +/datum/ai_goal/human/proc/reload(datum/component/knpc/HA, obj/item/gun) + var/mob/living/carbon/human/ai_boarder/H = HA.parent + if(istype(gun, /obj/item/gun/energy)) + var/obj/item/gun/energy/E = gun + if(E.selfcharge) //Okay good, it self charges we can just wait. + return TRUE + else //Discard it, we're not gonna teach them to use rechargers yet. + E.forceMove(get_turf(H)) + return FALSE + if(istype(gun, /obj/item/gun/ballistic)) + var/obj/item/gun/ballistic/B = gun + if(istype(B.mag_type, /obj/item/ammo_box/magazine/internal)) + if(!istype(B, /obj/item/gun/ballistic/rifle/polymer) || !istype(B, /obj/item/gun/ballistic/rifle/illestren)) + //Not dealing with this. They'll just ditch the revolver when they're done with it. + B.forceMove(get_turf(H)) + return FALSE + ///message_admins("Issa gun") + var/obj/item/storage/S = H.back + //Okay first off, is the gun already on our person? + var/list/expanded_contents = H.contents + if(S) + expanded_contents = S.contents + H.contents + var/obj/item/ammo_box/target_mag = locate(B.mag_type) in expanded_contents + if(istype(B, /obj/item/gun/ballistic/rifle/polymer)) + target_mag = locate(/obj/item/ammo_box/a762_stripper) in expanded_contents + if(istype(B, /obj/item/gun/ballistic/rifle)) + if(!B.bolt_locked) + B.rack(H) + return TRUE + //message_admins("Found [target_mag]") + if(target_mag) + if(istype(B, /obj/item/gun/ballistic/rifle/polymer)) + H.put_in_inactive_hand(target_mag) + B.attackby(target_mag, H) + target_mag.forceMove(get_turf(H)) + B.rack(H) //Rack the bolt. + else + //Dump that old mag + H.put_in_inactive_hand(target_mag) + B?.magazine?.forceMove(get_turf(H)) + B.attackby(target_mag, H) + B.rack(H) //Rack the bolt. +// addtimer(CALLBACK(B, PROC_REF(rack), H), 0.5 SECONDS, TIMER_UNIQUE) + else + if(!S) + gun.forceMove(get_turf(H)) + return FALSE + gun.forceMove(S) + +/datum/ai_goal/human/engage_targets/proc/check_ammo(obj/item/gun/ballistic/B = null) + if(!B) + return FALSE + if(!B.magazine) + return FALSE + if(!B.magazine.stored_ammo.len) + return FALSE + return TRUE + + +/datum/ai_goal/human/engage_targets/action(datum/component/knpc/HA) + if(!can_action(HA)) + return + HA.last_node = null //Reset their pathfinding + var/mob/living/carbon/human/ai_boarder/H = HA.parent + var/list/enemies = get_aggressors(HA) + var/obj/item/A = H.get_active_held_item() + var/closest_dist = 1000 + var/just_racked = FALSE + var/mob/living/target = null + for(var/mob/living/L as() in enemies) + var/dist = get_dist(L, H) + if(dist < closest_dist) + closest_dist = dist + target = L + if(!target) + return + var/dist = get_dist(H, target) + //We're holding a gun. See if we can shoot it.... + HA.pathfind_to(target) //Walk up close and YEET SLAM + var/obj/item/gun/G = null + if(istype(A, /obj/item/gun)) + G = A + if(iscarbon(target) || !(G && get_dist(target, H) < 3)) + HA.pathfind_to(target) //Walk up close and YEET SLAM + else + HA.dest = null + var/obj/item/gun/ballistic/B = null + if(istype(A, /obj/item/gun/ballistic)) + B = A + H.a_intent = (prob(65)) ? INTENT_HARM : INTENT_DISARM + if(G) + G.safety = FALSE + if(G && dist > 0 && !istype(target, /mob/living/simple_animal/hostile/asteroid/basilisk/whitesands)) + if(!G.can_shoot() || !G?.chambered.BB) + if(istype(B, /obj/item/gun/ballistic/rifle) && check_ammo(B)) + B.rack(H) + just_racked = TRUE + else + G.safety = FALSE + if(!G.can_shoot() || !G?.chambered.BB) //still cant shoot? + //We need to reload first.... + reload(HA, G) + just_racked = TRUE + //Fire! If they're in a ship, we don't want to scrap them directly. + if(!CheckFriendlyFire(H, target)) + //Okay, we have a line of sight, shoot! + if(B && !(B.semi_auto) && !G.chambered) + //Pump the shotty + G.attack_self(H) + //Let's help them use E-Guns.... + if(istype(G, /obj/item/gun/energy/e_gun)) + var/obj/item/gun/energy/e_gun/E = G + if(prob(20)) + E.unique_action(H) + + if(isobj(target.loc)) + if(!just_racked) + G.afterattack(target.loc, H) + else + if(!just_racked) + G.afterattack(target, H) + +// if(istype(B, /obj/item/gun/ballistic/rifle)) +// B.rack(H) //Rack the bolt. +// addtimer(CALLBACK(G, PROC_REF(rack), H), 0.5 SECONDS, TIMER_UNIQUE) + //Call your friends to help :)) + just_racked = FALSE + if(world.time >= HA.next_backup_call) + call_backup(HA) + + if(dist <= 1) + var/proc_fist = TRUE + var/override_gun_instincts = FALSE + if(istype(target, /mob/living/simple_animal/hostile/asteroid/basilisk/whitesands)) + override_gun_instincts = TRUE + if((!istype(A, /obj/item/gun) || override_gun_instincts) && H.a_intent == INTENT_HARM) + if(!override_gun_instincts) + if(grab_melee(HA) ) + A = H.get_active_held_item() + if(A) + target.attackby(A, H) + A.afterattack(target, H, TRUE) + proc_fist = FALSE + else + target.attackby(A, H) + A.afterattack(target, H, TRUE) + proc_fist = FALSE + + if(proc_fist && !isanimal(target)) + H.dna.species.spec_attack_hand(H, target) + + if(target.incapacitated()) + //I know kung-fu. + + var/obj/item/card/id/their_id = target.get_idcard() + if(their_id && !HA.stealing_id && H.knpc_traits & KNPC_STEAL_ID) + H.visible_message("[H] starts to take [their_id] from [target]!") + HA.stealing_id = TRUE + addtimer(CALLBACK(HA, TYPE_PROC_REF(/datum/component/knpc, steal_id), their_id), 5 SECONDS) + + if(istype(H) && H.knpc_traits & KNPC_IS_MARTIAL_ARTIST) + switch(rand(0, 2)) + //Throw! + if(0) + H.start_pulling(target, supress_message = FALSE) + H.setGrabState(GRAB_AGGRESSIVE) + H.visible_message("[H] judo throws [target]!") +// playsound(get_turf(target), 'nsv13/sound/effects/judo_throw.ogg', 100, TRUE) + target.shake_animation(10) + target.throw_at(get_turf(get_step(H, pick(GLOB.cardinals))), 5, 5) + if(1) + H.do_attack_animation(target, ATTACK_EFFECT_PUNCH) + target.visible_message("[H] grabs [target]'s wrist and wrenches it sideways!", \ + "[H] grabs your wrist and violently wrenches it to the side!") + playsound(get_turf(H), 'sound/weapons/thudswoosh.ogg', 50, 1, -1) + target.emote("scream") + target.dropItemToGround(target.get_active_held_item()) + target.apply_damage(5, BRUTE, pick(BODY_ZONE_L_ARM, BODY_ZONE_R_ARM)) + if(2) + H.do_attack_animation(target, ATTACK_EFFECT_KICK) + target.visible_message("[H] knees [target] in the stomach!", \ + "[H] winds you with a knee in the stomach!") + target.audible_message("[target] gags!") + target.losebreath += 3 + + else + //So they actually execute the curbstomp. + if(dist <= 1) + H.forceMove(get_turf(target)) + H.zone_selected = BODY_ZONE_HEAD + //Curbstomp! + H.MouseDrop(target) + return + if(H.knpc_traits & KNPC_IS_DODGER) + HA.kite(target) + +/datum/ai_goal/human/proc/grab_melee(datum/component/knpc/HA) + var/mob/living/carbon/human/ai_boarder/H = HA.parent + var/obj/item/A = H.get_active_held_item() + if(A && A.force > 10) + return TRUE + + var/obj/item/storage/back_storage = H.back + //Okay first off, is the gun already on our person? + var/list/expanded_contents = H.contents + if(back_storage) + expanded_contents = back_storage.contents + H.contents + var/obj/item/highest_force_score + for(var/obj/item/item_to_check as anything in expanded_contents) + if(!isobj(item_to_check)) + continue + if(item_to_check.force < 5) + continue + if(!highest_force_score) + highest_force_score = item_to_check + continue + if(item_to_check.force > highest_force_score.force) + highest_force_score = item_to_check + continue + if(H.put_in_hands(highest_force_score)) + return TRUE + return FALSE + + +/datum/ai_goal/human/proc/call_backup(datum/component/knpc/HA) + HA.next_backup_call = world.time + 30 SECONDS //So it's not horribly spammy. + var/mob/living/carbon/human/ai_boarder/H = HA.parent + var/obj/item/radio/headset/radio = H.ears + H.do_alert_animation(H) + playsound(H, 'sound/machines/chime.ogg', 50, 1, -1) + //Lets the AIs call for help over comms... This is quite deadly. + var/support_text = (radio) ? "; " : "" + if(H.knpc_traits & KNPC_IS_AREA_SPECIFIC) + var/text = pick(H.call_lines) + text += " [get_area(H)]!" + support_text += text + else + support_text += pick(H.call_lines) + if(!H.shut_up) + H.say(support_text) + + // Call for other intelligent AIs + for(var/datum/component/knpc/HH as() in GLOB.knpcs - HA) + var/mob/living/carbon/human/ai_boarder/other = HH.parent + var/obj/item/radio/headset/other_radio = other.ears + if(other.z != H.z || !other.can_hear() || other.incapacitated()) + continue //Yeah no. Radio is good, but not THAT good....yet + //They both have radios and can hear each other! + if(H.shut_up) + continue + if((radio?.on && other_radio?.on) || get_dist(other, H) <= HA.view_range || H.faction_check_mob(other, TRUE)) + var/thetext = (other_radio) ? "; " : "" + thetext += pick(H.response_lines) + HH.pathfind_to(H) + other.say(thetext) + //Firstly! Call for the simplemobs.. + for(var/mob/living/simple_animal/hostile/M in oview(HA.view_range, HA.parent)) + if(H.faction_check_mob(M, TRUE)) + if(M.AIStatus == AI_OFF) + return + else + M.Goto(HA.parent,M.move_to_delay,M.minimum_distance) + +/datum/ai_goal/human/patrol + name = "Patrol Nodes" + score = AI_SCORE_LOW_PRIORITY //The default task for most AIs is to just patrol + required_ai_flags = null //Set this if you want this task to only be achievable by certain types of ship. + + +/datum/ai_goal/human/patrol/action(datum/component/knpc/HA) + if(!can_action(HA)) + return + var/mob/living/carbon/human/ai_boarder/H = HA.parent + if(HA.last_node && get_dist(HA.last_node, H) > 2) + HA.pathfind_to(HA.last_node) //Restart pathfinding + return FALSE + get_next_patrol_node(HA) + +/obj/effect/landmark/patrol_node + name = "AI patrol node" +// icon = 'nsv13/icons/effects/mapping_helpers.dmi' + icon_state = "patrol_node" + var/id = null + var/next_id = null //id of the node that this one goes to. Alternatively, a list of ids which will all be possible next destinations. + var/previous_id = null //id of the node that precedes this one + var/obj/effect/landmark/patrol_node/previous //-- This isn't actually used anywhere.. - Delta + var/list/next_nodes = list() //List of possible followup nodes set by next_id. If multiple entities exist in the list, one will be chosen at random on every occasion. + invisibility = INVISIBILITY_OBSERVER + +/obj/effect/landmark/patrol_node/whitesands + id = "whitesands" + next_id = "whitesands" + icon_state = "x" + +/obj/effect/landmark/patrol_node/breach + name = "AI hold here node" + id = "hold" + next_id = "hold" + icon_state = "x4" + +/obj/effect/landmark/patrol_node/breach + name = "AI breach here node" + id = "breach1" + next_id = "breach2" + icon_state = "x" + +/obj/effect/landmark/patrol_node/breach/two + name = "AI after breach, go here node" + id = "breach2" + next_id = "breach3" + icon_state = "x2" + +/obj/effect/landmark/patrol_node/breach/three + name = "AI midway, go here node" + id = "breach3" + next_id = "breach4" + icon_state = "x3" + +/obj/effect/landmark/patrol_node/breach/four + name = "AI after sweeping, hold here node" + id = "breach4" + next_id = "breach4" + icon_state = "x4" + +/obj/effect/landmark/patrol_node/Initialize(mapload) + . = ..() + return INITIALIZE_HINT_LATELOAD + +/obj/effect/landmark/patrol_node/LateInitialize() + . = ..() + for(var/obj/effect/landmark/patrol_node/node in GLOB.landmarks_list) + if(!node.id) + continue + if(next_id) + if(islist(next_id)) + var/list/next_id_list = next_id + if(node.id in next_id_list) + next_nodes += node + else + if(node.id == next_id) + next_nodes += node + if(previous_id && node.id == previous_id) + previous = node + if(next_id && !length(next_nodes)) + CRASH("WARNING: Patrol node in [get_area(src)] has no next node(s) despite set id(s).") + if(previous_id && !previous) + CRASH("WARNING: Patrol node in [get_area(src)] has no previous node despite a set id.") + +/datum/ai_goal/human/proc/get_next_patrol_node(datum/component/knpc/HA) + //Okay, we need to pick a starting point. + var/mob/living/carbon/human/ai_boarder/H = HA.parent + if(!HA.last_node) + var/best_dist = 10000 + var/obj/effect/landmark/patrol_node/best + for(var/obj/effect/landmark/patrol_node/node in GLOB.landmarks_list) + var/dist = get_dist(H, node.loc) + if(dist < best_dist && node.z == H.z) + best_dist = dist + best = node + HA.last_node = best + //Start the patrol. + HA.pathfind_to(best) + return + + var/obj/effect/landmark/patrol_node/next_node = pick(HA.last_node.next_nodes) + if(HA.last_node.z != next_node.z) + var/obj/structure/ladder/L = locate(/obj/structure/ladder) in get_turf(HA.last_node) + if(!L) + L = locate(/obj/structure/ladder) in orange(1, get_turf(HA.last_node)) + if(L) + //Use the ladder.... + if(next_node.z > HA.last_node.z) + L.travel(TRUE, H, FALSE, L.up, FALSE) + else + L.travel(FALSE, H, FALSE, L.down, FALSE) + //No Ladder, lets check for stairs + else if(!L && next_node.z > HA.last_node.z) //If going up a Z + var/obj/structure/stairs/S = locate(/obj/structure/stairs) in orange(1, get_turf(HA.last_node)) + if(S) + step_towards(H, S) + step(H, S.dir) + else //If down + var/obj/structure/stairs/S = locate(/obj/structure/stairs) in orange(1, get_step_multiz(get_turf(HA.last_node), DOWN)) + if(S) + step_towards(H, S) + + HA.last_node = next_node + HA.pathfind_to(next_node) + +/datum/ai_goal/human/set_internals + name = "Set Internals" + score = AI_SCORE_CRITICAL //The lads need to be able to breathe. + required_ai_flags = null + +/datum/ai_goal/human/set_internals/check_score(datum/component/knpc/HA) + if(!..()) + return 0 + var/mob/living/carbon/human/H = HA.parent + //We need to breathe.... + if(H.failed_last_breath && world.time >= HA.next_internals_attempt) + HA.next_internals_attempt = world.time + 5 SECONDS + var/obj/item/storage/S = H.back + if(S && locate(/obj/item/tank/internals) in S.contents + H.contents) + return score + return 0 + +/datum/ai_goal/human/set_internals/action(datum/component/knpc/HA) + if(!can_action(HA)) + return + var/mob/living/carbon/human/ai_boarder/C = HA.parent + + if(C.incapacitated()) + return + + if(C.internal) + C.internal = null + else + if(!C.getorganslot(ORGAN_SLOT_BREATHING_TUBE)) + if(!istype(C.wear_mask)) + return 1 + else + var/obj/item/clothing/mask/M = C.wear_mask + if(M.mask_adjusted) // if mask on face but pushed down + M.adjustmask(C) // adjust it back + if(!(M.clothing_flags & ALLOWINTERNALS)) + return + + var/obj/item/I = C.is_holding_item_of_type(/obj/item/tank) + if(I) + C.internal = I + else if(ishuman(C)) + var/mob/living/carbon/human/H = C + H.internal = locate(/obj/item/tank/internals) in H + + //Separate so CO2 jetpacks are a little less cumbersome. + if(!C.internal && istype(C.back, /obj/item/tank)) + C.internal = C.back + +/datum/ai_goal/human/stop_drop_n_roll + name = "Stop drop & roll" + score = AI_SCORE_HIGH_PRIORITY //Putting out fires is important, but not more important than killing the target setting us on fire + required_ai_flags = null + +/datum/ai_goal/human/stop_drop_n_roll/check_score(datum/component/knpc/HA) + if(!..()) + return 0 + var/mob/living/carbon/human/H = HA.parent + //We need to breathe.... + if(H.fire_stacks > 0) + return score + return 0 + +/datum/ai_goal/human/stop_drop_n_roll/action(datum/component/knpc/HA) + var/mob/living/carbon/human/H = HA.parent + if(!can_action(HA)) + return + H.resist() //Stop drop and roll! + +/datum/ai_goal/human/escape_custody + name = "Escape Custody" + score = AI_SCORE_CRITICAL //Not being contained is fairly important + required_ai_flags = null + +/datum/ai_goal/human/escape_custody/check_score(datum/component/knpc/HA) + if(!..()) + return 0 + var/mob/living/carbon/human/H = HA.parent + if(istype(H.loc, /obj/structure/closet)) //Check if we are in a closet + return score + if(H.handcuffed || H.buckled || H.legcuffed) //Or locked up somehow + return score + return 0 + +/datum/ai_goal/human/escape_custody/action(datum/component/knpc/HA) + var/mob/living/carbon/human/H = HA.parent + if(!H.can_resist()) //We use a different check here + return + + if(istype(H.loc, /obj/structure/closet)) + var/obj/structure/closet/C = H.loc + C.open() //Open that closet + if(H.handcuffed || H.buckled || H.legcuffed) + H.resist() //Trigger the universal escape button + +/datum/ai_goal/human/heal_self + name = "Heal Self" + score = AI_SCORE_PRIORITY //Heal ourselves if outside of combat + required_ai_flags = null + +/datum/ai_goal/human/heal_self/check_score(datum/component/knpc/HA) + if(!..()) + return 0 + var/mob/living/carbon/human/H = HA.parent + if(H.health <= (H.maxHealth / 2)) //If half health + if(locate(/obj/item/reagent_containers/hypospray/medipen/survival) in H.contents) //And has survival pen + return score + return 0 + +/datum/ai_goal/human/heal_self/action(datum/component/knpc/HA) + if(!can_action(HA)) + return + var/mob/living/carbon/human/H = HA.parent + var/obj/item/storage/S = H.back + var/list/expanded_contents = H.contents + if(S) //Checking for a backpack + expanded_contents += S.contents + for(var/obj/item/I in H.held_items) //If we are holding anything + I.forceMove(S) //Put it in our pack + else + for(var/obj/item/I in H.held_items) //If we are holding anything + I.forceMove(get_turf(H)) //Drop it on the floor + var/obj/item/reagent_containers/hypospray/medipen/survival/P = locate(/obj/item/reagent_containers/hypospray/medipen/survival) in expanded_contents + if(P) + H.put_in_active_hand(P) //Roll Up Your Sleeve + P.attack(H, H) //Self Vax + P.forceMove(get_turf(H)) //Litter because doing one good thing is enugh for today + +#undef AI_TRAIT_BRAWLER +#undef AI_TRAIT_SUPPORT diff --git a/code/modules/mob/living/simple_animal/hostile/human/survivors.dm b/code/modules/mob/living/simple_animal/hostile/human/survivors.dm index 52863c1a5c64..b4b1eaa41224 100644 --- a/code/modules/mob/living/simple_animal/hostile/human/survivors.dm +++ b/code/modules/mob/living/simple_animal/hostile/human/survivors.dm @@ -26,14 +26,22 @@ /obj/effect/mob_spawn/human/corpse/damaged/whitesands/survivor ) +/mob/living/simple_animal/hostile/asteroid/whitesands/survivor/Initialize(mapload) + . = ..() + var/mob/living/carbon/human/ai_boarder/hermit/survivor/newhermit = new(loc) + newhermit.faction = faction.Copy() + return INITIALIZE_HINT_QDEL + + /mob/living/simple_animal/hostile/human/hermit/survivor/random/Initialize() . = ..() - if(prob(35)) - new /mob/living/simple_animal/hostile/human/hermit/ranged/hunter(loc) - return INITIALIZE_HINT_QDEL if(prob(10)) - new /mob/living/simple_animal/hostile/human/hermit/ranged/gunslinger(loc) - return INITIALIZE_HINT_QDEL + new /mob/living/carbon/human/ai_boarder/hermit/hunter(loc) + else if(prob(3)) + new /mob/living/carbon/human/ai_boarder/hermit/gunslinger(loc) + else if(prob(15)) + new /mob/living/carbon/human/ai_boarder/hermit/survivor(loc) + return INITIALIZE_HINT_QDEL /mob/living/simple_animal/hostile/human/hermit/ranged icon_state = "survivor_hunter" @@ -53,6 +61,12 @@ /obj/effect/mob_spawn/human/corpse/damaged/whitesands/hunter, ) +/mob/living/simple_animal/hostile/human/hermit/ranged/hunter/Initialize(mapload) + . = ..() + var/mob/living/carbon/human/ai_boarder/hermit/hunter/newhermit = new(loc) + newhermit.faction = faction.Copy() + return INITIALIZE_HINT_QDEL + /mob/living/simple_animal/hostile/human/hermit/ranged/gunslinger name = "Hermit Soldier" desc = "The miner's rebellion, though mostly underground, recieved a few good weapon shipments from an off-sector source. You should probably start running." @@ -67,6 +81,12 @@ /obj/effect/mob_spawn/human/corpse/damaged/whitesands/gunslinger, ) +/mob/living/simple_animal/hostile/asteroid/whitesands/ranged/gunslinger/Initialize(mapload) + . = ..() + var/mob/living/carbon/human/ai_boarder/hermit/gunslinger/newhermit = new(loc) + newhermit.faction = faction.Copy() + return INITIALIZE_HINT_QDEL + //survivor corpses /obj/effect/mob_spawn/human/corpse/damaged/whitesands diff --git a/shiptest.dme b/shiptest.dme index 55ad11a800d5..95f1efc6102d 100644 --- a/shiptest.dme +++ b/shiptest.dme @@ -2580,6 +2580,7 @@ #include "code\modules\mob\living\carbon\human\human_movement.dm" #include "code\modules\mob\living\carbon\human\human_say.dm" #include "code\modules\mob\living\carbon\human\inventory.dm" +#include "code\modules\mob\living\carbon\human\knpc.dm" #include "code\modules\mob\living\carbon\human\life.dm" #include "code\modules\mob\living\carbon\human\physiology.dm" #include "code\modules\mob\living\carbon\human\species.dm"