diff --git a/code/controllers/subsystems/mapping.dm b/code/controllers/subsystems/mapping.dm
index 5e13e28e8f..e48b9f557f 100644
--- a/code/controllers/subsystems/mapping.dm
+++ b/code/controllers/subsystems/mapping.dm
@@ -1,29 +1,29 @@
-// Handles map-related tasks, mostly here to ensure it does so after the MC initializes.
-SUBSYSTEM_DEF(mapping)
- name = "Mapping"
- init_order = INIT_ORDER_MAPPING
- flags = SS_NO_FIRE
-
- var/list/map_templates = list()
- var/dmm_suite/maploader = null
-
-/datum/controller/subsystem/mapping/Initialize(timeofday)
- if(subsystem_initialized)
- return
- world.max_z_changed() // This is to set up the player z-level list, maxz hasn't actually changed (probably)
- maploader = new()
- load_map_templates()
-
- if(config.generate_map)
- // Map-gen is still very specific to the map, however putting it here should ensure it loads in the correct order.
- using_map.perform_map_generation()
-
-
-/datum/controller/subsystem/mapping/proc/load_map_templates()
- for(var/T in subtypesof(/datum/map_template))
- var/datum/map_template/template = T
- if(!(initial(template.mappath))) // If it's missing the actual path its probably a base type or being used for inheritence.
- continue
- template = new T()
- map_templates[template.name] = template
- return TRUE
+// Handles map-related tasks, mostly here to ensure it does so after the MC initializes.
+SUBSYSTEM_DEF(mapping)
+ name = "Mapping"
+ init_order = INIT_ORDER_MAPPING
+ flags = SS_NO_FIRE
+
+ var/list/map_templates = list()
+ var/dmm_suite/maploader = null
+
+/datum/controller/subsystem/mapping/Initialize(timeofday)
+ if(subsystem_initialized)
+ return
+ world.max_z_changed() // This is to set up the player z-level list, maxz hasn't actually changed (probably)
+ maploader = new()
+ load_map_templates()
+
+ if(config.generate_map)
+ // Map-gen is still very specific to the map, however putting it here should ensure it loads in the correct order.
+ using_map.perform_map_generation()
+
+
+/datum/controller/subsystem/mapping/proc/load_map_templates()
+ for(var/T in subtypesof(/datum/map_template))
+ var/datum/map_template/template = T
+ if(!(initial(template.mappath))) // If it's missing the actual path its probably a base type or being used for inheritence.
+ continue
+ template = new T()
+ map_templates[template.name] = template
+ return TRUE
diff --git a/code/game/machinery/air_alarm.dm b/code/game/machinery/air_alarm.dm
index 86df5dd00f..5a7131dae9 100644
--- a/code/game/machinery/air_alarm.dm
+++ b/code/game/machinery/air_alarm.dm
@@ -1,791 +1,790 @@
-#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 = alarm_area.uid
- 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.loc, '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.loc, '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))
- playsound(src.loc, 'sound/machines/airalarm.ogg', 25, 0, 4)
- 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 = 1 //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 = 1
- 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" = "O2",
- "carbon dioxide" = "CO2",
- "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, "AI control for \the [src] interface has been disabled.")
- 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.IsIndexCut(AALARM_WIRE_IDSCAN))
- locked = !locked
- to_chat(user, "You [locked ? "lock" : "unlock"] the Air Alarm interface.")
- else
- to_chat(user, "Access denied.")
- return
-
-/obj/machinery/alarm/AltClick()
- ..()
- togglelock()
-
-/obj/machinery/alarm/power_change()
- ..()
- spawn(rand(0,15))
- update_icon()
+#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 = alarm_area.uid
+ 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.loc, '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.loc, '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 = 1 //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 = 1
+ 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" = "O2",
+ "carbon dioxide" = "CO2",
+ "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, "AI control for \the [src] interface has been disabled.")
+ 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.IsIndexCut(AALARM_WIRE_IDSCAN))
+ locked = !locked
+ to_chat(user, "You [locked ? "lock" : "unlock"] the Air Alarm interface.")
+ else
+ to_chat(user, "Access denied.")
+ return
+
+/obj/machinery/alarm/AltClick()
+ ..()
+ togglelock()
+
+/obj/machinery/alarm/power_change()
+ ..()
+ spawn(rand(0,15))
+ update_icon()
diff --git a/code/game/turfs/simulated/floor.dm b/code/game/turfs/simulated/floor.dm
index 7e8db4dbf9..af6d681122 100644
--- a/code/game/turfs/simulated/floor.dm
+++ b/code/game/turfs/simulated/floor.dm
@@ -20,7 +20,7 @@
'sound/effects/footstep/plating4.ogg',
'sound/effects/footstep/plating5.ogg'))
- var/list/old_decals = null // VOREStation Edit - Remember what decals we had between being pried up and replaced.
+ var/list/old_decals = null
// Flooring data.
var/flooring_override
diff --git a/code/modules/overmap/_defines.dm b/code/modules/overmap/_defines.dm
new file mode 100644
index 0000000000..6cfde46793
--- /dev/null
+++ b/code/modules/overmap/_defines.dm
@@ -0,0 +1,145 @@
+//How far from the edge of overmap zlevel could randomly placed objects spawn
+#define OVERMAP_EDGE 2
+
+#define SHIP_SIZE_TINY 1
+#define SHIP_SIZE_SMALL 2
+#define SHIP_SIZE_LARGE 3
+
+//multipliers for max_speed to find 'slow' and 'fast' speeds for the ship
+#define SHIP_SPEED_SLOW 1/(40 SECONDS)
+#define SHIP_SPEED_FAST 3/(20 SECONDS)// 15 speed
+
+#define OVERMAP_WEAKNESS_NONE 0
+#define OVERMAP_WEAKNESS_FIRE 1
+#define OVERMAP_WEAKNESS_EMP 2
+#define OVERMAP_WEAKNESS_MINING 4
+#define OVERMAP_WEAKNESS_EXPLOSIVE 8
+
+//Dimension of overmap (squares 4 lyfe)
+var/global/list/map_sectors = list()
+
+/area/overmap/
+ name = "System Map"
+ icon_state = "start"
+ requires_power = 0
+ base_turf = /turf/unsimulated/map
+
+/turf/unsimulated/map
+ icon = 'icons/turf/space.dmi'
+ icon_state = "map"
+ initialized = FALSE // TODO - Fix unsimulated turf initialization so this override is not necessary!
+
+/turf/unsimulated/map/edge
+ opacity = 1
+ density = 1
+
+/turf/unsimulated/map/Initialize()
+ . = ..()
+ name = "[x]-[y]"
+ var/list/numbers = list()
+
+ if(x == 1 || x == global.using_map.overmap_size)
+ numbers += list("[round(y/10)]","[round(y%10)]")
+ if(y == 1 || y == global.using_map.overmap_size)
+ numbers += "-"
+ if(y == 1 || y == global.using_map.overmap_size)
+ numbers += list("[round(x/10)]","[round(x%10)]")
+
+ for(var/i = 1 to numbers.len)
+ var/image/I = image('icons/effects/numbers.dmi',numbers[i])
+ I.pixel_x = 5*i - 2
+ I.pixel_y = world.icon_size/2 - 3
+ if(y == 1)
+ I.pixel_y = 3
+ I.pixel_x = 5*i + 4
+ if(y == global.using_map.overmap_size)
+ I.pixel_y = world.icon_size - 9
+ I.pixel_x = 5*i + 4
+ if(x == 1)
+ I.pixel_x = 5*i - 2
+ if(x == global.using_map.overmap_size)
+ I.pixel_x = 5*i + 2
+ add_overlay(I)
+
+//list used to track which zlevels are being 'moved' by the proc below
+var/list/moving_levels = list()
+//Proc to 'move' stars in spess
+//yes it looks ugly, but it should only fire when state actually change.
+//null direction stops movement
+proc/toggle_move_stars(zlevel, direction)
+ if(!zlevel)
+ return
+
+ if (moving_levels["[zlevel]"] != direction)
+ moving_levels["[zlevel]"] = direction
+
+ var/list/spaceturfs = block(locate(1, 1, zlevel), locate(world.maxx, world.maxy, zlevel))
+ for(var/turf/space/T in spaceturfs)
+ T.toggle_transit(direction)
+ CHECK_TICK
+/*
+//list used to cache empty zlevels to avoid nedless map bloat
+var/list/cached_space = list()
+
+proc/overmap_spacetravel(var/turf/space/T, var/atom/movable/A)
+ var/obj/effect/map/M = map_sectors["[T.z]"]
+ if (!M)
+ return
+ var/mapx = M.x
+ var/mapy = M.y
+ var/nx = 1
+ var/ny = 1
+ var/nz = M.map_z
+
+ if(T.x <= TRANSITIONEDGE)
+ nx = world.maxx - TRANSITIONEDGE - 2
+ ny = rand(TRANSITIONEDGE + 2, world.maxy - TRANSITIONEDGE - 2)
+ mapx = max(1, mapx-1)
+
+ else if (A.x >= (world.maxx - TRANSITIONEDGE - 1))
+ nx = TRANSITIONEDGE + 2
+ ny = rand(TRANSITIONEDGE + 2, world.maxy - TRANSITIONEDGE - 2)
+ mapx = min(world.maxx, mapx+1)
+
+ else if (T.y <= TRANSITIONEDGE)
+ ny = world.maxy - TRANSITIONEDGE -2
+ nx = rand(TRANSITIONEDGE + 2, world.maxx - TRANSITIONEDGE - 2)
+ mapy = max(1, mapy-1)
+
+ else if (A.y >= (world.maxy - TRANSITIONEDGE - 1))
+ ny = TRANSITIONEDGE + 2
+ nx = rand(TRANSITIONEDGE + 2, world.maxx - TRANSITIONEDGE - 2)
+ mapy = min(world.maxy, mapy+1)
+
+ testing("[A] moving from [M] ([M.x], [M.y]) to ([mapx],[mapy]).")
+
+ var/turf/map = locate(mapx,mapy,OVERMAP_ZLEVEL)
+ var/obj/effect/map/TM = locate() in map
+ if(TM)
+ nz = TM.map_z
+ testing("Destination: [TM]")
+ else
+ if(cached_space.len)
+ var/obj/effect/map/sector/temporary/cache = cached_space[cached_space.len]
+ cached_space -= cache
+ nz = cache.map_z
+ cache.x = mapx
+ cache.y = mapy
+ testing("Destination: *cached* [TM]")
+ else
+ world.maxz++
+ nz = world.maxz
+ TM = new /obj/effect/map/sector/temporary(mapx, mapy, nz)
+ testing("Destination: *new* [TM]")
+
+ var/turf/dest = locate(nx,ny,nz)
+ if(dest)
+ A.loc = dest
+
+ if(istype(M, /obj/effect/map/sector/temporary))
+ var/obj/effect/map/sector/temporary/source = M
+ if (source.can_die())
+ testing("Catching [M] for future use")
+ source.loc = null
+ cached_space += source
+*/
\ No newline at end of file