mirror of
https://github.com/Bubberstation/Bubberstation.git
synced 2026-06-06 06:40:42 +01:00
5057e65041
## About The Pull Request Adds some balloon alerts to the mop, /datum/component/cleaner, and the light replacer edits some balloon alerts on the janicart ## Why It's Good For The Game Doing your duties as a janitor tends to clog up chat, and this stuff seems better for balloon alerts anyway. Newer players are also often directed towards janitor, and this will help with feedback for someone unaccustomed to looking at chat. If any of these are too long let me know. ## Changelog 🆑 Seven qol: Mops, some cleaning items, and light replacers now use balloon alerts /🆑
340 lines
11 KiB
Plaintext
340 lines
11 KiB
Plaintext
|
|
// Light Replacer (LR)
|
|
//
|
|
// ABOUT THE DEVICE
|
|
//
|
|
// This is a device supposedly to be used by Janitors and Janitor Cyborgs which will
|
|
// allow them to easily replace lights. This was mostly designed for Janitor Cyborgs since
|
|
// they don't have hands or a way to replace lightbulbs.
|
|
//
|
|
// HOW IT WORKS
|
|
//
|
|
// You attack a light fixture with it, if the light fixture is broken it will replace the
|
|
// light fixture with a working light; the broken light is then placed on the floor for the
|
|
// user to then pickup with a trash bag. If it's empty then it will just place a light in the fixture.
|
|
//
|
|
// HOW TO REFILL THE DEVICE
|
|
//
|
|
// It will need to be manually refilled with lights.
|
|
// If it's part of a robot module, it will charge when the Robot is inside a Recharge Station.
|
|
//
|
|
// EMAGGED FEATURES
|
|
//
|
|
// I'm not sure everyone will react the emag's features so please say what your opinions are of it. (I'm pretty sure the players like it)
|
|
//
|
|
// When emagged it will rig every light it replaces with plasma, which will slowly heat up and ignite while the light is on.
|
|
// This is VERY noticable, even the device's name changes when you emag it so if anyone
|
|
// examines you when you're holding it in your hand, you will be discovered.
|
|
//
|
|
|
|
#define GLASS_SHEET_USES 5
|
|
#define LIGHTBULB_COST 1
|
|
#define BULB_SHARDS_REQUIRED 4
|
|
|
|
/obj/item/lightreplacer
|
|
name = "light replacer"
|
|
desc = "A device to automatically replace lights. Refill with broken or working light bulbs, or sheets of glass."
|
|
icon = 'icons/obj/janitor.dmi'
|
|
icon_state = "lightreplacer"
|
|
inhand_icon_state = "electronic"
|
|
worn_icon_state = "light_replacer"
|
|
lefthand_file = 'icons/mob/inhands/items/devices_lefthand.dmi'
|
|
righthand_file = 'icons/mob/inhands/items/devices_righthand.dmi'
|
|
w_class = WEIGHT_CLASS_SMALL
|
|
flags_1 = CONDUCT_1
|
|
slot_flags = ITEM_SLOT_BELT
|
|
force = 8
|
|
|
|
/// How many uses does our light replacer have?
|
|
var/uses = 10
|
|
/// The maximum number of lights this replacer can hold
|
|
var/max_uses = 20
|
|
/// The light replacer's charge increment (used for adding to cyborg light replacers)
|
|
var/charge = 1
|
|
|
|
/// Eating used bulbs gives us bulb shards. Requires BULB_SHARDS_MAXIMUM to produce a new light.
|
|
var/bulb_shards = 0
|
|
|
|
/// whether it is "bluespace powered" (can be used at a range)
|
|
var/bluespace_toggle = FALSE
|
|
|
|
/obj/item/lightreplacer/examine(mob/user)
|
|
. = ..()
|
|
. += status_string()
|
|
|
|
/obj/item/lightreplacer/pre_attack(atom/target, mob/living/user, params)
|
|
. = ..()
|
|
if(.)
|
|
return
|
|
return do_action(target, user) //if we are attacking a valid target[light, floodlight or turf] stop here
|
|
|
|
/obj/item/lightreplacer/attackby(obj/item/insert, mob/user, params)
|
|
. = ..()
|
|
if(uses >= max_uses)
|
|
user.balloon_alert(user, "already full!")
|
|
return TRUE
|
|
|
|
if(istype(insert, /obj/item/stack/sheet/glass))
|
|
var/obj/item/stack/sheet/glass/glass_to_insert = insert
|
|
if(glass_to_insert.use(LIGHTBULB_COST))
|
|
add_uses(GLASS_SHEET_USES)
|
|
user.balloon_alert(user, "glass inserted")
|
|
else
|
|
user.balloon_alert(user, "need [LIGHTBULB_COST] glass sheets!")
|
|
return TRUE
|
|
|
|
if(insert.type == /obj/item/shard) //we don't want to insert plasma, titanium or other types of shards
|
|
if(!user.temporarilyRemoveItemFromInventory(insert))
|
|
user.balloon_alert(user, "stuck in your hand!")
|
|
return TRUE
|
|
if(!add_shard(user)) //add_shard will display a message if it created a bulb from the shard so only display message when that does not happen
|
|
user.balloon_alert(user, "shard inserted")
|
|
qdel(insert)
|
|
return TRUE
|
|
|
|
if(istype(insert, /obj/item/light))
|
|
var/obj/item/light/light_to_insert = insert
|
|
//remove from player's hand
|
|
if(!user.temporarilyRemoveItemFromInventory(light_to_insert))
|
|
user.balloon_alert(user, "stuck in your hand!")
|
|
return TRUE
|
|
|
|
//insert light. display message only if adding a shard did not create a new bulb else the messages will conflict
|
|
var/display_msg = TRUE
|
|
if(light_to_insert.status == LIGHT_OK)
|
|
add_uses(1)
|
|
else if(add_shard(user))
|
|
display_msg = FALSE
|
|
if(display_msg)
|
|
user.balloon_alert(user, "light inserted")
|
|
qdel(light_to_insert)
|
|
|
|
return TRUE
|
|
|
|
if(istype(insert, /obj/item/storage))
|
|
var/replaced_something = FALSE
|
|
var/loaded = FALSE
|
|
|
|
var/obj/item/storage/storage_to_empty = insert
|
|
for(var/obj/item/item_to_check in storage_to_empty.contents)
|
|
//reached max capacity during insertion
|
|
if(src.uses >= max_uses)
|
|
break
|
|
|
|
//consume the item only if it's an light tube,bulb or shard
|
|
loaded = FALSE
|
|
if(istype(item_to_check, /obj/item/light))
|
|
var/obj/item/light/found_light = item_to_check
|
|
if(found_light.status == LIGHT_OK)
|
|
add_uses(1)
|
|
else
|
|
add_shard(user)
|
|
loaded = TRUE
|
|
else if(istype(item_to_check, /obj/item/stack/sheet/glass))
|
|
var/obj/item/stack/sheet/glass/glass_to_insert = item_to_check
|
|
if(glass_to_insert.use(LIGHTBULB_COST))
|
|
add_uses(GLASS_SHEET_USES)
|
|
loaded = TRUE
|
|
else if(item_to_check.type == /obj/item/shard)
|
|
add_shard(user)
|
|
loaded = TRUE
|
|
|
|
//if item was loaded delete it
|
|
if(loaded)
|
|
qdel(item_to_check)
|
|
replaced_something = TRUE
|
|
|
|
if(!replaced_something)
|
|
if(uses == max_uses)
|
|
user.balloon_alert(user, "already full!")
|
|
else
|
|
user.balloon_alert(user, "nothing usable in [storage_to_empty]!")
|
|
return TRUE
|
|
|
|
user.balloon_alert(user, "lights inserted")
|
|
return TRUE
|
|
|
|
/obj/item/lightreplacer/emag_act()
|
|
if(obj_flags & EMAGGED)
|
|
return
|
|
obj_flags |= EMAGGED
|
|
playsound(loc, SFX_SPARKS, 100, TRUE, SHORT_RANGE_SOUND_EXTRARANGE)
|
|
update_appearance()
|
|
|
|
/obj/item/lightreplacer/update_name(updates)
|
|
. = ..()
|
|
name = (obj_flags & EMAGGED) ? "shortcircuited [initial(name)]" : initial(name)
|
|
|
|
/obj/item/lightreplacer/update_icon_state()
|
|
icon_state = "[initial(icon_state)][(obj_flags & EMAGGED ? "-emagged" : "")]"
|
|
return ..()
|
|
|
|
/obj/item/lightreplacer/vv_edit_var(vname, vval)
|
|
if(vname == NAMEOF(src, obj_flags))
|
|
update_appearance()
|
|
else if(vname == NAMEOF(src, uses))
|
|
uses = clamp(vval, 0, max_uses)
|
|
return TRUE
|
|
else if(vname == NAMEOF(src, max_uses))
|
|
if(vval <= 0)
|
|
return FALSE
|
|
max_uses = vval
|
|
uses = clamp(uses, 0, max_uses)
|
|
return TRUE
|
|
else if(vname == NAMEOF(src, bulb_shards))
|
|
if(vval <= 0)
|
|
return FALSE
|
|
add_uses(round(vval / BULB_SHARDS_REQUIRED))
|
|
bulb_shards = vval % BULB_SHARDS_REQUIRED
|
|
return TRUE
|
|
return ..()
|
|
|
|
/obj/item/lightreplacer/attack_self(mob/user)
|
|
var/on_a_light = FALSE //For if we are on the same tile as a light when we do this
|
|
for(var/obj/machinery/light/target in user.loc)
|
|
replace_light(target, user)
|
|
on_a_light = TRUE
|
|
if(!on_a_light) //So we dont give a ballon alert when we just used replace_light
|
|
user.balloon_alert(user, "[uses] lights, [bulb_shards]/[BULB_SHARDS_REQUIRED] fragments")
|
|
|
|
/**
|
|
* attempts to fix lights, flood lights & lights on a turf
|
|
* Arguments
|
|
* * target - the target we are trying to fix
|
|
* * user - the mob performing this action
|
|
* returns TRUE if the target was valid[light, floodlight or turf] regardless if any light's were fixed or not
|
|
*/
|
|
/obj/item/lightreplacer/proc/do_action(atom/target, mob/user)
|
|
// if we are attacking an light fixture then replace it directly
|
|
if(istype(target, /obj/machinery/light))
|
|
if(replace_light(target, user) && bluespace_toggle)
|
|
user.Beam(target, icon_state = "rped_upgrade", time = 0.5 SECONDS)
|
|
playsound(src, 'sound/items/pshoom.ogg', 40, 1)
|
|
return TRUE
|
|
|
|
// if we are attacking a floodlight frame finish it
|
|
if(istype(target, /obj/structure/floodlight_frame))
|
|
var/obj/structure/floodlight_frame/frame = target
|
|
if(frame.state == FLOODLIGHT_NEEDS_LIGHTS && Use(user))
|
|
new /obj/machinery/power/floodlight(frame.loc)
|
|
if(bluespace_toggle)
|
|
user.Beam(target, icon_state = "rped_upgrade", time = 0.5 SECONDS)
|
|
playsound(src, 'sound/items/pshoom.ogg', 40, 1)
|
|
to_chat(user, span_notice("You finish \the [frame] with a light tube."))
|
|
qdel(frame)
|
|
return TRUE
|
|
|
|
//attempt to replace all light sources on the turf
|
|
if(isturf(target))
|
|
var/light_replaced = FALSE
|
|
for(var/atom/target_atom in target)
|
|
if(replace_light(target_atom, user))
|
|
light_replaced = TRUE
|
|
if(light_replaced && bluespace_toggle)
|
|
user.Beam(target, icon_state = "rped_upgrade", time = 0.5 SECONDS)
|
|
playsound(src, 'sound/items/pshoom.ogg', 40, 1)
|
|
return TRUE
|
|
|
|
return FALSE
|
|
|
|
/obj/item/lightreplacer/afterattack(atom/target, mob/user, proximity)
|
|
. = ..()
|
|
|
|
// has no bluespace capabilities
|
|
if(!bluespace_toggle)
|
|
return
|
|
// target not in range
|
|
if(target.z != user.z)
|
|
return
|
|
// target not in view
|
|
if(!(target in view(7, get_turf(user))))
|
|
user.balloon_alert(user, "out of range!")
|
|
return
|
|
|
|
//replace lights & stuff
|
|
do_action(target, user)
|
|
|
|
/obj/item/lightreplacer/proc/status_string()
|
|
return "It has [uses] light\s remaining (plus [bulb_shards]/[BULB_SHARDS_REQUIRED] fragment\s)."
|
|
|
|
/obj/item/lightreplacer/proc/Use(mob/user)
|
|
if(uses <= 0)
|
|
return FALSE
|
|
|
|
playsound(src.loc, 'sound/machines/click.ogg', 50, TRUE)
|
|
src.add_fingerprint(user)
|
|
add_uses(-1)
|
|
|
|
return TRUE
|
|
|
|
// Negative numbers will subtract
|
|
/obj/item/lightreplacer/proc/add_uses(amount = 1)
|
|
uses = clamp(uses + amount, 0, max_uses)
|
|
|
|
/obj/item/lightreplacer/proc/add_shard(user)
|
|
bulb_shards += 1
|
|
if(bulb_shards >= BULB_SHARDS_REQUIRED)
|
|
bulb_shards = 0
|
|
add_uses(1)
|
|
to_chat(user, span_notice("\The [src] fabricates a new bulb from the broken glass it has stored. [status_string()]"))
|
|
playsound(src.loc, 'sound/machines/ding.ogg', 50, TRUE)
|
|
return TRUE
|
|
return FALSE
|
|
|
|
/obj/item/lightreplacer/proc/Charge(mob/user)
|
|
charge += 1
|
|
if(charge > 3)
|
|
add_uses(1)
|
|
charge = 1
|
|
|
|
/obj/item/lightreplacer/proc/replace_light(obj/machinery/light/target, mob/living/user)
|
|
//Confirm that it's a light we're testing, because afterattack runs this for everything on a given turf and will runtime
|
|
if(!istype(target))
|
|
return FALSE
|
|
//If the light source is ok then what are we doing here
|
|
if(target.status == LIGHT_OK)
|
|
user.balloon_alert(user, "light already installed!")
|
|
return FALSE
|
|
//Were all out
|
|
if(!Use(user))
|
|
//This balloon alert text is a little redundant, but I want to avoid a new player "yeah i know the light is empty" moment
|
|
user.balloon_alert(user, "light replacer empty!")
|
|
return FALSE
|
|
|
|
//remove any broken light on the fixture & add it as a shard
|
|
if(target.status != LIGHT_EMPTY)
|
|
add_shard(user)
|
|
target.status = LIGHT_EMPTY
|
|
target.update()
|
|
//create a copy of the light type & copy it's params onto the target
|
|
var/obj/item/light/old_light = new target.light_type()
|
|
target.status = old_light.status
|
|
target.switchcount = old_light.switchcount
|
|
target.brightness = old_light.brightness
|
|
if(obj_flags & EMAGGED)
|
|
target.create_reagents(LIGHT_REAGENT_CAPACITY, SEALED_CONTAINER | TRANSPARENT)
|
|
target.reagents.add_reagent(/datum/reagent/toxin/plasma, 10)
|
|
target.on = target.has_power()
|
|
target.update()
|
|
//clean up
|
|
qdel(old_light)
|
|
|
|
return TRUE
|
|
|
|
/obj/item/lightreplacer/cyborg/Initialize(mapload)
|
|
. = ..()
|
|
ADD_TRAIT(src, TRAIT_NODROP, CYBORG_ITEM_TRAIT)
|
|
|
|
/obj/item/lightreplacer/blue
|
|
name = "bluespace light replacer"
|
|
desc = "A modified light replacer that zaps lights into place. Refill with broken or working lightbulbs, or sheets of glass."
|
|
icon_state = "lightreplacer_blue"
|
|
bluespace_toggle = TRUE
|
|
|
|
/obj/item/lightreplacer/blue/emag_act()
|
|
return // balancing against longrange explosions
|
|
|
|
#undef GLASS_SHEET_USES
|
|
#undef LIGHTBULB_COST
|
|
#undef BULB_SHARDS_REQUIRED
|