mirror of
https://github.com/Bubberstation/Bubberstation.git
synced 2026-01-28 10:01:58 +00:00
## About The Pull Request Melee attack chain now has a list passed along with it, `attack_modifiers`, which you can stick force modifiers to change the resulting attack This is basically a soft implementation of damage packets until a more definitive pr, but one that only applies to item attack chain, and not unarmed attacks. This change was done to facilitate a baton refactor - batons no longer hack together their own attack chain, and are now integrated straight into the real attack chain. This refactor itself was done because batons don't send any attack signals, which has been annoying in the past (for swing combat). ## Changelog 🆑 Melbert refactor: Batons have been refactored again. Baton stuns now properly count as an attack, when before it was a nothing. Report any oddities, particularly in regards to harmbatonning vs normal batonning. refactor: The method of adjusting item damage mid-attack has been refactored - some affected items include the Nullblade and knives. Report any strange happenings with damage numbers. refactor: A few objects have been moved to the new interaction chain - records consoles, mawed crucible, alien weeds and space vines, hedges, restaurant portals, and some mobs - to name a few. fix: Spears only deal bonus damage against secure lockers, not all closet types (including crates) /🆑
365 lines
12 KiB
Plaintext
365 lines
12 KiB
Plaintext
#define CART_HAS_MINIMUM_REAGENT_VOLUME !(reagents.total_volume < 1)
|
|
|
|
/obj/structure/mop_bucket
|
|
name = "mop bucket"
|
|
desc = "Fill it with water, but don't forget a mop!"
|
|
icon = 'icons/obj/service/janitor.dmi'
|
|
icon_state = "mopbucket"
|
|
density = TRUE
|
|
var/amount_per_transfer_from_this = 5 //shit I dunno, adding this so syringes stop runtime erroring. --NeoFite
|
|
/// The icon used for the water overlay
|
|
var/water_icon = "mopbucket_water"
|
|
|
|
/obj/structure/mop_bucket/Initialize(mapload)
|
|
. = ..()
|
|
create_reagents(100, OPENCONTAINER)
|
|
register_context()
|
|
|
|
/obj/structure/mop_bucket/add_context(atom/source, list/context, obj/item/held_item, mob/living/user)
|
|
. = ..()
|
|
|
|
if(istype(held_item, /obj/item/mop))
|
|
context[SCREENTIP_CONTEXT_RMB] = "Wet [held_item]"
|
|
return CONTEXTUAL_SCREENTIP_SET
|
|
|
|
if(istype(held_item, /obj/item/reagent_containers))
|
|
context[SCREENTIP_CONTEXT_LMB] = "Fill mop bucket"
|
|
return CONTEXTUAL_SCREENTIP_SET
|
|
|
|
return .
|
|
|
|
/obj/structure/mop_bucket/attackby(obj/item/weapon, mob/user, list/modifiers, list/attack_modifiers)
|
|
if(istype(weapon, /obj/item/reagent_containers))
|
|
update_appearance(UPDATE_OVERLAYS)
|
|
return FALSE // skip attack animation when refilling cart
|
|
|
|
return ..()
|
|
|
|
/obj/structure/mop_bucket/attackby_secondary(obj/item/weapon, mob/user, list/modifiers, list/attack_modifiers)
|
|
if(istype(weapon, /obj/item/mop))
|
|
if(weapon.reagents.total_volume >= weapon.reagents.maximum_volume)
|
|
balloon_alert(user, "already soaked!")
|
|
return SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN
|
|
if(!CART_HAS_MINIMUM_REAGENT_VOLUME)
|
|
balloon_alert(user, "empty!")
|
|
return SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN
|
|
reagents.trans_to(weapon, weapon.reagents.maximum_volume, transferred_by = user)
|
|
balloon_alert(user, "doused mop")
|
|
playsound(src, 'sound/effects/slosh.ogg', 25, vary = TRUE)
|
|
|
|
if(istype(weapon, /obj/item/reagent_containers) || istype(weapon, /obj/item/mop))
|
|
update_appearance(UPDATE_OVERLAYS)
|
|
return SECONDARY_ATTACK_CONTINUE_CHAIN // skip attack animations when refilling cart
|
|
|
|
return SECONDARY_ATTACK_CONTINUE_CHAIN
|
|
|
|
/obj/structure/mop_bucket/update_overlays()
|
|
. = ..()
|
|
if(reagents.total_volume > 0)
|
|
. += water_icon
|
|
|
|
/obj/structure/mop_bucket/janitorialcart
|
|
name = "janitorial cart"
|
|
desc = "This is the alpha and omega of sanitation."
|
|
icon_state = "cart"
|
|
water_icon = "cart_water"
|
|
var/obj/item/storage/bag/trash/mybag
|
|
var/obj/item/mop/mymop
|
|
var/obj/item/pushbroom/mybroom
|
|
var/obj/item/reagent_containers/spray/cleaner/myspray
|
|
var/obj/item/lightreplacer/myreplacer
|
|
var/list/obj/item/clothing/suit/caution/held_signs = list()
|
|
var/max_signs = 4
|
|
|
|
/obj/structure/mop_bucket/janitorialcart/Initialize(mapload)
|
|
. = ..()
|
|
reagents.maximum_volume *= 2.5
|
|
GLOB.janitor_devices += src
|
|
|
|
/obj/structure/mop_bucket/janitorialcart/Destroy()
|
|
GLOB.janitor_devices -= src
|
|
QDEL_NULL(myreplacer)
|
|
QDEL_NULL(myspray)
|
|
QDEL_NULL(mybroom)
|
|
QDEL_NULL(mymop)
|
|
QDEL_NULL(mybag)
|
|
QDEL_LIST(held_signs)
|
|
return ..()
|
|
|
|
/obj/structure/mop_bucket/janitorialcart/Entered(atom/movable/arrived, atom/old_loc, list/atom/old_locs)
|
|
if(istype(arrived, /obj/item/storage/bag/trash))
|
|
mybag = arrived
|
|
else if(istype(arrived, /obj/item/mop))
|
|
mymop = arrived
|
|
else if(istype(arrived, /obj/item/pushbroom))
|
|
mybroom = arrived
|
|
else if(istype(arrived, /obj/item/reagent_containers/spray/cleaner))
|
|
myspray = arrived
|
|
else if(istype(arrived, /obj/item/lightreplacer))
|
|
myreplacer = arrived
|
|
else if(istype(arrived, /obj/item/clothing/suit/caution))
|
|
held_signs += arrived
|
|
update_appearance(UPDATE_OVERLAYS)
|
|
return ..()
|
|
|
|
/obj/structure/mop_bucket/janitorialcart/Exited(atom/movable/gone, direction)
|
|
if(gone == mybag)
|
|
mybag = null
|
|
else if(gone == mymop)
|
|
mymop = null
|
|
else if(gone == mybroom)
|
|
mybroom = null
|
|
else if(gone == myspray)
|
|
myspray = null
|
|
else if(gone == myreplacer)
|
|
myreplacer = null
|
|
else if(gone in held_signs)
|
|
held_signs -= gone
|
|
if(!QDELING(src))
|
|
update_appearance(UPDATE_OVERLAYS)
|
|
return ..()
|
|
|
|
/obj/structure/mop_bucket/janitorialcart/examine(mob/user)
|
|
. = ..()
|
|
if(contents.len)
|
|
. += span_bold(span_info("\nIt is carrying:"))
|
|
for(var/thing in sort_names(contents))
|
|
if(thing in held_signs)
|
|
continue //we'll do this after.
|
|
. += "\t[icon2html(thing, user)] \a [thing]"
|
|
if(held_signs.len)
|
|
var/obj/item/clothing/suit/caution/sign_obj = held_signs[1]
|
|
if(held_signs.len > 1)
|
|
. += "\t[icon2html(sign_obj, user)] [convert_integer_to_words(length(held_signs))] [sign_obj.name]\s"
|
|
else
|
|
. += "\t[icon2html(sign_obj, user)] \a [sign_obj]"
|
|
. += span_notice("\n<b>Left-click</b> to [contents.len > 1 ? "search [src]" : "remove [contents[1]]"].")
|
|
if(mybag)
|
|
. += span_notice("<b>Right-click</b> with a <b>[weight_class_to_text(mybag.atom_storage.max_specific_storage)] item</b> to put it in [mybag].")
|
|
if(mymop)
|
|
. += span_notice("<b>Right-click</b> to quickly remove [mymop].")
|
|
if(CART_HAS_MINIMUM_REAGENT_VOLUME)
|
|
. += span_notice("<b>Right-click</b> with a <b>mop</b> to wet it.")
|
|
. += span_info("<b>Crowbar</b> it to dump its mop bucket onto [get_turf(src)].")
|
|
|
|
/obj/structure/mop_bucket/janitorialcart/add_context(atom/source, list/context, obj/item/held_item, mob/living/user)
|
|
. = ..()
|
|
|
|
if(isnull(held_item) && contents.len)
|
|
if(mymop)
|
|
context[SCREENTIP_CONTEXT_RMB] = "Remove [mymop]"
|
|
. = CONTEXTUAL_SCREENTIP_SET
|
|
if(contents.len == 1)
|
|
context[SCREENTIP_CONTEXT_LMB] = "Remove [contents[1]]"
|
|
. = CONTEXTUAL_SCREENTIP_SET
|
|
context[SCREENTIP_CONTEXT_LMB] = "Search cart"
|
|
. = CONTEXTUAL_SCREENTIP_SET
|
|
|
|
if(istype(held_item, /obj/item/mop) && !mymop)
|
|
context[SCREENTIP_CONTEXT_LMB] = "Insert [held_item]"
|
|
. = CONTEXTUAL_SCREENTIP_SET
|
|
if(istype(held_item, /obj/item/pushbroom) && !mybroom)
|
|
context[SCREENTIP_CONTEXT_LMB] = "Insert [held_item]"
|
|
. = CONTEXTUAL_SCREENTIP_SET
|
|
if(istype(held_item, /obj/item/storage/bag/trash) && !mybag)
|
|
context[SCREENTIP_CONTEXT_LMB] = "Insert [held_item]"
|
|
. = CONTEXTUAL_SCREENTIP_SET
|
|
if(istype(held_item, /obj/item/reagent_containers/spray/cleaner) && !myspray)
|
|
context[SCREENTIP_CONTEXT_LMB] = "Insert [held_item]"
|
|
. = CONTEXTUAL_SCREENTIP_SET
|
|
if(istype(held_item, /obj/item/clothing/suit/caution) && held_signs.len < max_signs)
|
|
context[SCREENTIP_CONTEXT_LMB] = "Insert [held_item]"
|
|
. = CONTEXTUAL_SCREENTIP_SET
|
|
if(istype(held_item, /obj/item/lightreplacer) && !myreplacer)
|
|
context[SCREENTIP_CONTEXT_LMB] = "Insert [held_item]"
|
|
. = CONTEXTUAL_SCREENTIP_SET
|
|
if(istype(held_item) && held_item.tool_behaviour == TOOL_CROWBAR && CART_HAS_MINIMUM_REAGENT_VOLUME)
|
|
context[SCREENTIP_CONTEXT_LMB] = "Dump [src]'s mop bucket on [get_turf(src)]"
|
|
. = CONTEXTUAL_SCREENTIP_SET
|
|
if(!isnull(held_item) && mybag?.atom_storage?.max_specific_storage >= held_item.w_class)
|
|
context[SCREENTIP_CONTEXT_RMB] = "Insert [held_item] into [mybag]"
|
|
. = CONTEXTUAL_SCREENTIP_SET
|
|
|
|
return . || NONE
|
|
|
|
/obj/structure/mop_bucket/janitorialcart/attackby(obj/item/attacking_item, mob/user, list/modifiers, list/attack_modifiers)
|
|
if(istype(attacking_item, /obj/item/mop))
|
|
if(mymop)
|
|
balloon_alert(user, "already has \a [mymop]!")
|
|
else if(user.transferItemToLoc(attacking_item, src))
|
|
balloon_alert(user, "placed [attacking_item]")
|
|
return
|
|
|
|
if(istype(attacking_item, /obj/item/pushbroom))
|
|
if(mybroom)
|
|
balloon_alert(user, "already has \a [mybroom]!")
|
|
else if(user.transferItemToLoc(attacking_item, src))
|
|
balloon_alert(user, "placed [attacking_item]")
|
|
return
|
|
|
|
if(istype(attacking_item, /obj/item/storage/bag/trash))
|
|
if(mybag)
|
|
balloon_alert(user, "already has \a [mybag]!")
|
|
return
|
|
|
|
var/obj/item/storage/bag/trash/insert = attacking_item
|
|
if(!insert.insertable)
|
|
balloon_alert(user, "cannot be inserted!")
|
|
return
|
|
|
|
if(user.transferItemToLoc(attacking_item, src))
|
|
balloon_alert(user, "attached [attacking_item]")
|
|
return
|
|
|
|
if(istype(attacking_item, /obj/item/reagent_containers/spray/cleaner))
|
|
if(myspray)
|
|
balloon_alert(user, "already has \a [myspray]!")
|
|
else if(user.transferItemToLoc(attacking_item, src))
|
|
balloon_alert(user, "placed [attacking_item]")
|
|
return
|
|
|
|
if(istype(attacking_item, /obj/item/lightreplacer))
|
|
if(myreplacer)
|
|
balloon_alert(user, "already has \a [myreplacer]!")
|
|
else if(user.transferItemToLoc(attacking_item, src))
|
|
balloon_alert(user, "placed [attacking_item]")
|
|
return
|
|
|
|
else if(istype(attacking_item, /obj/item/clothing/suit/caution))
|
|
if(held_signs.len >= max_signs)
|
|
balloon_alert(user, "sign rack is full!")
|
|
else if(user.transferItemToLoc(attacking_item, src))
|
|
balloon_alert(user, "placed [attacking_item]")
|
|
return
|
|
|
|
return ..()
|
|
|
|
/obj/structure/mop_bucket/janitorialcart/crowbar_act(mob/living/user, obj/item/tool)
|
|
if(!CART_HAS_MINIMUM_REAGENT_VOLUME)
|
|
balloon_alert(user, "mop bucket is empty!")
|
|
return ITEM_INTERACT_SUCCESS
|
|
user.balloon_alert_to_viewers("starts dumping [src]...", "started dumping [src]...")
|
|
user.visible_message(span_notice("[user] begins to dumping the contents of [src]'s mop bucket."), span_notice("You begin to dump the contents of [src]'s mop bucket..."))
|
|
if(tool.use_tool(src, user, 5 SECONDS, volume = 50))
|
|
balloon_alert(user, "dumped [src]")
|
|
to_chat(user, span_notice("You dumped the contents of [src]'s mop bucket onto the floor."))
|
|
reagents.expose(loc)
|
|
reagents.clear_reagents()
|
|
update_appearance(UPDATE_OVERLAYS)
|
|
return ITEM_INTERACT_SUCCESS
|
|
|
|
/obj/structure/mop_bucket/janitorialcart/attackby_secondary(obj/item/weapon, mob/user, list/modifiers, list/attack_modifiers)
|
|
. = ..()
|
|
if(. == SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN)
|
|
return SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN
|
|
|
|
if(istype(weapon, /obj/item/reagent_containers))
|
|
update_appearance(UPDATE_OVERLAYS)
|
|
return SECONDARY_ATTACK_CONTINUE_CHAIN //so we can empty the cart via our afterattack without trying to put the item in the bag
|
|
|
|
if(mybag?.attackby(weapon, user))
|
|
return SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN
|
|
|
|
return SECONDARY_ATTACK_CONTINUE_CHAIN
|
|
|
|
/obj/structure/mop_bucket/janitorialcart/attack_hand(mob/user, list/modifiers)
|
|
. = ..()
|
|
if(.)
|
|
return
|
|
|
|
var/list/items = list()
|
|
if(mybag)
|
|
items += list("Trash bag" = image(icon = mybag.icon, icon_state = mybag.icon_state))
|
|
if(mymop)
|
|
items += list("Mop" = image(icon = mymop.icon, icon_state = mymop.icon_state))
|
|
if(mybroom)
|
|
items += list("Broom" = image(icon = mybroom.icon, icon_state = mybroom.icon_state))
|
|
if(myspray)
|
|
items += list("Spray bottle" = image(icon = myspray.icon, icon_state = myspray.icon_state))
|
|
if(myreplacer)
|
|
items += list("Light replacer" = image(icon = myreplacer.icon, icon_state = myreplacer.icon_state))
|
|
if(held_signs.len)
|
|
var/obj/item/clothing/suit/caution/sign = held_signs[1]
|
|
items += list("Sign" = image(icon = sign.icon, icon_state = sign.icon_state))
|
|
|
|
if(!length(items))
|
|
return
|
|
|
|
var/pick = items[1]
|
|
if(length(items) > 1)
|
|
items = sort_list(items)
|
|
pick = show_radial_menu(user, src, items, custom_check = CALLBACK(src, PROC_REF(check_menu), user), radius = 38, require_near = TRUE)
|
|
|
|
if(!pick)
|
|
return
|
|
switch(pick)
|
|
if("Trash bag")
|
|
if(!mybag)
|
|
return
|
|
balloon_alert(user, "detached [mybag]")
|
|
user.put_in_hands(mybag)
|
|
if("Mop")
|
|
if(!mymop)
|
|
return
|
|
balloon_alert(user, "removed [mymop]")
|
|
user.put_in_hands(mymop)
|
|
if("Broom")
|
|
if(!mybroom)
|
|
return
|
|
balloon_alert(user, "removed [mybroom]")
|
|
user.put_in_hands(mybroom)
|
|
if("Spray bottle")
|
|
if(!myspray)
|
|
return
|
|
balloon_alert(user, "removed [myspray]")
|
|
user.put_in_hands(myspray)
|
|
if("Light replacer")
|
|
if(!myreplacer)
|
|
return
|
|
balloon_alert(user, "removed [myreplacer]")
|
|
user.put_in_hands(myreplacer)
|
|
if("Sign")
|
|
if(!held_signs.len)
|
|
return
|
|
var/obj/item/clothing/suit/caution/removed_sign = held_signs[1]
|
|
if(length(held_signs) > 1)
|
|
balloon_alert(user, "removed \a [removed_sign]")
|
|
else
|
|
balloon_alert(user, "removed [removed_sign]")
|
|
user.put_in_hands(removed_sign)
|
|
else
|
|
return
|
|
|
|
/obj/structure/mop_bucket/janitorialcart/attack_hand_secondary(mob/user, list/modifiers)
|
|
if(!mymop)
|
|
return SECONDARY_ATTACK_CONTINUE_CHAIN
|
|
balloon_alert(user, "removed [mymop]")
|
|
user.put_in_hands(mymop)
|
|
return SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN
|
|
|
|
/**
|
|
* check_menu: Checks if we are allowed to interact with a radial menu
|
|
*
|
|
* Arguments:
|
|
* * user The mob interacting with a menu
|
|
*/
|
|
/obj/structure/mop_bucket/janitorialcart/proc/check_menu(mob/living/user)
|
|
return istype(user) && !user.incapacitated
|
|
|
|
/obj/structure/mop_bucket/janitorialcart/update_overlays()
|
|
. = ..()
|
|
if(mybag)
|
|
. += istype(mybag, /obj/item/storage/bag/trash/bluespace) ? "cart_bluespace_garbage" : "cart_garbage"
|
|
if(mymop)
|
|
. += "cart_mop"
|
|
if(mybroom)
|
|
. += "cart_broom"
|
|
if(myspray)
|
|
. += "cart_spray"
|
|
if(myreplacer)
|
|
. += "cart_replacer"
|
|
if(held_signs.len)
|
|
. += "cart_sign[min(held_signs.len, 4)]"
|
|
|
|
#undef CART_HAS_MINIMUM_REAGENT_VOLUME
|