mirror of
https://github.com/Bubberstation/Bubberstation.git
synced 2026-01-27 17:41:50 +00:00
* Implements JPS (Jump Point Search) Pathfinding (#56780) So a month or so ago I wanted to make it so dogs in my dog AI PR could path through doors if they had access, and was told I'd need to improve our pathfinding efficiency if I wanted to use full pathfinding for them. Thus, enter JPS, a pathfinding algorithm that allows for massive timesavings in systems with uniform cost grids like ours. This code is still fairly rough and needs polishing, but it's fully functional and already shows massive savings over traditional A*! I plan for this to replace A* as our default pathing method, but I'll leave the A* code in place in case someone ever needs it for whatever reason, like if a specific case needs variable cost pathing. Note that this allows for diagonal pathing instead of the cardinal pathing our A* uses right now, and the current version of the code costs the same to move diagonally as it does to move laterally, which may change later. There's also a lot of dummy/test code in right now in general, but you should still be able to test it out for yourself by spawning a bot like a medibot and using your PDA to summon it. Preliminary Profile Results A preliminary profile is available here. Using one medibot by itself on Metastation, I generated a list of 500 random blob spawn points around the station, gave the medibot all access, then let each algorithm tackle the list. The old A* algorithm took a total of 86 seconds to complete the list and processed 978065 nodes, while JPS took a total of 46 seconds and processed only 100062 nodes, for a 47% decrease in total time and an almost 90% decrease in nodes processed! Why It's Good For The Game Significantly cheaper pathing, which will very much come in handy for the AI datums I'm looking to dig into, what's not to like? * Implements JPS (Jump Point Search) Pathfinding Co-authored-by: Ryll Ryll <3589655+Ryll-Ryll@users.noreply.github.com>
443 lines
16 KiB
Plaintext
443 lines
16 KiB
Plaintext
/obj/structure/girder
|
|
name = "girder"
|
|
icon_state = "girder"
|
|
desc = "A large structural assembly made out of metal; It requires a layer of iron before it can be considered a wall."
|
|
anchored = TRUE
|
|
density = TRUE
|
|
var/state = GIRDER_NORMAL
|
|
var/girderpasschance = 20 // percentage chance that a projectile passes through the girder.
|
|
var/can_displace = TRUE //If the girder can be moved around by wrenching it
|
|
var/next_beep = 0 //Prevents spamming of the construction sound
|
|
max_integrity = 200
|
|
flags_1 = RAD_PROTECT_CONTENTS_1 | RAD_NO_CONTAMINATE_1
|
|
rad_insulation = RAD_VERY_LIGHT_INSULATION
|
|
|
|
/obj/structure/girder/examine(mob/user)
|
|
. = ..()
|
|
switch(state)
|
|
if(GIRDER_REINF)
|
|
. += "<span class='notice'>The support struts are <b>screwed</b> in place.</span>"
|
|
if(GIRDER_REINF_STRUTS)
|
|
. += "<span class='notice'>The support struts are <i>unscrewed</i> and the inner <b>grille</b> is intact.</span>"
|
|
if(GIRDER_NORMAL)
|
|
if(can_displace)
|
|
. += "<span class='notice'>The bolts are <b>wrenched</b> in place.</span>"
|
|
if(GIRDER_DISPLACED)
|
|
. += "<span class='notice'>The bolts are <i>loosened</i>, but the <b>screws</b> are holding [src] together.</span>"
|
|
if(GIRDER_DISASSEMBLED)
|
|
. += "<span class='notice'>[src] is disassembled! You probably shouldn't be able to see this examine message.</span>"
|
|
|
|
/obj/structure/girder/attackby(obj/item/W, mob/user, params)
|
|
var/platingmodifier = 1
|
|
if(HAS_TRAIT(user, TRAIT_QUICK_BUILD))
|
|
platingmodifier = 0.7
|
|
if(next_beep <= world.time)
|
|
next_beep = world.time + 10
|
|
playsound(src, 'sound/machines/clockcult/integration_cog_install.ogg', 50, TRUE)
|
|
add_fingerprint(user)
|
|
|
|
if(istype(W, /obj/item/gun/energy/plasmacutter))
|
|
to_chat(user, "<span class='notice'>You start slicing apart the girder...</span>")
|
|
if(W.use_tool(src, user, 40, volume=100))
|
|
to_chat(user, "<span class='notice'>You slice apart the girder.</span>")
|
|
var/obj/item/stack/sheet/iron/M = new (loc, 2)
|
|
M.add_fingerprint(user)
|
|
qdel(src)
|
|
|
|
else if(istype(W, /obj/item/stack))
|
|
if(iswallturf(loc))
|
|
to_chat(user, "<span class='warning'>There is already a wall present!</span>")
|
|
return
|
|
if(!isfloorturf(src.loc))
|
|
to_chat(user, "<span class='warning'>A floor must be present to build a false wall!</span>")
|
|
return
|
|
if (locate(/obj/structure/falsewall) in src.loc.contents)
|
|
to_chat(user, "<span class='warning'>There is already a false wall present!</span>")
|
|
return
|
|
|
|
if(istype(W, /obj/item/stack/rods))
|
|
var/obj/item/stack/rods/S = W
|
|
if(state == GIRDER_DISPLACED)
|
|
if(S.get_amount() < 2)
|
|
to_chat(user, "<span class='warning'>You need at least two rods to create a false wall!</span>")
|
|
return
|
|
to_chat(user, "<span class='notice'>You start building a reinforced false wall...</span>")
|
|
if(do_after(user, 20, target = src))
|
|
if(S.get_amount() < 2)
|
|
return
|
|
S.use(2)
|
|
to_chat(user, "<span class='notice'>You create a false wall. Push on it to open or close the passage.</span>")
|
|
var/obj/structure/falsewall/iron/FW = new (loc)
|
|
transfer_fingerprints_to(FW)
|
|
qdel(src)
|
|
else
|
|
if(S.get_amount() < 5)
|
|
to_chat(user, "<span class='warning'>You need at least five rods to add plating!</span>")
|
|
return
|
|
to_chat(user, "<span class='notice'>You start adding plating...</span>")
|
|
if(do_after(user, 40, target = src))
|
|
if(S.get_amount() < 5)
|
|
return
|
|
S.use(5)
|
|
to_chat(user, "<span class='notice'>You add the plating.</span>")
|
|
var/turf/T = get_turf(src)
|
|
T.PlaceOnTop(/turf/closed/wall/mineral/iron)
|
|
transfer_fingerprints_to(T)
|
|
qdel(src)
|
|
return
|
|
|
|
if(!istype(W, /obj/item/stack/sheet))
|
|
return
|
|
|
|
var/obj/item/stack/sheet/S = W
|
|
if(istype(S, /obj/item/stack/sheet/iron))
|
|
if(state == GIRDER_DISPLACED)
|
|
if(S.get_amount() < 2)
|
|
to_chat(user, "<span class='warning'>You need two sheets of iron to create a false wall!</span>")
|
|
return
|
|
to_chat(user, "<span class='notice'>You start building a false wall...</span>")
|
|
if(do_after(user, 20*platingmodifier, target = src))
|
|
if(S.get_amount() < 2)
|
|
return
|
|
S.use(2)
|
|
to_chat(user, "<span class='notice'>You create a false wall. Push on it to open or close the passage.</span>")
|
|
var/obj/structure/falsewall/F = new (loc)
|
|
transfer_fingerprints_to(F)
|
|
qdel(src)
|
|
else if(state == GIRDER_REINF)
|
|
to_chat(user, "<span class='warning'>You can't finish a reinforced girder with regular iron. You need a plasteel sheet for that.</span>")
|
|
return
|
|
else
|
|
if(S.get_amount() < 2)
|
|
to_chat(user, "<span class='warning'>You need two sheets of iron to finish a wall!</span>")
|
|
return
|
|
to_chat(user, "<span class='notice'>You start adding plating...</span>")
|
|
if (do_after(user, 40*platingmodifier, target = src))
|
|
if(S.get_amount() < 2)
|
|
return
|
|
S.use(2)
|
|
to_chat(user, "<span class='notice'>You add the plating.</span>")
|
|
var/turf/T = get_turf(src)
|
|
T.PlaceOnTop(/turf/closed/wall)
|
|
transfer_fingerprints_to(T)
|
|
qdel(src)
|
|
return
|
|
|
|
if(istype(S, /obj/item/stack/sheet/plasteel))
|
|
if(state == GIRDER_DISPLACED)
|
|
if(S.get_amount() < 2)
|
|
to_chat(user, "<span class='warning'>You need at least two sheets to create a false wall!</span>")
|
|
return
|
|
to_chat(user, "<span class='notice'>You start building a reinforced false wall...</span>")
|
|
if(do_after(user, 20, target = src))
|
|
if(S.get_amount() < 2)
|
|
return
|
|
S.use(2)
|
|
to_chat(user, "<span class='notice'>You create a reinforced false wall. Push on it to open or close the passage.</span>")
|
|
var/obj/structure/falsewall/reinforced/FW = new (loc)
|
|
transfer_fingerprints_to(FW)
|
|
qdel(src)
|
|
else if(state == GIRDER_REINF)
|
|
if(S.get_amount() < 1)
|
|
return
|
|
to_chat(user, "<span class='notice'>You start finalizing the reinforced wall...</span>")
|
|
if(do_after(user, 50*platingmodifier, target = src))
|
|
if(S.get_amount() < 1)
|
|
return
|
|
S.use(1)
|
|
to_chat(user, "<span class='notice'>You fully reinforce the wall.</span>")
|
|
var/turf/T = get_turf(src)
|
|
T.PlaceOnTop(/turf/closed/wall/r_wall)
|
|
transfer_fingerprints_to(T)
|
|
qdel(src)
|
|
return
|
|
else
|
|
if(S.get_amount() < 1)
|
|
return
|
|
to_chat(user, "<span class='notice'>You start reinforcing the girder...</span>")
|
|
if(do_after(user, 60*platingmodifier, target = src))
|
|
if(S.get_amount() < 1)
|
|
return
|
|
S.use(1)
|
|
to_chat(user, "<span class='notice'>You reinforce the girder.</span>")
|
|
var/obj/structure/girder/reinforced/R = new (loc)
|
|
transfer_fingerprints_to(R)
|
|
qdel(src)
|
|
return
|
|
|
|
if(!S.has_unique_girder)
|
|
var/M = S.sheettype
|
|
if(state == GIRDER_DISPLACED)
|
|
var/falsewall_type = text2path("/obj/structure/falsewall/[M]")
|
|
if(!falsewall_type)
|
|
to_chat(user, "<span class='warning'>You can't seem to figure out how to make a false wall with [S]!</span>")
|
|
return
|
|
if(S.get_amount() < 2)
|
|
to_chat(user, "<span class='warning'>You need at least two sheets to create a false wall!</span>")
|
|
return
|
|
if(do_after(user, 20, target = src))
|
|
if(S.get_amount() < 2)
|
|
return
|
|
S.use(2)
|
|
to_chat(user, "<span class='notice'>You create a false wall. Push on it to open or close the passage.</span>")
|
|
var/obj/structure/falsewall/FW = new falsewall_type (loc)
|
|
transfer_fingerprints_to(FW)
|
|
qdel(src)
|
|
else
|
|
if(S.get_amount() < 2)
|
|
to_chat(user, "<span class='warning'>You need at least two sheets to add plating!</span>")
|
|
return
|
|
to_chat(user, "<span class='notice'>You start adding plating...</span>")
|
|
if (do_after(user, 40, target = src))
|
|
if(S.get_amount() < 2)
|
|
return
|
|
S.use(2)
|
|
to_chat(user, "<span class='notice'>You add the plating.</span>")
|
|
var/turf/T = get_turf(src)
|
|
if(S.walltype)
|
|
T.PlaceOnTop(S.walltype)
|
|
else
|
|
var/turf/newturf = T.PlaceOnTop(/turf/closed/wall/material)
|
|
var/list/material_list = list()
|
|
if(S.material_type)
|
|
material_list[GET_MATERIAL_REF(S.material_type)] = MINERAL_MATERIAL_AMOUNT * 2
|
|
if(material_list)
|
|
newturf.set_custom_materials(material_list)
|
|
|
|
transfer_fingerprints_to(T)
|
|
qdel(src)
|
|
return
|
|
|
|
add_hiddenprint(user)
|
|
|
|
else if(istype(W, /obj/item/pipe))
|
|
var/obj/item/pipe/P = W
|
|
if (P.pipe_type in list(0, 1, 5)) //simple pipes, simple bends, and simple manifolds.
|
|
if(!user.transferItemToLoc(P, drop_location()))
|
|
return
|
|
to_chat(user, "<span class='notice'>You fit the pipe into \the [src].</span>")
|
|
else
|
|
return ..()
|
|
|
|
// Screwdriver behavior for girders
|
|
/obj/structure/girder/screwdriver_act(mob/user, obj/item/tool)
|
|
if(..())
|
|
return TRUE
|
|
|
|
. = FALSE
|
|
if(state == GIRDER_DISPLACED)
|
|
user.visible_message("<span class='warning'>[user] disassembles the girder.</span>",
|
|
"<span class='notice'>You start to disassemble the girder...</span>",
|
|
"<span class='hear'>You hear clanking and banging noises.</span>")
|
|
if(tool.use_tool(src, user, 40, volume=100))
|
|
if(state != GIRDER_DISPLACED)
|
|
return
|
|
state = GIRDER_DISASSEMBLED
|
|
to_chat(user, "<span class='notice'>You disassemble the girder.</span>")
|
|
var/obj/item/stack/sheet/iron/M = new (loc, 2)
|
|
M.add_fingerprint(user)
|
|
qdel(src)
|
|
return TRUE
|
|
|
|
else if(state == GIRDER_REINF)
|
|
to_chat(user, "<span class='notice'>You start unsecuring support struts...</span>")
|
|
if(tool.use_tool(src, user, 40, volume=100))
|
|
if(state != GIRDER_REINF)
|
|
return
|
|
to_chat(user, "<span class='notice'>You unsecure the support struts.</span>")
|
|
state = GIRDER_REINF_STRUTS
|
|
return TRUE
|
|
|
|
else if(state == GIRDER_REINF_STRUTS)
|
|
to_chat(user, "<span class='notice'>You start securing support struts...</span>")
|
|
if(tool.use_tool(src, user, 40, volume=100))
|
|
if(state != GIRDER_REINF_STRUTS)
|
|
return
|
|
to_chat(user, "<span class='notice'>You secure the support struts.</span>")
|
|
state = GIRDER_REINF
|
|
return TRUE
|
|
|
|
// Wirecutter behavior for girders
|
|
/obj/structure/girder/wirecutter_act(mob/user, obj/item/tool)
|
|
. = ..()
|
|
if(state == GIRDER_REINF_STRUTS)
|
|
to_chat(user, "<span class='notice'>You start removing the inner grille...</span>")
|
|
if(tool.use_tool(src, user, 40, volume=100))
|
|
to_chat(user, "<span class='notice'>You remove the inner grille.</span>")
|
|
new /obj/item/stack/sheet/plasteel(get_turf(src))
|
|
var/obj/structure/girder/G = new (loc)
|
|
transfer_fingerprints_to(G)
|
|
qdel(src)
|
|
return TRUE
|
|
|
|
/obj/structure/girder/wrench_act(mob/user, obj/item/tool)
|
|
. = ..()
|
|
if(state == GIRDER_DISPLACED)
|
|
if(!isfloorturf(loc))
|
|
to_chat(user, "<span class='warning'>A floor must be present to secure the girder!</span>")
|
|
|
|
to_chat(user, "<span class='notice'>You start securing the girder...</span>")
|
|
if(tool.use_tool(src, user, 40, volume=100))
|
|
to_chat(user, "<span class='notice'>You secure the girder.</span>")
|
|
var/obj/structure/girder/G = new (loc)
|
|
transfer_fingerprints_to(G)
|
|
qdel(src)
|
|
return TRUE
|
|
else if(state == GIRDER_NORMAL && can_displace)
|
|
to_chat(user, "<span class='notice'>You start unsecuring the girder...</span>")
|
|
if(tool.use_tool(src, user, 40, volume=100))
|
|
to_chat(user, "<span class='notice'>You unsecure the girder.</span>")
|
|
var/obj/structure/girder/displaced/D = new (loc)
|
|
transfer_fingerprints_to(D)
|
|
qdel(src)
|
|
return TRUE
|
|
|
|
/obj/structure/girder/CanAllowThrough(atom/movable/mover, turf/target)
|
|
. = ..()
|
|
if((mover.pass_flags & PASSGRILLE) || istype(mover, /obj/projectile))
|
|
return prob(girderpasschance)
|
|
|
|
/obj/structure/girder/CanAStarPass(obj/item/card/id/ID, to_dir, atom/movable/caller)
|
|
. = !density
|
|
if(istype(caller))
|
|
. = . || (caller.pass_flags & PASSGRILLE)
|
|
|
|
/obj/structure/girder/deconstruct(disassembled = TRUE)
|
|
if(!(flags_1 & NODECONSTRUCT_1))
|
|
var/remains = pick(/obj/item/stack/rods, /obj/item/stack/sheet/iron)
|
|
new remains(loc)
|
|
qdel(src)
|
|
|
|
/obj/structure/girder/narsie_act()
|
|
new /obj/structure/girder/cult(loc)
|
|
qdel(src)
|
|
|
|
/obj/structure/girder/displaced
|
|
name = "displaced girder"
|
|
icon_state = "displaced"
|
|
anchored = FALSE
|
|
state = GIRDER_DISPLACED
|
|
girderpasschance = 25
|
|
max_integrity = 120
|
|
|
|
/obj/structure/girder/reinforced
|
|
name = "reinforced girder"
|
|
icon_state = "reinforced"
|
|
state = GIRDER_REINF
|
|
girderpasschance = 0
|
|
max_integrity = 350
|
|
|
|
|
|
|
|
//////////////////////////////////////////// cult girder //////////////////////////////////////////////
|
|
|
|
/obj/structure/girder/cult
|
|
name = "runed girder"
|
|
desc = "Framework made of a strange and shockingly cold metal. It doesn't seem to have any bolts."
|
|
icon = 'icons/obj/cult.dmi'
|
|
icon_state= "cultgirder"
|
|
can_displace = FALSE
|
|
|
|
/obj/structure/girder/cult/attackby(obj/item/W, mob/user, params)
|
|
add_fingerprint(user)
|
|
if(istype(W, /obj/item/melee/cultblade/dagger) && iscultist(user)) //Cultists can demolish cult girders instantly with their tomes
|
|
user.visible_message("<span class='warning'>[user] strikes [src] with [W]!</span>", "<span class='notice'>You demolish [src].</span>")
|
|
new /obj/item/stack/sheet/runed_metal(drop_location(), 1)
|
|
qdel(src)
|
|
|
|
else if(W.tool_behaviour == TOOL_WELDER)
|
|
if(!W.tool_start_check(user, amount=0))
|
|
return
|
|
|
|
to_chat(user, "<span class='notice'>You start slicing apart the girder...</span>")
|
|
if(W.use_tool(src, user, 40, volume=50))
|
|
to_chat(user, "<span class='notice'>You slice apart the girder.</span>")
|
|
var/obj/item/stack/sheet/runed_metal/R = new(drop_location(), 1)
|
|
transfer_fingerprints_to(R)
|
|
qdel(src)
|
|
|
|
else if(istype(W, /obj/item/stack/sheet/runed_metal))
|
|
var/obj/item/stack/sheet/runed_metal/R = W
|
|
if(R.get_amount() < 1)
|
|
to_chat(user, "<span class='warning'>You need at least one sheet of runed metal to construct a runed wall!</span>")
|
|
return
|
|
user.visible_message("<span class='notice'>[user] begins laying runed metal on [src]...</span>", "<span class='notice'>You begin constructing a runed wall...</span>")
|
|
if(do_after(user, 50, target = src))
|
|
if(R.get_amount() < 1)
|
|
return
|
|
user.visible_message("<span class='notice'>[user] plates [src] with runed metal.</span>", "<span class='notice'>You construct a runed wall.</span>")
|
|
R.use(1)
|
|
var/turf/T = get_turf(src)
|
|
T.PlaceOnTop(/turf/closed/wall/mineral/cult)
|
|
qdel(src)
|
|
|
|
else
|
|
return ..()
|
|
|
|
/obj/structure/girder/cult/narsie_act()
|
|
return
|
|
|
|
/obj/structure/girder/cult/deconstruct(disassembled = TRUE)
|
|
if(!(flags_1 & NODECONSTRUCT_1))
|
|
new /obj/item/stack/sheet/runed_metal(drop_location(), 1)
|
|
qdel(src)
|
|
|
|
/obj/structure/girder/rcd_vals(mob/user, obj/item/construction/rcd/the_rcd)
|
|
switch(the_rcd.mode)
|
|
if(RCD_FLOORWALL)
|
|
return list("mode" = RCD_FLOORWALL, "delay" = 20, "cost" = 8)
|
|
if(RCD_DECONSTRUCT)
|
|
return list("mode" = RCD_DECONSTRUCT, "delay" = 20, "cost" = 13)
|
|
return FALSE
|
|
|
|
/obj/structure/girder/rcd_act(mob/user, obj/item/construction/rcd/the_rcd, passed_mode)
|
|
var/turf/T = get_turf(src)
|
|
switch(passed_mode)
|
|
if(RCD_FLOORWALL)
|
|
to_chat(user, "<span class='notice'>You finish a wall.</span>")
|
|
T.PlaceOnTop(/turf/closed/wall)
|
|
qdel(src)
|
|
return TRUE
|
|
if(RCD_DECONSTRUCT)
|
|
to_chat(user, "<span class='notice'>You deconstruct the girder.</span>")
|
|
qdel(src)
|
|
return TRUE
|
|
return FALSE
|
|
|
|
/obj/structure/girder/bronze
|
|
name = "wall gear"
|
|
desc = "A girder made out of sturdy bronze, made to resemble a gear."
|
|
icon = 'icons/obj/clockwork_objects.dmi'
|
|
icon_state = "wall_gear"
|
|
can_displace = FALSE
|
|
|
|
/obj/structure/girder/bronze/attackby(obj/item/W, mob/living/user, params)
|
|
add_fingerprint(user)
|
|
if(W.tool_behaviour == TOOL_WELDER)
|
|
if(!W.tool_start_check(user, amount = 0))
|
|
return
|
|
to_chat(user, "<span class='notice'>You start slicing apart [src]...</span>")
|
|
if(W.use_tool(src, user, 40, volume=50))
|
|
to_chat(user, "<span class='notice'>You slice apart [src].</span>")
|
|
var/obj/item/stack/tile/bronze/B = new(drop_location(), 2)
|
|
transfer_fingerprints_to(B)
|
|
qdel(src)
|
|
|
|
else if(istype(W, /obj/item/stack/sheet/bronze))
|
|
var/obj/item/stack/sheet/bronze/B = W
|
|
if(B.get_amount() < 2)
|
|
to_chat(user, "<span class='warning'>You need at least two bronze sheets to build a bronze wall!</span>")
|
|
return
|
|
user.visible_message("<span class='notice'>[user] begins plating [src] with bronze...</span>", "<span class='notice'>You begin constructing a bronze wall...</span>")
|
|
if(do_after(user, 50, target = src))
|
|
if(B.get_amount() < 2)
|
|
return
|
|
user.visible_message("<span class='notice'>[user] plates [src] with bronze!</span>", "<span class='notice'>You construct a bronze wall.</span>")
|
|
B.use(2)
|
|
var/turf/T = get_turf(src)
|
|
T.PlaceOnTop(/turf/closed/wall/mineral/bronze)
|
|
qdel(src)
|
|
|
|
else
|
|
return ..()
|