diff --git a/baystation12.dme b/baystation12.dme
index b3f1f1a927b..31026a8d5dc 100644
--- a/baystation12.dme
+++ b/baystation12.dme
@@ -736,6 +736,7 @@
#include "code\game\machinery\magnet.dm"
#include "code\game\machinery\mass_driver.dm"
#include "code\game\machinery\mech_recharger.dm"
+#include "code\game\machinery\mind_engraver.dm"
#include "code\game\machinery\navbeacon.dm"
#include "code\game\machinery\newscaster.dm"
#include "code\game\machinery\nuclear_bomb.dm"
@@ -821,6 +822,7 @@
#include "code\game\machinery\computer\guestpass.dm"
#include "code\game\machinery\computer\law.dm"
#include "code\game\machinery\computer\message.dm"
+#include "code\game\machinery\computer\mind_engraver_control.dm"
#include "code\game\machinery\computer\Operating.dm"
#include "code\game\machinery\computer\pandemic.dm"
#include "code\game\machinery\computer\prisoner.dm"
@@ -986,6 +988,7 @@
#include "code\game\objects\items\devices\lightreplacer.dm"
#include "code\game\objects\items\devices\megaphone.dm"
#include "code\game\objects\items\devices\microlaser_radioactive.dm"
+#include "code\game\objects\items\devices\mind_engraver_chip.dm"
#include "code\game\objects\items\devices\modkit.dm"
#include "code\game\objects\items\devices\multitool.dm"
#include "code\game\objects\items\devices\oxycandle.dm"
@@ -1104,6 +1107,7 @@
#include "code\game\objects\items\weapons\circuitboards\machinery\household.dm"
#include "code\game\objects\items\weapons\circuitboards\machinery\mech_recharger.dm"
#include "code\game\objects\items\weapons\circuitboards\machinery\medical.dm"
+#include "code\game\objects\items\weapons\circuitboards\machinery\mind_engraver.dm"
#include "code\game\objects\items\weapons\circuitboards\machinery\mining.dm"
#include "code\game\objects\items\weapons\circuitboards\machinery\mining_drill.dm"
#include "code\game\objects\items\weapons\circuitboards\machinery\oxyregenerator.dm"
diff --git a/code/__defines/research.dm b/code/__defines/research.dm
index 722c17cff6b..a13d01887af 100644
--- a/code/__defines/research.dm
+++ b/code/__defines/research.dm
@@ -1,19 +1,20 @@
#define SHEET_MATERIAL_AMOUNT 2000
-#define TECH_MATERIAL "materials"
-#define TECH_ENGINEERING "engineering"
-#define TECH_PHORON "phorontech"
-#define TECH_POWER "powerstorage"
-#define TECH_BLUESPACE "bluespace"
-#define TECH_BIO "biotech"
-#define TECH_COMBAT "combat"
-#define TECH_MAGNET "magnets"
-#define TECH_DATA "programming"
-#define TECH_ESOTERIC "esoteric"
+#define TECH_MATERIAL "materials"
+#define TECH_ENGINEERING "engineering"
+#define TECH_PHORON "phorontech"
+#define TECH_POWER "powerstorage"
+#define TECH_BLUESPACE "bluespace"
+#define TECH_BIO "biotech"
+#define TECH_COMBAT "combat"
+#define TECH_MAGNET "magnets"
+#define TECH_DATA "programming"
+#define TECH_ESOTERIC "esoteric"
+#define TECH_DOMINION "dominion"
#define IMPRINTER 0x1 //For circuits. Uses glass/chemicals.
#define PROTOLATHE 0x2 //New stuff. Uses glass/metal/chemicals
#define MECHFAB 0x4 //Mechfab
#define CHASSIS 0x8 //For protolathe, but differently
-#define T_BOARD(name) "circuit board (" + (name) + ")"
\ No newline at end of file
+#define T_BOARD(name) "circuit board (" + (name) + ")"
diff --git a/code/game/machinery/computer/mind_engraver_control.dm b/code/game/machinery/computer/mind_engraver_control.dm
new file mode 100644
index 00000000000..82ff9a5d5a1
--- /dev/null
+++ b/code/game/machinery/computer/mind_engraver_control.dm
@@ -0,0 +1,106 @@
+/obj/machinery/computer/mind_engraver_control
+ name = "mind engraver control console"
+ density = TRUE
+ anchored = TRUE
+ icon_keyboard = "generic_key"
+ icon_screen = "holocontrol"
+ machine_name = "mind engraver control console"
+ machine_desc = "A control console used to interface with the mind engraver."
+
+ var/obj/machinery/mind_engraver/linked_engraver = null
+ var/model_name = "Mind Engraver v1.3"
+ var/model_producer = "Terran Dominion"
+
+/obj/machinery/computer/mind_engraver_control/Initialize()
+ . = ..()
+ for(var/obj/machinery/mind_engraver/ME in range(src, 1))
+ LinkEngraver(ME)
+ break
+
+/obj/machinery/computer/mind_engraver_control/Destroy()
+ linked_engraver = null
+ return ..()
+
+/obj/machinery/computer/mind_engraver_control/attackby(obj/item/I, mob/user)
+ if(istype(I, /obj/item/mind_engraver_chip))
+ to_chat(user, SPAN_WARNING("\The [I] must be inserted in the mind engraver, not the console!"))
+ return TRUE
+ if(istype(I, /obj/item/device/multitool))
+ var/obj/item/device/multitool/M = I
+ var/obj/machinery/mind_engraver/buffer = M.get_buffer(/obj/machinery/mind_engraver)
+ if(!istype(buffer))
+ return TRUE
+ linked_engraver = buffer
+ to_chat(user, SPAN_NOTICE("You link [src] with the machine in [I]'s buffer."))
+ return ..()
+
+/obj/machinery/computer/mind_engraver_control/interface_interact(user)
+ interact(user)
+ return TRUE
+
+/obj/machinery/computer/mind_engraver_control/interact(mob/living/user)
+ //If the computer is being hacked or is emagged, display the reboot message.
+ if(!istype(linked_engraver))
+ to_chat(user, SPAN_WARNING("\The [src] is not linked to any machine!"))
+ return
+
+ var/list/dat = list()
+ dat += "
[model_name]
"
+ dat += "[model_producer], all rights reserved.
"
+
+ dat += "Lock status: [linked_engraver.locked ? "Locked" : "Unlocked"]
"
+
+ if(linked_engraver.occupant)
+ dat += "Occupant: [linked_engraver.occupant.name]
"
+ if(linked_engraver.currently_imprinting)
+ dat += "\The [linked_engraver] is currently operating!
"
+ dat += "Approximate time remaining: [round((linked_engraver.imprint_end - world.time) / 10)] seconds
"
+ else if(istype(linked_engraver.nanochip))
+ dat += "Begin Imprinting Process
"
+
+ if(!istype(linked_engraver.nanochip))
+ dat += "Nanochip missing!
"
+ else if(!linked_engraver.nanochip.stored_data)
+ dat += "Nanochip data missing!
"
+ else
+ dat += "Currently installed nanochip:
"
+ dat += "Type: [linked_engraver.nanochip.stored_data.chip_type]
"
+ dat += "Function: [linked_engraver.nanochip.stored_data.PrintFunction()]
"
+
+ var/datum/browser/popup = new(user, "mind_engraver", "Mind Engraver Control Console", 480, 600)
+ popup.set_content(JOINTEXT(dat))
+ popup.open()
+ return
+
+/obj/machinery/computer/mind_engraver_control/Topic(href, href_list)
+ . = ..()
+ if(.)
+ return
+
+ if(href_list["lock"])
+ if(istype(linked_engraver))
+ linked_engraver.ToggleLock()
+
+ if(href_list["start"])
+ if(istype(linked_engraver))
+ if(linked_engraver.StartImprint())
+ to_chat(usr, SPAN_NOTICE("\The [linked_engraver] has begun the imprinting process!"))
+ else
+ to_chat(usr, SPAN_WARNING("\The [linked_engraver] failed to start a new imprinting process!"))
+
+ return interact(usr)
+
+/obj/machinery/computer/mind_engraver_control/proc/LinkEngraver(obj/machinery/mind_engraver/ME)
+ if(istype(linked_engraver))
+ UnregisterSignal(linked_engraver, COMSIG_PARENT_QDELETING)
+ if(!istype(ME))
+ return
+
+ linked_engraver = ME
+ RegisterSignal(ME, COMSIG_PARENT_QDELETING, .proc/OnLinkDestroyed)
+
+/obj/machinery/computer/mind_engraver_control/proc/OnLinkDestroyed()
+ SIGNAL_HANDLER
+
+ UnregisterSignal(linked_engraver, COMSIG_PARENT_QDELETING)
+ linked_engraver = null
diff --git a/code/game/machinery/mind_engraver.dm b/code/game/machinery/mind_engraver.dm
new file mode 100644
index 00000000000..89c63367ca5
--- /dev/null
+++ b/code/game/machinery/mind_engraver.dm
@@ -0,0 +1,310 @@
+/obj/machinery/mind_engraver
+ name = "mind engraver"
+ desc = "A complicated scanner connected to even more complicated machinery."
+ icon = 'icons/obj/machines/mind_engraver.dmi'
+ icon_state = "engraver_0"
+ density = FALSE
+ anchored = TRUE
+ construct_state = /decl/machine_construction/default/panel_closed
+
+ machine_name = "mind engraver"
+ machine_desc = "Mind Engravers are an ultra-tech device that was used by Terran Dominion during its power peak. \
+ Allows to imprint knowledge, skills or new memories into the mind of its occupant."
+
+ var/obj/item/mind_engraver_chip/nanochip = null
+ var/mob/living/carbon/human/occupant = null
+ var/currently_imprinting = FALSE
+ /// How much time it takes to finish a single process
+ var/imprint_time = 80 SECONDS
+ /// World time when imprint process is finished
+ var/imprint_end = 0
+ /// If TRUE - prevents the user from leaving, unless resisted out of
+ var/locked = FALSE
+ /// When the next breakout attempt can be done
+ var/breakout_cooldown = 0
+ /// Base chance on every Process() tick to do random damage
+ var/mistake_chance = 4
+
+/obj/machinery/mind_engraver/Destroy()
+ INVOKE_ASYNC(src, .proc/GoOut)
+ RemoveChip()
+ return ..()
+
+/obj/machinery/mind_engraver/Process()
+ if(stat & (NOPOWER|BROKEN))
+ if(occupant)
+ GoOutPowerLoss()
+ return
+
+ if(!currently_imprinting)
+ return
+
+ if(!occupant || !nanochip)
+ StopImprint()
+ return
+
+ if(world.time < imprint_end)
+ var/obj/item/organ/external/O_head = occupant.get_organ(BP_HEAD)
+ if(prob(mistake_chance) && istype(O_head))
+ var/choice = pick(1, 2, 3)
+ switch(choice)
+ if(1)
+ O_head.take_external_damage(rand(5, 10), 0, DAM_SHARP|DAM_EDGE)
+ to_chat(occupant, SPAN_DANGER("\The [src] misplaces a surgical incision, damaging your skin!"))
+ if(2)
+ occupant.adjustBrainLoss(rand(5, 15))
+ to_chat(occupant, SPAN_DANGER("\The [src]'s laser accidentaly passes over the exposed sections of your head, damaging your brain!"))
+ if(3)
+ occupant.custom_pain(SPAN_WARNING("You feel sudden sharp pain in your head!"), 20, affecting = O_head)
+ return
+
+ FinishImprint()
+
+ return
+
+/obj/machinery/mind_engraver/examine(mob/user, distance)
+ . = ..()
+ if(distance <= 1 && istype(nanochip))
+ to_chat(user, SPAN_NOTICE("It has a [nanochip] installed in it."))
+
+/obj/machinery/mind_engraver/attackby(obj/item/I, mob/user)
+ if(istype(I, /obj/item/mind_engraver_chip))
+ add_fingerprint(user)
+ if(nanochip)
+ to_chat(user, SPAN_WARNING("\The [src] has a nanochip already installed."))
+ return
+ if(!user.unEquip(I, src))
+ return
+ user.visible_message(
+ SPAN_NOTICE("\The [user] inserts \a [I] into \the [src]."),
+ SPAN_NOTICE("You insert \a [I] into \the [src]."),
+ )
+ nanochip = I
+ return TRUE
+ if(istype(I, /obj/item/device/multitool))
+ var/obj/item/device/multitool/M = I
+ M.set_buffer(src)
+ to_chat(user, SPAN_NOTICE("You save the data in [M]'s buffer. It can now be used to link with the mind engraver console."))
+ return TRUE
+ return ..()
+
+/obj/machinery/mind_engraver/AltClick(mob/user)
+ if(CanDefaultInteract(user))
+ RemoveChip(user)
+ return
+ return ..()
+
+/obj/machinery/mind_engraver/relaymove(mob/user)
+ . = ..()
+ if(locked)
+ to_chat(user, SPAN_WARNING("\The [src] is currently locked and will not let you out!"))
+ return
+ INVOKE_ASYNC(src, .proc/GoOut)
+
+/obj/machinery/mind_engraver/mob_breakout(mob/living/escapee)
+ . = ..()
+ if(breakout_cooldown > world.time || escapee != occupant || !locked)
+ return FALSE
+
+ . = TRUE
+
+ to_chat(escapee, SPAN_WARNING("You lean on the back of \the [src] and start pushing the door open."))
+ visible_message(SPAN_DANGER("\The [src] begins to shake violently!"))
+
+ for(var/i = 1 to 3)
+ breakout_cooldown = world.time + 20 SECONDS
+ if(!do_after(escapee, (10 SECONDS), incapacitation_flags = INCAPACITATION_DEFAULT & ~INCAPACITATION_RESTRAINED))
+ return FALSE
+
+ if(!locked)
+ INVOKE_ASYNC(src, .proc/GoOut)
+ return FALSE
+
+ playsound(src, 'sound/effects/glassbash.ogg', i * 30, 1)
+ shake_animation()
+ add_fingerprint(escapee)
+
+ ToggleLock(FALSE)
+ INVOKE_ASYNC(src, .proc/GoOut)
+ to_chat(escapee, SPAN_NOTICE("You successfully break out!"))
+ visible_message(SPAN_DANGER("\The [escapee] successfully broke out of \the [src]!"))
+ return TRUE
+
+/obj/machinery/mind_engraver/emp_act(severity)
+ if(stat & (BROKEN|NOPOWER))
+ return
+
+ . = ..()
+
+ if(prob(33))
+ GoOutPowerLoss()
+ return
+
+/obj/machinery/mind_engraver/MouseDrop_T(mob/living/carbon/human/H, mob/user)
+ if(!CanMouseDrop(H, user))
+ return
+ if(!istype(H))
+ return
+ if(H.buckled)
+ to_chat(user, SPAN_WARNING("Unbuckle the subject before attempting to move them."))
+ return
+ GoIn(H, user)
+
+/obj/machinery/mind_engraver/RefreshParts()
+ . = ..()
+ var/T = Clamp(total_component_rating_of_type(/obj/item/stock_parts/manipulator), 1, 12)
+ imprint_time = initial(imprint_time) / (T * 0.5)
+
+ T = Clamp(total_component_rating_of_type(/obj/item/stock_parts/capacitor), 1, 12)
+ mistake_chance = max(0, 1 + initial(mistake_chance) - (T * 0.25))
+
+/obj/machinery/mind_engraver/on_update_icon()
+ overlays.Cut()
+
+ if(!occupant)
+ icon_state = "engraver_0"
+ density = FALSE
+ if(locked)
+ icon_state = "engraver_1"
+ density = TRUE
+ else
+ icon_state = "engraver_0"
+ density = FALSE
+ else
+ icon_state = "engraver_2"
+ density = TRUE
+
+ if(locked)
+ overlays += image(icon, "locked_overlay")
+
+/obj/machinery/mind_engraver/proc/SetOccupant(mob/living/carbon/human/H)
+ occupant = H
+ update_icon()
+ if(!occupant)
+ update_use_power(POWER_USE_IDLE)
+ return
+ occupant.forceMove(src)
+ occupant.stop_pulling()
+ if(occupant.client)
+ occupant.client.perspective = EYE_PERSPECTIVE
+ occupant.client.eye = src
+
+ update_use_power(POWER_USE_ACTIVE)
+
+/obj/machinery/mind_engraver/proc/GoIn(mob/living/carbon/human/H, mob/user)
+ if(!istype(H))
+ return
+ if(stat & (BROKEN|NOPOWER))
+ return
+ if(occupant)
+ to_chat(user, SPAN_WARNING("\The [src] is already occupied."))
+ return
+ if(locked)
+ to_chat(user, SPAN_WARNING("\The [src] is locked."))
+ return
+
+ if(H == user)
+ visible_message(SPAN_NOTICE("\The [user] starts climbing into \the [src]."))
+ else
+ visible_message(SPAN_WARNING("\The [user] starts putting [H] into \the [src]."))
+
+ if(do_after(user, 20, src))
+ if(QDELETED(H))
+ return
+ if(get_dist(H, user) > 1)
+ return
+ if(occupant)
+ to_chat(user, SPAN_WARNING("\The [src] is already occupied."))
+ return
+ if(locked)
+ to_chat(user, SPAN_WARNING("\The [src] is locked."))
+ return
+ if(H.buckled)
+ to_chat(user, SPAN_WARNING("Unbuckle the subject before attempting to move them."))
+ return
+ SetOccupant(H)
+
+/obj/machinery/mind_engraver/proc/GoOut()
+ if(!occupant)
+ return
+
+ if(locked)
+ return
+
+ if(currently_imprinting)
+ StopImprint()
+ var/obj/item/organ/external/O_head = occupant.get_organ(BP_HEAD)
+ if(O_head)
+ to_chat(occupant, SPAN_DANGER("\The [src]'s tools quickly disengage away from your brain, leaving a terrible tear!"))
+ occupant.adjustBrainLoss(100)
+ occupant.custom_pain(SPAN_WARNING("It feels as if your head burst open!"), 50, affecting = O_head)
+ O_head.take_external_damage(35, 0, DAM_SHARP|DAM_EDGE)
+ if(!(stat & (NOPOWER|BROKEN)))
+ visible_message(SPAN_DANGER("\The [src] blares an alarm as its occupant leaves mid-procedure!"))
+ playsound(src, 'sound/machines/warning-buzzer.ogg', 50, TRUE, 7)
+
+ if(occupant.client)
+ occupant.client.eye = occupant.client.mob
+ occupant.client.perspective = MOB_PERSPECTIVE
+ occupant.dropInto(loc)
+ SetOccupant(null)
+
+/obj/machinery/mind_engraver/proc/GoOutPowerLoss()
+ if(!occupant)
+ return
+
+ visible_message(SPAN_DANGER("\The [src] blares an alarm as it loses power[locked ? "" : " and forcefuly ejects the occupant"]!"))
+ playsound(src, 'sound/machines/warning-buzzer.ogg', 50, TRUE, 7)
+ return INVOKE_ASYNC(src, .proc/GoOut)
+
+/obj/machinery/mind_engraver/proc/RemoveChip(mob/user)
+ if(!nanochip)
+ return
+
+ if(currently_imprinting)
+ to_chat(user, SPAN_WARNING("\The [src] is currently imprinting the data from the nanochip!"))
+ return
+
+ nanochip.dropInto(loc)
+ if(user)
+ user.put_in_any_hand_if_possible(nanochip)
+ nanochip = null
+
+/obj/machinery/mind_engraver/proc/StartImprint()
+ if(currently_imprinting)
+ return FALSE
+
+ if(!nanochip || !occupant)
+ return FALSE
+
+ currently_imprinting = TRUE
+ imprint_end = world.time + imprint_time
+ visible_message(SPAN_NOTICE("\The [src] chimes as it begins its operation!"))
+ playsound(src, 'sound/machines/synth_yes.ogg', 50, TRUE, 7)
+ return TRUE
+
+/obj/machinery/mind_engraver/proc/StopImprint()
+ currently_imprinting = FALSE
+
+/obj/machinery/mind_engraver/proc/FinishImprint()
+ StopImprint()
+
+ if(!nanochip || !occupant)
+ return
+
+ if(nanochip.stored_data)
+ nanochip.stored_data.ApplyEffect(occupant)
+
+ ToggleLock(FALSE)
+ INVOKE_ASYNC(src, .proc/GoOut)
+ visible_message(SPAN_NOTICE("\The [src] produces a signal as it finishes its operation!"))
+ playsound(src, 'sound/machines/ping.ogg', 50, TRUE, 7)
+
+/obj/machinery/mind_engraver/proc/ToggleLock(force_set = null)
+ if(force_set != null)
+ locked = force_set
+ else
+ locked = !locked
+ var/sound_path = locked ? 'sound/machines/bolts_down.ogg' : 'sound/machines/bolts_up.ogg'
+ playsound(src, sound_path, 35, TRUE)
+ update_icon()
diff --git a/code/game/objects/items/devices/mind_engraver_chip.dm b/code/game/objects/items/devices/mind_engraver_chip.dm
new file mode 100644
index 00000000000..ad6e54ba510
--- /dev/null
+++ b/code/game/objects/items/devices/mind_engraver_chip.dm
@@ -0,0 +1,293 @@
+/obj/item/mind_engraver_chip
+ name = "mind engraver nanochip"
+ desc = "A tiny chip that stores data used by mind engravers to imprint certain knowledge."
+ icon = 'icons/obj/stock_parts.dmi'
+ icon_state = "rom2"
+ item_state = "electronic"
+ w_class = ITEM_SIZE_SMALL
+
+ origin_tech = list(TECH_DATA = 8, TECH_DOMINION = 5)
+
+ var/datum/engraver_data/stored_data = null
+
+/obj/item/mind_engraver_chip/Initialize()
+ . = ..()
+ if(ispath(stored_data))
+ stored_data = new stored_data()
+
+/obj/item/mind_engraver_chip/Destroy()
+ QDEL_NULL(stored_data)
+ stored_data = null
+ return ..()
+
+/* Presets */
+/obj/item/mind_engraver_chip/human_languages
+ name = "mind engraver nanochip - \"Languages of the Humanity\""
+ desc = "A relatively big chip with tiny label saying \"Languages of the Humanity\"."
+ icon_state = "romos2"
+ stored_data = /datum/engraver_data/language/humanity
+
+/obj/item/mind_engraver_chip/diplomatic_languages
+ name = "mind engraver nanochip - \"Dominion Diplomacy\""
+ desc = "A bigger chip used by Terran Dominion to grant instant knowledge on all languages important to the Diplomatic \
+ procedures in the universe."
+ icon_state = "romos2"
+ stored_data = /datum/engraver_data/language/diplomatic
+
+/* Randomized spawns */
+/obj/item/mind_engraver_chip/random_skills
+ stored_data = /datum/engraver_data/skill/random
+
+/obj/item/mind_engraver_chip/random_languages
+ stored_data = /datum/engraver_data/language/random
+
+///////////
+/* Datum */
+///////////
+/datum/engraver_data
+ var/chip_type = "UNKNOWN"
+
+/datum/engraver_data/proc/PrintFunction()
+ return "UNKNOWN"
+
+/datum/engraver_data/proc/ApplyEffect(mob/living/carbon/human/target)
+ if(!istype(target))
+ return FALSE
+ return TRUE
+
+/* Skills */
+/datum/engraver_data/skill
+ chip_type = "Skill Imprint"
+ var/max_skills_count = 3
+ var/list/skills = list()
+
+/datum/engraver_data/skill/PrintFunction()
+ if(!LAZYLEN(skills))
+ return SPAN_DANGER("MISSING SKILL DATA")
+ var/func_text = "
"
+ for(var/decl/hierarchy/skill/S in GLOB.skills)
+ if(!(S.type in skills))
+ continue
+ func_text += " - Improves knowledge in the field of [S.name] to the [S.levels[skills[S.type]]] level
"
+ return func_text
+
+/datum/engraver_data/skill/ApplyEffect(mob/living/carbon/human/target)
+ if(!..())
+ return FALSE
+
+ if(!target.skillset)
+ return FALSE
+
+ for(var/decl/hierarchy/skill/S in GLOB.skills)
+ if(!(S.type in skills))
+ continue
+ if(target.get_skill_value(S.type) >= skills[S.type])
+ continue
+ target.skillset.skill_list[S.type] = skills[S.type]
+ to_chat(target, SPAN_NOTICE("You feel much more proficient in the field of [S.name]!"))
+
+ target.skillset.on_levels_change()
+ return TRUE
+
+// Random
+/datum/engraver_data/skill/random/New()
+ . = ..()
+ var/list/valid_skills = GLOB.skills.Copy()
+ for(var/i = 1 to rand(1, max_skills_count))
+ if(!LAZYLEN(valid_skills))
+ return
+ var/decl/hierarchy/skill/S = pick(valid_skills)
+ valid_skills -= S
+ skills[S.type] = rand(SKILL_BASIC, length(S.levels))
+
+// Presets
+/datum/engraver_data/skill/max_combat
+ skills = list(
+ SKILL_COMBAT = SKILL_MAX,
+ SKILL_WEAPONS = SKILL_MAX,
+ )
+
+/datum/engraver_data/skill/max_command
+ skills = list(
+ SKILL_BUREAUCRACY = SKILL_MAX,
+ SKILL_FINANCE = SKILL_MAX,
+ SKILL_PILOT = SKILL_MAX,
+ )
+
+/datum/engraver_data/skill/trained_medical
+ skills = list(
+ SKILL_MEDICAL = SKILL_TRAINED,
+ SKILL_ANATOMY = SKILL_TRAINED,
+ SKILL_CHEMISTRY = SKILL_TRAINED,
+ )
+
+/* Languages */
+/datum/engraver_data/language
+ chip_type = "Language Imprint"
+ var/max_language_count = 3
+ var/list/languages = list()
+
+/datum/engraver_data/language/PrintFunction()
+ if(!LAZYLEN(languages))
+ return SPAN_DANGER("MISSING SKILL DATA")
+ var/func_text = "
"
+ for(var/lang in languages)
+ if(!all_languages[lang])
+ continue
+ func_text += " - Imprints professional level knowledge on [lowertext(lang)] language
"
+ return func_text
+
+/datum/engraver_data/language/ApplyEffect(mob/living/carbon/human/target)
+ if(!..())
+ return FALSE
+
+ for(var/lang in languages)
+ if(!target.add_language(lang))
+ continue
+ to_chat(target, SPAN_NOTICE("You gain professional understanding of [lowertext(lang)] language!"))
+
+ return TRUE
+
+// Random
+/datum/engraver_data/language/random/New()
+ . = ..()
+ var/list/valid_languages = all_languages.Copy()
+ for(var/i = 1 to rand(1, max_language_count))
+ if(!LAZYLEN(valid_languages))
+ return
+ var/lang = pick(valid_languages)
+ valid_languages -= lang
+ languages |= lang
+
+// Presets
+/datum/engraver_data/language/humanity/New()
+ . = ..()
+ for(var/lang in subtypesof(/datum/language/human))
+ var/datum/language/L = lang
+ languages |= initial(L.name)
+
+/datum/engraver_data/language/diplomatic
+ languages = list(
+ LANGUAGE_COMMON,
+ LANGUAGE_SPACER,
+ LANGUAGE_HUMAN_ENGLISH,
+ LANGUAGE_SKRELLIAN,
+ LANGUAGE_UNATHI_SINTA,
+ )
+
+/datum/engraver_data/language/english
+ languages = list(LANGUAGE_HUMAN_ENGLISH)
+
+/datum/engraver_data/language/skrell
+ languages = list(LANGUAGE_SKRELLIAN)
+
+/datum/engraver_data/language/adherent
+ languages = list(LANGUAGE_ADHERENT)
+
+/* Fluff */
+/datum/engraver_data/memory
+ chip_type = "Memory Imprint"
+ var/function_text = "UNKNOWN"
+
+/datum/engraver_data/memory/PrintFunction()
+ if(!function_text)
+ return SPAN_DANGER("MISSING SKILL DATA")
+ return "[function_text]
"
+
+// Presets
+/datum/engraver_data/memory/dominion_info_one
+ function_text = "Imprints basic information about Terran Dominion, as of year 244, III e."
+ // List of text that will be sent to the user some time after applying the nanochip
+ var/list/after_thoughts = list(
+ "Soldier" = list(
+ "You feel like it might be good idea to join the Terran Navy...",
+ "You think that Dominion's army could use a soldier like you...",
+ "There's definitely a need for people like you in the Terran military!",
+ "Why are you not in the Dominion's army...? You are missing out on so much...",
+ "You could be a Lieutenant by now, and then aristocratic status is just a step away...",
+ "The Terran Dominion needs you in its military... You are a great soldier!",
+ ),
+ "Governor" = list(
+ "You feel like Terran Dominion could use someone like you for their government apparatus...",
+ "You think it'd be really nice if you were governing some planet out there...",
+ "Maybe your skills would be incredibly useful if put to some use..?",
+ "You start to wonder how high planet governor's salary is...",
+ "The Terran Dominion definitely needs you as one of their governors... You are an excellent strategist and leader!",
+ ),
+ "Scientist" = list(
+ "You feel like you are a wonderful scientist...",
+ "You wonder if Dominion's Ministry of Scientific Progress is where you belong...",
+ "Terran Dominion rewards their scientist handsomely, doesn't it?",
+ "Your mind is full of useful knowldge and ideas, the Dominion would certainly need you...",
+ "Your place is with other great minds in the Terran Dominion... You are an absolute genius!",
+ ),
+ "Doctor" = list(
+ "You feel like you could save a lot more lives as a doctor in the Dominion's military...",
+ "Terran Dominion has better medical equipment than the ones you are used to...",
+ "How great would it be if you were a combat medic? Or a field surgeon for the Terran navy?",
+ "You are an absolute master with surgical tools and medical equipment, after all...",
+ "The Terran Dominion would not say \"no\" to a great doctor like you, you are the best one out there!",
+ ),
+ // Lmao,
+ "Useless" = list(
+ "You try to think of how useful you'd be for the Terran Dominion...",
+ "You feel like you are just not fit for society...",
+ "Maybe you could be a... something...?",
+ "Now you just feel like the most useless person in universe...",
+ "You think it's a great idea to donate your brain to the Ministry of Scientific Progress...",
+ "Maybe becoming a cyborg will make you more useful...",
+ ),
+ )
+
+/datum/engraver_data/memory/dominion_info_one/ApplyEffect(mob/living/target)
+ if(!..())
+ return FALSE
+
+ var/list/dat = list()
+
+ dat += "Terran Dominion - Basic Information
"
+ dat += "This covers information about Terran Dominion as of year 244 of third era.
"
+ dat += "The following memories are all a part of the educational program initiated by the Ministry of Scientific Progress.
"
+ dat += "No other alterations will be present.
"
+ dat += "
"
+ dat += "Primary Language: English
"
+ dat += "Capital City: \[DATA EXPUNGED\] located on Earth, Sol system
"
+ dat += "Current Ruler: Emperor Jang Seo-joon"
+ dat += "
"
+ dat += "Terran Dominion is the official government of human species (homo-sapiens and its subtypes), \
+ currently in control of over a hundred populated star systems and around 500 operational \
+ space stations and motherships.
"
+ dat += "Terran Dominion is governed by the Emperor, which is appointed by the previous ruler as they see fit. \
+ Members of aristocracy fill in the positions of authority, such as sector administrators, \
+ ministers, advisors and military generals.
"
+ dat += "You, our dear citizen, is capable of joining the ranks of aristocracy by proving yourself to be a \
+ valuable asset to the Dominion. Current data recommends that you seek employment in the following \
+ fields if you are interested:
"
+ dat += "- Military;
"
+ dat += "- Planet Governor;
"
+ dat += "- Research & Development;
"
+
+ var/datum/browser/popup = new(target, "mind_engraver_chip_[type]", "New Memory", 360, 500)
+ popup.set_content(JOINTEXT(dat))
+ popup.open()
+
+ to_chat(target, SPAN_WARNING("Your mind is flooded with batches of new information. You feel like you always knew it..."))
+ // SIDE EFFECTS! You now want to serve the Terran Dominion! Kinda...
+ var/chosen_type = "Useless"
+ // Trying to be as on-point as possible
+ if(target.get_skill_value(SKILL_HAULING) >= SKILL_EXPERIENCED || target.get_skill_value(SKILL_COMBAT) >= SKILL_TRAINED || target.get_skill_value(SKILL_WEAPONS) >= SKILL_TRAINED)
+ chosen_type = "Soldier"
+ else if(target.get_skill_value(SKILL_PILOT) >= SKILL_EXPERIENCED || target.get_skill_value(SKILL_BUREAUCRACY) >= SKILL_EXPERIENCED || target.get_skill_value(SKILL_FINANCE) >= SKILL_EXPERIENCED)
+ chosen_type = "Governor"
+ else if(target.get_skill_value(SKILL_SCIENCE) >= SKILL_EXPERIENCED || target.get_skill_value(SKILL_CHEMISTRY) >= SKILL_TRAINED || target.get_skill_value(SKILL_DEVICES) >= SKILL_EXPERIENCED)
+ chosen_type = "Scientist"
+ else if(target.get_skill_value(SKILL_MEDICAL) >= SKILL_TRAINED && target.get_skill_value(SKILL_ANATOMY) >= SKILL_TRAINED)
+ chosen_type = "Doctor"
+
+ if(chosen_type in after_thoughts)
+ var/list/chosen_list = after_thoughts[chosen_type]
+ var/message_delay = rand(30 SECONDS, 80 SECONDS)
+ for(var/i = 1 to length(chosen_list))
+ addtimer(CALLBACK(GLOBAL_PROC, /proc/to_chat, target, SPAN_SUBTLE(chosen_list[i])), (rand(90 SECONDS, 300 SECONDS) + (i * message_delay)))
+
+ return TRUE
diff --git a/code/game/objects/items/weapons/circuitboards/computer/computer.dm b/code/game/objects/items/weapons/circuitboards/computer/computer.dm
index 4a40a7685d5..8e2274184ec 100644
--- a/code/game/objects/items/weapons/circuitboards/computer/computer.dm
+++ b/code/game/objects/items/weapons/circuitboards/computer/computer.dm
@@ -114,3 +114,13 @@
/obj/item/stock_parts/circuitboard/pandemic
name = T_BOARD("PanD.E.M.I.C 2200 console")
build_path = /obj/machinery/computer/pandemic
+
+/obj/item/stock_parts/circuitboard/aiupload
+ name = T_BOARD("AI upload console")
+ build_path = /obj/machinery/computer/upload/ai
+ origin_tech = list(TECH_DATA = 4)
+
+/obj/item/stock_parts/circuitboard/mind_engraver_console
+ name = T_BOARD("mind engraver console")
+ build_path = /obj/machinery/computer/mind_engraver_control
+ origin_tech = list(TECH_ENGINEERING = 7, TECH_DATA = 5, TECH_DOMINION = 2)
diff --git a/code/game/objects/items/weapons/circuitboards/machinery/mind_engraver.dm b/code/game/objects/items/weapons/circuitboards/machinery/mind_engraver.dm
new file mode 100644
index 00000000000..f0e3e7772cc
--- /dev/null
+++ b/code/game/objects/items/weapons/circuitboards/machinery/mind_engraver.dm
@@ -0,0 +1,13 @@
+/obj/item/stock_parts/circuitboard/mind_engraver
+ name = T_BOARD("mind engraver")
+ build_path = /obj/machinery/mind_engraver
+ board_type = "machine"
+ origin_tech = list(TECH_ENGINEERING = 7, TECH_DATA = 5, TECH_DOMINION = 2)
+ req_components = list(
+ /obj/item/stock_parts/capacitor = 4,
+ /obj/item/stock_parts/scanning_module = 2,
+ /obj/item/stock_parts/manipulator = 2
+ )
+ additional_spawn_components = list(
+ /obj/item/stock_parts/power/apc/buildable = 1
+ )
diff --git a/code/game/objects/objs.dm b/code/game/objects/objs.dm
index d6b75a553d3..f6139d7fca7 100644
--- a/code/game/objects/objs.dm
+++ b/code/game/objects/objs.dm
@@ -188,3 +188,7 @@
/obj/get_mass()
return min(2**(w_class-1), 100)
+
+/obj/proc/mob_breakout(mob/living/escapee)
+ set waitfor = FALSE
+ return FALSE
diff --git a/code/game/objects/structures.dm b/code/game/objects/structures.dm
index 5d2c42b1300..fd0762f8816 100644
--- a/code/game/objects/structures.dm
+++ b/code/game/objects/structures.dm
@@ -21,10 +21,6 @@
return TRUE
return ..()
-/obj/structure/proc/mob_breakout(mob/living/escapee)
- set waitfor = FALSE
- return FALSE
-
/obj/structure/Destroy()
reset_mobs_offset()
var/turf/T = get_turf(src)
diff --git a/code/modules/mob/living/living.dm b/code/modules/mob/living/living.dm
index 644401db125..e5a7de03883 100644
--- a/code/modules/mob/living/living.dm
+++ b/code/modules/mob/living/living.dm
@@ -667,9 +667,9 @@ default behaviour is:
return TRUE
//Breaking out of a structure?
- if(istype(loc, /obj/structure))
- var/obj/structure/C = loc
- if(C.mob_breakout(src))
+ if(istype(loc, /obj))
+ var/obj/O = loc
+ if(O.mob_breakout(src))
return TRUE
/mob/living/proc/escape_inventory(obj/item/holder/H)
diff --git a/code/modules/research/designs/designs_circuits.dm b/code/modules/research/designs/designs_circuits.dm
index f5d8871a521..e5e1a05b957 100644
--- a/code/modules/research/designs/designs_circuits.dm
+++ b/code/modules/research/designs/designs_circuits.dm
@@ -667,6 +667,13 @@
build_path = /obj/item/stock_parts/circuitboard/disperser
sort_string = "KCAAG"
+/datum/design/circuit/shipmap
+ name = "ship holomap"
+ id = "shipmap"
+ req_tech = list(TECH_ENGINEERING = 1)
+ build_path = /obj/item/stock_parts/circuitboard/shipmap
+ sort_string = "LAAAA"
+
/datum/design/circuit/tcom
req_tech = list(TECH_DATA = 4, TECH_ENGINEERING = 4)
@@ -904,9 +911,26 @@
build_path = /obj/item/stock_parts/circuitboard/radio_beacon
sort_string = "XAAAI"
-/datum/design/circuit/shipmap
- name = "ship holomap"
- id = "shipmap"
- req_tech = list(TECH_ENGINEERING = 1)
- build_path = /obj/item/stock_parts/circuitboard/shipmap
- sort_string = "SAAAT"
+/datum/design/circuit/dominion/AssembleDesignName()
+ if(build_path)
+ var/obj/item/stock_parts/circuitboard/C = build_path
+ if(initial(C.board_type) == "machine")
+ name = "\[Dominion Machine Circuit\] - [item_name]"
+ else if(initial(C.board_type) == "computer")
+ name = "\[Dominion Computer Circuit\] - [item_name]"
+ else
+ name = "\[Dominion Circuit\] - [item_name]"
+
+/datum/design/circuit/mind_engraver
+ name = "mind engraver"
+ id = "mind_engraver"
+ req_tech = list(TECH_ENGINEERING = 8, TECH_DATA = 6, TECH_DOMINION = 5)
+ build_path = /obj/item/stock_parts/circuitboard/mind_engraver
+ sort_string = "YAAAA"
+
+/datum/design/circuit/mind_engraver_console
+ name = "mind engraver console"
+ id = "mind_engraver_console"
+ req_tech = list(TECH_ENGINEERING = 8, TECH_DATA = 6, TECH_DOMINION = 5)
+ build_path = /obj/item/stock_parts/circuitboard/mind_engraver_console
+ sort_string = "YAAAB"
diff --git a/code/modules/research/part_replacer.dm b/code/modules/research/part_replacer.dm
index 6fc0242e44b..3ce85f9be8b 100644
--- a/code/modules/research/part_replacer.dm
+++ b/code/modules/research/part_replacer.dm
@@ -44,9 +44,9 @@
remote_interaction = TRUE
remote_distance = 5
-/obj/item/research
+/obj/item/research_debug
name = "research debugging device"
desc = "Instant research tool. For testing purposes only."
icon = 'icons/obj/stock_parts.dmi'
icon_state = "smes_coil"
- origin_tech = list(TECH_MATERIAL = 19, TECH_ENGINEERING = 19, TECH_PHORON = 19, TECH_POWER = 19, TECH_BLUESPACE = 19, TECH_BIO = 19, TECH_COMBAT = 19, TECH_MAGNET = 19, TECH_DATA = 19, TECH_ESOTERIC = 19)
\ No newline at end of file
+ origin_tech = list(TECH_MATERIAL = 19, TECH_ENGINEERING = 19, TECH_PHORON = 19, TECH_POWER = 19, TECH_BLUESPACE = 19, TECH_BIO = 19, TECH_COMBAT = 19, TECH_MAGNET = 19, TECH_DATA = 19, TECH_ESOTERIC = 19, TECH_DOMINION = 19)
diff --git a/code/modules/research/research.dm b/code/modules/research/research.dm
index f8ef4897f8a..c1c49e6e1de 100644
--- a/code/modules/research/research.dm
+++ b/code/modules/research/research.dm
@@ -197,6 +197,12 @@ research holder datum.
id = TECH_ESOTERIC
level = 0
+/datum/tech/dominion
+ name = "Dominion Technology"
+ desc = "Mostly forgotten and lost technology used during Terran Dominion's prime era, containing powerful designs and tools."
+ id = TECH_DOMINION
+ level = 0
+
/obj/item/disk/tech_disk
name = "fabricator data disk"
desc = "A disk for storing fabricator learning data for backup."
@@ -207,7 +213,6 @@ research holder datum.
matter = list(MATERIAL_PLASTIC = 30, MATERIAL_STEEL = 30, MATERIAL_GLASS = 10)
var/datum/tech/stored
-
/obj/item/disk/design_disk
name = "component design disk"
desc = "A disk for storing device design data for construction in lathes."
diff --git a/icons/obj/machines/mind_engraver.dmi b/icons/obj/machines/mind_engraver.dmi
new file mode 100644
index 00000000000..7e9788c45a9
Binary files /dev/null and b/icons/obj/machines/mind_engraver.dmi differ