mirror of
https://github.com/CHOMPStation2/CHOMPStation2.git
synced 2025-12-10 02:09:41 +00:00
791 lines
29 KiB
Plaintext
791 lines
29 KiB
Plaintext
#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 magic
|
|
/area
|
|
var/obj/machinery/alarm/master_air_alarm
|
|
var/list/air_vent_names = list()
|
|
var/list/air_scrub_names = list()
|
|
var/list/air_vent_info = list()
|
|
var/list/air_scrub_info = list()
|
|
|
|
/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' //VOREStation Edit - Other icons
|
|
icon_state = "alarm0"
|
|
plane = TURF_PLANE
|
|
layer = ABOVE_TURF_LAYER
|
|
anchored = 1
|
|
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
|
|
var/alarm_id = null
|
|
var/breach_detection = 1 // Whether to use automatic breach detection or not
|
|
var/frequency = 1439
|
|
//var/skipprocess = 0 //Experimenting
|
|
var/alarm_frequency = 1437
|
|
var/remote_control = 0
|
|
var/rcon_setting = 2
|
|
var/rcon_time = 0
|
|
var/locked = 1
|
|
panel_open = 0 // If it's been screwdrivered open.
|
|
var/aidisabled = 0
|
|
var/shorted = 0
|
|
circuit = /obj/item/weapon/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
|
|
|
|
var/list/TLV = list()
|
|
var/list/trace_gas = list("sleeping_agent", "volatile_fuel") //list of other gases that this air alarm is able to detect
|
|
|
|
var/danger_level = 0
|
|
var/pressure_dangerlevel = 0
|
|
var/oxygen_dangerlevel = 0
|
|
var/co2_dangerlevel = 0
|
|
var/phoron_dangerlevel = 0
|
|
var/temperature_dangerlevel = 0
|
|
var/other_dangerlevel = 0
|
|
|
|
var/report_danger_level = 1
|
|
|
|
var/alarms_hidden = FALSE //If the alarms from this machine are visible on consoles
|
|
|
|
/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/server/Initialize(mapload)
|
|
. = ..()
|
|
req_access = list(access_rd, access_atmospherics, access_engine_equip)
|
|
TLV["oxygen"] = list(-1.0, -1.0,-1.0,-1.0) // Partial pressure, kpa
|
|
TLV["carbon dioxide"] = list(-1.0, -1.0, 5, 10) // Partial pressure, kpa
|
|
TLV["phoron"] = 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/Initialize(mapload)
|
|
. = ..()
|
|
first_run()
|
|
|
|
/obj/machinery/alarm/Destroy()
|
|
unregister_radio(src, frequency)
|
|
qdel(wires)
|
|
wires = null
|
|
if(alarm_area && alarm_area.master_air_alarm == src)
|
|
alarm_area.master_air_alarm = null
|
|
elect_master(exclude_self = TRUE)
|
|
return ..()
|
|
|
|
/obj/machinery/alarm/proc/first_run()
|
|
alarm_area = get_area(src)
|
|
area_uid = "\ref[alarm_area]"
|
|
if(name == "alarm")
|
|
name = "[alarm_area.name] Air Alarm"
|
|
|
|
if(!wires)
|
|
wires = new(src)
|
|
|
|
// breathable air according to human/Life()
|
|
TLV["oxygen"] = list(16, 19, 135, 140) // Partial pressure, kpa
|
|
TLV["carbon dioxide"] = list(-1.0, -1.0, 5, 10) // Partial pressure, kpa
|
|
TLV["phoron"] = 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
|
|
|
|
//VOREStation Add
|
|
pixel_x = (src.dir & 3)? 0 : (src.dir == 4 ? -28 : 28)
|
|
pixel_y = (src.dir & 3)? (src.dir ==1 ? -28 : 28) : 0
|
|
//VOREStation Add End
|
|
|
|
/obj/machinery/alarm/Initialize()
|
|
. = ..()
|
|
set_frequency(frequency)
|
|
if(!master_is_operating())
|
|
elect_master()
|
|
|
|
/obj/machinery/alarm/process()
|
|
if((stat & (NOPOWER|BROKEN)) || shorted)
|
|
return
|
|
|
|
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()
|
|
|
|
//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
|
|
|
|
return
|
|
|
|
/obj/machinery/alarm/proc/handle_heating_cooling(var/datum/gas_mixture/environment)
|
|
if(!regulating_temperature)
|
|
//check for when we should start adjusting temperature
|
|
if(!get_danger_level(target_temperature, TLV["temperature"]) && abs(environment.temperature - target_temperature) > 2.0)
|
|
update_use_power(USE_POWER_ACTIVE)
|
|
regulating_temperature = 1
|
|
audible_message("\The [src] clicks as it starts [environment.temperature > target_temperature ? "cooling" : "heating"] the room.",\
|
|
"You hear a click and a faint electronic hum.")
|
|
playsound(src, 'sound/machines/click.ogg', 50, 1)
|
|
else
|
|
//check for when we should stop adjusting temperature
|
|
if(get_danger_level(target_temperature, TLV["temperature"]) || abs(environment.temperature - target_temperature) <= 0.5)
|
|
update_use_power(USE_POWER_IDLE)
|
|
regulating_temperature = 0
|
|
audible_message("\The [src] clicks quietly as it stops [environment.temperature > target_temperature ? "cooling" : "heating"] the room.",\
|
|
"You hear a click as a faint electronic humming stops.")
|
|
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.
|
|
|
|
pressure_dangerlevel = get_danger_level(environment_pressure, TLV["pressure"])
|
|
oxygen_dangerlevel = get_danger_level(environment.gas["oxygen"]*partial_pressure, TLV["oxygen"])
|
|
co2_dangerlevel = get_danger_level(environment.gas["carbon_dioxide"]*partial_pressure, TLV["carbon dioxide"])
|
|
phoron_dangerlevel = get_danger_level(environment.gas["phoron"]*partial_pressure, TLV["phoron"])
|
|
temperature_dangerlevel = get_danger_level(environment.temperature, TLV["temperature"])
|
|
other_dangerlevel = get_danger_level(other_moles*partial_pressure, TLV["other"])
|
|
|
|
return max(
|
|
pressure_dangerlevel,
|
|
oxygen_dangerlevel,
|
|
co2_dangerlevel,
|
|
phoron_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/proc/master_is_operating()
|
|
return alarm_area && alarm_area.master_air_alarm && !(alarm_area.master_air_alarm.stat & (NOPOWER | BROKEN))
|
|
|
|
/obj/machinery/alarm/proc/elect_master(exclude_self = FALSE)
|
|
for(var/obj/machinery/alarm/AA in alarm_area)
|
|
if(exclude_self && AA == src)
|
|
continue
|
|
if(!(AA.stat & (NOPOWER|BROKEN)))
|
|
alarm_area.master_air_alarm = AA
|
|
return 1
|
|
return 0
|
|
|
|
/obj/machinery/alarm/proc/get_danger_level(var/current_value, var/list/danger_levels)
|
|
if((current_value >= danger_levels[4] && danger_levels[4] > 0) || current_value <= danger_levels[1])
|
|
return 2
|
|
if((current_value >= danger_levels[3] && danger_levels[3] > 0) || current_value <= danger_levels[2])
|
|
return 1
|
|
return 0
|
|
|
|
/obj/machinery/alarm/update_icon()
|
|
if(panel_open)
|
|
icon_state = "alarmx"
|
|
set_light(0)
|
|
return
|
|
if((stat & (NOPOWER|BROKEN)) || shorted)
|
|
icon_state = "alarmp"
|
|
set_light(0)
|
|
return
|
|
|
|
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 = "alarm0"
|
|
new_color = "#03A728"
|
|
if(1)
|
|
icon_state = "alarm2" //yes, alarm2 is yellow alarm
|
|
new_color = "#EC8B2F"
|
|
if(2)
|
|
icon_state = "alarm1"
|
|
new_color = "#DA0205"
|
|
|
|
set_light(l_range = 2, l_power = 0.25, l_color = new_color)
|
|
|
|
/obj/machinery/alarm/receive_signal(datum/signal/signal)
|
|
if(stat & (NOPOWER|BROKEN))
|
|
return
|
|
if(alarm_area.master_air_alarm != src)
|
|
if(master_is_operating())
|
|
return
|
|
elect_master()
|
|
if(alarm_area.master_air_alarm != src)
|
|
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
|
|
spawn(10)
|
|
send_signal(m_id, list("init" = new_name))
|
|
|
|
/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)
|
|
radio_controller.remove_object(src, frequency)
|
|
frequency = new_frequency
|
|
radio_connection = radio_controller.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()
|
|
//propagate mode to other air alarms in the area
|
|
//TODO: make it so that players can choose between applying the new mode to the room they are in (related area) vs the entire alarm area
|
|
for(var/obj/machinery/alarm/AA in alarm_area)
|
|
AA.mode = mode
|
|
|
|
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)
|
|
|
|
update_icon()
|
|
|
|
/obj/machinery/alarm/proc/post_alert(alert_level)
|
|
var/datum/radio_frequency/frequency = radio_controller.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)
|
|
ui_interact(user)
|
|
|
|
/obj/machinery/alarm/attack_hand(mob/user)
|
|
. = ..()
|
|
if(.)
|
|
return
|
|
return interact(user)
|
|
|
|
/obj/machinery/alarm/interact(mob/user)
|
|
ui_interact(user)
|
|
wires.Interact(user)
|
|
|
|
/obj/machinery/alarm/ui_interact(mob/user, ui_key = "main", datum/nanoui/ui = null, force_open = 1, var/master_ui = null, var/datum/topic_state/state = default_state)
|
|
var/data[0]
|
|
var/remote_connection = 0
|
|
var/remote_access = 0
|
|
if(state)
|
|
var/list/href = state.href_list(user)
|
|
remote_connection = href["remote_connection"] // Remote connection means we're non-adjacent/connecting from another computer
|
|
remote_access = href["remote_access"] // Remote access means we also have the privilege to alter the air alarm.
|
|
|
|
data["locked"] = locked && !issilicon(user)
|
|
data["remote_connection"] = remote_connection
|
|
data["remote_access"] = remote_access
|
|
data["rcon"] = rcon_setting
|
|
data["screen"] = screen
|
|
|
|
populate_status(data)
|
|
|
|
if(!(locked && !remote_connection) || remote_access || issilicon(user))
|
|
populate_controls(data)
|
|
|
|
ui = SSnanoui.try_update_ui(user, src, ui_key, ui, data, force_open)
|
|
if(!ui)
|
|
ui = new(user, src, ui_key, "air_alarm.tmpl", name, 325, 625, master_ui = master_ui, state = state)
|
|
ui.set_initial_data(data)
|
|
ui.open()
|
|
ui.set_auto_update(1)
|
|
|
|
/obj/machinery/alarm/proc/populate_status(var/data)
|
|
var/turf/location = get_turf(src)
|
|
var/datum/gas_mixture/environment = location.return_air()
|
|
var/total = environment.total_moles
|
|
|
|
var/list/environment_data = new
|
|
data["has_environment"] = total
|
|
if(total)
|
|
var/pressure = environment.return_pressure()
|
|
environment_data[++environment_data.len] = list("name" = "Pressure", "value" = pressure, "unit" = "kPa", "danger_level" = pressure_dangerlevel)
|
|
environment_data[++environment_data.len] = list("name" = "Oxygen", "value" = environment.gas["oxygen"] / total * 100, "unit" = "%", "danger_level" = oxygen_dangerlevel)
|
|
environment_data[++environment_data.len] = list("name" = "Carbon dioxide", "value" = environment.gas["carbon_dioxide"] / total * 100, "unit" = "%", "danger_level" = co2_dangerlevel)
|
|
environment_data[++environment_data.len] = list("name" = "Toxins", "value" = environment.gas["phoron"] / total * 100, "unit" = "%", "danger_level" = phoron_dangerlevel)
|
|
environment_data[++environment_data.len] = list("name" = "Temperature", "value" = environment.temperature, "unit" = "K ([round(environment.temperature - T0C, 0.1)]C)", "danger_level" = temperature_dangerlevel)
|
|
data["total_danger"] = danger_level
|
|
data["environment"] = environment_data
|
|
data["atmos_alarm"] = alarm_area.atmosalm
|
|
data["fire_alarm"] = alarm_area.fire != null
|
|
data["target_temperature"] = "[target_temperature - T0C]C"
|
|
|
|
/obj/machinery/alarm/proc/populate_controls(var/list/data)
|
|
switch(screen)
|
|
if(AALARM_SCREEN_MAIN)
|
|
data["mode"] = mode
|
|
if(AALARM_SCREEN_VENT)
|
|
var/vents[0]
|
|
for(var/id_tag in alarm_area.air_vent_names)
|
|
var/long_name = alarm_area.air_vent_names[id_tag]
|
|
var/list/info = alarm_area.air_vent_info[id_tag]
|
|
if(!info)
|
|
continue
|
|
vents[++vents.len] = list(
|
|
"id_tag" = id_tag,
|
|
"long_name" = sanitize(long_name),
|
|
"power" = info["power"],
|
|
"checks" = info["checks"],
|
|
"direction" = info["direction"],
|
|
"external" = info["external"]
|
|
)
|
|
data["vents"] = vents
|
|
if(AALARM_SCREEN_SCRUB)
|
|
var/scrubbers[0]
|
|
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[++scrubbers.len] = list(
|
|
"id_tag" = id_tag,
|
|
"long_name" = sanitize(long_name),
|
|
"power" = info["power"],
|
|
"scrubbing" = info["scrubbing"],
|
|
"panic" = info["panic"],
|
|
"filters" = list()
|
|
)
|
|
scrubbers[scrubbers.len]["filters"] += list(list("name" = "Oxygen", "command" = "o2_scrub", "val" = info["filter_o2"]))
|
|
scrubbers[scrubbers.len]["filters"] += list(list("name" = "Nitrogen", "command" = "n2_scrub", "val" = info["filter_n2"]))
|
|
scrubbers[scrubbers.len]["filters"] += list(list("name" = "Carbon Dioxide", "command" = "co2_scrub","val" = info["filter_co2"]))
|
|
scrubbers[scrubbers.len]["filters"] += list(list("name" = "Toxin" , "command" = "tox_scrub","val" = info["filter_phoron"]))
|
|
scrubbers[scrubbers.len]["filters"] += list(list("name" = "Nitrous Oxide", "command" = "n2o_scrub","val" = info["filter_n2o"]))
|
|
scrubbers[scrubbers.len]["filters"] += list(list("name" = "Fuel", "command" = "fuel_scrub","val" = info["filter_fuel"]))
|
|
data["scrubbers"] = scrubbers
|
|
if(AALARM_SCREEN_MODE)
|
|
var/modes[0]
|
|
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)
|
|
data["modes"] = modes
|
|
data["mode"] = mode
|
|
if(AALARM_SCREEN_SENSORS)
|
|
var/list/selected
|
|
var/thresholds[0]
|
|
|
|
var/list/gas_names = list(
|
|
"oxygen" = "O<sub>2</sub>",
|
|
"carbon dioxide" = "CO<sub>2</sub>",
|
|
"phoron" = "Toxin",
|
|
"other" = "Other")
|
|
for(var/g in gas_names)
|
|
thresholds[++thresholds.len] = list("name" = gas_names[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
|
|
|
|
/obj/machinery/alarm/CanUseTopic(var/mob/user, var/datum/topic_state/state, var/href_list = list())
|
|
if(aidisabled && isAI(user))
|
|
to_chat(user, "<span class='warning'>AI control for \the [src] interface has been disabled.</span>")
|
|
return STATUS_CLOSE
|
|
|
|
. = shorted ? STATUS_DISABLED : STATUS_INTERACTIVE
|
|
|
|
if(. == STATUS_INTERACTIVE)
|
|
var/extra_href = state.href_list(usr)
|
|
// Prevent remote users from altering RCON settings unless they already have access
|
|
if(href_list["rcon"] && extra_href["remote_connection"] && !extra_href["remote_access"])
|
|
. = STATUS_UPDATE
|
|
|
|
return min(..(), .)
|
|
|
|
/obj/machinery/alarm/Topic(href, href_list, var/datum/topic_state/state)
|
|
if(..(href, href_list, state))
|
|
return 1
|
|
|
|
// hrefs that can always be called -walter0o
|
|
if(href_list["rcon"])
|
|
var/attempted_rcon_setting = text2num(href_list["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
|
|
return 1
|
|
|
|
if(href_list["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 = input("What temperature would you like the system to mantain? (Capped between [min_temperature] and [max_temperature]C)", "Thermostat Controls", target_temperature - T0C) as num|null
|
|
if(isnum(input_temperature))
|
|
if(input_temperature > max_temperature || input_temperature < min_temperature)
|
|
to_chat(usr, "Temperature must be between [min_temperature]C and [max_temperature]C")
|
|
else
|
|
target_temperature = input_temperature + T0C
|
|
return 1
|
|
|
|
// hrefs that need the AA unlocked -walter0o
|
|
var/extra_href = state.href_list(usr)
|
|
if(!(locked && !extra_href["remote_connection"]) || extra_href["remote_access"] || issilicon(usr))
|
|
if(href_list["command"])
|
|
var/device_id = href_list["id_tag"]
|
|
switch(href_list["command"])
|
|
if("set_external_pressure")
|
|
var/input_pressure = input("What pressure you like the system to mantain?", "Pressure Controls") as num|null
|
|
if(isnum(input_pressure))
|
|
send_signal(device_id, list(href_list["command"] = input_pressure))
|
|
return 1
|
|
|
|
if("reset_external_pressure")
|
|
send_signal(device_id, list(href_list["command"] = ONE_ATMOSPHERE))
|
|
return 1
|
|
|
|
if( "power",
|
|
"adjust_external_pressure",
|
|
"checks",
|
|
"o2_scrub",
|
|
"n2_scrub",
|
|
"co2_scrub",
|
|
"tox_scrub",
|
|
"n2o_scrub",
|
|
"fuel_scrub",
|
|
"panic_siphon",
|
|
"scrubbing",
|
|
"direction")
|
|
|
|
send_signal(device_id, list(href_list["command"] = text2num(href_list["val"])))
|
|
return 1
|
|
|
|
if("set_threshold")
|
|
var/env = href_list["env"]
|
|
var/threshold = text2num(href_list["var"])
|
|
var/list/selected = TLV[env]
|
|
var/list/thresholds = list("lower bound", "low warning", "high warning", "upper bound")
|
|
var/newval = input("Enter [thresholds[threshold]] for [env]", "Alarm triggers", selected[threshold]) as null | num
|
|
if(isnull(newval))
|
|
return 1
|
|
if(newval<0)
|
|
selected[threshold] = -1.0
|
|
else if(env=="temperature" && newval>5000)
|
|
selected[threshold] = 5000
|
|
else if(env=="pressure" && newval>50*ONE_ATMOSPHERE)
|
|
selected[threshold] = 50*ONE_ATMOSPHERE
|
|
else if(env!="temperature" && env!="pressure" && newval>200)
|
|
selected[threshold] = 200
|
|
else
|
|
newval = round(newval,0.01)
|
|
selected[threshold] = newval
|
|
if(threshold == 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(threshold == 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(threshold == 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(threshold == 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]
|
|
|
|
apply_mode()
|
|
return 1
|
|
|
|
if(href_list["screen"])
|
|
screen = text2num(href_list["screen"])
|
|
return 1
|
|
|
|
if(href_list["atmos_unlock"])
|
|
switch(href_list["atmos_unlock"])
|
|
if("0")
|
|
alarm_area.firedoors_close()
|
|
if("1")
|
|
alarm_area.firedoors_open()
|
|
return 1
|
|
|
|
if(href_list["atmos_alarm"])
|
|
if(alarm_area.atmosalert(2, src))
|
|
apply_danger_level(2)
|
|
update_icon()
|
|
return 1
|
|
|
|
if(href_list["atmos_reset"])
|
|
if(alarm_area.atmosalert(0, src))
|
|
apply_danger_level(0)
|
|
update_icon()
|
|
return 1
|
|
|
|
if(href_list["mode"])
|
|
mode = text2num(href_list["mode"])
|
|
apply_mode()
|
|
return 1
|
|
|
|
/obj/machinery/alarm/attackby(obj/item/W as obj, mob/user as mob)
|
|
add_fingerprint(user)
|
|
if(alarm_deconstruction_screwdriver(user, W))
|
|
return
|
|
if(alarm_deconstruction_wirecutters(user, W))
|
|
return
|
|
|
|
if(istype(W, /obj/item/weapon/card/id) || istype(W, /obj/item/device/pda))// trying to unlock the interface with an ID card
|
|
togglelock()
|
|
return ..()
|
|
|
|
/obj/machinery/alarm/verb/togglelock(mob/user as mob)
|
|
if(stat & (NOPOWER|BROKEN))
|
|
to_chat(user, "It does nothing.")
|
|
return
|
|
else
|
|
if(allowed(usr) && !wires.is_cut(WIRE_IDSCAN))
|
|
locked = !locked
|
|
to_chat(user, "<span class='notice'>You [locked ? "lock" : "unlock"] the Air Alarm interface.</span>")
|
|
else
|
|
to_chat(user, "<span class='warning'>Access denied.</span>")
|
|
return
|
|
|
|
/obj/machinery/alarm/AltClick()
|
|
..()
|
|
togglelock()
|
|
|
|
/obj/machinery/alarm/power_change()
|
|
..()
|
|
spawn(rand(0,15))
|
|
update_icon()
|