Skip to content

Commit

Permalink
Merge branch 'master' into blood_emissive
Browse files Browse the repository at this point in the history
  • Loading branch information
MrMelbert committed Nov 24, 2024
2 parents 8fb68b3 + c235908 commit ea9a09c
Show file tree
Hide file tree
Showing 136 changed files with 2,934 additions and 1,285 deletions.
1 change: 0 additions & 1 deletion code/__DEFINES/alerts.dm
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,6 @@
#define ALERT_BUCKLED "buckled"
#define ALERT_HANDCUFFED "handcuffed"
#define ALERT_LEGCUFFED "legcuffed"
#define ALERT_WOUNDED "wound"
#define ALERT_IRRADIATED "irradiated"
#define ALERT_EMBEDDED_OBJECT "embeddedobject"
#define ALERT_SHOES_KNOT "shoealert"
Expand Down
4 changes: 2 additions & 2 deletions code/__DEFINES/bodyparts.dm
Original file line number Diff line number Diff line change
Expand Up @@ -19,10 +19,10 @@
/// A mutiplication of the burn and brute damage that the limb's stored damage contributes to its attached mob's overall wellbeing.
/// For instance, if a limb has 50 damage, and has a coefficient of 50%, the human is considered to have suffered 25 damage to their total health.

#define LIMB_BODY_DAMAGE_COEFFICIENT_ADVANCED 0.5 //Used by advanced robotic limbs.
#define LIMB_BODY_DAMAGE_COEFFICIENT_ADVANCED 0.66 //Used by advanced robotic limbs.
#define LIMB_BODY_DAMAGE_COEFFICIENT_DEFAULT 0.75 //Used by all limbs by default.
#define LIMB_BODY_DAMAGE_COEFFICIENT_TOTAL 1 //Used by heads and torsos
#define LIMB_BODY_DAMAGE_COEFFICIENT_PROSTHESIS 2.5 //Used by surplus prosthesis limbs
#define LIMB_BODY_DAMAGE_COEFFICIENT_PROSTHESIS 1.5 //Used by surplus prosthesis limbs

// EMP
// Note most of these values are doubled on heavy EMP
Expand Down
4 changes: 4 additions & 0 deletions code/__DEFINES/chat.dm
Original file line number Diff line number Diff line change
Expand Up @@ -46,3 +46,7 @@
#define debug_world_log(msg) if (GLOB.Debug2) log_world("DEBUG: [msg]")
/// Adds a generic box around whatever message you're sending in chat. Really makes things stand out.
#define examine_block(str) ("<div class='examine_block'>" + str + "</div>")
/// Makes a horizontal line with text in the middle
#define separator_hr(str) ("<div class='separator'>" + str + "</div>")
/// Helper which creates a chat message which may have a tooltip in some contexts, but not others.
#define conditional_tooltip(normal_text, tooltip_text, condition) ((condition) ? (span_tooltip(tooltip_text, normal_text)) : (normal_text))
4 changes: 2 additions & 2 deletions code/__DEFINES/dcs/signals/signals_mob/signals_mob_carbon.dm
Original file line number Diff line number Diff line change
Expand Up @@ -42,8 +42,8 @@

/// Called from update_health_hud, whenever a bodypart is being updated on the health doll
#define COMSIG_BODYPART_UPDATING_HEALTH_HUD "bodypart_updating_health_hud"
/// Return to override that bodypart's health hud with your own icon
#define COMPONENT_OVERRIDE_BODYPART_HEALTH_HUD (1<<0)
/// Return to override that bodypart's health hud with whatever is returned by the list
#define OVERRIDE_BODYPART_HEALTH_HUD (1<<0)

