Files
CHOMPStation2/code/game/machinery/air_alarm.dm
CHOMPStation2StaffMirrorBot f7de0bb70b [MIRROR] Start of TG Click Code Port (#12071)
Co-authored-by: Cameron Lennox <killer65311@gmail.com>
2025-12-06 03:18:32 -05:00

953 lines
33 KiB
Plaintext

#define DECLARE_TLV_VALUES var/red_min; var/yel_min; var/yel_max; var/red_max; var/tlv_comparitor;
#define LOAD_TLV_VALUES(x, y) red_min = x[1]; yel_min = x[2]; yel_max = x[3]; red_max = x[4]; tlv_comparitor = y;
#define TEST_TLV_VALUES (((tlv_comparitor > red_max && red_max > 0) || tlv_comparitor < red_min) ? 2 : ((tlv_comparitor > yel_max && yel_max > 0) || tlv_comparitor < yel_min) ? 1 : 0)
#define AALARM_MODE_SCRUBBING 1
#define AALARM_MODE_REPLACEMENT 2 //like scrubbing, but faster.
#define AALARM_MODE_PANIC 3 //constantly sucks all air
#define AALARM_MODE_CYCLE 4 //sucks off all air, then refill and switches to scrubbing
#define AALARM_MODE_FILL 5 //emergency fill
#define AALARM_MODE_OFF 6 //Shuts it all down.
#define AALARM_SCREEN_MAIN 1
#define AALARM_SCREEN_VENT 2
#define AALARM_SCREEN_SCRUB 3
#define AALARM_SCREEN_MODE 4
#define AALARM_SCREEN_SENSORS 5
#define AALARM_REPORT_TIMEOUT 100
#define MAX_TEMPERATURE 90
#define MIN_TEMPERATURE -40
//all air alarms in area are connected via freq 1439
/area
var/datum/weakref/main_air_alarm // The air alarm currently managing the others in the area, settings changes go to this one and propogate
var/list/air_vent_names = list()
var/list/air_scrub_names = list()
var/list/air_vent_info = list()
var/list/air_scrub_info = list()
var/list/air_alarms = list()
/area/proc/elect_main_air_alarm(var/exclude_self = FALSE)
// loop through all sensors to update the area's sensor list as well
main_air_alarm = null
var/list/checks = list()
for(var/obj/machinery/alarm/AA in air_alarms)
if(exclude_self && AA == src)
continue
if(!(AA.stat & (NOPOWER|BROKEN)))
checks += AA
if(!checks.len)
return
main_air_alarm = WEAKREF(pick(checks))
for(var/obj/machinery/alarm/AA in checks)
AA.update_icon()
/area/proc/main_air_alarm_is_operating()
var/obj/machinery/alarm/AM = main_air_alarm?.resolve()
return AM && !(AM.stat & (NOPOWER | BROKEN))
/obj/machinery/alarm
name = "alarm"
desc = "Used to control various station atmospheric systems. The light indicates the current air status of the area."
icon = 'icons/obj/monitors_vr.dmi'
icon_state = "alarm_0"
layer = ABOVE_WINDOW_LAYER
vis_flags = VIS_HIDE // They have an emissive that looks bad in openspace due to their wall-mounted nature
anchored = TRUE
unacidable = TRUE
use_power = USE_POWER_IDLE
idle_power_usage = 80
active_power_usage = 1000 //For heating/cooling rooms. 1000 joules equates to about 1 degree every 2 seconds for a single tile of air.
power_channel = ENVIRON
req_one_access = list(ACCESS_ATMOSPHERICS, ACCESS_ENGINE_EQUIP)
clicksound = "button"
clickvol = 30
blocks_emissive = NONE
light_power = 0.25
var/alarm_id = null
var/breach_detection = 1 // Whether to use automatic breach detection or not
var/frequency = PUMPS_FREQ
//var/skipprocess = 0 //Experimenting
var/alarm_frequency = ALERT_FREQ
var/remote_control = 0
var/rcon_setting = 2
var/rcon_time = 0
var/locked = 1
panel_open = FALSE // If it's been screwdrivered open.
var/aidisabled = 0
var/shorted = 0
circuit = /obj/item/circuitboard/airalarm
var/datum/wires/alarm/wires
var/mode = AALARM_MODE_SCRUBBING
var/screen = AALARM_SCREEN_MAIN
var/area_uid
var/area/alarm_area
var/target_temperature = T0C+20
var/regulating_temperature = 0
var/datum/radio_frequency/radio_connection
/// Keys are things like temperature and certain gasses. Values are lists, which contain, in order:
/// red warning minimum value, yellow warning minimum value, yellow warning maximum value, red warning maximum value
/// Use code\defines\gases.dm as reference for id/name. Please keep it consistent
var/list/TLV = list()
var/list/trace_gas = list(GAS_N2O, GAS_VOLATILE_FUEL) //list of other gases that this air alarm is able to detect
var/danger_level = 0
var/pressure_dangerlevel = 0
var/report_danger_level = 1
var/alarms_hidden = FALSE //If the alarms from this machine are visible on consoles
var/datum/looping_sound/alarm/decompression_alarm/soundloop // CHOMPEdit: Looping Alarms
var/atmoswarn = FALSE // CHOMPEdit: Looping Alarms
/obj/machinery/alarm/nobreach
breach_detection = 0
/obj/machinery/alarm/monitor
report_danger_level = 0
breach_detection = 0
/obj/machinery/alarm/alarms_hidden
alarms_hidden = TRUE
/obj/machinery/alarm/angled
icon = 'icons/obj/wall_machines_angled.dmi'
/obj/machinery/alarm/angled/hidden
alarms_hidden = TRUE
/obj/machinery/alarm/angled/offset_airalarm()
pixel_x = (dir & 3) ? 0 : (dir == 4 ? -21 : 21)
pixel_y = (dir & 3) ? (dir == 1 ? -18 : 20) : 0
/obj/machinery/alarm/Initialize(mapload)
. = ..()
update_area()
set_frequency(frequency)
if(!pixel_x && !pixel_y)
offset_airalarm()
if(!wires)
wires = new(src)
alarm_area.air_alarms += src
if(!alarm_area.main_air_alarm_is_operating()) // select main alarm
alarm_area.elect_main_air_alarm()
set_initial_TLV()
/obj/machinery/alarm/Destroy()
unregister_radio(src, frequency)
qdel(wires)
wires = null
alarm_area.air_alarms -= src
if(alarm_area.main_air_alarm?.resolve() == src)
alarm_area.elect_main_air_alarm(TRUE)
alarm_area = null
QDEL_NULL(soundloop) // CHOMPEdit: Looping Alarms
. = ..()
/obj/machinery/alarm/proc/offset_airalarm()
pixel_x = (dir & 3) ? 0 : (dir == 4 ? -26 : 26)
pixel_y = (dir & 3) ? (dir == 1 ? -26 : 26) : 0
/obj/machinery/alarm/proc/set_initial_TLV()
// breathable air according to human/Life()
TLV[GAS_O2] = list(16, 19, 135, 140) // Partial pressure, kpa
TLV[GAS_N2] = list(0, 0, 135, 140) // Partial pressure, kpa
TLV[GAS_CO2] = list(-1.0, -1.0, 5, 10) // Partial pressure, kpa
TLV[GAS_PHORON] = list(-1.0, -1.0, 0, 0.5) // Partial pressure, kpa
TLV[GAS_CH4] = list(-1.0, -1.0, 0, 0.5) // Partial pressure, kpa
TLV["other"] = list(-1.0, -1.0, 0.5, 1.0) // Partial pressure, kpa
TLV["pressure"] = list(ONE_ATMOSPHERE * 0.80, ONE_ATMOSPHERE * 0.90, ONE_ATMOSPHERE * 1.10, ONE_ATMOSPHERE * 1.20) /* kpa */
TLV["temperature"] = list(T0C - 26, T0C, T0C + 40, T0C + 66) // K
update_icon()
/obj/machinery/alarm/proc/update_area()
alarm_area = get_area(src)
area_uid = "\ref[alarm_area]"
if(name == "alarm")
name = "[alarm_area.name] Air Alarm \[[rand(9999)]\]" // random number id to help with players locating alarms, cosmetic
// CHOMPAdd Start
/obj/machinery/alarm/Initialize(mapload)
. = ..()
soundloop = new(list(src), FALSE)
// CHOMPAdd ENd
/obj/machinery/alarm/proc/scan_atmo()
var/turf/simulated/location = src.loc
if(!istype(location)) return//returns if loc is not simulated
var/datum/gas_mixture/environment = location.return_air()
//Handle temperature adjustment here.
handle_heating_cooling(environment)
var/old_level = danger_level
var/old_pressurelevel = pressure_dangerlevel
danger_level = overall_danger_level(environment)
if(old_level != danger_level)
apply_danger_level(danger_level)
if(old_pressurelevel != pressure_dangerlevel)
if(breach_detected())
mode = AALARM_MODE_OFF
apply_mode()
if(mode == AALARM_MODE_CYCLE && environment.return_pressure() < ONE_ATMOSPHERE * 0.05)
mode = AALARM_MODE_FILL
apply_mode()
// CHOMPAdd Start
if(alarm_area?.atmosalm || danger_level > 0) // Looping Alarms (Trigger Decompression alarm here, on detection of any breach in the area)
soundloop.start()
atmoswarn = TRUE
else if(danger_level == 0 && alarm_area?.atmosalm == 0) // Looping Alarms (Cancel Decompression alarm here)
soundloop.stop()
atmoswarn = FALSE
// CHOMPAdd End
//atmos computer remote controll stuff
switch(rcon_setting)
if(RCON_NO)
remote_control = 0
if(RCON_AUTO)
if(danger_level == 2)
remote_control = 1
else
remote_control = 0
if(RCON_YES)
remote_control = 1
/obj/machinery/alarm/process()
if(!alarm_area)
return
var/obj/machinery/alarm/MA = alarm_area.main_air_alarm?.resolve()
if(!MA)
alarm_area.elect_main_air_alarm()
MA = alarm_area.main_air_alarm?.resolve() // try again
if(!MA || (stat & (NOPOWER|BROKEN)) || shorted || MA.shorted)
return
scan_atmo()
/obj/machinery/alarm/proc/handle_heating_cooling(var/datum/gas_mixture/environment)
DECLARE_TLV_VALUES
LOAD_TLV_VALUES(TLV["temperature"], target_temperature)
if(!regulating_temperature)
//check for when we should start adjusting temperature
if(!TEST_TLV_VALUES && abs(environment.temperature - target_temperature) > 2.0 && environment.return_pressure() >= 1)
update_use_power(USE_POWER_ACTIVE)
regulating_temperature = (environment.temperature > target_temperature ? 1 : 2)
audible_message("\The [src] clicks as it starts [regulating_temperature == 1 ? "cooling" : "heating"] the room.",\
"You hear a click and a faint electronic hum.", runemessage = "* click *")
playsound(src, 'sound/machines/click.ogg', 50, 1)
else
//check for when we should stop adjusting temperature
if(TEST_TLV_VALUES || abs(environment.temperature - target_temperature) <= 0.5 || environment.return_pressure() < 1)
update_use_power(USE_POWER_IDLE)
audible_message("\The [src] clicks quietly as it stops [regulating_temperature == 1 ? "cooling" : "heating"] the room.",\
"You hear a click as a faint electronic humming stops.", runemessage = "* click *")
regulating_temperature = 0
playsound(src, 'sound/machines/click.ogg', 50, 1)
if(regulating_temperature)
if(target_temperature > T0C + MAX_TEMPERATURE)
target_temperature = T0C + MAX_TEMPERATURE
if(target_temperature < T0C + MIN_TEMPERATURE)
target_temperature = T0C + MIN_TEMPERATURE
var/datum/gas_mixture/gas
gas = environment.remove(0.25 * environment.total_moles)
if(gas)
if(gas.temperature <= target_temperature) //gas heating
var/energy_used = min(gas.get_thermal_energy_change(target_temperature) , active_power_usage)
gas.add_thermal_energy(energy_used)
//use_power(energy_used, ENVIRON) //handle by update_use_power instead
else //gas cooling
var/heat_transfer = min(abs(gas.get_thermal_energy_change(target_temperature)), active_power_usage)
//Assume the heat is being pumped into the hull which is fixed at 20 C
//none of this is really proper thermodynamics but whatever
var/cop = gas.temperature / T20C //coefficient of performance -> power used = heat_transfer/cop
heat_transfer = min(heat_transfer, cop * active_power_usage) //this ensures that we don't use more than active_power_usage amount of power
heat_transfer = -gas.add_thermal_energy(-heat_transfer) //get the actual heat transfer
//use_power(heat_transfer / cop, ENVIRON) //handle by update_use_power instead
environment.merge(gas)
/obj/machinery/alarm/proc/overall_danger_level(var/datum/gas_mixture/environment)
var/partial_pressure = R_IDEAL_GAS_EQUATION * environment.temperature/environment.volume
var/environment_pressure = environment.return_pressure()
var/other_moles = 0
for(var/g in trace_gas)
other_moles += environment.gas[g] //this is only going to be used in a partial pressure calc, so we don't need to worry about group_multiplier here.
DECLARE_TLV_VALUES
LOAD_TLV_VALUES(TLV["pressure"], environment_pressure)
pressure_dangerlevel = TEST_TLV_VALUES // not local because it's used in process()
LOAD_TLV_VALUES(TLV[GAS_O2], environment.gas[GAS_O2]*partial_pressure)
var/oxygen_dangerlevel = TEST_TLV_VALUES
LOAD_TLV_VALUES(TLV[GAS_CO2], environment.gas[GAS_CO2]*partial_pressure)
var/co2_dangerlevel = TEST_TLV_VALUES
LOAD_TLV_VALUES(TLV[GAS_PHORON], environment.gas[GAS_PHORON]*partial_pressure)
var/phoron_dangerlevel = TEST_TLV_VALUES
LOAD_TLV_VALUES(TLV[GAS_CH4], environment.gas[GAS_CH4]*partial_pressure)
var/methane_dangerlevel = TEST_TLV_VALUES
LOAD_TLV_VALUES(TLV["temperature"], environment.temperature)
var/temperature_dangerlevel = TEST_TLV_VALUES
LOAD_TLV_VALUES(TLV["other"], other_moles*partial_pressure)
var/other_dangerlevel = TEST_TLV_VALUES
return max(
pressure_dangerlevel,
oxygen_dangerlevel,
co2_dangerlevel,
phoron_dangerlevel,
methane_dangerlevel,
other_dangerlevel,
temperature_dangerlevel
)
// Returns whether this air alarm thinks there is a breach, given the sensors that are available to it.
/obj/machinery/alarm/proc/breach_detected()
var/turf/simulated/location = src.loc
if(!istype(location))
return 0
if(breach_detection == 0)
return 0
var/datum/gas_mixture/environment = location.return_air()
var/environment_pressure = environment.return_pressure()
var/pressure_levels = TLV["pressure"]
if(environment_pressure <= pressure_levels[1]) //low pressures
if(!(mode == AALARM_MODE_PANIC || mode == AALARM_MODE_CYCLE))
return 1
return 0
/obj/machinery/alarm/update_icon()
// start actual update!
cut_overlays()
if(panel_open)
icon_state = "alarmx"
set_light(0)
set_light_on(FALSE)
return
if(!alarm_area || (stat & (NOPOWER|BROKEN)) || shorted)
icon_state = "alarmp"
set_light(0)
set_light_on(FALSE)
return
// sub light!
var/obj/machinery/alarm/MA = alarm_area.main_air_alarm?.resolve()
if(MA == src)
// I am the main alarm
add_overlay(mutable_appearance(icon, "alarm_Mmode"))
add_overlay(emissive_appearance(icon, "alarm_Mmode"))
if(!MA || MA.shorted)
// main alarm is out! don't show display!
icon_state = "alarmp"
add_overlay(mutable_appearance(icon, "alarm_Xmode"))
add_overlay(emissive_appearance(icon, "alarm_Xmode"))
set_light(0)
set_light_on(FALSE)
return
// passive light on
add_overlay(mutable_appearance(icon, "alarm_Pmode"))
add_overlay(emissive_appearance(icon, "alarm_Pmode"))
var/icon_level = danger_level
if(alarm_area.atmosalm)
icon_level = max(icon_level, 1) //if there's an atmos alarm but everything is okay locally, no need to go past yellow
var/new_color = null
switch(icon_level)
if(0)
icon_state = "alarm_0"
if(alarm_area.main_air_alarm?.resolve() == src)
// active controller
add_overlay(mutable_appearance(icon, "alarm_ov0"))
add_overlay(emissive_appearance(icon, "alarm_ov0"))
new_color = "#03A728"
else
// passive mode
add_overlay(mutable_appearance(icon, "alarm_ovP"))
add_overlay(emissive_appearance(icon, "alarm_ovP"))
new_color = "#0033FF"
if(1)
icon_state = "alarm_2" //yes, alarm2 is yellow alarm
add_overlay(mutable_appearance(icon, "alarm_ov2"))
add_overlay(emissive_appearance(icon, "alarm_ov2"))
new_color = "#EC8B2F"
if(2)
icon_state = "alarm_1"
add_overlay(mutable_appearance(icon, "alarm_ov1"))
add_overlay(emissive_appearance(icon, "alarm_ov1"))
new_color = "#DA0205"
set_light(l_range = 2, l_power = 0.25, l_color = new_color)
set_light_on(TRUE)
/obj/machinery/alarm/receive_signal(datum/signal/signal)
if(stat & (NOPOWER|BROKEN))
return
if(!signal || signal.encryption)
return
var/id_tag = signal.data["tag"]
if(!id_tag)
return
if(signal.data["area"] != area_uid)
return
if(signal.data["sigtype"] != "status")
return
var/dev_type = signal.data["device"]
if(!(id_tag in alarm_area.air_scrub_names) && !(id_tag in alarm_area.air_vent_names))
register_env_machine(id_tag, dev_type)
if(dev_type == "AScr")
alarm_area.air_scrub_info[id_tag] = signal.data
else if(dev_type == "AVP")
alarm_area.air_vent_info[id_tag] = signal.data
/obj/machinery/alarm/proc/register_env_machine(var/m_id, var/device_type)
var/new_name
if(device_type == "AVP")
new_name = "[alarm_area.name] Vent Pump #[alarm_area.air_vent_names.len+1]"
alarm_area.air_vent_names[m_id] = new_name
else if(device_type == "AScr")
new_name = "[alarm_area.name] Air Scrubber #[alarm_area.air_scrub_names.len+1]"
alarm_area.air_scrub_names[m_id] = new_name
else
return
addtimer(CALLBACK(src, PROC_REF(send_signal),m_id, list("init" = new_name)), 10, TIMER_DELETE_ME)
/obj/machinery/alarm/proc/refresh_all()
for(var/id_tag in alarm_area.air_vent_names)
var/list/I = alarm_area.air_vent_info[id_tag]
if(I && I["timestamp"] + AALARM_REPORT_TIMEOUT / 2 > world.time)
continue
send_signal(id_tag, list("status"))
for(var/id_tag in alarm_area.air_scrub_names)
var/list/I = alarm_area.air_scrub_info[id_tag]
if(I && I["timestamp"] + AALARM_REPORT_TIMEOUT / 2 > world.time)
continue
send_signal(id_tag, list("status"))
/obj/machinery/alarm/proc/set_frequency(new_frequency)
SSradio.remove_object(src, frequency)
frequency = new_frequency
radio_connection = SSradio.add_object(src, frequency, RADIO_TO_AIRALARM)
/obj/machinery/alarm/proc/send_signal(var/target, var/list/command)//sends signal 'command' to 'target'. Returns 0 if no radio connection, 1 otherwise
if(!radio_connection)
return 0
var/datum/signal/signal = new
signal.transmission_method = TRANSMISSION_RADIO //radio signal
signal.source = src
signal.data = command
signal.data["tag"] = target
signal.data["sigtype"] = "command"
radio_connection.post_signal(src, signal, RADIO_FROM_AIRALARM)
// to_world("Signal [command] Broadcasted to [target]")
return 1
/obj/machinery/alarm/proc/apply_mode()
for(var/obj/machinery/alarm/AA in alarm_area.air_alarms)
AA.mode = mode //propagate mode to other air alarms in the area
switch(mode)
if(AALARM_MODE_SCRUBBING)
for(var/device_id in alarm_area.air_scrub_names)
send_signal(device_id, list("power"= 1, "co2_scrub"= 1, "scrubbing"= 1, "panic_siphon"= 0))
for(var/device_id in alarm_area.air_vent_names)
send_signal(device_id, list("power"= 1, "checks"= "default", "set_external_pressure"= "default"))
if(AALARM_MODE_PANIC, AALARM_MODE_CYCLE)
for(var/device_id in alarm_area.air_scrub_names)
send_signal(device_id, list("power"= 1, "panic_siphon"= 1))
for(var/device_id in alarm_area.air_vent_names)
send_signal(device_id, list("power"= 0))
if(AALARM_MODE_REPLACEMENT)
for(var/device_id in alarm_area.air_scrub_names)
send_signal(device_id, list("power"= 1, "panic_siphon"= 1))
for(var/device_id in alarm_area.air_vent_names)
send_signal(device_id, list("power"= 1, "checks"= "default", "set_external_pressure"= "default"))
if(AALARM_MODE_FILL)
for(var/device_id in alarm_area.air_scrub_names)
send_signal(device_id, list("power"= 0))
for(var/device_id in alarm_area.air_vent_names)
send_signal(device_id, list("power"= 1, "checks"= "default", "set_external_pressure"= "default"))
if(AALARM_MODE_OFF)
for(var/device_id in alarm_area.air_scrub_names)
send_signal(device_id, list("power"= 0))
for(var/device_id in alarm_area.air_vent_names)
send_signal(device_id, list("power"= 0))
/obj/machinery/alarm/proc/apply_danger_level(var/new_danger_level)
if(report_danger_level && alarm_area.atmosalert(new_danger_level, src))
post_alert(new_danger_level)
for(var/obj/machinery/alarm/AA in alarm_area.air_alarms)
update_icon()
/obj/machinery/alarm/proc/post_alert(alert_level)
var/datum/radio_frequency/frequency = SSradio.return_frequency(alarm_frequency)
if(!frequency)
return
var/datum/signal/alert_signal = new
alert_signal.source = src
alert_signal.transmission_method = TRANSMISSION_RADIO
alert_signal.data["zone"] = alarm_area.name
alert_signal.data["type"] = "Atmospheric"
if(alert_level==2)
alert_signal.data["alert"] = "severe"
else if(alert_level==1)
alert_signal.data["alert"] = "minor"
else if(alert_level==0)
alert_signal.data["alert"] = "clear"
frequency.post_signal(src, alert_signal)
/obj/machinery/alarm/attack_ai(mob/user)
tgui_interact(user)
/obj/machinery/alarm/attack_hand(mob/user)
. = ..()
if(.)
return
return interact(user)
/obj/machinery/alarm/interact(mob/user)
tgui_interact(user)
wires.Interact(user)
/obj/machinery/alarm/tgui_status(mob/user)
if(isAI(user) && aidisabled)
to_chat(user, "AI control has been disabled.")
else if(!shorted)
return ..()
return STATUS_CLOSE
/obj/machinery/alarm/tgui_interact(mob/user, datum/tgui/ui, datum/tgui/parent_ui, datum/tgui_state/state)
ui = SStgui.try_update_ui(user, src, ui)
if(!ui)
ui = new(user, src, "AirAlarm", name, parent_ui)
if(state)
ui.set_state(state)
ui.open()
/obj/machinery/alarm/tgui_data(mob/user, datum/tgui/ui, datum/tgui_state/state)
var/list/data = list(
"locked" = locked,
"siliconUser" = siliconaccess(user) || (isobserver(user) && is_admin(user)),
"remoteUser" = !!ui.parent_ui,
"danger_level" = danger_level,
"target_temperature" = "[target_temperature - T0C]C",
"rcon" = rcon_setting,
)
var/area/A = get_area(src)
data["atmos_alarm"] = A?.atmosalm
data["fire_alarm"] = A?.fire
var/turf/T = get_turf(src)
var/datum/gas_mixture/environment = T.return_air()
var/list/list/environment_data = list()
data["environment_data"] = environment_data
DECLARE_TLV_VALUES
var/pressure = environment.return_pressure()
LOAD_TLV_VALUES(TLV["pressure"], pressure)
environment_data.Add(list(list(
"name" = "Pressure",
"value" = pressure,
"unit" = "kPa",
"danger_level" = TEST_TLV_VALUES
)))
var/temperature = environment.temperature
LOAD_TLV_VALUES(TLV["temperature"], temperature)
environment_data.Add(list(list(
"name" = "Temperature",
"value" = temperature,
"unit" = "K ([round(temperature - T0C, 0.1)]C)",
"danger_level" = TEST_TLV_VALUES
)))
var/total_moles = environment.total_moles
var/partial_pressure = R_IDEAL_GAS_EQUATION * environment.temperature / environment.volume
for(var/gas_id in environment.gas)
if(!(gas_id in TLV))
continue
LOAD_TLV_VALUES(TLV[gas_id], environment.gas[gas_id] * partial_pressure)
environment_data.Add(list(list(
"name" = gas_id,
"value" = environment.gas[gas_id] / total_moles * 100,
"unit" = "%",
"danger_level" = TEST_TLV_VALUES
)))
if(!locked || siliconaccess(user) || data["remoteUser"] || (isobserver(user) && is_admin(user)))
var/list/list/vents = list()
data["vents"] = vents
for(var/id_tag in A.air_vent_names)
var/long_name = A.air_vent_names[id_tag]
var/list/info = A.air_vent_info[id_tag]
if(!info)
continue
vents.Add(list(list(
"id_tag" = id_tag,
"long_name" = sanitize(long_name),
"power" = info["power"],
"checks" = info["checks"],
"excheck" = info["checks"]&1,
"incheck" = info["checks"]&2,
"direction" = info["direction"],
"external" = info["external"],
"internal" = info["internal"],
"extdefault"= (info["external"] == ONE_ATMOSPHERE),
"intdefault"= (info["internal"] == 0),
)))
var/list/list/scrubbers = list()
data["scrubbers"] = scrubbers
for(var/id_tag in alarm_area.air_scrub_names)
var/long_name = alarm_area.air_scrub_names[id_tag]
var/list/info = alarm_area.air_scrub_info[id_tag]
if(!info)
continue
scrubbers += list(list(
"id_tag" = id_tag,
"long_name" = sanitize(long_name),
"power" = info["power"],
"scrubbing" = info["scrubbing"],
"panic" = info["panic"],
"filters" = list(
list("name" = GASNAME_O2, "command" = "o2_scrub", "val" = info["filter_o2"]),
list("name" = GASNAME_N2, "command" = "n2_scrub", "val" = info["filter_n2"]),
list("name" = GASNAME_CO2, "command" = "co2_scrub", "val" = info["filter_co2"]),
list("name" = GASNAME_PHORON, "command" = "tox_scrub", "val" = info["filter_phoron"]),
list("name" = GASNAME_CH4, "command" = "ch4_scrub", "val" = info["filter_ch4"]),
list("name" = GASNAME_N2O, "command" = "n2o_scrub", "val" = info["filter_n2o"]),
list("name" = GASNAME_VOLATILE_FUEL,"command" = "fuel_scrub", "val" = info["filter_fuel"])
)
))
data["scrubbers"] = scrubbers
data["mode"] = mode
var/list/list/modes = list()
data["modes"] = modes
modes[++modes.len] = list("name" = "Filtering - Scrubs out contaminants", "mode" = AALARM_MODE_SCRUBBING, "selected" = mode == AALARM_MODE_SCRUBBING, "danger" = 0)
modes[++modes.len] = list("name" = "Replace Air - Siphons out air while replacing", "mode" = AALARM_MODE_REPLACEMENT, "selected" = mode == AALARM_MODE_REPLACEMENT, "danger" = 0)
modes[++modes.len] = list("name" = "Panic - Siphons air out of the room", "mode" = AALARM_MODE_PANIC, "selected" = mode == AALARM_MODE_PANIC, "danger" = 1)
modes[++modes.len] = list("name" = "Cycle - Siphons air before replacing", "mode" = AALARM_MODE_CYCLE, "selected" = mode == AALARM_MODE_CYCLE, "danger" = 1)
modes[++modes.len] = list("name" = "Fill - Shuts off scrubbers and opens vents", "mode" = AALARM_MODE_FILL, "selected" = mode == AALARM_MODE_FILL, "danger" = 0)
modes[++modes.len] = list("name" = "Off - Shuts off vents and scrubbers", "mode" = AALARM_MODE_OFF, "selected" = mode == AALARM_MODE_OFF, "danger" = 0)
var/list/selected
var/list/thresholds = list()
var/list/gas_names = list(GAS_O2, GAS_CO2, GAS_PHORON, GAS_CH4, "other") //Gas ids made to match code\defines\gases.dm
for(var/g in gas_names)
thresholds[++thresholds.len] = list("name" = g, "settings" = list())
selected = TLV[g]
for(var/i = 1, i <= 4, i++)
thresholds[thresholds.len]["settings"] += list(list("env" = g, "val" = i, "selected" = selected[i]))
selected = TLV["pressure"]
thresholds[++thresholds.len] = list("name" = "Pressure", "settings" = list())
for(var/i = 1, i <= 4, i++)
thresholds[thresholds.len]["settings"] += list(list("env" = "pressure", "val" = i, "selected" = selected[i]))
selected = TLV["temperature"]
thresholds[++thresholds.len] = list("name" = "Temperature", "settings" = list())
for(var/i = 1, i <= 4, i++)
thresholds[thresholds.len]["settings"] += list(list("env" = "temperature", "val" = i, "selected" = selected[i]))
data["thresholds"] = thresholds
return data
/obj/machinery/alarm/tgui_act(action, params, datum/tgui/ui, datum/tgui_state/state)
if(..())
return TRUE
if(action == "rcon")
var/attempted_rcon_setting = text2num(params["rcon"])
switch(attempted_rcon_setting)
if(RCON_NO)
rcon_setting = RCON_NO
if(RCON_AUTO)
rcon_setting = RCON_AUTO
if(RCON_YES)
rcon_setting = RCON_YES
for(var/obj/machinery/alarm/AA in alarm_area.air_alarms)
AA.rcon_setting = rcon_setting
return TRUE
if(action == "temperature")
var/list/selected = TLV["temperature"]
var/max_temperature = min(selected[3] - T0C, MAX_TEMPERATURE)
var/min_temperature = max(selected[2] - T0C, MIN_TEMPERATURE)
var/input_temperature = tgui_input_number(ui.user, "What temperature would you like the system to mantain? (Capped between [min_temperature] and [max_temperature]C)", "Thermostat Controls", target_temperature - T0C, max_temperature, min_temperature, round_value = FALSE)
if(isnum(input_temperature))
if(input_temperature > max_temperature || input_temperature < min_temperature)
to_chat(ui.user, "Temperature must be between [min_temperature]C and [max_temperature]C")
else
for(var/obj/machinery/alarm/AA in alarm_area.air_alarms)
AA.target_temperature = input_temperature + T0C
return TRUE
// Account for remote users here.
// Yes, this is kinda snowflaky; however, I would argue it would be far more snowflakey
// to include "custom hrefs" and all the other bullshit that nano states have just for the
// like, two UIs, that want remote access to other UIs.
if((locked && !(siliconaccess(ui.user) || (isobserver(ui.user) && is_admin(ui.user))) && !istype(state, /datum/tgui_state/air_alarm_remote)) || (issilicon(ui.user) && aidisabled))
return
var/device_id = params["id_tag"]
switch(action)
if("lock")
if((siliconaccess(ui.user) && !wires.is_cut(WIRE_IDSCAN)) || (isobserver(ui.user) && is_admin(ui.user)))
locked = !locked
. = TRUE
if( "power",
"o2_scrub",
"n2_scrub",
"co2_scrub",
"tox_scrub",
"n2o_scrub",
"fuel_scrub",
"ch4_scrub",
"panic_siphon",
"scrubbing",
"direction")
send_signal(device_id, list("[action]" = text2num(params["val"])), ui.user)
. = TRUE
if("excheck")
send_signal(device_id, list("checks" = text2num(params["val"])^1), ui.user)
. = TRUE
if("incheck")
send_signal(device_id, list("checks" = text2num(params["val"])^2), ui.user)
. = TRUE
if("set_external_pressure", "set_internal_pressure")
var/target = params["value"]
if(!isnull(target))
send_signal(device_id, list("[action]" = target), ui.user)
. = TRUE
if("reset_external_pressure")
send_signal(device_id, list("reset_external_pressure"), ui.user)
. = TRUE
if("reset_internal_pressure")
send_signal(device_id, list("reset_internal_pressure"), ui.user)
. = TRUE
if("threshold")
var/env = params["env"]
var/name = params["var"]
var/value = tgui_input_number(ui.user, "New [name] for [env]:", name, TLV[env][name], min_value=-1, round_value = FALSE)
if(!isnull(value) && !..())
if(value < 0)
TLV[env][name] = -1
else
TLV[env][name] = round(value, 0.01)
clamp_tlv_values(env, name)
// investigate_log(" treshold value for [env]:[name] was set to [value] by [key_name(ui.user)]",INVESTIGATE_ATMOS)
for(var/obj/machinery/alarm/AA in alarm_area.air_alarms)
AA.TLV[env][name] = TLV[env][name]
. = TRUE
if("mode")
mode = text2num(params["mode"])
// investigate_log("was turned to [get_mode_name(mode)] mode by [key_name(ui.user)]",INVESTIGATE_ATMOS)
apply_mode(ui.user)
. = TRUE
if("alarm")
if(alarm_area.atmosalert(2, src))
apply_danger_level(2)
. = TRUE
if("reset")
atmos_reset()
. = TRUE
for(var/obj/machinery/alarm/AA in alarm_area.air_alarms)
update_icon()
// This big ol' mess just ensures that TLV always makes sense. If you set the max value below the min value,
// it'll automatically update all the other values to keep it sane.
/obj/machinery/alarm/proc/clamp_tlv_values(env, changed_threshold)
var/list/selected = TLV[env]
switch(changed_threshold)
if(1)
if(selected[1] > selected[2])
selected[2] = selected[1]
if(selected[1] > selected[3])
selected[3] = selected[1]
if(selected[1] > selected[4])
selected[4] = selected[1]
if(2)
if(selected[1] > selected[2])
selected[1] = selected[2]
if(selected[2] > selected[3])
selected[3] = selected[2]
if(selected[2] > selected[4])
selected[4] = selected[2]
if(3)
if(selected[1] > selected[3])
selected[1] = selected[3]
if(selected[2] > selected[3])
selected[2] = selected[3]
if(selected[3] > selected[4])
selected[4] = selected[3]
if(4)
if(selected[1] > selected[4])
selected[1] = selected[4]
if(selected[2] > selected[4])
selected[2] = selected[4]
if(selected[3] > selected[4])
selected[3] = selected[4]
/obj/machinery/alarm/proc/atmos_reset()
if(alarm_area.atmosalert(0, src))
apply_danger_level(0)
for(var/obj/machinery/alarm/AA in alarm_area.air_alarms)
update_icon()
/obj/machinery/alarm/attackby(obj/item/W as obj, mob/user)
add_fingerprint(user)
if(alarm_deconstruction_screwdriver(user, W))
return
if(alarm_deconstruction_wirecutters(user, W))
return
if(istype(W, /obj/item/card/id) || istype(W, /obj/item/pda))// trying to unlock the interface with an ID card
togglelock(user)
return ..()
/obj/machinery/alarm/proc/togglelock(mob/user)
if(stat & (NOPOWER|BROKEN))
to_chat(user, "It does nothing.")
return
else
if(allowed(user) && !wires.is_cut(WIRE_IDSCAN))
locked = !locked
to_chat(user, span_notice("You [locked ? "lock" : "unlock"] the Air Alarm interface."))
else
to_chat(user, span_warning("Access denied."))
return
/obj/machinery/alarm/click_alt(mob/user)
..()
togglelock(user)
/obj/machinery/alarm/power_change()
..()
var/delay_time = rand(0,15)
if(delay_time)
addtimer(CALLBACK(src, PROC_REF(process_power_change)), delay_time, TIMER_DELETE_ME) // CHOMPEdit
return
process_power_change()
// CHOMPAdd Start
/obj/machinery/alarm/proc/process_power_change()
update_icon()
if(!soundloop)
return
if(stat & (NOPOWER | BROKEN))
soundloop.stop()
else if(atmoswarn)
soundloop.start()
// CHOMPAdd End
/obj/machinery/alarm/server/Initialize(mapload)
. = ..()
req_access = list(ACCESS_RD, ACCESS_ATMOSPHERICS, ACCESS_ENGINE_EQUIP)
TLV[GAS_O2] = list(-1.0, -1.0,-1.0,-1.0) // Partial pressure, kpa
TLV[GAS_CO2] = list(-1.0, -1.0, 5, 10) // Partial pressure, kpa
TLV[GAS_PHORON] = list(-1.0, -1.0, 0, 0.5) // Partial pressure, kpa
TLV[GAS_CH4] = list(-1.0, -1.0, 0, 0.5) // Partial pressure, kpa
TLV["other"] = list(-1.0, -1.0, 0.5, 1.0) // Partial pressure, kpa
TLV["pressure"] = list(0,ONE_ATMOSPHERE*0.10,ONE_ATMOSPHERE*1.40,ONE_ATMOSPHERE*1.60) /* kpa */
TLV["temperature"] = list(20, 40, 140, 160) // K
target_temperature = 90
/obj/machinery/alarm/freezer
target_temperature = T0C - 13.15 // Chilly freezer room
/obj/machinery/alarm/freezer/set_initial_TLV()
. = ..()
TLV["temperature"] = list(T0C - 40, T0C - 20, T0C + 40, T0C + 66) // K, Lower Temperature for Freezer Air Alarms (This is because TLV is hardcoded to be generated on first_run, and therefore the only way to modify this without changing TLV generation)
// CHOMPEdit START
/obj/machinery/alarm/sifwilderness
breach_detection = 0
report_danger_level = 0
/obj/machinery/alarm/sifwilderness/set_initial_TLV()
. = ..()
TLV["oxygen"] = list(16, 17, 135, 140)
TLV["pressure"] = list(0,ONE_ATMOSPHERE*0.10,ONE_ATMOSPHERE*1.50,ONE_ATMOSPHERE*1.60)
TLV["temperature"] = list(T0C - 40, T0C - 31, T0C + 40, T0C + 120)
// CHOMPEdit END
#undef LOAD_TLV_VALUES
#undef TEST_TLV_VALUES
#undef DECLARE_TLV_VALUES
#undef AALARM_MODE_SCRUBBING
#undef AALARM_MODE_REPLACEMENT
#undef AALARM_MODE_PANIC
#undef AALARM_MODE_CYCLE
#undef AALARM_MODE_FILL
#undef AALARM_MODE_OFF
#undef AALARM_SCREEN_MAIN
#undef AALARM_SCREEN_VENT
#undef AALARM_SCREEN_SCRUB
#undef AALARM_SCREEN_MODE
#undef AALARM_SCREEN_SENSORS
#undef AALARM_REPORT_TIMEOUT
#undef MAX_TEMPERATURE
#undef MIN_TEMPERATURE