using System; using System.Collections.Generic; using System.Diagnostics; using System.Reflection; using System.Runtime.CompilerServices; using System.Runtime.Versioning; using System.Security; using System.Security.Permissions; using BepInEx; using BepInEx.Logging; using HarmonyLib; using Microsoft.CodeAnalysis; using Photon.Pun; using Photon.Realtime; using UnityEngine; [assembly: CompilationRelaxations(8)] [assembly: RuntimeCompatibility(WrapNonExceptionThrows = true)] [assembly: Debuggable(DebuggableAttribute.DebuggingModes.Default | DebuggableAttribute.DebuggingModes.DisableOptimizations | DebuggableAttribute.DebuggingModes.IgnoreSymbolStoreSequencePoints | DebuggableAttribute.DebuggingModes.EnableEditAndContinue)] [assembly: TargetFramework(".NETStandard,Version=v2.1", FrameworkDisplayName = ".NET Standard 2.1")] [assembly: AssemblyCompany("TumbleweedThrottle")] [assembly: AssemblyConfiguration("Debug")] [assembly: AssemblyFileVersion("1.0.0.0")] [assembly: AssemblyInformationalVersion("1.0.0+8ea5883b0352137320aef213490b484f7644d538")] [assembly: AssemblyProduct("TumbleweedThrottle")] [assembly: AssemblyTitle("TumbleweedThrottle")] [assembly: SecurityPermission(SecurityAction.RequestMinimum, SkipVerification = true)] [assembly: AssemblyVersion("1.0.0.0")] [module: UnverifiableCode] [module: RefSafetyRules(11)] namespace Microsoft.CodeAnalysis { [CompilerGenerated] [Microsoft.CodeAnalysis.Embedded] internal sealed class EmbeddedAttribute : Attribute { } } namespace System.Runtime.CompilerServices { [CompilerGenerated] [Microsoft.CodeAnalysis.Embedded] [AttributeUsage(AttributeTargets.Class | AttributeTargets.Property | AttributeTargets.Field | AttributeTargets.Event | AttributeTargets.Parameter | AttributeTargets.ReturnValue | AttributeTargets.GenericParameter, AllowMultiple = false, Inherited = false)] internal sealed class NullableAttribute : Attribute { public readonly byte[] NullableFlags; public NullableAttribute(byte P_0) { NullableFlags = new byte[1] { P_0 }; } public NullableAttribute(byte[] P_0) { NullableFlags = P_0; } } [CompilerGenerated] [Microsoft.CodeAnalysis.Embedded] [AttributeUsage(AttributeTargets.Class | AttributeTargets.Struct | AttributeTargets.Method | AttributeTargets.Interface | AttributeTargets.Delegate, AllowMultiple = false, Inherited = false)] internal sealed class NullableContextAttribute : Attribute { public readonly byte Flag; public NullableContextAttribute(byte P_0) { Flag = P_0; } } [CompilerGenerated] [Microsoft.CodeAnalysis.Embedded] [AttributeUsage(AttributeTargets.Module, AllowMultiple = false, Inherited = false)] internal sealed class RefSafetyRulesAttribute : Attribute { public readonly int Version; public RefSafetyRulesAttribute(int P_0) { Version = P_0; } } } namespace TumbleweedThrottle { [BepInPlugin("jill920.tumbleweedthrottle", "Tumbleweed Throttle", "1.0.0")] public class TumbleweedThrottlePlugin : BaseUnityPlugin { public const string MOD_GUID = "jill920.tumbleweedthrottle"; public const string MOD_NAME = "Tumbleweed Throttle"; public const string MOD_VERSION = "1.0.0"; public static ManualLogSource Logger; public static bool DebugMode = false; public const float BASE_SYNC_INTERVAL = 1f; public const float MIN_SYNC_INTERVAL = 0.22f; public const float MAX_SYNC_INTERVAL = 0.9f; public const float TUMBLEWEED_LIFETIME_SECONDS = 6f; public const float TUMBLEWEED_TARGET_SEARCH_INTERVAL = 8f; public const float POSITION_TOLERANCE = 1.5f; public const int LOW_CLIENT_THRESHOLD = 2; public const int HIGH_CLIENT_THRESHOLD = 12; public static float CurrentSyncInterval = 1f; private void Awake() { Logger = ((BaseUnityPlugin)this).Logger; string[] commandLineArgs = Environment.GetCommandLineArgs(); string[] array = commandLineArgs; foreach (string text in array) { if (text.Equals("-PerfDebug", StringComparison.OrdinalIgnoreCase)) { DebugMode = true; break; } } Logger.LogInfo((object)$"Photon SendRate: {PhotonNetwork.SendRate}"); Logger.LogInfo((object)$"Photon SerializationRate: {PhotonNetwork.SerializationRate}"); Logger.LogInfo((object)$"Time.fixedDeltaTime: {Time.fixedDeltaTime} ({1f / Time.fixedDeltaTime:F0} Hz)"); UpdateSyncInterval(); Harmony.CreateAndPatchAll(typeof(TumbleweedThrottlePlugin).Assembly, "jill920.tumbleweedthrottle"); Logger.LogInfo((object)"[Tumbleweed Throttle 1.0.0] Loaded"); Logger.LogInfo((object)$" Adaptive sync: {CurrentSyncInterval:F2}s (adjusts with player count)"); Logger.LogInfo((object)$" Tumbleweed lifetime: {6f}s"); } private void Update() { Room currentRoom = PhotonNetwork.CurrentRoom; int num = ((currentRoom == null) ? 1 : currentRoom.PlayerCount); float num2 = CalculateSyncInterval(num); if (Mathf.Abs(num2 - CurrentSyncInterval) > 0.01f) { CurrentSyncInterval = num2; if (DebugMode) { Logger.LogInfo((object)$"Player count: {num}, Sync interval adjusted to {CurrentSyncInterval:F2}s"); } } } private static float CalculateSyncInterval(int playerCount) { if (playerCount <= 2) { return 0.22f; } if (playerCount >= 12) { return 0.9f; } float num = (float)(playerCount - 2) / 10f; return Mathf.Lerp(0.22f, 0.9f, num); } public static void UpdateSyncInterval() { Room currentRoom = PhotonNetwork.CurrentRoom; int playerCount = ((currentRoom == null) ? 1 : currentRoom.PlayerCount); CurrentSyncInterval = CalculateSyncInterval(playerCount); } } public class ThrottledTumbleweedSync : MonoBehaviour, IPunObservable { private PhotonView photonView; private Rigidbody body; private Vector3 lastSyncedPosition; private Vector3 lastSyncedVelocity; private float lastSyncTime; private void Start() { photonView = ((Component)this).GetComponent(); body = ((Component)this).GetComponent(); PhotonRigidbodyView component = ((Component)this).GetComponent(); if ((Object)(object)component != (Object)null) { ((Behaviour)component).enabled = false; } ItemPhysicsSyncer component2 = ((Component)this).GetComponent(); if ((Object)(object)component2 != (Object)null) { ((Behaviour)component2).enabled = false; FieldInfo field = typeof(ItemPhysicsSyncer).GetField("shouldSync", BindingFlags.Instance | BindingFlags.NonPublic); if (field != null) { field.SetValue(component2, false); } } if ((Object)(object)photonView != (Object)null) { if (photonView.ObservedComponents != null) { photonView.ObservedComponents.Clear(); } photonView.ObservedComponents.Add((Component)(object)this); } if (TumbleweedThrottlePlugin.DebugMode) { TumbleweedThrottlePlugin.Logger.LogInfo((object)$"Throttled sync added to tumbleweed {((Object)this).GetInstanceID()}"); } } void IPunObservable.OnPhotonSerializeView(PhotonStream stream, PhotonMessageInfo info) { //IL_010f: Unknown result type (might be due to invalid IL or missing references) //IL_0114: Unknown result type (might be due to invalid IL or missing references) //IL_011c: Unknown result type (might be due to invalid IL or missing references) //IL_0121: Unknown result type (might be due to invalid IL or missing references) //IL_0143: Unknown result type (might be due to invalid IL or missing references) //IL_0148: Unknown result type (might be due to invalid IL or missing references) //IL_014c: Unknown result type (might be due to invalid IL or missing references) //IL_015d: Unknown result type (might be due to invalid IL or missing references) //IL_004c: Unknown result type (might be due to invalid IL or missing references) //IL_0052: Unknown result type (might be due to invalid IL or missing references) //IL_006b: Unknown result type (might be due to invalid IL or missing references) //IL_0071: Unknown result type (might be due to invalid IL or missing references) //IL_00a9: Unknown result type (might be due to invalid IL or missing references) //IL_00ae: Unknown result type (might be due to invalid IL or missing references) //IL_00ba: Unknown result type (might be due to invalid IL or missing references) //IL_00bf: Unknown result type (might be due to invalid IL or missing references) //IL_00cb: Unknown result type (might be due to invalid IL or missing references) //IL_00e2: Unknown result type (might be due to invalid IL or missing references) if ((Object)(object)body == (Object)null) { return; } float time = Time.time; float currentSyncInterval = TumbleweedThrottlePlugin.CurrentSyncInterval; if (stream.IsWriting) { if (!(time - lastSyncTime < currentSyncInterval)) { bool flag = Vector3.Distance(body.position, lastSyncedPosition) > 1.5f; bool flag2 = Vector3.Distance(body.linearVelocity, lastSyncedVelocity) > 0.5f; if (flag || flag2) { lastSyncTime = time; lastSyncedPosition = body.position; lastSyncedVelocity = body.linearVelocity; stream.SendNext((object)body.position); stream.SendNext((object)body.linearVelocity); } } } else if (stream.Count >= 2) { Vector3 val = (Vector3)stream.ReceiveNext(); Vector3 linearVelocity = (Vector3)stream.ReceiveNext(); float num = Mathf.Clamp01(Time.deltaTime * (1f / currentSyncInterval)); body.position = Vector3.Lerp(body.position, val, num); body.linearVelocity = linearVelocity; } } } [HarmonyPatch] internal static class PerformanceFixPatches { private static Dictionary lastMobTargeting; private static Dictionary lastMobRaycast; private static Dictionary lastTargetSearch; private static Dictionary cachedTarget; private static HashSet patchedTumbleweeds; private static FieldInfo tumbleweedPhotonViewField; private static FieldInfo tumbleweedRigField; private static FieldInfo tumbleweedRollForceField; private static MethodInfo tumbleweedGetTargetMethod; static PerformanceFixPatches() { lastMobTargeting = new Dictionary(); lastMobRaycast = new Dictionary(); lastTargetSearch = new Dictionary(); cachedTarget = new Dictionary(); patchedTumbleweeds = new HashSet(); tumbleweedPhotonViewField = typeof(TumbleWeed).GetField("photonView", BindingFlags.Instance | BindingFlags.NonPublic); tumbleweedRigField = typeof(TumbleWeed).GetField("rig", BindingFlags.Instance | BindingFlags.NonPublic); tumbleweedRollForceField = typeof(TumbleWeed).GetField("rollForce", BindingFlags.Instance | BindingFlags.NonPublic); tumbleweedGetTargetMethod = typeof(TumbleWeed).GetMethod("GetTarget", BindingFlags.Instance | BindingFlags.NonPublic); } private static void LogDebug(string message) { if (TumbleweedThrottlePlugin.DebugMode) { TumbleweedThrottlePlugin.Logger.LogInfo((object)("[DEBUG] " + message)); } } [HarmonyPatch(typeof(TumbleWeed), "Start")] [HarmonyPostfix] private static void Postfix_TumbleWeed_Start(TumbleWeed __instance) { int instanceID = ((Object)__instance).GetInstanceID(); if (!patchedTumbleweeds.Contains(instanceID)) { patchedTumbleweeds.Add(instanceID); RemoveAfterSeconds component = ((Component)__instance).GetComponent(); if ((Object)(object)component == (Object)null) { component = ((Component)__instance).gameObject.AddComponent(); component.Config(true, 6f); component.photonRemove = true; LogDebug($"Added RemoveAfterSeconds to tumbleweed {instanceID}"); } if ((Object)(object)((Component)__instance).GetComponent() == (Object)null) { ((Component)__instance).gameObject.AddComponent(); LogDebug($"Added ThrottledTumbleweedSync to tumbleweed {instanceID}"); } } } [HarmonyPatch(typeof(TumbleWeed), "FixedUpdate")] [HarmonyPrefix] private static bool Prefix_TumbleWeed_FixedUpdate(TumbleWeed __instance) { //IL_0101: Unknown result type (might be due to invalid IL or missing references) //IL_0106: Unknown result type (might be due to invalid IL or missing references) //IL_010b: Unknown result type (might be due to invalid IL or missing references) //IL_0162: Unknown result type (might be due to invalid IL or missing references) //IL_0166: Unknown result type (might be due to invalid IL or missing references) //IL_0140: Unknown result type (might be due to invalid IL or missing references) //IL_014b: Unknown result type (might be due to invalid IL or missing references) //IL_0150: Unknown result type (might be due to invalid IL or missing references) //IL_0155: Unknown result type (might be due to invalid IL or missing references) //IL_0159: Unknown result type (might be due to invalid IL or missing references) //IL_015e: Unknown result type (might be due to invalid IL or missing references) object? obj = tumbleweedPhotonViewField?.GetValue(__instance); PhotonView val = (PhotonView)((obj is PhotonView) ? obj : null); if ((Object)(object)val == (Object)null || !val.IsMine) { return true; } int instanceID = ((Object)__instance).GetInstanceID(); float time = Time.time; if (!lastTargetSearch.ContainsKey(instanceID) || time - lastTargetSearch[instanceID] >= 8f) { lastTargetSearch[instanceID] = time; if (tumbleweedGetTargetMethod != null) { Dictionary dictionary = cachedTarget; object? obj2 = tumbleweedGetTargetMethod.Invoke(__instance, null); dictionary[instanceID] = (Character)((obj2 is Character) ? obj2 : null); } } object? obj3 = tumbleweedRigField?.GetValue(__instance); Rigidbody val2 = (Rigidbody)((obj3 is Rigidbody) ? obj3 : null); float num = ((tumbleweedRollForceField != null) ? ((float)tumbleweedRollForceField.GetValue(__instance)) : 10f); if ((Object)(object)val2 != (Object)null) { Vector3 val3 = -Vector3.right; if (cachedTarget.ContainsKey(instanceID) && (Object)(object)cachedTarget[instanceID] != (Object)null) { Vector3 val4 = cachedTarget[instanceID].Center - ((Component)__instance).transform.position; val3 = ((Vector3)(ref val4)).normalized; } val2.AddForce(val3 * num, (ForceMode)5); } return false; } } }