/// Called from /obj/item/bodypart/check_for_injuries (mob/living/carbon/examiner, list/check_list)
#define COMSIG_BODYPART_CHECKED_FOR_INJURY "bodypart_injury_checked"
Expand Down
4 changes: 4 additions & 0 deletions code/__DEFINES/mobs.dm
Original file line number Diff line number Diff line change
Expand Up @@ -978,3 +978,7 @@ GLOBAL_LIST_INIT(layers_to_offset, list(

/// Types of bullets that mining mobs take full damage from
#define MINING_MOB_PROJECTILE_VULNERABILITY list(BRUTE)

/// Distance which you can see someone's ID card
/// Short enough that you can inspect over tables (bartender checking age)
#define ID_EXAMINE_DISTANCE 3
5 changes: 4 additions & 1 deletion code/__DEFINES/span.dm
Original file line number Diff line number Diff line change
Expand Up @@ -117,18 +117,21 @@
#define span_secradio(str) ("<span class='secradio'>" + str + "</span>")
#define span_servradio(str) ("<span class='servradio'>" + str + "</span>")
#define span_singing(str) ("<span class='singing'>" + str + "</span>")
#define span_slightly_larger(str) ("<span class='slightly_larger'>" + str + "</span>")
#define span_slime(str) ("<span class='slime'>" + str + "</span>")
#define span_small(str) ("<span class='small'>" + str + "</span>")
#define span_smalldanger(str) ("<span class='smalldanger'>" + str + "</span>")
#define span_smallnotice(str) ("<span class='smallnotice'>" + str + "</span>")
#define span_smallnoticeital(str) ("<span class='smallnoticeital'>" + str + "</span>")
#define span_spiderbreacher(str) ("<span class='spiderbreacher'>" + str + "</span>")
#define span_spiderbroodmother(str) ("<span class='spiderbroodmother'>" + str + "</span>")
#define span_spiderscout(str) ("<span class='spiderscout'>" + str + "</span>")
#define span_spiderbreacher(str) ("<span class='spiderbreacher'>" + str + "</span>")
#define span_subheader_announcement_text(str) ("<span class='subheader_announcement_text'>" + str + "</span>")
#define span_suicide(str) ("<span class='suicide'>" + str + "</span>")
#define span_suppradio(str) ("<span class='suppradio'>" + str + "</span>")
#define span_syndradio(str) ("<span class='syndradio'>" + str + "</span>")
#define span_tape_recorder(str) ("<span class='tape_recorder'>" + str + "</span>")
#define span_tinydanger(str) ("<span class='tinydanger'>" + str + "</span>")
#define span_tinynotice(str) ("<span class='tinynotice'>" + str + "</span>")
#define span_tinynoticeital(str) ("<span class='tinynoticeital'>" + str + "</span>")
#define span_unconscious(str) ("<span class='unconscious'>" + str + "</span>")
Expand Down
6 changes: 5 additions & 1 deletion code/__DEFINES/surgery.dm
Original file line number Diff line number Diff line change
Expand Up @@ -26,8 +26,12 @@
#define ORGAN_HIDDEN (1<<9)
/// Has the organ already been inserted inside someone
#define ORGAN_VIRGIN (1<<10)
/// ALWAYS show this when scanned by advanced scanners, even if it is totally healthy
#define ORGAN_PROMINENT (1<<11)
/// An organ that is ostensibly dangerous when inside a body
#define ORGAN_HAZARDOUS (1<<12)

#define ORGAN_IRRADIATED (1<<11)
#define ORGAN_IRRADIATED (1<<13)

/// Helper to figure out if a limb is organic
#define IS_ORGANIC_LIMB(limb) (limb.bodytype & BODYTYPE_ORGANIC)
Expand Down
2 changes: 1 addition & 1 deletion code/_onclick/hud/hud.dm
Original file line number Diff line number Diff line change
Expand Up @@ -93,7 +93,7 @@ GLOBAL_LIST_INIT(available_ui_styles, list(

var/atom/movable/screen/healths
var/atom/movable/screen/stamina
var/atom/movable/screen/healthdoll
var/atom/movable/screen/healthdoll/healthdoll
var/atom/movable/screen/spacesuit
var/atom/movable/screen/hunger
// subtypes can override this to force a specific UI style
Expand Down
2 changes: 1 addition & 1 deletion code/_onclick/hud/human.dm
Original file line number Diff line number Diff line change
Expand Up @@ -271,7 +271,7 @@
hunger = new /atom/movable/screen/hunger(null, src)
infodisplay += hunger

healthdoll = new /atom/movable/screen/healthdoll(null, src)
healthdoll = new /atom/movable/screen/healthdoll/human(null, src)
infodisplay += healthdoll

stamina = new /atom/movable/screen/stamina(null, src)
Expand Down
84 changes: 83 additions & 1 deletion code/_onclick/hud/screen_objects.dm
Original file line number Diff line number Diff line change
Expand Up @@ -663,6 +663,88 @@
screen_loc = ui_living_healthdoll
var/filtered = FALSE //so we don't repeatedly create the mask of the mob every update

/atom/movable/screen/healthdoll/human
/// Tracks components of our doll, each limb is a separate atom in our vis_contents
VAR_PRIVATE/list/atom/movable/screen/limbs
/// Lazylist, tracks all body zones that are wounded currently
/// Used so we can sync animations should the list be updated
VAR_PRIVATE/list/animated_zones

/atom/movable/screen/healthdoll/human/Initialize(mapload, datum/hud/hud_owner)
. = ..()
limbs = list()
for(var/i in BODY_ZONES_ALL)
var/atom/movable/screen/healthdoll_limb/limb = new(src, null)
// layer chest above other limbs, it's the center after all
limb.layer = i == BODY_ZONE_CHEST ? layer + 0.05 : layer
limbs[i] = limb
// why viscontents? why not overlays? - because i want to animate filters
vis_contents += limb
update_appearance()

/atom/movable/screen/healthdoll/human/Destroy()
QDEL_LIST_ASSOC_VAL(limbs)
vis_contents.Cut()
return ..()

/atom/movable/screen/healthdoll/human/update_icon_state()
. = ..()
var/mob/living/carbon/human/owner = hud?.mymob
if(isnull(owner))
return
if(owner.stat == DEAD)
for(var/limb in limbs)
limbs[limb].icon_state = "[limb]DEAD"
return

var/list/current_animated = LAZYLISTDUPLICATE(animated_zones)

for(var/obj/item/bodypart/body_part as anything in owner.bodyparts)
var/icon_key = 0
var/part_zone = body_part.body_zone

var/list/overridable_key = list(icon_key)
if(body_part.bodypart_disabled)
icon_key = 7
else if(owner.stat == DEAD)
icon_key = "DEAD"
else if(SEND_SIGNAL(body_part, COMSIG_BODYPART_UPDATING_HEALTH_HUD, owner, overridable_key) & OVERRIDE_BODYPART_HEALTH_HUD)
icon_key = overridable_key[1] // thanks i hate it
else if(!owner.has_status_effect(/datum/status_effect/grouped/screwy_hud/fake_healthy))
var/damage = body_part.get_damage() / body_part.max_damage
// calculate what icon state (1-5, or 0 if undamaged) to use based on damage
icon_key = clamp(ceil(damage * 5), 0, 5)

if(length(body_part.wounds))
LAZYSET(animated_zones, part_zone, TRUE)
else
LAZYREMOVE(animated_zones, part_zone)
limbs[part_zone].icon_state = "[part_zone][icon_key]"
// handle leftovers
for(var/missing_zone in owner.get_missing_limbs())
limbs[missing_zone].icon_state = "[missing_zone]6"
LAZYREMOVE(animated_zones, missing_zone)
// time to re-sync animations, something changed
if(animated_zones ~! current_animated)
for(var/animated_zone in animated_zones)
var/atom/wounded_zone = limbs[animated_zone]
var/existing_filter = wounded_zone.get_filter("wound_outline")
if(existing_filter)
animate(existing_filter) // stop animation so we can resync
else
wounded_zone.add_filter("wound_outline", 1, list("type" = "outline", "color" = "#FF0033", "alpha" = 0, "size" = 1.2))
existing_filter = wounded_zone.get_filter("wound_outline")
animate(existing_filter, alpha = 200, time = 1.5 SECONDS, loop = -1)
animate(alpha = 0, time = 1.5 SECONDS)
if(LAZYLEN(current_animated)) // avoid null - list() runtimes please
for(var/lost_zone in current_animated - animated_zones)
limbs[lost_zone].remove_filter("wound_outline")

// Basically just holds an icon we can put a filter on
/atom/movable/screen/healthdoll_limb
screen_loc = ui_living_healthdoll
vis_flags = VIS_INHERIT_ID | VIS_INHERIT_PLANE

/atom/movable/screen/mood
name = "mood"
icon_state = "mood5"
Expand Down Expand Up @@ -846,7 +928,7 @@ INITIALIZE_IMMEDIATE(/atom/movable/screen/splash)
animate(get_filter("hunger_outline"), alpha = 200, time = 1.5 SECONDS, loop = -1)
animate(alpha = 0, time = 1.5 SECONDS)

else if(get_filter("hunger_outline"))
else
remove_filter("hunger_outline")

// Update color of the food
Expand Down
8 changes: 4 additions & 4 deletions code/datums/brain_damage/creepy_trauma.dm
Original file line number Diff line number Diff line change
Expand Up @@ -111,13 +111,13 @@
to_chat(owner, span_userdanger("You gag and swallow a bit of bile..."))

// if the creep examines first, then the obsession examines them, have a 50% chance to possibly blow their cover. wearing a mask avoids this risk
/datum/brain_trauma/special/obsessed/proc/stare(datum/source, mob/living/examining_mob, triggering_examiner)
/datum/brain_trauma/special/obsessed/proc/stare(datum/source, mob/living/examining_mob)
SIGNAL_HANDLER

if(examining_mob != owner || !triggering_examiner || prob(50))
return
if(prob(50))
return NONE

addtimer(CALLBACK(GLOBAL_PROC, GLOBAL_PROC_REF(to_chat), obsession, span_warning("You catch [examining_mob] staring at you..."), 3))
addtimer(CALLBACK(GLOBAL_PROC, GLOBAL_PROC_REF(to_chat), obsession, span_warning("You notice [examining_mob] staring at you oddly..."), 0.6 SECONDS))
return COMSIG_BLOCK_EYECONTACT

/datum/brain_trauma/special/obsessed/proc/find_obsession()
Expand Down
52 changes: 46 additions & 6 deletions code/datums/components/butchering.dm
Original file line number Diff line number Diff line change
Expand Up @@ -42,11 +42,13 @@
SIGNAL_HANDLER

if(!user.combat_mode)
return
return NONE
if(M.stat == DEAD && (M.butcher_results || M.guaranteed_butcher_results)) //can we butcher it?
if(butchering_enabled && (can_be_blunt || source.get_sharpness()))
INVOKE_ASYNC(src, PROC_REF(startButcher), source, M, user)
return COMPONENT_CANCEL_ATTACK_CHAIN
if(!DOING_INTERACTION(user, "[REF(M)]_butchering"))
INVOKE_ASYNC(src, PROC_REF(startButcher), source, M, user)
return COMPONENT_CANCEL_ATTACK_CHAIN
return NONE

if(ishuman(M) && source.force && source.get_sharpness())
var/mob/living/carbon/human/H = M
Expand All @@ -56,17 +58,34 @@
return COMPONENT_CANCEL_ATTACK_CHAIN

if(H.has_status_effect(/datum/status_effect/neck_slice))
return
return NONE

INVOKE_ASYNC(src, PROC_REF(startNeckSlice), source, H, user)
return COMPONENT_CANCEL_ATTACK_CHAIN

return NONE

/datum/component/butchering/proc/startButcher(obj/item/source, mob/living/M, mob/living/user)
to_chat(user, span_notice("You begin to butcher [M]..."))
playsound(M.loc, butcher_sound, 50, TRUE, -1)
if(do_after(user, speed, M) && M.Adjacent(source))
playsound(M, butcher_sound, 50, TRUE, -1)
if(do_after(user, speed, M, extra_checks = CALLBACK(src, PROC_REF(butcher_effects), source, M, user), interaction_key = "[REF(M)]_butchering") && M.Adjacent(source))
on_butchering(user, M)

/datum/component/butchering/proc/butcher_effects(obj/item/source, mob/living/butchering, mob/living/user)
if(user.next_move <= world.time)
user.face_atom(butchering)
// performs an attack against the mob
var/old_combat_mode = user.combat_mode
user.set_combat_mode(TRUE, TRUE)
source.melee_attack_chain(user, butchering)
user.set_combat_mode(old_combat_mode, TRUE)
// and play the sound
playsound(butchering, butcher_sound, 50, TRUE, -1)
// start our own click cd
user.changeNext_move(0.6 SECONDS)
// so the butchering doesn't stop
return TRUE

/datum/component/butchering/proc/startNeckSlice(obj/item/source, mob/living/carbon/human/H, mob/living/user)
if(DOING_INTERACTION_WITH_TARGET(user, H))
to_chat(user, span_warning("You're already interacting with [H]!"))
Expand Down Expand Up @@ -221,6 +240,9 @@
if(victim.stat == DEAD && (victim.butcher_results || victim.guaranteed_butcher_results))
on_butchering(parent, victim)

/datum/component/butchering/recycler/butcher_effects(obj/item/source, mob/living/butchering, mob/living/user)
return TRUE // p sure this won't work

/datum/component/butchering/mecha

/datum/component/butchering/mecha/RegisterWithParent()
Expand All @@ -242,6 +264,9 @@
SIGNAL_HANDLER
INVOKE_ASYNC(src, PROC_REF(on_butchering), chassis, target)

/datum/component/butchering/mecha/butcher_effects(obj/item/source, mob/living/butchering, mob/living/user)
return TRUE // could give this one, unsure if it's needed though

/datum/component/butchering/wearable

/datum/component/butchering/wearable/RegisterWithParent()
Expand Down Expand Up @@ -276,3 +301,18 @@
if(!isliving(target))
return NONE
return onItemAttack(parent, target, user)

/datum/component/butchering/wearable/butcher_effects(obj/item/source, mob/living/butchering, mob/living/user)
if(user.next_move <= world.time)
user.face_atom(butchering)
// performs an attack against the mob
var/old_combat_mode = user.combat_mode
user.set_combat_mode(TRUE, TRUE)
user.UnarmedAttack(butchering, TRUE)
user.set_combat_mode(old_combat_mode, TRUE)
// and play the sound
playsound(butchering, butcher_sound, 50, TRUE, -1)
// start our own click cd
user.changeNext_move(0.6 SECONDS)
// so the butchering doesn't stop
return TRUE
1 change: 1 addition & 0 deletions code/datums/components/sticker.dm
Original file line number Diff line number Diff line change
Expand Up @@ -80,6 +80,7 @@
var/atom/parent_atom = parent

sticker_overlay = mutable_appearance(icon = our_sticker.icon, icon_state = our_sticker.icon_state, layer = parent_atom.layer + 0.01, appearance_flags = RESET_COLOR)
sticker_overlay.color = our_sticker.color
sticker_overlay.pixel_w = px - world.icon_size / 2
sticker_overlay.pixel_z = py - world.icon_size / 2

Expand Down
7 changes: 6 additions & 1 deletion code/datums/datum.dm
Original file line number Diff line number Diff line change
Expand Up @@ -404,10 +404,15 @@

var/list/names = islist(name_or_names) ? name_or_names : list(name_or_names)

. = FALSE
for(var/name in names)
if(filter_data[name])
filter_data -= name
update_filters()
. = TRUE

if(.)
update_filters()
return .

/datum/proc/clear_filters()
ASSERT(isatom(src) || isimage(src))
Expand Down
1 change: 1 addition & 0 deletions code/datums/dna.dm
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ GLOBAL_LIST_INIT(identity_block_lengths, list(
GLOBAL_LIST_INIT(features_block_lengths, list(
"[DNA_MUTANT_COLOR_BLOCK]" = DNA_BLOCK_SIZE_COLOR,
"[DNA_ETHEREAL_COLOR_BLOCK]" = DNA_BLOCK_SIZE_COLOR,
"[DNA_FEATHER_COLOR_BLOCK]" = DNA_BLOCK_SIZE_COLOR, // NON-MODULE CHANGE
))

/**
Expand Down
4 changes: 2 additions & 2 deletions code/datums/elements/ai_held_item.dm
Original file line number Diff line number Diff line change
Expand Up @@ -52,9 +52,9 @@
SIGNAL_HANDLER

var/obj/item/carried_item = get_held_item(source)
if (!carried_item)
if (!carried_item || (carried_item.item_flags & (ABSTRACT|EXAMINE_SKIP|HAND_ITEM)))
return
examine_text += span_notice("[source.p_They()] [source.p_are()] carrying [carried_item.get_examine_string(user)].")
examine_text += span_notice("[source.p_They()] [source.p_are()] carrying [carried_item.examine_title(user)].")

/// If we died, drop anything we were carrying
/datum/element/ai_held_item/proc/on_death(mob/living/ol_yeller)
Expand Down
2 changes: 1 addition & 1 deletion code/datums/elements/dextrous.dm
Original file line number Diff line number Diff line change
Expand Up @@ -69,5 +69,5 @@
for(var/obj/item/held_item in examined.held_items)
if(held_item.item_flags & (ABSTRACT|EXAMINE_SKIP|HAND_ITEM))
continue
examine_list += span_info("[examined.p_They()] [examined.p_have()] [held_item.get_examine_string(user)] in [examined.p_their()] \
examine_list += span_info("[examined.p_They()] [examined.p_have()] [held_item.examine_title(user)] in [examined.p_their()] \
[examined.get_held_index_name(examined.get_held_index_of_item(held_item))].")
28 changes: 28 additions & 0 deletions code/datums/greyscale/json_configs/_umbrella.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
{
"umbrella_on": [
{
"type": "icon_state",
"icon_state": "umbrella_on_handle",
"blend_mode": "overlay"
},
{
"type": "icon_state",
"icon_state": "umbrella_on",
"blend_mode": "overlay",
"color_ids": [ 1 ]
}
],
"umbrella": [
{
"type": "icon_state",
"icon_state": "umbrella_handle",
"blend_mode": "overlay"
},
{
"type": "icon_state",
"icon_state": "umbrella",
"blend_mode": "overlay",
"color_ids": [ 1 ]
}
]
}
Loading

0 comments on commit ea9a09c

Please sign in to comment.