using ABI_RC.Core.InteractionSystem; using ABI_RC.Core.IO; using ABI_RC.Core.Player; using cohtml; using HarmonyLib; using MelonLoader; using System.Collections; using UnityEngine; namespace NAK.FuckMetrics; public class FuckMetrics : MelonMod { internal static MelonLogger.Instance Logger; public static readonly MelonPreferences_Category CategoryFuckMetrics = MelonPreferences.CreateCategory(nameof(FuckMetrics)); public static readonly MelonPreferences_Entry EntryDisableCohtmlViewOnIdle = CategoryFuckMetrics.CreateEntry("Disable CohtmlView On Idle", false, description: "Disables CohtmlView on the menus when idle. Takes up to 6 seconds after menu exit. This can give a huge performance boost."); public static readonly MelonPreferences_Entry EntryDisableMetrics = CategoryFuckMetrics.CreateEntry("Menu Metrics", FuckMetrics.SettingState.MenuOnly, description: "Disables menu metrics (FPS & Ping). Updates once on menu open if disabled."); public static readonly MelonPreferences_Entry EntryDisableCoreUpdates = CategoryFuckMetrics.CreateEntry("Menu Core Updates", FuckMetrics.SettingState.MenuOnly, description: "Disables menu core updates (Gamerule Icons & Debug Status). Updates once on menu open if disabled."); public static readonly MelonPreferences_Entry EntryMetricsUpdateRate = CategoryFuckMetrics.CreateEntry("Metrics Update Rate", 1f, description: "Sets the update rate for the menu metrics. Default is 0.5f. Recommended to be 1f or higher."); public static readonly MelonPreferences_Entry EntryCoreUpdateRate = CategoryFuckMetrics.CreateEntry("Core Update Rate", 2f, description: "Sets the update rate for the menu core updates. Default is 0.1f. Recommended to be 2f or higher."); public override void OnInitializeMelon() { Logger = LoggerInstance; EntryDisableMetrics.OnEntryValueChangedUntyped.Subscribe(OnDisableMetrics); EntryDisableCoreUpdates.OnEntryValueChangedUntyped.Subscribe(OnDisableCoreUpdates); EntryMetricsUpdateRate.OnEntryValueChangedUntyped.Subscribe(OnChangeMetricsUpdateRate); EntryCoreUpdateRate.OnEntryValueChangedUntyped.Subscribe(OnChangeCoreUpdateRate); ApplyPatches(typeof(HarmonyPatches.CVR_MenuManagerPatches)); ApplyPatches(typeof(HarmonyPatches.ViewManagerPatches)); ApplyPatches(typeof(HarmonyPatches.CohtmlViewPatches)); MelonCoroutines.Start(WaitForLocalPlayer()); } private IEnumerator WaitForLocalPlayer() { while (PlayerSetup.Instance == null) yield return null; InitializeSettings(); } private void InitializeSettings() { FuckMetrics.ToggleMetrics(false); FuckMetrics.ToggleCoreUpdates(false); FuckMetrics.ToggleMetrics(EntryDisableMetrics.Value == FuckMetrics.SettingState.Always); FuckMetrics.ToggleCoreUpdates(EntryDisableCoreUpdates.Value == FuckMetrics.SettingState.Always); } private void OnDisableMetrics(object arg1, object arg2) { FuckMetrics.ToggleMetrics(EntryDisableMetrics.Value != FuckMetrics.SettingState.Disabled); } private void OnDisableCoreUpdates(object arg1, object arg2) { FuckMetrics.ToggleCoreUpdates(EntryDisableCoreUpdates.Value != FuckMetrics.SettingState.Disabled); } private void OnChangeMetricsUpdateRate(object arg1, object arg2) { if (EntryDisableMetrics.Value != FuckMetrics.SettingState.Disabled) { FuckMetrics.ToggleMetrics(false); FuckMetrics.ToggleMetrics(true); } } private void OnChangeCoreUpdateRate(object arg1, object arg2) { if (EntryDisableCoreUpdates.Value != FuckMetrics.SettingState.Disabled) { FuckMetrics.ToggleCoreUpdates(false); FuckMetrics.ToggleCoreUpdates(true); } } private void ApplyPatches(Type type) { try { HarmonyInstance.PatchAll(type); } catch (Exception e) { Logger.Msg($"Failed while patching {type.Name}!"); Logger.Error(e); } } public enum SettingState { Always, MenuOnly, Disabled } public static void ToggleMetrics(bool enable) { var job = SchedulerSystem.Instance.activeJobs.FirstOrDefault(pair => pair.Job.Method.Name == "UpdateMetrics").Job; if (enable && job == null) { SchedulerSystem.AddJob(new SchedulerSystem.Job(ViewManager.Instance.UpdateMetrics), 0f, FuckMetrics.EntryMetricsUpdateRate.Value, -1); } else if (!enable && job != null) { SchedulerSystem.RemoveJob(job); } } public static void ToggleCoreUpdates(bool enable) { var job = SchedulerSystem.Instance.activeJobs.FirstOrDefault(pair => pair.Job.Method.Name == "SendCoreUpdate").Job; if (enable && job == null) { SchedulerSystem.AddJob(new SchedulerSystem.Job(CVR_MenuManager.Instance.SendCoreUpdate), 0f, FuckMetrics.EntryCoreUpdateRate.Value, -1); } else if (!enable && job != null) { SchedulerSystem.RemoveJob(job); } } public static void ApplyMetricsSettings(bool show) { var disableMetrics = FuckMetrics.EntryDisableMetrics.Value; if (disableMetrics == FuckMetrics.SettingState.Always) return; if (disableMetrics == FuckMetrics.SettingState.MenuOnly) { FuckMetrics.ToggleMetrics(show); } else if (disableMetrics == FuckMetrics.SettingState.Disabled && show) { ViewManager.Instance.UpdateMetrics(); } } public static void ApplyCoreUpdatesSettings(bool show) { var disableCoreUpdates = FuckMetrics.EntryDisableCoreUpdates.Value; if (disableCoreUpdates == FuckMetrics.SettingState.Always) return; if (disableCoreUpdates == FuckMetrics.SettingState.MenuOnly) { FuckMetrics.ToggleCoreUpdates(show); } else if (disableCoreUpdates == FuckMetrics.SettingState.Disabled && show) { CVR_MenuManager.Instance.SendCoreUpdate(); } } public static void CohtmlAdvanceView(CohtmlView cohtmlView, Traverse menuOpenTraverse) { if (!FuckMetrics.EntryDisableCohtmlViewOnIdle.Value) return; if (cohtmlView != null && !menuOpenTraverse.GetValue()) { cohtmlView.enabled = false; try { cohtmlView.View.Advance(cohtmlView.CohtmlUISystem?.Id ?? 0, (double)Time.unscaledTime * 1000.0); } catch (Exception e) { FuckMetrics.Logger.Error($"An exception was thrown while calling CohtmlView.Advance(). Error message: {e.Message}"); } } } }