From c14b84ca321e20b9d8b4012ba7aaccf50eff2541 Mon Sep 17 00:00:00 2001 From: data-bomb Date: Wed, 20 Mar 2024 07:05:26 -0700 Subject: [PATCH] Switching to Unity-Friendly Timers - Removes System.Timers from Admin-Mod - Removes System.Timers from Announcements --- Si_AdminMod/ChatVotes.cs | 101 ++++++++++++++++----------- Si_Announcements/Si_Announcements.cs | 63 +++++++++-------- 2 files changed, 95 insertions(+), 69 deletions(-) diff --git a/Si_AdminMod/ChatVotes.cs b/Si_AdminMod/ChatVotes.cs index 1c32945..65ed548 100644 --- a/Si_AdminMod/ChatVotes.cs +++ b/Si_AdminMod/ChatVotes.cs @@ -25,11 +25,12 @@ You should have received a copy of the GNU General Public License using Il2Cpp; #endif +using HarmonyLib; using MelonLoader; using System; using System.Collections.Generic; using System.Linq; -using System.Timers; +using UnityEngine; namespace SilicaAdminMod { @@ -41,7 +42,7 @@ public static class ChatVotes public delegate void VoteHandler(ChatVoteResults results); - private static Timer? Timer_VoteDuration; + private static float Timer_TallyVote; static HelperMethods.CommandCallback voteChatCallback = Command_VoteChat; @@ -123,52 +124,74 @@ public static void HoldVote(ChatVoteBallot ballot) private static void StartVoteDurationTimer() { - double interval = SiAdminMod.Pre_Admin_VoteDuration.Value * 1000.0f; - MelonLogger.Msg("Starting vote timer with duration " + interval.ToString()); - Timer_VoteDuration = new Timer(interval); + MelonLogger.Msg("Starting vote timer with duration " + SiAdminMod.Pre_Admin_VoteDuration.Value.ToString()); voteInProgress = true; - Timer_VoteDuration.Elapsed += new ElapsedEventHandler(HandleVoteTimerExpired); - Timer_VoteDuration.AutoReset = false; - Timer_VoteDuration.Enabled = true; + HelperMethods.StartTimer(ref Timer_TallyVote); } - // don't access anything from melonloader inside the timer callback - private static void HandleVoteTimerExpired(object? source, ElapsedEventArgs e) +#if NET6_0 + [HarmonyPatch(typeof(MusicJukeboxHandler), nameof(MusicJukeboxHandler.Update))] +#else + [HarmonyPatch(typeof(MusicJukeboxHandler), "Update")] +#endif + private static class ApplyPatch_MusicJukeboxHandlerUpdate { - - if (currentVoteResults == null) - { - MelonLogger.Warning("Current vote results unavailable for timer expiration."); - voteInProgress = false; - return; - } - - OptionVoteResult winningResult = new OptionVoteResult - { - Votes = -1, - Command = "invalid" - }; - - foreach (OptionVoteResult currentVoteResult in currentVoteResults.DetailedResults) + private static void Postfix(MusicJukeboxHandler __instance) { - // determine winner - if (currentVoteResult.Votes > winningResult.Votes) + try { - winningResult = currentVoteResult; - } - MelonLogger.Msg("Found command " + currentVoteResult.Command + " with votes " + currentVoteResult.Votes); + // check if timer expired while the game is in-progress + if (HelperMethods.IsTimerActive(Timer_TallyVote)) + { + Timer_TallyVote += Time.deltaTime; + + if (Timer_TallyVote >= SiAdminMod.Pre_Admin_VoteDuration.Value) + { + Timer_TallyVote = HelperMethods.Timer_Inactive; + + if (currentVoteResults == null) + { + MelonLogger.Warning("Current vote results unavailable for timer expiration."); + voteInProgress = false; + return; + } + + OptionVoteResult winningResult = new OptionVoteResult + { + Votes = -1, + Command = "invalid" + }; + + foreach (OptionVoteResult currentVoteResult in currentVoteResults.DetailedResults) + { + // determine winner + if (currentVoteResult.Votes > winningResult.Votes) + { + winningResult = currentVoteResult; + } + + MelonLogger.Msg("Found command " + currentVoteResult.Command + " with votes " + currentVoteResult.Votes); + + // unregister commands for current vote + PlayerMethods.UnregisterPlayerPhrase(currentVoteResult.Command); + } + + // assign winner + currentVoteResults.WinningCommand = winningResult.Command; + + // call the vote handler + voteInProgress = false; + currentVoteResults.VoteHandler(currentVoteResults); + } - // unregister commands for current vote - PlayerMethods.UnregisterPlayerPhrase(currentVoteResult.Command); + } + } + catch (Exception exception) + { + HelperMethods.PrintError(exception, "Failed in MusicJukeboxHandler::Update"); + } } - - // assign winner - currentVoteResults.WinningCommand = winningResult.Command; - - // call the vote handler - voteInProgress = false; - currentVoteResults.VoteHandler(currentVoteResults); } } } \ No newline at end of file diff --git a/Si_Announcements/Si_Announcements.cs b/Si_Announcements/Si_Announcements.cs index 3af98ce..28810f7 100644 --- a/Si_Announcements/Si_Announcements.cs +++ b/Si_Announcements/Si_Announcements.cs @@ -29,14 +29,14 @@ You should have received a copy of the GNU General Public License using MelonLoader; using MelonLoader.Utils; using Si_Announcements; -using System.Timers; using System; using System.IO; using System.Collections.Generic; using SilicaAdminMod; using System.Linq; +using UnityEngine; -[assembly: MelonInfo(typeof(Announcements), "Server Announcements", "1.1.7", "databomb", "https://github.com/data-bomb/Silica")] +[assembly: MelonInfo(typeof(Announcements), "Server Announcements", "1.1.8", "databomb", "https://github.com/data-bomb/Silica")] [assembly: MelonGame("Bohemia Interactive", "Silica")] [assembly: MelonOptionalDependencies("Admin Mod")] @@ -48,11 +48,12 @@ public class Announcements : MelonMod static MelonPreferences_Entry _Announcements_SecondsBetweenMessages = null!; static MelonPreferences_Entry _Announcements_ShowIfLastChatWasAnnouncement = null!; - static Timer announcementTimer = null!; + static readonly float Timer_Inactive = -123.0f; + + static float Timer_Announcement = Timer_Inactive; static int announcementCount; static string[]? announcementsText; static string? lastChatMessage; - static bool timerExpired; public override void OnInitializeMelon() { @@ -89,13 +90,6 @@ public override void OnInitializeMelon() } MelonLogger.Msg("Loaded announcements.txt file with " + announcementsText.Length + " announcements."); - - double interval = _Announcements_SecondsBetweenMessages.Value * 1000.0f; - announcementTimer = new System.Timers.Timer(interval); - announcementTimer.Elapsed += new ElapsedEventHandler(TimerCallbackAnnouncement); - announcementTimer.AutoReset = true; - announcementTimer.Enabled = true; - } catch (Exception exception) { @@ -106,6 +100,8 @@ public override void OnInitializeMelon() #if NET6_0 public override void OnLateInitializeMelon() { + HelperMethods.StartTimer(ref Timer_Announcement); + bool QListLoaded = RegisteredMelons.Any(m => m.Info.Name == "QList"); if (!QListLoaded) { @@ -122,11 +118,6 @@ public override void OnLateInitializeMelon() } #endif - private static void TimerCallbackAnnouncement(object? source, ElapsedEventArgs e) - { - timerExpired = true; - } - #if NET6_0 [HarmonyPatch(typeof(MusicJukeboxHandler), nameof(MusicJukeboxHandler.Update))] #else @@ -138,30 +129,42 @@ private static void Postfix(MusicJukeboxHandler __instance) { try { + // check if timer expired while the game is in-progress - if (GameMode.CurrentGameMode.GameOngoing == true && timerExpired == true) + if (HelperMethods.IsTimerActive(Timer_Announcement)) { - timerExpired = false; + Timer_Announcement += Time.deltaTime; - if (announcementsText == null) + if (Timer_Announcement >= _Announcements_SecondsBetweenMessages.Value) { - return; - } + Timer_Announcement = Timer_Inactive; - if (!_Announcements_ShowIfLastChatWasAnnouncement.Value) - { - // check if the last chat message was an announcement - if (IsPreviousChatMessageAnnouncement(lastChatMessage)) + if (announcementsText == null) { - MelonLogger.Msg("Skipping Announcement - Repeated Message"); return; } - } - string nextAnnouncement = GetNextAnnouncement(); + // skip if game is not ongoign + if (!GameMode.CurrentGameMode.GameOngoing) + { + return; + } - Player broadcastPlayer = HelperMethods.FindBroadcastPlayer(); - broadcastPlayer.SendChatMessage(nextAnnouncement); + if (!_Announcements_ShowIfLastChatWasAnnouncement.Value) + { + // check if the last chat message was an announcement + if (IsPreviousChatMessageAnnouncement(lastChatMessage)) + { + MelonLogger.Msg("Skipping Announcement - Repeated Message"); + return; + } + } + + string nextAnnouncement = GetNextAnnouncement(); + + Player broadcastPlayer = HelperMethods.FindBroadcastPlayer(); + broadcastPlayer.SendChatMessage(nextAnnouncement); + } } } catch (Exception exception)