mirror of
https://github.com/Bubberstation/Bubberstation.git
synced 2025-12-19 14:12:55 +00:00
506 lines
16 KiB
Plaintext
506 lines
16 KiB
Plaintext
// the SMES
|
|
// stores power
|
|
|
|
|
|
//Cache defines
|
|
#define SMES_CLEVEL_1 1
|
|
#define SMES_CLEVEL_2 2
|
|
#define SMES_CLEVEL_3 3
|
|
#define SMES_CLEVEL_4 4
|
|
#define SMES_CLEVEL_5 5
|
|
#define SMES_OUTPUTTING 6
|
|
#define SMES_NOT_OUTPUTTING 7
|
|
#define SMES_INPUTTING 8
|
|
#define SMES_INPUT_ATTEMPT 9
|
|
|
|
// SKYRAT EDIT COMMENT: Modularized Power change in modular_nova\master_files\code\modules\power\smes.dm
|
|
/obj/machinery/power/smes
|
|
name = "power storage unit"
|
|
desc = "A high-capacity superconducting magnetic energy storage (SMES) unit."
|
|
icon_state = "smes"
|
|
density = TRUE
|
|
use_power = NO_POWER_USE
|
|
circuit = /obj/item/circuitboard/machine/smes
|
|
can_change_cable_layer = TRUE
|
|
|
|
/// The charge capacity.
|
|
var/capacity = 50 * STANDARD_BATTERY_CHARGE // The board defaults with 5 high capacity batteries.
|
|
/// The current charge.
|
|
var/charge = 0
|
|
|
|
var/input_attempt = TRUE // TRUE = attempting to charge, FALSE = not attempting to charge
|
|
var/inputting = TRUE // TRUE = actually inputting, FALSE = not inputting
|
|
var/input_level = 50 KILO WATTS // amount of power the SMES attempts to charge by
|
|
var/input_level_max = 200 KILO WATTS // cap on input_level
|
|
var/input_available = 0 // amount of charge available from input last tick
|
|
|
|
var/output_attempt = TRUE // TRUE = attempting to output, FALSE = not attempting to output
|
|
var/outputting = TRUE // TRUE = actually outputting, FALSE = not outputting
|
|
var/output_level = 50 KILO WATTS // amount of power the SMES attempts to output
|
|
var/output_level_max = 200 KILO WATTS // cap on output_level
|
|
var/output_used = 0 // amount of power actually outputted. may be less than output_level if the powernet returns excess power
|
|
|
|
/// does this SMES show its input/output lights?
|
|
var/show_display_lights = TRUE
|
|
|
|
var/obj/machinery/power/terminal/terminal = null
|
|
|
|
/obj/machinery/power/smes/examine(user)
|
|
. = ..()
|
|
if(!terminal)
|
|
. += span_warning("This [src] has no power terminal!")
|
|
|
|
/obj/machinery/power/smes/Initialize(mapload)
|
|
. = ..()
|
|
dir_loop:
|
|
for(var/direction in GLOB.cardinals)
|
|
var/turf/turf = get_step(src, direction)
|
|
for(var/obj/machinery/power/terminal/term in turf)
|
|
if(term && term.dir == REVERSE_DIR(direction))
|
|
terminal = term
|
|
break dir_loop
|
|
|
|
if(!terminal)
|
|
atom_break()
|
|
return
|
|
terminal.master = src
|
|
update_appearance()
|
|
|
|
/obj/machinery/power/smes/RefreshParts()
|
|
SHOULD_CALL_PARENT(FALSE)
|
|
var/power_coefficient = 1 // Bubber edit
|
|
var/max_charge = 0
|
|
var/new_charge = 0
|
|
for(var/datum/stock_part/capacitor/capacitor in component_parts)
|
|
// SKYRAT EDIT CHANGE START - Original: power_coefficient += capacitor.tier
|
|
power_coefficient = 2 ** (capacitor.tier - 1)
|
|
// SKYRAT EDIT CHANGE END
|
|
input_level_max = initial(input_level_max) * power_coefficient
|
|
output_level_max = initial(output_level_max) * power_coefficient
|
|
for(var/obj/item/stock_parts/power_store/power_cell in component_parts)
|
|
max_charge += power_cell.maxcharge
|
|
new_charge += power_cell.charge
|
|
capacity = max_charge
|
|
if(!initial(charge) && !charge)
|
|
charge = new_charge
|
|
|
|
/obj/machinery/power/smes/should_have_node()
|
|
return TRUE
|
|
|
|
// adapted from APC item interacts for cable act handling
|
|
/obj/machinery/power/smes/item_interaction(mob/living/user, obj/item/tool, list/modifiers)
|
|
. = NONE
|
|
if(istype(tool, /obj/item/stack/cable_coil))
|
|
. = cable_act(user, tool, LAZYACCESS(modifiers, RIGHT_CLICK))
|
|
if(.)
|
|
return .
|
|
return .
|
|
|
|
/obj/machinery/power/smes/cable_layer_act(mob/living/user, obj/item/tool)
|
|
if(!QDELETED(terminal))
|
|
balloon_alert(user, "cut the terminal first!")
|
|
return ITEM_INTERACT_BLOCKING
|
|
return ..()
|
|
|
|
//opening using screwdriver
|
|
/obj/machinery/power/smes/screwdriver_act(mob/living/user, obj/item/tool)
|
|
. = ITEM_INTERACT_BLOCKING
|
|
if(default_deconstruction_screwdriver(user, "[initial(icon_state)]-o", initial(icon_state), tool))
|
|
update_appearance()
|
|
return ITEM_INTERACT_SUCCESS
|
|
|
|
//changing direction using wrench
|
|
/obj/machinery/power/smes/wrench_act(mob/living/user, obj/item/tool)
|
|
. = ITEM_INTERACT_BLOCKING
|
|
if(default_change_direction_wrench(user, tool))
|
|
terminal = null
|
|
var/turf/turf = get_step(src, dir)
|
|
for(var/obj/machinery/power/terminal/term in turf)
|
|
if(term && term.dir == REVERSE_DIR(dir))
|
|
terminal = term
|
|
terminal.master = src
|
|
to_chat(user, span_notice("Terminal found."))
|
|
break
|
|
if(!terminal)
|
|
to_chat(user, span_alert("No power terminal found."))
|
|
return ITEM_INTERACT_SUCCESS
|
|
set_machine_stat(machine_stat & ~BROKEN)
|
|
update_appearance()
|
|
return ITEM_INTERACT_SUCCESS
|
|
|
|
//building and linking a terminal
|
|
/obj/machinery/power/smes/proc/cable_act(mob/living/user, obj/item/stack/cable_coil/installing_cable, is_right_clicking)
|
|
. = ITEM_INTERACT_BLOCKING
|
|
if(!can_place_terminal(user, installing_cable, silent = FALSE))
|
|
return ITEM_INTERACT_BLOCKING
|
|
var/terminal_cable_layer
|
|
if(is_right_clicking)
|
|
var/choice = tgui_input_list(user, "Select Power Input Cable Layer", "Select Cable Layer", GLOB.cable_name_to_layer)
|
|
if(isnull(choice) \
|
|
|| !user.is_holding(installing_cable) \
|
|
|| !user.Adjacent(src) \
|
|
|| user.incapacitated \
|
|
|| !can_place_terminal(user, installing_cable, silent = TRUE) \
|
|
)
|
|
return ITEM_INTERACT_BLOCKING
|
|
terminal_cable_layer = GLOB.cable_name_to_layer[choice]
|
|
user.visible_message(span_notice("[user.name] starts adding cables to [src]."))
|
|
balloon_alert(user, "adding cables...")
|
|
playsound(src, 'sound/items/deconstruct.ogg', 50, TRUE)
|
|
|
|
if(!do_after(user, 2 SECONDS, target = src))
|
|
return ITEM_INTERACT_BLOCKING
|
|
if(!can_place_terminal(user, installing_cable, silent = TRUE))
|
|
return ITEM_INTERACT_BLOCKING
|
|
var/obj/item/stack/cable_coil/cable = installing_cable
|
|
var/turf/turf = get_turf(user)
|
|
var/obj/structure/cable/connected_cable = turf.get_cable_node(terminal_cable_layer) //get the connecting node cable, if there's one
|
|
if (prob(50) && electrocute_mob(user, connected_cable, connected_cable, 1, TRUE)) //animate the electrocution if uncautious and unlucky
|
|
do_sparks(5, TRUE, src)
|
|
return ITEM_INTERACT_BLOCKING
|
|
cable.use(10)
|
|
user.visible_message(span_notice("[user.name] adds cables to [src]."))
|
|
balloon_alert(user, "cables added")
|
|
//build the terminal and link it to the network
|
|
make_terminal(turf, terminal_cable_layer)
|
|
terminal.connect_to_network()
|
|
return ITEM_INTERACT_SUCCESS
|
|
|
|
//crowbarring it!
|
|
/obj/machinery/power/smes/crowbar_act(mob/living/user, obj/item/tool)
|
|
if(!panel_open)
|
|
return
|
|
var/turf/turf = get_turf(src)
|
|
if(default_deconstruction_crowbar(tool))
|
|
message_admins("[src] has been deconstructed by [ADMIN_LOOKUPFLW(user)] in [ADMIN_VERBOSEJMP(turf)].")
|
|
user.log_message("deconstructed [src]", LOG_GAME)
|
|
investigate_log("deconstructed by [key_name(user)] at [AREACOORD(src)].", INVESTIGATE_ENGINE)
|
|
return
|
|
|
|
/// Checks if we're in a valid state to place a terminal
|
|
/obj/machinery/power/smes/proc/can_place_terminal(mob/living/user, obj/item/stack/cable_coil/installing_cable, silent = TRUE)
|
|
var/set_dir = get_dir(user, src)
|
|
if(set_dir & (set_dir - 1))//we don't want diagonal click
|
|
return FALSE
|
|
|
|
var/turf/terminal_turf = get_turf(user)
|
|
if(!panel_open)
|
|
if(!silent && user)
|
|
balloon_alert(user, "open the maintenance panel!")
|
|
return FALSE
|
|
if(terminal_turf.underfloor_accessibility < UNDERFLOOR_INTERACTABLE)
|
|
if(!silent && user)
|
|
balloon_alert(user, "remove the floor plating!")
|
|
return FALSE
|
|
if(terminal)
|
|
if(!silent && user)
|
|
balloon_alert(user, "already wired!")
|
|
return FALSE
|
|
if(installing_cable.get_amount() < 10)
|
|
if(!silent && user)
|
|
balloon_alert(user, "need ten lengths of cable!")
|
|
return FALSE
|
|
return TRUE
|
|
|
|
/obj/machinery/power/smes/wirecutter_act(mob/living/user, obj/item/item)
|
|
//disassembling the terminal
|
|
. = ..()
|
|
if(terminal && panel_open)
|
|
terminal.dismantle(user, item)
|
|
return TRUE
|
|
|
|
|
|
/obj/machinery/power/smes/default_deconstruction_crowbar(obj/item/crowbar/crowbar)
|
|
if(istype(crowbar) && terminal)
|
|
balloon_alert(usr, "remove the power terminal!")
|
|
return FALSE
|
|
|
|
return ..()
|
|
|
|
/obj/machinery/power/smes/on_deconstruction(disassembled)
|
|
for(var/obj/item/stock_parts/power_store/cell in component_parts)
|
|
cell.charge = (charge / capacity) * cell.maxcharge
|
|
|
|
/obj/machinery/power/smes/Destroy()
|
|
if(SSticker.IsRoundInProgress())
|
|
var/turf/turf = get_turf(src)
|
|
message_admins("[src] deleted at [ADMIN_VERBOSEJMP(turf)]")
|
|
log_game("[src] deleted at [AREACOORD(turf)]")
|
|
investigate_log("deleted at [AREACOORD(turf)]", INVESTIGATE_ENGINE)
|
|
if(terminal)
|
|
disconnect_terminal()
|
|
return ..()
|
|
|
|
// create a terminal object pointing towards the SMES
|
|
// wires will attach to this
|
|
/obj/machinery/power/smes/proc/make_terminal(turf/turf, terminal_cable_layer = cable_layer)
|
|
terminal = new/obj/machinery/power/terminal(turf)
|
|
terminal.cable_layer = terminal_cable_layer
|
|
terminal.setDir(get_dir(turf,src))
|
|
terminal.master = src
|
|
set_machine_stat(machine_stat & ~BROKEN)
|
|
|
|
/obj/machinery/power/smes/disconnect_terminal()
|
|
if(terminal)
|
|
terminal.master = null
|
|
terminal = null
|
|
atom_break()
|
|
|
|
/// is this SMES in a suitable state to display overlays?
|
|
/obj/machinery/power/smes/proc/display_ready()
|
|
if(machine_stat & BROKEN)
|
|
return FALSE
|
|
if(panel_open)
|
|
return FALSE
|
|
return TRUE
|
|
|
|
/obj/machinery/power/smes/update_overlays()
|
|
. = ..()
|
|
if(!display_ready())
|
|
return
|
|
|
|
if(show_display_lights)
|
|
. += "smes-op[outputting ? 1 : 0]"
|
|
. += "smes-oc[inputting ? 1 : 0]"
|
|
|
|
var/clevel = chargedisplay()
|
|
if(clevel > 0)
|
|
. += "smes-og[clevel]"
|
|
|
|
|
|
/obj/machinery/power/smes/proc/chargedisplay()
|
|
if(capacity <= 0)
|
|
return 0
|
|
return clamp(round(5.5*charge/capacity),0,5)
|
|
|
|
/obj/machinery/power/smes/process(seconds_per_tick)
|
|
if(machine_stat & BROKEN)
|
|
return
|
|
|
|
//store machine state to see if we need to update the icon overlays
|
|
var/last_disp = chargedisplay()
|
|
var/last_chrg = inputting
|
|
var/last_onln = outputting
|
|
var/input_energy = power_to_energy(input_level)
|
|
var/output_energy = power_to_energy(output_level)
|
|
|
|
//outputting
|
|
if(output_attempt)
|
|
if(outputting)
|
|
output_used = min(charge, output_energy) //limit output to that stored
|
|
|
|
if (add_avail(output_used)) // add output to powernet if it exists (smes side)
|
|
adjust_charge(-output_used) // reduce the storage (may be recovered in /restore() if excessive)
|
|
else
|
|
outputting = FALSE
|
|
|
|
if(output_used < 0.1) // either from no charge or set to 0
|
|
outputting = FALSE
|
|
investigate_log("lost power and turned off", INVESTIGATE_ENGINE)
|
|
else if(output_attempt && charge > output_energy && output_level > 0)
|
|
outputting = TRUE
|
|
else
|
|
output_used = 0
|
|
else
|
|
outputting = FALSE
|
|
|
|
//inputting
|
|
if(terminal && input_attempt)
|
|
input_available = terminal.surplus()
|
|
|
|
if(inputting)
|
|
if(input_available > 0) // if there's power available, try to charge
|
|
|
|
var/load = min((capacity-charge), input_energy, input_available) // charge at set rate, limited to spare capacity
|
|
|
|
adjust_charge(load) // increase the charge
|
|
|
|
terminal.add_load(load) // add the load to the terminal side network
|
|
|
|
else // if not enough capcity
|
|
inputting = FALSE // stop inputting
|
|
|
|
else
|
|
if(input_attempt && input_available > 0)
|
|
inputting = TRUE
|
|
else
|
|
inputting = FALSE
|
|
|
|
// only update icon if state changed
|
|
if(last_disp != chargedisplay() || last_chrg != inputting || last_onln != outputting)
|
|
update_appearance()
|
|
|
|
/// Adjusts the charge in this SMES, used instead of directly adjusting the charge value. Mainly for the benefit of the power connector/portable SMES system.
|
|
/obj/machinery/power/smes/proc/adjust_charge(charge_adjust)
|
|
charge += charge_adjust
|
|
|
|
/// Sets the charge in this SMES, used instead of directly adjusting the charge value. Mainly for the benefit of the power connector/portable SMES system.
|
|
/obj/machinery/power/smes/proc/set_charge(charge_set)
|
|
charge = charge_set
|
|
|
|
// called after all power processes are finished
|
|
// restores charge level to smes if there was excess this ptick
|
|
/obj/machinery/power/smes/proc/restore()
|
|
if(machine_stat & BROKEN)
|
|
return
|
|
|
|
if(!outputting)
|
|
output_used = 0
|
|
return
|
|
|
|
var/excess = powernet.netexcess // this was how much wasn't used on the network last ptick, minus any removed by other SMESes
|
|
|
|
excess = min(output_used, excess) // clamp it to how much was actually output by this SMES last ptick
|
|
|
|
excess = min((capacity-charge), excess) // for safety, also limit recharge by space capacity of SMES (shouldn't happen)
|
|
|
|
// now recharge this amount
|
|
|
|
var/clev = chargedisplay()
|
|
|
|
adjust_charge(excess) // restore unused power
|
|
powernet.netexcess -= excess // remove the excess from the powernet, so later SMESes don't try to use it
|
|
|
|
output_used -= excess
|
|
|
|
if(clev != chargedisplay() ) //if needed updates the icons overlay
|
|
update_appearance()
|
|
return
|
|
|
|
|
|
/obj/machinery/power/smes/ui_interact(mob/user, datum/tgui/ui)
|
|
ui = SStgui.try_update_ui(user, src, ui)
|
|
if(!ui)
|
|
ui = new(user, src, "Smes", name)
|
|
ui.open()
|
|
|
|
/obj/machinery/power/smes/ui_data()
|
|
var/list/data = list(
|
|
"capacity" = capacity,
|
|
"capacityPercent" = round(100*charge/capacity, 0.1),
|
|
"charge" = charge,
|
|
"inputAttempt" = input_attempt,
|
|
"inputting" = inputting,
|
|
"inputLevel" = input_level,
|
|
"inputLevel_text" = display_power(input_level, convert = FALSE),
|
|
"inputLevelMax" = input_level_max,
|
|
"inputAvailable" = energy_to_power(input_available),
|
|
"outputAttempt" = output_attempt,
|
|
"outputting" = energy_to_power(outputting),
|
|
"outputLevel" = output_level,
|
|
"outputLevel_text" = display_power(output_level, convert = FALSE),
|
|
"outputLevelMax" = output_level_max,
|
|
"outputUsed" = energy_to_power(output_used),
|
|
)
|
|
return data
|
|
|
|
/obj/machinery/power/smes/ui_act(action, list/params, datum/tgui/ui, datum/ui_state/state)
|
|
. = ..()
|
|
if(.)
|
|
return
|
|
switch(action)
|
|
if("tryinput")
|
|
input_attempt = !input_attempt
|
|
log_smes(usr)
|
|
update_appearance()
|
|
. = TRUE
|
|
if("tryoutput")
|
|
output_attempt = !output_attempt
|
|
log_smes(usr)
|
|
update_appearance()
|
|
. = TRUE
|
|
if("input")
|
|
var/target = params["target"]
|
|
var/adjust = text2num(params["adjust"])
|
|
if(target == "min")
|
|
target = 0
|
|
. = TRUE
|
|
else if(target == "max")
|
|
target = input_level_max
|
|
. = TRUE
|
|
else if(adjust)
|
|
target = input_level + adjust
|
|
. = TRUE
|
|
else if(text2num(target) != null)
|
|
target = text2num(target)
|
|
. = TRUE
|
|
if(.)
|
|
input_level = clamp(target, 0, input_level_max)
|
|
log_smes(usr)
|
|
if("output")
|
|
var/target = params["target"]
|
|
var/adjust = text2num(params["adjust"])
|
|
if(target == "min")
|
|
target = 0
|
|
. = TRUE
|
|
else if(target == "max")
|
|
target = output_level_max
|
|
. = TRUE
|
|
else if(adjust)
|
|
target = output_level + adjust
|
|
. = TRUE
|
|
else if(text2num(target) != null)
|
|
target = text2num(target)
|
|
. = TRUE
|
|
if(.)
|
|
output_level = clamp(target, 0, output_level_max)
|
|
log_smes(usr)
|
|
|
|
/obj/machinery/power/smes/proc/log_smes(mob/user)
|
|
investigate_log("Input/Output: [input_level]/[output_level] | Charge: [charge] | Output-mode: [output_attempt?"ON":"OFF"] | Input-mode: [input_attempt?"AUTO":"OFF"] by [user ? key_name(user) : "outside forces"]", INVESTIGATE_ENGINE)
|
|
|
|
/obj/machinery/power/smes/emp_act(severity)
|
|
. = ..()
|
|
if(. & EMP_PROTECT_SELF)
|
|
return
|
|
input_attempt = rand(0,1)
|
|
inputting = input_attempt
|
|
output_attempt = rand(0,1)
|
|
outputting = output_attempt
|
|
output_level = rand(0, output_level_max)
|
|
input_level = rand(0, input_level_max)
|
|
adjust_charge(-STANDARD_BATTERY_CHARGE/severity)
|
|
if (charge < 0)
|
|
set_charge(0)
|
|
update_appearance()
|
|
log_smes()
|
|
|
|
// Variant of SMES that starts with super power cells for higher longevity
|
|
/obj/machinery/power/smes/super
|
|
name = "super capacity power storage unit"
|
|
desc = "A super-capacity superconducting magnetic energy storage (SMES) unit. Relatively rare, and typically installed in long-range outposts where minimal maintenance is expected."
|
|
circuit = /obj/item/circuitboard/machine/smes/super
|
|
capacity = 100 * STANDARD_BATTERY_CHARGE
|
|
|
|
/obj/machinery/power/smes/super/full
|
|
charge = 100 * STANDARD_BATTERY_CHARGE
|
|
|
|
/obj/machinery/power/smes/full
|
|
charge = 50 * STANDARD_BATTERY_CHARGE
|
|
|
|
/obj/machinery/power/smes/ship
|
|
charge = 20 * STANDARD_BATTERY_CHARGE
|
|
|
|
/obj/machinery/power/smes/engineering
|
|
charge = 50 * STANDARD_BATTERY_CHARGE // Engineering starts with some charge for singulo //sorry little one, singulo as engine is gone
|
|
output_level = 90 KILO WATTS
|
|
|
|
/obj/machinery/power/smes/magical
|
|
name = "magical power storage unit"
|
|
desc = "A high-capacity superconducting magnetic energy storage (SMES) unit. Magically produces power."
|
|
|
|
/obj/machinery/power/smes/magical/process()
|
|
capacity = INFINITY
|
|
charge = INFINITY
|
|
..()
|
|
|
|
#undef SMES_CLEVEL_1
|
|
#undef SMES_CLEVEL_2
|
|
#undef SMES_CLEVEL_3
|
|
#undef SMES_CLEVEL_4
|
|
#undef SMES_CLEVEL_5
|
|
#undef SMES_OUTPUTTING
|
|
#undef SMES_NOT_OUTPUTTING
|
|
#undef SMES_INPUTTING
|
|
#undef SMES_INPUT_ATTEMPT
|