diff --git a/_maps/map_files/BoxStation/BoxStation.dmm b/_maps/map_files/BoxStation/BoxStation.dmm
index c157be159a4e..c1472d132ff8 100644
--- a/_maps/map_files/BoxStation/BoxStation.dmm
+++ b/_maps/map_files/BoxStation/BoxStation.dmm
@@ -9908,8 +9908,9 @@
"doR" = (
/obj/machinery/power/apc/auto_name/directional/south,
/obj/structure/cable,
-/obj/effect/mapping_helpers/apc/cell_5k,
/obj/machinery/plumbing/ooze_compressor,
+/obj/effect/mapping_helpers/apc/cell_10k,
+/obj/effect/mapping_helpers/apc/full_charge,
/turf/open/floor/iron/white,
/area/station/science/xenobiology)
"doW" = (
diff --git a/_maps/map_files/Deltastation/DeltaStation2.dmm b/_maps/map_files/Deltastation/DeltaStation2.dmm
index 29a07339b6ba..43768adc6127 100644
--- a/_maps/map_files/Deltastation/DeltaStation2.dmm
+++ b/_maps/map_files/Deltastation/DeltaStation2.dmm
@@ -42678,6 +42678,8 @@
/obj/effect/turf_decal/siding/purple{
dir = 9
},
+/obj/effect/mapping_helpers/apc/cell_10k,
+/obj/effect/mapping_helpers/apc/full_charge,
/turf/open/floor/iron,
/area/station/science/xenobiology)
"kpy" = (
@@ -89711,6 +89713,11 @@
/obj/effect/decal/cleanable/dirt,
/turf/open/floor/iron,
/area/station/engineering/atmos)
+"vUy" = (
+/obj/machinery/duct,
+/obj/machinery/light/floor/has_bulb,
+/turf/open/floor/engine,
+/area/station/science/xenobiology)
"vUz" = (
/obj/structure/disposalpipe/segment,
/turf/open/floor/iron/white,
@@ -121069,7 +121076,7 @@ feF
gcr
pTC
iLv
-gvX
+vUy
duq
fMl
fMl
@@ -121079,7 +121086,7 @@ taA
fMl
fMl
xsN
-gvX
+vUy
iLv
mAt
aAx
@@ -122868,7 +122875,7 @@ woj
rrU
pTC
iLv
-gvX
+vUy
pQz
fMl
fMl
@@ -122878,7 +122885,7 @@ taA
fMl
fMl
esm
-gvX
+vUy
iLv
mAt
cZl
diff --git a/_maps/map_files/IceBoxStation/IceBoxStation.dmm b/_maps/map_files/IceBoxStation/IceBoxStation.dmm
index 6cae56e2d386..10dc68395186 100644
--- a/_maps/map_files/IceBoxStation/IceBoxStation.dmm
+++ b/_maps/map_files/IceBoxStation/IceBoxStation.dmm
@@ -1194,6 +1194,8 @@
},
/obj/machinery/power/apc/auto_name/directional/south,
/obj/structure/cable,
+/obj/effect/mapping_helpers/apc/cell_10k,
+/obj/effect/mapping_helpers/apc/full_charge,
/turf/open/floor/iron/white,
/area/station/science/xenobiology)
"aut" = (
diff --git a/_maps/map_files/MetaStation/MetaStation.dmm b/_maps/map_files/MetaStation/MetaStation.dmm
index bcc8844dcc25..b477ff8f1011 100644
--- a/_maps/map_files/MetaStation/MetaStation.dmm
+++ b/_maps/map_files/MetaStation/MetaStation.dmm
@@ -13278,6 +13278,8 @@
/obj/effect/turf_decal/tile/purple/half/contrasted{
dir = 8
},
+/obj/effect/mapping_helpers/apc/cell_10k,
+/obj/effect/mapping_helpers/apc/full_charge,
/turf/open/floor/iron/white,
/area/station/science/cytology)
"eYw" = (
@@ -32577,8 +32579,6 @@
/obj/machinery/atmospherics/pipe/smart/manifold4w/supply/hidden/layer4,
/obj/machinery/atmospherics/pipe/smart/manifold4w/scrubbers/hidden/layer2,
/obj/machinery/duct,
-/obj/machinery/power/apc/auto_name/directional/west,
-/obj/structure/cable,
/turf/open/floor/stone,
/area/station/science/xenobiology)
"lJh" = (
@@ -49930,6 +49930,10 @@
/obj/effect/turf_decal/stripes/line{
dir = 9
},
+/obj/effect/mapping_helpers/apc/cell_10k,
+/obj/machinery/power/apc/auto_name/directional/north,
+/obj/effect/mapping_helpers/apc/full_charge,
+/obj/structure/cable,
/turf/open/floor/stone,
/area/station/science/xenobiology)
"rMz" = (
diff --git a/_maps/map_files/tramstation/tramstation.dmm b/_maps/map_files/tramstation/tramstation.dmm
index 096c0677e899..cef5e292f535 100644
--- a/_maps/map_files/tramstation/tramstation.dmm
+++ b/_maps/map_files/tramstation/tramstation.dmm
@@ -44309,6 +44309,16 @@
/obj/structure/cable,
/turf/open/floor/carpet,
/area/station/command/heads_quarters/hos)
+"nme" = (
+/obj/effect/turf_decal/trimline/purple/filled/line{
+ dir = 1
+ },
+/obj/machinery/power/apc/auto_name/directional/north,
+/obj/structure/cable,
+/obj/effect/mapping_helpers/apc/cell_10k,
+/obj/effect/mapping_helpers/apc/full_charge,
+/turf/open/floor/stone,
+/area/station/science/xenobiology)
"nmf" = (
/obj/structure/table/wood,
/obj/machinery/computer/security/wooden_tv,
@@ -54558,14 +54568,6 @@
/obj/structure/cable,
/turf/open/space/basic,
/area/space/nearstation)
-"qCW" = (
-/obj/effect/turf_decal/trimline/purple/filled/line{
- dir = 1
- },
-/obj/machinery/power/apc/auto_name/directional/north,
-/obj/structure/cable,
-/turf/open/floor/stone,
-/area/station/science/xenobiology)
"qCX" = (
/obj/effect/turf_decal/stripes/line{
dir = 9
@@ -67170,6 +67172,16 @@
/obj/machinery/door/firedoor,
/turf/open/floor/iron,
/area/station/cargo/drone_bay)
+"uAW" = (
+/obj/effect/decal/cleanable/dirt,
+/obj/machinery/atmospherics/pipe/smart/manifold4w/scrubbers/hidden/layer2,
+/obj/machinery/atmospherics/pipe/smart/manifold4w/supply/hidden/layer4,
+/obj/structure/cable,
+/obj/structure/disposalpipe/segment{
+ dir = 10
+ },
+/turf/open/floor/catwalk_floor,
+/area/station/maintenance/starboard/central)
"uAZ" = (
/obj/effect/turf_decal/trimline/white/warning{
dir = 8
@@ -124257,7 +124269,7 @@ xPQ
vud
tml
qVr
-qCW
+nme
pbt
xcM
dtN
@@ -126540,8 +126552,8 @@ hFC
bwz
xks
qxm
-icR
-icR
+uAW
+lLW
iix
tuN
tuN
@@ -127825,7 +127837,7 @@ oys
oys
gDw
gDw
-nop
+bay
nop
hPn
eyE
@@ -128080,8 +128092,8 @@ kHX
gDw
gDw
gDw
-nop
-nop
+bay
+uJu
nop
oys
oys
diff --git a/code/__DEFINES/maths.dm b/code/__DEFINES/maths.dm
index 18a07c0d0197..b86931a592a5 100644
--- a/code/__DEFINES/maths.dm
+++ b/code/__DEFINES/maths.dm
@@ -7,6 +7,9 @@
#define IS_FINITE__UNSAFE(a) (a-a == a-a)
#define IS_FINITE(a) (isnum(a) && IS_FINITE__UNSAFE(a))
+
+#define IS_SAFE_NUM(a) (isnum(a) && !IS_INF__UNSAFE(a) && IS_FINITE__UNSAFE(a))
+
// ------------------------------------
// Aight dont remove the rest
diff --git a/code/__DEFINES/~monkestation/clock_cult.dm b/code/__DEFINES/~monkestation/clock_cult.dm
index 8a1bb92a7446..e9c9424b50dd 100644
--- a/code/__DEFINES/~monkestation/clock_cult.dm
+++ b/code/__DEFINES/~monkestation/clock_cult.dm
@@ -5,5 +5,16 @@
#define SIGIL_TRANSMISSION_RANGE 4
+///base state the ark is created in, any state besides this will be a hostile environment
+#define ARK_STATE_BASE 0
+///state for the grace period after the cult has reached its member count max and have enough activing anchoring crystals to summon
+#define ARK_STATE_CHARGING 1
+///state for after the cult has been annouced as well as the first half of the assault
+#define ARK_STATE_ACTIVE 2
+///state for the halfway point of ark activation
+#define ARK_STATE_SUMMONING 3
+///the ark has either finished opening or been destroyed in this state
+#define ARK_STATE_FINAL 4
+
///max damage taken per hit by "important" clock structures
#define MAX_IMPORTANT_CLOCK_DAMAGE 30
diff --git a/code/controllers/subsystem/polling.dm b/code/controllers/subsystem/polling.dm
index 8341181e926a..745477986109 100644
--- a/code/controllers/subsystem/polling.dm
+++ b/code/controllers/subsystem/polling.dm
@@ -179,6 +179,9 @@ SUBSYSTEM_DEF(polling)
/datum/controller/subsystem/polling/proc/poll_ghost_candidates_for_mob(question, role, check_jobban, poll_time = 30 SECONDS, mob/target_mob, ignore_category = null, flashwindow = TRUE, pic_source, role_name_text)
var/static/list/mob/currently_polling_mobs = list()
+ if(!isnull(target_mob) && !ismob(target_mob))
+ stack_trace("attempted to use a non-mob as the target mob ([target_mob] | [target_mob.type])")
+
if(currently_polling_mobs.Find(target_mob))
return list()
diff --git a/code/game/objects/items/storage/bags.dm b/code/game/objects/items/storage/bags.dm
index 2ef99e3dc5f2..99d5fd4e91c9 100644
--- a/code/game/objects/items/storage/bags.dm
+++ b/code/game/objects/items/storage/bags.dm
@@ -508,6 +508,7 @@
/obj/item/reagent_containers/syringe,
/obj/item/slime_extract,
/obj/item/swab,
+ /obj/item/stack/biomass // monke: make science bags able to hold biomass cubes
))
/*
diff --git a/code/modules/atmospherics/gasmixtures/reactions.dm b/code/modules/atmospherics/gasmixtures/reactions.dm
index 736e438ab6c1..b1c08db0e731 100644
--- a/code/modules/atmospherics/gasmixtures/reactions.dm
+++ b/code/modules/atmospherics/gasmixtures/reactions.dm
@@ -1227,8 +1227,8 @@
var/particle_chance = ((PARTICLE_CHANCE_CONSTANT)/(reaction_energy-PARTICLE_CHANCE_CONSTANT)) + 1//Asymptopically approaches 100% as the energy of the reaction goes up.
if(prob(PERCENT(particle_chance)))
location.fire_nuclear_particle()
- var/rad_power = max((FUSION_RAD_COEFFICIENT/instability) + 5000, 0)
- radiation_pulse(location,rad_power)
+ var/rad_power = max((FUSION_RAD_COEFFICIENT/instability) + 250, 15000) //Wow!
+ radiation_pulse(location, rand(1,30), 9, 60, 2 SECONDS, rad_power, 1) // Nonmodular. Fixes fusion going wacko mode - makes it significantly harder to irradiate people through walls, and fixes the range issue.
var/new_heat_capacity = air.heat_capacity()
if(new_heat_capacity > MINIMUM_HEAT_CAPACITY && (air.temperature <= FUSION_MAXIMUM_TEMPERATURE || reaction_energy <= 0)) //If above FUSION_MAXIMUM_TEMPERATURE, will only adjust temperature for endothermic reactions.
air.temperature = clamp(((air.temperature*old_heat_capacity + reaction_energy)/new_heat_capacity),TCMB,INFINITY)
diff --git a/code/modules/mob/living/basic/guardian/guardian_verbs.dm b/code/modules/mob/living/basic/guardian/guardian_verbs.dm
index abf536503b2a..f2e81306c515 100644
--- a/code/modules/mob/living/basic/guardian/guardian_verbs.dm
+++ b/code/modules/mob/living/basic/guardian/guardian_verbs.dm
@@ -170,9 +170,8 @@
"Do you want to play as [owner.real_name]'s [chosen_guardian.theme.name]?",
check_jobban = ROLE_PAI,
poll_time = 10 SECONDS,
- target_mob = src,
+ target_mob = owner,
ignore_category = POLL_IGNORE_HOLOPARASITE,
- pic_source = src,
role_name_text = chosen_guardian.theme.name
)
if (!LAZYLEN(ghost_candidates))
diff --git a/code/modules/mob/living/basic/pets/parrot/parrot_ai/parroting_action.dm b/code/modules/mob/living/basic/pets/parrot/parrot_ai/parroting_action.dm
index 52f516ee4060..d1488a60b3bb 100644
--- a/code/modules/mob/living/basic/pets/parrot/parrot_ai/parroting_action.dm
+++ b/code/modules/mob/living/basic/pets/parrot/parrot_ai/parroting_action.dm
@@ -36,13 +36,6 @@
modified_speech = copytext_char(speech, 3)
else
- // monke start - stfu poly
- var/speak_prob = speaking_pawn.ears.use_command ? 25 : 50
- for(var/channel in list(RADIO_KEY_COMMON, RADIO_TOKEN_COMMAND))
- if(channel in available_channels)
- speak_prob = max(FLOOR(speak_prob * 0.4, 5), 5)
- use_radio = prob(speak_prob)
- // monke end
if(HAS_CHANNEL_PREFIX)
modified_speech = "[use_radio ? pick(available_channels) : ""][copytext_char(speech, 3)]"
else
diff --git a/code/modules/research/xenobiology/crossbreeding/_status_effects.dm b/code/modules/research/xenobiology/crossbreeding/_status_effects.dm
index a7c6c3baed07..366e4d2e692a 100644
--- a/code/modules/research/xenobiology/crossbreeding/_status_effects.dm
+++ b/code/modules/research/xenobiology/crossbreeding/_status_effects.dm
@@ -470,8 +470,8 @@
colour = "grey"
/datum/status_effect/stabilized/grey/tick()
- for(var/mob/living/basic/slime/S in range(1, get_turf(owner)))
- SEND_SIGNAL(S, COMSIG_FRIENDSHIP_CHANGE, owner, 1)
+ for(var/mob/living/basic/slime/new_friend in range(3, get_turf(owner)))
+ SEND_SIGNAL(new_friend, COMSIG_FRIENDSHIP_CHANGE, owner, 1)
return ..()
/datum/status_effect/stabilized/orange
diff --git a/code/modules/vending/autodrobe.dm b/code/modules/vending/autodrobe.dm
index deae8acc20fe..6a8f4fa52bb8 100644
--- a/code/modules/vending/autodrobe.dm
+++ b/code/modules/vending/autodrobe.dm
@@ -14,6 +14,8 @@
"products" = list(
/obj/item/clothing/under/costume/gladiator = 1,
/obj/item/clothing/head/helmet/gladiator = 1,
+ /obj/item/clothing/head/viking/fake_helmet = 1, //monkestation edit
+ /obj/item/clothing/under/viking/fake_tunic = 1, //monkestation edit
/obj/item/clothing/suit/toggle/labcoat/mad = 1,
/obj/item/clothing/suit/bio_suit/plaguedoctorsuit = 1,
/obj/item/clothing/head/bio_hood/plague = 1,
diff --git a/html/changelogs/AutoChangeLog-pr-1638.yml b/html/changelogs/AutoChangeLog-pr-1638.yml
new file mode 100644
index 000000000000..f4143ee6fb7c
--- /dev/null
+++ b/html/changelogs/AutoChangeLog-pr-1638.yml
@@ -0,0 +1,16 @@
+author: "Absolucy"
+delete-after: True
+changes:
+ - rscadd: "Added a HUD that appears when hovering your mouse over ooze compressors, showing the current extract it is making, and the progress on making said extract."
+ - bugfix: "Fixed the normal slime extract menu coming up on ooze compressors after selecting a crossbreed."
+ - bugfix: "Fixed xenobiology machinery taking up absurd amounts of power while doing nothing."
+ - balance: "Increased range of stabilized grey extract from 1 to 3."
+ - qol: "Increased biomass cube stack size (5 -> 15)."
+ - qol: "Biocubes now fit in science bags."
+ - qol: "You can dump biocubes from a science bag into the biomass recycler."
+ - qol: "You can right click the slime market with a science bag to sell all the extracts without the confirmation prompt."
+ - rscadd: "Added balloon alerts to lots of xenobio-related actions."
+ - qol: "You can toggle ooze suckers directly via right click."
+ - qol: "You can now right click an active ooze compressor to cancel the current recipe."
+ - qol: "Slime mutation syringes now show whether they've been used or not in the examine text."
+ - bugfix: "Removing accessories from slimes now works properly."
\ No newline at end of file
diff --git a/html/changelogs/AutoChangeLog-pr-1654.yml b/html/changelogs/AutoChangeLog-pr-1654.yml
deleted file mode 100644
index 75bf62c0e1e9..000000000000
--- a/html/changelogs/AutoChangeLog-pr-1654.yml
+++ /dev/null
@@ -1,4 +0,0 @@
-author: "KnigTheThrasher"
-delete-after: True
-changes:
- - bugfix: "Fixed Icebox tests"
\ No newline at end of file
diff --git a/html/changelogs/AutoChangeLog-pr-1658.yml b/html/changelogs/AutoChangeLog-pr-1658.yml
new file mode 100644
index 000000000000..6e6f666425ed
--- /dev/null
+++ b/html/changelogs/AutoChangeLog-pr-1658.yml
@@ -0,0 +1,5 @@
+author: "Absolucy"
+delete-after: True
+changes:
+ - bugfix: "Fix holoparasite resetting not working."
+ - code_imp: "Added a safety check that'll emit a (harmless) runtime if something tries to pass an invalid target mob to `SSpolling.poll_ghost_candidates_for_mob`."
\ No newline at end of file
diff --git a/html/changelogs/archive/2024-04.yml b/html/changelogs/archive/2024-04.yml
index 28d6899e2947..b77717544386 100644
--- a/html/changelogs/archive/2024-04.yml
+++ b/html/changelogs/archive/2024-04.yml
@@ -178,3 +178,27 @@
- qol: Mute people who can still use sign language are now capable of using sign
language to invoke spells. Wizards are still too egotistical to even consider
doing so, tho.
+2024-04-16:
+ KnigTheThrasher:
+ - bugfix: Fixed Icebox tests
+2024-04-17:
+ Absolucy:
+ - qol: Heavily increases and randomizes the speech cooldown of parrots.
+ DimWhat:
+ - rscadd: Added two new tails to the anime quirk
+ KittyNoodle:
+ - balance: Hardlight spears can no longer be used with scarp.
+ - balance: Hardlight spear damage and wounding reduced.
+ - balance: Hardlight spear homing removed.
+ - balance: Hardlight spear embed chance halved
+ - balance: Hardlight spears now have cooldown after firing before you can use another
+ spear.
+ KnigTheThrasher:
+ - bugfix: Tram disposals should work now
+ TTNT789:
+ - rscadd: adds a viking costume to the autodrobe
+ Yawet330:
+ - bugfix: Fusion, why?
+ wraith-54321:
+ - balance: Clockwork airlocks lose their damage deflection when placed on reebe
+ - balance: Clockwork walls take half as long to deconstruct on reebe
diff --git a/monkestation/code/__HELPERS/reagents.dm b/monkestation/code/__HELPERS/reagents.dm
index 4803b2044356..5400bdedafed 100644
--- a/monkestation/code/__HELPERS/reagents.dm
+++ b/monkestation/code/__HELPERS/reagents.dm
@@ -1,7 +1,8 @@
///Returns a random reagent object minus ethanol reagents
/proc/get_random_reagent_id_unrestricted_non_ethanol()
- var/static/list/random_reagents = list()
- if(!random_reagents.len)
+ var/static/list/random_reagents
+ if(!random_reagents)
+ random_reagents = list()
for(var/datum/reagent/reagent_path as anything in subtypesof(/datum/reagent))
if(istype(reagent_path, /datum/reagent/consumable) && !initial(reagent_path.bypass_restriction))
continue
diff --git a/monkestation/code/game/objects/items/implants/hardlight.dm b/monkestation/code/game/objects/items/implants/hardlight.dm
index 5d7518b40a53..c77e5885472c 100644
--- a/monkestation/code/game/objects/items/implants/hardlight.dm
+++ b/monkestation/code/game/objects/items/implants/hardlight.dm
@@ -234,12 +234,6 @@
back_spear_overlay.pixel_x = -32
. += back_spear_overlay
-
-/obj/item/gun/magic/hardlight_spear/can_trigger_gun(mob/living/user, akimbo_usage) // This isn't really a gun, so it shouldn't be checking for TRAIT_NOGUNS, a firing pin (pinless), or a trigger guard (guardless)
- if(akimbo_usage)
- return FALSE //this would be kinda weird while shooting someone down.
- return TRUE
-
/obj/item/gun/magic/hardlight_spear/process_fire(atom/target, mob/living/user, message = TRUE, params = null, zone_override = "", bonus_spread = 0)
. = ..()
if(!.)
@@ -247,11 +241,16 @@
if(spears_left)
var/obj/item/gun/magic/hardlight_spear/spear = new type
spear.spears_left = spears_left - 1
+ spear.apply_cooldown()
qdel(src)
user.put_in_hands(spear)
else
user.dropItemToGround(src, TRUE)
+/obj/item/gun/magic/hardlight_spear/proc/apply_cooldown()
+ semicd = TRUE
+ addtimer(CALLBACK(src, PROC_REF(reset_semicd)), 1.5 SECONDS)
+
/obj/item/ammo_casing/magic/hardlight_spear
name = "please god report this"
desc = "Why god why"
@@ -259,34 +258,23 @@
projectile_type = /obj/projectile/bullet/hardlight_spear
heavy_metal = FALSE
-/obj/item/ammo_casing/magic/hardlight_spear/ready_proj(atom/target, mob/living/user, quiet, zone_override, atom/fired_from)
- if(!loaded_projectile)
- return
-
- if(isliving(target))
- loaded_projectile.homing = TRUE
- loaded_projectile.homing_turn_speed = 40
- loaded_projectile.set_homing_target(target)
-
- return ..()
-
/obj/projectile/bullet/hardlight_spear
name = "hardlight spear"
icon = 'monkestation/icons/obj/guns/projectiles.dmi'
icon_state = "lightspear"
- damage = 45
- armour_penetration = 10
- wound_bonus = 5
- bare_wound_bonus = 60
- wound_falloff_tile = 0
- embed_falloff_tile = 0
- speed = 0.4 //lower = faster
+ damage = 40
+ armour_penetration = 5
+ wound_bonus = -5
+ bare_wound_bonus = 50
+ wound_falloff_tile = -1
+ embed_falloff_tile = -1
+ speed = 0.7 //lower = faster
shrapnel_type = /obj/item/shrapnel/bullet/spear
light_outer_range = 1
light_power = 1
hitsound = 'sound/weapons/bladeslice.ogg'
hitsound_wall = 'sound/weapons/parry.ogg'
- embedding = list(embed_chance=100, fall_chance=2, jostle_chance=4, ignore_throwspeed_threshold=TRUE, pain_stam_pct=0.5, pain_mult=5, jostle_pain_mult=6, rip_time=10)
+ embedding = list(embed_chance=70, fall_chance=6, jostle_chance=4, ignore_throwspeed_threshold=TRUE, pain_stam_pct=0.5, pain_mult=2, jostle_pain_mult=3, rip_time=10)
/obj/item/shrapnel/bullet/spear
name = "hardlight spear"
diff --git a/monkestation/code/modules/antagonists/clock_cult/machines/observation_console.dm b/monkestation/code/modules/antagonists/clock_cult/machines/observation_console.dm
index be7c8c793de2..42b7795e2773 100644
--- a/monkestation/code/modules/antagonists/clock_cult/machines/observation_console.dm
+++ b/monkestation/code/modules/antagonists/clock_cult/machines/observation_console.dm
@@ -4,18 +4,23 @@
icon_screen = "ratvar1"
icon_keyboard = "ratvar_key1"
icon_state = "ratvarcomputer"
+ resistance_flags = INDESTRUCTIBLE
clockwork = TRUE
lock_override = TRUE
circuit = /obj/item/circuitboard/machine/camera_console_ratvar
+ ///List of areas we are allowed to warp to
+ var/static/list/allowed_areas = list()
/obj/machinery/computer/camera_advanced/ratvar/Initialize(mapload)
. = ..()
START_PROCESSING(SSobj, src)
actions += new /datum/action/innate/clockcult/warp(src)
+ actions += new /datum/action/innate/clockcult/console_info(src)
+ actions += new /datum/action/innate/clockcult/add_warp_area(src, src)
/obj/machinery/computer/camera_advanced/ratvar/Destroy()
STOP_PROCESSING(SSobj, src)
- . = ..()
+ return ..()
/obj/machinery/computer/camera_advanced/ratvar/process(seconds_per_tick)
if(SPT_PROB(3, seconds_per_tick))
@@ -52,7 +57,7 @@
/datum/action/innate/clockcult/warp/Activate()
if(!isliving(owner))
return
- if(GLOB.clock_ark && GLOB.clock_ark.current_state >= 2) //equal to ARK_STATE_ACTIVE, not using the defines so we can #undef them
+ if(GLOB.clock_ark && GLOB.clock_ark.current_state >= ARK_STATE_ACTIVE)
to_chat(owner, span_brass("You cannot warp while the gateway is opening!"))
return
if(warping)
@@ -76,14 +81,34 @@
warping = TRUE
button_icon_state = "warp_cancel"
build_all_button_icons(UPDATE_BUTTON_ICON)
- if(do_after(cam_user, 5 SECONDS, target = target_loc, extra_checks = CALLBACK(src, PROC_REF(special_check))))
+ if(do_after(cam_user, 5 SECONDS, target = target_loc, extra_checks = CALLBACK(src, PROC_REF(warping_check))))
try_servant_warp(cam_user, target_loc)
button_icon_state = "warp_down"
build_all_button_icons(UPDATE_BUTTON_ICON)
warping = FALSE
-/datum/action/innate/clockcult/warp/proc/special_check()
+/datum/action/innate/clockcult/warp/proc/warping_check()
return warping
+/datum/action/innate/clockcult/console_info
+ name = "Console info"
+ desc = "Get info on this console."
+ button_icon_state = "console_info"
+
+/datum/action/innate/clockcult/add_warp_area
+ name = "Add Warp Area"
+ desc = "Add an additional area you can warp to."
+ button_icon_state = "Spatial Warp"
+ ///Ref to the console we are linked to
+ var/obj/machinery/computer/camera_advanced/ratvar/linked_console
+
+/datum/action/innate/clockcult/add_warp_area/New(Target, console)
+ . = ..()
+ linked_console = console
+
+/datum/action/innate/clockcult/add_warp_area/Destroy()
+ linked_console = null
+ return ..()
+
/obj/item/circuitboard/machine/camera_console_ratvar
build_path = /obj/machinery/computer/camera_advanced/ratvar
diff --git a/monkestation/code/modules/antagonists/clock_cult/structures/airlock.dm b/monkestation/code/modules/antagonists/clock_cult/structures/airlock.dm
index 0b678db0a8e0..c14bf31356c6 100644
--- a/monkestation/code/modules/antagonists/clock_cult/structures/airlock.dm
+++ b/monkestation/code/modules/antagonists/clock_cult/structures/airlock.dm
@@ -11,6 +11,11 @@
req_access = list(ACCESS_CLOCKCULT)
damage_deflection = 10
+/obj/machinery/door/airlock/bronze/clock/Initialize(mapload)
+ . = ..()
+ if(on_reebe(src))
+ damage_deflection = 0
+
/obj/machinery/door/airlock/bronze/clock/canAIControl(mob/user)
return (IS_CLOCK(user) && !isAllPowerCut())
diff --git a/monkestation/code/modules/antagonists/clock_cult/structures/the_ark.dm b/monkestation/code/modules/antagonists/clock_cult/structures/the_ark.dm
index 48fb92ad8923..a10d48c3f560 100644
--- a/monkestation/code/modules/antagonists/clock_cult/structures/the_ark.dm
+++ b/monkestation/code/modules/antagonists/clock_cult/structures/the_ark.dm
@@ -1,11 +1,6 @@
GLOBAL_DATUM(clock_ark, /obj/structure/destructible/clockwork/the_ark) //set to be equal to the ark on creation if none
GLOBAL_VAR_INIT(ratvar_risen, FALSE)
-#define ARK_STATE_BASE 0 //base state the ark is created in, any state besides this will be a hostile environment
-#define ARK_STATE_CHARGING 1 //state for the grace period after the cult has reached its member count max and have an active anchor crystal
-#define ARK_STATE_ACTIVE 2 //state for after the cult has been annouced as well as the first half of the assault
-#define ARK_STATE_SUMMONING 3 //state for the halfway point of ark activation
-#define ARK_STATE_FINAL 4 //the ark has either finished opening or been destroyed in this state
#define ARK_READY_PERIOD 300 SECONDS //how long until the cult is annouced after they reach max members, 5 minutes
#define ARK_GRACE_PERIOD 210 SECONDS //how long until the portals open after the cult is annouced, 3 minutes 30 seconds
#define ARK_ASSAULT_PERIOD 600 //how long the crew has to destroy the ark after the assault begins, 10 minutes
@@ -239,11 +234,6 @@ GLOBAL_VAR_INIT(ratvar_risen, FALSE)
explosion(pick(GLOB.abscond_markers), 50, 40, 30, 30, FALSE, TRUE)
SSticker.force_ending = TRUE
-#undef ARK_STATE_BASE
-#undef ARK_STATE_CHARGING
-#undef ARK_STATE_ACTIVE
-#undef ARK_STATE_SUMMONING
-#undef ARK_STATE_FINAL
#undef ARK_READY_PERIOD
#undef ARK_GRACE_PERIOD
#undef ARK_ASSAULT_PERIOD
diff --git a/monkestation/code/modules/antagonists/clock_cult/turf.dm b/monkestation/code/modules/antagonists/clock_cult/turf.dm
index c8e33fe5e01d..8bd077ff4d42 100644
--- a/monkestation/code/modules/antagonists/clock_cult/turf.dm
+++ b/monkestation/code/modules/antagonists/clock_cult/turf.dm
@@ -195,6 +195,9 @@
//do the deconstruction stuff, this really should be a proc on Rwalls as well
/turf/closed/wall/clockwork/proc/next_decon_state(obj/item/used_tool, mob/user, current_state, set_state, sent_message, use_time = 4 SECONDS)
+ if(on_reebe(src))
+ use_time = round(use_time / 2, 0.1) //each stage takes half as long on reebe
+
if(used_tool.use_tool(src, user, use_time, volume=100))
if(!istype(src, /turf/closed/wall/clockwork) || d_state != current_state)
return TRUE
diff --git a/monkestation/code/modules/mob/dead/new_player/sprite_accessories/anime.dm b/monkestation/code/modules/mob/dead/new_player/sprite_accessories/anime.dm
index 2ffaa4b5f2ef..6faea383a344 100644
--- a/monkestation/code/modules/mob/dead/new_player/sprite_accessories/anime.dm
+++ b/monkestation/code/modules/mob/dead/new_player/sprite_accessories/anime.dm
@@ -165,3 +165,13 @@
/datum/sprite_accessory/anime_bottom/bunny
name = "Bunny Puff"
icon_state = "playbunny"
+
+/datum/sprite_accessory/anime_bottom/mouse
+ name = "Mouse Tail"
+ icon_state = "mouse"
+ color_src = null
+
+/datum/sprite_accessory/anime_bottom/plug
+ name = "Plug"
+ icon_state = "plug"
+ hasinner = TRUE
diff --git a/monkestation/code/modules/mob/living/basic/pets/parrot/parrot_ai/parroting_action.dm b/monkestation/code/modules/mob/living/basic/pets/parrot/parrot_ai/parroting_action.dm
new file mode 100644
index 000000000000..96c6ed44340c
--- /dev/null
+++ b/monkestation/code/modules/mob/living/basic/pets/parrot/parrot_ai/parroting_action.dm
@@ -0,0 +1,5 @@
+/datum/ai_behavior/perform_speech/parrot
+ action_cooldown = 45 SECONDS // SHUT UP
+
+/datum/ai_behavior/perform_speech/parrot/perform(seconds_per_tick, datum/ai_controller/controller, ...)
+ controller.behavior_cooldowns[src] = world.time + rand(45 SECONDS, 3 MINUTES)
diff --git a/monkestation/code/modules/slimecore/items/mutation_syringe.dm b/monkestation/code/modules/slimecore/items/mutation_syringe.dm
index fd3998c06be8..75883df8a3ff 100644
--- a/monkestation/code/modules/slimecore/items/mutation_syringe.dm
+++ b/monkestation/code/modules/slimecore/items/mutation_syringe.dm
@@ -5,25 +5,41 @@
icon = 'monkestation/code/modules/slimecore/icons/slimes.dmi'
icon_state = "mutation_syringe"
- ///the path we infuse
+ w_class = WEIGHT_CLASS_SMALL
+
+ /// Type path of the slime trait to infuse.
var/datum/slime_trait/infusing_trait_path
- /// have we been used?
- var/used = FALSE
+ /// Amount of uses remaining.
+ var/uses = 1
+/obj/item/slime_mutation_syringe/examine(mob/user)
+ . = ..()
+ if(uses)
+ . += span_notice("It has [uses] uses left.")
+ else
+ . += span_warning("It has been completely used up.")
/obj/item/slime_mutation_syringe/afterattack(atom/target, mob/user, proximity_flag, click_parameters)
. = ..()
- if(!infusing_trait_path || used)
+ if(!ispath(infusing_trait_path))
+ return
+ if(!uses)
+ user.balloon_alert(user, "used up")
+ to_chat(user, span_warning("[src] has been completely used up!"))
return
if(!istype(target, /mob/living/basic/slime))
return
var/mob/living/basic/slime/slime = target
if(slime.add_trait(infusing_trait_path))
- used = TRUE
- icon_state = "mutation_syringe-empty"
+ uses--
+ update_icon_state()
+ user.balloon_alert_to_viewers("injected mutator")
to_chat(user, span_notice("You inject [target] with [src]."))
+/obj/item/slime_mutation_syringe/update_icon_state()
+ . = ..()
+ icon_state = uses ? initial(icon_state) : "[initial(icon_state)]-empty"
/obj/item/slime_mutation_syringe/cleaner
name = "cleaner slime mutation syringe"
@@ -64,19 +80,32 @@
icon = 'monkestation/code/modules/slimecore/icons/slimes.dmi'
icon_state = "mutation_syringe"
- /// have we been used?
- var/used = FALSE
+ /// Amount of uses remaining.
+ var/uses = 1
+/obj/item/slime_mutation_syringe_random/examine(mob/user)
+ . = ..()
+ if(uses)
+ . += span_notice("It has [uses] uses left.")
+ else
+ . += span_warning("It has been completely used up.")
/obj/item/slime_mutation_syringe_random/afterattack(atom/target, mob/user, proximity_flag, click_parameters)
. = ..()
- if(used)
+ if(!uses)
+ user.balloon_alert(user, "used up")
+ to_chat(user, span_warning("[src] has been completely used up!"))
return
if(!istype(target, /mob/living/basic/slime))
return
var/mob/living/basic/slime/slime = target
slime.start_mutating(TRUE)
- used = TRUE
- icon_state = "mutation_syringe-empty"
+ uses--
+ update_icon_state()
+ user.balloon_alert_to_viewers("injected mutator")
to_chat(user, span_notice("You inject [target] with [src]."))
+
+/obj/item/slime_mutation_syringe_random/update_icon_state()
+ . = ..()
+ icon_state = uses ? initial(icon_state) : "[initial(icon_state)]-empty"
diff --git a/monkestation/code/modules/slimecore/machines/biomass_recycler.dm b/monkestation/code/modules/slimecore/machines/biomass_recycler.dm
index be89b669ffe2..bdb0a50209e1 100644
--- a/monkestation/code/modules/slimecore/machines/biomass_recycler.dm
+++ b/monkestation/code/modules/slimecore/machines/biomass_recycler.dm
@@ -34,37 +34,38 @@ GLOBAL_LIST_INIT(biomass_unlocks, list())
power_change()
return TOOL_ACT_TOOLTYPE_SUCCESS
-/obj/machinery/biomass_recycler/attackby(obj/item/O, mob/user, params)
- if(default_deconstruction_screwdriver(user, "grinder_open", "grinder", O))
- return
-
- if(default_pry_open(O))
- return
-
- if(default_deconstruction_crowbar(O))
+/obj/machinery/biomass_recycler/attackby(obj/item/item, mob/user, params)
+ . = ..()
+ if(. || default_deconstruction_screwdriver(user, "grinder_open", "grinder", item) || default_pry_open(item) || default_deconstruction_crowbar(item) || machine_stat)
return
-
- if(machine_stat) //NOPOWER etc
+ if(istype(item, /obj/item/storage/bag/xeno))
+ var/total_biomass = 0
+ for(var/obj/item/stack/biomass/biomass in item)
+ total_biomass += biomass.amount
+ stored_matter += biomass.amount
+ qdel(biomass)
+ if(total_biomass > 0)
+ to_chat(user, span_notice("You dump [total_biomass] cube\s of biomass from [item] into [src]."))
+ user.balloon_alert_to_viewers("inserted biomass")
return
-
- if(HAS_TRAIT(O, TRAIT_NODROP))
+ else if(HAS_TRAIT(item, TRAIT_NODROP))
return
-
- if(istype(O, /obj/item/stack/biomass))
- var/obj/item/stack/biomass/biomass = O
+ else if(istype(item, /obj/item/stack/biomass))
+ var/obj/item/stack/biomass/biomass = item
to_chat(user, span_notice("You insert [biomass.amount] cube\s of biomass into [src]."))
+ user.balloon_alert_to_viewers("inserted biomass")
stored_matter += biomass.amount
qdel(biomass)
return
var/can_recycle
for(var/recycable_type in recyclable_types)
- if(istype(O, recycable_type))
+ if(istype(item, recycable_type))
can_recycle = recycable_type
break
if(can_recycle)
- recycle(O, user, can_recycle)
+ recycle(item, user, can_recycle)
/obj/machinery/biomass_recycler/MouseDrop_T(mob/living/target, mob/living/user)
if(!istype(target))
@@ -125,6 +126,7 @@ GLOBAL_LIST_INIT(biomass_unlocks, list())
var/spawn_type = item_names[pick]
if(stored_matter < printable_types[spawn_type])
to_chat(user, span_warning("[src] does not have enough stored biomass for that! It currently has [stored_matter] out of [printable_types[spawn_type]] unit\s required."))
+ balloon_alert(user, "not enough biomass")
return
var/spawned = new spawn_type(user.loc)
@@ -139,7 +141,7 @@ GLOBAL_LIST_INIT(biomass_unlocks, list())
icon = 'monkestation/code/modules/slimecore/icons/stack_objects.dmi'
icon_state = "biomass"
base_icon_state = "biomass"
- max_amount = 5
+ max_amount = 15
singular_name = "biomass cube"
merge_type = /obj/item/stack/biomass
flags_1 = CONDUCT_1
diff --git a/monkestation/code/modules/slimecore/machines/extract_requestor.dm b/monkestation/code/modules/slimecore/machines/extract_requestor.dm
index 1e254d48b130..faed4fff0690 100644
--- a/monkestation/code/modules/slimecore/machines/extract_requestor.dm
+++ b/monkestation/code/modules/slimecore/machines/extract_requestor.dm
@@ -6,8 +6,8 @@
base_icon_state = "civilian_pad"
density = TRUE
use_power = IDLE_POWER_USE
- idle_power_usage = 10
- active_power_usage = 2000
+ idle_power_usage = BASE_MACHINE_IDLE_CONSUMPTION
+ active_power_usage = BASE_MACHINE_ACTIVE_CONSUMPTION
circuit = /obj/item/circuitboard/machine/slime_extract_requestor
var/obj/machinery/computer/slime_market/console
var/list/current_requests = list()
diff --git a/monkestation/code/modules/slimecore/machines/ooze_compressor/ooze_compressor.dm b/monkestation/code/modules/slimecore/machines/ooze_compressor/ooze_compressor.dm
index 8baef0b70d46..5ed9b18e56d8 100644
--- a/monkestation/code/modules/slimecore/machines/ooze_compressor/ooze_compressor.dm
+++ b/monkestation/code/modules/slimecore/machines/ooze_compressor/ooze_compressor.dm
@@ -18,12 +18,13 @@
icon = 'monkestation/code/modules/slimecore/icons/machinery.dmi'
base_icon_state = "cross_compressor"
icon_state = "cross_compressor"
- category="Distribution"
+ category = "Distribution"
anchored = TRUE
- idle_power_usage = 10
- active_power_usage = 1000
+ use_power = IDLE_POWER_USE
+ idle_power_usage = BASE_MACHINE_IDLE_CONSUMPTION
+ active_power_usage = BASE_MACHINE_ACTIVE_CONSUMPTION
buffer = 5000
reagent_flags = NO_REACT
@@ -74,9 +75,12 @@
/obj/machinery/plumbing/ooze_compressor/add_context(atom/source, list/context, obj/item/held_item, mob/user)
. = ..()
- context[SCREENTIP_CONTEXT_ALT_LMB] = "Toggle Repeated Extract Compression"
- context[SCREENTIP_CONTEXT_LMB] = "Select a normal extract to make"
- context[SCREENTIP_CONTEXT_RMB] = "Select a crossbreed to make"
+ if(current_recipe)
+ context[SCREENTIP_CONTEXT_RMB] = "Cancel current recipe"
+ else
+ context[SCREENTIP_CONTEXT_LMB] = "Select a normal extract to make"
+ context[SCREENTIP_CONTEXT_RMB] = "Select a crossbreed to make"
+ context[SCREENTIP_CONTEXT_CTRL_LMB] = "[repeat_recipe ? "Disable" : "Enable"] repeated extract compression"
return CONTEXTUAL_SCREENTIP_SET
/obj/machinery/plumbing/ooze_compressor/create_reagents(max_vol, flags)
@@ -86,10 +90,7 @@
/obj/machinery/plumbing/ooze_compressor/update_icon_state()
. = ..()
- if(compressing)
- icon_state = "cross_compressor_running"
- else
- icon_state = base_icon_state
+ icon_state = compressing ? "cross_compressor_running" : base_icon_state
/obj/machinery/plumbing/ooze_compressor/examine(mob/user)
. = ..()
@@ -152,44 +153,73 @@
update_appearance()
if(holder.total_volume == 0 && !compressing) //we were emptying, but now we aren't
holder.flags |= NO_REACT
+ manage_hud_as_needed()
return NONE
-/obj/machinery/plumbing/ooze_compressor/process(seconds_per_tick)
- if(!compressing)
- use_power(active_power_usage * seconds_per_tick)
-
/obj/machinery/plumbing/ooze_compressor/proc/compress_recipe()
compressing = TRUE
update_appearance()
if(!repeat_recipe)
reagents_for_recipe = list()
+ manage_hud_as_needed()
+ update_power_usage()
addtimer(CALLBACK(src, PROC_REF(finish_compressing)), 3 SECONDS)
/obj/machinery/plumbing/ooze_compressor/proc/finish_compressing()
for(var/i in 1 to current_recipe.created_amount)
- new current_recipe.output_item(loc)
+ new current_recipe.output_item(drop_location())
compressing = FALSE
update_appearance()
reagents.clear_reagents()
if(!repeat_recipe)
current_recipe = null
+ update_power_usage()
+ manage_hud_as_needed()
+
+/obj/machinery/plumbing/ooze_compressor/proc/update_power_usage()
+ if(compressing)
+ update_use_power(ACTIVE_POWER_USE)
+ else if(current_recipe)
+ update_use_power(IDLE_POWER_USE)
+ else
+ update_use_power(NO_POWER_USE)
/obj/machinery/plumbing/ooze_compressor/attack_hand(mob/living/user, list/modifiers)
. = ..()
+ if(. || !can_interact(user))
+ return
+ if(!anchored)
+ balloon_alert(user, "unanchored!")
+ return TRUE
if(change_recipe(user))
reagents.clear_reagents()
+ return TRUE
/obj/machinery/plumbing/ooze_compressor/attack_hand_secondary(mob/living/user, list/modifiers)
. = ..()
- if(change_recipe(user, TRUE))
- reagents.clear_reagents()
+ if(. == SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN || !can_interact(user))
+ return
+ . = SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN
+ if(!anchored)
+ balloon_alert(user, "unanchored!")
+ return
+ if(current_recipe)
+ repeat_recipe = FALSE
+ if(!compressing)
+ current_recipe = null
+ reagents.clear_reagents()
+ update_power_usage()
+ balloon_alert_to_viewers("cancelled recipe")
+ else
+ if(change_recipe(user, TRUE))
+ reagents.clear_reagents()
-/obj/machinery/plumbing/ooze_compressor/AltClick(mob/user)
- if(anchored)
- visible_message(span_notice("[user] presses a button turning the repeat recipe system [repeat_recipe ? "Off" : "On"]"))
+/obj/machinery/plumbing/ooze_compressor/CtrlClick(mob/user)
+ if(anchored && can_interact(user))
repeat_recipe = !repeat_recipe
- return TRUE
- . = ..()
+ balloon_alert_to_viewers("[repeat_recipe ? "enabled" : "disabled"] repeating")
+ visible_message(span_notice("[user] presses a button turning the repeat recipe system [repeat_recipe ? span_green("on") : span_red("off")]"))
+ return ..()
/obj/machinery/plumbing/ooze_compressor/proc/change_recipe(mob/user, cross_breed = FALSE)
var/choice
@@ -201,14 +231,14 @@
else
choice = show_radial_menu(user, src, recipe_choices, require_near = TRUE, tooltips = TRUE)
- if(!(choice in choice_to_datum))
- return
-
- if(compressing)
+ if(compressing || !(choice in choice_to_datum))
return
current_recipe = choice_to_datum[choice]
reagents_for_recipe = list()
reagents_for_recipe += current_recipe.required_oozes
+ balloon_alert_to_viewers("set extract recipe")
+ manage_hud_as_needed()
+ update_power_usage()
#undef CROSSBREED_BASE_PATHS
diff --git a/monkestation/code/modules/slimecore/machines/ooze_compressor/ooze_compressor_hud.dm b/monkestation/code/modules/slimecore/machines/ooze_compressor/ooze_compressor_hud.dm
new file mode 100644
index 000000000000..bec098500afa
--- /dev/null
+++ b/monkestation/code/modules/slimecore/machines/ooze_compressor/ooze_compressor_hud.dm
@@ -0,0 +1,127 @@
+/obj/machinery/plumbing/ooze_compressor
+ var/image/hover_appearance
+ var/datum/atom_hud/alternate_appearance/basic/ooze_compressor/hover_popup
+
+/obj/machinery/plumbing/ooze_compressor/Destroy()
+ QDEL_NULL(hover_popup)
+ return ..()
+
+/obj/machinery/plumbing/ooze_compressor/MouseEntered(location, control, params)
+ . = ..()
+ if(!QDELETED(usr) && anchored)
+ manage_hud_as_needed()
+ hover_popup?.show_to(usr)
+
+/obj/machinery/plumbing/ooze_compressor/MouseExited(location, control, params)
+ . = ..()
+ if(!QDELETED(usr) && !QDELETED(hover_popup))
+ hover_popup.hide_from(usr)
+ manage_hud_as_needed(cleanup = TRUE)
+
+/obj/machinery/plumbing/ooze_compressor/set_anchored(anchorvalue)
+ . = ..()
+ if(!anchored)
+ manage_hud_as_needed()
+
+/obj/machinery/plumbing/ooze_compressor/proc/manage_hud_as_needed(cleanup = FALSE)
+ if(!anchored || (cleanup && !QDELETED(hover_popup) && !length(hover_popup.hud_users_all_z_levels)))
+ // don't bother keeping the hud around if it isn't needed
+ QDEL_NULL(hover_popup)
+ return
+ setup_hud()
+
+/obj/machinery/plumbing/ooze_compressor/proc/setup_hud()
+ // delete old hud if it exists and collect a list of its users
+ var/list/mob/old_users
+ if(!QDELETED(hover_popup))
+ old_users = hover_popup.hud_users_all_z_levels.Copy()
+ QDEL_NULL(hover_popup)
+
+ if(!length(GLOB.compressor_recipe_previews)) // we can't initialize this normally bc it will shit itself if initialized early
+ GLOB.compressor_recipe_previews = create_compressor_previews()
+
+ // setup new hover appearance
+ if(current_recipe)
+ hover_appearance = image(GLOB.compressor_recipe_previews[current_recipe.type], loc = src, layer = CHAT_LAYER)
+ hover_appearance.add_filter("extract_outline", 1, outline_filter(size = 1, color = COLOR_WHITE))
+ hover_appearance.pixel_y = 10
+ else
+ hover_appearance = image(loc = src, layer = CHAT_LAYER)
+ hover_appearance.pixel_y = 18
+ SET_PLANE_EXPLICIT(hover_appearance, HUD_PLANE, src)
+ hover_appearance.plane = HUD_PLANE
+ hover_appearance.appearance_flags = RESET_COLOR
+
+ // now setup the actual hud
+ hover_popup = add_alt_appearance(/datum/atom_hud/alternate_appearance/basic/ooze_compressor, "ooze_compressor", hover_appearance)
+ // and the cooldown maptext
+ refresh_info_maptext()
+
+ for(var/mob/old_user as anything in old_users)
+ if(QDELETED(old_user))
+ continue
+ hover_popup.show_to(old_user)
+
+/obj/machinery/plumbing/ooze_compressor/proc/refresh_info_maptext()
+ if(!hover_popup)
+ return
+ if(!hover_popup.info_maptext)
+ hover_popup.info_maptext = image(loc = src, layer = CHAT_LAYER + 0.1)
+ SET_PLANE_EXPLICIT(hover_popup.info_maptext, HUD_PLANE, src)
+ hover_popup.info_maptext.plane = HUD_PLANE
+ hover_popup.info_maptext.appearance_flags = APPEARANCE_UI_IGNORE_ALPHA | KEEP_APART
+ hover_popup.info_maptext.maptext_height = world.icon_size / 2
+ hover_popup.info_maptext.maptext_y = 4
+ var/maptext
+ if(compressing)
+ hover_popup.info_maptext.maptext_width = world.icon_size * 2
+ hover_popup.info_maptext.maptext_x = -(world.icon_size / 2)
+ maptext = "compressing..."
+ else
+ hover_popup.info_maptext.maptext_width = world.icon_size
+ hover_popup.info_maptext.maptext_x = 0
+ maptext = current_recipe ? "[get_progress()]%" : "inactive"
+ hover_popup.info_maptext.maptext = MAPTEXT_TINY_UNICODE("[maptext]")
+ hover_popup.give_info()
+
+/obj/machinery/plumbing/ooze_compressor/proc/get_progress()
+ if(!current_recipe || compressing)
+ return 0
+ var/current = 0
+ var/needed = 0
+ for(var/datum/reagent/reagent as anything in current_recipe.required_oozes)
+ needed += current_recipe.required_oozes[reagent] || 0
+ for(var/datum/reagent/listed_reagent as anything in reagents.reagent_list)
+ if(listed_reagent.type != reagent)
+ continue
+ current += listed_reagent.volume
+ return clamp(round((current / needed) * 100, 1), 0, 100)
+
+/datum/atom_hud/alternate_appearance/basic/ooze_compressor
+ var/image/info_maptext
+
+/datum/atom_hud/alternate_appearance/basic/ooze_compressor/show_to(mob/new_viewer)
+ . = ..()
+ if(info_maptext && !QDELETED(new_viewer) && !QDELETED(new_viewer.client))
+ new_viewer.client.images |= info_maptext
+
+/datum/atom_hud/alternate_appearance/basic/ooze_compressor/hide_from(mob/former_viewer, absolute)
+ . = ..()
+ if(info_maptext && !QDELETED(former_viewer) && !QDELETED(former_viewer.client))
+ former_viewer.client.images -= info_maptext
+
+/datum/atom_hud/alternate_appearance/basic/ooze_compressor/proc/give_info()
+ if(!info_maptext)
+ return
+ for(var/mob/user as anything in hud_users_all_z_levels)
+ if(QDELETED(user) || QDELETED(user.client))
+ continue
+ user.client.images |= info_maptext
+
+/datum/atom_hud/alternate_appearance/basic/ooze_compressor/proc/take_cooldowns()
+ if(!info_maptext)
+ return
+ for(var/mob/user as anything in hud_users_all_z_levels)
+ if(QDELETED(user) || QDELETED(user.client))
+ continue
+ user.client.images -= info_maptext
diff --git a/monkestation/code/modules/slimecore/machines/ooze_compressor/recipes/_base_recipe.dm b/monkestation/code/modules/slimecore/machines/ooze_compressor/recipes/_base_recipe.dm
index a60288732cb3..dbc827f25b35 100644
--- a/monkestation/code/modules/slimecore/machines/ooze_compressor/recipes/_base_recipe.dm
+++ b/monkestation/code/modules/slimecore/machines/ooze_compressor/recipes/_base_recipe.dm
@@ -1,6 +1,18 @@
+GLOBAL_LIST_EMPTY_TYPED(compressor_recipe_previews, /image)
+
/datum/compressor_recipe
var/list/required_oozes = list()
var/obj/item/output_item
var/created_amount = 1
/datum/compressor_recipe/crossbreed
+
+/proc/create_compressor_previews()
+ . = list()
+ for(var/datum/compressor_recipe/recipe as anything in subtypesof(/datum/compressor_recipe))
+ var/output_type = recipe::output_item
+ if(!ispath(output_type, /obj/item))
+ continue
+ var/obj/item/preview = new output_type(null)
+ .[recipe] = image(getFlatIcon(preview))
+ qdel(preview)
diff --git a/monkestation/code/modules/slimecore/machines/ooze_sucker.dm b/monkestation/code/modules/slimecore/machines/ooze_sucker.dm
index 216edb7dfdbe..69f2f497ee76 100644
--- a/monkestation/code/modules/slimecore/machines/ooze_sucker.dm
+++ b/monkestation/code/modules/slimecore/machines/ooze_sucker.dm
@@ -7,10 +7,13 @@
icon_state = "ooze_sucker"
anchored = FALSE
density = FALSE
- idle_power_usage = 10
- active_power_usage = 1000
+
+ use_power = IDLE_POWER_USE
+ idle_power_usage = BASE_MACHINE_IDLE_CONSUMPTION * 0.1
+ active_power_usage = BASE_MACHINE_ACTIVE_CONSUMPTION * 0.15
+
buffer = 3000
- category="Distribution"
+ category = "Distribution"
reagent_flags = NO_REACT
/// Pump is turned on by engineer, etc.
@@ -31,6 +34,9 @@
/// Additional ratio of liquid volume to drain
var/drain_percent = 1
+ /// Whether draining was performed last process or not.
+ var/drained_last_process = FALSE
+
/obj/machinery/plumbing/ooze_sucker/Initialize(mapload, bolt, layer)
. = ..()
AddComponent(/datum/component/plumbing/simple_supply, bolt, layer)
@@ -68,6 +74,22 @@
turned_on = FALSE
update_icon_state()
+/obj/machinery/plumbing/ooze_sucker/create_reagents(max_vol, flags)
+ . = ..()
+ RegisterSignals(reagents, list(COMSIG_REAGENTS_REM_REAGENT, COMSIG_REAGENTS_DEL_REAGENT, COMSIG_REAGENTS_CLEAR_REAGENTS, COMSIG_REAGENTS_REACTED), PROC_REF(on_reagent_change))
+ RegisterSignal(reagents, COMSIG_QDELETING, PROC_REF(on_reagents_del))
+
+/// Handles properly detaching signal hooks.
+/obj/machinery/plumbing/ooze_sucker/proc/on_reagents_del(datum/reagents/reagents)
+ SIGNAL_HANDLER
+ UnregisterSignal(reagents, list(COMSIG_REAGENTS_REM_REAGENT, COMSIG_REAGENTS_DEL_REAGENT, COMSIG_REAGENTS_CLEAR_REAGENTS, COMSIG_REAGENTS_REACTED, COMSIG_QDELETING))
+ return NONE
+
+/// Handles ensuring power usage becomes idle when reagents are full.
+/obj/machinery/plumbing/ooze_sucker/proc/on_reagent_change(datum/reagents/holder, ...)
+ SIGNAL_HANDLER
+ update_power_usage()
+ return NONE
/obj/machinery/plumbing/ooze_sucker/proc/toggle_state()
turned_on = !turned_on
@@ -109,12 +131,25 @@
// We're good, actually pump.
for(var/turf/affected_turf as anything in affected_turfs)
pump_turf(affected_turf, seconds_per_tick, multiplier)
+ update_power_usage()
+
+/obj/machinery/plumbing/ooze_sucker/proc/update_power_usage()
+ if(!turned_on)
+ update_use_power(NO_POWER_USE)
+ else if(reagents && reagents.total_volume >= reagents.maximum_volume)
+ update_use_power(IDLE_POWER_USE)
+ else if(drained_last_process)
+ update_use_power(ACTIVE_POWER_USE)
+ else
+ update_use_power(IDLE_POWER_USE)
/obj/machinery/plumbing/ooze_sucker/proc/pump_turf(turf/affected_turf, seconds_per_tick, multiplier)
if(processes < processes_required)
processes++
return
processes = 0
+ drained_last_process = FALSE
+
if(!affected_turf.liquids || !affected_turf.liquids.liquid_group)
return
@@ -128,3 +163,12 @@
if(!targeted_group.reagents_per_turf)
return
targeted_group.transfer_specific_reagents(reagents, target_value, reagents_to_check = typesof(/datum/reagent/slime_ooze), merge = TRUE)
+ drained_last_process = TRUE
+
+/obj/machinery/plumbing/ooze_sucker/attack_hand_secondary(mob/user, list/modifiers)
+ . = ..()
+ if(. == SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN)
+ return
+ toggle_state()
+ balloon_alert_to_viewers("[turned_on ? "enabled" : "disabled"] ooze sucker")
+ return SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN
diff --git a/monkestation/code/modules/slimecore/machines/slime_grinder.dm b/monkestation/code/modules/slimecore/machines/slime_grinder.dm
index 4cb79b472b19..b5ef7280e7c1 100644
--- a/monkestation/code/modules/slimecore/machines/slime_grinder.dm
+++ b/monkestation/code/modules/slimecore/machines/slime_grinder.dm
@@ -7,8 +7,11 @@
icon = 'monkestation/code/modules/slimecore/icons/slime_grinder.dmi'
icon_state = "slime_grinder_backdrop"
base_icon_state = "slime_grinder_backdrop"
- idle_power_usage = 10
- active_power_usage = 1000
+
+ use_power = IDLE_POWER_USE
+ idle_power_usage = BASE_MACHINE_IDLE_CONSUMPTION
+ active_power_usage = BASE_MACHINE_ACTIVE_CONSUMPTION
+
buffer = 3000
category="Distribution"
diff --git a/monkestation/code/modules/slimecore/machines/slime_market.dm b/monkestation/code/modules/slimecore/machines/slime_market.dm
index 8e299bec9cda..57cd96c8397b 100644
--- a/monkestation/code/modules/slimecore/machines/slime_market.dm
+++ b/monkestation/code/modules/slimecore/machines/slime_market.dm
@@ -6,25 +6,11 @@
base_icon_state = "market_pad"
density = TRUE
use_power = IDLE_POWER_USE
- idle_power_usage = 10
- active_power_usage = 2000
+ idle_power_usage = BASE_MACHINE_IDLE_CONSUMPTION
+ active_power_usage = BASE_MACHINE_ACTIVE_CONSUMPTION
circuit = /obj/item/circuitboard/machine/slime_market_pad
var/obj/machinery/computer/slime_market/console
-/obj/machinery/slime_market_pad/attackby(obj/item/I, mob/user, params)
- if(default_deconstruction_screwdriver(user, icon_state, icon_state, I))
- user.visible_message(span_notice("\The [user] [panel_open ? "opens" : "closes"] the hatch on \the [src]."), span_notice("You [panel_open ? "open" : "close"] the hatch on \the [src]."))
- update_appearance()
- return TRUE
-
- if(default_unfasten_wrench(user, I))
- return TRUE
-
- if(default_deconstruction_crowbar(I))
- return TRUE
-
- . = ..()
-
/obj/machinery/slime_market_pad/examine(mob/user)
. = ..()
if(!panel_open)
@@ -41,6 +27,8 @@
/obj/machinery/slime_market_pad/AltClick(mob/user)
. = ..()
+ if(!.)
+ return
link_console()
/obj/machinery/slime_market_pad/proc/link_console()
@@ -53,30 +41,39 @@
console.link_market_pad()
break
-/obj/machinery/slime_market_pad/attackby(obj/item/I, mob/living/user, params)
+/obj/machinery/slime_market_pad/attackby(obj/item/item, mob/living/user, params)
. = ..()
- if(!console)
+ if(. || !can_interact(user))
+ return
+ if(default_deconstruction_screwdriver(user, icon_state, icon_state, item))
+ user.visible_message(span_notice("\The [user] [panel_open ? "opens" : "closes"] the hatch on \the [src]."), span_notice("You [panel_open ? "open" : "close"] the hatch on \the [src]."))
+ update_appearance()
+ return TRUE
+ if(default_unfasten_wrench(user, item) || default_deconstruction_crowbar(item))
+ return TRUE
+ if(QDELETED(console))
to_chat(user, span_warning("[src] does not have a console linked to it!"))
return
-
- if(istype(I, /obj/item/slime_extract))
- var/obj/item/slime_extract/extract = I
+ if(istype(item, /obj/item/slime_extract))
+ var/obj/item/slime_extract/extract = item
if(extract.tier == 0)
to_chat(user, span_warning("[src] doesn't seem to accept this extract!"))
return
flick("[base_icon_state]_vend", src)
sell_extract(extract)
return
-
- else if(istype(I, /obj/item/storage/bag/xeno))
- if(tgui_alert(user, "Are you sure you want to sell all extracts from [I]?", "<3?", list("Yes", "No")) != "Yes")
+ else if(istype(item, /obj/item/storage/bag/xeno))
+ if(tgui_alert(user, "Are you sure you want to sell all extracts from [item]?", "<3?", list("Yes", "No")) != "Yes")
return
-
flick("[base_icon_state]_vend", src)
- for(var/obj/item/slime_extract/extract in I)
+ var/sold_extracts = 0
+ for(var/obj/item/slime_extract/extract in item)
if(extract.tier == 0)
continue
sell_extract(extract)
+ sold_extracts++
+ if(sold_extracts > 0)
+ user.balloon_alert_to_viewers("sold [sold_extracts] extracts")
return
/obj/machinery/slime_market_pad/proc/sell_extract(obj/item/slime_extract/extract)
@@ -97,23 +94,38 @@
SSresearch.slime_core_prices[core_type] = (1 + price_mod * price_limiter) * SSresearch.slime_core_prices[core_type]
qdel(extract)
-/obj/machinery/slime_market_pad/attackby_secondary(obj/item/weapon, mob/user, params)
- if(!console)
- to_chat(user, span_warning("[src] does not have a console linked to it!"))
+/obj/machinery/slime_market_pad/attackby_secondary(obj/item/item, mob/user, params)
+ . = ..()
+ if(. == SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN || !can_interact(user))
return
+ if(QDELETED(console))
+ to_chat(user, span_warning("[src] does not have a console linked to it!"))
+ return SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN
- if(!console.request_pad)
+ if(istype(item, /obj/item/storage/bag/xeno))
+ flick("[base_icon_state]_vend", src)
+ var/sold_extracts = 0
+ for(var/obj/item/slime_extract/extract in item)
+ if(extract.tier == 0)
+ continue
+ sell_extract(extract)
+ sold_extracts++
+ if(sold_extracts > 0)
+ user.balloon_alert_to_viewers("sold [sold_extracts] extracts")
+ return SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN
+
+ if(QDELETED(console.request_pad))
to_chat(user, span_warning("[console] does not have a request_pad linked to it!"))
- return
+ return SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN
if(!length(console.request_pad.current_requests))
to_chat(user, span_warning("There are no current extract requests!"))
- return
+ return SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN
- if(istype(weapon, /obj/item/slime_extract))
+ if(istype(item, /obj/item/slime_extract))
var/list/radial_choices = list()
var/list/choice_to_request = list()
- var/obj/item/slime_extract/extract = weapon
+ var/obj/item/slime_extract/extract = item
for(var/datum/extract_request_data/current as anything in console.request_pad.current_requests)
if((current.extract_path != extract.type) || current.ready_for_pickup)
continue
@@ -134,4 +146,4 @@
flick("[base_icon_state]_vend", src)
qdel(extract)
- return
+ return SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN
diff --git a/monkestation/code/modules/slimecore/machines/slime_market_computer.dm b/monkestation/code/modules/slimecore/machines/slime_market_computer.dm
index 3c86eb44af7a..a3394c7a0105 100644
--- a/monkestation/code/modules/slimecore/machines/slime_market_computer.dm
+++ b/monkestation/code/modules/slimecore/machines/slime_market_computer.dm
@@ -66,7 +66,7 @@ GLOBAL_DATUM(default_slime_market, /obj/machinery/computer/slime_market)
/obj/machinery/computer/slime_market/ui_assets(mob/user)
return list(
- get_asset_datum(/datum/asset/spritesheet/xenobio_market),
+ get_asset_datum(/datum/asset/spritesheet/xenobio_market)
)
/obj/machinery/computer/slime_market/ui_interact(mob/user, datum/tgui/ui)
@@ -110,19 +110,21 @@ GLOBAL_DATUM(default_slime_market, /obj/machinery/computer/slime_market)
iter += 1
var/obj/item/slime_extract/core = core_type
- var/list/core_data = list("icon" = "[initial(core.icon_state)]",
- "price" = SSresearch.slime_core_prices[core_type],
- "key" = iter % 4,
- )
+ var/list/core_data = list(
+ "icon" = "[initial(core.icon_state)]",
+ "price" = SSresearch.slime_core_prices[core_type],
+ "key" = iter % 4
+ )
price_row.Add(list(core_data))
iter += 1
if(core_type == /obj/item/slime_extract/grey)
core = /obj/item/slime_extract/rainbow
- var/list/rainbow_core_data = list("icon" = "[initial(core.icon_state)]",
- "price" = SSresearch.slime_core_prices[/obj/item/slime_extract/rainbow],
- "key" = iter % 4,
- )
+ var/list/rainbow_core_data = list(
+ "icon" = "[initial(core.icon_state)]",
+ "price" = SSresearch.slime_core_prices[/obj/item/slime_extract/rainbow],
+ "key" = iter % 4
+ )
price_row.Add(list(rainbow_core_data))
iter += 1
price_row.Add(list(list("key" = iter % 4)))
diff --git a/monkestation/code/modules/slimecore/machines/slime_pen_controller.dm b/monkestation/code/modules/slimecore/machines/slime_pen_controller.dm
index e071ac929ecb..862b41467872 100644
--- a/monkestation/code/modules/slimecore/machines/slime_pen_controller.dm
+++ b/monkestation/code/modules/slimecore/machines/slime_pen_controller.dm
@@ -28,7 +28,6 @@
. = ..()
locate_machinery()
-
/obj/machinery/slime_pen_controller/add_context(atom/source, list/context, obj/item/held_item, mob/user)
. = ..()
@@ -69,13 +68,13 @@
var/list/mutation_info = list()
var/mob_string
for(var/mob/living/mob as anything in mutation_data.latch_needed)
- mob_string += "[mutation_data.latch_needed[mob]] units of genetic data from [initial(mob.name)]. \n"
+ mob_string += "[mutation_data.latch_needed[mob]] units of genetic data from [mob::name]. \n"
var/item_string
for(var/obj/item/item as anything in mutation_data.needed_items)
- item_string += "[initial(item.name)]. \n"
+ item_string += "[item:name]. \n"
mutation_info += list(
- "color" = capitalize(initial(mutation_data.output.name)),
+ "color" = capitalize(mutation_data.output::name),
"weight" = mutation_data.weight,
"mutate_chance" = mutation_data.mutate_probability,
"mobs_needed" = mob_string,
@@ -135,7 +134,6 @@
. = ..()
if(.)
return
-
switch(action)
if("buy")
for(var/datum/corral_upgrade/item as anything in subtypesof(/datum/corral_upgrade))
@@ -146,7 +144,7 @@
/obj/machinery/slime_pen_controller/proc/try_buy(datum/corral_upgrade/item)
if(!linked_data)
return
- if(SSresearch.xenobio_points < initial(item.cost))
+ if(SSresearch.xenobio_points < item::cost)
return
var/datum/corral_upgrade/new_upgrade = new item
@@ -165,25 +163,23 @@
return
/obj/machinery/slime_pen_controller/attack_hand_secondary(mob/user, list/modifiers)
- if(linked_sucker)
- visible_message(span_notice("[user] fiddles with the [src] toggling the pens ooze sucker."))
- linked_sucker.toggle_state()
- return TRUE
. = ..()
-
-/obj/machinery/slime_pen_controller/attackby(obj/item/weapon, mob/user, params)
- if(weapon.tool_behaviour == TOOL_MULTITOOL)
- if(!multitool_check_buffer(user, weapon))
- return
- var/obj/item/multitool/M = weapon
- if(!M.buffer)
- return
- var/obj/machinery/corral_corner/pad = M.buffer
- if(!istype(pad))
- return
- if(!pad.connected_data)
- return
- linked_data = pad.connected_data
- to_chat(user, span_notice("You link the [pad] to the [src]."))
+ if(. == SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN)
return
+ if(!QDELETED(linked_sucker))
+ linked_sucker.toggle_state()
+ balloon_alert_to_viewers("[linked_sucker.turned_on ? "enabled" : "disabled"] ooze sucker")
+ visible_message(span_notice("[user] fiddles with the [src], [linked_sucker.turned_on ? "enabling" : "disabling"] the pens ooze sucker."))
+ return SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN
+
+/obj/machinery/slime_pen_controller/multitool_act(mob/living/user, obj/item/multitool/multitool)
. = ..()
+ if(!multitool_check_buffer(user, multitool) || QDELETED(multitool.buffer))
+ return
+ var/obj/machinery/corral_corner/pad = multitool.buffer
+ if(!istype(pad) || !pad.connected_data)
+ return
+ linked_data = pad.connected_data
+ balloon_alert_to_viewers("linked pad")
+ pad.balloon_alert_to_viewers("linked to controller")
+ to_chat(user, span_notice("You link the [pad] to the [src]."))
diff --git a/monkestation/code/modules/slimecore/mobs/_base_slime.dm b/monkestation/code/modules/slimecore/mobs/_base_slime.dm
index dfbe5d102b8a..5b21399ca0ba 100644
--- a/monkestation/code/modules/slimecore/mobs/_base_slime.dm
+++ b/monkestation/code/modules/slimecore/mobs/_base_slime.dm
@@ -162,6 +162,7 @@
/mob/living/basic/slime/mob_try_pickup(mob/living/user, instant)
if(!SEND_SIGNAL(src, COMSIG_FRIENDSHIP_CHECK_LEVEL, user, FRIENDSHIP_FRIEND))
to_chat(user, span_notice("[src] doesn't trust you enough to let you pick them up"))
+ balloon_alert(user, "not enough trust!")
return FALSE
. = ..()
@@ -188,10 +189,9 @@
/mob/living/basic/slime/proc/recompile_ai_tree()
var/list/new_planning_subtree = list()
+ RemoveElement(/datum/element/basic_eating, food_types = compiled_liked_foods)
rebuild_foods()
- RemoveElement(/datum/element/basic_eating)
-
new_planning_subtree |= add_or_replace_tree(/datum/ai_planning_subtree/pet_planning)
if(!HAS_TRAIT(src, TRAIT_SLIME_RABID))
@@ -310,6 +310,7 @@
slime_flags |= SPLITTING_SLIME
visible_message(span_notice("[name] starts to flatten, it looks to be splitting."))
+ balloon_alert_to_viewers("splitting...")
addtimer(CALLBACK(src, PROC_REF(finish_splitting)), 15 SECONDS)
@@ -331,6 +332,7 @@
ai_controller.set_ai_status(AI_STATUS_OFF)
visible_message(span_notice("[name] starts to undulate, it looks to be mutating."))
+ balloon_alert_to_viewers("mutating...")
slime_flags |= MUTATING_SLIME
ungulate()
@@ -414,8 +416,9 @@
. = ..()
if(worn_accessory)
visible_message("[user] takes the [worn_accessory] off the [src].")
+ balloon_alert_to_viewers("removed accessory")
+ worn_accessory.forceMove(user.drop_location())
worn_accessory = null
- worn_accessory.forceMove(get_turf(user))
update_appearance()
/mob/living/basic/slime/Life(seconds_per_tick, times_fired)
diff --git a/monkestation/code/modules/trading/lootbox_odds.dm b/monkestation/code/modules/trading/lootbox_odds.dm
index 33619926ab6c..18773422b2bb 100644
--- a/monkestation/code/modules/trading/lootbox_odds.dm
+++ b/monkestation/code/modules/trading/lootbox_odds.dm
@@ -32,8 +32,9 @@
user.client.client_token_holder.adjust_antag_tokens(LOW_THREAT, 1)
if("Loadout Item")
- var/static/list/viable_types = list()
- if(!length(viable_types))
+ var/static/list/viable_types
+ if(!viable_types)
+ viable_types = list()
for(var/datum/loadout_item/type as anything in subtypesof(/datum/loadout_item))
var/datum/loadout_item/listed = new type()
if(!istype(listed))
diff --git a/monkestation/code/modules/viking/viking_armour.dm b/monkestation/code/modules/viking/viking_armour.dm
index 9c5bacd94e5e..046378082285 100644
--- a/monkestation/code/modules/viking/viking_armour.dm
+++ b/monkestation/code/modules/viking/viking_armour.dm
@@ -97,6 +97,18 @@
clothing_flags = STOPSPRESSUREDAMAGE
female_sprite_flags = FEMALE_UNIFORM_TOP_ONLY
+/obj/item/clothing/under/viking/fake_tunic
+ name = "viking pelt"
+ desc = "A faux wolf pelt made to cosplay with"
+ worn_icon_state = "pelts"
+ icon_state = "pelts"
+ w_class = WEIGHT_CLASS_NORMAL
+
+/obj/item/clothing/head/viking/fake_helmet
+ name = "foam helmet"
+ desc = "a foam helmet that looks like that of a vikings helmet"
+ worn_icon_state = "honeless_helm_worn"
+ icon_state = "hornless_helm_item"
/datum/armor/viking
melee = 45
bullet = 30
diff --git a/monkestation/icons/mob/anime/anime_bottom.dmi b/monkestation/icons/mob/anime/anime_bottom.dmi
index bc6d43e412f0..03db82988211 100644
Binary files a/monkestation/icons/mob/anime/anime_bottom.dmi and b/monkestation/icons/mob/anime/anime_bottom.dmi differ
diff --git a/monkestation/icons/mob/clock_cult/actions_clock.dmi b/monkestation/icons/mob/clock_cult/actions_clock.dmi
index 76cda1eed569..a6b6f3832af3 100644
Binary files a/monkestation/icons/mob/clock_cult/actions_clock.dmi and b/monkestation/icons/mob/clock_cult/actions_clock.dmi differ
diff --git a/tgstation.dme b/tgstation.dme
index 9221c60952f6..b780c5fe2a5d 100644
--- a/tgstation.dme
+++ b/tgstation.dme
@@ -6422,6 +6422,7 @@
#include "monkestation\code\modules\mob\living\basic\animatronic.dm"
#include "monkestation\code\modules\mob\living\basic\ggg\glerm.dm"
#include "monkestation\code\modules\mob\living\basic\ggg\susflash.dm"
+#include "monkestation\code\modules\mob\living\basic\pets\parrot\parrot_ai\parroting_action.dm"
#include "monkestation\code\modules\mob\living\basic\space_fauna\fugu_gland.dm"
#include "monkestation\code\modules\mob\living\basic\vermin\mouse.dm"
#include "monkestation\code\modules\mob\living\carbon\carbon_death.dm"
@@ -6768,6 +6769,7 @@
#include "monkestation\code\modules\slimecore\machines\slime_market_computer.dm"
#include "monkestation\code\modules\slimecore\machines\slime_pen_controller.dm"
#include "monkestation\code\modules\slimecore\machines\ooze_compressor\ooze_compressor.dm"
+#include "monkestation\code\modules\slimecore\machines\ooze_compressor\ooze_compressor_hud.dm"
#include "monkestation\code\modules\slimecore\machines\ooze_compressor\ooze_compressor_plumbing.dm"
#include "monkestation\code\modules\slimecore\machines\ooze_compressor\shitcode.dm"
#include "monkestation\code\modules\slimecore\machines\ooze_compressor\recipes\_base_recipe.dm"