mirror of
https://github.com/ParadiseSS13/Paradise.git
synced 2026-01-01 05:02:33 +00:00
839 lines
27 KiB
Plaintext
839 lines
27 KiB
Plaintext
#define DNA_BLOCK_SIZE 3
|
|
|
|
#define NEGATE_MUTATION_THRESHOLD 30 // Occupants with over ## percent radiation threshold will not gain mutations
|
|
|
|
#define PAGE_UI "ui"
|
|
#define PAGE_SE "se"
|
|
#define PAGE_BUFFER "buffer"
|
|
#define PAGE_REJUVENATORS "rejuvenators"
|
|
|
|
//list("data" = null, "owner" = null, "label" = null, "type" = null, "ue" = 0),
|
|
/datum/dna2_record
|
|
var/datum/dna/dna = null
|
|
var/types = 0
|
|
var/name = "Empty"
|
|
|
|
// Stuff for cloners
|
|
var/id = null
|
|
var/implant = null
|
|
var/ckey = null
|
|
var/mind = null
|
|
var/languages = null
|
|
|
|
/datum/dna2_record/proc/GetData()
|
|
var/list/ser=list("data" = null, "owner" = null, "label" = null, "type" = null, "ue" = 0)
|
|
if(dna)
|
|
ser["ue"] = (types & DNA2_BUF_UE) == DNA2_BUF_UE
|
|
if(types & DNA2_BUF_SE)
|
|
ser["data"] = dna.SE
|
|
else
|
|
ser["data"] = dna.UI
|
|
ser["owner"] = dna.real_name
|
|
ser["label"] = name
|
|
if(types & DNA2_BUF_UI)
|
|
ser["type"] = "ui"
|
|
else
|
|
ser["type"] = "se"
|
|
return ser
|
|
|
|
/datum/dna2_record/proc/copy()
|
|
var/datum/dna2_record/newrecord = new /datum/dna2_record
|
|
newrecord.dna = dna.Clone()
|
|
newrecord.types = types
|
|
newrecord.name = name
|
|
newrecord.mind = mind
|
|
newrecord.ckey = ckey
|
|
newrecord.languages = languages
|
|
newrecord.implant = implant
|
|
return newrecord
|
|
|
|
|
|
/////////////////////////// DNA MACHINES
|
|
/obj/machinery/dna_scannernew
|
|
name = "\improper DNA modifier"
|
|
desc = "It scans DNA structures."
|
|
icon = 'icons/obj/cryogenic2.dmi'
|
|
icon_state = "scanner_open"
|
|
density = TRUE
|
|
anchored = TRUE
|
|
idle_power_consumption = 50
|
|
active_power_consumption = 300
|
|
interact_offline = TRUE
|
|
var/locked = FALSE
|
|
var/mob/living/carbon/occupant = null
|
|
var/obj/item/reagent_containers/glass/beaker = null
|
|
var/opened = FALSE
|
|
var/damage_coeff
|
|
var/scan_level
|
|
var/precision_coeff
|
|
|
|
/obj/machinery/dna_scannernew/examine(mob/user)
|
|
. = ..()
|
|
. += "<span class='notice'>You can <b>Alt-Click</b> [src] to eject its occupant.</span>"
|
|
. += "<span class='notice'>You can <b>Click-drag</b> someone to [src] to put them in.</span>"
|
|
|
|
/obj/machinery/dna_scannernew/Initialize(mapload)
|
|
. = ..()
|
|
component_parts = list()
|
|
component_parts += new /obj/item/circuitboard/dna_scanner(null)
|
|
component_parts += new /obj/item/stock_parts/scanning_module(null)
|
|
component_parts += new /obj/item/stock_parts/manipulator(null)
|
|
component_parts += new /obj/item/stock_parts/micro_laser(null)
|
|
component_parts += new /obj/item/stack/sheet/glass(null)
|
|
component_parts += new /obj/item/stack/cable_coil(null, 1)
|
|
component_parts += new /obj/item/stack/cable_coil(null, 1)
|
|
RefreshParts()
|
|
|
|
/obj/machinery/dna_scannernew/upgraded/Initialize(mapload)
|
|
. = ..()
|
|
component_parts = list()
|
|
component_parts += new /obj/item/circuitboard/dna_scanner(null)
|
|
component_parts += new /obj/item/stock_parts/scanning_module/phasic(null)
|
|
component_parts += new /obj/item/stock_parts/manipulator/pico(null)
|
|
component_parts += new /obj/item/stock_parts/micro_laser/ultra(null)
|
|
component_parts += new /obj/item/stack/sheet/glass(null)
|
|
component_parts += new /obj/item/stack/cable_coil(null, 1)
|
|
component_parts += new /obj/item/stack/cable_coil(null, 1)
|
|
RefreshParts()
|
|
|
|
/obj/machinery/dna_scannernew/RefreshParts()
|
|
scan_level = 0
|
|
damage_coeff = 0
|
|
precision_coeff = 0
|
|
for(var/obj/item/stock_parts/scanning_module/P in component_parts)
|
|
scan_level += P.rating
|
|
for(var/obj/item/stock_parts/manipulator/P in component_parts)
|
|
precision_coeff = P.rating
|
|
for(var/obj/item/stock_parts/micro_laser/P in component_parts)
|
|
damage_coeff = P.rating
|
|
|
|
/obj/machinery/dna_scannernew/AllowDrop()
|
|
return FALSE
|
|
|
|
/obj/machinery/dna_scannernew/relaymove(mob/user)
|
|
if(user.incapacitated())
|
|
return FALSE //maybe they should be able to get out with cuffs, but whatever
|
|
go_out()
|
|
|
|
/obj/machinery/dna_scannernew/AltClick(mob/user)
|
|
if(user.stat || HAS_TRAIT(user, TRAIT_HANDS_BLOCKED) || !Adjacent(user))
|
|
return
|
|
eject_occupant(user)
|
|
add_fingerprint(user)
|
|
|
|
/obj/machinery/dna_scannernew/Destroy()
|
|
eject_occupant(null, TRUE)
|
|
return ..()
|
|
|
|
/obj/machinery/dna_scannernew/proc/eject_occupant(user, force)
|
|
go_out(user, force)
|
|
for(var/obj/O in src)
|
|
if(!istype(O,/obj/item/circuitboard/clonescanner) && \
|
|
!istype(O,/obj/item/stock_parts) && \
|
|
!istype(O,/obj/item/stack/cable_coil) && \
|
|
O != beaker)
|
|
O.forceMove(get_turf(src))//Ejects items that manage to get in there (exluding the components and beaker)
|
|
if(!occupant)
|
|
for(var/mob/M in src)//Failsafe so you can get mobs out
|
|
M.forceMove(get_turf(src))
|
|
|
|
/obj/machinery/dna_scannernew/update_icon_state()
|
|
if(panel_open)
|
|
icon_state = "scanner_[occupant ? "" : "open_"]maintenance"
|
|
return
|
|
if(stat & NOPOWER)
|
|
icon_state = "scanner"
|
|
return
|
|
if(!occupant)
|
|
icon_state = "scanner_open"
|
|
return
|
|
icon_state = "scanner_occupied"
|
|
|
|
/obj/machinery/dna_scannernew/MouseDrop_T(atom/movable/O, mob/user)
|
|
if(!istype(O))
|
|
return
|
|
if(O.loc == user) //no you can't pull things out of your ass
|
|
return
|
|
if(user.incapacitated()) //are you cuffed, dying, lying, stunned or other
|
|
return
|
|
if(O.anchored || get_dist(user, src) > 1 || get_dist(user, O) > 1 || user.contents.Find(src)) // is the mob anchored, too far away from you, or are you too far away from the source
|
|
return
|
|
if(!ismob(O)) //humans only
|
|
return
|
|
if(isanimal_or_basicmob(O) || issilicon(O)) //animals and robutts dont fit
|
|
return
|
|
if(!ishuman(user) && !isrobot(user)) //No ghosts or mice putting people into the sleeper
|
|
return
|
|
if(user.loc==null) // just in case someone manages to get a closet into the blue light dimension, as unlikely as that seems
|
|
return
|
|
if(!isturf(user.loc) || !isturf(O.loc)) // are you in a container/closet/pod/etc?
|
|
return
|
|
if(occupant)
|
|
to_chat(user, "<span class='boldnotice'>[src] is already occupied!</span>")
|
|
return TRUE
|
|
var/mob/living/L = O
|
|
if(!istype(L) || L.buckled)
|
|
return
|
|
if(L.abiotic())
|
|
to_chat(user, "<span class='danger'>Subject may not hold anything in their hands.</span>")
|
|
return TRUE
|
|
if(L.has_buckled_mobs()) //mob attached to us
|
|
to_chat(user, "<span class='warning'>[L] will not fit into [src] because [L.p_they()] [L.p_have()] a slime latched onto [L.p_their()] head.</span>")
|
|
return TRUE
|
|
if(L == user)
|
|
visible_message("<span class='notice'>[user] climbs into [src].</span>")
|
|
else
|
|
visible_message("<span class='notice'>[user] puts [L.name] into [src].</span>")
|
|
put_in(L)
|
|
if(user.pulling == L)
|
|
user.stop_pulling()
|
|
QDEL_LIST_CONTENTS(L.grabbed_by)
|
|
return TRUE
|
|
|
|
/obj/machinery/dna_scannernew/item_interaction(mob/living/user, obj/item/used, list/modifiers)
|
|
if(istype(used, /obj/item/reagent_containers/glass))
|
|
if(beaker)
|
|
to_chat(user, "<span class='warning'>A beaker is already loaded into the machine.</span>")
|
|
return ITEM_INTERACT_COMPLETE
|
|
|
|
if(!user.drop_item())
|
|
to_chat(user, "<span class='warning'>\The [used] is stuck to you!</span>")
|
|
return ITEM_INTERACT_COMPLETE
|
|
|
|
beaker = used
|
|
SStgui.update_uis(src)
|
|
used.forceMove(src)
|
|
user.visible_message("[user] adds \a [used] to \the [src]!", "You add \a [used] to \the [src]!")
|
|
return ITEM_INTERACT_COMPLETE
|
|
|
|
if(istype(used, /obj/item/grab))
|
|
var/obj/item/grab/G = used
|
|
if(!ismob(G.affecting))
|
|
return ITEM_INTERACT_COMPLETE
|
|
|
|
if(occupant)
|
|
to_chat(user, "<span class='boldnotice'>The scanner is already occupied!</span>")
|
|
return ITEM_INTERACT_COMPLETE
|
|
|
|
if(G.affecting.abiotic())
|
|
to_chat(user, "<span class='boldnotice'>Subject may not hold anything in their hands.</span>")
|
|
return ITEM_INTERACT_COMPLETE
|
|
|
|
if(G.affecting.has_buckled_mobs()) //mob attached to us
|
|
to_chat(user, "<span class='warning'>[G] will not fit into [src] because [G.affecting.p_they()] [G.affecting.p_have()] a slime latched onto [G.affecting.p_their()] head.</span>")
|
|
return ITEM_INTERACT_COMPLETE
|
|
|
|
if(panel_open)
|
|
to_chat(usr, "<span class='boldnotice'>Close the maintenance panel first.</span>")
|
|
return ITEM_INTERACT_COMPLETE
|
|
|
|
put_in(G.affecting)
|
|
add_fingerprint(user)
|
|
qdel(G)
|
|
return ITEM_INTERACT_COMPLETE
|
|
|
|
return ..()
|
|
|
|
/obj/machinery/dna_scannernew/crowbar_act(mob/user, obj/item/I)
|
|
if(!panel_open)
|
|
return
|
|
. = TRUE
|
|
if(!I.use_tool(src, user, 0, volume = I.tool_volume))
|
|
return
|
|
if(flags & NODECONSTRUCT)//We need to check for this early or the contents could be moved before it checks for the flag normally
|
|
return
|
|
for(var/obj/thing in contents) // in case there is something in the scanner
|
|
thing.forceMove(loc)
|
|
default_deconstruction_crowbar(user, I)
|
|
|
|
/obj/machinery/dna_scannernew/screwdriver_act(mob/user, obj/item/I)
|
|
if(occupant)
|
|
to_chat(user, "<span class='notice'>The maintenance panel is locked.</span>")
|
|
return TRUE
|
|
if(default_deconstruction_screwdriver(user, "[icon_state]_maintenance", "[initial(icon_state)]", I))
|
|
return TRUE
|
|
|
|
/obj/machinery/dna_scannernew/proc/put_in(mob/M)
|
|
M.forceMove(src)
|
|
occupant = M
|
|
update_icon(UPDATE_ICON_STATE)
|
|
SStgui.update_uis(src)
|
|
|
|
/obj/machinery/dna_scannernew/proc/go_out(mob/user, force)
|
|
if(!occupant)
|
|
if(user)
|
|
to_chat(user, "<span class='warning'>The scanner is empty!</span>")
|
|
return
|
|
if(locked && !force)
|
|
if(user)
|
|
to_chat(user, "<span class='warning'>The scanner is locked!</span>")
|
|
return
|
|
occupant.forceMove(loc)
|
|
occupant = null
|
|
update_icon(UPDATE_ICON_STATE)
|
|
SStgui.update_uis(src)
|
|
|
|
/obj/machinery/dna_scannernew/force_eject_occupant(mob/target)
|
|
go_out(null, TRUE)
|
|
|
|
/obj/machinery/dna_scannernew/ex_act(severity)
|
|
if(occupant)
|
|
occupant.ex_act(severity)
|
|
..()
|
|
|
|
/obj/machinery/dna_scannernew/handle_atom_del(atom/A)
|
|
..()
|
|
if(A == occupant)
|
|
occupant = null
|
|
update_icon(UPDATE_ICON_STATE)
|
|
SStgui.update_uis(src)
|
|
|
|
// Checks if occupants can be irradiated/mutated - prevents exploits where wearing full rad protection would still let you gain mutations
|
|
/obj/machinery/dna_scannernew/proc/radiation_check()
|
|
if(!occupant)
|
|
return TRUE
|
|
|
|
if(HAS_TRAIT(occupant, TRAIT_GENELESS))
|
|
return TRUE
|
|
|
|
var/radiation_protection = occupant.run_armor_check(null, RAD)
|
|
if(radiation_protection > NEGATE_MUTATION_THRESHOLD)
|
|
return TRUE
|
|
return FALSE
|
|
|
|
/obj/machinery/computer/scan_consolenew
|
|
name = "\improper DNA Modifier access console"
|
|
desc = "Allows you to scan and modify DNA."
|
|
icon_screen = "dna"
|
|
icon_keyboard = "med_key"
|
|
circuit = /obj/item/circuitboard/scan_consolenew
|
|
var/selected_ui_block = 1.0
|
|
var/selected_ui_subblock = 1.0
|
|
var/selected_se_block = 1.0
|
|
var/selected_se_subblock = 1.0
|
|
var/selected_ui_target = 1
|
|
var/selected_ui_target_hex = 1
|
|
var/radiation_duration = 2.0
|
|
var/radiation_intensity = 1.0
|
|
var/list/datum/dna2_record/buffers[3]
|
|
var/irradiating = 0
|
|
var/injector_ready = FALSE //Quick fix for issue 286 (screwdriver the screen twice to restore injector) -Pete
|
|
var/obj/machinery/dna_scannernew/connected = null
|
|
var/obj/item/disk/data/disk = null
|
|
var/selected_menu_key = PAGE_UI
|
|
idle_power_consumption = 10
|
|
active_power_consumption = 400
|
|
|
|
/obj/machinery/computer/scan_consolenew/item_interaction(mob/living/user, obj/item/used, list/modifiers)
|
|
if(istype(used, /obj/item/disk/data)) //INSERT SOME diskS
|
|
if(!disk)
|
|
user.drop_item()
|
|
used.forceMove(src)
|
|
disk = used
|
|
to_chat(user, "You insert [used].")
|
|
SStgui.update_uis(src)
|
|
|
|
return ITEM_INTERACT_COMPLETE
|
|
|
|
return ..()
|
|
|
|
/obj/machinery/computer/scan_consolenew/Initialize(mapload)
|
|
. = ..()
|
|
for(var/i=0;i<3;i++)
|
|
buffers[i+1]=new /datum/dna2_record
|
|
addtimer(CALLBACK(src, PROC_REF(find_machine)), 1 SECONDS)
|
|
addtimer(CALLBACK(src, PROC_REF(ready)), 25 SECONDS)
|
|
|
|
/obj/machinery/computer/scan_consolenew/proc/find_machine()
|
|
for(var/obj/machinery/dna_scannernew/scanner in orange(1, src))
|
|
connected = scanner
|
|
return
|
|
|
|
/obj/machinery/computer/scan_consolenew/proc/ready()
|
|
injector_ready = TRUE
|
|
|
|
/obj/machinery/computer/scan_consolenew/proc/all_dna_blocks(list/buffer)
|
|
var/list/arr = list()
|
|
for(var/i = 1, i <= length(buffer), i++)
|
|
arr += "[i]:[ENCODE_DNA_BLOCK(buffer[i])]"
|
|
return arr
|
|
|
|
/obj/machinery/computer/scan_consolenew/proc/setInjectorBlock(obj/item/dnainjector/I, blk, datum/dna2_record/buffer)
|
|
var/pos = findtext(blk,":")
|
|
if(!pos)
|
|
return FALSE
|
|
var/id = text2num(copytext(blk,1,pos))
|
|
if(!id)
|
|
return FALSE
|
|
I.block = id
|
|
I.buf = buffer
|
|
return TRUE
|
|
|
|
/obj/machinery/computer/scan_consolenew/attack_ai(mob/user)
|
|
add_hiddenprint(user)
|
|
attack_hand(user)
|
|
|
|
/obj/machinery/computer/scan_consolenew/attack_hand(mob/user)
|
|
if(isnull(connected))
|
|
for(dir in list(NORTH,EAST,SOUTH,WEST))
|
|
connected = locate(/obj/machinery/dna_scannernew, get_step(src, dir))
|
|
if(!isnull(connected))
|
|
attack_hand(user)
|
|
break
|
|
else
|
|
if(..(user))
|
|
return
|
|
|
|
if(stat & (NOPOWER|BROKEN))
|
|
return
|
|
|
|
ui_interact(user)
|
|
|
|
/obj/machinery/computer/scan_consolenew/ui_state(mob/user)
|
|
return GLOB.default_state
|
|
|
|
/obj/machinery/computer/scan_consolenew/ui_interact(mob/user, datum/tgui/ui = null)
|
|
if(user == connected.occupant)
|
|
return
|
|
|
|
ui = SStgui.try_update_ui(user, src, ui)
|
|
if(!ui)
|
|
ui = new(user, src, "DNAModifier", name)
|
|
ui.open()
|
|
|
|
/obj/machinery/computer/scan_consolenew/ui_data(mob/user)
|
|
var/list/data = list()
|
|
data["selectedMenuKey"] = selected_menu_key
|
|
data["locked"] = connected.locked
|
|
data["hasOccupant"] = connected.occupant ? 1 : 0
|
|
|
|
data["isInjectorReady"] = injector_ready
|
|
|
|
data["hasDisk"] = disk ? 1 : 0
|
|
|
|
var/diskData[0]
|
|
if(!disk || !disk.buf)
|
|
diskData["data"] = null
|
|
diskData["owner"] = null
|
|
diskData["label"] = null
|
|
diskData["type"] = null
|
|
diskData["ue"] = null
|
|
else
|
|
diskData = disk.buf.GetData()
|
|
data["disk"] = diskData
|
|
|
|
var/list/new_buffers = list()
|
|
for(var/datum/dna2_record/buf in buffers)
|
|
new_buffers += list(buf.GetData())
|
|
data["buffers"]=new_buffers
|
|
|
|
data["radiationIntensity"] = radiation_intensity
|
|
data["radiationDuration"] = radiation_duration
|
|
data["irradiating"] = irradiating
|
|
|
|
data["dnaBlockSize"] = DNA_BLOCK_SIZE
|
|
data["selectedUIBlock"] = selected_ui_block
|
|
data["selectedUISubBlock"] = selected_ui_subblock
|
|
data["selectedSEBlock"] = selected_se_block
|
|
data["selectedSESubBlock"] = selected_se_subblock
|
|
data["selectedUITarget"] = selected_ui_target
|
|
data["selectedUITargetHex"] = selected_ui_target_hex
|
|
|
|
var/occupantData[0]
|
|
if(!connected.occupant || !connected.occupant.dna)
|
|
occupantData["name"] = null
|
|
occupantData["stat"] = null
|
|
occupantData["isViableSubject"] = null
|
|
occupantData["health"] = null
|
|
occupantData["maxHealth"] = null
|
|
occupantData["minHealth"] = null
|
|
occupantData["uniqueEnzymes"] = null
|
|
occupantData["uniqueIdentity"] = null
|
|
occupantData["structuralEnzymes"] = null
|
|
occupantData["radiationLevel"] = null
|
|
else
|
|
occupantData["name"] = connected.occupant.dna.real_name
|
|
occupantData["stat"] = connected.occupant.stat
|
|
occupantData["isViableSubject"] = 1
|
|
if((HAS_TRAIT(connected.occupant, TRAIT_BADDNA) && connected.scan_level < 3) || !connected.occupant.dna || HAS_TRAIT(connected.occupant, TRAIT_GENELESS))
|
|
occupantData["isViableSubject"] = 0
|
|
occupantData["health"] = connected.occupant.health
|
|
occupantData["maxHealth"] = connected.occupant.maxHealth
|
|
occupantData["minHealth"] = HEALTH_THRESHOLD_DEAD
|
|
occupantData["uniqueEnzymes"] = connected.occupant.dna.unique_enzymes
|
|
occupantData["uniqueIdentity"] = connected.occupant.dna.uni_identity
|
|
occupantData["structuralEnzymes"] = connected.occupant.dna.struc_enzymes
|
|
occupantData["radiationLevel"] = connected.occupant.radiation
|
|
data["occupant"] = occupantData
|
|
|
|
data["isBeakerLoaded"] = connected.beaker ? 1 : 0
|
|
data["beakerLabel"] = null
|
|
data["beakerVolume"] = 0
|
|
if(connected.beaker)
|
|
data["beakerLabel"] = connected.beaker.label_text ? connected.beaker.label_text : null
|
|
if(length(connected.beaker.reagents.reagent_list))
|
|
for(var/datum/reagent/R in connected.beaker.reagents.reagent_list)
|
|
data["beakerVolume"] += R.volume
|
|
|
|
// Transfer modal information if there is one
|
|
data["modal"] = ui_modal_data(src)
|
|
|
|
return data
|
|
|
|
/obj/machinery/computer/scan_consolenew/ui_act(action, params)
|
|
if(..())
|
|
return FALSE // don't update uis
|
|
if(!isturf(usr.loc))
|
|
return FALSE // don't update uis
|
|
if(!src || !connected)
|
|
return FALSE // don't update uis
|
|
if(irradiating) // Make sure that it isn't already irradiating someone...
|
|
return FALSE // don't update uis
|
|
if(stat & (NOPOWER|BROKEN))
|
|
return
|
|
|
|
add_fingerprint(usr)
|
|
|
|
if(ui_act_modal(action, params))
|
|
return TRUE
|
|
|
|
. = TRUE
|
|
switch(action)
|
|
if("selectMenuKey")
|
|
var/key = params["key"]
|
|
if(!(key in list(PAGE_UI, PAGE_SE, PAGE_BUFFER, PAGE_REJUVENATORS)))
|
|
return
|
|
selected_menu_key = key
|
|
if("toggleLock")
|
|
if(connected && connected.occupant)
|
|
connected.locked = !(connected.locked)
|
|
if("pulseRadiation")
|
|
irradiating = radiation_duration
|
|
var/lock_state = connected.locked
|
|
connected.locked = TRUE //lock it
|
|
|
|
SStgui.update_uis(src)
|
|
sleep(10 * radiation_duration) // sleep for radiation_duration seconds
|
|
|
|
irradiating = 0
|
|
connected.locked = lock_state
|
|
|
|
if(!connected.occupant)
|
|
return
|
|
|
|
var/radiation = (((radiation_intensity * 3) + radiation_duration * 3) / connected.damage_coeff)
|
|
connected.occupant.apply_effect(radiation, IRRADIATE)
|
|
if(connected.radiation_check())
|
|
return
|
|
|
|
if(prob(95))
|
|
if(prob(75))
|
|
randmutb(connected.occupant)
|
|
else
|
|
randmuti(connected.occupant)
|
|
else
|
|
if(prob(95))
|
|
randmutg(connected.occupant)
|
|
else
|
|
randmuti(connected.occupant)
|
|
if("radiationDuration")
|
|
radiation_duration = clamp(text2num(params["value"]), 1, 20)
|
|
if("radiationIntensity")
|
|
radiation_intensity = clamp(text2num(params["value"]), 1, 10)
|
|
////////////////////////////////////////////////////////
|
|
if("changeUITarget")
|
|
selected_ui_target = clamp(text2num(params["value"]), 1, 15)
|
|
selected_ui_target_hex = num2text(selected_ui_target, 1, 16)
|
|
if("selectUIBlock") // This chunk of code updates selected block / sub-block based on click
|
|
var/select_block = text2num(params["block"])
|
|
var/select_subblock = text2num(params["subblock"])
|
|
if(!select_block || !select_subblock)
|
|
return
|
|
|
|
selected_ui_block = clamp(select_block, 1, DNA_UI_LENGTH)
|
|
selected_ui_subblock = clamp(select_subblock, 1, DNA_BLOCK_SIZE)
|
|
if("pulseUIRadiation")
|
|
var/block = connected.occupant.dna.GetUISubBlock(selected_ui_block, selected_ui_subblock)
|
|
|
|
irradiating = radiation_duration
|
|
var/lock_state = connected.locked
|
|
connected.locked = TRUE //lock it
|
|
|
|
SStgui.update_uis(src)
|
|
sleep(10 * radiation_duration) // sleep for radiation_duration seconds
|
|
|
|
irradiating = 0
|
|
connected.locked = lock_state
|
|
|
|
if(!connected.occupant)
|
|
return
|
|
|
|
if(prob((80 + (radiation_duration / 2))))
|
|
var/radiation = (radiation_intensity + radiation_duration)
|
|
connected.occupant.apply_effect(radiation, IRRADIATE)
|
|
|
|
if(connected.radiation_check())
|
|
return
|
|
|
|
block = miniscrambletarget(num2text(selected_ui_target), radiation_intensity, radiation_duration)
|
|
connected.occupant.dna.SetUISubBlock(selected_ui_block, selected_ui_subblock, block)
|
|
connected.occupant.UpdateAppearance()
|
|
else
|
|
var/radiation = ((radiation_intensity * 2) + radiation_duration)
|
|
connected.occupant.apply_effect(radiation, IRRADIATE)
|
|
if(connected.radiation_check())
|
|
return
|
|
|
|
if(prob(20 + radiation_intensity))
|
|
randmutb(connected.occupant)
|
|
domutcheck(connected.occupant)
|
|
else
|
|
randmuti(connected.occupant)
|
|
connected.occupant.UpdateAppearance()
|
|
////////////////////////////////////////////////////////
|
|
if("injectRejuvenators")
|
|
if(!connected.occupant || !connected.beaker)
|
|
return
|
|
var/inject_amount = clamp(round(text2num(params["amount"]), 5), 0, 50) // round to nearest 5 and clamp to 0-50
|
|
if(!inject_amount)
|
|
return
|
|
connected.beaker.reagents.trans_to(connected.occupant, inject_amount)
|
|
connected.beaker.reagents.reaction(connected.occupant)
|
|
////////////////////////////////////////////////////////
|
|
if("selectSEBlock") // This chunk of code updates selected block / sub-block based on click (se stands for strutural enzymes)
|
|
var/select_block = text2num(params["block"])
|
|
var/select_subblock = text2num(params["subblock"])
|
|
if(!select_block || !select_subblock)
|
|
return
|
|
|
|
selected_se_block = clamp(select_block, 1, DNA_SE_LENGTH)
|
|
selected_se_subblock = clamp(select_subblock, 1, DNA_BLOCK_SIZE)
|
|
if("pulseSERadiation")
|
|
var/block = connected.occupant.dna.GetSESubBlock(selected_se_block, selected_se_subblock)
|
|
//var/original_block=block
|
|
//testing("Irradiating SE block [selected_se_block]:[selected_se_subblock] ([block])...")
|
|
|
|
irradiating = radiation_duration
|
|
var/lock_state = connected.locked
|
|
connected.locked = TRUE //lock it
|
|
|
|
SStgui.update_uis(src)
|
|
sleep(10 * radiation_duration) // sleep for radiation_duration seconds
|
|
|
|
irradiating = 0
|
|
connected.locked = lock_state
|
|
|
|
if(connected.occupant)
|
|
if(prob((80 + ((radiation_duration / 2) + (connected.precision_coeff ** 3)))))
|
|
var/radiation = ((radiation_intensity + radiation_duration) / connected.damage_coeff)
|
|
connected.occupant.apply_effect(radiation, IRRADIATE)
|
|
|
|
if(connected.radiation_check())
|
|
return 1
|
|
|
|
var/real_SE_block=selected_se_block
|
|
block = miniscramble(block, radiation_intensity, radiation_duration)
|
|
if(prob(20))
|
|
if(selected_se_block > 1 && selected_se_block < DNA_SE_LENGTH/2)
|
|
real_SE_block++
|
|
else if(selected_se_block > DNA_SE_LENGTH/2 && selected_se_block < DNA_SE_LENGTH)
|
|
real_SE_block--
|
|
|
|
//testing("Irradiated SE block [real_SE_block]:[selected_se_subblock] ([original_block] now [block]) [(real_SE_block!=selected_se_block) ? "(SHIFTED)":""]!")
|
|
connected.occupant.dna.SetSESubBlock(real_SE_block, selected_se_subblock, block)
|
|
domutcheck(connected.occupant)
|
|
else
|
|
var/radiation = (((radiation_intensity * 2) + radiation_duration) / connected.damage_coeff)
|
|
connected.occupant.apply_effect(radiation, IRRADIATE)
|
|
|
|
if(connected.radiation_check())
|
|
return
|
|
|
|
if(prob(80 - radiation_duration))
|
|
//testing("Random bad mut!")
|
|
randmutb(connected.occupant)
|
|
domutcheck(connected.occupant)
|
|
else
|
|
randmuti(connected.occupant)
|
|
//testing("Random identity mut!")
|
|
connected.occupant.UpdateAppearance()
|
|
if("ejectBeaker")
|
|
if(connected.beaker)
|
|
var/obj/item/reagent_containers/glass/B = connected.beaker
|
|
B.forceMove(connected.loc)
|
|
connected.beaker = null
|
|
if("ejectOccupant")
|
|
connected.eject_occupant()
|
|
// Transfer Buffer Management
|
|
if("bufferOption")
|
|
var/bufferOption = params["option"]
|
|
var/bufferId = text2num(params["id"])
|
|
if(bufferId < 1 || bufferId > 3) // Not a valid buffer id
|
|
return
|
|
|
|
var/datum/dna2_record/buffer = buffers[bufferId]
|
|
switch(bufferOption)
|
|
if("saveUI")
|
|
if(connected.occupant && connected.occupant.dna)
|
|
var/datum/dna2_record/databuf = new
|
|
databuf.types = DNA2_BUF_UI // DNA2_BUF_UE
|
|
databuf.dna = connected.occupant.dna.Clone()
|
|
if(ishuman(connected.occupant))
|
|
databuf.dna.real_name=connected.occupant.name
|
|
databuf.name = "Unique Identifier"
|
|
buffers[bufferId] = databuf
|
|
if("saveUIAndUE")
|
|
if(connected.occupant && connected.occupant.dna)
|
|
var/datum/dna2_record/databuf = new
|
|
databuf.types = DNA2_BUF_UI|DNA2_BUF_UE
|
|
databuf.dna = connected.occupant.dna.Clone()
|
|
if(ishuman(connected.occupant))
|
|
databuf.dna.real_name=connected.occupant.dna.real_name
|
|
databuf.name = "Unique Identifier + Unique Enzymes"
|
|
buffers[bufferId] = databuf
|
|
if("saveSE")
|
|
if(connected.occupant && connected.occupant.dna)
|
|
var/datum/dna2_record/databuf = new
|
|
databuf.types = DNA2_BUF_SE
|
|
databuf.dna = connected.occupant.dna.Clone()
|
|
if(ishuman(connected.occupant))
|
|
databuf.dna.real_name = connected.occupant.dna.real_name
|
|
databuf.name = "Structural Enzymes"
|
|
buffers[bufferId] = databuf
|
|
if("clear")
|
|
buffers[bufferId] = new /datum/dna2_record()
|
|
if("changeLabel")
|
|
ui_modal_input(src, "changeBufferLabel", "Please enter the new buffer label:", null, list("id" = bufferId), buffer.name, UI_MODAL_INPUT_MAX_LENGTH_NAME)
|
|
if("transfer")
|
|
if(!connected.occupant || (HAS_TRAIT(connected.occupant, TRAIT_BADDNA) && connected.scan_level < 3) || !connected.occupant.dna)
|
|
return
|
|
|
|
irradiating = 2
|
|
var/lock_state = connected.locked
|
|
connected.locked = TRUE //lock it
|
|
|
|
SStgui.update_uis(src)
|
|
sleep(2 SECONDS)
|
|
|
|
irradiating = 0
|
|
connected.locked = lock_state
|
|
|
|
var/radiation = (rand(20,50) / connected.damage_coeff)
|
|
connected.occupant.apply_effect(radiation, IRRADIATE)
|
|
|
|
if(connected.radiation_check())
|
|
return
|
|
|
|
var/datum/dna2_record/buf = buffers[bufferId]
|
|
|
|
if(buf.types & DNA2_BUF_UI)
|
|
if(buf.types & DNA2_BUF_UE)
|
|
connected.occupant.real_name = buf.dna.real_name
|
|
connected.occupant.name = buf.dna.real_name
|
|
connected.occupant.UpdateAppearance(buf.dna.UI.Copy())
|
|
else if(buf.types & DNA2_BUF_SE)
|
|
connected.occupant.dna.SE = buf.dna.SE.Copy()
|
|
connected.occupant.dna.UpdateSE()
|
|
domutcheck(connected.occupant)
|
|
if("createInjector")
|
|
if(!injector_ready)
|
|
return
|
|
if(text2num(params["block"]) > 0)
|
|
var/list/choices = all_dna_blocks((buffer.types & DNA2_BUF_SE) ? buffer.dna.SE : buffer.dna.UI)
|
|
ui_modal_choice(src, "createInjectorBlock", "Please select the block to create an injector from:", null, list("id" = bufferId), null, choices)
|
|
else
|
|
create_injector(bufferId, TRUE)
|
|
if("loadDisk")
|
|
if(isnull(disk) || disk.read_only)
|
|
return
|
|
buffers[bufferId] = disk.buf.copy()
|
|
if("saveDisk")
|
|
if(isnull(disk) || disk.read_only)
|
|
return
|
|
var/datum/dna2_record/buf = buffers[bufferId]
|
|
disk.buf = buf.copy()
|
|
disk.name = "data disk - '[buf.name]'"
|
|
if("wipeDisk")
|
|
if(isnull(disk) || disk.read_only)
|
|
return
|
|
disk.buf = null
|
|
if("ejectDisk")
|
|
if(!disk)
|
|
return
|
|
disk.forceMove(get_turf(src))
|
|
disk = null
|
|
|
|
/**
|
|
* Creates a blank injector with the name of the buffer at the given buffer_id
|
|
*
|
|
* Arguments:
|
|
* * buffer_id - The ID of the buffer
|
|
* * copy_buffer - Whether the injector should copy the buffer contents
|
|
*/
|
|
/obj/machinery/computer/scan_consolenew/proc/create_injector(buffer_id, copy_buffer = FALSE)
|
|
if(buffer_id < 1 || buffer_id > length(buffers))
|
|
return
|
|
|
|
// Cooldown
|
|
injector_ready = FALSE
|
|
addtimer(CALLBACK(src, PROC_REF(injector_cooldown_finish)), 10 SECONDS)
|
|
|
|
// Create it
|
|
var/datum/dna2_record/buf = buffers[buffer_id]
|
|
var/obj/item/dnainjector/I = new()
|
|
I.forceMove(loc)
|
|
I.name += " ([buf.name])"
|
|
if(copy_buffer)
|
|
I.buf = buf.copy()
|
|
if(connected)
|
|
I.damage_coeff = connected.damage_coeff
|
|
return I
|
|
|
|
/**
|
|
* Called when the injector creation cooldown finishes
|
|
*/
|
|
/obj/machinery/computer/scan_consolenew/proc/injector_cooldown_finish()
|
|
injector_ready = TRUE
|
|
|
|
/**
|
|
* Called in ui_act() to process modal actions
|
|
*
|
|
* Arguments:
|
|
* * action - The action passed by tgui
|
|
* * params - The params passed by tgui
|
|
*/
|
|
/obj/machinery/computer/scan_consolenew/proc/ui_act_modal(action, params)
|
|
. = TRUE
|
|
var/id = params["id"] // The modal's ID
|
|
var/list/arguments = istext(params["arguments"]) ? json_decode(params["arguments"]) : params["arguments"]
|
|
switch(ui_modal_act(src, action, params))
|
|
if(UI_MODAL_ANSWER)
|
|
var/answer = params["answer"]
|
|
switch(id)
|
|
if("createInjectorBlock")
|
|
var/buffer_id = text2num(arguments["id"])
|
|
if(buffer_id < 1 || buffer_id > length(buffers))
|
|
return
|
|
var/datum/dna2_record/buf = buffers[buffer_id]
|
|
var/obj/item/dnainjector/I = create_injector(buffer_id)
|
|
setInjectorBlock(I, answer, buf.copy())
|
|
if("changeBufferLabel")
|
|
var/buffer_id = text2num(arguments["id"])
|
|
if(buffer_id < 1 || buffer_id > length(buffers))
|
|
return
|
|
var/datum/dna2_record/buf = buffers[buffer_id]
|
|
buf.name = answer
|
|
buffers[buffer_id] = buf
|
|
else
|
|
return FALSE
|
|
else
|
|
return FALSE
|
|
|
|
|
|
#undef PAGE_UI
|
|
#undef PAGE_SE
|
|
#undef PAGE_BUFFER
|
|
#undef PAGE_REJUVENATORS
|
|
#undef DNA_BLOCK_SIZE
|
|
#undef NEGATE_MUTATION_THRESHOLD
|