Merge pull request #7905 from MistakeNot4892/beepboop

Added verbs/handling for removing and attaching your own robotic bodyparts.
This commit is contained in:
Atermonera
2021-02-23 14:16:59 -08:00
committed by GitHub
9 changed files with 257 additions and 19 deletions

View File

@@ -0,0 +1,191 @@
/*
A system for easily and quickly removing your own bodyparts, with a view towards
swapping them out for new ones, or just doing it as a party trick to horrify an
audience. Current implementation only supports robolimbs and uses a modular_bodypart
value on the manufacturer datum, but I have tried to keep it generic for future work.
PS. jesus christ this was meant to be a half an hour port
*/
// External organ procs:
// Does this bodypart count as a modular limb, and if so, what kind?
/obj/item/organ/external/proc/get_modular_limb_category()
. = MODULAR_BODYPART_INVALID
if(robotic >= ORGAN_ROBOT && model)
var/datum/robolimb/manufacturer = all_robolimbs[model]
if(!isnull(manufacturer?.modular_bodyparts))
. = manufacturer.modular_bodyparts
// Checks if a limb could theoretically be removed.
// Note that this does not currently bother checking if a child or internal organ is vital.
/obj/item/organ/external/proc/can_remove_modular_limb(var/mob/living/carbon/human/user)
if(vital || cannot_amputate)
return FALSE
var/bodypart_cat = get_modular_limb_category()
if(bodypart_cat == MODULAR_BODYPART_CYBERNETIC)
if(!parent_organ)
return FALSE
var/obj/item/organ/external/parent = user?.get_organ(parent_organ)
if(!parent || parent.get_modular_limb_category(user) < MODULAR_BODYPART_CYBERNETIC)
return FALSE
. = (bodypart_cat != MODULAR_BODYPART_INVALID)
// Note that this proc is checking if the organ can be attached -to-, not attached itself.
/obj/item/organ/external/proc/can_attach_modular_limb_here(var/mob/living/carbon/human/user)
var/list/limb_data = user?.species?.has_limbs[organ_tag]
if(islist(limb_data) && limb_data["has_children"] > 0)
. = (length(children) < limb_data["has_children"])
/obj/item/organ/external/proc/can_be_attached_modular_limb(var/mob/living/carbon/user)
var/bodypart_cat = get_modular_limb_category()
if(bodypart_cat == MODULAR_BODYPART_INVALID)
return FALSE
if(!parent_organ)
return FALSE
var/obj/item/organ/external/parent = user?.get_organ(parent_organ)
if(!parent)
return FALSE
if(!parent.can_attach_modular_limb_here(user))
return FALSE
if(bodypart_cat == MODULAR_BODYPART_CYBERNETIC && parent.get_modular_limb_category(src) < MODULAR_BODYPART_CYBERNETIC)
return FALSE
return TRUE
// Checks if an organ (or the parent of one) is in a fit state for modular limb stuff to happen.
/obj/item/organ/external/proc/check_modular_limb_damage(var/mob/living/carbon/human/user)
. = damage >= min_broken_damage || (status & ORGAN_BROKEN) || is_stump() // can't use is_broken() as the limb has ORGAN_CUT_AWAY
// Human mob procs:
// Checks the organ list for limbs meeting a predicate. Way overengineered for such a limited use
// case but I can see it being expanded in the future if meat limbs or doona limbs use it.
/mob/living/carbon/human/proc/get_modular_limbs(var/return_first_found = FALSE, var/validate_proc)
for(var/bp in organs)
var/obj/item/organ/external/E = bp
if(!validate_proc || call(E, validate_proc)(src) > MODULAR_BODYPART_INVALID)
LAZYADD(., E)
if(return_first_found)
return
// Prune children so we can't remove every individual component of an entire prosthetic arm
// piece by piece. Technically a circular dependency here would remove the limb entirely but
// if there's a parent whose child is also its parent, there's something wrong regardless.
for(var/bp in .)
var/obj/item/organ/external/E = bp
if(length(E.children))
. -= E.children
// Called in robotize(), replaced() and removed() to update our modular limb verbs.
/mob/living/carbon/human/proc/refresh_modular_limb_verbs()
if(length(get_modular_limbs(return_first_found = TRUE, validate_proc = /obj/item/organ/external/proc/can_attach_modular_limb_here)))
verbs |= .proc/attach_limb_verb
else
verbs -= .proc/attach_limb_verb
if(length(get_modular_limbs(return_first_found = TRUE, validate_proc = /obj/item/organ/external/proc/can_remove_modular_limb)))
verbs |= .proc/detach_limb_verb
else
verbs -= .proc/detach_limb_verb
// Proc helper for attachment verb.
/mob/living/carbon/human/proc/check_can_attach_modular_limb(var/obj/item/organ/external/E)
if(world.time < last_special + (2 SECONDS) || get_active_hand() != E)
return FALSE
if(incapacitated() || restrained())
to_chat(src, SPAN_WARNING("You can't do that in your current state!"))
return FALSE
if(QDELETED(E) || !istype(E))
to_chat(src, SPAN_WARNING("You are not holding a compatible limb to attach."))
return FALSE
if(!E.can_be_attached_modular_limb(src))
to_chat(src, SPAN_WARNING("\The [E] cannot be attached to your current body."))
return FALSE
if(E.get_modular_limb_category() <= MODULAR_BODYPART_INVALID)
to_chat(src, SPAN_WARNING("\The [E] cannot be attached by your own hand."))
return FALSE
var/install_to_zone = E.organ_tag
if(!isnull(get_organ(install_to_zone)))
to_chat(src, SPAN_WARNING("There is already a limb attached at that part of your body."))
return FALSE
if(E.check_modular_limb_damage(src))
to_chat(src, SPAN_WARNING("\The [E] is too damaged to be attached."))
return FALSE
var/obj/item/organ/external/parent = E.parent_organ && get_organ(E.parent_organ)
if(!parent)
to_chat(src, SPAN_WARNING("\The [E] needs an existing limb to be attached to."))
return FALSE
if(parent.check_modular_limb_damage(src))
to_chat(src, SPAN_WARNING("Your [parent.name] is too damaged to have anything attached."))
return FALSE
return TRUE
// Proc helper for detachment verb.
/mob/living/carbon/human/proc/check_can_detach_modular_limb(var/obj/item/organ/external/E)
if(world.time < last_special + (2 SECONDS))
return FALSE
if(incapacitated() || restrained())
to_chat(src, SPAN_WARNING("You can't do that in your current state!"))
return FALSE
if(!istype(E) || QDELETED(src) || QDELETED(E) || E.owner != src || E.loc != src)
return FALSE
if(E.check_modular_limb_damage(src))
to_chat(src, SPAN_WARNING("That limb is too damaged to be removed!"))
return FALSE
var/obj/item/organ/external/parent = E.parent_organ && get_organ(E.parent_organ)
if(!parent)
return FALSE
if(parent.check_modular_limb_damage(src))
to_chat(src, SPAN_WARNING("Your [parent.name] is too damaged to detach anything from it."))
return FALSE
return (E in get_modular_limbs(return_first_found = FALSE, validate_proc = /obj/item/organ/external/proc/can_remove_modular_limb))
// Verbs below:
// Add or remove robotic limbs; check refresh_modular_limb_verbs() above.
/mob/living/carbon/human/proc/attach_limb_verb()
set name = "Attach Limb"
set category = "Object"
set desc = "Attach a replacement limb."
set usr = src
var/obj/item/organ/external/E = get_active_hand()
if(!check_can_attach_modular_limb(E))
return FALSE
if(!do_after(src, 2 SECONDS, src))
return FALSE
if(!check_can_attach_modular_limb(E))
return FALSE
last_special = world.time
drop_from_inventory(E)
E.replaced(src)
E.status &= ~ORGAN_CUT_AWAY
var/datum/gender/G = gender_datums[gender]
visible_message(
SPAN_NOTICE("\The [src] attaches \the [E] to [G.his] body!"),
SPAN_NOTICE("You attach \the [E] to your body!"))
regenerate_icons() // Not sure why this isn't called by removed(), but without it we don't update our limb appearance.
return TRUE
/mob/living/carbon/human/proc/detach_limb_verb()
set name = "Remove Limb"
set category = "Object"
set desc = "Detach one of your limbs."
set usr = src
var/list/detachable_limbs = get_modular_limbs(return_first_found = FALSE, validate_proc = /obj/item/organ/external/proc/can_remove_modular_limb)
if(!length(detachable_limbs))
to_chat(src, SPAN_WARNING("You have no detachable limbs."))
return FALSE
var/obj/item/organ/external/E = input(usr, "Which limb do you wish to detach?", "Limb Removal") as null|anything in detachable_limbs
if(!check_can_detach_modular_limb(E))
return FALSE
if(!do_after(src, 2 SECONDS, src))
return FALSE
if(!check_can_detach_modular_limb(E))
return FALSE
last_special = world.time
E.removed(src)
E.dropInto(loc)
put_in_hands(E)
var/datum/gender/G = gender_datums[gender]
visible_message(
SPAN_NOTICE("\The [src] detaches [G.his] [E.name]!"),
SPAN_NOTICE("You detach your [E.name]!"))
return TRUE

