diff --git a/code/controllers/subsystem/blackbox.dm b/code/controllers/subsystem/blackbox.dm
index 5b2c67c296..b30bc41759 100644
--- a/code/controllers/subsystem/blackbox.dm
+++ b/code/controllers/subsystem/blackbox.dm
@@ -1,242 +1,246 @@
-SUBSYSTEM_DEF(blackbox)
- name = "Blackbox"
- wait = 6000
- flags = SS_NO_TICK_CHECK
-
- var/list/msg_common = list()
- var/list/msg_science = list()
- var/list/msg_command = list()
- var/list/msg_medical = list()
- var/list/msg_engineering = list()
- var/list/msg_security = list()
- var/list/msg_deathsquad = list()
- var/list/msg_syndicate = list()
- var/list/msg_service = list()
- var/list/msg_cargo = list()
- var/list/msg_other = list()
-
- var/list/feedback = list() //list of datum/feedback_variable
-
-//poll population
-/datum/controller/subsystem/blackbox/fire()
- if(!SSdbcore.Connect())
- return
- var/playercount = 0
- for(var/mob/M in GLOB.player_list)
- if(M.client)
- playercount += 1
- var/admincount = GLOB.admins.len
- var/datum/DBQuery/query_record_playercount = SSdbcore.NewQuery("INSERT INTO [format_table_name("legacy_population")] (playercount, admincount, time, server_ip, server_port) VALUES ([playercount], [admincount], '[SQLtime()]', INET_ATON('[world.internet_address]'), '[world.port]')")
- query_record_playercount.Execute()
-
-/datum/controller/subsystem/blackbox/Recover()
- msg_common = SSblackbox.msg_common
- msg_science = SSblackbox.msg_science
- msg_command = SSblackbox.msg_command
- msg_medical = SSblackbox.msg_medical
- msg_engineering = SSblackbox.msg_engineering
- msg_security = SSblackbox.msg_security
- msg_deathsquad = SSblackbox.msg_deathsquad
- msg_syndicate = SSblackbox.msg_syndicate
- msg_service = SSblackbox.msg_service
- msg_cargo = SSblackbox.msg_cargo
- msg_other = SSblackbox.msg_other
-
- feedback = SSblackbox.feedback
-
-//no touchie
-/datum/controller/subsystem/blackbox/can_vv_get(var_name)
- if(var_name == "feedback")
- return FALSE
- return ..()
-
-/datum/controller/subsystem/blackbox/vv_edit_var(var_name, var_value)
- return FALSE
-
-/datum/controller/subsystem/blackbox/Shutdown()
- var/pda_msg_amt = 0
- var/rc_msg_amt = 0
-
- for (var/obj/machinery/message_server/MS in GLOB.message_servers)
- if (MS.pda_msgs.len > pda_msg_amt)
- pda_msg_amt = MS.pda_msgs.len
- if (MS.rc_msgs.len > rc_msg_amt)
- rc_msg_amt = MS.rc_msgs.len
-
- set_details("radio_usage","")
-
- add_details("radio_usage","COM-[msg_common.len]")
- add_details("radio_usage","SCI-[msg_science.len]")
- add_details("radio_usage","HEA-[msg_command.len]")
- add_details("radio_usage","MED-[msg_medical.len]")
- add_details("radio_usage","ENG-[msg_engineering.len]")
- add_details("radio_usage","SEC-[msg_security.len]")
- add_details("radio_usage","DTH-[msg_deathsquad.len]")
- add_details("radio_usage","SYN-[msg_syndicate.len]")
- add_details("radio_usage","SRV-[msg_service.len]")
- add_details("radio_usage","CAR-[msg_cargo.len]")
- add_details("radio_usage","OTH-[msg_other.len]")
- add_details("radio_usage","PDA-[pda_msg_amt]")
- add_details("radio_usage","RC-[rc_msg_amt]")
-
- set_details("round_end","[time2text(world.realtime)]") //This one MUST be the last one that gets set.
-
- if (!SSdbcore.Connect())
- return
-
- var/list/sqlrowlist = list()
-
- for (var/datum/feedback_variable/FV in feedback)
- sqlrowlist += list("time" = "Now()", "round_id" = GLOB.round_id, "var_name" = "'[sanitizeSQL(FV.get_variable())]'", "var_value" = FV.get_value(), "details" = "'[sanitizeSQL(FV.get_details())]'")
-
- if (!length(sqlrowlist))
- return
-
- SSdbcore.MassInsert(format_table_name("feedback"), sqlrowlist, ignore_errors = TRUE, delayed = TRUE)
-
-
-/datum/controller/subsystem/blackbox/proc/LogBroadcast(blackbox_msg, freq)
- switch(freq)
- if(1459)
- msg_common += blackbox_msg
- if(1351)
- msg_science += blackbox_msg
- if(1353)
- msg_command += blackbox_msg
- if(1355)
- msg_medical += blackbox_msg
- if(1357)
- msg_engineering += blackbox_msg
- if(1359)
- msg_security += blackbox_msg
- if(1441)
- msg_deathsquad += blackbox_msg
- if(1213)
- msg_syndicate += blackbox_msg
- if(1349)
- msg_service += blackbox_msg
- if(1347)
- msg_cargo += blackbox_msg
- else
- msg_other += blackbox_msg
-
-/datum/controller/subsystem/blackbox/proc/find_feedback_datum(variable)
- for(var/datum/feedback_variable/FV in feedback)
- if(FV.get_variable() == variable)
- return FV
-
- var/datum/feedback_variable/FV = new(variable)
- feedback += FV
- return FV
-
-/datum/controller/subsystem/blackbox/proc/set_val(variable, value)
- var/datum/feedback_variable/FV = find_feedback_datum(variable)
- FV.set_value(value)
-
-/datum/controller/subsystem/blackbox/proc/inc(variable, value)
- var/datum/feedback_variable/FV = find_feedback_datum(variable)
- FV.inc(value)
-
-/datum/controller/subsystem/blackbox/proc/dec(variable,value)
- var/datum/feedback_variable/FV = find_feedback_datum(variable)
- FV.dec(value)
-
-/datum/controller/subsystem/blackbox/proc/set_details(variable,details)
- var/datum/feedback_variable/FV = find_feedback_datum(variable)
- FV.set_details(details)
-
-/datum/controller/subsystem/blackbox/proc/add_details(variable,details)
- var/datum/feedback_variable/FV = find_feedback_datum(variable)
- FV.add_details(details)
-
-/datum/controller/subsystem/blackbox/proc/ReportDeath(mob/living/L)
- if(!SSdbcore.Connect())
- return
- if(!L || !L.key || !L.mind)
- return
- var/turf/T = get_turf(L)
- var/area/placeofdeath = get_area(T.loc)
- var/sqlname = sanitizeSQL(L.real_name)
- var/sqlkey = sanitizeSQL(L.ckey)
- var/sqljob = sanitizeSQL(L.mind.assigned_role)
- var/sqlspecial = sanitizeSQL(L.mind.special_role)
- var/sqlpod = sanitizeSQL(placeofdeath.name)
- var/laname
- var/lakey
- if(L.lastattacker && ismob(L.lastattacker))
- var/mob/LA = L.lastattacker
- laname = sanitizeSQL(LA.real_name)
- lakey = sanitizeSQL(LA.key)
- var/sqlgender = sanitizeSQL(L.gender)
- var/sqlbrute = sanitizeSQL(L.getBruteLoss())
- var/sqlfire = sanitizeSQL(L.getFireLoss())
- var/sqlbrain = sanitizeSQL(L.getBrainLoss())
- var/sqloxy = sanitizeSQL(L.getOxyLoss())
- var/sqltox = sanitizeSQL(L.getStaminaLoss())
- var/sqlclone = sanitizeSQL(L.getStaminaLoss())
- var/sqlstamina = sanitizeSQL(L.getStaminaLoss())
- var/coord = sanitizeSQL("[L.x], [L.y], [L.z]")
- var/map = sanitizeSQL(SSmapping.config.map_name)
- var/datum/DBQuery/query_report_death = SSdbcore.NewQuery("INSERT INTO [format_table_name("death")] (name, byondkey, job, special, pod, tod, laname, lakey, gender, bruteloss, fireloss, brainloss, oxyloss, toxloss, cloneloss, staminaloss, coord, mapname, server_ip, server_port) VALUES ('[sqlname]', '[sqlkey]', '[sqljob]', '[sqlspecial]', '[sqlpod]', '[SQLtime()]', '[laname]', '[lakey]', '[sqlgender]', [sqlbrute], [sqlfire], [sqlbrain], [sqloxy], [sqltox], [sqlclone], [sqlstamina], '[coord]', '[map]', INET_ATON('[world.internet_address]'), '[world.port]')")
- query_report_death.Execute()
-
-
-//feedback variable datum, for storing all kinds of data
-/datum/feedback_variable
- var/variable
- var/value
- var/details
-
-/datum/feedback_variable/New(param_variable, param_value = 0)
- variable = param_variable
- value = param_value
-
-/datum/feedback_variable/proc/inc(num = 1)
- if (isnum(value))
- value += num
- else
- value = text2num(value)
- if (isnum(value))
- value += num
- else
- value = num
-
-/datum/feedback_variable/proc/dec(num = 1)
- if (isnum(value))
- value -= num
- else
- value = text2num(value)
- if (isnum(value))
- value -= num
- else
- value = -num
-
-/datum/feedback_variable/proc/set_value(num)
- if (isnum(num))
- value = num
-
-/datum/feedback_variable/proc/get_value()
- if (!isnum(value))
- return 0
- return value
-
-/datum/feedback_variable/proc/get_variable()
- return variable
-
-/datum/feedback_variable/proc/set_details(text)
- if (istext(text))
- details = text
-
-/datum/feedback_variable/proc/add_details(text)
- if (istext(text))
- text = replacetext(text, " ", "_")
- if (!details)
- details = text
- else
- details += " [text]"
-
-/datum/feedback_variable/proc/get_details()
- return details
-
-/datum/feedback_variable/proc/get_parsed()
- return list(variable,value,details)
+SUBSYSTEM_DEF(blackbox)
+ name = "Blackbox"
+ wait = 6000
+ flags = SS_NO_TICK_CHECK
+
+ var/list/msg_common = list()
+ var/list/msg_science = list()
+ var/list/msg_command = list()
+ var/list/msg_medical = list()
+ var/list/msg_engineering = list()
+ var/list/msg_security = list()
+ var/list/msg_deathsquad = list()
+ var/list/msg_syndicate = list()
+ var/list/msg_service = list()
+ var/list/msg_cargo = list()
+ var/list/msg_other = list()
+
+ var/list/feedback = list() //list of datum/feedback_variable
+
+//poll population
+/datum/controller/subsystem/blackbox/fire()
+ if(!SSdbcore.Connect())
+ return
+ var/playercount = 0
+ for(var/mob/M in GLOB.player_list)
+ if(M.client)
+ playercount += 1
+ var/admincount = GLOB.admins.len
+ var/datum/DBQuery/query_record_playercount = SSdbcore.NewQuery("INSERT INTO [format_table_name("legacy_population")] (playercount, admincount, time, server_ip, server_port) VALUES ([playercount], [admincount], '[SQLtime()]', INET_ATON('[world.internet_address]'), '[world.port]')")
+ query_record_playercount.Execute()
+
+/datum/controller/subsystem/blackbox/Recover()
+ msg_common = SSblackbox.msg_common
+ msg_science = SSblackbox.msg_science
+ msg_command = SSblackbox.msg_command
+ msg_medical = SSblackbox.msg_medical
+ msg_engineering = SSblackbox.msg_engineering
+ msg_security = SSblackbox.msg_security
+ msg_deathsquad = SSblackbox.msg_deathsquad
+ msg_syndicate = SSblackbox.msg_syndicate
+ msg_service = SSblackbox.msg_service
+ msg_cargo = SSblackbox.msg_cargo
+ msg_other = SSblackbox.msg_other
+
+
+ feedback = SSblackbox.feedback
+
+//no touchie
+/datum/controller/subsystem/blackbox/can_vv_get(var_name)
+ if(var_name == "feedback")
+ return FALSE
+ return ..()
+
+/datum/controller/subsystem/blackbox/vv_edit_var(var_name, var_value)
+ return FALSE
+
+/datum/controller/subsystem/blackbox/Shutdown()
+ var/pda_msg_amt = 0
+ var/rc_msg_amt = 0
+
+ for (var/obj/machinery/message_server/MS in GLOB.message_servers)
+ if (MS.pda_msgs.len > pda_msg_amt)
+ pda_msg_amt = MS.pda_msgs.len
+ if (MS.rc_msgs.len > rc_msg_amt)
+ rc_msg_amt = MS.rc_msgs.len
+
+ set_details("radio_usage","")
+
+ add_details("radio_usage","COM-[msg_common.len]")
+ add_details("radio_usage","SCI-[msg_science.len]")
+ add_details("radio_usage","HEA-[msg_command.len]")
+ add_details("radio_usage","MED-[msg_medical.len]")
+ add_details("radio_usage","ENG-[msg_engineering.len]")
+ add_details("radio_usage","SEC-[msg_security.len]")
+ add_details("radio_usage","DTH-[msg_deathsquad.len]")
+ add_details("radio_usage","SYN-[msg_syndicate.len]")
+ add_details("radio_usage","SRV-[msg_service.len]")
+ add_details("radio_usage","CAR-[msg_cargo.len]")
+ add_details("radio_usage","OTH-[msg_other.len]")
+ add_details("radio_usage","PDA-[pda_msg_amt]")
+ add_details("radio_usage","RC-[rc_msg_amt]")
+
+ set_details("round_end","[time2text(world.realtime)]") //This one MUST be the last one that gets set.
+
+ if (!SSdbcore.Connect())
+ return
+
+ var/sqlrowlist = ""
+
+ for (var/datum/feedback_variable/FV in feedback)
+ if (sqlrowlist != "")
+ sqlrowlist += ", " //a comma (,) at the start of the first row to insert will trigger a SQL error
+
+ sqlrowlist += list(list("time" = "Now()", "round_id" = GLOB.round_id, "var_name" = "'[sanitizeSQL(FV.get_variable())]'", "var_value" = FV.get_value(), "details" = "'[sanitizeSQL(FV.get_details())]'"))
+
+ if (sqlrowlist == "")
+ return
+
+ var/datum/DBQuery/query_feedback_save = SSdbcore.NewQuery("INSERT DELAYED IGNORE INTO [format_table_name("feedback")] VALUES " + sqlrowlist)
+ query_feedback_save.Execute()
+
+/datum/controller/subsystem/blackbox/proc/LogBroadcast(blackbox_msg, freq)
+ switch(freq)
+ if(1459)
+ msg_common += blackbox_msg
+ if(1351)
+ msg_science += blackbox_msg
+ if(1353)
+ msg_command += blackbox_msg
+ if(1355)
+ msg_medical += blackbox_msg
+ if(1357)
+ msg_engineering += blackbox_msg
+ if(1359)
+ msg_security += blackbox_msg
+ if(1441)
+ msg_deathsquad += blackbox_msg
+ if(1213)
+ msg_syndicate += blackbox_msg
+ if(1349)
+ msg_service += blackbox_msg
+ if(1347)
+ msg_cargo += blackbox_msg
+ else
+ msg_other += blackbox_msg
+
+/datum/controller/subsystem/blackbox/proc/find_feedback_datum(variable)
+ for(var/datum/feedback_variable/FV in feedback)
+ if(FV.get_variable() == variable)
+ return FV
+
+ var/datum/feedback_variable/FV = new(variable)
+ feedback += FV
+ return FV
+
+/datum/controller/subsystem/blackbox/proc/set_val(variable, value)
+ var/datum/feedback_variable/FV = find_feedback_datum(variable)
+ FV.set_value(value)
+
+/datum/controller/subsystem/blackbox/proc/inc(variable, value)
+ var/datum/feedback_variable/FV = find_feedback_datum(variable)
+ FV.inc(value)
+
+/datum/controller/subsystem/blackbox/proc/dec(variable,value)
+ var/datum/feedback_variable/FV = find_feedback_datum(variable)
+ FV.dec(value)
+
+/datum/controller/subsystem/blackbox/proc/set_details(variable,details)
+ var/datum/feedback_variable/FV = find_feedback_datum(variable)
+ FV.set_details(details)
+
+/datum/controller/subsystem/blackbox/proc/add_details(variable,details)
+ var/datum/feedback_variable/FV = find_feedback_datum(variable)
+ FV.add_details(details)
+
+/datum/controller/subsystem/blackbox/proc/ReportDeath(mob/living/L)
+ if(!SSdbcore.Connect())
+ return
+ if(!L || !L.key || !L.mind)
+ return
+ var/turf/T = get_turf(L)
+ var/area/placeofdeath = get_area(T.loc)
+ var/sqlname = sanitizeSQL(L.real_name)
+ var/sqlkey = sanitizeSQL(L.ckey)
+ var/sqljob = sanitizeSQL(L.mind.assigned_role)
+ var/sqlspecial = sanitizeSQL(L.mind.special_role)
+ var/sqlpod = sanitizeSQL(placeofdeath.name)
+ var/laname
+ var/lakey
+ if(L.lastattacker && ismob(L.lastattacker))
+ var/mob/LA = L.lastattacker
+ laname = sanitizeSQL(LA.real_name)
+ lakey = sanitizeSQL(LA.key)
+ var/sqlgender = sanitizeSQL(L.gender)
+ var/sqlbrute = sanitizeSQL(L.getBruteLoss())
+ var/sqlfire = sanitizeSQL(L.getFireLoss())
+ var/sqlbrain = sanitizeSQL(L.getBrainLoss())
+ var/sqloxy = sanitizeSQL(L.getOxyLoss())
+ var/sqltox = sanitizeSQL(L.getStaminaLoss())
+ var/sqlclone = sanitizeSQL(L.getStaminaLoss())
+ var/sqlstamina = sanitizeSQL(L.getStaminaLoss())
+ var/coord = sanitizeSQL("[L.x], [L.y], [L.z]")
+ var/map = sanitizeSQL(SSmapping.config.map_name)
+ var/datum/DBQuery/query_report_death = SSdbcore.NewQuery("INSERT INTO [format_table_name("death")] (name, byondkey, job, special, pod, tod, laname, lakey, gender, bruteloss, fireloss, brainloss, oxyloss, toxloss, cloneloss, staminaloss, coord, mapname, server_ip, server_port) VALUES ('[sqlname]', '[sqlkey]', '[sqljob]', '[sqlspecial]', '[sqlpod]', '[SQLtime()]', '[laname]', '[lakey]', '[sqlgender]', [sqlbrute], [sqlfire], [sqlbrain], [sqloxy], [sqltox], [sqlclone], [sqlstamina], '[coord]', '[map]', INET_ATON('[world.internet_address]'), '[world.port]')")
+ query_report_death.Execute()
+
+
+//feedback variable datum, for storing all kinds of data
+/datum/feedback_variable
+ var/variable
+ var/value
+ var/details
+
+/datum/feedback_variable/New(param_variable, param_value = 0)
+ variable = param_variable
+ value = param_value
+
+/datum/feedback_variable/proc/inc(num = 1)
+ if (isnum(value))
+ value += num
+ else
+ value = text2num(value)
+ if (isnum(value))
+ value += num
+ else
+ value = num
+
+/datum/feedback_variable/proc/dec(num = 1)
+ if (isnum(value))
+ value -= num
+ else
+ value = text2num(value)
+ if (isnum(value))
+ value -= num
+ else
+ value = -num
+
+/datum/feedback_variable/proc/set_value(num)
+ if (isnum(num))
+ value = num
+
+/datum/feedback_variable/proc/get_value()
+ if (!isnum(value))
+ return 0
+ return value
+
+/datum/feedback_variable/proc/get_variable()
+ return variable
+
+/datum/feedback_variable/proc/set_details(text)
+ if (istext(text))
+ details = text
+
+/datum/feedback_variable/proc/add_details(text)
+ if (istext(text))
+ text = replacetext(text, " ", "_")
+ if (!details)
+ details = text
+ else
+ details += " [text]"
+
+/datum/feedback_variable/proc/get_details()
+ return details
+
+/datum/feedback_variable/proc/get_parsed()
+ return list(variable,value,details)
\ No newline at end of file
diff --git a/code/game/gamemodes/cult/cult_comms.dm b/code/game/gamemodes/cult/cult_comms.dm
index 2d2b5d1fa9..6651f68556 100644
--- a/code/game/gamemodes/cult/cult_comms.dm
+++ b/code/game/gamemodes/cult/cult_comms.dm
@@ -298,4 +298,5 @@
if(GLOB.blood_target)
to_chat(B.current,"The blood mark has expired!")
B.current.client.images -= GLOB.blood_target_image
- QDEL_NULL(GLOB.blood_target)
\ No newline at end of file
+ QDEL_NULL(GLOB.blood_target_image)
+ GLOB.blood_target = null
diff --git a/code/modules/mining/mine_items.dm b/code/modules/mining/mine_items.dm
index d15896a8db..3df4d521ff 100644
--- a/code/modules/mining/mine_items.dm
+++ b/code/modules/mining/mine_items.dm
@@ -300,31 +300,47 @@
//Door
/obj/machinery/door/airlock/survival_pod
name = "airlock"
- icon = 'icons/obj/doors/airlocks/survival/horizontal/survival.dmi'
- overlays_file = 'icons/obj/doors/airlocks/survival/horizontal/survival_overlays.dmi'
+ icon = 'icons/obj/doors/airlocks/survival/survival.dmi'
+ overlays_file = 'icons/obj/doors/airlocks/survival/survival_overlays.dmi'
assemblytype = /obj/structure/door_assembly/door_assembly_pod
opacity = 0
glass = 1
+ var/expected_dir = SOUTH //we visually turn when shuttle rotated, but need to not turn for any other reason
+
+/obj/machinery/door/airlock/survival_pod/setDir(direction)
+ direction = expected_dir
+ ..()
+
+/obj/machinery/door/airlock/survival_pod/shuttleRotate(rotation)
+ expected_dir = angle2dir(rotation+dir2angle(dir))
+ ..()
/obj/machinery/door/airlock/survival_pod/vertical
- icon = 'icons/obj/doors/airlocks/survival/vertical/survival.dmi'
- overlays_file = 'icons/obj/doors/airlocks/survival/vertical/survival_overlays.dmi'
- assemblytype = /obj/structure/door_assembly/door_assembly_pod/vertical
+ dir = EAST
+ expected_dir = EAST
/obj/structure/door_assembly/door_assembly_pod
name = "pod airlock assembly"
- icon = 'icons/obj/doors/airlocks/survival/horizontal/survival.dmi'
- overlays_file = 'icons/obj/doors/airlocks/survival/horizontal/survival_overlays.dmi'
+ icon = 'icons/obj/doors/airlocks/survival/survival.dmi'
+ overlays_file = 'icons/obj/doors/airlocks/survival/survival_overlays.dmi'
airlock_type = /obj/machinery/door/airlock/survival_pod
anchored = 1
state = 1
mineral = "glass"
material = "glass"
+ var/expected_dir = SOUTH
+
+/obj/structure/door_assembly/door_assembly_pod/setDir(direction)
+ direction = expected_dir
+ ..()
+
+/obj/structure/door_assembly/door_assembly_pod/shuttleRotate(rotation)
+ expected_dir = angle2dir(rotation+dir2angle(dir))
+ ..()
/obj/structure/door_assembly/door_assembly_pod/vertical
- icon = 'icons/obj/doors/airlocks/survival/vertical/survival.dmi'
- overlays_file = 'icons/obj/doors/airlocks/survival/vertical/survival_overlays.dmi'
- airlock_type = /obj/machinery/door/airlock/survival_pod/vertical
+ dir = EAST
+ expected_dir = EAST
//Table
/obj/structure/table/survival_pod
diff --git a/code/modules/mob/living/simple_animal/friendly/drone/_drone.dm b/code/modules/mob/living/simple_animal/friendly/drone/_drone.dm
index e98a57d0a6..d675458485 100644
--- a/code/modules/mob/living/simple_animal/friendly/drone/_drone.dm
+++ b/code/modules/mob/living/simple_animal/friendly/drone/_drone.dm
@@ -55,7 +55,7 @@
var/laws = \
"1. You may not involve yourself in the matters of another being, even if such matters conflict with Law Two or Law Three, unless the other being is another Drone.\n"+\
"2. You may not harm any being, regardless of intent or circumstance.\n"+\
- "3. Your goals are to build, maintain, repair, improve, and power the station to the best of your abilities, You must never actively work against these goals."
+ "3. Your goals are to build, maintain, repair, improve, and provide power to the best of your abilities, You must never actively work against these goals."
var/light_on = 0
var/heavy_emp_damage = 25 //Amount of damage sustained if hit by a heavy EMP pulse
var/alarms = list("Atmosphere" = list(), "Fire" = list(), "Power" = list())
diff --git a/code/modules/paperwork/folders.dm b/code/modules/paperwork/folders.dm
index 1fd71e9bba..f98cb5d0e9 100644
--- a/code/modules/paperwork/folders.dm
+++ b/code/modules/paperwork/folders.dm
@@ -78,8 +78,8 @@
name = "folder- 'TOP SECRET'"
desc = "A folder stamped \"Top Secret - Property of Nanotrasen Corporation. Unauthorized distribution is punishable by death.\""
-/obj/item/weapon/folder/documents/New()
- ..()
+/obj/item/weapon/folder/documents/Initialize()
+ . = ..()
new /obj/item/documents/nanotrasen(src)
update_icon()
@@ -91,20 +91,20 @@
/obj/item/weapon/folder/syndicate/red
icon_state = "folder_sred"
-/obj/item/weapon/folder/syndicate/red/New()
- ..()
+/obj/item/weapon/folder/syndicate/red/Initialize()
+ . = ..()
new /obj/item/documents/syndicate/red(src)
update_icon()
/obj/item/weapon/folder/syndicate/blue
icon_state = "folder_sblue"
-/obj/item/weapon/folder/syndicate/blue/New()
- ..()
+/obj/item/weapon/folder/syndicate/blue/Initialize()
+ . = ..()
new /obj/item/documents/syndicate/blue(src)
update_icon()
-/obj/item/weapon/folder/syndicate/mining/New()
- ..()
+/obj/item/weapon/folder/syndicate/mining/Initialize()
+ . = ..()
new /obj/item/documents/syndicate/mining(src)
- update_icon()
\ No newline at end of file
+ update_icon()
diff --git a/code/modules/projectiles/guns/energy/kinetic_accelerator.dm b/code/modules/projectiles/guns/energy/kinetic_accelerator.dm
index 4a7115b0c6..ca8b0b85c0 100644
--- a/code/modules/projectiles/guns/energy/kinetic_accelerator.dm
+++ b/code/modules/projectiles/guns/energy/kinetic_accelerator.dm
@@ -61,6 +61,7 @@
for(var/A in get_modkits())
var/obj/item/borg/upgrade/modkit/M = A
M.modify_projectile(K)
+ K.kinetic_modules += M //do something special on-hit, easy!
/obj/item/weapon/gun/energy/kinetic_accelerator/cyborg
holds_charge = TRUE
@@ -165,9 +166,11 @@
log_override = TRUE
var/pressure_decrease = 0.25
- var/turf_aoe = FALSE
- var/mob_aoe = 0
- var/list/hit_overlays = list()
+ var/list/kinetic_modules = list()
+
+/obj/item/projectile/kinetic/Destroy()
+ QDEL_NULL(kinetic_modules)
+ return ..()
/obj/item/projectile/kinetic/prehit(atom/target)
var/turf/target_turf = get_turf(target)
@@ -197,23 +200,15 @@
M.gets_drilled(firer)
var/obj/effect/temp_visual/kinetic_blast/K = new /obj/effect/temp_visual/kinetic_blast(target_turf)
K.color = color
- for(var/type in hit_overlays)
- new type(target_turf)
- if(turf_aoe)
- for(var/T in RANGE_TURFS(1, target_turf) - target_turf)
- if(ismineralturf(T))
- var/turf/closed/mineral/M = T
- M.gets_drilled(firer)
- if(mob_aoe)
- for(var/mob/living/L in range(1, target_turf) - firer - target)
- var/armor = L.run_armor_check(def_zone, flag, "", "", armour_penetration)
- L.apply_damage(damage*mob_aoe, damage_type, def_zone, armor)
- to_chat(L, "You're struck by a [name]!")
+ for(var/obj/item/borg/upgrade/modkit/M in kinetic_modules)
+ if(QDELETED(M)) //whoever shot this was very, very unfortunate.
+ continue
+ M.projectile_strike(src, target_turf, target)
//Modkits
/obj/item/borg/upgrade/modkit
- name = "modification kit"
+ name = "kinetic accelerator modification kit"
desc = "An upgrade for kinetic accelerators."
icon = 'icons/obj/objects.dmi'
icon_state = "modkit"
@@ -266,14 +261,13 @@
to_chat(user, "You don't have room([KA.get_remaining_mod_capacity()]% remaining, [cost]% needed) to install this modkit. Use a crowbar to remove existing modkits.")
. = FALSE
-
-
/obj/item/borg/upgrade/modkit/proc/uninstall(obj/item/weapon/gun/energy/kinetic_accelerator/KA)
forceMove(get_turf(KA))
KA.modkits -= src
/obj/item/borg/upgrade/modkit/proc/modify_projectile(obj/item/projectile/kinetic/K)
+/obj/item/borg/upgrade/modkit/proc/projectile_strike(obj/item/projectile/kinetic/K, turf/target_turf, atom/target)
//Range
/obj/item/borg/upgrade/modkit/range
@@ -315,21 +309,72 @@
//AoE blasts
/obj/item/borg/upgrade/modkit/aoe
modifier = 0
+ var/turf_aoe = FALSE
+ var/stats_stolen = FALSE
+
+/obj/item/borg/upgrade/modkit/aoe/install(obj/item/weapon/gun/energy/kinetic_accelerator/KA, mob/user)
+ . = ..()
+ if(.)
+ for(var/obj/item/borg/upgrade/modkit/aoe/AOE in KA.modkits) //make sure only one of the aoe modules has values if somebody has multiple
+ if(AOE.stats_stolen)
+ continue
+ modifier += AOE.modifier //take its modifiers
+ AOE.modifier = 0
+ turf_aoe += AOE.turf_aoe
+ AOE.turf_aoe = FALSE
+ AOE.stats_stolen = TRUE
+
+/obj/item/borg/upgrade/modkit/aoe/uninstall(obj/item/weapon/gun/energy/kinetic_accelerator/KA)
+ ..()
+ modifier = initial(modifier) //get our modifiers back
+ turf_aoe = initial(turf_aoe)
+ if(stats_stolen) //if we had our stats stolen, find the stealer and take them from it
+ for(var/obj/item/borg/upgrade/modkit/aoe/AOE in KA.modkits)
+ if(AOE.stats_stolen)
+ continue
+ AOE.modifier -= modifier
+ AOE.turf_aoe -= turf_aoe
+ else //otherwise, reset the stolen stats and have it recalculate
+ var/obj/item/borg/upgrade/modkit/aoe/new_stealer
+ for(var/obj/item/borg/upgrade/modkit/aoe/AOE in KA.modkits)
+ if(!new_stealer)
+ new_stealer = AOE //just make the first one a stealer
+ AOE.modifier = initial(AOE.modifier)
+ AOE.turf_aoe = initial(AOE.turf_aoe)
+ AOE.stats_stolen = FALSE
+ if(new_stealer) //if there's no stealer, then there's no other aoe modkits
+ for(var/obj/item/borg/upgrade/modkit/aoe/AOE in KA.modkits)
+ if(AOE != new_stealer)
+ new_stealer.modifier += AOE.modifier
+ AOE.modifier = 0
+ new_stealer.turf_aoe += AOE.turf_aoe
+ AOE.turf_aoe = FALSE
+ AOE.stats_stolen = TRUE
+ stats_stolen = FALSE
/obj/item/borg/upgrade/modkit/aoe/modify_projectile(obj/item/projectile/kinetic/K)
K.name = "kinetic explosion"
- if(!K.turf_aoe && !K.mob_aoe)
- K.hit_overlays += /obj/effect/temp_visual/explosion/fast
- K.mob_aoe += modifier
+
+/obj/item/borg/upgrade/modkit/aoe/projectile_strike(obj/item/projectile/kinetic/K, turf/target_turf, atom/target)
+ if(stats_stolen)
+ return
+ new /obj/effect/overlay/temp_visual/explosion/fast(target_turf)
+ if(turf_aoe)
+ for(var/T in RANGE_TURFS(1, target_turf) - target_turf)
+ if(ismineralturf(T))
+ var/turf/closed/mineral/M = T
+ M.gets_drilled(K.firer)
+ if(modifier)
+ for(var/mob/living/L in range(1, target_turf) - K.firer - target)
+ var/armor = L.run_armor_check(K.def_zone, K.flag, "", "", K.armour_penetration)
+ L.apply_damage(K.damage*modifier, K.damage_type, K.def_zone, armor)
+ to_chat(L, "You're struck by a [K.name]!")
/obj/item/borg/upgrade/modkit/aoe/turfs
name = "mining explosion"
desc = "Causes the kinetic accelerator to destroy rock in an AoE."
denied_type = /obj/item/borg/upgrade/modkit/aoe/turfs
-
-/obj/item/borg/upgrade/modkit/aoe/turfs/modify_projectile(obj/item/projectile/kinetic/K)
- ..()
- K.turf_aoe = TRUE
+ turf_aoe = TRUE
/obj/item/borg/upgrade/modkit/aoe/turfs/andmobs
name = "offensive mining explosion"
diff --git a/html/changelogs/AutoChangeLog-pr-1075.yml b/html/changelogs/AutoChangeLog-pr-1075.yml
new file mode 100644
index 0000000000..d3fe3a44bf
--- /dev/null
+++ b/html/changelogs/AutoChangeLog-pr-1075.yml
@@ -0,0 +1,4 @@
+author: "RandomMarine"
+delete-after: True
+changes:
+ - tweak: "Drone laws no longer restrict drones to the station."
diff --git a/icons/obj/doors/airlocks/survival/horizontal/survival.dmi b/icons/obj/doors/airlocks/survival/horizontal/survival.dmi
deleted file mode 100644
index 5dbe3de771..0000000000
Binary files a/icons/obj/doors/airlocks/survival/horizontal/survival.dmi and /dev/null differ
diff --git a/icons/obj/doors/airlocks/survival/horizontal/survival_overlays.dmi b/icons/obj/doors/airlocks/survival/horizontal/survival_overlays.dmi
deleted file mode 100644
index 9bdb3be3cd..0000000000
Binary files a/icons/obj/doors/airlocks/survival/horizontal/survival_overlays.dmi and /dev/null differ
diff --git a/icons/obj/doors/airlocks/survival/survival.dmi b/icons/obj/doors/airlocks/survival/survival.dmi
new file mode 100644
index 0000000000..f96d174c47
Binary files /dev/null and b/icons/obj/doors/airlocks/survival/survival.dmi differ
diff --git a/icons/obj/doors/airlocks/survival/survival_overlays.dmi b/icons/obj/doors/airlocks/survival/survival_overlays.dmi
new file mode 100644
index 0000000000..e8bede8217
Binary files /dev/null and b/icons/obj/doors/airlocks/survival/survival_overlays.dmi differ
diff --git a/icons/obj/doors/airlocks/survival/vertical/survival.dmi b/icons/obj/doors/airlocks/survival/vertical/survival.dmi
deleted file mode 100644
index c5063da5a0..0000000000
Binary files a/icons/obj/doors/airlocks/survival/vertical/survival.dmi and /dev/null differ
diff --git a/icons/obj/doors/airlocks/survival/vertical/survival_overlays.dmi b/icons/obj/doors/airlocks/survival/vertical/survival_overlays.dmi
deleted file mode 100644
index 26d0c0d403..0000000000
Binary files a/icons/obj/doors/airlocks/survival/vertical/survival_overlays.dmi and /dev/null differ