From 69f20e250b9ec5f665a36301d51791014537cee1 Mon Sep 17 00:00:00 2001
From: AffectedArc07 <25063394+AffectedArc07@users.noreply.github.com>
Date: Tue, 14 May 2024 10:29:38 +0100
Subject: [PATCH] Removes Nanomob (#25417)
* Nukes nanomob
* Forgot this
* Update code/modules/client/preference/loadout/loadout_general.dm
Co-authored-by: GDN <96800819+GDNgit@users.noreply.github.com>
Signed-off-by: Burzah <116982774+Burzah@users.noreply.github.com>
---------
Signed-off-by: Burzah <116982774+Burzah@users.noreply.github.com>
Co-authored-by: Burzah <116982774+Burzah@users.noreply.github.com>
Co-authored-by: GDN <96800819+GDNgit@users.noreply.github.com>
---
_maps/map_files/stations/boxstation.dmm | 29 +-
_maps/map_files/stations/cerestation.dmm | 10 +-
_maps/map_files/stations/metastation.dmm | 9 -
code/__DEFINES/nanomob_defines.dm | 17 -
code/__DEFINES/subsystems.dm | 1 -
.../subsystem/SSnano_mob_hunter.dm | 155 ------
.../game/machinery/computer/communications.dm | 13 -
.../game/machinery/vendors/generic_vendors.dm | 2 -
code/game/objects/effects/landmarks.dm | 3 -
.../arcade/mob_hunt/battle_computer.dm | 309 -----------
code/modules/arcade/mob_hunt/mob_avatar.dm | 168 ------
code/modules/arcade/mob_hunt/mob_cards.dm | 43 --
code/modules/arcade/mob_hunt/mob_datums.dm | 480 ------------------
.../arcade/mob_hunt/mob_type_datums.dm | 241 ---------
code/modules/arcade/prize_datums.dm | 6 -
.../asset_cache/assets/asset_mob_hunt.dm | 7 -
.../preference/loadout/loadout_general.dm | 5 -
code/modules/economy/merch_items.dm | 7 -
code/modules/pda/PDA.dm | 3 -
code/modules/pda/cart.dm | 37 --
code/modules/pda/mob_hunt_game_app.dm | 205 --------
code/modules/pda/pda_tgui.dm | 5 -
icons/effects/mob_hunt.dmi | Bin 51165 -> 0 bytes
icons/obj/computer.dmi | Bin 147566 -> 147649 bytes
paradise.dme | 9 -
.../tgui/interfaces/CommunicationsComputer.js | 8 -
.../tgui/interfaces/pda/pda_mob_hunt.js | 131 -----
tgui/public/tgui.bundle.js | 130 ++---
28 files changed, 71 insertions(+), 1962 deletions(-)
delete mode 100644 code/__DEFINES/nanomob_defines.dm
delete mode 100644 code/controllers/subsystem/SSnano_mob_hunter.dm
delete mode 100644 code/modules/arcade/mob_hunt/battle_computer.dm
delete mode 100644 code/modules/arcade/mob_hunt/mob_avatar.dm
delete mode 100644 code/modules/arcade/mob_hunt/mob_cards.dm
delete mode 100644 code/modules/arcade/mob_hunt/mob_datums.dm
delete mode 100644 code/modules/arcade/mob_hunt/mob_type_datums.dm
delete mode 100644 code/modules/asset_cache/assets/asset_mob_hunt.dm
delete mode 100644 code/modules/pda/mob_hunt_game_app.dm
delete mode 100644 icons/effects/mob_hunt.dmi
delete mode 100644 tgui/packages/tgui/interfaces/pda/pda_mob_hunt.js
diff --git a/_maps/map_files/stations/boxstation.dmm b/_maps/map_files/stations/boxstation.dmm
index 412826b1377c..e04d48bfe0af 100644
--- a/_maps/map_files/stations/boxstation.dmm
+++ b/_maps/map_files/stations/boxstation.dmm
@@ -11923,7 +11923,6 @@
/area/station/hallway/secondary/entry/north)
"aMC" = (
/obj/structure/table,
-/obj/machinery/computer/mob_healer_terminal,
/turf/simulated/floor/carpet/arcade,
/area/station/public/arcade)
"aMD" = (
@@ -14876,9 +14875,6 @@
/area/station/medical/reception)
"aVy" = (
/obj/machinery/economy/vending/coffee,
-/obj/machinery/computer/mob_healer_terminal{
- pixel_x = 28
- },
/obj/structure/sign/poster/official/random{
pixel_y = 32
},
@@ -55778,13 +55774,6 @@
icon_state = "darkgreen"
},
/area/station/medical/medbay3)
-"fes" = (
-/obj/structure/chair{
- dir = 4
- },
-/obj/machinery/computer/mob_battle_terminal/red,
-/turf/simulated/floor/plasteel,
-/area/station/public/dorms)
"feN" = (
/turf/simulated/floor/plasteel,
/area/station/hallway/secondary/entry/north)
@@ -65342,12 +65331,6 @@
/obj/effect/spawner/window/reinforced,
/turf/simulated/floor/plating,
/area/station/supply/lobby)
-"jPN" = (
-/obj/machinery/computer/mob_battle_terminal/blue{
- pixel_x = -32
- },
-/turf/simulated/floor/plasteel,
-/area/station/public/dorms)
"jPZ" = (
/obj/structure/sign/poster/official/cleanliness{
pixel_x = 32
@@ -76992,10 +76975,6 @@
},
/turf/simulated/floor/plating,
/area/station/security/permabrig)
-"pwU" = (
-/obj/effect/landmark/battle_mob_point,
-/turf/simulated/floor/engine,
-/area/holodeck/alphadeck)
"pxv" = (
/obj/machinery/atmospherics/unary/outlet_injector/on{
dir = 8;
@@ -132341,7 +132320,7 @@ nXr
aKQ
fRL
aPm
-jPN
+aPm
lkw
fho
aPm
@@ -132590,7 +132569,7 @@ awE
ayf
azn
arR
-fes
+tbC
tbC
tbC
aPm
@@ -133621,10 +133600,10 @@ atG
sHt
sHt
sHt
-pwU
sHt
sHt
-pwU
+sHt
+sHt
sHt
sHt
sHt
diff --git a/_maps/map_files/stations/cerestation.dmm b/_maps/map_files/stations/cerestation.dmm
index 817cd6ed17c9..0eba8dd99f61 100644
--- a/_maps/map_files/stations/cerestation.dmm
+++ b/_maps/map_files/stations/cerestation.dmm
@@ -82602,12 +82602,6 @@
icon_state = "whitegreen"
},
/area/station/medical/virology)
-"tpw" = (
-/obj/effect/landmark/battle_mob_point,
-/turf/simulated/floor/engine{
- name = "Holodeck Projector Floor"
- },
-/area/holodeck/alphadeck)
"tpA" = (
/turf/simulated/floor/plasteel{
icon_state = "dark"
@@ -116641,10 +116635,10 @@ bEM
pDn
pDn
pDn
-tpw
pDn
pDn
-tpw
+pDn
+pDn
pDn
pDn
pDn
diff --git a/_maps/map_files/stations/metastation.dmm b/_maps/map_files/stations/metastation.dmm
index ca66c423b305..82aac31c65a6 100644
--- a/_maps/map_files/stations/metastation.dmm
+++ b/_maps/map_files/stations/metastation.dmm
@@ -806,9 +806,6 @@
/turf/simulated/wall,
/area/station/public/arcade)
"ajb" = (
-/obj/machinery/computer/mob_battle_terminal/red{
- pixel_y = 30
- },
/obj/item/kirbyplants/plant20,
/turf/simulated/floor/plasteel{
icon_state = "darkgrey"
@@ -3389,9 +3386,6 @@
name = "east bump";
pixel_x = 27
},
-/obj/machinery/computer/mob_battle_terminal/blue{
- pixel_y = -30
- },
/turf/simulated/floor/plasteel{
icon_state = "darkgrey"
},
@@ -53400,9 +53394,6 @@
/obj/structure/table,
/obj/item/stack/medical/bruise_pack,
/obj/item/stack/medical/ointment,
-/obj/machinery/computer/mob_healer_terminal{
- pixel_y = 30
- },
/turf/simulated/floor/plasteel{
dir = 1;
icon_state = "whiteblue"
diff --git a/code/__DEFINES/nanomob_defines.dm b/code/__DEFINES/nanomob_defines.dm
deleted file mode 100644
index 0e3510e62c3e..000000000000
--- a/code/__DEFINES/nanomob_defines.dm
+++ /dev/null
@@ -1,17 +0,0 @@
-#define NANOMOB_TYPE_FIRE /datum/mob_type/fire
-#define NANOMOB_TYPE_WATER /datum/mob_type/water
-#define NANOMOB_TYPE_GRASS /datum/mob_type/grass
-#define NANOMOB_TYPE_ELECTRIC /datum/mob_type/electric
-#define NANOMOB_TYPE_GROUND /datum/mob_type/ground
-#define NANOMOB_TYPE_ROCK /datum/mob_type/rock
-#define NANOMOB_TYPE_BUG /datum/mob_type/bug
-#define NANOMOB_TYPE_POISON /datum/mob_type/poison
-#define NANOMOB_TYPE_NORMAL /datum/mob_type/normal
-#define NANOMOB_TYPE_FIGHTING /datum/mob_type/fighting
-#define NANOMOB_TYPE_PSYCHIC /datum/mob_type/psychic
-#define NANOMOB_TYPE_GHOST /datum/mob_type/ghost
-#define NANOMOB_TYPE_ICE /datum/mob_type/ice
-#define NANOMOB_TYPE_FLYING /datum/mob_type/flying
-#define NANOMOB_TYPE_BLUESPACE /datum/mob_type/bluespace
-#define NANOMOB_TYPE_DARK /datum/mob_type/dark
-#define NANOMOB_TYPE_STEEL /datum/mob_type/steel
diff --git a/code/__DEFINES/subsystems.dm b/code/__DEFINES/subsystems.dm
index 335ac141fc8f..b213db5ac2dc 100644
--- a/code/__DEFINES/subsystems.dm
+++ b/code/__DEFINES/subsystems.dm
@@ -91,7 +91,6 @@
// If the subsystem isn't listed here it's either DEFAULT or PROCESS (if it's a processing subsystem child)
#define FIRE_PRIORITY_PING 10
-#define FIRE_PRIORITY_NANOMOB 10
#define FIRE_PRIORITY_NIGHTSHIFT 10
#define FIRE_PRIORITY_IDLE_NPC 10
#define FIRE_PRIORITY_CLEANUP 10
diff --git a/code/controllers/subsystem/SSnano_mob_hunter.dm b/code/controllers/subsystem/SSnano_mob_hunter.dm
deleted file mode 100644
index 93e622f3ae1b..000000000000
--- a/code/controllers/subsystem/SSnano_mob_hunter.dm
+++ /dev/null
@@ -1,155 +0,0 @@
-SUBSYSTEM_DEF(mob_hunt)
- name = "Nano-Mob Hunter GO Server"
- priority = FIRE_PRIORITY_NANOMOB // Low priority, no need for MC_TICK_CHECK due to extremely low performance impact.
- flags = SS_NO_INIT
- offline_implications = "Nano-Mob Hunter will no longer spawn mobs. No immediate action is needed."
- cpu_display = SS_CPUDISPLAY_LOW
- var/max_normal_spawns = 15 //change this to adjust the number of normal spawns that can exist at one time. trapped spawns (from traitors) don't count towards this
- var/list/normal_spawns = list()
- var/max_trap_spawns = 15 //change this to adjust the number of trap spawns that can exist at one time. traps spawned beyond this point clear the oldest traps
- var/list/trap_spawns = list()
- var/list/connected_clients = list()
- var/server_status = 1 //1 is online, 0 is offline
- var/reset_cooldown = 0 //number of controller cycles before the manual_reboot proc can be used again (ignored if server is offline so you can always boot back up)
- var/obj/machinery/computer/mob_battle_terminal/red_terminal
- var/obj/machinery/computer/mob_battle_terminal/blue_terminal
- var/battle_turn = null
-
-/datum/controller/subsystem/mob_hunt/fire(resumed = FALSE)
- if(reset_cooldown) //if reset_cooldown is set (we are on cooldown, duh), reduce the remaining cooldown every cycle
- reset_cooldown--
- if(!server_status)
- return
- client_mob_update()
- if(length(normal_spawns) < max_normal_spawns)
- spawn_mob()
-
-//leaving this here in case admins want to use it for a random mini-event or something
-/datum/controller/subsystem/mob_hunt/proc/server_crash(recover_time = 3000)
- server_status = 0
- for(var/datum/data/pda/app/mob_hunter_game/client in connected_clients)
- client.disconnect("Server Crash")
- for(var/obj/effect/nanomob/N in trap_spawns)
- N.despawn()
- for(var/obj/effect/nanomob/N in normal_spawns)
- N.despawn()
- //just in case
- normal_spawns.Cut()
- trap_spawns.Cut()
- connected_clients.Cut()
- if(!isnum(recover_time))
- recover_time = 3000
- if(recover_time > 0) //when provided with a negative or zero valued recover_time argument, the server won't auto-restart but can be manually rebooted still
- //set a timer to automatically recover after recover_time has passed (can be manually restarted if you get impatient too)
- addtimer(CALLBACK(src, PROC_REF(auto_recover)), recover_time, TIMER_UNIQUE)
-
-/datum/controller/subsystem/mob_hunt/proc/client_mob_update()
- var/list/ex_players = list()
- for(var/datum/data/pda/app/mob_hunter_game/client in connected_clients)
- var/mob/living/carbon/human/H = client.get_player()
- if(connected_clients[client])
- if(!H || H != connected_clients[client])
- ex_players |= connected_clients[client]
- connected_clients[client] = H
- if(length(ex_players)) //to make sure we don't do this if we didn't lose any player since the last update
- for(var/obj/effect/nanomob/N in (normal_spawns + trap_spawns))
- N.conceal(ex_players)
-
-/datum/controller/subsystem/mob_hunt/proc/auto_recover()
- if(server_status != 0)
- return
- server_status = 1
- while(length(normal_spawns) < max_normal_spawns) //repopulate the server's spawns completely if we auto-recover from crash
- spawn_mob()
-
-/datum/controller/subsystem/mob_hunt/proc/manual_reboot()
- if(server_status && reset_cooldown)
- return 0
- for(var/obj/effect/nanomob/N in trap_spawns)
- N.despawn()
- for(var/obj/effect/nanomob/N in normal_spawns)
- N.despawn()
- server_status = 1
- reset_cooldown = 25 //25 controller cycle cooldown for manual restarts
- return 1
-
-/datum/controller/subsystem/mob_hunt/proc/spawn_mob()
- var/list/nanomob_types = subtypesof(/datum/mob_hunt)
- var/datum/mob_hunt/mob_info = pick(nanomob_types)
- new mob_info()
-
-/datum/controller/subsystem/mob_hunt/proc/register_spawn(datum/mob_hunt/mob_info)
- if(!mob_info)
- return 0
- var/obj/effect/nanomob/new_mob = new /obj/effect/nanomob(mob_info.spawn_point, mob_info)
- normal_spawns += new_mob
- new_mob.reveal()
- return 1
-
-/datum/controller/subsystem/mob_hunt/proc/register_trap(datum/mob_hunt/mob_info)
- if(!mob_info)
- return 0
- if(!mob_info.is_trap)
- return register_spawn(mob_info)
- var/obj/effect/nanomob/new_mob = new /obj/effect/nanomob(mob_info.spawn_point, mob_info)
- trap_spawns += new_mob
- new_mob.reveal()
- if(length(trap_spawns) > max_trap_spawns)
- var/obj/effect/nanomob/old_trap = trap_spawns[1]
- old_trap.despawn()
- return 1
-
-/datum/controller/subsystem/mob_hunt/proc/start_check()
- if(battle_turn) //somehow we got called mid-battle, so lets just stop now
- return
- if(red_terminal && red_terminal.ready && blue_terminal && blue_terminal.ready)
- battle_turn = pick("Red", "Blue")
- red_terminal.atom_say("Battle starting!")
- blue_terminal.atom_say("Battle starting!")
- if(battle_turn == "Red")
- red_terminal.atom_say("Red Player's Turn!")
- else if(battle_turn == "Blue")
- blue_terminal.atom_say("Blue Player's Turn!")
-
-/datum/controller/subsystem/mob_hunt/proc/launch_attack(team, raw_damage, datum/mob_type/attack_type)
- if(!team || !raw_damage)
- return
- var/obj/machinery/computer/mob_battle_terminal/target = null
- if(team == "Red")
- target = blue_terminal
- else if(team == "Blue")
- target = red_terminal
- else
- return
- target.receive_attack(raw_damage, attack_type)
-
-/datum/controller/subsystem/mob_hunt/proc/end_battle(loser, surrender = 0)
- var/obj/machinery/computer/mob_battle_terminal/winner_terminal = null
- var/obj/machinery/computer/mob_battle_terminal/loser_terminal = null
- if(loser == "Red")
- loser_terminal = red_terminal
- winner_terminal = blue_terminal
- else if(loser == "Blue")
- loser_terminal = blue_terminal
- winner_terminal = red_terminal
- battle_turn = null
- winner_terminal.ready = FALSE
- loser_terminal.ready = FALSE
- if(surrender) //surrender doesn't give exp, to avoid people just farming exp without actually doing a battle
- winner_terminal.atom_say("Your rival surrendered!")
- else
- var/progress_message = winner_terminal.mob_info.gain_exp()
- winner_terminal.atom_say("[winner_terminal.team] Player wins!")
- winner_terminal.atom_say(progress_message)
-
-/datum/controller/subsystem/mob_hunt/proc/end_turn()
- red_terminal.updateUsrDialog()
- blue_terminal.updateUsrDialog()
- if(!battle_turn)
- return
- if(battle_turn == "Red")
- battle_turn = "Blue"
- blue_terminal.atom_say("Blue's turn.")
- else if(battle_turn == "Blue")
- battle_turn = "Red"
- blue_terminal.atom_say("Red's turn.")
diff --git a/code/game/machinery/computer/communications.dm b/code/game/machinery/computer/communications.dm
index 4b2541251358..36a0a848a2f2 100644
--- a/code/game/machinery/computer/communications.dm
+++ b/code/game/machinery/computer/communications.dm
@@ -300,20 +300,7 @@
emagged = FALSE
setMenuState(ui.user, COMM_SCREEN_MAIN)
- if("RestartNanoMob")
- if(SSmob_hunt)
- if(SSmob_hunt.manual_reboot())
- var/loading_msg = pick("Respawning spawns", "Reticulating splines", "Flipping hat",
- "Capturing all of them", "Fixing minor text issues", "Being the very best",
- "Nerfing this", "Not communicating with playerbase", "Coding a ripoff in a 2D spaceman game")
- to_chat(ui.user, "Restarting Nano-Mob Hunter GO! game server. [loading_msg]...")
- else
- to_chat(ui.user, "Nano-Mob Hunter GO! game server reboot failed due to recent restart. Please wait before re-attempting.")
- else
- to_chat(ui.user, "Nano-Mob Hunter GO! game server is offline for extended maintenance. Contact your Central Command administrators for more info if desired.")
-
// ADMIN CENTCOMM ONLY STUFF
-
if("send_to_cc_announcement_page")
if(!ADMIN_CHECK(ui.user))
return
diff --git a/code/game/machinery/vendors/generic_vendors.dm b/code/game/machinery/vendors/generic_vendors.dm
index a25e252705fc..d817a93c1fa8 100644
--- a/code/game/machinery/vendors/generic_vendors.dm
+++ b/code/game/machinery/vendors/generic_vendors.dm
@@ -1301,7 +1301,6 @@
icon_panel = "wide_vendor"
category = VENDOR_TYPE_SUPPLIES
products = list(/obj/item/pda =10,
- /obj/item/cartridge/mob_hunt_game = 25,
/obj/item/cartridge/medical = 10,
/obj/item/cartridge/chemistry = 10,
/obj/item/cartridge/engineering = 10,
@@ -1314,7 +1313,6 @@
/obj/item/cartridge/mime = 1)
prices = list(/obj/item/pda = 300,
- /obj/item/cartridge/mob_hunt_game = 50,
/obj/item/cartridge/medical = 200,
/obj/item/cartridge/chemistry = 150,
/obj/item/cartridge/engineering = 100,
diff --git a/code/game/objects/effects/landmarks.dm b/code/game/objects/effects/landmarks.dm
index a87d29096548..374a2fb1114d 100644
--- a/code/game/objects/effects/landmarks.dm
+++ b/code/game/objects/effects/landmarks.dm
@@ -678,8 +678,5 @@ INITIALIZE_IMMEDIATE(/obj/effect/landmark/newplayer_start) //Without this you sp
var/turf/simulated/T = get_turf(src)
T.burn_tile()
-/obj/effect/landmark/battle_mob_point
- name = "Nanomob Battle Avatar Spawn Point"
-
/obj/effect/landmark/free_golem_spawn
name = "Free Golem Spawn Point"
diff --git a/code/modules/arcade/mob_hunt/battle_computer.dm b/code/modules/arcade/mob_hunt/battle_computer.dm
deleted file mode 100644
index 228686fa951c..000000000000
--- a/code/modules/arcade/mob_hunt/battle_computer.dm
+++ /dev/null
@@ -1,309 +0,0 @@
-
-/obj/machinery/computer/mob_battle_terminal
- name = "Nano-Mob Hunter GO! Battle Terminal"
- desc = "Insert a mob card to partake in life-like Nano-Mob Battle Action!"
- icon_state = "mob_battle_empty"
- icon_screen = null
- icon_keyboard = null
- density = FALSE
- anchored = TRUE
- var/obj/item/nanomob_card/card
- var/datum/mob_hunt/mob_info
- var/obj/effect/landmark/battle_mob_point/avatar_point
- var/obj/effect/nanomob/battle/avatar
- var/ready = FALSE
- var/team = "Grey"
-
-/obj/machinery/computer/mob_battle_terminal/red
- pixel_y = 24
- dir = SOUTH
- team = "Red"
-
-/obj/machinery/computer/mob_battle_terminal/blue
- pixel_y = -24
- dir = NORTH
- team = "Blue"
-
-/obj/machinery/computer/mob_battle_terminal/red/Initialize()
- ..()
- check_connection()
-
-/obj/machinery/computer/mob_battle_terminal/blue/Initialize()
- ..()
- check_connection()
-
-/obj/machinery/computer/mob_battle_terminal/update_icon_state()
- if(card)
- icon_state = "mob_battle_loaded"
- else
- icon_state = "mob_battle_empty"
-
-/obj/machinery/computer/mob_battle_terminal/Destroy()
- eject_card(1)
- if(SSmob_hunt)
- if(SSmob_hunt.battle_turn)
- SSmob_hunt.battle_turn = null
- if(SSmob_hunt.red_terminal == src)
- SSmob_hunt.red_terminal = null
- if(SSmob_hunt.blue_terminal == src)
- SSmob_hunt.blue_terminal = null
- QDEL_NULL(avatar)
- return ..()
-
-/obj/machinery/computer/mob_battle_terminal/attackby(obj/item/O, mob/user)
- if(istype(O, /obj/item/nanomob_card))
- insert_card(O, user)
- return
- return ..()
-
-/obj/machinery/computer/mob_battle_terminal/proc/insert_card(obj/item/nanomob_card/new_card, mob/user)
- if(!new_card)
- return
- if(card)
- to_chat(user, "The card slot is currently filled.")
- return
- if(!new_card.mob_data)
- to_chat(user, "This is a blank mob card.")
- return
- if(new_card.mob_data && !new_card.mob_data.cur_health)
- to_chat(user, "This mob is incapacitated! Heal it before attempting to use it in battle!")
- return
- user.unEquip(new_card)
- new_card.forceMove(src)
- card = new_card
- mob_info = card.mob_data
- update_icon()
- update_avatar()
- updateUsrDialog()
-
-/obj/machinery/computer/mob_battle_terminal/proc/eject_card(override = 0)
- if(!override)
- if(ready && SSmob_hunt.battle_turn != team)
- atom_say("You can't recall on your rival's turn!")
- return
- if(!card)
- return
- card.mob_data = mob_info
- mob_info = null
- card.forceMove(get_turf(src))
- card = null
- update_avatar()
- update_icon()
- updateUsrDialog()
-
-/obj/machinery/computer/mob_battle_terminal/proc/update_avatar()
- //if we don't have avatars yet, spawn them
- if(!avatar)
- if(!avatar_point)
- log_debug("[src] attempted to spawn a battle mob avatar without a spawn point!")
- return
- else
- avatar = new(get_turf(avatar_point))
- //update avatar info from card
- if(mob_info)
- avatar.mob_info = mob_info
- else
- avatar.mob_info = null
- //tell the avatar to update themself with the new info
- avatar.update_self()
-
-/obj/machinery/computer/mob_battle_terminal/attack_hand(mob/user)
- add_fingerprint(user)
- interact(user)
-
-/obj/machinery/computer/mob_battle_terminal/attack_ai(mob/user)
- to_chat(user, "You cannot interface with this portion of the simulation.")
- return
-
-/obj/machinery/computer/mob_battle_terminal/interact(mob/user)
- check_connection()
- var/datum/asset/mob_hunt_asset = get_asset_datum(/datum/asset/simple/mob_hunt)
- mob_hunt_asset.send(user)
-
- var/dat = ""
- dat += "
"
- dat += ""
- dat += ""
- dat += "[team] PLAYER"
- dat += " | "
- dat += "
"
- dat += ""
- dat += ""
- if(!card)
- dat += "No Nano-Mob card loaded."
- dat += " | "
- dat += "
"
- if(ready && SSmob_hunt.battle_turn) //offer the surrender option if they are in a battle (ready), but don't have a card loaded
- dat += ""
- dat += "Surrender! | "
- dat += "
"
- else
- dat += ""
- dat += ""
- dat += ""
- dat += "[mob_info.mob_name]"
- dat += " | "
- if(mob_info.nickname)
- dat += ""
- else
- dat += " | "
- var/img_src = SSassets.transport.get_asset_url("[mob_info.icon_state_normal].png")
- if(mob_info.is_shiny)
- dat += SSassets.transport.get_asset_url("[mob_info.icon_state_shiny].png")
- dat += ""
- dat += " | "
- dat += "
"
- if(mob_info.nickname)
- dat += ""
- dat += ""
- dat += "[mob_info.nickname]"
- dat += " | "
- dat += "
"
- dat += "
"
- dat += "
"
- dat += "Health: [mob_info.cur_health] / [mob_info.max_health]
"
- dat += ""
- dat += ""
- if(mob_info.cur_health)
- dat += "Attack! | "
- else
- dat += "Incapacitated! | "
- dat += "Recall! | "
- dat += "
"
- dat += "
"
- dat += ""
- dat += ""
- if(!ready)
- dat += ""
- dat += "Battle! | "
- dat += "
"
- if(ready && !SSmob_hunt.battle_turn)
- dat += ""
- dat += "Cancel Battle! | "
- dat += "
"
- dat += "
"
-
- var/datum/browser/popup = new(user, "mob_battle_terminal", "Nano-Mob Hunter GO! Battle Terminal", 575, 400)
- popup.set_content(dat)
- popup.open()
-
-/obj/machinery/computer/mob_battle_terminal/Topic(href, href_list)
- if(..())
- return 1
-
- if(href_list["attack"])
- do_attack()
-
- if(href_list["eject"])
- eject_card()
-
- if(href_list["surrender"])
- surrender()
-
- if(href_list["ready"])
- var/option = text2num(href_list["ready"])
- if(option == 1)
- start_battle()
- else if(option == 2)
- ready = FALSE
- atom_say("[team] Player cancels their battle challenge.")
-
- updateUsrDialog()
-
-/obj/machinery/computer/mob_battle_terminal/proc/check_connection()
- if(team == "Red")
- if(SSmob_hunt && !SSmob_hunt.red_terminal)
- SSmob_hunt.red_terminal = src
- else if(team == "Blue")
- if(SSmob_hunt && !SSmob_hunt.blue_terminal)
- SSmob_hunt.blue_terminal = src
-
- find_avatar_spawn_point()
-
-/obj/machinery/computer/mob_battle_terminal/proc/find_avatar_spawn_point()
- if(avatar_point)
- return
- var/obj/effect/landmark/battle_mob_point/closest
- for(var/obj/effect/landmark/battle_mob_point/bmp in GLOB.landmarks_list)
- if(!istype(bmp, /obj/effect/landmark/battle_mob_point))
- continue
- if(bmp.z != z) //only match with points on the same z-level)
- continue
- if(!closest || isnull(closest))
- closest = bmp
- continue
- if(closest == bmp)
- continue
- if((abs(x-bmp.x)+abs(y-bmp.y)) < (abs(x-closest.x)+abs(y-closest.y))) //get_dist would be preferable if it didn't count diagonals as 1 distance, so we have to rely on actual x/y distances in this janky way.
- closest = bmp
- if(closest)
- avatar_point = closest
- else
- log_debug("[src] was unable to locate a nearby mob battle landmark! Unable to spawn battle avatars!")
-
-/obj/machinery/computer/mob_battle_terminal/proc/do_attack()
- if(!ready) //no attacking if you arent ready to fight (duh)
- return
- if(!SSmob_hunt || team != SSmob_hunt.battle_turn) //don't attack unless it is actually our turn
- return
- else
- var/message = "[mob_info.mob_name] attacks!"
- if(mob_info.nickname)
- message = "[mob_info.nickname] attacks!"
- atom_say(message)
- SSmob_hunt.launch_attack(team, mob_info.get_raw_damage(), mob_info.get_attack_type())
-
-/obj/machinery/computer/mob_battle_terminal/proc/start_battle()
- if(ready) //don't do anything if we are still ready
- return
- if(!card) //don't do anything if there isn't a card inserted
- return
- ready = TRUE
- atom_say("[team] Player is ready for battle! Waiting for rival...")
- SSmob_hunt.start_check()
-
-/obj/machinery/computer/mob_battle_terminal/proc/receive_attack(raw_damage, datum/mob_type/attack_type)
- var/message = mob_info.take_damage(raw_damage, attack_type)
- avatar.audible_message(message, null, 5)
- if(!mob_info.cur_health)
- SSmob_hunt.end_battle(team)
- eject_card(1) //force the card out, they were defeated
- else
- SSmob_hunt.end_turn()
-
-/obj/machinery/computer/mob_battle_terminal/proc/surrender()
- atom_say("[team] Player surrenders the battle!")
- SSmob_hunt.end_battle(team, 1)
-
-//////////////////////////////
-// Mob Healing Terminal //
-// (Pokemon Center) //
-//////////////////////////////
-
-/obj/machinery/computer/mob_healer_terminal
- name = "Nano-Mob Hunter GO! Restoration Terminal"
- desc = "Swipe a mob card to instantly restore it to full health!"
- icon_state = "mob_battle_loaded"
- icon_screen = null
- icon_keyboard = null
- density = FALSE
- anchored = TRUE
- dir = EAST
-
-/obj/machinery/computer/mob_healer_terminal/attackby(obj/item/O, mob/user)
- if(istype(O, /obj/item/nanomob_card))
- heal_card(O, user)
- return
- return ..()
-
-/obj/machinery/computer/mob_healer_terminal/proc/heal_card(obj/item/nanomob_card/patient, mob/user)
- if(!patient)
- return
- if(!patient.mob_data)
- to_chat(user, "This is a blank mob card.")
- return
- if(patient.mob_data && patient.mob_data.cur_health == patient.mob_data.max_health)
- to_chat(user, "This mob is already at maximum health!")
- return
- patient.mob_data.cur_health = patient.mob_data.max_health
- to_chat(user, "[patient.mob_data.nickname ? patient.mob_data.nickname : patient.mob_data.mob_name] has been restored to full health!")
diff --git a/code/modules/arcade/mob_hunt/mob_avatar.dm b/code/modules/arcade/mob_hunt/mob_avatar.dm
deleted file mode 100644
index ecab58523dbf..000000000000
--- a/code/modules/arcade/mob_hunt/mob_avatar.dm
+++ /dev/null
@@ -1,168 +0,0 @@
-/obj/effect/nanomob
- name = "Nano-Mob Avatar" //will be overridden by the mob datum name value when created
- desc = "A wild Nano-Mob appeared! Hit it with your PDA with the game open to attempt to capture it!"
- invisibility = 101
- alpha = 128
- anchored = TRUE //just in case
- density = FALSE
- icon = 'icons/effects/mob_hunt.dmi'
- var/state_name
- var/datum/mob_hunt/mob_info = null
- var/list/clients_encountered = list() //tracks who has already interacted with us, so they can't attempt a second capture
- var/image/avatar
-
-/obj/effect/nanomob/Initialize(mapload, datum/mob_hunt/new_info)
- . = ..()
- if(!new_info)
- return INITIALIZE_HINT_QDEL
- mob_info = new_info
- RegisterSignal(mob_info, COMSIG_PARENT_QDELETING, PROC_REF(delete_wrapper))
- update_self()
- forceMove(mob_info.spawn_point)
- if(!mob_info.is_trap)
- addtimer(CALLBACK(src, PROC_REF(despawn)), mob_info.lifetime)
-
-/obj/effect/nanomob/Destroy()
- SSmob_hunt.trap_spawns -= src
- SSmob_hunt.normal_spawns -= src
- mob_info = null // Can't delete this since multiple players can get the exact same /datum/mob_hunt. (This should be refactored)
- clients_encountered.Cut()
- QDEL_NULL(avatar)
- return ..()
-
-/obj/effect/nanomob/proc/delete_wrapper()
- SIGNAL_HANDLER
- qdel(src)
-
-/obj/effect/nanomob/proc/update_self()
- if(!mob_info)
- return
- name = mob_info.mob_name
- desc = "A wild [name] (level [mob_info.level]) appeared! Hit it with your PDA with the game open to attempt to capture it!"
- if(mob_info.is_shiny)
- state_name = mob_info.icon_state_shiny
- else
- state_name = mob_info.icon_state_normal
- avatar = image(icon, src, state_name)
- avatar.override = 1
- add_alt_appearance("nanomob_avatar", avatar)
-
-/obj/effect/nanomob/attackby(obj/item/O, mob/user)
- if(istype(O, /obj/item/pda))
- var/obj/item/pda/P = O
- attempt_capture(P, -20) //attempting a melee capture reduces the mob's effective run_chance by 20% to balance the risk of triggering a trap mob
- return 1
-
-/obj/effect/nanomob/hitby(atom/movable/AM, skipcatch, hitpush, blocked, datum/thrownthing/throwingdatum)
- if(istype(AM, /obj/item/pda))
- var/obj/item/pda/P = AM
- attempt_capture(P) //attempting a ranged capture does not affect the mob's effective run_chance but does prevent you from being shocked by a trap mob
- return 1
-
-/obj/effect/nanomob/proc/attempt_capture(obj/item/pda/P, catch_mod = 0) //negative catch_mods lower effective run chance,
- if(!P || !P.current_app || !istype(P.current_app, /datum/data/pda/app/mob_hunter_game) || !P.cartridge)
- return
-
- var/datum/data/pda/app/mob_hunter_game/client = P.current_app
- var/total_catch_mod = client.catch_mod + catch_mod //negative values decrease the chance of the mob running, positive values makes it more likely to flee
- if(!client.connected) //must be connected to attempt captures
- P.atom_say("No server connection. Capture aborted.")
- return
-
- if(mob_info.is_trap) //traps work even if you ran into them before, which is why this is before the clients_encountered check
- if(client.hacked) //hacked copies of the game (copies capable of setting traps) are protected from traps
- return
- if(iscarbon(P.loc))
- var/mob/living/carbon/C = P.loc
- //Strike them down with a lightning bolt to complete the illusion (copied from the surge reagent overdose, probably could make this a general-use proc in the future)
- playsound(get_turf(C), 'sound/effects/eleczap.ogg', 75, 1)
- var/icon/I=new('icons/obj/zap.dmi',"lightningend")
- I.Turn(-135)
- var/obj/effect/overlay/beam/B = new(get_turf(C))
- B.pixel_x = rand(-20, 0)
- B.pixel_y = rand(-20, 0)
- B.icon = I
- //then actually do the damage/stun
- C.electrocute_act(20, P, 1) //same damage as a revenant's overload ability, except subject to gloves/species shock resistance (for human mobs at least)
- return
-
- if(client in clients_encountered) //we've already dealt with you, go away!
- return
- else //deal with the new hunter by either running away or getting caught
- clients_encountered += client
- var/message = null
- var/effective_run_chance = mob_info.run_chance + total_catch_mod
- if((effective_run_chance > 0) && prob(effective_run_chance))
- message += "Capture failed! [name] escaped [P.owner ? "from [P.owner]" : "from this hunter"]!"
- conceal(client)
- else
- if(client.register_capture(mob_info, 1))
- message += "Capture success! [P.owner ? P.owner : "This hunter"] captured [name]!"
- conceal(client)
- else
- message += "Capture error! Try again."
- clients_encountered -= client //if the capture registration failed somehow, let them have another chance with this mob
- P.atom_say(message)
-
-/obj/effect/nanomob/proc/despawn()
- if(SSmob_hunt)
- if(mob_info.is_trap)
- SSmob_hunt.trap_spawns -= src
- else
- SSmob_hunt.normal_spawns -= src
- qdel(src)
-
-/obj/effect/nanomob/proc/reveal()
- if(!SSmob_hunt)
- return
- var/list/show_to = list()
- for(var/A in SSmob_hunt.connected_clients)
- if((A in clients_encountered) || !SSmob_hunt.connected_clients[A])
- continue
- show_to |= SSmob_hunt.connected_clients[A]
- display_alt_appearance("nanomob_avatar", show_to)
-
-/obj/effect/nanomob/proc/conceal(list/hide_from)
- if(!SSmob_hunt)
- return
- var/list/hiding_from = list()
- if(hide_from)
- hiding_from = hide_from
- else
- for(var/A in SSmob_hunt.connected_clients)
- if((A in clients_encountered) && SSmob_hunt.connected_clients[A])
- hiding_from |= SSmob_hunt.connected_clients[A]
- hide_alt_appearance("nanomob_avatar", hiding_from)
-
-// BATTLE MOB AVATARS
-
-/obj/effect/nanomob/battle
- name = "Nano-Mob Battle Avatar"
- desc = "A new challenger approaches!"
- invisibility = 0
- icon_state = "placeholder"
- var/obj/machinery/computer/mob_battle_terminal/my_terminal
-
-/obj/effect/nanomob/battle/update_self()
- if(!mob_info)
- name = "Nano-Mob Battle Avatar"
- desc = "A new challenger approaches"
- icon_state = "placeholder"
- else
- name = mob_info.mob_name
- desc = "A tamed [name] (level [mob_info.level]) ready for battle!"
- if(mob_info.is_shiny)
- icon_state = mob_info.icon_state_shiny
- else
- icon_state = mob_info.icon_state_normal
-
-/obj/effect/nanomob/battle/attempt_capture(obj/item/pda/P, catch_mod = 0)
- //you can't capture battle avatars, since they belong to someone already
- return
-
-//battle avatars are always visible, so we can ignore reveal and conceal calls for them
-/obj/effect/nanomob/battle/reveal()
- return
-
-/obj/effect/nanomob/battle/conceal()
- return
diff --git a/code/modules/arcade/mob_hunt/mob_cards.dm b/code/modules/arcade/mob_hunt/mob_cards.dm
deleted file mode 100644
index 6acfe826f445..000000000000
--- a/code/modules/arcade/mob_hunt/mob_cards.dm
+++ /dev/null
@@ -1,43 +0,0 @@
-
-//Standard Cards
-/obj/item/nanomob_card
- name = "Nano-Mob Hunter Trading Card"
- desc = "A blank Nano-Mob Hunter Trading Card. Worthless!"
- icon = 'icons/obj/card.dmi'
- icon_state = "trade_card"
- force = 0
- throwforce = 1
- w_class = WEIGHT_CLASS_TINY
- var/datum/mob_hunt/mob_data
-
-/obj/item/nanomob_card/proc/update_info()
- if(!mob_data)
- return
- if(mob_data.is_shiny)
- name = "Holographic [mob_data.mob_name] Nano-Mob Hunter Card"
- desc = "WOW! A holographic trading card containing a level [mob_data.level] [mob_data.mob_name]!"
- icon_state = "trade_card_holo"
- else
- name = "[mob_data.mob_name] Nano-Mob Hunter Card"
- desc = "A trading card containing a level [mob_data.level] [mob_data.mob_name]!"
-
-//Booster Pack Cards (random mob data)
-/obj/item/nanomob_card/booster
- name = "Nano-Mob Hunter Booster Pack Card"
- desc = "A random Nano-Mob Trading Card from a Booster Pack. Wonder what it is?"
-
-/obj/item/nanomob_card/booster/New()
- ..()
- var/datum/mob_hunt/mob_info = pick(subtypesof(/datum/mob_hunt))
- mob_data = new mob_info(0,null,1)
- update_info()
-
-//Booster Packs (Box of booster pack cards)
-/obj/item/storage/box/nanomob_booster_pack
- name = "Nano-Mob Hunter Trading Card Booster Pack"
- desc = "Contains 6 random Nano-Mob Hunter Trading Cards. May contain a holographic card!"
- can_hold = list(/obj/item/nanomob_card)
-
-/obj/item/storage/box/nanomob_booster_pack/populate_contents()
- for(var/i in 1 to 6)
- new /obj/item/nanomob_card/booster(src)
diff --git a/code/modules/arcade/mob_hunt/mob_datums.dm b/code/modules/arcade/mob_hunt/mob_datums.dm
deleted file mode 100644
index fc60e17bbbea..000000000000
--- a/code/modules/arcade/mob_hunt/mob_datums.dm
+++ /dev/null
@@ -1,480 +0,0 @@
-
-/datum/mob_hunt
- //GENERAL STATS AND VARIABLES
- var/mob_name = "Generic Mob" //the mob's original name (its species/type/whatever)
- var/nickname = "" //the mob's nickname (if given by the owner)
- var/run_chance = 0 //percent chance the mob will escape capture attempts (higher is obviously more likely to get away)
-
- //COMBAT STATS AND VARIABLES
- var/level = 0 //actual level of this mob (don't set this, it gets overwritten in New())
- var/min_level = 1 //minimum level of this mob (used for randomizing the actual level)
- var/max_level = 1 //maximum level of this mob (used for randomizing the actual level)
- var/exp = 0 //number of battles the mob has won towards the next level (resets to 0 on level-up)
- var/exp_to_level = 3 //number of battles the mob must win to level up (in case we want to make some mobs harder or easier to level)
- //the types of the mob will be used for battles to determine damage resistance or weakness (mob_type_datums.dm)
- var/datum/mob_type/primary_type //Make sure you set this or the mob will be unable to deal damage and will take absurd damage in battles
- var/datum/mob_type/secondary_type //Don't set if not a dual-type mob so the mob will only calculate damage based on primary type
- var/base_attack = 5 //base damage dealt by the mob's attacks for battling (effectively damage dealt at level 0)
- var/base_health = 5 //base health of the mob for battling (effectively max health at level 0)
- var/attack_multiplier = 1 //how much additional damage per level the mob deals (level * attack_multiplier)
- var/health_multiplier = 1 //how much additional health per level the mob gets (level * health_multiplier) for calculating max health
- var/cur_health = 0
- var/max_health = 0
-
- //SPAWN PREFERENCES AND VARIABLES
- //A note on mob spawn preferences: The mob types also have preferences, which are handled prior to per-mob preferences, so ultimately you use a combined set of preferences
- var/list/area_blacklist = list() //list of areas this mob can NOT spawn in (such as the bridge)
- var/list/turf_blacklist = list() //list of turfs this mob can NOT spawn on (such as wood floors)
- var/list/area_whitelist = list() //list of areas this mob is more likely to spawn in (can be used to reinclude subtypes of blacklisted areas)
- var/list/turf_whitelist = list() //list of turfs this mob is more likely to spawn on (can be used to reinclude subtypes of blacklisted turfs)
- var/turf/spawn_point //gets set and sent to the game server to spawn its avatar (generated in select_spawn or assigned via set_trap)
- var/lifetime = 6000 //number of deciseconds the mob will remain before despawning (REMEMBER: DECISECONDS! So 6000 is 600 seconds which is 10 minutes)
- var/is_shiny = 0 //if this gets set at spawn (super rare), the mob is considered "shiny" and will use the shiny icon_state and holographic cards
-
- //the icon file for mob_hunt stuff is 'icons/effects/mob_hunt.dmi' so reference that for the following vars
- var/icon_state_normal = "placeholder" //the icon_state for this mob's normal version
- var/icon_state_shiny = "placeholder" //the icon_state for this mob's rare shiney version
-
- var/is_trap = 0 //if this gets set, the mob is a booby-trap and will electrocute any players that dare attempt to catch it
-
-/datum/mob_hunt/New(set_trap = 0, turf/trap_turf = null, no_register = 0)
- if(set_trap)
- level = max_level
- is_trap = 1
- spawn_point = trap_turf
- else
- level = rand(min_level, max_level)
- if(prob(1) && prob(1))
- is_shiny = 1
- max_health = base_health + (level * health_multiplier)
- cur_health = max_health
- if(primary_type)
- primary_type = new primary_type()
- if(secondary_type)
- secondary_type = new secondary_type()
- if(no_register) //for booster pack cards
- return
- if(SSmob_hunt)
- if(set_trap)
- if(SSmob_hunt.register_trap(src))
- return
- else if(select_spawn())
- if(SSmob_hunt.register_spawn(src))
- return
- qdel(src) //if you reach this, the datum is just pure clutter, so delete it
-
-/datum/mob_hunt/proc/select_spawn()
- var/list/possible_areas = get_possible_areas()
- if(!length(possible_areas))
- log_admin("No possible areas to spawn [type] found. Possible code/mapping error?")
- return 0
- while(length(possible_areas))
- //randomly select an area from our possible_areas list to try spawning in, then remove it from possible_areas so it won't get picked over and over forever.
- var/spawn_area_path = pickweight(possible_areas)
- var/area/spawn_area = locate(spawn_area_path)
- possible_areas -= spawn_area_path
- if(!spawn_area)
- break
- //clear and generate a fresh list of turfs in the selected area, weighted based on white/black lists
- var/list/possible_turfs = get_possible_turfs(spawn_area)
- if(!length(possible_turfs)) //If we don't have any possible turfs, this attempt was a failure. Try again.
- continue
- //if we got this far, we're spawning on this attempt, hooray!
- spawn_point = pickweight(possible_turfs)
- break
- if(!spawn_point)
- //if we get to this, we failed every attempt to find a suitable turf for EVERY area in our list of possible areas. DAMN.
- log_admin("No acceptable turfs to spawn [type] on could be located. Possible code/mapping error, or someone replaced/destroyed all the acceptable turf types?")
- return 0
- return 1
-
-/datum/mob_hunt/proc/get_possible_areas()
- var/list/possible_areas = list()
- //setup, sets all station areas (and subtypes) to weight 1
- for(var/area/A in SSmapping.existing_station_areas)
- if(istype(A, /area/holodeck)) //don't allow holodeck areas as possible spawns since it will allow it to spawn in the holodeck rooms on CC level as well
- continue
- if(A in possible_areas)
- continue
- for(var/areapath in typesof(A))
- possible_areas[areapath] = 1
- //primary type preferences
- if(primary_type)
- for(var/A in primary_type.area_whitelist)
- for(var/areapath in typesof(A))
- possible_areas[areapath] += 4
- for(var/A in primary_type.area_blacklist)
- for(var/areapath in typesof(A))
- possible_areas[areapath] -= 2
- //secondary type preferences
- if(secondary_type)
- for(var/A in secondary_type.area_whitelist)
- for(var/areapath in typesof(A))
- possible_areas[areapath] += 4
- for(var/A in secondary_type.area_blacklist)
- for(var/areapath in typesof(A))
- possible_areas[areapath] -= 2
- //mob preferences
- for(var/A in area_whitelist)
- for(var/areapath in typesof(A))
- possible_areas[areapath] += 4
- for(var/A in area_blacklist)
- for(var/areapath in typesof(A))
- possible_areas[areapath] -= 2
- //weight check, remove negative or zero weight areas from the list, then return the list.
- for(var/areapath in possible_areas)
- //remove any areas that shouldn't be on the station-level
- if(possible_areas[areapath] < 1)
- possible_areas -= areapath
- continue
- return possible_areas
-
-/datum/mob_hunt/proc/get_possible_turfs(area/spawn_area)
- if(!spawn_area)
- return list()
- var/list/possible_turfs = list()
- //setup, sets all turfs in spawn_area to weight 1
- for(var/turf/T in spawn_area)
- if(!is_station_level(T.z)) //mobs will only consider station-level turfs for spawning. Largely here so we won't have to worry about mapping errors or mobs on the derelict solars
- continue
- possible_turfs[T] = 1
- //primary type preferences
- if(primary_type)
- if(is_type_in_list(T, primary_type.turf_whitelist))
- possible_turfs[T] += 4
- if(is_type_in_list(T, primary_type.turf_blacklist))
- possible_turfs[T] -= 2
- //secondary type preferences
- if(secondary_type)
- if(is_type_in_list(T, secondary_type.turf_whitelist))
- possible_turfs[T] += 4
- if(is_type_in_list(T, secondary_type.turf_blacklist))
- possible_turfs[T] -= 2
- //mob preferences
- if(is_type_in_list(T, turf_whitelist))
- possible_turfs[T] += 4
- if(is_type_in_list(T, turf_blacklist))
- possible_turfs[T] -= 2
- //weight check, remove negative or zero weight turfs from the list, then return the list
- if(possible_turfs[T] < 1)
- possible_turfs -= T
- return possible_turfs
-
-/datum/mob_hunt/proc/calc_def_multiplier(datum/mob_type/attack_type)
- if(!primary_type)
- return 99 //typeless mobs are weak to everything since they shouldn't exist
- if(!attack_type) //typeless attacks will return a multiplier of 1 in case we want to use this for calculating unmodified damage for some reason (UI maybe?)
- return 1
-
- var/multiplier = 1
- if(attack_type in primary_type.immunity)
- return 0 //a single immunity negates all damage
- else if(attack_type in primary_type.resistance)
- multiplier *= 0.5
- else if(attack_type in primary_type.weakness)
- multiplier *= 2
- else
- multiplier *= 1
-
- if(!secondary_type) //if we don't have a second type, we're done here
- return multiplier
- if(attack_type in secondary_type.immunity)
- return 0 //a single immunity negates all damage
- else if(attack_type in secondary_type.resistance)
- multiplier *= 0.5
- else if(attack_type in secondary_type.weakness)
- multiplier *= 2
- else
- multiplier *= 1
- return multiplier
-
-/datum/mob_hunt/proc/take_damage(raw_damage, datum/mob_type/attack_type)
- var/message = ""
- var/multiplier = calc_def_multiplier(attack_type)
- var/total_damage = raw_damage * multiplier
- if(!cur_health) //it's already downed, quit hitting it
- return null
- if(!total_damage)
- message += "The attack is completely ineffective! "
- else
- cur_health = max(cur_health - total_damage, 0)
- switch(multiplier)
- if(0)
- message += "The attack is completely ineffective! "
- if(0.25)
- message += "It's barely effective... "
- if(0.5)
- message += "It's not very effective... "
- if(2)
- message += "It's super effective! "
- if(4)
- message += "It's ultra effective! "
- if(99)
- message += pick("REKT! ", "DUNKED! ", "DEFENSE BREAK! ", "WOMBO-COMBO'D!")
- if(!cur_health)
- message += "[nickname ? nickname : mob_name] is downed!"
- return message
-
-/datum/mob_hunt/proc/get_raw_damage()
- return (level * attack_multiplier)
-
-/datum/mob_hunt/proc/get_attack_type()
- var/datum/mob_type/attack_type = primary_type
- if(secondary_type && prob(40))
- attack_type = secondary_type
- return attack_type
-
-/datum/mob_hunt/proc/gain_exp()
- exp++
- var/message = "[nickname ? nickname : mob_name] gained EXP! ([exp] / [exp_to_level] EXP)"
- if(exp >= exp_to_level)
- message = levelup()
- return message
-
-/datum/mob_hunt/proc/levelup()
- var/message = ""
- level++
- exp = 0
- if(level > max_level) //This is where we would trigger an evolution, when those are added (need to add evolved forms first)
- level = max_level //for now though, we'll just cap them back at their max_level
- message += "[nickname ? nickname : mob_name] can't get any stronger right now!"
- else
- max_health = base_health + (level * health_multiplier)
- cur_health = max_health //full heal on level-up
- message += "[nickname ? nickname : mob_name] has reached level [level]!"
- return message
-
-/datum/mob_hunt/proc/get_type1()
- if(!primary_type)
- return "Typeless"
- else
- return primary_type.name
-
-/datum/mob_hunt/proc/get_type2()
- if(!secondary_type)
- return null
- else
- return secondary_type.name
-
-
-/datum/mob_hunt/nemabug
- mob_name = "Nemabug"
- run_chance = 50
- min_level = 1
- max_level = 10
- primary_type = NANOMOB_TYPE_BUG
- icon_state_normal = "nemabug"
- icon_state_shiny = "nemabug_shiny"
- lifetime = 6000
-
-/datum/mob_hunt/stoutquill
- mob_name = "Stoutquill"
- run_chance = 50
- min_level = 5
- max_level = 15
- primary_type = NANOMOB_TYPE_ICE
- icon_state_normal = "stoutquill"
- icon_state_shiny = "stoutquill_shiny"
- lifetime = 4500
-
-/datum/mob_hunt/spectra
- mob_name = "Spectra"
- run_chance = 35
- min_level = 1
- max_level = 10
- primary_type = NANOMOB_TYPE_POISON
- icon_state_normal = "spectra"
- icon_state_shiny = "spectra_shiny"
- lifetime = 6000
-
-/datum/mob_hunt/dunny
- mob_name = "Dunny"
- run_chance = 35
- min_level = 1
- max_level = 10
- primary_type = NANOMOB_TYPE_FIRE
- icon_state_normal = "dunny"
- icon_state_shiny = "dunny_shiny"
- lifetime = 6000
-
-/datum/mob_hunt/buffsel
- mob_name = "Buffsel"
- run_chance = 35
- min_level = 1
- max_level = 10
- primary_type = NANOMOB_TYPE_ROCK
- icon_state_normal = "buffsel"
- icon_state_shiny = "buffsel_shiny"
- lifetime = 6000
-
-/datum/mob_hunt/quarrel
- mob_name = "Quarrel"
- run_chance = 35
- min_level = 1
- max_level = 10
- primary_type = NANOMOB_TYPE_NORMAL
- icon_state_normal = "quarrel"
- icon_state_shiny = "quarrel_shiny"
- lifetime = 6000
-
-/datum/mob_hunt/vulerrt
- mob_name = "Vulerrt"
- run_chance = 50
- min_level = 5
- max_level = 15
- primary_type = NANOMOB_TYPE_DARK
- icon_state_normal = "vulerrt"
- icon_state_shiny = "vulerrt_shiny"
- turf_whitelist = list()
- lifetime = 4500
-
-/datum/mob_hunt/strudel
- mob_name = "Strudel"
- run_chance = 35
- min_level = 1
- max_level = 10
- primary_type = NANOMOB_TYPE_ELECTRIC
- icon_state_normal = "strudel"
- icon_state_shiny = "strudel_shiny"
- lifetime = 4500
-
-/datum/mob_hunt/folstick
- mob_name = "Folstick"
- run_chance = 35
- min_level = 1
- max_level = 10
- primary_type = NANOMOB_TYPE_WATER
- icon_state_normal = "folstick"
- icon_state_shiny = "folstick_shiny"
- lifetime = 6000
-
-/datum/mob_hunt/glimmerflare
- mob_name = "Glimmerflare"
- run_chance = 50
- min_level = 5
- max_level = 15
- primary_type = NANOMOB_TYPE_PSYCHIC
- icon_state_normal = "glimmerflare"
- icon_state_shiny = "glimmerflare_shiny"
- lifetime = 4500
-
-/datum/mob_hunt/leecoon
- mob_name = "Leecoon"
- run_chance = 35
- min_level = 1
- max_level = 10
- primary_type = NANOMOB_TYPE_GRASS
- icon_state_normal = "leecoon"
- icon_state_shiny = "leecoon_shiny"
- lifetime = 6000
-
-/datum/mob_hunt/halk
- mob_name = "Halk"
- run_chance = 35
- min_level = 1
- max_level = 10
- primary_type = NANOMOB_TYPE_FLYING
- icon_state_normal = "halk"
- icon_state_shiny = "halk_shiny"
- lifetime = 6000
-
-/datum/mob_hunt/gooby
- mob_name = "Gooby"
- run_chance = 65
- min_level = 5
- max_level = 20
- primary_type = NANOMOB_TYPE_ELECTRIC
- secondary_type = NANOMOB_TYPE_BUG
- icon_state_normal = "gooby"
- icon_state_shiny = "gooby_shiny"
- lifetime = 3000
-
-/datum/mob_hunt/pandoom
- mob_name = "Pandoom"
- run_chance = 50
- min_level = 5
- max_level = 15
- primary_type = NANOMOB_TYPE_GHOST
- icon_state_normal = "pandoom"
- icon_state_shiny = "pandoom_shiny"
- lifetime = 4500
-
-/datum/mob_hunt/relish
- mob_name = "Relish"
- run_chance = 65
- min_level = 5
- max_level = 20
- primary_type = NANOMOB_TYPE_FIRE
- secondary_type = NANOMOB_TYPE_GROUND
- icon_state_normal = "relish"
- icon_state_shiny = "relish_shiny"
- lifetime = 3000
-
-/datum/mob_hunt/xofine
- mob_name = "Xofine"
- run_chance = 50
- min_level = 5
- max_level = 10
- primary_type = NANOMOB_TYPE_FIRE
- secondary_type = NANOMOB_TYPE_NORMAL
- icon_state_normal = "xofine"
- icon_state_shiny = "xofine_shiny"
- lifetime = 3000
-
-/datum/mob_hunt/gitten
- mob_name = "Gitten"
- run_chance = 65
- min_level = 5
- max_level = 20
- primary_type = NANOMOB_TYPE_WATER
- secondary_type = NANOMOB_TYPE_POISON
- icon_state_normal = "gitten"
- icon_state_shiny = "gitten_shiny"
- lifetime = 3000
-
-/datum/mob_hunt/nai
- mob_name = "Nai"
- run_chance = 65
- min_level = 5
- max_level = 20
- primary_type = NANOMOB_TYPE_GRASS
- secondary_type = NANOMOB_TYPE_NORMAL
- icon_state_normal = "nai"
- icon_state_shiny = "nai_shiny"
- lifetime = 3000
-
-/datum/mob_hunt/pyroghast
- mob_name = "Pyroghast"
- run_chance = 65
- min_level = 5
- max_level = 20
- primary_type = NANOMOB_TYPE_FIRE
- secondary_type = NANOMOB_TYPE_GHOST
- icon_state_normal = "pyroghast"
- icon_state_shiny = "pyroghast"
- lifetime = 4500
-
-/datum/mob_hunt/starslam
- mob_name = "Starslam"
- run_chance = 65
- min_level = 5
- max_level = 20
- primary_type = NANOMOB_TYPE_FIGHTING
- secondary_type = NANOMOB_TYPE_ICE
- icon_state_normal = "starslam"
- icon_state_shiny = "starslam_shiny"
- lifetime = 2500
-
-/datum/mob_hunt/pheron
- mob_name = "Pheron"
- run_chance = 85
- min_level = 10
- max_level = 20
- primary_type = NANOMOB_TYPE_BLUESPACE
- icon_state_normal = "pheron"
- icon_state_shiny = "pheron_shiny"
- area_blacklist = list()
- turf_blacklist = list()
- area_whitelist = list()
- turf_whitelist = list()
- lifetime = 2000
diff --git a/code/modules/arcade/mob_hunt/mob_type_datums.dm b/code/modules/arcade/mob_hunt/mob_type_datums.dm
deleted file mode 100644
index 1789ac679126..000000000000
--- a/code/modules/arcade/mob_hunt/mob_type_datums.dm
+++ /dev/null
@@ -1,241 +0,0 @@
-
-/datum/mob_type
- var/name = "Typeless"
- var/list/weakness = list()
- var/list/resistance = list()
- var/list/immunity = list()
-
- //Type-based spawn preferences (to eliminate copy-pasting the same area and turf lists on each mob type)
- var/list/area_blacklist = list() //areas to be avoided
- var/list/area_whitelist = list() //areas to be more preferred
- var/list/turf_blacklist = list() //turf types to be avoided
- var/list/turf_whitelist = list() //turf types to be more preferred
-
-//Type defines, to avoid spelling mistakes
-/datum/mob_type/fire
- name = "Fire"
- weakness = list(NANOMOB_TYPE_WATER,
- NANOMOB_TYPE_ROCK,
- NANOMOB_TYPE_GROUND)
- resistance = list(NANOMOB_TYPE_BUG,
- NANOMOB_TYPE_FIRE,
- NANOMOB_TYPE_GRASS,
- NANOMOB_TYPE_ICE,
- NANOMOB_TYPE_STEEL)
- area_blacklist = list(/area/station/public/toilet,
- /area/station/science/server/coldroom)
- area_whitelist = list(/area/station/maintenance/turbine,
- /area/station/maintenance/incinerator,
- /area/station/service/kitchen)
- turf_blacklist = list(/turf/simulated/floor/beach/water)
-
-/datum/mob_type/water
- name = "Water"
- weakness = list(NANOMOB_TYPE_ELECTRIC,
- NANOMOB_TYPE_GRASS)
- resistance = list(NANOMOB_TYPE_FIRE,
- NANOMOB_TYPE_ICE,
- NANOMOB_TYPE_STEEL,
- NANOMOB_TYPE_WATER)
- area_blacklist = list(/area/station/maintenance/turbine,
- /area/station/maintenance/incinerator,
- /area/station/service/kitchen)
- area_whitelist = list(/area/station/public/toilet)
- turf_whitelist = list(/turf/simulated/floor/beach/water)
-
-/datum/mob_type/grass
- name = "Grass"
- weakness = list(NANOMOB_TYPE_FIRE,
- NANOMOB_TYPE_BUG,
- NANOMOB_TYPE_POISON,
- NANOMOB_TYPE_ICE,
- NANOMOB_TYPE_FLYING)
- resistance = list(NANOMOB_TYPE_WATER,
- NANOMOB_TYPE_GRASS,
- NANOMOB_TYPE_ELECTRIC,
- NANOMOB_TYPE_GROUND)
- area_blacklist = list(/area/station/science)
- area_whitelist = list(/area/station/service/hydroponics,
- /area/station/public/construction) // dont ask me why grass types can appear in construction sites, I'm just migrating areas
- turf_whitelist = list(/turf/simulated/floor/grass)
-
-/datum/mob_type/electric
- name = "Electric"
- weakness = list(NANOMOB_TYPE_GROUND)
- resistance = list(NANOMOB_TYPE_ELECTRIC,
- NANOMOB_TYPE_FLYING,
- NANOMOB_TYPE_STEEL)
- area_whitelist = list(/area/station/engineering/engine,
- /area/station/science/server,
- /area/station/maintenance,
- /area/station/turret_protected/ai,
- /area/station/turret_protected/ai_upload,
- /area/station/turret_protected/aisat/interior,
- /area/station/aisat,
- /area/station/science/robotics)
- turf_whitelist = list(/turf/simulated/floor/bluegrid)
-
-/datum/mob_type/ground
- name = "Ground"
- weakness = list(NANOMOB_TYPE_WATER,
- NANOMOB_TYPE_GRASS,
- NANOMOB_TYPE_ICE)
- resistance = list(NANOMOB_TYPE_ROCK,
- NANOMOB_TYPE_POISON)
- immunity = list(NANOMOB_TYPE_ELECTRIC)
-
-/datum/mob_type/rock
- name = "Rock"
- weakness = list(NANOMOB_TYPE_WATER,
- NANOMOB_TYPE_GRASS,
- NANOMOB_TYPE_GROUND,
- NANOMOB_TYPE_FIGHTING,
- NANOMOB_TYPE_STEEL)
- resistance = list(NANOMOB_TYPE_FIRE,
- NANOMOB_TYPE_FLYING,
- NANOMOB_TYPE_POISON,
- NANOMOB_TYPE_NORMAL)
- area_whitelist = list(/area/station/supply,
- /area/station/maintenance/disposal)
- turf_whitelist = list(/turf/simulated/wall,
- /turf/simulated/floor/mineral)
-
-/datum/mob_type/bug
- name = "Bug"
- weakness = list(NANOMOB_TYPE_FIRE,
- NANOMOB_TYPE_ROCK,
- NANOMOB_TYPE_FLYING)
- resistance = list(NANOMOB_TYPE_GRASS,
- NANOMOB_TYPE_GROUND,
- NANOMOB_TYPE_FIGHTING)
- area_blacklist = list(/area/station/science)
- area_whitelist = list(/area/station/service/hydroponics,
- /area/station/public/construction) // dont ask me why bug types can appear in construction sites, I'm just migrating areas
- turf_whitelist = list(/turf/simulated/floor/grass)
-
-/datum/mob_type/poison
- name = "Poison"
- weakness = list(NANOMOB_TYPE_GROUND,
- NANOMOB_TYPE_PSYCHIC)
- resistance = list(NANOMOB_TYPE_GRASS,
- NANOMOB_TYPE_BUG,
- NANOMOB_TYPE_POISON,
- NANOMOB_TYPE_FIGHTING)
- area_blacklist = list(/area/station/medical,
- /area/station/service/janitor)
- area_whitelist = list(/area/station/medical/virology,
- /area/station/science,
- /area/station/command/office/rd,
- /area/station/maintenance/asmaint2)
-
-/datum/mob_type/normal
- name = "Normal"
- weakness = list(NANOMOB_TYPE_FIGHTING)
- immunity = list(NANOMOB_TYPE_GHOST)
-
-/datum/mob_type/fighting
- name = "Fighting"
- weakness = list(NANOMOB_TYPE_PSYCHIC,
- NANOMOB_TYPE_FLYING)
- resistance = list(NANOMOB_TYPE_ROCK,
- NANOMOB_TYPE_BUG,
- NANOMOB_TYPE_DARK)
- area_blacklist = list(/area/station/medical)
- area_whitelist = list(/area/station/service/bar,
- /area/station/public/fitness,
- /area/station/security)
- turf_whitelist = list(/turf/simulated/floor/wood)
-
-/datum/mob_type/psychic
- name = "Psychic"
- weakness = list(NANOMOB_TYPE_BUG,
- NANOMOB_TYPE_GHOST,
- NANOMOB_TYPE_DARK)
- resistance = list(NANOMOB_TYPE_FIGHTING,
- NANOMOB_TYPE_PSYCHIC)
- area_blacklist = list(/area/station/science,
- /area/station/command/office/rd,
- /area/station/maintenance/asmaint2,
- /area/station/command/teleporter,
- /area/station/command/gateway)
- area_whitelist = list(/area/station/service/library,
- /area/station/service/chapel,
- /area/station/medical/psych)
-
-/datum/mob_type/ghost
- name = "Ghost"
- weakness = list(NANOMOB_TYPE_GHOST,
- NANOMOB_TYPE_DARK)
- resistance = list(NANOMOB_TYPE_BUG,
- NANOMOB_TYPE_POISON)
- immunity = list(NANOMOB_TYPE_NORMAL,
- NANOMOB_TYPE_FIGHTING)
- area_whitelist = list(/area/station/medical/morgue,
- /area/station/service/chapel,
- /area/station/medical/cloning)
-
-/datum/mob_type/ice
- name = "Ice"
- weakness = list(NANOMOB_TYPE_FIRE,
- NANOMOB_TYPE_ROCK,
- NANOMOB_TYPE_FIGHTING,
- NANOMOB_TYPE_STEEL)
- resistance = list(NANOMOB_TYPE_ICE)
- area_blacklist = list(/area/station/maintenance/turbine,
- /area/station/maintenance/incinerator,
- /area/station/service/kitchen)
- area_whitelist = list(/area/station/science/server/coldroom)
-
-/datum/mob_type/flying
- name = "Flying"
- weakness = list(NANOMOB_TYPE_ELECTRIC,
- NANOMOB_TYPE_ROCK,
- NANOMOB_TYPE_ICE)
- resistance = list(NANOMOB_TYPE_GRASS,
- NANOMOB_TYPE_BUG,
- NANOMOB_TYPE_FIGHTING)
- immunity = list(NANOMOB_TYPE_GROUND)
- area_blacklist = list(/area/station/maintenance)
- area_whitelist = list(/area/station/hallway)
-
-/datum/mob_type/bluespace
- name = "Bluespace"
- weakness = list(NANOMOB_TYPE_ICE,
- NANOMOB_TYPE_BLUESPACE)
- resistance = list(NANOMOB_TYPE_FIRE,
- NANOMOB_TYPE_WATER,
- NANOMOB_TYPE_GRASS,
- NANOMOB_TYPE_ELECTRIC)
-
-/datum/mob_type/dark
- name = "Dark"
- weakness = list(NANOMOB_TYPE_BUG,
- NANOMOB_TYPE_FIGHTING)
- resistance = list(NANOMOB_TYPE_GHOST,
- NANOMOB_TYPE_DARK)
- immunity = list(NANOMOB_TYPE_PSYCHIC)
- area_blacklist = list(/area/station/engineering/solar,
- /area/station/maintenance/auxsolarport,
- /area/station/maintenance/starboardsolar,
- /area/station/maintenance/portsolar,
- /area/station/maintenance/auxsolarstarboard,
- /area/station/service/clown)
- area_whitelist = list(/area/station/maintenance,
- /area/station/service/mime)
- turf_blacklist = list(/turf/simulated/floor/light)
-
-/datum/mob_type/steel
- name = "Steel"
- weakness = list(NANOMOB_TYPE_FIRE,
- NANOMOB_TYPE_GROUND,
- NANOMOB_TYPE_FIGHTING)
- resistance = list(NANOMOB_TYPE_GRASS,
- NANOMOB_TYPE_ROCK,
- NANOMOB_TYPE_BUG,
- NANOMOB_TYPE_NORMAL,
- NANOMOB_TYPE_PSYCHIC,
- NANOMOB_TYPE_ICE,
- NANOMOB_TYPE_FLYING,
- NANOMOB_TYPE_BLUESPACE,
- NANOMOB_TYPE_STEEL)
- immunity = list(NANOMOB_TYPE_POISON)
diff --git a/code/modules/arcade/prize_datums.dm b/code/modules/arcade/prize_datums.dm
index 40d8f4d54e30..7e9481bceee8 100644
--- a/code/modules/arcade/prize_datums.dm
+++ b/code/modules/arcade/prize_datums.dm
@@ -275,12 +275,6 @@ GLOBAL_DATUM_INIT(global_prizes, /datum/prizes, new())
typepath = /obj/item/clothing/under/syndicate/tacticool
cost = 90
-/datum/prize_item/nanomob_booster
- name = "Nano-Mob Hunter Trading Card Booster Pack"
- desc = "Contains 6 random Nano-Mob Hunter Trading Cards. May contain a holographic card!"
- typepath = /obj/item/storage/box/nanomob_booster_pack
- cost = 100
-
/datum/prize_item/fakespell
name = "Fake Spellbook"
desc = "Perform magic! Astound your friends! Get mistaken for an enemy of the corporation!"
diff --git a/code/modules/asset_cache/assets/asset_mob_hunt.dm b/code/modules/asset_cache/assets/asset_mob_hunt.dm
deleted file mode 100644
index ed1f7494e332..000000000000
--- a/code/modules/asset_cache/assets/asset_mob_hunt.dm
+++ /dev/null
@@ -1,7 +0,0 @@
-/datum/asset/simple/mob_hunt/register()
- for(var/state in icon_states('icons/effects/mob_hunt.dmi'))
- if(state == "Placeholder")
- continue
- assets["[state].png"] = icon('icons/effects/mob_hunt.dmi', state)
-
- return ..()
diff --git a/code/modules/client/preference/loadout/loadout_general.dm b/code/modules/client/preference/loadout/loadout_general.dm
index 2549eff6d9c1..e243d1318742 100644
--- a/code/modules/client/preference/loadout/loadout_general.dm
+++ b/code/modules/client/preference/loadout/loadout_general.dm
@@ -162,11 +162,6 @@
path = /obj/item/paicard
cost = 2
-/datum/gear/mob_hunt_game
- display_name = "Nano-Mob Hunter GO! Cartridge"
- path = /obj/item/cartridge/mob_hunt_game
- cost = 2
-
//////////////////////
// Mugs //
//////////////////////
diff --git a/code/modules/economy/merch_items.dm b/code/modules/economy/merch_items.dm
index 4b2c97487d5a..b15a0eb6b303 100644
--- a/code/modules/economy/merch_items.dm
+++ b/code/modules/economy/merch_items.dm
@@ -47,13 +47,6 @@
cost = 100
category = MERCH_CAT_TOY
-/datum/merch_item/nanomob_booster
- name = "Nano-Mob Hunter Trading Card Booster Pack"
- desc = "Contains 6 random Nano-Mob Hunter Trading Cards. May contain a holographic card!"
- typepath = /obj/item/storage/box/nanomob_booster_pack
- cost = 125
- category = MERCH_CAT_TOY
-
/datum/merch_item/crayons
name = "Crayons"
desc = "Let security know how they're doing by scrawling love notes all over their hallways."
diff --git a/code/modules/pda/PDA.dm b/code/modules/pda/PDA.dm
index 231dafbef73b..e21c00727b2e 100644
--- a/code/modules/pda/PDA.dm
+++ b/code/modules/pda/PDA.dm
@@ -312,9 +312,6 @@ GLOBAL_LIST_EMPTY(PDAs)
add_pen(C)
to_chat(user, "You slide \the [C] into \the [src].")
playsound(src, 'sound/machines/pda_button1.ogg', 50, TRUE)
- else if(istype(C, /obj/item/nanomob_card))
- if(cartridge && istype(cartridge, /obj/item/cartridge/mob_hunt_game))
- cartridge.attackby(C, user, params)
/obj/item/pda/proc/add_pen(obj/item/P)
P.forceMove(src)
diff --git a/code/modules/pda/cart.dm b/code/modules/pda/cart.dm
index 4a66081f9563..75deb8f685da 100644
--- a/code/modules/pda/cart.dm
+++ b/code/modules/pda/cart.dm
@@ -306,40 +306,3 @@
messenger_plugins = list(
new /datum/data/pda/messenger_plugin/virus/frame
)
-
-/obj/item/cartridge/mob_hunt_game
- name = "Nano-Mob Hunter GO! Cartridge"
- desc = "The hit new PDA game that lets you track down and capture your favorite Nano-Mobs living in your world!"
- icon_state = "cart-eye"
- programs = list(
- new /datum/data/pda/app/mob_hunter_game
- )
-
-/obj/item/cartridge/mob_hunt_game/examine(mob/user)
- . = ..()
- if(emagged)
- if(isAntag(user))
- . += "This copy of Nano-Mob Hunter GO! has been hacked to allow the creation of trap mobs which will cause any PDA that attempts to capture it to shock anyone holding it. Hacked copies of the game will not trigger the trap. Provided you actually find someone playing nanomobs, that is."
-
-/obj/item/cartridge/mob_hunt_game/attackby(obj/item/O, mob/user, params)
- if(istype(O, /obj/item/nanomob_card))
- var/obj/item/nanomob_card/card = O
- var/datum/data/pda/app/mob_hunter_game/my_game = programs[1]
-
- if(my_game.register_capture(card.mob_data))
- to_chat(user, "Transfer successful!")
- qdel(card)
- else
- to_chat(user, "Transfer failed. Could not read mob data from card.")
-
- else
- ..()
-
-/obj/item/cartridge/mob_hunt_game/emag_act(mob/user)
- if(!emagged)
- emagged = TRUE
- var/datum/data/pda/app/mob_hunter_game/my_game = programs[1]
- my_game.hacked = TRUE
- to_chat(user, "TR4P_M45T3R.mod successfully initialized. ToS violated. User Agreement nullified. Gotta pwn them all.")
- to_chat(user, "You can now create trapped versions of any mob in your collection that will damage hunters who attempt to capture it.")
- return TRUE
diff --git a/code/modules/pda/mob_hunt_game_app.dm b/code/modules/pda/mob_hunt_game_app.dm
deleted file mode 100644
index 73eab8cf1a52..000000000000
--- a/code/modules/pda/mob_hunt_game_app.dm
+++ /dev/null
@@ -1,205 +0,0 @@
-
-/*
- This stuff isn't included with in core_apps.dm because it is so distinct and would end up making the file huge (more than it already is).
- I put it in it's own file in order to help it stay organized and easier to locate the procs and such.
- For reference, the other Nano-mob Hunter GO files are in /code/game/modules/arcade/mob_hunt
-*/
-
-/datum/data/pda/app/mob_hunter_game
- name = "Nano-Mob Hunter GO"
- icon = "gamepad"
- template = "pda_mob_hunt"
- category = "Games"
-
- var/list/my_collection = list()
- var/current_index = 0
- var/connected = FALSE
- var/hacked = FALSE //if set, this cartridge is able to spawn trap mobs from its collection (set via emag_act on the cartridge)
- var/catch_mod = 0 //used to adjust the likelihood of a mob running from this client, a negative value means it is less likely to run (support for temporary bonuses)
- var/wild_captures = 0 //used to track the total number of mobs captured from the wild (does not count card mobs) by this client
- var/scan_range = 3 //maximum distance (in tiles) from which the client can reveal nearby mobs
-
-/datum/data/pda/app/mob_hunter_game/start()
- ..()
- START_PROCESSING(SSobj, pda)
-
-/datum/data/pda/app/mob_hunter_game/stop()
- ..()
- disconnect("Program Terminated")
- STOP_PROCESSING(SSobj, pda)
-
-/datum/data/pda/app/mob_hunter_game/Destroy()
- if(pda)
- STOP_PROCESSING(SSobj, pda)
- SSmob_hunt.connected_clients -= src
- return ..()
-
-/datum/data/pda/app/mob_hunter_game/proc/scan_nearby()
- if(!SSmob_hunt || !connected)
- return
- for(var/turf/T in range(scan_range, get_turf(pda)))
- for(var/obj/effect/nanomob/N in T.contents)
- if(src in N.clients_encountered)
- //hide the mob
- N.conceal()
- else
- //reveal the mob
- N.reveal()
-
-/datum/data/pda/app/mob_hunter_game/proc/reconnect()
- if(!SSmob_hunt || !SSmob_hunt.server_status || connected)
- //show a message about the server being unavailable (because it doesn't exist / didn't get set to the global var / is offline)
- return 0
- SSmob_hunt.connected_clients += src
- connected = TRUE
- if(pda)
- pda.atom_say("Connection established. Capture all of the mobs, [pda.owner ? pda.owner : "hunter"]!")
- return 1
-
-/datum/data/pda/app/mob_hunter_game/proc/get_player()
- if(!pda)
- return null
- if(ishuman(pda.loc))
- var/mob/living/carbon/human/H = pda.loc
- return H
- return null
-
-/datum/data/pda/app/mob_hunter_game/proc/disconnect(reason = null)
- if(!SSmob_hunt || !connected)
- return
- SSmob_hunt.connected_clients -= src
- for(var/obj/effect/nanomob/N in (SSmob_hunt.normal_spawns + SSmob_hunt.trap_spawns))
- N.conceal(list(get_player()))
- connected = FALSE
- //show a disconnect message if we were disconnected involuntarily (reason argument provided)
- if(pda && reason)
- pda.atom_say("Disconnected from server. Reason: [reason].")
-
-/datum/data/pda/app/mob_hunter_game/program_process()
- if(!SSmob_hunt || !connected)
- return
- scan_nearby()
-
-/datum/data/pda/app/mob_hunter_game/proc/register_capture(datum/mob_hunt/captured, wild = FALSE)
- if(!captured)
- return FALSE
- my_collection += captured
- RegisterSignal(captured, COMSIG_PARENT_QDELETING, PROC_REF(remove_mob))
- if(wild)
- wild_captures++
- return TRUE
-
-/datum/data/pda/app/mob_hunter_game/update_ui(mob/user, list/data)
- if(!SSmob_hunt || !(src in SSmob_hunt.connected_clients))
- data["connected"] = 0
- else
- data["connected"] = 1
- data["wild_captures"] = wild_captures
- data["no_collection"] = 0
- if(!length(my_collection))
- data["no_collection"] = 1
- return
- var/datum/mob_hunt/mob_info
- if(!current_index)
- current_index = 1
- if(current_index > length(my_collection))
- current_index = 1
- if(current_index < 1)
- current_index = length(my_collection)
- mob_info = my_collection[current_index]
- var/list/entry = list(
- "nickname" = mob_info.nickname,
- "real_name" = mob_info.mob_name,
- "level" = mob_info.level,
- "type1" = mob_info.get_type1(),
- "type2" = mob_info.get_type2(),
- "sprite" = "[mob_info.icon_state_normal].png",
- "is_hacked" = hacked
- )
- if(mob_info.is_shiny)
- entry["sprite"] = "[mob_info.icon_state_shiny].png"
- data["entry"] = entry
-
-/datum/data/pda/app/mob_hunter_game/proc/assign_nickname()
- if(!length(my_collection))
- return
- var/datum/mob_hunt/mob_info = my_collection[current_index]
- var/old_name = mob_info.mob_name
- if(mob_info.nickname)
- old_name = mob_info.nickname
- mob_info.nickname = tgui_input_text(usr, "Give a nickname to [old_name]?", "Nickname", old_name)
-
-/datum/data/pda/app/mob_hunter_game/proc/release()
- if(!length(my_collection))
- return
- if(tgui_alert(usr, "Are you sure you want to release this mob back into the wild?", "Confirm Release", list("Yes", "No")) == "Yes")
- remove_mob()
-
-/datum/data/pda/app/mob_hunter_game/proc/print_card()
- if(!pda || !length(my_collection))
- return
- var/obj/item/nanomob_card/card = new/obj/item/nanomob_card(null)
- var/datum/mob_hunt/mob_info = my_collection[current_index]
- card.mob_data = mob_info
- card.update_info()
- card.forceMove(get_turf(pda))
- remove_mob()
-
-/**
- * Removes a Nanomob from the [my_collection] list.
- *
- * The Nanomob that is currently selected in the app ([current_index]) will be removed from the list unless a `mob_override` argument is given, in which case that will be removed instead.
- *
- * Arguments:
- * * mob_override - A specific Nanomob to remove from the list. (Optional)
- */
-/datum/data/pda/app/mob_hunter_game/proc/remove_mob(datum/mob_hunt/mob_override = null)
- SIGNAL_HANDLER
- if(!length(my_collection))
- return
-
- if(mob_override)
- my_collection -= mob_override
- else
- my_collection -= my_collection[current_index]
-
- var/collection_length = length(my_collection)
- if(current_index > collection_length)
- current_index = collection_length
-
-/datum/data/pda/app/mob_hunter_game/proc/set_trap()
- if(!length(my_collection) || !pda || !hacked)
- return
- var/datum/mob_hunt/bait = my_collection[current_index]
- bait = bait.type
- new bait(1, get_turf(pda))
-
-/datum/data/pda/app/mob_hunter_game/ui_act(action, list/params, datum/tgui/ui, datum/ui_state/state)
- if(..())
- return
-
- if(!pda.silent)
- playsound(pda, 'sound/machines/terminal_select.ogg', 15, TRUE)
-
- . = TRUE
- switch(action)
- if("Rename")
- assign_nickname()
- if("Release")
- release()
- if("Next")
- current_index++
- if(current_index > length(my_collection))
- current_index = 1
- if("Prev")
- current_index--
- if(current_index < 1)
- current_index = length(my_collection)
- if("Reconnect")
- reconnect()
- if("Disconnect")
- disconnect()
- if("Transfer")
- print_card()
- if("Set_Trap")
- set_trap()
diff --git a/code/modules/pda/pda_tgui.dm b/code/modules/pda/pda_tgui.dm
index 27310410d31e..96c1a8f52961 100644
--- a/code/modules/pda/pda_tgui.dm
+++ b/code/modules/pda/pda_tgui.dm
@@ -60,11 +60,6 @@
return data
-/obj/item/pda/ui_assets(mob/user)
- return list(
- get_asset_datum(/datum/asset/simple/mob_hunt)
- )
-
// Yes the stupid amount of args here is important, see L102
/obj/item/pda/ui_act(action, list/params, datum/tgui/ui, datum/ui_state/state)
if(..())
diff --git a/icons/effects/mob_hunt.dmi b/icons/effects/mob_hunt.dmi
deleted file mode 100644
index 1eea83a9afa4406f7f1a46d966e201887e83ee48..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001
literal 51165
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%0DwmgUNiyp9ZR3-h(DBR|>nsf$BcMgJG
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