View File

@@ -366,7 +366,7 @@
return
var/datum/robolimb/robohead = all_robolimbs[E.model]
if(!robohead.monitor_styles || !robohead.monitor_icon)
to_chat(src,"<span class='warning'>Your head doesn't have a monitor or it doens't support to be changed!</span>")
to_chat(src,"<span class='warning'>Your head doesn't have a monitor, or it doesn't support being changed!</span>")
return
var/list/states
if(!states)

View File

@@ -122,7 +122,7 @@ Variables you may want to make use of are:
has_limbs = list(
BP_TORSO = list("path" = /obj/item/organ/external/chest),
BP_GROIN = list("path" = /obj/item/organ/external/groin),
BP_HEAD = list("path" = /obj/item/organ/external/head),
BP_HEAD = list("path" = /obj/item/organ/external/head),
BP_L_ARM = list("path" = /obj/item/organ/external/arm),
BP_R_ARM = list("path" = /obj/item/organ/external/arm/right),
BP_L_LEG = list("path" = /obj/item/organ/external/leg),

View File

@@ -370,6 +370,9 @@
var/limb_path = organ_data["path"]
var/obj/item/organ/O = new limb_path(H)
organ_data["descriptor"] = O.name
if(O.parent_organ)
organ_data = has_limbs[O.parent_organ]
organ_data["has_children"] = organ_data["has_children"]+1
for(var/organ_tag in has_organ)
var/organ_type = has_organ[organ_tag]