diff --git a/EXILED/Exiled.API/Features/Player.cs b/EXILED/Exiled.API/Features/Player.cs index d5e3d3eae..c6cd27514 100644 --- a/EXILED/Exiled.API/Features/Player.cs +++ b/EXILED/Exiled.API/Features/Player.cs @@ -589,9 +589,18 @@ public PlayerPermissions RemoteAdminPermissions public Role Role { get => role ??= Role.Create(RoleManager.CurrentRole); - internal set => role = value; + internal set + { + PreviousRole = role.Type; + role = value; + } } + /// + /// Gets the role that player had before changing role. + /// + public RoleTypeId PreviousRole { get; private set; } + /// /// Gets or sets the player's SCP preferences. /// diff --git a/EXILED/Exiled.Events/EventArgs/Player/EscapedEventArgs.cs b/EXILED/Exiled.Events/EventArgs/Player/EscapedEventArgs.cs new file mode 100644 index 000000000..f185f7845 --- /dev/null +++ b/EXILED/Exiled.Events/EventArgs/Player/EscapedEventArgs.cs @@ -0,0 +1,70 @@ +// ----------------------------------------------------------------------- +// +// Copyright (c) Exiled Team. All rights reserved. +// Licensed under the CC BY-SA 3.0 license. +// +// ----------------------------------------------------------------------- + +namespace Exiled.Events.EventArgs.Player +{ + using System; + using System.Collections.Generic; + + using Exiled.API.Enums; + using Exiled.API.Features; + using Exiled.API.Features.Roles; + using Exiled.Events.EventArgs.Interfaces; + using PlayerRoles; + using Respawning; + + /// + /// Contains all information after player has escaped. + /// + public class EscapedEventArgs : IPlayerEvent + { + /// + /// Initializes a new instance of the class. + /// + /// + /// + /// + /// , . + public EscapedEventArgs(Player player, EscapeScenario escapeScenario, Role role, KeyValuePair kvp) + { + Player = player; + EscapeScenario = escapeScenario; + Team = kvp.Key; + Tickets = kvp.Value; + OldRole = role; + EscapeTime = (int)Math.Ceiling(role.ActiveTime.TotalSeconds); + } + + /// + public Player Player { get; } + + /// + /// Gets the type of escape. + /// + public EscapeScenario EscapeScenario { get; } + + /// + /// Gets the that gained tickets for this escape. + /// + public SpawnableTeamType Team { get; } + + /// + /// Gets the amount of tickets gained for this escape. + /// + public float Tickets { get; } + + /// + /// Gets the previous role for this player. + /// + public Role OldRole { get; } + + /// + /// Gets the time in seconds since round started. + /// + public int EscapeTime { get; } + } +} \ No newline at end of file diff --git a/EXILED/Exiled.Events/Handlers/Player.cs b/EXILED/Exiled.Events/Handlers/Player.cs index 2e6209741..576cc3bb0 100644 --- a/EXILED/Exiled.Events/Handlers/Player.cs +++ b/EXILED/Exiled.Events/Handlers/Player.cs @@ -239,6 +239,11 @@ public class Player /// public static Event Escaping { get; set; } = new(); + /// + /// Invoked after a escapes. + /// + public static Event Escaped { get; set; } = new(); + /// /// Invoked before a begins speaking to the intercom. /// @@ -734,6 +739,12 @@ public class Player /// The instance. public static void OnEscaping(EscapingEventArgs ev) => Escaping.InvokeSafely(ev); + /// + /// Called after a escapes. + /// + /// The instance. + public static void OnEscaped(EscapedEventArgs ev) => Escaped.InvokeSafely(ev); + /// /// Called before a begins speaking to the intercom. /// diff --git a/EXILED/Exiled.Events/Patches/Events/Player/Escaping.cs b/EXILED/Exiled.Events/Patches/Events/Player/EscapingAndEscaped.cs similarity index 74% rename from EXILED/Exiled.Events/Patches/Events/Player/Escaping.cs rename to EXILED/Exiled.Events/Patches/Events/Player/EscapingAndEscaped.cs index 048de1136..a48e531d9 100644 --- a/EXILED/Exiled.Events/Patches/Events/Player/Escaping.cs +++ b/EXILED/Exiled.Events/Patches/Events/Player/EscapingAndEscaped.cs @@ -1,5 +1,5 @@ // ----------------------------------------------------------------------- -// +// // Copyright (c) Exiled Team. All rights reserved. // Licensed under the CC BY-SA 3.0 license. // @@ -8,6 +8,7 @@ namespace Exiled.Events.Patches.Events.Player { #pragma warning disable SA1402 // File may only contain a single type +#pragma warning disable IDE0060 using System; using System.Collections.Generic; @@ -18,8 +19,8 @@ namespace Exiled.Events.Patches.Events.Player using API.Enums; using API.Features; using API.Features.Pools; - using EventArgs.Player; + using Exiled.API.Features.Roles; using Exiled.Events.Attributes; using HarmonyLib; using PlayerRoles.FirstPersonControl; @@ -28,11 +29,12 @@ namespace Exiled.Events.Patches.Events.Player using static HarmonyLib.AccessTools; /// - /// Patches for . + /// Patches for and . /// [EventPatch(typeof(Handlers.Player), nameof(Handlers.Player.Escaping))] + [EventPatch(typeof(Handlers.Player), nameof(Handlers.Player.Escaped))] [HarmonyPatch(typeof(Escape), nameof(Escape.ServerHandlePlayer))] - internal static class Escaping + internal static class EscapingAndEscaped { private static IEnumerable Transpiler(IEnumerable instructions, ILGenerator generator) { @@ -42,6 +44,7 @@ private static IEnumerable Transpiler(IEnumerable instruction.opcode == OpCodes.Newobj) + offset; @@ -97,9 +100,44 @@ private static IEnumerable Transpiler(IEnumerable x.Is(OpCodes.Stfld, Field(typeof(Escape.EscapeMessage), nameof(Escape.EscapeMessage.EscapeTime)))) + offset; + + newInstructions.InsertRange(index, new CodeInstruction[] + { + // role = ev.Player.Role + new(OpCodes.Ldloc_S, ev.LocalIndex), + new(OpCodes.Callvirt, PropertyGetter(typeof(EscapingEventArgs), nameof(EscapingEventArgs.Player))), + new(OpCodes.Callvirt, PropertyGetter(typeof(Player), nameof(Player.Role))), + new(OpCodes.Stloc_S, role.LocalIndex), + }); + + newInstructions.InsertRange(newInstructions.Count - 1, new CodeInstruction[] + { + // ev.Player + new(OpCodes.Ldloc_S, ev.LocalIndex), + new(OpCodes.Callvirt, PropertyGetter(typeof(EscapingEventArgs), nameof(EscapingEventArgs.Player))), + + // escapeScenario + new(OpCodes.Ldloc_1), + + // role + new(OpCodes.Ldloc_S, role.LocalIndex), + + // ev.RespawnTickets.Key + new(OpCodes.Ldloc_S, ev.LocalIndex), + new(OpCodes.Callvirt, PropertyGetter(typeof(EscapingEventArgs), nameof(EscapingEventArgs.RespawnTickets))), + + // EscapedEventArgs ev2 = new(ev.Player, ev.EscapeScenario, role, float, SpawnableTeamType); + new(OpCodes.Newobj, GetDeclaredConstructors(typeof(EscapedEventArgs))[0]), + + // Handlers.Player.OnEscaped(ev); + new(OpCodes.Call, Method(typeof(Handlers.Player), nameof(Handlers.Player.OnEscaped))), + }); + newInstructions[newInstructions.Count - 1].WithLabels(returnLabel); for (int z = 0; z < newInstructions.Count; z++) @@ -120,6 +158,7 @@ private static void GrantAllTickets(EscapingEventArgs ev) /// Replaces last returned to . /// [EventPatch(typeof(Handlers.Player), nameof(Handlers.Player.Escaping))] + [EventPatch(typeof(Handlers.Player), nameof(Handlers.Player.Escaped))] [HarmonyPatch(typeof(Escape), nameof(Escape.ServerGetScenario))] internal static class GetScenario { @@ -151,4 +190,4 @@ private static IEnumerable Transpiler(IEnumerable.Pool.Return(newInstructions); } } -} +} \ No newline at end of file