From ad1e0ac7ece26a902b99449b83941000af1ae919 Mon Sep 17 00:00:00 2001 From: Casper Date: Mon, 8 Jul 2024 10:02:07 +0100 Subject: [PATCH 1/7] Checkpoint --- code/modules/admin/admin_verbs.dm | 2 +- code/modules/mob/mob.dm | 7 + code/modules/mob/mob_defines.dm | 3 + modular_zzplurt/code/modules/admin/ban.dm | 0 modular_zzplurt/code/modules/admin/jobban.dm | 179 +++ modular_zzplurt/code/modules/admin/kick.dm | 20 + .../code/modules/admin/player_panel.dm | 414 ++++++ .../code/modules/admin/transform.dm | 127 ++ modular_zzplurt/code/modules/admin/verbs.dm | 7 + .../code/modules/client/client_procs.dm | 60 + .../code/modules/mob/living/living.dm | 27 + .../code/modules/mob/living/living_defines.dm | 4 + tgstation.dme | 9 + tgui/packages/tgui/components/Collapsible.tsx | 6 +- tgui/packages/tgui/interfaces/PlayerPanel.tsx | 1109 +++++++++++++++++ 15 files changed, 1971 insertions(+), 3 deletions(-) create mode 100644 modular_zzplurt/code/modules/admin/ban.dm create mode 100644 modular_zzplurt/code/modules/admin/jobban.dm create mode 100644 modular_zzplurt/code/modules/admin/kick.dm create mode 100644 modular_zzplurt/code/modules/admin/player_panel.dm create mode 100644 modular_zzplurt/code/modules/admin/transform.dm create mode 100644 modular_zzplurt/code/modules/admin/verbs.dm create mode 100644 modular_zzplurt/code/modules/client/client_procs.dm create mode 100644 modular_zzplurt/code/modules/mob/living/living.dm create mode 100644 modular_zzplurt/code/modules/mob/living/living_defines.dm create mode 100644 tgui/packages/tgui/interfaces/PlayerPanel.tsx diff --git a/code/modules/admin/admin_verbs.dm b/code/modules/admin/admin_verbs.dm index 0f7d7a3f39b30..4a9bb413e3f5f 100644 --- a/code/modules/admin/admin_verbs.dm +++ b/code/modules/admin/admin_verbs.dm @@ -341,7 +341,7 @@ ADMIN_VERB(remove_mob_action, R_FUN, "Remove Mob Action", ADMIN_VERB_NO_DESCRIPT if(!length(target_abilities)) return - var/chosen_ability = tgui_input_list(user, "Choose the spell to remove from [removal_target]", "Depower", sort_list(target_abilities)) + var/chosen_ability = tgui_input_list(user, "Choose the action to remove from [removal_target]", "Depower", sort_list(target_abilities)) if(isnull(chosen_ability)) return var/datum/action/cooldown/mob_cooldown/to_remove = target_abilities[chosen_ability] diff --git a/code/modules/mob/mob.dm b/code/modules/mob/mob.dm index 0922ac6ae51eb..37cb1af25238a 100644 --- a/code/modules/mob/mob.dm +++ b/code/modules/mob/mob.dm @@ -1,3 +1,8 @@ +/mob/proc/create_player_panel() + QDEL_NULL(mob_panel) + + mob_panel = new(src) + /** * Delete a mob * @@ -44,6 +49,7 @@ observe.reset_perspective(null) qdel(hud_used) + QDEL_NULL(mob_panel) QDEL_LIST(client_colours) ghostize(can_reenter_corpse = FALSE) //False, since we're deleting it currently if(mind?.current == src) //Let's just be safe yeah? This will occasionally be cleared, but not always. Can't do it with ghostize without changing behavior @@ -95,6 +101,7 @@ update_config_movespeed() initialize_actionspeed() update_movespeed(TRUE) + create_player_panel() become_hearing_sensitive() log_mob_tag("TAG: [tag] CREATED: [key_name(src)] \[[type]\]") diff --git a/code/modules/mob/mob_defines.dm b/code/modules/mob/mob_defines.dm index 2206efd0e13ce..3f95defe17a85 100644 --- a/code/modules/mob/mob_defines.dm +++ b/code/modules/mob/mob_defines.dm @@ -171,6 +171,9 @@ var/list/client_colours = list() var/hud_type = /datum/hud + // Admin player panel for this mob + var/datum/player_panel/mob_panel + var/datum/focus //What receives our keyboard inputs. src by default /// Used for tracking last uses of emotes for cooldown purposes diff --git a/modular_zzplurt/code/modules/admin/ban.dm b/modular_zzplurt/code/modules/admin/ban.dm new file mode 100644 index 0000000000000..e69de29bb2d1d diff --git a/modular_zzplurt/code/modules/admin/jobban.dm b/modular_zzplurt/code/modules/admin/jobban.dm new file mode 100644 index 0000000000000..c9d3577225c46 --- /dev/null +++ b/modular_zzplurt/code/modules/admin/jobban.dm @@ -0,0 +1,179 @@ +// // Stuff that helps the TGUI player panel jobban section to work + +// GLOBAL_LIST_INIT(jobban_panel_data, list( +// list( +// "name" = "Command", +// "color" = "yellow", +// "roles" = GLOB.command_positions +// ), +// list( +// "name" = "Security", +// "color" = "red", +// "roles" = GLOB.security_positions +// ), +// list( +// "name" = "Engineering", +// "color" = "orange", +// "roles" = GLOB.engineering_positions +// ), +// list( +// "name" = "Medical", +// "color" = "blue", +// "roles" = GLOB.medical_positions +// ), +// list( +// "name" = "Science", +// "color" = "violet", +// "roles" = GLOB.science_positions +// ), +// list( +// "name" = "Supply", +// "color" = "brown", +// "roles" = GLOB.supply_positions +// ), +// list( +// "name" = "Service", +// "color" = "green", +// "roles" = GLOB.civilian_positions +// ), +// list( +// "name" = "Silicon", +// "color" = "purple", +// "roles" = GLOB.nonhuman_positions +// ), +// list( +// "name" = "Ghost Roles", +// "color" = "teal", +// "roles" = list( +// ROLE_PAI, +// ROLE_POSIBRAIN, +// ROLE_DRONE, +// ROLE_DEATHSQUAD, +// ROLE_LAVALAND, +// ROLE_GHOSTCAFE, +// ROLE_SENTIENCE, +// ROLE_MIND_TRANSFER +// ) +// ), +// list( +// "name" = "Antagonists", +// "color" = "red", +// "roles" = list( +// ROLE_TRAITOR, +// ROLE_CHANGELING, +// ROLE_HERETIC, +// ROLE_OPERATIVE, +// ROLE_REV, +// ROLE_CULTIST, +// ROLE_SERVANT_OF_RATVAR, +// ROLE_WIZARD, +// ROLE_ABDUCTOR, +// ROLE_ALIEN, +// ROLE_FAMILIES, +// ROLE_BLOODSUCKER, +// ROLE_SLAVER +// ) +// ), +// list( +// "name" = "Other", +// "color" = "red", +// "roles" = list( +// "pacifist", +// "appearance", +// "emote", +// "OOC", +// ROLE_RESPAWN +// ) +// ) +// )) + +// // notbannedlist is just a list of strings of the job titles you want to ban. +// /datum/admins/proc/Jobban(mob/M, list/notbannedlist) +// if (!check_rights(R_BAN)) +// to_chat(usr, "Error: You do not have sufficient admin rights to ban players.") +// return + +// var/severity = null +// var/reason = null + +// switch(tgui_alert(usr, "Job ban type", buttons = list("Temporary", "Permanent", "Cancel"))) +// if("Temporary") +// var/mins = input(usr,"How long (in minutes)?","Ban time",1440) as num|null +// if(mins <= 0) +// to_chat(usr, span_danger("[mins] is not a valid duration.")) +// return +// reason = input(usr,"Please State Reason For Banning [M.key].","Reason") as message|null +// if(!reason) +// return +// severity = tgui_alert(usr, "Set the severity of the note/ban", buttons = list("High", "Medium", "Minor", "None")) +// if(!severity) +// return +// var/msg +// for(var/job in notbannedlist) +// if(!DB_ban_record(BANTYPE_JOB_TEMP, M, mins, reason, job)) +// to_chat(usr, span_danger("Failed to apply ban.")) +// return +// if(M.client) +// jobban_buildcache(M.client) +// ban_unban_log_save("[key_name(usr)] temp-jobbanned [key_name(M)] from [job] for [mins] minutes. reason: [reason]") +// log_admin_private("[key_name(usr)] temp-jobbanned [key_name(M)] from [job] for [mins] minutes.") +// if(!msg) +// msg = job +// else +// msg += ", [job]" +// create_message("note", M.key, null, "Banned from [msg] - [reason]", null, null, 0, 0, null, 0, severity) +// message_admins(span_adminnotice("[key_name_admin(usr)] banned [key_name_admin(M)] from [msg] for [mins] minutes.")) +// to_chat(M, span_boldannounce("You have been [((msg == "ooc") || (msg == "appearance") || (msg == "pacifist")) ? "banned" : "jobbanned"] by [usr.client.key] from: [msg == "pacifist" ? "using violence" : msg].")) +// to_chat(M, span_boldannounce("The reason is: [reason]")) +// to_chat(M, span_danger("This jobban will be lifted in [mins] minutes.")) + +// if("Permanent") +// reason = input(usr,"Please State Reason For Banning [M.key].","Reason") as message|null +// if (!reason) +// return +// severity = tgui_alert(usr, "Please State Reason For Banning", buttons = list("High", "Medium", "Minor", "None")) +// if (!severity) +// return + +// var/msg +// for(var/job in notbannedlist) +// if(!DB_ban_record(BANTYPE_JOB_PERMA, M, -1, reason, job)) +// to_chat(usr, span_danger("Failed to apply ban.")) +// return +// if(M.client) +// jobban_buildcache(M.client) +// ban_unban_log_save("[key_name(usr)] perma-jobbanned [key_name(M)] from [job]. reason: [reason]") +// log_admin_private("[key_name(usr)] perma-banned [key_name(M)] from [job]") +// if(!msg) +// msg = job +// else +// msg += ", [job]" +// create_message("note", M.key, null, "Banned from [msg] - [reason]", null, null, 0, 0, null, 0, severity) +// message_admins(span_adminnotice("[key_name_admin(usr)] banned [key_name_admin(M)] from [msg].")) +// to_chat(M, span_boldannounce("You have been [((msg == "ooc") || (msg == "appearance") || (msg == "pacifist")) ? "banned" : "jobbanned"] by [usr.client.key] from: [msg == "pacifist" ? "using violence" : msg].")) +// to_chat(M, span_boldannounce("The reason is: [reason]")) +// to_chat(M, span_danger("This jobban can be lifted only upon request.")) + +// // notbannedlist is just a list of strings of the job titles you want to unban. +// /datum/admins/proc/UnJobban(mob/M, list/bannedlist) + +// if (!check_rights(R_BAN)) +// to_chat(usr, "Error: You do not have sufficient admin rights to unban players.") +// return + +// var/msg +// for(var/job in bannedlist) +// var/reason = jobban_isbanned(M, job) +// if (tgui_alert(usr, "Job: '[job]' Ban reason: '[reason]'", "Un-Jobban This Player?", list("Yes", "No")) == "Yes") +// ban_unban_log_save("[key_name(usr)] unjobbanned [key_name(M)] from [job]") +// log_admin_private("[key_name(usr)] unbanned [key_name(M)] from [job]") +// DB_ban_unban(M.ckey, BANTYPE_ANY_JOB, job) +// if(M.client) +// jobban_buildcache(M.client) +// if(!msg) +// msg = job +// else +// msg += ", [job]" +// if(msg) +// message_admins(span_adminnotice("[key_name_admin(usr)] unbanned [key_name_admin(M)] from [msg].")) +// to_chat(M, span_boldannounce("You have been un-jobbanned by [usr.client.key] from [msg].")) diff --git a/modular_zzplurt/code/modules/admin/kick.dm b/modular_zzplurt/code/modules/admin/kick.dm new file mode 100644 index 0000000000000..4e4ace65f8cf2 --- /dev/null +++ b/modular_zzplurt/code/modules/admin/kick.dm @@ -0,0 +1,20 @@ +ADMIN_VERB(kick, R_ADMIN, "Kick", "Kicks a client from the server.", ADMIN_CATEGORY_GAME, mob/target in world) + if(!ismob(target)) + to_chat(usr, span_danger("Error: Target is not a mob. Please try again.")) + return + if(!check_if_greater_rights_than(target.client)) + to_chat(usr, span_danger("Error: They have more rights than you do."), confidential = TRUE) + return + if(tgui_alert(usr, "Kick [key_name(target)]?", "Confirm", list("Yes", "No")) != "Yes") + return + if(!target) + to_chat(usr, span_danger("Error: [target] no longer exists!"), confidential = TRUE) + return + if(!target.client) + to_chat(usr, span_danger("Error: [target] no longer has a client!"), confidential = TRUE) + return + to_chat(target, span_danger("You have been kicked from the server by [usr.client.holder.fakekey ? "an Administrator" : "[usr.client.key]"]."), confidential = TRUE) + log_admin("[key_name(usr)] kicked [key_name(target)].") + message_admins(span_adminnotice("[key_name_admin(usr)] kicked [key_name_admin(target)].")) + qdel(target.client) + BLACKBOX_LOG_ADMIN_VERB("Kick") diff --git a/modular_zzplurt/code/modules/admin/player_panel.dm b/modular_zzplurt/code/modules/admin/player_panel.dm new file mode 100644 index 0000000000000..ad9d2c3d5c84f --- /dev/null +++ b/modular_zzplurt/code/modules/admin/player_panel.dm @@ -0,0 +1,414 @@ +GLOBAL_LIST_INIT(mute_bits, list( + list(name = "IC", bitflag = MUTE_IC), + list(name = "OOC", bitflag = MUTE_OOC), + list(name = "Pray", bitflag = MUTE_PRAY), + list(name = "Ahelp", bitflag = MUTE_ADMINHELP), + list(name = "Deadchat", bitflag = MUTE_DEADCHAT) +)) + +GLOBAL_LIST_INIT(pp_limbs, list( + "Head" = BODY_ZONE_HEAD, + "Left leg" = BODY_ZONE_L_LEG, + "Right leg" = BODY_ZONE_R_LEG, + "Left arm" = BODY_ZONE_L_ARM, + "Right arm" = BODY_ZONE_R_ARM +)) + +ADMIN_VERB_ONLY_CONTEXT_MENU(show_player_panel2, R_ADMIN, "Show Player Panel 2", mob/player in world) + if(!player) + to_chat(user, span_warning("You seem to be selecting a mob that doesn't exist anymore."), confidential = TRUE) + return + + if(!player.mob_panel) + player.create_player_panel() + + player.mob_panel.ui_interact(user.mob) + +/datum/player_panel + var/mob/targetMob + var/client/targetClient + +/datum/player_panel/New(mob/target) + . = ..() + targetMob = target + +/datum/player_panel/Destroy(force, ...) + targetMob = null + targetClient = null + + SStgui.close_uis(src) + return ..() + +/datum/player_panel/ui_interact(mob/user, datum/tgui/ui) + if(!targetMob) + return + + ui = SStgui.try_update_ui(user, src, ui) + if (!ui) + ui = new(user, src, "PlayerPanel", "[targetMob.name] Player Panel") + ui.open() + +/datum/player_panel/ui_state(mob/user) + return GLOB.admin_state + +/datum/player_panel/ui_data(mob/user) + . = list() + .["mob_name"] = targetMob.real_name + .["mob_type"] = targetMob.type + .["admin_mob_type"] = user.client?.mob.type + .["godmode"] = targetMob.status_flags & GODMODE + + var/mob/living/L = targetMob + if (istype(L)) + .["is_frozen"] = L.admin_frozen + .["is_slept"] = L.admin_sleeping + .["mob_scale"] = L.current_size + + if(targetMob.client) + targetClient = targetMob.client + .["client_ckey"] = targetClient.ckey + .["client_muted"] = targetClient.prefs.muted + .["client_rank"] = targetClient.holder ? targetClient.holder.ranks : "Player" + else + targetClient = null + .["client_ckey"] = null + .["last_ckey"] = targetMob.ckey + +/datum/player_panel/ui_static_data() + . = list() + + .["transformables"] = GLOB.pp_transformables + .["glob_limbs"] = GLOB.pp_limbs + .["glob_mute_bits"] = GLOB.mute_bits + .["current_time"] = time2text(world.timeofday, "YYYY-MM-DD hh:mm:ss") + + if(targetClient) + var/byond_version = "Unknown" + if(targetClient.byond_version) + byond_version = "[targetClient.byond_version].[targetClient.byond_build ? targetClient.byond_build : "xxx"]" + .["data_byond_version"] = byond_version + .["data_player_join_date"] = targetClient.player_join_date + .["data_account_join_date"] = targetClient.account_join_date + .["data_related_cid"] = targetClient.related_accounts_cid + .["data_related_ip"] = targetClient.related_accounts_ip + + if(CONFIG_GET(flag/use_exp_tracking)) + .["playtimes_enabled"] = TRUE + .["playtime"] = targetMob.client.get_exp_living() + +/datum/player_panel/ui_act(action, params, datum/tgui/ui) + . = ..() + + var/mob/adminMob = ui.user + var/client/adminClient = adminMob.client + + if(. || !check_rights_for(adminClient, R_ADMIN)) + message_admins(span_adminhelp("WARNING: NON-ADMIN [ADMIN_LOOKUPFLW(adminMob)] ATTEMPTED TO ACCESS ADMIN PANEL. WARN Casper3044.")) + to_chat(adminClient, "Error: you are not an admin!") + return + + switch(action) + // If this mob used to be player controlled but isn't anymore, this action will open the player panel for the mob that player is now controlling. + if ("open_latest_panel") + if (targetMob.client || !targetMob.ckey) + return + + var/mob/latestMob = get_mob_by_ckey(targetMob.ckey) + + if(!latestMob) + to_chat(adminClient, span_warning("That ckey is not controlling a mob.")) + return + + if(targetMob == latestMob) + return + + to_chat(adminClient, span_notice("New mob found for player: [targetMob.ckey] ([latestMob]).")) + SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/show_player_panel, latestMob) + + if ("edit_rank") + if (!targetMob.client?.ckey) + return + + var/list/context = list() + + context["key"] = targetMob.client.ckey + + if (GLOB.admin_datums[targetMob.client.ckey] || GLOB.deadmins[targetMob.client.ckey]) + context["editrights"] = "rank" + else + context["editrights"] = "add" + + adminClient.holder.edit_rights_topic(context) + + if ("access_variables") + adminClient.debug_variables(targetMob) + + if ("access_playtimes") + if (targetMob.client) + adminClient.holder.cmd_show_exp_panel(targetMob.client) + + if ("private_message") + SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/cmd_admin_pm_context, targetMob) + + if ("subtle_message") + SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/cmd_admin_subtle_message, targetMob) + + if ("set_name") + targetMob.vv_auto_rename(params["name"]) + + if ("heal") + SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/cmd_admin_rejuvenate, targetMob) + + if ("ghost") + if(targetMob.client) + log_admin("[key_name(adminClient)] ejected [key_name(targetMob)] from their body.") + message_admins("[key_name_admin(adminClient)] ejected [key_name_admin(targetMob)] from their body.") + to_chat(targetMob, span_danger("An admin has ejected you from your body.")) + targetMob.ghostize(FALSE) + + if ("offer_control") + offer_control(targetMob) + + if ("take_control") + // Disassociates observer mind from the body mind + if(targetMob.client) + targetMob.ghostize(FALSE) + else + for(var/mob/dead/observer/ghost in GLOB.dead_mob_list) + if(targetMob.mind == ghost.mind) + ghost.mind = null + + targetMob.ckey = adminMob.ckey + qdel(adminMob) + + message_admins(span_adminnotice("[key_name_admin(adminClient)] took control of [targetMob].")) + log_admin("[key_name(adminClient)] took control of [targetMob].") + addtimer(CALLBACK(targetMob.mob_panel, TYPE_PROC_REF(/datum, ui_interact), targetMob), 0.1 SECONDS) + + if ("smite") + SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/admin_smite, targetMob) + + if ("bring") + SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/get_mob, targetMob) + + if ("orbit") + if(!isobserver(adminMob)) + SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/admin_ghost) + var/mob/dead/observer/O = adminClient.mob + O.ManualFollow(targetMob) + + if ("jump_to") + SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/jump_to_mob, targetMob) + + if ("freeze") + var/mob/living/L = targetMob + if (istype(L)) + SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/toggle_admin_freeze, L) + + if ("sleep") + var/mob/living/L = targetMob + if (istype(L)) + SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/toggle_admin_sleep, L) + + if ("lobby") + if(!isobserver(targetMob)) + to_chat(adminClient, span_notice("You can only send ghost players back to the Lobby.")) + return + + if(!targetMob.client) + to_chat(adminClient, span_warning("[targetMob] doesn't seem to have an active client.")) + return + + log_admin("[key_name(adminClient)] has sent [key_name(targetMob)] back to the Lobby.") + message_admins("[key_name(adminClient)] has sent [key_name(targetMob)] back to the Lobby.") + + var/mob/dead/new_player/NP = new() + NP.ckey = targetMob.ckey + qdel(targetMob) + + if ("select_equipment") + SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/select_equipment, targetMob) + + if ("strip") + for(var/obj/item/I in targetMob) + targetMob.dropItemToGround(I, TRUE) //The TRUE forces all items to drop, since this is an admin undress. + + if ("cryo") + targetMob.vv_send_cryo() + + if ("force_say") + targetMob.say(params["to_say"], forced="admin") + + if ("force_emote") + message_admins("force emote called") + targetMob.emote("me", EMOTE_VISIBLE|EMOTE_AUDIBLE, "Test 123") + targetMob.emote("me", EMOTE_VISIBLE|EMOTE_AUDIBLE, params["to_emote"]) + QUEUE_OR_CALL_VERB_FOR(VERB_CALLBACK(targetMob, TYPE_PROC_REF(/mob, emote), "me", EMOTE_VISIBLE|EMOTE_AUDIBLE, params["to_emote"]), SSspeech_controller) + + if ("prison") + if(isAI(targetMob)) + to_chat(adminClient, "This cannot be used on instances of type /mob/living/silicon/ai.") + return + + targetMob.forceMove(pick(GLOB.prisonwarp)) + to_chat(targetMob, span_userdanger("You have been sent to Prison!")) + + log_admin("[key_name(adminClient)] has sent [key_name(targetMob)] to Prison!") + message_admins("[key_name_admin(adminClient)] has sent [key_name_admin(targetMob)] to Prison!") + + if ("kick") + SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/kick, targetMob) + + if ("ban") + var/player_key = targetMob.key + var/player_ip = targetMob.client.address + var/player_cid = targetMob.client.computer_id + adminClient.holder.ban_panel(player_key, player_ip, player_cid) + + if ("sticky_ban") + var/list/ban_settings = list() + if(targetMob.client) + ban_settings["ckey"] = targetMob.ckey + adminClient.holder.stickyban("add", ban_settings) + + if ("notes") + if (targetMob.client) + browse_messages(target_ckey = ckey(targetMob.ckey)) + + if ("logs") + var/source = targetMob.client ? LOGSRC_CKEY : LOGSRC_MOB + show_individual_logging_panel(targetMob, source) + + if ("mute") + if(!targetMob.client) + return + + targetMob.client.prefs.muted = text2num(params["mute_flag"]) + log_admin("[key_name(adminClient)] set the mute flags for [key_name(targetMob)] to [targetMob.client.prefs.muted].") + + if ("mute_all") + if(!targetMob.client) + return + + for(var/bit in GLOB.mute_bits) + targetMob.client.prefs.muted |= bit["bitflag"] + + log_admin("[key_name(adminClient)] mass-muted [key_name(targetMob)].") + + if ("unmute_all") + if(!targetMob.client) + return + + for(var/bit in GLOB.mute_bits) + targetMob.client.prefs.muted &= ~bit["bitflag"] + + log_admin("[key_name(adminClient)] mass-unmuted [key_name(targetMob)].") + + if ("related_accounts") + if(targetMob.client) + var/related_accounts + if (params["related_thing"] == "CID") + related_accounts = targetMob.client.related_accounts_cid + else + related_accounts = targetMob.client.related_accounts_ip + + related_accounts = splittext(related_accounts, ", ") + + var/list/dat = list("Related accounts by [params["related_thing"]]:") + dat += related_accounts + adminClient << browse(dat.Join("
"), "window=related_[targetMob.client];size=420x300") + + if ("transform") + var/choice = params["newType"] + if (choice == "/mob/living") + choice = tgui_input_list(adminClient, "What should this mob transform into", "Mob Transform", subtypesof(choice)) + if (!choice) + return + + adminClient.holder.transformMob(targetMob, adminMob, choice, params["newTypeName"]) + + if ("toggle_godmode") + adminClient.cmd_admin_godmode(targetMob) + + if ("spell") + SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/give_spell, targetMob) + + if ("martial_art") + adminClient.teach_martial_art(targetMob) + + if ("quirk") + adminClient.toggle_quirk(targetMob) + + if ("species") + adminClient.set_species(targetMob) + + if ("limb") + if(!params["limbs"] || !ishuman(targetMob)) + return + + var/mob/living/carbon/human/H = targetMob + + for(var/limb in params["limbs"]) + if (!limb) + continue + + if (params["delimb_mode"]) + var/obj/item/bodypart/L = H.get_bodypart(limb) + if (!L) + continue + L.dismember() + playsound(H, 'sound/effects/cartoon_pop.ogg', 70) + else + H.regenerate_limb(limb) + + if ("scale") + var/mob/living/L = targetMob + if(!isnull(params["new_scale"]) && istype(L)) + L.vv_edit_var("current_size", params["new_scale"]) + // mobSize = params["new_scale"] + + if ("explode") + var/power = text2num(params["power"]) + var/empMode = text2num(params["emp_mode"]) + + + var/turf/T = get_turf(adminMob) + message_admins("[ADMIN_LOOKUPFLW(adminClient)] created an admin [empMode ? "EMP" : "explosion"] at [ADMIN_VERBOSEJMP(T)].") + log_admin("[key_name(adminClient)] created an admin [empMode ? "EMP" : "explosion"] at [adminMob.loc].") + + if (empMode) + empulse(adminMob, power, power / 2, TRUE) + else + explosion(adminMob, power / 3, power / 2, power, power, ignorecap = TRUE) + + if ("narrate") + var/list/stylesRaw = params["classes"] + + var/styles = "" + for(var/style in stylesRaw) + styles += "[style]:[stylesRaw[style]];" + + if (params["mode_global"]) + to_chat(world, "[params["message"]]") + log_admin("GlobalNarrate: [key_name(adminClient)] : [params["message"]]") + message_admins(span_adminnotice("[key_name_admin(adminClient)] Sent a global narrate")) + else + for(var/mob/M in view(params["range"], adminMob)) + to_chat(M, "[params["message"]]") + + log_admin("LocalNarrate: [key_name(adminClient)] at [AREACOORD(adminMob)]: [params["message"]]") + message_admins(span_adminnotice(" LocalNarrate: [key_name_admin(adminClient)] at [ADMIN_VERBOSEJMP(adminMob)]: [params["message"]]
")) + + if ("languages") + var/datum/language_holder/H = targetMob.get_language_holder() + H.open_language_menu(adminMob) + + if ("traitor_panel") + SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/show_traitor_panel, targetMob) + + if ("job_exemption_panel") + show_job_exempt_menu(adminMob, targetMob.ckey) + + if ("skill_panel") + SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/show_skill_panel, targetMob) + diff --git a/modular_zzplurt/code/modules/admin/transform.dm b/modular_zzplurt/code/modules/admin/transform.dm new file mode 100644 index 0000000000000..f6158e3683870 --- /dev/null +++ b/modular_zzplurt/code/modules/admin/transform.dm @@ -0,0 +1,127 @@ +// Stuff that helps the TGUI player panel transform section to work + +GLOBAL_LIST_INIT(pp_transformables, list( + list( + name = "Common", + color = "", + types = list( + list( + name = "Human", + key = /mob/living/carbon/human + ), + list( + name = "Monkey", + key = /mob/living/carbon/human/species/monkey + ), + list( + name = "Cyborg", + key = /mob/living/silicon/robot + ) + ) + ), + list( + name = "Xenomorph", + color = "purple", + types = list( + list( + name = "Larva", + key = /mob/living/carbon/alien/larva + ), + list( + name = "Drone", + key = /mob/living/carbon/alien/adult/drone + ), + list( + name = "Hunter", + key = /mob/living/carbon/alien/adult/hunter + ), + list( + name = "Sentinel", + key = /mob/living/carbon/alien/adult/sentinel + ), + list( + name = "Praetorian", + key = /mob/living/carbon/alien/adult/royal/praetorian + ), + list( + name = "Queen", + key = /mob/living/carbon/alien/adult/royal/queen + ) + ) + ), + list( + name = "Lavaland", + color = "orange", + types = list( + list( + name = "Goliath", + key = /mob/living/basic/mining/goliath + ), + list( + name = "Legion", + key = /mob/living/simple_animal/hostile/megafauna/legion/small + ), + list( + name = "Blood-Drunk Miner", + key = /mob/living/simple_animal/hostile/megafauna/blood_drunk_miner + ), + list( + name = "Gladiator", + key = /mob/living/simple_animal/hostile/megafauna/gladiator + ), + list( + name = "Dragon", + key = /mob/living/simple_animal/hostile/megafauna/dragon + ), + list( + name = "Legion Hivelord", + key = /mob/living/simple_animal/hostile/asteroid/elite/legionnaire + ) + ) + ), + list( + name = "Cultist", + color = "violet", + types = list( + list( + name = "Shade", + key = /mob/living/basic/shade + ), + list( + name = "Artificer", + key = /mob/living/basic/construct/artificer + ), + list( + name = "Wraith", + key = /mob/living/basic/construct/wraith + ), + list( + name = "Juggernaut", + key = /mob/living/basic/construct/juggernaut + ) + ) + ) +)) + +// M: Mob to change +// newType: Path of new type e.g: /mob/living/carbon/alien/humanoid/drone +// newTypeName (optional): Name of the new type (used in logging): e.g: "Drone" +/datum/admins/proc/transformMob(mob/M, mob/adminMob, newType, newTypeName) + if(!check_rights(R_SPAWN)) + return + + if(!ismob(M)) + to_chat(usr, "This can only be used on instances of type /mob.") + return + + if (!newTypeName) + newTypeName = newType + + log_admin("[key_name(usr)] transformed [key_name(M)] into a [newTypeName].") + message_admins("[key_name_admin(usr)] transformed [key_name_admin(M)] into a [newTypeName].") + + var/mob/newMob = M.change_mob_type(newType, delete_old_mob = TRUE) + + if (M == adminMob) + adminMob = newMob + addtimer(CALLBACK(newMob.mob_panel, PROC_REF(ui_interact), adminMob), 0.1 SECONDS) diff --git a/modular_zzplurt/code/modules/admin/verbs.dm b/modular_zzplurt/code/modules/admin/verbs.dm new file mode 100644 index 0000000000000..7548462ec5f69 --- /dev/null +++ b/modular_zzplurt/code/modules/admin/verbs.dm @@ -0,0 +1,7 @@ +ADMIN_VERB(toggle_admin_freeze, R_ADMIN, "Toggle Freeze Mob", "Freezes / unfreezes a mob's movement", ADMIN_CATEGORY_GAME, mob/living/target in world) + target.toggle_admin_freeze(user) + BLACKBOX_LOG_ADMIN_VERB("Toggle Freeze Mob") + +ADMIN_VERB(toggle_admin_sleep, R_ADMIN, "Toggle Sleep Mob", "Sleeps / unsleeps a mob", ADMIN_CATEGORY_GAME, mob/living/target in world) + target.toggle_admin_sleep(user) + BLACKBOX_LOG_ADMIN_VERB("Toggle Sleep Mob") diff --git a/modular_zzplurt/code/modules/client/client_procs.dm b/modular_zzplurt/code/modules/client/client_procs.dm new file mode 100644 index 0000000000000..706320fdf1a9d --- /dev/null +++ b/modular_zzplurt/code/modules/client/client_procs.dm @@ -0,0 +1,60 @@ + +/client/proc/toggle_quirk(mob/living/carbon/human/H) + if (!istype(H)) + to_chat(usr, "This can only be used on /mob/living/carbon/human.") + return + + var/list/options = list("Clear"="Clear") + for(var/x in subtypesof(/datum/quirk)) + var/datum/quirk/T = x + var/qname = initial(T.name) + options[H.has_quirk(T) ? "[qname] (Remove)" : "[qname] (Add)"] = T + + var/result = tgui_input_list(usr, "Choose quirk to add/remove", "Mob Quirks", options) + + if(QDELETED(H)) + to_chat(usr, "Mob doesn't exist anymore") + return + + if(result) + if(result == "Clear") + for(var/datum/quirk/q in H.quirks) + H.remove_quirk(q.type) + else + var/T = options[result] + if(H.has_quirk(T)) + H.remove_quirk(T) + else + H.add_quirk(T,TRUE) + +/client/proc/teach_martial_art(mob/living/carbon/C) + if (!istype(C)) + to_chat(usr, "This can only be used on /mob/living/carbon.") + return + + var/list/artpaths = subtypesof(/datum/martial_art) + var/list/artnames = list() + for(var/i in artpaths) + var/datum/martial_art/M = i + artnames[initial(M.name)] = M + var/result = tgui_input_list(usr, "Choose the martial art to teach", "JUDO CHOP", artnames) // input(usr, "Choose the martial art to teach","JUDO CHOP") as null|anything in artnames + + if(QDELETED(C)) + to_chat(usr, "Mob doesn't exist anymore") + return + if(result) + var/chosenart = artnames[result] + var/datum/martial_art/MA = new chosenart + MA.teach(C) + log_admin("[key_name(usr)] has taught [MA] to [key_name(C)].") + message_admins(span_notice("[key_name_admin(usr)] has taught [MA] to [key_name_admin(C)].")) + +/client/proc/set_species(mob/living/carbon/human/H) + if (istype(H)) + var/result = tgui_input_list(usr, "Choose a new species","Species", GLOB.species_list) + if(QDELETED(H)) + to_chat(usr, "Mob doesn't exist anymore") + return + if(result) + admin_ticket_log("[key_name_admin(usr)] has modified the bodyparts of [H] to [result]") + H.set_species(GLOB.species_list[result]) diff --git a/modular_zzplurt/code/modules/mob/living/living.dm b/modular_zzplurt/code/modules/mob/living/living.dm new file mode 100644 index 0000000000000..44e8544302ad1 --- /dev/null +++ b/modular_zzplurt/code/modules/mob/living/living.dm @@ -0,0 +1,27 @@ +/// Toggle admin frozen +/mob/living/proc/toggle_admin_freeze(client/admin) + admin_frozen = !admin_frozen + + if(admin_frozen) + SetStun(INFINITY, ignore_canstun = TRUE) + else + SetStun(0, ignore_canstun = TRUE) + + if(client && admin) + to_chat(src, span_userdanger("An admin has [!admin_frozen ? "un" : ""]frozen you.")) + log_admin("[key_name(admin)] toggled admin-freeze on [key_name(src)].") + message_admins("[key_name_admin(admin)] toggled admin-freeze on [key_name_admin(src)].") + +/// Toggle admin sleeping +/mob/living/proc/toggle_admin_sleep(client/admin) + admin_sleeping = !admin_sleeping + + if(admin_sleeping) + SetSleeping(INFINITY) + else + SetSleeping(0) + + if(client && admin) + to_chat(src, span_userdanger("An admin has [!admin_sleeping ? "un": ""]slept you.")) + log_admin("[key_name(admin)] toggled admin-sleep on [key_name(src)].") + message_admins("[key_name_admin(admin)] toggled admin-sleep on [key_name_admin(src)].") diff --git a/modular_zzplurt/code/modules/mob/living/living_defines.dm b/modular_zzplurt/code/modules/mob/living/living_defines.dm new file mode 100644 index 0000000000000..bfc77190412cf --- /dev/null +++ b/modular_zzplurt/code/modules/mob/living/living_defines.dm @@ -0,0 +1,4 @@ +/mob/living + // Admin CC + var/admin_frozen = FALSE + var/admin_sleeping = FALSE diff --git a/tgstation.dme b/tgstation.dme index 1522d37464652..01dfc2591e855 100644 --- a/tgstation.dme +++ b/tgstation.dme @@ -9077,9 +9077,18 @@ #include "modular_zzplurt\code\_globalvars\mobs.dm" #include "modular_zzplurt\code\controllers\configuration\entries\discord.dm" #include "modular_zzplurt\code\controllers\subsystem\discord.dm" +#include "modular_zzplurt\code\modules\admin\ban.dm" +#include "modular_zzplurt\code\modules\admin\jobban.dm" +#include "modular_zzplurt\code\modules\admin\kick.dm" +#include "modular_zzplurt\code\modules\admin\transform.dm" +#include "modular_zzplurt\code\modules\admin\verbs.dm" +#include "modular_zzplurt\code\modules\client\client_procs.dm" #include "modular_zzplurt\code\modules\client\verbs\looc.dm" #include "modular_zzplurt\code\modules\client\verbs\ooc.dm" #include "modular_zzplurt\code\modules\discord\tgs_commands.dm" #include "modular_zzplurt\code\modules\discord\verbs.dm" #include "modular_zzplurt\code\modules\mob\dead\new_player\new_player.dm" +#include "modular_zzplurt\code\modules\admin\player_panel.dm" +#include "modular_zzplurt\code\modules\mob\living\living.dm" +#include "modular_zzplurt\code\modules\mob\living\living_defines.dm" // END_INCLUDE diff --git a/tgui/packages/tgui/components/Collapsible.tsx b/tgui/packages/tgui/components/Collapsible.tsx index 9f5f944b0ec04..e2dc046c95fa1 100644 --- a/tgui/packages/tgui/components/Collapsible.tsx +++ b/tgui/packages/tgui/components/Collapsible.tsx @@ -14,11 +14,12 @@ type Props = Partial<{ open: boolean; title: ReactNode; icon: string; + disabled: boolean; }> & BoxProps; export function Collapsible(props: Props) { - const { children, color, title, buttons, icon, ...rest } = props; + const { children, color, title, buttons, icon, disabled, ...rest } = props; const [open, setOpen] = useState(props.open); return ( @@ -29,6 +30,7 @@ export function Collapsible(props: Props) { fluid color={color} icon={icon ? icon : open ? 'chevron-down' : 'chevron-right'} + disabled={disabled} onClick={() => setOpen(!open)} {...rest} > @@ -39,7 +41,7 @@ export function Collapsible(props: Props) {
{buttons}
)} - {open && {children}} + {open && !disabled && {children}} ); } diff --git a/tgui/packages/tgui/interfaces/PlayerPanel.tsx b/tgui/packages/tgui/interfaces/PlayerPanel.tsx new file mode 100644 index 0000000000000..8d3b85d9f885e --- /dev/null +++ b/tgui/packages/tgui/interfaces/PlayerPanel.tsx @@ -0,0 +1,1109 @@ +import { useState } from 'react'; + +import { useBackend } from '../backend'; +import { + Box, + Button, + Collapsible, + Dropdown, + Flex, + Input, + LabeledList, + NoticeBox, + NumberInput, + Section, + Slider, + Tabs, +} from '../components'; +import { Window } from '../layouts'; + +type Data = { + mob_name: string; + mob_type: string; + admin_mob_type: string; + client_ckey: string; + client_rank: string; + last_ckey: string; + playtimes_enabled: boolean; + playtime: string; + godmode: boolean; + is_frozen: boolean; + is_slept: boolean; + client_muted: number; + current_time: string; + data_related_ip: string; + data_related_cid: string; + data_player_join_date: string; + data_account_join_date: string; + data_byond_version: string; + + glob_mute_bits: { + name: string; + bitflag: number; + }[]; + + glob_limbs: { + [key: string]: string; + }; + + transformables: { + name: string; + color: string; + types: { + name: string; + key: string; + }[]; + }[]; +}; + +const PAGES = [ + { + title: 'General', + component: () => GeneralActions, + color: 'green', + icon: 'tools', + }, + { + title: 'Mob', + component: () => PhysicalActions, + color: 'yellow', + icon: 'bolt', + canAccess: (data) => { + return !!data.mob_type.includes('/mob/living'); + }, + }, + { + title: 'Transform', + component: () => TransformActions, + color: 'orange', + icon: 'exchange-alt', + }, + { + title: 'Punish', + component: () => PunishmentActions, + color: 'red', + icon: 'gavel', + }, + { + title: 'Fun', + component: () => FunActions, + color: 'blue', + icon: 'laugh', + }, + { + title: 'Other', + component: () => OtherActions, + color: 'blue', + icon: 'crosshairs', + }, +]; + +export const PlayerPanel = () => { + const { act, data } = useBackend(); + const [pageIndex, setPageIndex] = useState(0); + const PageComponent = PAGES[pageIndex].component(); + + const { + mob_name, + mob_type, + client_ckey, + client_rank, + last_ckey, + playtimes_enabled, + playtime, + } = data; + + return ( + + +
+ + + Name: + + + act('set_name', { name: value })} + /> + + {!!client_ckey && ( + + + Rank: + + + + + + )} + + + + Mob Type: + + + {mob_type} + + + + + {!!client_ckey && ( + + + + )} + + {(!!client_ckey || !!last_ckey) && ( + + + {client_ckey ? 'Client:' : 'Last client:'} + + {client_ckey || last_ckey} + + {!client_ckey && last_ckey && ( + + + + )} + + {!!client_ckey && ( + + + + + )} + + )} +
+ + +
+ + {PAGES.map((page, i) => { + if (page.canAccess && !page.canAccess(data)) { + return; + } + + return ( + setPageIndex(i)} + > + {page.title} + + ); + })} + +
+
+ + + +
+
+
+ ); +}; + +const GeneralActions = () => { + const { act, data } = useBackend(); + const { client_ckey, mob_type, admin_mob_type } = data; + return ( +
+
+ + + + +
+ +
+ + act('bring')} + > + Bring + + + act('jump_to')} + > + Jump To + + +
+ +
+ + + act('strip')} + > + Drop All Items + + + + act('cryo')} + > + Send To Cryo + + act('lobby')} + > + Send To Lobby + + +
+
+ + act('ghost')} + > + Eject Ghost + + act('take_control')} + > + Take Control + + act('offer_control')} + > + Offer Control + + +
+
+ ); +}; + +const PhysicalActions = () => { + const { act, data } = useBackend(); + const { glob_limbs, godmode, mob_type } = data; + const [mobScale, setMobScale] = useState(1); + const limbs = Object.keys(glob_limbs); + const limb_flags = limbs.map((_, i) => 1 << i); + const [delimbOption, setDelimbOption] = useState(0); + + return ( +
+
act('toggle_godmode')} + > + God Mode + + } + > + + + + + + +
+
+ {limbs.map((val, index) => ( + + setDelimbOption( + delimbOption & limb_flags[index] + ? delimbOption & ~limb_flags[index] + : delimbOption | limb_flags[index], + ) + } + > + {val} + + ))} + + } + > + + + act('limb', { + limbs: limb_flags.map( + (val, index) => + !!(delimbOption & val) && glob_limbs[limbs[index]], + ), + delimb_mode: true, + }) + } + > + Delimb + + + act('limb', { + limbs: limb_flags.map( + (val, index) => + !!(delimbOption & val) && glob_limbs[limbs[index]], + ), + }) + } + > + Relimb + + +
+
{ + setMobScale(1); + act('scale', { new_scale: 1 }); + }} + > + Reset + + } + > + + { + setMobScale(value); // Update slider value + act('scale', { new_scale: value }); // Update mob's value + }} + unit="x" + /> + +
+
+ + + Force Say: + + + act('force_say', { to_say: value })} + /> + + + + + Force Emote: + + + act('force_emote', { to_emote: value })} + /> + + +
+
+ ); +}; + +const TransformActions = () => { + const { act, data } = useBackend(); + const { transformables, mob_type } = data; + return ( +
+ + + {transformables.map((transformables_category) => { + return ( +
+ + {transformables_category.types.map((transformables_type) => { + return ( + + + act('transform', { + newType: transformables_type.key, + newTypeName: transformables_type.name, + }) + } + > + {transformables_type.name} + + + ); + })} + +
+ ); + })} +
+ ); +}; + +const PunishmentActions = () => { + const { act, data } = useBackend(); + const { + client_ckey, + mob_type, + is_frozen, + is_slept, + glob_mute_bits, + client_muted, + data_related_cid, + data_related_ip, + data_byond_version, + data_player_join_date, + data_account_join_date, + current_time, + } = data; + return ( +
+ + + + +
+ + + + act('prison')} + > + Admin Prison + + +
+ +
+ + act('kick')} + > + Kick + + + act('sticky_ban')} + > + Sticky Ban + + +
+ +
+ + + + } + > + + {glob_mute_bits.map((bit, i) => { + const isMuted = client_muted && client_muted & bit.bitflag; + return ( + + ); + })} + +
+
+ + Related accounts by: + + + + + } + > + + + + {current_time} + + + {data_account_join_date} + + + {data_player_join_date} + + + {data_byond_version} + + + +
+
+ ); +}; + +const FunActions = () => { + const { act } = useBackend(); + + const colours = { + White: '#a4bad6', + Dark: '#42474D', + Red: '#c51e1e', + 'Red Bright': '#FF0000', + Velvet: '#660015', + Green: '#059223', + Blue: '#6685f5', + Purple: '#800080', + 'Purple Dark': '#5000A0', + Narsie: '#973e3b', + Ratvar: '#BE8700', + }; + + const [lockExplode, setLockExplode] = useState(true); + const [empMode, setEmpMode] = useState(false); + const [expPower, setExpPower] = useState(8); + const [narrateSize, setNarrateSize] = useState(1); + const [narrateMessage, setNarrateMessage] = useState(''); + const [narrateColour, setNarrateColour] = useState(Object.keys(colours)[0]); + const [narrateFont, setNarrateFont] = useState('Verdana'); + const [narrateBold, setNarrateBold] = useState(false); + const [narrateItalic, setNarrateItalic] = useState(false); + const [narrateGlobal, setNarrateGlobal] = useState(false); + const [narrateRange, setNarrateRange] = useState(7); + + const narrateStyles = { + color: colours[narrateColour], + 'font-size': narrateSize + 'rem', + 'font-weight': narrateBold ? 'bold' : '', + 'font-family': narrateFont, + 'font-style': narrateItalic ? 'italic' : '', + }; + + return ( +
+ + These features are centred on YOUR viewport + + +
+ setEmpMode(!empMode)} + > + EMP Mode + + + + } + > + + + + + + setExpPower(value)} + ranges={{ + green: [0, 8], + orange: [8, 15], + red: [15, 30], + }} + minValue={1} + maxValue={30} + height="100%" + /> + + +
+
setNarrateGlobal(!narrateGlobal)} + > + Global Narrate + + } + > + + + + + + setNarrateColour(value)} + /> + + + setNarrateFont(value)} + /> + + + + + + + setNarrateBold(!narrateBold)} + /> + + + setNarrateItalic(!narrateItalic)} + /> + + + + + + + setNarrateSize(value)} + /> + + {!narrateGlobal && ( + + setNarrateRange(value)} + /> + + )} + + + + + + + + setNarrateMessage(value)} + /> + + + + + + + {narrateMessage} + +
+
+ ); +}; + +const OtherActions = () => { + const { act, data } = useBackend(); + const { mob_type, client_ckey } = data; + + return ( +
+
+ + + + +
+
+ ); +}; From eea296d263b92f830c047fb987c2f286716c192a Mon Sep 17 00:00:00 2001 From: Casper Date: Mon, 8 Jul 2024 13:05:19 +0100 Subject: [PATCH 2/7] Seems to work --- code/modules/admin/admin_verbs.dm | 2 +- code/modules/admin/topic.dm | 7 -- code/modules/admin/verbs/admingame.dm | 14 ++- code/modules/mob/mob.dm | 7 -- code/modules/mob/mob_defines.dm | 3 - .../code/modules/admin/player_panel.dm | 102 ++++++++++++++---- modular_zzplurt/code/modules/admin/verbs.dm | 7 -- .../client/preferences/player_panel.dm | 11 ++ modular_zzplurt/code/modules/mob/mob.dm | 12 +++ .../code/modules/mob/mob_defines.dm | 3 + tgstation.dme | 4 +- tgui/packages/tgui/interfaces/PlayerPanel.tsx | 92 ++++++++++++---- .../features/game_preferences/admin.tsx | 7 ++ 13 files changed, 203 insertions(+), 68 deletions(-) delete mode 100644 modular_zzplurt/code/modules/admin/verbs.dm create mode 100644 modular_zzplurt/code/modules/client/preferences/player_panel.dm create mode 100644 modular_zzplurt/code/modules/mob/mob.dm create mode 100644 modular_zzplurt/code/modules/mob/mob_defines.dm diff --git a/code/modules/admin/admin_verbs.dm b/code/modules/admin/admin_verbs.dm index 4a9bb413e3f5f..0f7d7a3f39b30 100644 --- a/code/modules/admin/admin_verbs.dm +++ b/code/modules/admin/admin_verbs.dm @@ -341,7 +341,7 @@ ADMIN_VERB(remove_mob_action, R_FUN, "Remove Mob Action", ADMIN_VERB_NO_DESCRIPT if(!length(target_abilities)) return - var/chosen_ability = tgui_input_list(user, "Choose the action to remove from [removal_target]", "Depower", sort_list(target_abilities)) + var/chosen_ability = tgui_input_list(user, "Choose the spell to remove from [removal_target]", "Depower", sort_list(target_abilities)) if(isnull(chosen_ability)) return var/datum/action/cooldown/mob_cooldown/to_remove = target_abilities[chosen_ability] diff --git a/code/modules/admin/topic.dm b/code/modules/admin/topic.dm index d8547e8afae4a..c80a3a581afb3 100644 --- a/code/modules/admin/topic.dm +++ b/code/modules/admin/topic.dm @@ -1032,13 +1032,6 @@ var/mob/M = locate(href_list["SyndicateReply"]) usr.client.admin_headset_message(M, RADIO_CHANNEL_SYNDICATE) - else if(href_list["HeadsetMessage"]) - if(!check_rights(R_ADMIN)) - return - - var/mob/M = locate(href_list["HeadsetMessage"]) - usr.client.admin_headset_message(M) - else if(href_list["reject_custom_name"]) if(!check_rights(R_ADMIN)) return diff --git a/code/modules/admin/verbs/admingame.dm b/code/modules/admin/verbs/admingame.dm index 26e3758955471..f8a20b02cb29c 100644 --- a/code/modules/admin/verbs/admingame.dm +++ b/code/modules/admin/verbs/admingame.dm @@ -2,12 +2,22 @@ ADMIN_VERB(cmd_player_panel, R_ADMIN, "Player Panel", "See all players and their user.holder.player_panel_new() ADMIN_VERB_ONLY_CONTEXT_MENU(show_player_panel, R_ADMIN, "Show Player Panel", mob/player in world) - log_admin("[key_name(user)] checked the individual player panel for [key_name(player)][isobserver(user.mob)?"":" while in game"].") - if(!player) to_chat(user, span_warning("You seem to be selecting a mob that doesn't exist anymore."), confidential = TRUE) return + var/useModern = user.prefs.read_preference(/datum/preference/toggle/use_tgui_player_panel) + + message_admins(useModern) + + if (useModern) + if(!player.mob_panel) + player.create_player_panel() + player.mob_panel.ui_interact(user.mob) + return + + log_admin("[key_name(user)] checked the individual player panel for [key_name(player)][isobserver(user.mob)?"":" while in game"].") + var/body = "Options for [player.key]" body += "Options panel for [player]" if(player.client) diff --git a/code/modules/mob/mob.dm b/code/modules/mob/mob.dm index 37cb1af25238a..0922ac6ae51eb 100644 --- a/code/modules/mob/mob.dm +++ b/code/modules/mob/mob.dm @@ -1,8 +1,3 @@ -/mob/proc/create_player_panel() - QDEL_NULL(mob_panel) - - mob_panel = new(src) - /** * Delete a mob * @@ -49,7 +44,6 @@ observe.reset_perspective(null) qdel(hud_used) - QDEL_NULL(mob_panel) QDEL_LIST(client_colours) ghostize(can_reenter_corpse = FALSE) //False, since we're deleting it currently if(mind?.current == src) //Let's just be safe yeah? This will occasionally be cleared, but not always. Can't do it with ghostize without changing behavior @@ -101,7 +95,6 @@ update_config_movespeed() initialize_actionspeed() update_movespeed(TRUE) - create_player_panel() become_hearing_sensitive() log_mob_tag("TAG: [tag] CREATED: [key_name(src)] \[[type]\]") diff --git a/code/modules/mob/mob_defines.dm b/code/modules/mob/mob_defines.dm index 3f95defe17a85..2206efd0e13ce 100644 --- a/code/modules/mob/mob_defines.dm +++ b/code/modules/mob/mob_defines.dm @@ -171,9 +171,6 @@ var/list/client_colours = list() var/hud_type = /datum/hud - // Admin player panel for this mob - var/datum/player_panel/mob_panel - var/datum/focus //What receives our keyboard inputs. src by default /// Used for tracking last uses of emotes for cooldown purposes diff --git a/modular_zzplurt/code/modules/admin/player_panel.dm b/modular_zzplurt/code/modules/admin/player_panel.dm index ad9d2c3d5c84f..f7b7c73b43fbb 100644 --- a/modular_zzplurt/code/modules/admin/player_panel.dm +++ b/modular_zzplurt/code/modules/admin/player_panel.dm @@ -1,6 +1,7 @@ GLOBAL_LIST_INIT(mute_bits, list( list(name = "IC", bitflag = MUTE_IC), list(name = "OOC", bitflag = MUTE_OOC), + list(name = "LOOC", bitflag = MUTE_LOOC), list(name = "Pray", bitflag = MUTE_PRAY), list(name = "Ahelp", bitflag = MUTE_ADMINHELP), list(name = "Deadchat", bitflag = MUTE_DEADCHAT) @@ -14,16 +15,6 @@ GLOBAL_LIST_INIT(pp_limbs, list( "Right arm" = BODY_ZONE_R_ARM )) -ADMIN_VERB_ONLY_CONTEXT_MENU(show_player_panel2, R_ADMIN, "Show Player Panel 2", mob/player in world) - if(!player) - to_chat(user, span_warning("You seem to be selecting a mob that doesn't exist anymore."), confidential = TRUE) - return - - if(!player.mob_panel) - player.create_player_panel() - - player.mob_panel.ui_interact(user.mob) - /datum/player_panel var/mob/targetMob var/client/targetClient @@ -72,7 +63,9 @@ ADMIN_VERB_ONLY_CONTEXT_MENU(show_player_panel2, R_ADMIN, "Show Player Panel 2", else targetClient = null .["client_ckey"] = null - .["last_ckey"] = targetMob.ckey + + if (targetMob.ckey) + .["last_ckey"] = copytext(targetMob.ckey, 2) /datum/player_panel/ui_static_data() . = list() @@ -92,6 +85,20 @@ ADMIN_VERB_ONLY_CONTEXT_MENU(show_player_panel2, R_ADMIN, "Show Player Panel 2", .["data_related_cid"] = targetClient.related_accounts_cid .["data_related_ip"] = targetClient.related_accounts_ip + var/datum/player_details/deets = GLOB.player_details[targetClient.ckey] + .["data_old_names"] = deets.get_played_names() || null + + var/list/player_ranks = list() + if(SSplayer_ranks.is_donator(targetClient, admin_bypass = FALSE)) + player_ranks += "Donator" + if(SSplayer_ranks.is_mentor(targetClient, admin_bypass = FALSE)) + player_ranks += "Mentor" + if(SSplayer_ranks.is_veteran(targetClient, admin_bypass = FALSE)) + player_ranks += "Veteran" + if(SSplayer_ranks.is_vetted(targetClient, admin_bypass = FALSE)) + player_ranks |= "Vetted" + .["ranks"] = length(player_ranks) ? player_ranks.Join(", ") : null + if(CONFIG_GET(flag/use_exp_tracking)) .["playtimes_enabled"] = TRUE .["playtime"] = targetMob.client.get_exp_living() @@ -113,7 +120,9 @@ ADMIN_VERB_ONLY_CONTEXT_MENU(show_player_panel2, R_ADMIN, "Show Player Panel 2", if (targetMob.client || !targetMob.ckey) return - var/mob/latestMob = get_mob_by_ckey(targetMob.ckey) + // Remove '@' from the start of the ckey. + var/ckey = copytext(targetMob.ckey, 2) + var/mob/latestMob = get_mob_by_ckey(ckey) if(!latestMob) to_chat(adminClient, span_warning("That ckey is not controlling a mob.")) @@ -151,7 +160,35 @@ ADMIN_VERB_ONLY_CONTEXT_MENU(show_player_panel2, R_ADMIN, "Show Player Panel 2", SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/cmd_admin_pm_context, targetMob) if ("subtle_message") - SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/cmd_admin_subtle_message, targetMob) + var/list/subtle_message_options = list("Voice in head", RADIO_CHANNEL_CENTCOM, RADIO_CHANNEL_SYNDICATE) + var/sender = tgui_input_list(adminClient, "Choose the method of subtle messaging", "Subtle Message", subtle_message_options) + if (!sender) + return + + var/msg = input("Contents of the message", text("Subtle PM to [targetMob.key]")) as text + if (!msg) + return + + if (sender == "Voice in head") + to_chat(targetMob, "You hear a voice in your head... [msg]") + else + var/mob/living/carbon/human/H = targetMob + + if(!istype(H)) + to_chat(adminClient, "The person you are trying to contact is not human. Unsent message: [msg]") + return + + if(!istype(H.ears, /obj/item/radio/headset)) + to_chat(adminClient, "The person you are trying to contact is not wearing a headset. Unsent message: [msg]") + return + + to_chat(H, "You hear something crackle in your ears for a moment before a voice speaks. \"Please stand by for a message from [sender == RADIO_CHANNEL_SYNDICATE ? "your benefactor" : "Central Command"]. Message as follows[sender == RADIO_CHANNEL_SYNDICATE ? ", agent." : ":"] [msg]. Message ends.\"") + + + log_admin("SubtlePM ([sender]): [key_name(adminClient)] -> [key_name(targetMob)] : [msg]") + msg = span_adminnotice(" SubtleMessage ([sender]): [key_name_admin(adminClient)] -> [key_name_admin(targetMob)] : [msg]") + message_admins(msg) + admin_ticket_log(targetMob, msg) if ("set_name") targetMob.vv_auto_rename(params["name"]) @@ -203,12 +240,12 @@ ADMIN_VERB_ONLY_CONTEXT_MENU(show_player_panel2, R_ADMIN, "Show Player Panel 2", if ("freeze") var/mob/living/L = targetMob if (istype(L)) - SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/toggle_admin_freeze, L) + L.toggle_admin_freeze(adminClient) if ("sleep") var/mob/living/L = targetMob if (istype(L)) - SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/toggle_admin_sleep, L) + L.toggle_admin_sleep(adminClient) if ("lobby") if(!isobserver(targetMob)) @@ -240,10 +277,8 @@ ADMIN_VERB_ONLY_CONTEXT_MENU(show_player_panel2, R_ADMIN, "Show Player Panel 2", targetMob.say(params["to_say"], forced="admin") if ("force_emote") - message_admins("force emote called") - targetMob.emote("me", EMOTE_VISIBLE|EMOTE_AUDIBLE, "Test 123") - targetMob.emote("me", EMOTE_VISIBLE|EMOTE_AUDIBLE, params["to_emote"]) - QUEUE_OR_CALL_VERB_FOR(VERB_CALLBACK(targetMob, TYPE_PROC_REF(/mob, emote), "me", EMOTE_VISIBLE|EMOTE_AUDIBLE, params["to_emote"]), SSspeech_controller) + if (params["to_emote"]) + QUEUE_OR_CALL_VERB_FOR(VERB_CALLBACK(targetMob, TYPE_PROC_REF(/mob, emote), "me", EMOTE_VISIBLE|EMOTE_AUDIBLE, params["to_emote"], TRUE), SSspeech_controller) if ("prison") if(isAI(targetMob)) @@ -412,3 +447,32 @@ ADMIN_VERB_ONLY_CONTEXT_MENU(show_player_panel2, R_ADMIN, "Show Player Panel 2", if ("skill_panel") SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/show_skill_panel, targetMob) + if ("commend") + if(!targetMob.ckey) + to_chat(adminClient, span_warning("This mob either no longer exists or no longer is being controlled by someone!")) + return + + switch(tgui_alert(adminMob, "Would you like the effects to apply immediately or at the end of the round? Applying them now will make it clear it was an admin commendation.", "<3?", list("Apply now", "Apply at round end", "Cancel"))) + if("Apply now") + targetMob.receive_heart(adminMob, instant = TRUE) + if("Apply at round end") + targetMob.receive_heart(adminMob) + + if ("play_sound_to") + var/soundFile = input("", "Select a sound file",) as null|sound + + if(soundFile && targetMob) + SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/play_direct_mob_sound, soundFile, targetMob) + + if ("apply_client_quirks") + var/mob/living/carbon/human/H = targetMob + if(!istype(H)) + to_chat(adminClient, "this can only be used on instances of type /mob/living/carbon/human.", confidential = TRUE) + return + if(!H.client) + to_chat(adminClient, "[H] has no client!", confidential = TRUE) + return + SSquirks.AssignQuirks(H, H.client) + log_admin("[key_name(adminClient)] applied client quirks to [key_name(H)].") + message_admins(span_adminnotice("[key_name_admin(adminClient)] applied client quirks to [key_name_admin(H)].")) + diff --git a/modular_zzplurt/code/modules/admin/verbs.dm b/modular_zzplurt/code/modules/admin/verbs.dm deleted file mode 100644 index 7548462ec5f69..0000000000000 --- a/modular_zzplurt/code/modules/admin/verbs.dm +++ /dev/null @@ -1,7 +0,0 @@ -ADMIN_VERB(toggle_admin_freeze, R_ADMIN, "Toggle Freeze Mob", "Freezes / unfreezes a mob's movement", ADMIN_CATEGORY_GAME, mob/living/target in world) - target.toggle_admin_freeze(user) - BLACKBOX_LOG_ADMIN_VERB("Toggle Freeze Mob") - -ADMIN_VERB(toggle_admin_sleep, R_ADMIN, "Toggle Sleep Mob", "Sleeps / unsleeps a mob", ADMIN_CATEGORY_GAME, mob/living/target in world) - target.toggle_admin_sleep(user) - BLACKBOX_LOG_ADMIN_VERB("Toggle Sleep Mob") diff --git a/modular_zzplurt/code/modules/client/preferences/player_panel.dm b/modular_zzplurt/code/modules/client/preferences/player_panel.dm new file mode 100644 index 0000000000000..b02ef40796b3e --- /dev/null +++ b/modular_zzplurt/code/modules/client/preferences/player_panel.dm @@ -0,0 +1,11 @@ +/datum/preference/toggle/use_tgui_player_panel + category = PREFERENCE_CATEGORY_GAME_PREFERENCES + savefile_key = "use_tgui_player_panel" + savefile_identifier = PREFERENCE_PLAYER + default_value = TRUE + +/datum/preference/toggle/use_tgui_player_panel/is_accessible(datum/preferences/preferences) + if (!..(preferences)) + return FALSE + + return is_admin(preferences.parent) diff --git a/modular_zzplurt/code/modules/mob/mob.dm b/modular_zzplurt/code/modules/mob/mob.dm new file mode 100644 index 0000000000000..ba7a76e8d41ba --- /dev/null +++ b/modular_zzplurt/code/modules/mob/mob.dm @@ -0,0 +1,12 @@ +/mob/proc/create_player_panel() + QDEL_NULL(mob_panel) + + mob_panel = new(src) + +/mob/Initialize() + . = ..() + create_player_panel() + +/mob/Destroy() + QDEL_NULL(mob_panel) + . = ..() diff --git a/modular_zzplurt/code/modules/mob/mob_defines.dm b/modular_zzplurt/code/modules/mob/mob_defines.dm new file mode 100644 index 0000000000000..86f47c2b7dad7 --- /dev/null +++ b/modular_zzplurt/code/modules/mob/mob_defines.dm @@ -0,0 +1,3 @@ +/mob + // Admin player panel for this mob + var/datum/player_panel/mob_panel diff --git a/tgstation.dme b/tgstation.dme index 01dfc2591e855..5d39f6e107ae7 100644 --- a/tgstation.dme +++ b/tgstation.dme @@ -9081,12 +9081,14 @@ #include "modular_zzplurt\code\modules\admin\jobban.dm" #include "modular_zzplurt\code\modules\admin\kick.dm" #include "modular_zzplurt\code\modules\admin\transform.dm" -#include "modular_zzplurt\code\modules\admin\verbs.dm" #include "modular_zzplurt\code\modules\client\client_procs.dm" +#include "modular_zzplurt\code\modules\client\preferences\player_panel.dm" #include "modular_zzplurt\code\modules\client\verbs\looc.dm" #include "modular_zzplurt\code\modules\client\verbs\ooc.dm" #include "modular_zzplurt\code\modules\discord\tgs_commands.dm" #include "modular_zzplurt\code\modules\discord\verbs.dm" +#include "modular_zzplurt\code\modules\mob\mob.dm" +#include "modular_zzplurt\code\modules\mob\mob_defines.dm" #include "modular_zzplurt\code\modules\mob\dead\new_player\new_player.dm" #include "modular_zzplurt\code\modules\admin\player_panel.dm" #include "modular_zzplurt\code\modules\mob\living\living.dm" diff --git a/tgui/packages/tgui/interfaces/PlayerPanel.tsx b/tgui/packages/tgui/interfaces/PlayerPanel.tsx index 8d3b85d9f885e..93098f8220e85 100644 --- a/tgui/packages/tgui/interfaces/PlayerPanel.tsx +++ b/tgui/packages/tgui/interfaces/PlayerPanel.tsx @@ -14,6 +14,7 @@ import { Section, Slider, Tabs, + Tooltip, } from '../components'; import { Window } from '../layouts'; @@ -23,6 +24,7 @@ type Data = { admin_mob_type: string; client_ckey: string; client_rank: string; + ranks: string; last_ckey: string; playtimes_enabled: boolean; playtime: string; @@ -36,6 +38,7 @@ type Data = { data_player_join_date: string; data_account_join_date: string; data_byond_version: string; + data_old_names: string; glob_mute_bits: { name: string; @@ -108,6 +111,7 @@ export const PlayerPanel = () => { mob_type, client_ckey, client_rank, + ranks, last_ckey, playtimes_enabled, playtime, @@ -183,21 +187,32 @@ export const PlayerPanel = () => { {client_ckey ? 'Client:' : 'Last client:'} - {client_ckey || last_ckey} + + + + {client_ckey || last_ckey} + + - {!client_ckey && last_ckey && ( - + {!client_ckey && !!last_ckey && ( - - )} + /> + )} + {!!client_ckey && ( @@ -432,18 +447,39 @@ const PhysicalActions = () => { > Quirks - + + + +
{ data_byond_version, data_player_join_date, data_account_join_date, + data_old_names, current_time, } = data; return ( @@ -815,6 +852,9 @@ const PunishmentActions = () => { {data_byond_version} + + {data_old_names} +
@@ -1078,10 +1118,20 @@ const OtherActions = () => { p=".5rem" mb=".5rem" textAlign="center" - disabled={!mob_type.includes('/mob/living')} - onClick={(e) => act('languages')} + disabled={!client_ckey} + onClick={(e) => act('job_exemption_panel')} + > + Job Exemption Panel + + diff --git a/tgui/packages/tgui/interfaces/PreferencesMenu/preferences/features/game_preferences/admin.tsx b/tgui/packages/tgui/interfaces/PreferencesMenu/preferences/features/game_preferences/admin.tsx index 7888a53e0038b..14d308928f7a4 100644 --- a/tgui/packages/tgui/interfaces/PreferencesMenu/preferences/features/game_preferences/admin.tsx +++ b/tgui/packages/tgui/interfaces/PreferencesMenu/preferences/features/game_preferences/admin.tsx @@ -6,6 +6,13 @@ import { } from '../base'; import { FeatureDropdownInput } from '../dropdowns'; +export const use_tgui_player_panel: FeatureToggle = { + name: 'Use modern player panel', + category: 'ADMIN', + description: 'Whether to use the new TGUI player panel or the old HTML one.', + component: CheckboxInput, +}; + export const asaycolor: Feature = { name: 'Admin chat color', category: 'ADMIN', From a35dfcafaeee9516e9fb7dbeb0356a6315b170b6 Mon Sep 17 00:00:00 2001 From: Casper Date: Mon, 8 Jul 2024 13:17:02 +0100 Subject: [PATCH 3/7] Cleanup --- code/modules/admin/topic.dm | 7 + code/modules/admin/verbs/admingame.dm | 3 - modular_zzplurt/code/modules/admin/ban.dm | 0 modular_zzplurt/code/modules/admin/jobban.dm | 179 ------------------ modular_zzplurt/code/modules/admin/kick.dm | 20 -- .../code/modules/admin/player_panel.dm | 16 +- tgstation.dme | 3 - 7 files changed, 22 insertions(+), 206 deletions(-) delete mode 100644 modular_zzplurt/code/modules/admin/ban.dm delete mode 100644 modular_zzplurt/code/modules/admin/jobban.dm delete mode 100644 modular_zzplurt/code/modules/admin/kick.dm diff --git a/code/modules/admin/topic.dm b/code/modules/admin/topic.dm index c80a3a581afb3..d8547e8afae4a 100644 --- a/code/modules/admin/topic.dm +++ b/code/modules/admin/topic.dm @@ -1032,6 +1032,13 @@ var/mob/M = locate(href_list["SyndicateReply"]) usr.client.admin_headset_message(M, RADIO_CHANNEL_SYNDICATE) + else if(href_list["HeadsetMessage"]) + if(!check_rights(R_ADMIN)) + return + + var/mob/M = locate(href_list["HeadsetMessage"]) + usr.client.admin_headset_message(M) + else if(href_list["reject_custom_name"]) if(!check_rights(R_ADMIN)) return diff --git a/code/modules/admin/verbs/admingame.dm b/code/modules/admin/verbs/admingame.dm index f8a20b02cb29c..de5b2eac68958 100644 --- a/code/modules/admin/verbs/admingame.dm +++ b/code/modules/admin/verbs/admingame.dm @@ -7,9 +7,6 @@ ADMIN_VERB_ONLY_CONTEXT_MENU(show_player_panel, R_ADMIN, "Show Player Panel", mo return var/useModern = user.prefs.read_preference(/datum/preference/toggle/use_tgui_player_panel) - - message_admins(useModern) - if (useModern) if(!player.mob_panel) player.create_player_panel() diff --git a/modular_zzplurt/code/modules/admin/ban.dm b/modular_zzplurt/code/modules/admin/ban.dm deleted file mode 100644 index e69de29bb2d1d..0000000000000 diff --git a/modular_zzplurt/code/modules/admin/jobban.dm b/modular_zzplurt/code/modules/admin/jobban.dm deleted file mode 100644 index c9d3577225c46..0000000000000 --- a/modular_zzplurt/code/modules/admin/jobban.dm +++ /dev/null @@ -1,179 +0,0 @@ -// // Stuff that helps the TGUI player panel jobban section to work - -// GLOBAL_LIST_INIT(jobban_panel_data, list( -// list( -// "name" = "Command", -// "color" = "yellow", -// "roles" = GLOB.command_positions -// ), -// list( -// "name" = "Security", -// "color" = "red", -// "roles" = GLOB.security_positions -// ), -// list( -// "name" = "Engineering", -// "color" = "orange", -// "roles" = GLOB.engineering_positions -// ), -// list( -// "name" = "Medical", -// "color" = "blue", -// "roles" = GLOB.medical_positions -// ), -// list( -// "name" = "Science", -// "color" = "violet", -// "roles" = GLOB.science_positions -// ), -// list( -// "name" = "Supply", -// "color" = "brown", -// "roles" = GLOB.supply_positions -// ), -// list( -// "name" = "Service", -// "color" = "green", -// "roles" = GLOB.civilian_positions -// ), -// list( -// "name" = "Silicon", -// "color" = "purple", -// "roles" = GLOB.nonhuman_positions -// ), -// list( -// "name" = "Ghost Roles", -// "color" = "teal", -// "roles" = list( -// ROLE_PAI, -// ROLE_POSIBRAIN, -// ROLE_DRONE, -// ROLE_DEATHSQUAD, -// ROLE_LAVALAND, -// ROLE_GHOSTCAFE, -// ROLE_SENTIENCE, -// ROLE_MIND_TRANSFER -// ) -// ), -// list( -// "name" = "Antagonists", -// "color" = "red", -// "roles" = list( -// ROLE_TRAITOR, -// ROLE_CHANGELING, -// ROLE_HERETIC, -// ROLE_OPERATIVE, -// ROLE_REV, -// ROLE_CULTIST, -// ROLE_SERVANT_OF_RATVAR, -// ROLE_WIZARD, -// ROLE_ABDUCTOR, -// ROLE_ALIEN, -// ROLE_FAMILIES, -// ROLE_BLOODSUCKER, -// ROLE_SLAVER -// ) -// ), -// list( -// "name" = "Other", -// "color" = "red", -// "roles" = list( -// "pacifist", -// "appearance", -// "emote", -// "OOC", -// ROLE_RESPAWN -// ) -// ) -// )) - -// // notbannedlist is just a list of strings of the job titles you want to ban. -// /datum/admins/proc/Jobban(mob/M, list/notbannedlist) -// if (!check_rights(R_BAN)) -// to_chat(usr, "Error: You do not have sufficient admin rights to ban players.") -// return - -// var/severity = null -// var/reason = null - -// switch(tgui_alert(usr, "Job ban type", buttons = list("Temporary", "Permanent", "Cancel"))) -// if("Temporary") -// var/mins = input(usr,"How long (in minutes)?","Ban time",1440) as num|null -// if(mins <= 0) -// to_chat(usr, span_danger("[mins] is not a valid duration.")) -// return -// reason = input(usr,"Please State Reason For Banning [M.key].","Reason") as message|null -// if(!reason) -// return -// severity = tgui_alert(usr, "Set the severity of the note/ban", buttons = list("High", "Medium", "Minor", "None")) -// if(!severity) -// return -// var/msg -// for(var/job in notbannedlist) -// if(!DB_ban_record(BANTYPE_JOB_TEMP, M, mins, reason, job)) -// to_chat(usr, span_danger("Failed to apply ban.")) -// return -// if(M.client) -// jobban_buildcache(M.client) -// ban_unban_log_save("[key_name(usr)] temp-jobbanned [key_name(M)] from [job] for [mins] minutes. reason: [reason]") -// log_admin_private("[key_name(usr)] temp-jobbanned [key_name(M)] from [job] for [mins] minutes.") -// if(!msg) -// msg = job -// else -// msg += ", [job]" -// create_message("note", M.key, null, "Banned from [msg] - [reason]", null, null, 0, 0, null, 0, severity) -// message_admins(span_adminnotice("[key_name_admin(usr)] banned [key_name_admin(M)] from [msg] for [mins] minutes.")) -// to_chat(M, span_boldannounce("You have been [((msg == "ooc") || (msg == "appearance") || (msg == "pacifist")) ? "banned" : "jobbanned"] by [usr.client.key] from: [msg == "pacifist" ? "using violence" : msg].")) -// to_chat(M, span_boldannounce("The reason is: [reason]")) -// to_chat(M, span_danger("This jobban will be lifted in [mins] minutes.")) - -// if("Permanent") -// reason = input(usr,"Please State Reason For Banning [M.key].","Reason") as message|null -// if (!reason) -// return -// severity = tgui_alert(usr, "Please State Reason For Banning", buttons = list("High", "Medium", "Minor", "None")) -// if (!severity) -// return - -// var/msg -// for(var/job in notbannedlist) -// if(!DB_ban_record(BANTYPE_JOB_PERMA, M, -1, reason, job)) -// to_chat(usr, span_danger("Failed to apply ban.")) -// return -// if(M.client) -// jobban_buildcache(M.client) -// ban_unban_log_save("[key_name(usr)] perma-jobbanned [key_name(M)] from [job]. reason: [reason]") -// log_admin_private("[key_name(usr)] perma-banned [key_name(M)] from [job]") -// if(!msg) -// msg = job -// else -// msg += ", [job]" -// create_message("note", M.key, null, "Banned from [msg] - [reason]", null, null, 0, 0, null, 0, severity) -// message_admins(span_adminnotice("[key_name_admin(usr)] banned [key_name_admin(M)] from [msg].")) -// to_chat(M, span_boldannounce("You have been [((msg == "ooc") || (msg == "appearance") || (msg == "pacifist")) ? "banned" : "jobbanned"] by [usr.client.key] from: [msg == "pacifist" ? "using violence" : msg].")) -// to_chat(M, span_boldannounce("The reason is: [reason]")) -// to_chat(M, span_danger("This jobban can be lifted only upon request.")) - -// // notbannedlist is just a list of strings of the job titles you want to unban. -// /datum/admins/proc/UnJobban(mob/M, list/bannedlist) - -// if (!check_rights(R_BAN)) -// to_chat(usr, "Error: You do not have sufficient admin rights to unban players.") -// return - -// var/msg -// for(var/job in bannedlist) -// var/reason = jobban_isbanned(M, job) -// if (tgui_alert(usr, "Job: '[job]' Ban reason: '[reason]'", "Un-Jobban This Player?", list("Yes", "No")) == "Yes") -// ban_unban_log_save("[key_name(usr)] unjobbanned [key_name(M)] from [job]") -// log_admin_private("[key_name(usr)] unbanned [key_name(M)] from [job]") -// DB_ban_unban(M.ckey, BANTYPE_ANY_JOB, job) -// if(M.client) -// jobban_buildcache(M.client) -// if(!msg) -// msg = job -// else -// msg += ", [job]" -// if(msg) -// message_admins(span_adminnotice("[key_name_admin(usr)] unbanned [key_name_admin(M)] from [msg].")) -// to_chat(M, span_boldannounce("You have been un-jobbanned by [usr.client.key] from [msg].")) diff --git a/modular_zzplurt/code/modules/admin/kick.dm b/modular_zzplurt/code/modules/admin/kick.dm deleted file mode 100644 index 4e4ace65f8cf2..0000000000000 --- a/modular_zzplurt/code/modules/admin/kick.dm +++ /dev/null @@ -1,20 +0,0 @@ -ADMIN_VERB(kick, R_ADMIN, "Kick", "Kicks a client from the server.", ADMIN_CATEGORY_GAME, mob/target in world) - if(!ismob(target)) - to_chat(usr, span_danger("Error: Target is not a mob. Please try again.")) - return - if(!check_if_greater_rights_than(target.client)) - to_chat(usr, span_danger("Error: They have more rights than you do."), confidential = TRUE) - return - if(tgui_alert(usr, "Kick [key_name(target)]?", "Confirm", list("Yes", "No")) != "Yes") - return - if(!target) - to_chat(usr, span_danger("Error: [target] no longer exists!"), confidential = TRUE) - return - if(!target.client) - to_chat(usr, span_danger("Error: [target] no longer has a client!"), confidential = TRUE) - return - to_chat(target, span_danger("You have been kicked from the server by [usr.client.holder.fakekey ? "an Administrator" : "[usr.client.key]"]."), confidential = TRUE) - log_admin("[key_name(usr)] kicked [key_name(target)].") - message_admins(span_adminnotice("[key_name_admin(usr)] kicked [key_name_admin(target)].")) - qdel(target.client) - BLACKBOX_LOG_ADMIN_VERB("Kick") diff --git a/modular_zzplurt/code/modules/admin/player_panel.dm b/modular_zzplurt/code/modules/admin/player_panel.dm index f7b7c73b43fbb..bcebc8a2493fa 100644 --- a/modular_zzplurt/code/modules/admin/player_panel.dm +++ b/modular_zzplurt/code/modules/admin/player_panel.dm @@ -292,7 +292,21 @@ GLOBAL_LIST_INIT(pp_limbs, list( message_admins("[key_name_admin(adminClient)] has sent [key_name_admin(targetMob)] to Prison!") if ("kick") - SSadmin_verbs.dynamic_invoke_verb(adminClient, /datum/admin_verb/kick, targetMob) + if(!check_if_greater_rights_than(targetClient)) + to_chat(adminClient, span_danger("Error: They have more rights than you do."), confidential = TRUE) + return + if(tgui_alert(adminMob, "Kick [key_name(targetMob)]?", "Confirm", list("Yes", "No")) != "Yes") + return + if(!targetMob) + to_chat(adminClient, span_danger("Error: [targetMob] no longer exists!"), confidential = TRUE) + return + if(!targetClient) + to_chat(adminClient, span_danger("Error: [targetMob] no longer has a client!"), confidential = TRUE) + return + to_chat(targetMob, span_danger("You have been kicked from the server by [adminClient.holder.fakekey ? "an Administrator" : "[adminClient.key]"]."), confidential = TRUE) + log_admin("[key_name(adminClient)] kicked [key_name(targetMob)].") + message_admins(span_adminnotice("[key_name_admin(adminClient)] kicked [key_name_admin(targetMob)].")) + qdel(targetClient) if ("ban") var/player_key = targetMob.key diff --git a/tgstation.dme b/tgstation.dme index 5d39f6e107ae7..27b98fda14b55 100644 --- a/tgstation.dme +++ b/tgstation.dme @@ -9077,9 +9077,6 @@ #include "modular_zzplurt\code\_globalvars\mobs.dm" #include "modular_zzplurt\code\controllers\configuration\entries\discord.dm" #include "modular_zzplurt\code\controllers\subsystem\discord.dm" -#include "modular_zzplurt\code\modules\admin\ban.dm" -#include "modular_zzplurt\code\modules\admin\jobban.dm" -#include "modular_zzplurt\code\modules\admin\kick.dm" #include "modular_zzplurt\code\modules\admin\transform.dm" #include "modular_zzplurt\code\modules\client\client_procs.dm" #include "modular_zzplurt\code\modules\client\preferences\player_panel.dm" From ac1721809eeacc2ce956666d46e8826ade14a0bf Mon Sep 17 00:00:00 2001 From: Casper Date: Mon, 8 Jul 2024 13:38:03 +0100 Subject: [PATCH 4/7] Cleanup 2 --- .../code/modules/admin/player_panel.dm | 5 ++--- tgui/packages/tgui/interfaces/PlayerPanel.tsx | 22 ++++++++++--------- 2 files changed, 14 insertions(+), 13 deletions(-) diff --git a/modular_zzplurt/code/modules/admin/player_panel.dm b/modular_zzplurt/code/modules/admin/player_panel.dm index bcebc8a2493fa..053866ce6769f 100644 --- a/modular_zzplurt/code/modules/admin/player_panel.dm +++ b/modular_zzplurt/code/modules/admin/player_panel.dm @@ -36,7 +36,7 @@ GLOBAL_LIST_INIT(pp_limbs, list( ui = SStgui.try_update_ui(user, src, ui) if (!ui) - ui = new(user, src, "PlayerPanel", "[targetMob.name] Player Panel") + ui = new(user, src, "PlayerPanel", "[targetMob.real_name] Player Panel") ui.open() /datum/player_panel/ui_state(mob/user) @@ -110,7 +110,7 @@ GLOBAL_LIST_INIT(pp_limbs, list( var/client/adminClient = adminMob.client if(. || !check_rights_for(adminClient, R_ADMIN)) - message_admins(span_adminhelp("WARNING: NON-ADMIN [ADMIN_LOOKUPFLW(adminMob)] ATTEMPTED TO ACCESS ADMIN PANEL. WARN Casper3044.")) + message_admins(span_adminhelp("WARNING: NON-ADMIN [ADMIN_LOOKUPFLW(adminMob)] ATTEMPTED TO ACCESS ADMIN PANEL. NOTIFY Casper3044.")) to_chat(adminClient, "Error: you are not an admin!") return @@ -414,7 +414,6 @@ GLOBAL_LIST_INIT(pp_limbs, list( var/mob/living/L = targetMob if(!isnull(params["new_scale"]) && istype(L)) L.vv_edit_var("current_size", params["new_scale"]) - // mobSize = params["new_scale"] if ("explode") var/power = text2num(params["power"]) diff --git a/tgui/packages/tgui/interfaces/PlayerPanel.tsx b/tgui/packages/tgui/interfaces/PlayerPanel.tsx index 93098f8220e85..de11517e5a9ea 100644 --- a/tgui/packages/tgui/interfaces/PlayerPanel.tsx +++ b/tgui/packages/tgui/interfaces/PlayerPanel.tsx @@ -118,7 +118,7 @@ export const PlayerPanel = () => { } = data; return ( - +
@@ -468,7 +468,7 @@ const PhysicalActions = () => { @@ -476,7 +476,7 @@ const PhysicalActions = () => { width="100%" height="100%" icon="comment-dots" - onClick={(e) => act('languages')} + onClick={() => act('languages')} > Languages @@ -1069,7 +1069,7 @@ const FunActions = () => { p=".5rem" textAlign="center" disabled={!narrateMessage} - onClick={(e) => + onClick={() => act('narrate', { message: narrateMessage, classes: narrateStyles, @@ -1109,7 +1109,7 @@ const OtherActions = () => { mb=".5rem" textAlign="center" disabled={!client_ckey} - onClick={(e) => act('traitor_panel')} + onClick={() => act('traitor_panel')} > Traitor Panel @@ -1119,7 +1119,7 @@ const OtherActions = () => { mb=".5rem" textAlign="center" disabled={!client_ckey} - onClick={(e) => act('job_exemption_panel')} + onClick={() => act('job_exemption_panel')} > Job Exemption Panel @@ -1129,7 +1129,7 @@ const OtherActions = () => { mb=".5rem" textAlign="center" disabled={!client_ckey} - onClick={(e) => act('commend')} + onClick={() => act('commend')} > Commend Behavior @@ -1139,7 +1139,7 @@ const OtherActions = () => { mb=".5rem" textAlign="center" disabled={!client_ckey} - onClick={(e) => act('play_sound_to')} + onClick={() => act('play_sound_to')} > Play Sound To @@ -1148,8 +1148,10 @@ const OtherActions = () => { p=".5rem" mb=".5rem" textAlign="center" - disabled={!client_ckey} - onClick={(e) => act('apply_client_quirks')} + disabled={ + !client_ckey || !mob_type.includes('/mob/living/carbon/human') + } + onClick={() => act('apply_client_quirks')} > Apply Client Quirks From ea25694031255fc4a829dae587dadfa64403bb78 Mon Sep 17 00:00:00 2001 From: Casper Date: Mon, 8 Jul 2024 13:54:48 +0100 Subject: [PATCH 5/7] Lint fix --- tgstation.dme | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tgstation.dme b/tgstation.dme index 27b98fda14b55..8814e55d483af 100644 --- a/tgstation.dme +++ b/tgstation.dme @@ -9077,6 +9077,7 @@ #include "modular_zzplurt\code\_globalvars\mobs.dm" #include "modular_zzplurt\code\controllers\configuration\entries\discord.dm" #include "modular_zzplurt\code\controllers\subsystem\discord.dm" +#include "modular_zzplurt\code\modules\admin\player_panel.dm" #include "modular_zzplurt\code\modules\admin\transform.dm" #include "modular_zzplurt\code\modules\client\client_procs.dm" #include "modular_zzplurt\code\modules\client\preferences\player_panel.dm" @@ -9087,7 +9088,6 @@ #include "modular_zzplurt\code\modules\mob\mob.dm" #include "modular_zzplurt\code\modules\mob\mob_defines.dm" #include "modular_zzplurt\code\modules\mob\dead\new_player\new_player.dm" -#include "modular_zzplurt\code\modules\admin\player_panel.dm" #include "modular_zzplurt\code\modules\mob\living\living.dm" #include "modular_zzplurt\code\modules\mob\living\living_defines.dm" // END_INCLUDE From 652bc6ae639a9c496b414dd14c31f3df0ed6529a Mon Sep 17 00:00:00 2001 From: Casper Date: Tue, 9 Jul 2024 21:18:41 +0100 Subject: [PATCH 6/7] . --- modular_zzplurt/code/modules/admin/transform.dm | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modular_zzplurt/code/modules/admin/transform.dm b/modular_zzplurt/code/modules/admin/transform.dm index f6158e3683870..cc18b8e81ca24 100644 --- a/modular_zzplurt/code/modules/admin/transform.dm +++ b/modular_zzplurt/code/modules/admin/transform.dm @@ -118,7 +118,7 @@ GLOBAL_LIST_INIT(pp_transformables, list( newTypeName = newType log_admin("[key_name(usr)] transformed [key_name(M)] into a [newTypeName].") - message_admins("[key_name_admin(usr)] transformed [key_name_admin(M)] into a [newTypeName].") + message_admins(span_adminnotice("[key_name_admin(usr)] transformed [key_name_admin(M)] into a [newTypeName].")) var/mob/newMob = M.change_mob_type(newType, delete_old_mob = TRUE) From ef2e066b4b0311dc6faaa2b1ad9a555ba949cbf5 Mon Sep 17 00:00:00 2001 From: Casper Date: Thu, 11 Jul 2024 09:41:02 +0100 Subject: [PATCH 7/7] Modularisation edits. --- code/modules/admin/verbs/admingame.dm | 2 ++ tgui/packages/tgui/components/Collapsible.tsx | 11 ++++++++++- .../preferences/features/game_preferences/admin.tsx | 2 ++ 3 files changed, 14 insertions(+), 1 deletion(-) diff --git a/code/modules/admin/verbs/admingame.dm b/code/modules/admin/verbs/admingame.dm index de5b2eac68958..1713059a922f9 100644 --- a/code/modules/admin/verbs/admingame.dm +++ b/code/modules/admin/verbs/admingame.dm @@ -6,12 +6,14 @@ ADMIN_VERB_ONLY_CONTEXT_MENU(show_player_panel, R_ADMIN, "Show Player Panel", mo to_chat(user, span_warning("You seem to be selecting a mob that doesn't exist anymore."), confidential = TRUE) return + // SPLURT EDIT START var/useModern = user.prefs.read_preference(/datum/preference/toggle/use_tgui_player_panel) if (useModern) if(!player.mob_panel) player.create_player_panel() player.mob_panel.ui_interact(user.mob) return + // SPLURT EDIT END log_admin("[key_name(user)] checked the individual player panel for [key_name(player)][isobserver(user.mob)?"":" while in game"].") diff --git a/tgui/packages/tgui/components/Collapsible.tsx b/tgui/packages/tgui/components/Collapsible.tsx index e2dc046c95fa1..712ed77e72387 100644 --- a/tgui/packages/tgui/components/Collapsible.tsx +++ b/tgui/packages/tgui/components/Collapsible.tsx @@ -14,7 +14,9 @@ type Props = Partial<{ open: boolean; title: ReactNode; icon: string; + // SPLURT EDIT START disabled: boolean; + // SPLURT EDIT END }> & BoxProps; @@ -30,7 +32,9 @@ export function Collapsible(props: Props) { fluid color={color} icon={icon ? icon : open ? 'chevron-down' : 'chevron-right'} + // SPLURT EDIT START disabled={disabled} + // SPLURT EDIT END onClick={() => setOpen(!open)} {...rest} > @@ -41,7 +45,12 @@ export function Collapsible(props: Props) {
{buttons}
)} - {open && !disabled && {children}} + {open && + // SPLURT EDIT START + !disabled && ( + // SPLURT EDIT END + {children} + )} ); } diff --git a/tgui/packages/tgui/interfaces/PreferencesMenu/preferences/features/game_preferences/admin.tsx b/tgui/packages/tgui/interfaces/PreferencesMenu/preferences/features/game_preferences/admin.tsx index 14d308928f7a4..6307c635dbb9e 100644 --- a/tgui/packages/tgui/interfaces/PreferencesMenu/preferences/features/game_preferences/admin.tsx +++ b/tgui/packages/tgui/interfaces/PreferencesMenu/preferences/features/game_preferences/admin.tsx @@ -6,12 +6,14 @@ import { } from '../base'; import { FeatureDropdownInput } from '../dropdowns'; +// SPLURT EDIT START export const use_tgui_player_panel: FeatureToggle = { name: 'Use modern player panel', category: 'ADMIN', description: 'Whether to use the new TGUI player panel or the old HTML one.', component: CheckboxInput, }; +// SPLURT EDIT END export const asaycolor: Feature = { name: 'Admin chat color',