Decompiled source of ExtraSlotsCustomSlots v1.0.0
ExtraSlotsCustomSlots.dll
Decompiled 3 hours ago
The result has been truncated due to the large size, download it to view full contents!
using System; using System.Collections; using System.Collections.Generic; using System.Diagnostics; using System.Globalization; using System.IO; using System.IO.Compression; using System.Linq; using System.Reflection; using System.Runtime.CompilerServices; using System.Runtime.InteropServices; using System.Runtime.Serialization; using System.Runtime.Versioning; using System.Security; using System.Security.Permissions; using BepInEx; using BepInEx.Bootstrap; using BepInEx.Configuration; using ExtraSlots; using ExtraSlotsCustomSlots.AdventureBackpacksCustomSlot; using HarmonyLib; using JetBrains.Annotations; using Microsoft.CodeAnalysis; using ServerSync; using TMPro; using UnityEngine; using UnityEngine.SceneManagement; [assembly: CompilationRelaxations(8)] [assembly: RuntimeCompatibility(WrapNonExceptionThrows = true)] [assembly: Debuggable(DebuggableAttribute.DebuggingModes.Default | DebuggableAttribute.DebuggingModes.DisableOptimizations | DebuggableAttribute.DebuggingModes.IgnoreSymbolStoreSequencePoints | DebuggableAttribute.DebuggingModes.EnableEditAndContinue)] [assembly: AssemblyTitle("ExtraSlotsCustomSlots")] [assembly: AssemblyDescription("")] [assembly: AssemblyConfiguration("")] [assembly: AssemblyCompany("")] [assembly: AssemblyProduct("ExtraSlotsCustomSlots")] [assembly: AssemblyCopyright("Copyright © 2024")] [assembly: AssemblyTrademark("")] [assembly: ComVisible(false)] [assembly: Guid("ee3d2a93-2e64-4701-bfd2-efd5a55cfe52")] [assembly: AssemblyFileVersion("1.0.0")] [assembly: TargetFramework(".NETFramework,Version=v4.8", FrameworkDisplayName = ".NET Framework 4.8")] [assembly: SecurityPermission(SecurityAction.RequestMinimum, SkipVerification = true)] [assembly: AssemblyVersion("1.0.0.0")] [module: UnverifiableCode] 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; } } } namespace ExtraSlotsCustomSlots { public class AdventureBackpacksSlot : CustomSlot { public const string ID = "AdventureBackpacks"; public const string pluginID = "vapok.mods.adventurebackpacks"; public static bool IsActive => Chainloader.PluginInfos.ContainsKey("vapok.mods.adventurebackpacks") && ExtraSlotsCustomSlots.adventureBackpackSlotEnabled.Value; public AdventureBackpacksSlot() { CustomSlot.slots.Add(this); GUID = "vapok.mods.adventurebackpacks"; if (!base.PluginInstalled) { return; } Assembly assembly = Assembly.GetAssembly(((object)Chainloader.PluginInfos["vapok.mods.adventurebackpacks"].Instance).GetType()); MethodInfo isValid = AccessTools.Method(assembly.GetType("AdventureBackpacks.API.ABAPI"), "IsBackpack", (Type[])null, (Type[])null); if (isValid == null) { ExtraSlotsCustomSlots.LogWarning("AdventureBackpacks mod is loaded but AdventureBackpacks.API.ABAPI:IsBackpack is not found"); return; } slotID = "AdventureBackpacks"; itemIsValid = delegate(ItemData item) { int result; if (item != null) { MethodInfo methodInfo = isValid; object[] parameters = (object[])(object)new ItemData[1] { item }; result = (((bool)methodInfo.Invoke(null, parameters)) ? 1 : 0); } else { result = 0; } return (byte)result != 0; }; getName = () => ExtraSlotsCustomSlots.adventureBackpackSlotName.Value; isActive = () => API.IsAnyGlobalKeyActive(ExtraSlotsCustomSlots.adventureBackpackSlotGlobalKey.Value) || API.IsAnyMaterialDiscovered(ExtraSlotsCustomSlots.adventureBackpackSlotItemDiscovered.Value); initialized = true; CustomItemType.InitBackpackFunc(itemIsValid, assembly); } } public static class AdventureBackpacksPatches { [HarmonyPatch] public static class AdventureBackpacks_PlayerExtensions_CustomSlotItem { public static List<MethodBase> targets = new List<MethodBase>(); public static bool Prepare() { if (!Chainloader.PluginInfos.TryGetValue("vapok.mods.adventurebackpacks", out var value)) { return false; } Assembly assembly = Assembly.GetAssembly(((object)value.Instance).GetType()); MethodInfo methodInfo = AccessTools.Method(assembly.GetType("AdventureBackpacks.Extensions.PlayerExtensions"), "IsBackpackEquipped", (Type[])null, (Type[])null); if ((object)methodInfo != null) { ExtraSlotsCustomSlots.LogInfo("AdventureBackpacks.Extensions.PlayerExtensions:IsBackpackEquipped method will be patched to make it work with custom slot"); targets.Add(methodInfo); } MethodInfo methodInfo2 = AccessTools.Method(assembly.GetType("AdventureBackpacks.Extensions.PlayerExtensions"), "IsThisBackpackEquipped", (Type[])null, (Type[])null); if ((object)methodInfo2 != null) { ExtraSlotsCustomSlots.LogInfo("AdventureBackpacks.Extensions.PlayerExtensions:IsThisBackpackEquipped method will be patched to make it work with custom slot"); targets.Add(methodInfo2); } MethodInfo methodInfo3 = AccessTools.Method(assembly.GetType("AdventureBackpacks.Extensions.PlayerExtensions"), "GetEquippedBackpack", (Type[])null, (Type[])null); if ((object)methodInfo3 != null) { ExtraSlotsCustomSlots.LogInfo("AdventureBackpacks.Extensions.PlayerExtensions:GetEquippedBackpack method will be patched to make it work with custom slot"); targets.Add(methodInfo3); } if (targets.Count == 0) { return false; } return true; } private static IEnumerable<MethodBase> TargetMethods() { return targets; } public static void Prefix(Player player, ref ItemData __state) { if (AdventureBackpacksSlot.IsActive && (__state = ((Humanoid)player).m_shoulderItem) != null) { ((Humanoid)player).m_shoulderItem = ((Humanoid)(object)player).GetAdventureBackpack(); } } public static void Postfix(Player player, ItemData __state) { if (__state != null) { ((Humanoid)player).m_shoulderItem = __state; } } } [HarmonyPatch(typeof(Humanoid), "UnequipItem")] public static class Humanoid_UnequipItem_CustomItemType_FirstPrefix { public static ItemData m_shoulderItem; public static bool Prepare() { return Chainloader.PluginInfos.ContainsKey("vapok.mods.adventurebackpacks"); } [HarmonyPriority(800)] [HarmonyBefore(new string[] { "vapok.mods.adventurebackpacks" })] private static void Prefix(Humanoid __instance, ItemData item) { //IL_0026: Unknown result type (might be due to invalid IL or missing references) //IL_002b: Unknown result type (might be due to invalid IL or missing references) if (AdventureBackpacksSlot.IsActive && item != null && !((Object)(object)Player.m_localPlayer != (Object)(object)__instance)) { Scene activeScene = SceneManager.GetActiveScene(); if (!((Scene)(ref activeScene)).name.Equals("start") && CustomItemType.IsBackpack(item) && (m_shoulderItem = __instance.m_shoulderItem) != null) { __instance.m_shoulderItem = __instance.GetAdventureBackpack(); } } } } [HarmonyPatch(typeof(Humanoid), "UnequipItem")] public static class Humanoid_UnequipItem_CustomItemType_LastPrefix { public static bool Prepare() { return Chainloader.PluginInfos.ContainsKey("vapok.mods.adventurebackpacks"); } [HarmonyPriority(0)] [HarmonyAfter(new string[] { "vapok.mods.adventurebackpacks" })] private static void Prefix(Humanoid __instance) { if (Humanoid_UnequipItem_CustomItemType_FirstPrefix.m_shoulderItem != null) { __instance.m_shoulderItem = Humanoid_UnequipItem_CustomItemType_FirstPrefix.m_shoulderItem; } } } [HarmonyPatch] public static class EpicLoot_Player_GetEquipment_AddBackpackItem { public const string epicLootGUID = "randyknapp.mods.epicloot"; public static MethodBase target; public static bool Prepare() { if (!Chainloader.PluginInfos.ContainsKey("randyknapp.mods.epicloot")) { return false; } target = AccessTools.Method("EpicLoot.PlayerExtensions:GetEquipment", (Type[])null, (Type[])null); if (target == null) { return false; } ExtraSlotsCustomSlots.LogInfo("EpicLoot.PlayerExtensions:GetEquipment method will be patched to add adventure backpack"); return true; } public static MethodBase TargetMethod() { return target; } public static void Postfix(Player player, List<ItemData> __result) { if (AdventureBackpacksSlot.IsActive) { ItemData adventureBackpack = ((Humanoid)(object)player).GetAdventureBackpack(); if (adventureBackpack != null) { __result.Add(adventureBackpack); } } } } } public class HipLanternSlot : CustomSlot { public const string ID = "HipLantern"; public const string pluginID = "shudnal.HipLantern"; public HipLanternSlot() { CustomSlot.slots.Add(this); GUID = "shudnal.HipLantern"; if (!base.PluginInstalled) { return; } Assembly assembly = Assembly.GetAssembly(((object)Chainloader.PluginInfos["shudnal.HipLantern"].Instance).GetType()); MethodInfo isValid = AccessTools.Method(assembly.GetType("HipLantern.LanternItem"), "IsLanternItem", new Type[1] { typeof(ItemData) }, (Type[])null); if (isValid == null) { ExtraSlotsCustomSlots.LogWarning("HipLantern mod is loaded but HipLantern.LanternItem:IsLanternItem is not found"); return; } slotID = "HipLantern"; itemIsValid = delegate(ItemData item) { int result; if (item != null) { MethodInfo methodInfo = isValid; object[] parameters = (object[])(object)new ItemData[1] { item }; result = (((bool)methodInfo.Invoke(null, parameters)) ? 1 : 0); } else { result = 0; } return (byte)result != 0; }; getName = () => ExtraSlotsCustomSlots.hipLanternSlotName.Value; isActive = () => API.IsAnyGlobalKeyActive(ExtraSlotsCustomSlots.hipLanternSlotGlobalKey.Value) || API.IsAnyMaterialDiscovered(ExtraSlotsCustomSlots.hipLanternSlotItemDiscovered.Value); initialized = true; } } public class CircletExtendedSlot : CustomSlot { public const string ID = "CircletExtended"; public const string pluginID = "shudnal.CircletExtended"; public CircletExtendedSlot() { CustomSlot.slots.Add(this); GUID = "shudnal.CircletExtended"; if (!base.PluginInstalled) { return; } Assembly assembly = Assembly.GetAssembly(((object)Chainloader.PluginInfos["shudnal.CircletExtended"].Instance).GetType()); MethodInfo isValid = AccessTools.Method(assembly.GetType("CircletExtended.CircletItem"), "IsCircletItem", new Type[1] { typeof(ItemData) }, (Type[])null); if (isValid == null) { ExtraSlotsCustomSlots.LogWarning("CircletExtended mod is loaded but CircletExtended.CircletItem:IsCircletItem is not found"); return; } slotID = "CircletExtended"; itemIsValid = delegate(ItemData item) { int result; if (item != null) { MethodInfo methodInfo = isValid; object[] parameters = (object[])(object)new ItemData[1] { item }; result = (((bool)methodInfo.Invoke(null, parameters)) ? 1 : 0); } else { result = 0; } return (byte)result != 0; }; getName = () => ExtraSlotsCustomSlots.circletExtendedSlotName.Value; isActive = () => API.IsAnyGlobalKeyActive(ExtraSlotsCustomSlots.circletExtendedSlotGlobalKey.Value) || API.IsAnyMaterialDiscovered(ExtraSlotsCustomSlots.circletExtendedSlotItemDiscovered.Value); initialized = true; } } public class BowsBeforeHoesSlot : CustomSlot { public const string ID = "BowsBeforeHoes"; public const string pluginID = "Azumatt.BowsBeforeHoes"; public BowsBeforeHoesSlot() { CustomSlot.slots.Add(this); GUID = "Azumatt.BowsBeforeHoes"; if (!base.PluginInstalled) { return; } Assembly assembly = Assembly.GetAssembly(((object)Chainloader.PluginInfos["Azumatt.BowsBeforeHoes"].Instance).GetType()); MethodInfo isValid = AccessTools.Method(assembly.GetType("BowsBeforeHoes.Util.Functions"), "IsQuiverSlot", (Type[])null, (Type[])null); if (isValid == null) { ExtraSlotsCustomSlots.LogWarning("BowsBeforeHoes mod is loaded but BowsBeforeHoes.Util.Functions:IsQuiverSlot is not found"); return; } slotID = "BowsBeforeHoes"; itemIsValid = delegate(ItemData item) { int result; if (item != null) { MethodInfo methodInfo = isValid; object[] parameters = (object[])(object)new ItemData[1] { item }; result = (((bool)methodInfo.Invoke(null, parameters)) ? 1 : 0); } else { result = 0; } return (byte)result != 0; }; getName = () => ExtraSlotsCustomSlots.bbhQuiverSlotName.Value; isActive = () => API.IsAnyGlobalKeyActive(ExtraSlotsCustomSlots.bbhQuiverSlotGlobalKey.Value) || API.IsAnyMaterialDiscovered(ExtraSlotsCustomSlots.bbhQuiverSlotItemDiscovered.Value); initialized = true; } } public class JewelcraftingNeckSlot : CustomSlot { public const string ID = "JewelcraftingNeck"; public const string pluginID = "org.bepinex.plugins.jewelcrafting"; public JewelcraftingNeckSlot() { CustomSlot.slots.Add(this); GUID = "org.bepinex.plugins.jewelcrafting"; if (!base.PluginInstalled) { return; } Assembly assembly = Assembly.GetAssembly(((object)Chainloader.PluginInfos["org.bepinex.plugins.jewelcrafting"].Instance).GetType()); MethodInfo isValid = AccessTools.Method(assembly.GetType("Jewelcrafting.Visual"), "IsNeckItem", (Type[])null, (Type[])null); if (isValid == null) { ExtraSlotsCustomSlots.LogWarning("Jewelcrafting mod is loaded but Jewelcrafting.Visual:IsNeckItem is not found"); return; } slotID = "JewelcraftingNeck"; itemIsValid = delegate(ItemData item) { int result; if (item != null) { MethodInfo methodInfo = isValid; object[] parameters = (object[])(object)new ItemData[1] { item }; result = (((bool)methodInfo.Invoke(null, parameters)) ? 1 : 0); } else { result = 0; } return (byte)result != 0; }; getName = () => ExtraSlotsCustomSlots.jewelcraftingNeckSlotName.Value; isActive = () => API.IsAnyGlobalKeyActive(ExtraSlotsCustomSlots.jewelcraftingNeckSlotGlobalKey.Value) || API.IsAnyMaterialDiscovered(ExtraSlotsCustomSlots.jewelcraftingNeckSlotItemDiscovered.Value); initialized = true; } } public class JewelcraftingRingSlot : CustomSlot { public const string ID = "JewelcraftingRing"; public const string pluginID = "org.bepinex.plugins.jewelcrafting"; public JewelcraftingRingSlot() { CustomSlot.slots.Add(this); GUID = "org.bepinex.plugins.jewelcrafting"; if (!base.PluginInstalled) { return; } Assembly assembly = Assembly.GetAssembly(((object)Chainloader.PluginInfos["org.bepinex.plugins.jewelcrafting"].Instance).GetType()); MethodInfo isValid = AccessTools.Method(assembly.GetType("Jewelcrafting.Visual"), "IsFingerItem", (Type[])null, (Type[])null); if (isValid == null) { ExtraSlotsCustomSlots.LogWarning("Jewelcrafting mod is loaded but Jewelcrafting.Visual:IsFingerItem is not found"); return; } slotID = "JewelcraftingRing"; itemIsValid = delegate(ItemData item) { int result; if (item != null) { MethodInfo methodInfo = isValid; object[] parameters = (object[])(object)new ItemData[1] { item }; result = (((bool)methodInfo.Invoke(null, parameters)) ? 1 : 0); } else { result = 0; } return (byte)result != 0; }; getName = () => ExtraSlotsCustomSlots.jewelcraftingRingSlotName.Value; isActive = () => API.IsAnyGlobalKeyActive(ExtraSlotsCustomSlots.jewelcraftingRingSlotGlobalKey.Value) || API.IsAnyMaterialDiscovered(ExtraSlotsCustomSlots.jewelcraftingRingSlotItemDiscovered.Value); initialized = true; } } public class BackpacksSlot : CustomSlot { public const string ID = "Backpacks"; public const string pluginID = "org.bepinex.plugins.backpacks"; public BackpacksSlot() { CustomSlot.slots.Add(this); GUID = "org.bepinex.plugins.backpacks"; if (!base.PluginInstalled) { return; } Assembly assembly = Assembly.GetAssembly(((object)Chainloader.PluginInfos["org.bepinex.plugins.backpacks"].Instance).GetType()); MethodInfo isValid = AccessTools.Method(assembly.GetType("Backpacks.Backpacks"), "validateBackpack", (Type[])null, (Type[])null); if (isValid == null) { ExtraSlotsCustomSlots.LogWarning("Backpacks mod is loaded but Backpacks.Backpacks:validateBackpack is not found"); return; } slotID = "Backpacks"; itemIsValid = delegate(ItemData item) { int result; if (item != null) { MethodInfo methodInfo = isValid; object[] parameters = (object[])(object)new ItemData[1] { item }; result = (((bool)methodInfo.Invoke(null, parameters)) ? 1 : 0); } else { result = 0; } return (byte)result != 0; }; getName = () => ExtraSlotsCustomSlots.backpacksSlotName.Value; isActive = () => API.IsAnyGlobalKeyActive(ExtraSlotsCustomSlots.backpacksSlotGlobalKey.Value) || API.IsAnyMaterialDiscovered(ExtraSlotsCustomSlots.backpacksSlotItemDiscovered.Value); initialized = true; } } internal class CustomConfigs { internal class ConfigurationManagerAttributes { [UsedImplicitly] public Action<ConfigEntryBase>? CustomDrawer; } internal static object? configManager; internal static Type? configManagerStyles; internal static GUIStyle GetStyle(GUIStyle other) { //IL_0034: Unknown result type (might be due to invalid IL or missing references) //IL_0039: Unknown result type (might be due to invalid IL or missing references) //IL_0051: Expected O, but got Unknown if (configManagerStyles == null) { return other; } FieldInfo fieldInfo = AccessTools.Field(configManagerStyles, "fontSize"); if (fieldInfo == null) { return other; } return new GUIStyle(other) { fontSize = (int)fieldInfo.GetValue(configManagerStyles) }; } internal static void Awake() { Assembly assembly = AppDomain.CurrentDomain.GetAssemblies().FirstOrDefault((Assembly a) => a.GetName().Name == "ConfigurationManager"); Type type = assembly?.GetType("ConfigurationManager.ConfigurationManager"); configManager = ((type == null) ? null : Chainloader.ManagerObject.GetComponent(type)); configManagerStyles = assembly?.GetType("ConfigurationManager.ConfigurationManagerStyles"); } internal static Action<ConfigEntryBase> DrawSeparatedStrings(string splitString) { string splitString2 = splitString; return delegate(ConfigEntryBase cfg) { //IL_010a: Unknown result type (might be due to invalid IL or missing references) //IL_010f: Unknown result type (might be due to invalid IL or missing references) //IL_0125: Expected O, but got Unknown //IL_0155: Unknown result type (might be due to invalid IL or missing references) //IL_015a: Unknown result type (might be due to invalid IL or missing references) //IL_0170: Expected O, but got Unknown bool valueOrDefault = cfg.Description.Tags.Select((object a) => (a.GetType().Name == "ConfigurationManagerAttributes") ? ((bool?)a.GetType().GetField("ReadOnly")?.GetValue(a)) : null).FirstOrDefault((bool? v) => v.HasValue).GetValueOrDefault(); bool flag = false; GUILayout.BeginVertical(Array.Empty<GUILayoutOption>()); List<string> list = new List<string>(); List<string> list2 = ((string)cfg.BoxedValue).Split(new string[1] { splitString2 }, StringSplitOptions.None).ToList(); for (int i = 0; i < list2.Count; i++) { GUILayout.BeginHorizontal(Array.Empty<GUILayoutOption>()); string text = list2[i]; string text2 = GUILayout.TextField(text, GetStyle(GUI.skin.textArea), (GUILayoutOption[])(object)new GUILayoutOption[1] { GUILayout.ExpandWidth(true) }); if (text2 != text && !valueOrDefault) { flag = true; } if (GUILayout.Button("x", new GUIStyle(GetStyle(GUI.skin.button)) { fixedWidth = 21f }, Array.Empty<GUILayoutOption>()) && !valueOrDefault) { flag = true; } else { list.Add(text2); } if (GUILayout.Button("+", new GUIStyle(GetStyle(GUI.skin.button)) { fixedWidth = 21f }, Array.Empty<GUILayoutOption>()) && !valueOrDefault) { flag = true; list.Add(""); } GUILayout.EndHorizontal(); } GUILayout.EndVertical(); if (flag) { cfg.BoxedValue = string.Join(splitString2, list); } }; } internal static Action<ConfigEntryBase> DrawOrderedFixedStrings(string splitString) { string splitString2 = splitString; return delegate(ConfigEntryBase cfg) { //IL_00e4: Unknown result type (might be due to invalid IL or missing references) //IL_00e9: Unknown result type (might be due to invalid IL or missing references) //IL_00ff: Expected O, but got Unknown //IL_015b: Unknown result type (might be due to invalid IL or missing references) //IL_0160: Unknown result type (might be due to invalid IL or missing references) //IL_0176: Expected O, but got Unknown bool valueOrDefault = cfg.Description.Tags.Select((object a) => (a.GetType().Name == "ConfigurationManagerAttributes") ? ((bool?)a.GetType().GetField("ReadOnly")?.GetValue(a)) : null).FirstOrDefault((bool? v) => v.HasValue).GetValueOrDefault(); bool flag = false; GUILayout.BeginVertical(Array.Empty<GUILayoutOption>()); string[] array = ((string)cfg.BoxedValue).Split(new string[1] { splitString2 }, StringSplitOptions.None).ToArray(); for (int i = 0; i < array.Length; i++) { GUILayout.BeginHorizontal(Array.Empty<GUILayoutOption>()); string text = array[i]; GUILayout.Label(text, GetStyle(GUI.skin.textArea), (GUILayoutOption[])(object)new GUILayoutOption[1] { GUILayout.ExpandWidth(true) }); if (GUILayout.Button("ʌ", new GUIStyle(GetStyle(GUI.skin.button)) { fixedWidth = 21f }, Array.Empty<GUILayoutOption>()) && !valueOrDefault && (flag = i > 0)) { ref string reference = ref array[i]; ref string reference2 = ref array[i - 1]; string text2 = array[i - 1]; string text3 = array[i]; reference = text2; reference2 = text3; } if (GUILayout.Button("v", new GUIStyle(GetStyle(GUI.skin.button)) { fixedWidth = 21f }, Array.Empty<GUILayoutOption>()) && !valueOrDefault && (flag = i < array.Length - 1)) { ref string reference = ref array[i]; ref string reference3 = ref array[i + 1]; string text3 = array[i + 1]; string text2 = array[i]; reference = text3; reference3 = text2; } GUILayout.EndHorizontal(); } GUILayout.EndVertical(); if (flag) { cfg.BoxedValue = string.Join(splitString2, array); } }; } } public class CustomSlot { public const string slotPrefix = "ESCS"; public string GUID; public bool initialized; public string slotID; public Func<ItemData, bool> itemIsValid; public Func<string> getName; public Func<bool> isActive; public static readonly List<CustomSlot> slots = new List<CustomSlot>(); public static readonly string VanillaOrder = "Backpacks,AdventureBackpacks,CircletExtended,JewelcraftingNeck,MagicPluginEarring,JewelcraftingRing,MagicPluginTome,BowsBeforeHoes,HipLantern"; public bool PluginInstalled => Chainloader.PluginInfos.ContainsKey(GUID); public bool AddSlot() { return initialized && API.AddSlot(GetSlotID(slotID), getName, itemIsValid, isActive); } public bool RemoveSlot() { return initialized && (API.RemoveSlot(GetSlotID(slotID)) || API.RemoveSlot(slotID)); } public static string GetSlotID(string slotID) { return "ESCS" + slotID.ToString(); } public override string ToString() { return initialized ? slotID.ToString() : (GUID + " (inactive)"); } } [BepInDependency(/*Could not decode attribute arguments.*/)] [BepInDependency(/*Could not decode attribute arguments.*/)] [BepInDependency(/*Could not decode attribute arguments.*/)] [BepInDependency(/*Could not decode attribute arguments.*/)] [BepInDependency(/*Could not decode attribute arguments.*/)] [BepInDependency(/*Could not decode attribute arguments.*/)] [BepInDependency(/*Could not decode attribute arguments.*/)] [BepInDependency(/*Could not decode attribute arguments.*/)] [BepInPlugin("shudnal.ExtraSlotsCustomSlots", "Extra Slots Custom Slots", "1.0.0")] public class ExtraSlotsCustomSlots : BaseUnityPlugin { public const string pluginID = "shudnal.ExtraSlotsCustomSlots"; public const string pluginName = "Extra Slots Custom Slots"; public const string pluginVersion = "1.0.0"; internal readonly Harmony harmony = new Harmony("shudnal.ExtraSlotsCustomSlots"); internal static readonly ConfigSync configSync = new ConfigSync("shudnal.ExtraSlotsCustomSlots") { DisplayName = "Extra Slots Custom Slots", CurrentVersion = "1.0.0", MinimumRequiredVersion = "1.0.0" }; internal static ExtraSlotsCustomSlots instance; public static ConfigEntry<bool> configLocked; public static ConfigEntry<bool> loggingEnabled; public static ConfigEntry<string> slotsOrder; public static ConfigEntry<bool> adventureBackpackSlotEnabled; public static ConfigEntry<string> adventureBackpackSlotName; public static ConfigEntry<string> adventureBackpackSlotGlobalKey; public static ConfigEntry<string> adventureBackpackSlotItemDiscovered; public static ConfigEntry<bool> backpacksSlotEnabled; public static ConfigEntry<string> backpacksSlotName; public static ConfigEntry<string> backpacksSlotGlobalKey; public static ConfigEntry<string> backpacksSlotItemDiscovered; public static ConfigEntry<bool> bbhQuiverSlotEnabled; public static ConfigEntry<string> bbhQuiverSlotName; public static ConfigEntry<string> bbhQuiverSlotGlobalKey; public static ConfigEntry<string> bbhQuiverSlotItemDiscovered; public static ConfigEntry<bool> circletExtendedSlotEnabled; public static ConfigEntry<string> circletExtendedSlotName; public static ConfigEntry<string> circletExtendedSlotGlobalKey; public static ConfigEntry<string> circletExtendedSlotItemDiscovered; public static ConfigEntry<bool> hipLanternSlotEnabled; public static ConfigEntry<string> hipLanternSlotName; public static ConfigEntry<string> hipLanternSlotGlobalKey; public static ConfigEntry<string> hipLanternSlotItemDiscovered; public static ConfigEntry<bool> jewelcraftingNeckSlotEnabled; public static ConfigEntry<string> jewelcraftingNeckSlotName; public static ConfigEntry<string> jewelcraftingNeckSlotGlobalKey; public static ConfigEntry<string> jewelcraftingNeckSlotItemDiscovered; public static ConfigEntry<bool> jewelcraftingRingSlotEnabled; public static ConfigEntry<string> jewelcraftingRingSlotName; public static ConfigEntry<string> jewelcraftingRingSlotGlobalKey; public static ConfigEntry<string> jewelcraftingRingSlotItemDiscovered; public static ConfigEntry<bool> magicPluginTomeSlotEnabled; public static ConfigEntry<string> magicPluginTomeSlotName; public static ConfigEntry<string> magicPluginTomeSlotGlobalKey; public static ConfigEntry<string> magicPluginTomeSlotItemDiscovered; public static ConfigEntry<bool> magicPluginEarringSlotEnabled; public static ConfigEntry<string> magicPluginEarringSlotName; public static ConfigEntry<string> magicPluginEarringSlotGlobalKey; public static ConfigEntry<string> magicPluginEarringSlotItemDiscovered; private void Awake() { instance = this; ConfigInit(); configSync.AddLockingConfigEntry<bool>(configLocked); harmony.PatchAll(); UpdateSlots(); } private void OnDestroy() { ((BaseUnityPlugin)this).Config.Save(); instance = null; Harmony obj = harmony; if (obj != null) { obj.UnpatchSelf(); } } public void ConfigInit() { //IL_006d: Unknown result type (might be due to invalid IL or missing references) //IL_0078: Expected O, but got Unknown configLocked = config("General", "Lock Configuration", defaultValue: true, "Configuration is locked and can be changed by server admins only. "); loggingEnabled = config("General", "Logging enabled", defaultValue: false, "Enable logging. [Not synced with Server]", synchronizedSetting: false); slotsOrder = config("General", "Slots order", CustomSlot.VanillaOrder, new ConfigDescription("Comma-separated slot ID order of custom slots", (AcceptableValueBase)null, new object[1] { new CustomConfigs.ConfigurationManagerAttributes { CustomDrawer = CustomConfigs.DrawOrderedFixedStrings(",") } })); slotsOrder.SettingChanged += delegate { UpdateSlots(); }; adventureBackpackSlotEnabled = config("Mod - Adventure Backpacks", "Enabled", defaultValue: true, "Enable adventure backpack slot. Restart the game after change to avoid potential issues."); adventureBackpackSlotName = config("Mod - Adventure Backpacks", "Name", "Backpack", "Slot name"); adventureBackpackSlotGlobalKey = config("Mod - Adventure Backpacks", "Global keys", "", "Comma-separated list of global keys and player unique keys. Slot will be active only if any key is enabled or list is not set."); adventureBackpackSlotItemDiscovered = config("Mod - Adventure Backpacks", "Items discovered", "$vapok_mod_item_backpack_meadows,$vapok_mod_item_backpack_blackforest,$vapok_mod_item_backpack_swamp,$vapok_mod_item_backpack_mountains,$vapok_mod_item_backpack_plains,$vapok_mod_item_backpack_mistlands,$vapok_mod_item_rugged_backpack,$vapok_mod_item_arctic_backpack", "Comma-separated list of items. Slot will be active only if any item is discovered or list is not set."); adventureBackpackSlotEnabled.SettingChanged += delegate { UpdateSlots(); }; backpacksSlotEnabled = config("Mod - Backpacks", "Enabled", defaultValue: true, "Enable backpack slot"); backpacksSlotName = config("Mod - Backpacks", "Name", "$bp_backpack_slot_name", "Slot name"); backpacksSlotGlobalKey = config("Mod - Backpacks", "Global keys", "", "Comma-separated list of global keys and player unique keys. Slot will be active only if any key is enabled or list is not set."); backpacksSlotItemDiscovered = config("Mod - Backpacks", "Items discovered", "$item_explorer", "Comma-separated list of items. Slot will be active only if any item is discovered or list is not set."); backpacksSlotEnabled.SettingChanged += delegate { UpdateSlots(); }; bbhQuiverSlotEnabled = config("Mod - BowsBeforeHoes", "Enabled", defaultValue: true, "Enable quiver slot"); bbhQuiverSlotName = config("Mod - BowsBeforeHoes", "Name", "$bbh_slot_quiver", "Slot name"); bbhQuiverSlotGlobalKey = config("Mod - BowsBeforeHoes", "Global keys", "", "Comma-separated list of global keys and player unique keys. Slot will be active only if any key is enabled or list is not set."); bbhQuiverSlotItemDiscovered = config("Mod - BowsBeforeHoes", "Items discovered", "$item_quiver_blackforest,$item_quiver_seeker,$item_quiver_leather,$item_quiver_odinplus,$item_quiver_plainslox", "Comma-separated list of items. Slot will be active only if any item is discovered or list is not set."); bbhQuiverSlotEnabled.SettingChanged += delegate { UpdateSlots(); }; circletExtendedSlotEnabled = config("Mod - CircletExtended", "Enabled", defaultValue: true, "Enable circlet slot"); circletExtendedSlotName = config("Mod - CircletExtended", "Name", "Circlet", "Slot name"); circletExtendedSlotGlobalKey = config("Mod - CircletExtended", "Global keys", "", "Comma-separated list of global keys and player unique keys. Slot will be active only if any key is enabled or list is not set."); circletExtendedSlotItemDiscovered = config("Mod - CircletExtended", "Items discovered", "$item_helmet_dverger", "Comma-separated list of items. Slot will be active only if any item is discovered or list is not set."); circletExtendedSlotEnabled.SettingChanged += delegate { UpdateSlots(); }; hipLanternSlotEnabled = config("Mod - HipLantern", "Enabled", defaultValue: true, "Enable hip lantern slot"); hipLanternSlotName = config("Mod - HipLantern", "Name", "Lantern", "Slot name"); hipLanternSlotGlobalKey = config("Mod - HipLantern", "Global keys", "", "Comma-separated list of global keys and player unique keys. Slot will be active only if any key is enabled or list is not set."); hipLanternSlotItemDiscovered = config("Mod - HipLantern", "Items discovered", "$item_hiplantern", "Comma-separated list of items. Slot will be active only if any item is discovered or list is not set."); hipLanternSlotEnabled.SettingChanged += delegate { UpdateSlots(); }; jewelcraftingNeckSlotEnabled = config("Mod - Jewelcrafting - Neck", "Enabled", defaultValue: true, "Enable neck slot"); jewelcraftingNeckSlotName = config("Mod - Jewelcrafting - Neck", "Name", "Neck", "Slot name"); jewelcraftingNeckSlotGlobalKey = config("Mod - Jewelcrafting - Neck", "Global keys", "", "Comma-separated list of global keys and player unique keys. Slot will be active only if any key is enabled or list is not set."); jewelcraftingNeckSlotItemDiscovered = config("Mod - Jewelcrafting - Neck", "Items discovered", "$jc_necklace_red,$jc_necklace_green,$jc_necklace_blue,$jc_necklace_yellow,$jc_necklace_purple,$jc_necklace_orange,$jc_necklace_dvergrnecklace,$jc_necklace_eitrnecklace,$jc_necklace_fireresistnecklace,$jc_necklace_frostresistnecklace,$jc_necklace_poisonresistnecklace,", "Comma-separated list of items. Slot will be active only if any item is discovered or list is not set."); jewelcraftingNeckSlotEnabled.SettingChanged += delegate { UpdateSlots(); }; jewelcraftingRingSlotEnabled = config("Mod - Jewelcrafting - Ring", "Enabled", defaultValue: true, "Enable Ring slot"); jewelcraftingRingSlotName = config("Mod - Jewelcrafting - Ring", "Name", "Finger", "Slot name"); jewelcraftingRingSlotGlobalKey = config("Mod - Jewelcrafting - Ring", "Global keys", "", "Comma-separated list of global keys and player unique keys. Slot will be active only if any key is enabled or list is not set."); jewelcraftingRingSlotItemDiscovered = config("Mod - Jewelcrafting - Ring", "Items discovered", "$jc_ring_purple,$jc_ring_green,$jc_ring_red,$jc_ring_blue,$jc_ring_black,$jc_ring_dvergrring,$jc_ring_eitrring,$jc_ring_fireresistring,$jc_ring_frostresistring,$jc_ring_poisonresistring", "Comma-separated list of items. Slot will be active only if any item is discovered or list is not set."); jewelcraftingRingSlotEnabled.SettingChanged += delegate { UpdateSlots(); }; magicPluginTomeSlotEnabled = config("Mod - Magic Plugin - Tome", "Enabled", defaultValue: true, "Enable tome slot"); magicPluginTomeSlotName = config("Mod - Magic Plugin - Tome", "Name", "$bmp_tomeslot", "Slot name"); magicPluginTomeSlotGlobalKey = config("Mod - Magic Plugin - Tome", "Global keys", "", "Comma-separated list of global keys and player unique keys. Slot will be active only if any key is enabled or list is not set."); magicPluginTomeSlotItemDiscovered = config("Mod - Magic Plugin - Tome", "Items discovered", "$bmp_advance_magicbook,$bmp_beginners_magicbook", "Comma-separated list of items. Slot will be active only if any item is discovered or list is not set."); magicPluginTomeSlotEnabled.SettingChanged += delegate { UpdateSlots(); }; magicPluginEarringSlotEnabled = config("Mod - Magic Plugin - Earring", "Enabled", defaultValue: true, "Enable earring slot"); magicPluginEarringSlotName = config("Mod - Magic Plugin - Earring", "Name", "$bmp_earringslot", "Slot name"); magicPluginEarringSlotGlobalKey = config("Mod - Magic Plugin - Earring", "Global keys", "", "Comma-separated list of global keys and player unique keys. Slot will be active only if any key is enabled or list is not set."); magicPluginEarringSlotItemDiscovered = config("Mod - Magic Plugin - Earring", "Items discovered", "$bmp_dvergr_earring,$bmp_fireresist_earring,$bmp_frostresist_earring,$bmp_poisonresist_earring,$bmp_eitr_earring", "Comma-separated list of items. Slot will be active only if any item is discovered or list is not set."); magicPluginEarringSlotEnabled.SettingChanged += delegate { UpdateSlots(); }; } public static void UpdateSlots() { CollectionExtensions.Do<CustomSlot>((IEnumerable<CustomSlot>)CustomSlot.slots, (Action<CustomSlot>)delegate(CustomSlot slot) { slot.RemoveSlot(); }); CustomSlot.slots.Clear(); CollectionExtensions.Do<string>(from s in slotsOrder.Value.Split(new char[1] { ',' }) select s.Trim() into s where !Utility.IsNullOrWhiteSpace(s) select s, (Action<string>)InitSlot); List<string> vanillaSlots = CustomSlot.VanillaOrder.Split(new char[1] { ',' }).ToList(); CollectionExtensions.Do<CustomSlot>((IEnumerable<CustomSlot>)CustomSlot.slots, (Action<CustomSlot>)delegate(CustomSlot slot) { vanillaSlots.Remove(slot.slotID); }); CollectionExtensions.Do<string>((IEnumerable<string>)vanillaSlots, (Action<string>)InitSlot); CollectionExtensions.Do<CustomSlot>((IEnumerable<CustomSlot>)CustomSlot.slots, (Action<CustomSlot>)TryAddSlot); } public static void TryAddSlot(CustomSlot slot) { if (slot.RemoveSlot()) { LogInfo($"Slot {slot} was removed"); } if (slot.AddSlot()) { LogInfo($"Slot {slot} was added"); } else if (slot.initialized) { LogWarning($"Error while trying to add new slot {slot}."); } } public static void InitSlot(string slot) { if (slot == null) { return; } switch (slot.Length) { case 18: switch (slot[0]) { case 'A': if (slot == "AdventureBackpacks" && adventureBackpackSlotEnabled.Value) { new AdventureBackpacksSlot(); } break; case 'M': if (slot == "MagicPluginEarring" && magicPluginEarringSlotEnabled.Value) { new MagicPluginEarringSlot(); } break; } break; case 15: switch (slot[0]) { case 'M': if (slot == "MagicPluginTome" && magicPluginTomeSlotEnabled.Value) { new MagicPluginTomeSlot(); } break; case 'C': if (slot == "CircletExtended" && circletExtendedSlotEnabled.Value) { new CircletExtendedSlot(); } break; } break; case 17: switch (slot[13]) { case 'N': if (slot == "JewelcraftingNeck" && jewelcraftingNeckSlotEnabled.Value) { new JewelcraftingNeckSlot(); } break; case 'R': if (slot == "JewelcraftingRing" && jewelcraftingRingSlotEnabled.Value) { new JewelcraftingRingSlot(); } break; } break; case 9: if (slot == "Backpacks" && backpacksSlotEnabled.Value) { new BackpacksSlot(); } break; case 14: if (slot == "BowsBeforeHoes" && bbhQuiverSlotEnabled.Value) { new BowsBeforeHoesSlot(); } break; case 10: if (slot == "HipLantern" && hipLanternSlotEnabled.Value) { new HipLanternSlot(); } break; case 11: case 12: case 13: case 16: break; } } public static void LogInfo(object data) { if (loggingEnabled.Value) { ((BaseUnityPlugin)instance).Logger.LogInfo(data); } } public static void LogMessage(object data) { ((BaseUnityPlugin)instance).Logger.LogMessage(data); } public static void LogWarning(object data) { ((BaseUnityPlugin)instance).Logger.LogWarning(data); } private ConfigEntry<T> config<T>(string group, string name, T defaultValue, ConfigDescription description, bool synchronizedSetting = true) { ConfigEntry<T> val = ((BaseUnityPlugin)this).Config.Bind<T>(group, name, defaultValue, description); SyncedConfigEntry<T> syncedConfigEntry = configSync.AddConfigEntry<T>(val); syncedConfigEntry.SynchronizedConfig = synchronizedSetting; return val; } private ConfigEntry<T> config<T>(string group, string name, T defaultValue, string description, bool synchronizedSetting = true) { //IL_000c: Unknown result type (might be due to invalid IL or missing references) //IL_0018: Expected O, but got Unknown return config(group, name, defaultValue, new ConfigDescription(description, (AcceptableValueBase)null, Array.Empty<object>()), synchronizedSetting); } } public class MagicPluginTomeSlot : CustomSlot { public const string ID = "MagicPluginTome"; public const string pluginID = "blacks7ar.MagicPlugin"; public MagicPluginTomeSlot() { CustomSlot.slots.Add(this); GUID = "blacks7ar.MagicPlugin"; if (!base.PluginInstalled) { return; } Assembly assembly = Assembly.GetAssembly(((object)Chainloader.PluginInfos["blacks7ar.MagicPlugin"].Instance).GetType()); MethodInfo isValid = AccessTools.Method(assembly.GetType("MagicPlugin.Functions.MagicSlot"), "IsTomeItem", (Type[])null, (Type[])null); if (isValid == null) { ExtraSlotsCustomSlots.LogWarning("MagicPlugin mod is loaded but MagicPlugin.Functions.MagicSlot:IsTomeItem is not found"); return; } slotID = "MagicPluginTome"; itemIsValid = delegate(ItemData item) { int result; if (item != null) { MethodInfo methodInfo = isValid; object[] parameters = (object[])(object)new ItemData[1] { item }; result = (((bool)methodInfo.Invoke(null, parameters)) ? 1 : 0); } else { result = 0; } return (byte)result != 0; }; getName = () => ExtraSlotsCustomSlots.magicPluginTomeSlotName.Value; isActive = () => API.IsAnyGlobalKeyActive(ExtraSlotsCustomSlots.magicPluginTomeSlotGlobalKey.Value) || API.IsAnyMaterialDiscovered(ExtraSlotsCustomSlots.magicPluginTomeSlotItemDiscovered.Value); initialized = true; } } public class MagicPluginEarringSlot : CustomSlot { public const string ID = "MagicPluginEarring"; public const string pluginID = "blacks7ar.MagicPlugin"; public MagicPluginEarringSlot() { CustomSlot.slots.Add(this); GUID = "blacks7ar.MagicPlugin"; if (!base.PluginInstalled) { return; } Assembly assembly = Assembly.GetAssembly(((object)Chainloader.PluginInfos["blacks7ar.MagicPlugin"].Instance).GetType()); MethodInfo isValid = AccessTools.Method(assembly.GetType("MagicPlugin.Functions.MagicSlot"), "IsEarringItem", (Type[])null, (Type[])null); if (isValid == null) { ExtraSlotsCustomSlots.LogWarning("MagicPlugin mod is loaded but MagicPlugin.Functions.MagicSlot:IsEarringItem is not found"); return; } slotID = "MagicPluginEarring"; itemIsValid = delegate(ItemData item) { int result; if (item != null) { MethodInfo methodInfo = isValid; object[] parameters = (object[])(object)new ItemData[1] { item }; result = (((bool)methodInfo.Invoke(null, parameters)) ? 1 : 0); } else { result = 0; } return (byte)result != 0; }; getName = () => ExtraSlotsCustomSlots.magicPluginEarringSlotName.Value; isActive = () => API.IsAnyGlobalKeyActive(ExtraSlotsCustomSlots.magicPluginEarringSlotGlobalKey.Value) || API.IsAnyMaterialDiscovered(ExtraSlotsCustomSlots.magicPluginEarringSlotItemDiscovered.Value); initialized = true; } } [HarmonyPatch] public static class MagicPlugin_MagicSlot_AddCustomSlot_PreventCustomSlotAddition { public static MethodBase target; public static bool Prepare() { if (!Chainloader.PluginInfos.TryGetValue("blacks7ar.MagicPlugin", out var value)) { return false; } target = AccessTools.Method(Assembly.GetAssembly(((object)value.Instance).GetType()).GetType("MagicPlugin.Functions.MagicSlot"), "AddCustomSlot", (Type[])null, (Type[])null); if (target == null) { return false; } ExtraSlotsCustomSlots.LogInfo("MagicPlugin.Functions.MagicSlot:AddCustomSlot method will be patched to prevent adding custom slot call"); return true; } public static MethodBase TargetMethod() { return target; } public static bool Prefix() { return false; } } [HarmonyPatch] public static class MagicPlugin_AzuEPI_IsLoaded_CustomSlotHandle { public static MethodBase target; public static bool Prepare() { if (!Chainloader.PluginInfos.TryGetValue("blacks7ar.MagicPlugin", out var value)) { return false; } target = AccessTools.Method(Assembly.GetAssembly(((object)value.Instance).GetType()).GetType("AzuExtendedPlayerInventory.API"), "IsLoaded", (Type[])null, (Type[])null); if (target == null) { return false; } ExtraSlotsCustomSlots.LogInfo("MagicPlugin.AzuExtendedPlayerInventory.API:IsLoaded method will be patched to enable custom slot handling"); return true; } public static MethodBase TargetMethod() { return target; } public static void Postfix(ref bool __result) { __result = true; } } } namespace ExtraSlotsCustomSlots.AdventureBackpacksCustomSlot { [Serializable] public class HumanoidAdventureBackpack { public ItemData backpack; } public static class HumanoidExtension { private static readonly ConditionalWeakTable<Humanoid, HumanoidAdventureBackpack> data = new ConditionalWeakTable<Humanoid, HumanoidAdventureBackpack>(); public static HumanoidAdventureBackpack GetBackpackData(this Humanoid humanoid) { return data.GetOrCreateValue(humanoid); } public static ItemData GetAdventureBackpack(this Humanoid humanoid) { return humanoid.GetBackpackData().backpack; } public static ItemData SetAdventureBackpack(this Humanoid humanoid, ItemData item) { return humanoid.GetBackpackData().backpack = item; } } [Serializable] public class VisEquipmentAdventureBackpack { public string m_backpackItem = ""; public List<GameObject> m_backpackItemInstances; public int m_currentbackpackItemHash = 0; public static readonly int s_backpackItem = StringExtensionMethods.GetStableHashCode("AdventureBackpackItem"); } public static class VisEquipmentExtension { [HarmonyPatch(typeof(VisEquipment), "UpdateEquipmentVisuals")] public static class VisEquipment_UpdateEquipmentVisuals_CustomItemType { private static void Prefix(VisEquipment __instance) { if (!AdventureBackpacksSlot.IsActive) { return; } int hash = 0; ZDO zDO = __instance.m_nview.GetZDO(); if (zDO != null) { hash = zDO.GetInt(VisEquipmentAdventureBackpack.s_backpackItem, 0); } else { VisEquipmentAdventureBackpack backpackData = __instance.GetBackpackData(); if (!string.IsNullOrEmpty(backpackData.m_backpackItem)) { hash = StringExtensionMethods.GetStableHashCode(backpackData.m_backpackItem); } } if (__instance.SetBackpackEquipped(hash)) { __instance.UpdateLodgroup(); } } } [HarmonyPatch(typeof(Humanoid), "SetupVisEquipment")] public static class Humanoid_SetupVisEquipment_CustomItemType { private static void Postfix(Humanoid __instance, VisEquipment visEq) { if (AdventureBackpacksSlot.IsActive) { ItemData adventureBackpack = __instance.GetAdventureBackpack(); visEq.SetBackpackItem((adventureBackpack != null && (Object)(object)adventureBackpack.m_dropPrefab != (Object)null) ? ((Object)adventureBackpack.m_dropPrefab).name : ""); } } } private static readonly ConditionalWeakTable<VisEquipment, VisEquipmentAdventureBackpack> data = new ConditionalWeakTable<VisEquipment, VisEquipmentAdventureBackpack>(); public static VisEquipmentAdventureBackpack GetBackpackData(this VisEquipment visEquipment) { return data.GetOrCreateValue(visEquipment); } public static void SetBackpackItem(this VisEquipment visEquipment, string name) { VisEquipmentAdventureBackpack backpackData = visEquipment.GetBackpackData(); if (!(backpackData.m_backpackItem == name)) { backpackData.m_backpackItem = name; if (visEquipment.m_nview.GetZDO() != null && visEquipment.m_nview.IsOwner()) { visEquipment.m_nview.GetZDO().Set(VisEquipmentAdventureBackpack.s_backpackItem, (!string.IsNullOrEmpty(name)) ? StringExtensionMethods.GetStableHashCode(name) : 0, false); } } } public static bool SetBackpackEquipped(this VisEquipment visEquipment, int hash) { VisEquipmentAdventureBackpack backpackData = visEquipment.GetBackpackData(); if (backpackData.m_currentbackpackItemHash == hash) { return false; } if (backpackData.m_backpackItemInstances != null) { foreach (GameObject backpackItemInstance in backpackData.m_backpackItemInstances) { if (Object.op_Implicit((Object)(object)visEquipment.m_lodGroup)) { Utils.RemoveFromLodgroup(visEquipment.m_lodGroup, backpackItemInstance); } Object.Destroy((Object)(object)backpackItemInstance); } backpackData.m_backpackItemInstances = null; } backpackData.m_currentbackpackItemHash = hash; if (hash != 0) { backpackData.m_backpackItemInstances = visEquipment.AttachArmor(hash, -1); CustomItemType.ReorderBones?.Invoke(visEquipment, hash, backpackData.m_backpackItemInstances); } return true; } } public static class CustomItemType { [HarmonyPatch(typeof(Humanoid), "EquipItem")] public static class Humanoid_EquipItem_CustomItemType { [HarmonyPriority(800)] [HarmonyBefore(new string[] { "vapok.mods.adventurebackpacks" })] private static void Postfix(Humanoid __instance, ItemData item, ref bool __result, bool triggerEquipEffects) { if (AdventureBackpacksSlot.IsActive && !__instance.IsItemEquiped(item) && IsBackpack(item)) { if (__instance.GetAdventureBackpack() != null) { __instance.UnequipItem(__instance.GetAdventureBackpack(), triggerEquipEffects); __instance.m_visEquipment.UpdateEquipmentVisuals(); } __instance.SetAdventureBackpack(item); if (__instance.IsItemEquiped(item)) { item.m_equipped = true; __result = true; } __instance.SetupEquipment(); } } } [HarmonyPatch(typeof(Humanoid), "UnequipItem")] public static class Humanoid_UnequipItem_CustomItemType { private static void Postfix(Humanoid __instance, ItemData item) { if (AdventureBackpacksSlot.IsActive && IsBackpack(item) && __instance.GetAdventureBackpack() == item) { __instance.SetAdventureBackpack(null); __instance.SetupEquipment(); } } } [HarmonyPatch(typeof(Humanoid), "UnequipAllItems")] public class Humanoid_UnequipAllItems_CustomItemType { public static void Postfix(Humanoid __instance) { if (AdventureBackpacksSlot.IsActive) { __instance.UnequipItem(__instance.GetAdventureBackpack(), false); } } } [HarmonyPatch(typeof(Humanoid), "IsItemEquiped")] public static class Humanoid_IsItemEquiped_CustomItemType { private static void Postfix(Humanoid __instance, ItemData item, ref bool __result) { if (AdventureBackpacksSlot.IsActive && IsBackpack(item)) { __result = __result || __instance.GetAdventureBackpack() == item; } } } [HarmonyPatch(typeof(ItemData), "IsEquipable")] public static class ItemDropItemData_IsEquipable_CustomItemType { private static void Postfix(ItemData __instance, ref bool __result) { //IL_001a: Unknown result type (might be due to invalid IL or missing references) //IL_001f: Unknown result type (might be due to invalid IL or missing references) if (AdventureBackpacksSlot.IsActive) { __result = __result || __instance.m_shared.m_itemType == AdventureBackpackItem.GetItemType(); } } } [HarmonyPatch(typeof(Player), "UnequipDeathDropItems")] private static class Player_UnequipDeathDropItems_AdventureBackpack { private static void Prefix(Player __instance) { if (AdventureBackpacksSlot.IsActive) { ItemData adventureBackpack = ((Humanoid)(object)__instance).GetAdventureBackpack(); if (adventureBackpack != null) { adventureBackpack.m_equipped = false; ((Humanoid)(object)__instance).SetAdventureBackpack(null); ((Humanoid)__instance).SetupEquipment(); } } } } [HarmonyPatch(typeof(Player), "GetEquipmentEitrRegenModifier")] private static class Player_GetEquipmentEitrRegenModifier_AdventureBackpack { private static void Postfix(Player __instance, ref float __result) { if (AdventureBackpacksSlot.IsActive) { ItemData adventureBackpack = ((Humanoid)(object)__instance).GetAdventureBackpack(); if (adventureBackpack != null) { __result += adventureBackpack.m_shared.m_eitrRegenModifier; } } } } [HarmonyPatch(typeof(Inventory), "Changed")] public static class Inventory_Changed_CustomItemType { private static void Prefix(Inventory __instance) { if (!AdventureBackpacksSlot.IsActive) { return; } Player localPlayer = Player.m_localPlayer; if (__instance == ((localPlayer != null) ? ((Humanoid)localPlayer).GetInventory() : null)) { ItemData adventureBackpack = ((Humanoid)(object)Player.m_localPlayer).GetAdventureBackpack(); if (adventureBackpack != null && !__instance.ContainsItem(adventureBackpack)) { ((Humanoid)(object)Player.m_localPlayer).SetAdventureBackpack(null); ((Humanoid)Player.m_localPlayer).SetupEquipment(); } } } } [HarmonyPatch(typeof(Humanoid), "GetEquipmentWeight")] public static class Humanoid_GetEquipmentWeight_CustomItemType { private static void Postfix(Humanoid __instance, ref float __result) { if (AdventureBackpacksSlot.IsActive) { ItemData adventureBackpack = __instance.GetAdventureBackpack(); if (adventureBackpack != null) { __result += adventureBackpack.m_shared.m_weight; } } } } [HarmonyPatch(typeof(Humanoid), "UpdateEquipment")] private static class Humanoid_UpdateEquipment_AdventureBackpackDurabilityDrain { private static void Postfix(Humanoid __instance, float dt) { if (AdventureBackpacksSlot.IsActive) { ItemData adventureBackpack = __instance.GetAdventureBackpack(); if (adventureBackpack != null && adventureBackpack.m_shared.m_useDurability) { __instance.DrainEquipedItemDurability(adventureBackpack, dt); } } } } [HarmonyPatch(typeof(Player), "ApplyArmorDamageMods")] private static class Player_ApplyArmorDamageMods_AdventureBackpack { private static void Postfix(Player __instance, ref DamageModifiers mods) { if (AdventureBackpacksSlot.IsActive) { ItemData adventureBackpack = ((Humanoid)(object)__instance).GetAdventureBackpack(); if (adventureBackpack != null) { ((DamageModifiers)(ref mods)).Apply(adventureBackpack.m_shared.m_damageModifiers); } } } } [HarmonyPatch(typeof(Player), "UpdateModifiers")] private static class Player_UpdateModifiers_AdventureBackpack { private static void Postfix(Player __instance) { if (!AdventureBackpacksSlot.IsActive || Player.s_equipmentModifierSourceFields == null) { return; } ItemData adventureBackpack = ((Humanoid)(object)__instance).GetAdventureBackpack(); if (adventureBackpack != null) { for (int i = 0; i < __instance.m_equipmentModifierValues.Length; i++) { __instance.m_equipmentModifierValues[i] += (float)Player.s_equipmentModifierSourceFields[i].GetValue(adventureBackpack.m_shared); } } } } [HarmonyPatch(typeof(Humanoid), "GetSetCount")] private static class Humanoid_GetSetCount_AdventureBackpack { private static void Postfix(Humanoid __instance, string setName, ref int __result) { if (AdventureBackpacksSlot.IsActive) { ItemData adventureBackpack = __instance.GetAdventureBackpack(); if (adventureBackpack != null && adventureBackpack.m_shared.m_setName == setName) { __result++; } } } } [HarmonyPatch(typeof(Humanoid), "UpdateEquipmentStatusEffects")] private static class Humanoid_UpdateEquipmentStatusEffects_AdventureBackpack { private static void Prefix(Humanoid __instance) { if (!AdventureBackpacksSlot.IsActive) { return; } tempEffects.Clear(); ItemData adventureBackpack = __instance.GetAdventureBackpack(); if (adventureBackpack != null) { if (Object.op_Implicit((Object)(object)adventureBackpack.m_shared.m_equipStatusEffect)) { tempEffects.Add(adventureBackpack.m_shared.m_equipStatusEffect); } if (__instance.HaveSetEffect(adventureBackpack)) { tempEffects.Add(adventureBackpack.m_shared.m_setStatusEffect); } } } private static void Postfix(Humanoid __instance) { foreach (StatusEffect item in tempEffects.Where((StatusEffect item) => !__instance.m_equipmentStatusEffects.Contains(item))) { ((Character)__instance).m_seman.AddStatusEffect(item, false, 0, 0f); } __instance.m_equipmentStatusEffects.UnionWith(tempEffects); tempEffects.Clear(); } } [HarmonyPatch(typeof(SEMan), "RemoveStatusEffect", new Type[] { typeof(int), typeof(bool) })] private static class SEMan_RemoveStatusEffect_AdventureBackpackPreventRemoval { private static void Prefix(SEMan __instance, ref int nameHash) { Player localPlayer = Player.m_localPlayer; if (__instance != ((localPlayer != null) ? ((Character)localPlayer).GetSEMan() : null) || tempEffects.Count == 0) { return; } foreach (StatusEffect tempEffect in tempEffects) { if (tempEffect.NameHash() == nameHash) { nameHash = 0; } } } } private static readonly HashSet<StatusEffect> tempEffects = new HashSet<StatusEffect>(); public static Func<ItemData, bool> IsBackpack; public static Action<VisEquipment, int, List<GameObject>> ReorderBones; internal static void InitBackpackFunc(Func<ItemData, bool> isValid, Assembly assembly) { IsBackpack = isValid; MethodInfo reorderBones = AccessTools.Method(assembly.GetType("Vapok.Common.Tools.BoneReorder"), "ReorderBones", (Type[])null, (Type[])null); if (reorderBones != null) { ReorderBones = delegate(VisEquipment visEq, int hash, List<GameObject> gameObjects) { reorderBones.Invoke(null, new object[3] { visEq, hash, gameObjects }); }; } else { ExtraSlotsCustomSlots.LogWarning("AdventureBackpacks mod is loaded but Vapok.Common.Tools:ReorderBones is not found"); } } } public static class AdventureBackpackItem { [HarmonyPatch(typeof(Player), "AddKnownItem")] public static class Player_AddKnownItem_AdventureBackpackStats { private static void Postfix(Player __instance, ref ItemData item) { if (AdventureBackpacksSlot.IsActive && !__instance.m_knownMaterial.Contains(item.m_shared.m_name) && CustomItemType.IsBackpack(item)) { PatchBackpackItemData(item); } } } [HarmonyPatch(typeof(Player), "OnSpawned")] public class Player_OnSpawned_AdventureBackpackStats { public static void Postfix(Player __instance) { if (AdventureBackpacksSlot.IsActive && !((Object)(object)__instance != (Object)(object)Player.m_localPlayer)) { PatchInventory(((Humanoid)__instance).GetInventory()); } } } [HarmonyPatch(typeof(Inventory), "Load")] public class Inventory_Load_AdventureBackpackStats { public static void Postfix(Inventory __instance) { if (AdventureBackpacksSlot.IsActive) { PatchInventory(__instance); } } } [HarmonyPatch(typeof(ItemDrop), "Start")] public static class ItemDrop_Start_AdventureBackpackStats { private static void Postfix(ref ItemDrop __instance) { if (AdventureBackpacksSlot.IsActive && CustomItemType.IsBackpack(__instance.m_itemData)) { PatchBackpackItemData(__instance.m_itemData); } } } [HarmonyPatch(typeof(ObjectDB), "Awake")] public static class ObjectDB_Awake_ChangeBackpackItemType { [HarmonyPriority(0)] [HarmonyAfter(new string[] { "vapok.mods.adventurebackpacks" })] private static void Postfix(ObjectDB __instance) { if (AdventureBackpacksSlot.IsActive && __instance.m_items.Count != 0 && !((Object)(object)__instance.GetItemPrefab("Wood") == (Object)null)) { UpdateBackpacksItemType(); } } } [HarmonyPatch(typeof(ObjectDB), "CopyOtherDB")] public static class ObjectDB_CopyOtherDB_AddPrefab { [HarmonyPriority(0)] [HarmonyAfter(new string[] { "vapok.mods.adventurebackpacks" })] private static void Postfix(ObjectDB __instance) { if (AdventureBackpacksSlot.IsActive && __instance.m_items.Count != 0 && !((Object)(object)__instance.GetItemPrefab("Wood") == (Object)null)) { UpdateBackpacksItemType(); } } } internal static ItemType GetItemType() { //IL_0014: Unknown result type (might be due to invalid IL or missing references) //IL_000f: Unknown result type (might be due to invalid IL or missing references) //IL_0017: Unknown result type (might be due to invalid IL or missing references) if (!AdventureBackpacksSlot.IsActive) { return (ItemType)17; } return (ItemType)16; } internal static void PatchBackpackItemData(ItemData itemData) { //IL_0011: Unknown result type (might be due to invalid IL or missing references) //IL_0016: Unknown result type (might be due to invalid IL or missing references) //IL_0021: Unknown result type (might be due to invalid IL or missing references) //IL_0026: Unknown result type (might be due to invalid IL or missing references) if (itemData != null) { itemData.m_shared.m_itemType = GetItemType(); itemData.m_shared.m_attachOverride = GetItemType(); } } internal static void PatchInventory(Inventory inventory) { if (!AdventureBackpacksSlot.IsActive || inventory == null) { return; } foreach (ItemData item in from item in inventory.GetAllItems() where CustomItemType.IsBackpack(item) select item) { PatchBackpackItemData(item); } } internal static void PatchBackpackItemOnConfigChange() { UpdateBackpacksItemType(); Player localPlayer = Player.m_localPlayer; PatchInventory((localPlayer != null) ? ((Humanoid)localPlayer).GetInventory() : null); } internal static void UpdateBackpacksItemType() { if (!AdventureBackpacksSlot.IsActive || !Object.op_Implicit((Object)(object)ObjectDB.instance)) { return; } foreach (GameObject item in ObjectDB.instance.m_items) { if ((Object)(object)item != (Object)null) { ItemData val = item.GetComponent<ItemDrop>()?.m_itemData; if (val != null && CustomItemType.IsBackpack(val)) { PatchBackpackItemData(val); } } } } } } namespace System.Runtime.CompilerServices { [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 ServerSync { [PublicAPI] internal abstract class OwnConfigEntryBase { public object? LocalBaseValue; public bool SynchronizedConfig = true; public abstract ConfigEntryBase BaseConfig { get; } } [PublicAPI] internal class SyncedConfigEntry<T> : OwnConfigEntryBase { public readonly ConfigEntry<T> SourceConfig; public override ConfigEntryBase BaseConfig => (ConfigEntryBase)(object)SourceConfig; public T Value { get { return SourceConfig.Value; } set { SourceConfig.Value = value; } } public SyncedConfigEntry(ConfigEntry<T> sourceConfig) { SourceConfig = sourceConfig; } public void AssignLocalValue(T value) { if (LocalBaseValue == null) { Value = value; } else { LocalBaseValue = value; } } } internal abstract class CustomSyncedValueBase { public object? LocalBaseValue; public readonly string Identifier; public readonly Type Type; private object? boxedValue; protected bool localIsOwner; public readonly int Priority; public object? BoxedValue { get { return boxedValue; } set { boxedValue = value; this.ValueChanged?.Invoke(); } } public event Action? ValueChanged; protected CustomSyncedValueBase(ConfigSync configSync, string identifier, Type type, int priority) { Priority = priority; Identifier = identifier; Type = type; configSync.AddCustomValue(this); localIsOwner = configSync.IsSourceOfTruth; configSync.SourceOfTruthChanged += delegate(bool truth) { localIsOwner = truth; }; } } [PublicAPI] internal sealed class CustomSyncedValue<T> : CustomSyncedValueBase { public T Value { get { return (T)base.BoxedValue; } set { base.BoxedValue = value; } } public CustomSyncedValue(ConfigSync configSync, string identifier, T value = default(T), int priority = 0) : base(configSync, identifier, typeof(T), priority) { Value = value; } public void AssignLocalValue(T value) { if (localIsOwner) { Value = value; } else { LocalBaseValue = value; } } } internal class ConfigurationManagerAttributes { [UsedImplicitly] public bool? ReadOnly = false; } [PublicAPI] internal class ConfigSync { [HarmonyPatch(typeof(ZRpc), "HandlePackage")] private static class SnatchCurrentlyHandlingRPC { public static ZRpc? currentRpc; [HarmonyPrefix] private static void Prefix(ZRpc __instance) { currentRpc = __instance; } } [HarmonyPatch(typeof(ZNet), "Awake")] internal static class RegisterRPCPatch { [HarmonyPostfix] private static void Postfix(ZNet __instance) { isServer = __instance.IsServer(); foreach (ConfigSync configSync2 in configSyncs) { ZRoutedRpc.instance.Register<ZPackage>(configSync2.Name + " ConfigSync", (Action<long, ZPackage>)configSync2.RPC_FromOtherClientConfigSync); if (isServer) { configSync2.InitialSyncDone = true; Debug.Log((object)("Registered '" + configSync2.Name + " ConfigSync' RPC - waiting for incoming connections")); } } if (isServer) { ((MonoBehaviour)__instance).StartCoroutine(WatchAdminListChanges()); } static void SendAdmin(List<ZNetPeer> peers, bool isAdmin) { ZPackage package = ConfigsToPackage(null, null, new PackageEntry[1] { new PackageEntry { section = "Internal", key = "lockexempt", type = typeof(bool), value = isAdmin } }); ConfigSync configSync = configSyncs.First(); if (configSync != null) { ((MonoBehaviour)ZNet.instance).StartCoroutine(configSync.sendZPackage(peers, package)); } } static IEnumerator WatchAdminListChanges() { MethodInfo listContainsId = AccessTools.DeclaredMethod(typeof(ZNet), "ListContainsId", (Type[])null, (Type[])null); SyncedList adminList = (SyncedList)AccessTools.DeclaredField(typeof(ZNet), "m_adminList").GetValue(ZNet.instance); List<string> CurrentList = new List<string>(adminList.GetList()); while (true) { yield return (object)new WaitForSeconds(30f); if (!adminList.GetList().SequenceEqual(CurrentList)) { CurrentList = new List<string>(adminList.GetList()); List<ZNetPeer> adminPeer = ZNet.instance.GetPeers().Where(delegate(ZNetPeer p) { string hostName = p.m_rpc.GetSocket().GetHostName(); return ((object)listContainsId == null) ? adminList.Contains(hostName) : ((bool)listContainsId.Invoke(ZNet.instance, new object[2] { adminList, hostName })); }).ToList(); List<ZNetPeer> nonAdminPeer = ZNet.instance.GetPeers().Except(adminPeer).ToList(); SendAdmin(nonAdminPeer, isAdmin: false); SendAdmin(adminPeer, isAdmin: true); } } } } } [HarmonyPatch(typeof(ZNet), "OnNewConnection")] private static class RegisterClientRPCPatch { [HarmonyPostfix] private static void Postfix(ZNet __instance, ZNetPeer peer) { if (__instance.IsServer()) { return; } foreach (ConfigSync configSync in configSyncs) { peer.m_rpc.Register<ZPackage>(configSync.Name + " ConfigSync", (Action<ZRpc, ZPackage>)configSync.RPC_FromServerConfigSync); } } } private class ParsedConfigs { public readonly Dictionary<OwnConfigEntryBase, object?> configValues = new Dictionary<OwnConfigEntryBase, object>(); public readonly Dictionary<CustomSyncedValueBase, object?> customValues = new Dictionary<CustomSyncedValueBase, object>(); } [HarmonyPatch(typeof(ZNet), "Shutdown")] private class ResetConfigsOnShutdown { [HarmonyPostfix] private static void Postfix() { ProcessingServerUpdate = true; foreach (ConfigSync configSync in configSyncs) { configSync.resetConfigsFromServer(); configSync.IsSourceOfTruth = true; configSync.InitialSyncDone = false; } ProcessingServerUpdate = false; } } [HarmonyPatch(typeof(ZNet), "RPC_PeerInfo")] private class SendConfigsAfterLogin { private class BufferingSocket : ISocket { public volatile bool finished = false; public volatile int versionMatchQueued = -1; public readonly List<ZPackage> Package = new List<ZPackage>(); public readonly ISocket Original; public BufferingSocket(ISocket original) { Original = original; } public bool IsConnected() { return Original.IsConnected(); } public ZPackage Recv() { return Original.Recv(); } public int GetSendQueueSize() { return Original.GetSendQueueSize(); } public int GetCurrentSendRate() { return Original.GetCurrentSendRate(); } public bool IsHost() { return Original.IsHost(); } public void Dispose() { Original.Dispose(); } public bool GotNewData() { return Original.GotNewData(); } public void Close() { Original.Close(); } public string GetEndPointString() { return Original.GetEndPointString(); } public void GetAndResetStats(out int totalSent, out int totalRecv) { Original.GetAndResetStats(ref totalSent, ref totalRecv); } public void GetConnectionQuality(out float localQuality, out float remoteQuality, out int ping, out float outByteSec, out float inByteSec) { Original.GetConnectionQuality(ref localQuality, ref remoteQuality, ref ping, ref outByteSec, ref inByteSec); } public ISocket Accept() { return Original.Accept(); } public int GetHostPort() { return Original.GetHostPort(); } public bool Flush() { return Original.Flush(); } public string GetHostName() { return Original.GetHostName(); } public void VersionMatch() { if (finished) { Original.VersionMatch(); } else { versionMatchQueued = Package.Count; } } public void Send(ZPackage pkg) { //IL_0057: Unknown result type (might be due to invalid IL or missing references) //IL_005d: Expected O, but got Unknown int pos = pkg.GetPos(); pkg.SetPos(0); int num = pkg.ReadInt(); if ((num == StringExtensionMethods.GetStableHashCode("PeerInfo") || num == StringExtensionMethods.GetStableHashCode("RoutedRPC") || num == StringExtensionMethods.GetStableHashCode("ZDOData")) && !finished) { ZPackage val = new ZPackage(pkg.GetArray()); val.SetPos(pos); Package.Add(val); } else { pkg.SetPos(pos); Original.Send(pkg); } } } [HarmonyPriority(800)] [HarmonyPrefix] private static void Prefix(ref Dictionary<Assembly, BufferingSocket>? __state, ZNet __instance, ZRpc rpc) { //IL_0078: Unknown result type (might be due to invalid IL or missing references) //IL_007e: Invalid comparison between Unknown and I4 if (__instance.IsServer()) { BufferingSocket value = new BufferingSocket(rpc.GetSocket()); AccessTools.DeclaredField(typeof(ZRpc), "m_socket").SetValue(rpc, value); object? obj = AccessTools.DeclaredMethod(typeof(ZNet), "GetPeer", new Type[1] { typeof(ZRpc) }, (Type[])null).Invoke(__instance, new object[1] { rpc }); ZNetPeer val = (ZNetPeer)((obj is ZNetPeer) ? obj : null); if (val != null && (int)ZNet.m_onlineBackend > 0) { AccessTools.DeclaredField(typeof(ZNetPeer), "m_socket").SetValue(val, value); } if (__state == null) { __state = new Dictionary<Assembly, BufferingSocket>(); } __state[Assembly.GetExecutingAssembly()] = value; } } [HarmonyPostfix] private static void Postfix(Dictionary<Assembly, BufferingSocket> __state, ZNet __instance, ZRpc rpc) { ZRpc rpc2 = rpc; ZNet __instance2 = __instance; Dictionary<Assembly, BufferingSocket> __state2 = __state; ZNetPeer peer; if (__instance2.IsServer()) { object obj = AccessTools.DeclaredMethod(typeof(ZNet), "GetPeer", new Type[1] { typeof(ZRpc) }, (Type[])null).Invoke(__instance2, new object[1] { rpc2 }); peer = (ZNetPeer)((obj is ZNetPeer) ? obj : null); if (peer == null) { SendBufferedData(); } else { ((MonoBehaviour)__instance2).StartCoroutine(sendAsync()); } } void SendBufferedData() { if (rpc2.GetSocket() is BufferingSocket bufferingSocket) { AccessTools.DeclaredField(typeof(ZRpc), "m_socket").SetValue(rpc2, bufferingSocket.Original); object? obj2 = AccessTools.DeclaredMethod(typeof(ZNet), "GetPeer", new Type[1] { typeof(ZRpc) }, (Type[])null).Invoke(__instance2, new object[1] { rpc2 }); ZNetPeer val = (ZNetPeer)((obj2 is ZNetPeer) ? obj2 : null); if (val != null) { AccessTools.DeclaredField(typeof(ZNetPeer), "m_socket").SetValue(val, bufferingSocket.Original); } } BufferingSocket bufferingSocket2 = __state2[Assembly.GetExecutingAssembly()]; bufferingSocket2.finished = true; for (int i = 0; i < bufferingSocket2.Package.Count; i++) { if (i == bufferingSocket2.versionMatchQueued) { bufferingSocket2.Original.VersionMatch(); } bufferingSocket2.Original.Send(bufferingSocket2.Package[i]); } if (bufferingSocket2.Package.Count == bufferingSocket2.versionMatchQueued) { bufferingSocket2.Original.VersionMatch(); } } IEnumerator sendAsync() { foreach (ConfigSync configSync in configSyncs) { List<PackageEntry> entries = new List<PackageEntry>(); if (configSync.CurrentVersion != null) { entries.Add(new PackageEntry { section = "Internal", key = "serverversion", type = typeof(string), value = configSync.CurrentVersion }); } MethodInfo listContainsId = AccessTools.DeclaredMethod(typeof(ZNet), "ListContainsId", (Type[])null, (Type[])null); SyncedList adminList = (SyncedList)AccessTools.DeclaredField(typeof(ZNet), "m_adminList").GetValue(ZNet.instance); entries.Add(new PackageEntry { section = "Internal", key = "lockexempt", type = typeof(bool), value = (((object)listContainsId == null) ? ((object)adminList.Contains(rpc2.GetSocket().GetHostName())) : listContainsId.Invoke(ZNet.instance, new object[2] { adminList, rpc2.GetSocket().GetHostName() })) }); ZPackage package = ConfigsToPackage(configSync.allConfigs.Select((OwnConfigEntryBase c) => c.BaseConfig), configSync.allCustomValues, entries, partial: false); yield return ((MonoBehaviour)__instance2).StartCoroutine(configSync.sendZPackage(new List<ZNetPeer> { peer }, package)); } SendBufferedData(); } } } private class PackageEntry { public string section = null; public string key = null; public Type type = null; public object? value; } [HarmonyPatch(typeof(ConfigEntryBase), "GetSerializedValue")] private static class PreventSavingServerInfo { [HarmonyPrefix] private static bool Prefix(ConfigEntryBase __instance, ref string __result) { OwnConfigEntryBase ownConfigEntryBase = configData(__instance); if (ownConfigEntryBase == null || isWritableConfig(ownConfigEntryBase)) { return true; } __result = TomlTypeConverter.ConvertToString(ownConfigEntryBase.LocalBaseValue, __instance.SettingType); return false; } } [HarmonyPatch(typeof(ConfigEntryBase), "SetSerializedValue")] private static class PreventConfigRereadChangingValues { [HarmonyPrefix] private static bool Prefix(ConfigEntryBase __instance, string value) { OwnConfigEntryBase ownConfigEntryBase = configData(__instance); if (ownConfigEntryBase == null || ownConfigEntryBase.LocalBaseValue == null) { return true; } try { ownConfigEntryBase.LocalBaseValue = TomlTypeConverter.ConvertToValue(value, __instance.SettingType); } catch (Exception ex) { Debug.LogWarning((object)$"Config value of setting \"{__instance.Definition}\" could not be parsed and will be ignored. Reason: {ex.Message}; Value: {value}"); } return false; } } private class InvalidDeserializationTypeException : Exception { public string expected = null; public string received = null; public string field = ""; } public static bool ProcessingServerUpdate; public readonly string Name; public string? DisplayName; public string? CurrentVersion; public string? MinimumRequiredVersion; public bool ModRequired = false; private bool? forceConfigLocking; private bool isSourceOfTruth = true; private static readonly HashSet<ConfigSync> configSyncs; private readonly HashSet<OwnConfigEntryBase> allConfigs = new HashSet<OwnConfigEntryBase>(); private HashSet<CustomSyncedValueBase> allCustomValues = new HashSet<CustomSyncedValueBase>(); private static bool isServer; private static bool lockExempt; private OwnConfigEntryBase? lockedConfig = null; private const byte PARTIAL_CONFIGS = 1; private const byte FRAGMENTED_CONFIG = 2; private const byte COMPRESSED_CONFIG = 4; private readonly Dictionary<string, SortedDictionary<int, byte[]>> configValueCache = new Dictionary<string, SortedDictionary<int, byte[]>>(); private readonly List<KeyValuePair<long, string>> cacheExpirations = new List<KeyValuePair<long, string>>(); private static long packageCounter; public bool IsLocked { get { bool? flag = forceConfigLocking; bool num; if (!flag.HasValue) { if (lockedConfig == null) { goto IL_0052; } num = ((IConvertible)lockedConfig.BaseConfig.BoxedValue).ToInt32(CultureInfo.InvariantCulture) != 0; } else { num = flag.GetValueOrDefault(); } if (!num) { goto IL_0052; } int result = ((!lockExempt) ? 1 : 0); goto IL_0053; IL_0053: return (byte)result != 0; IL_0052: result = 0; goto IL_0053; } set { forceConfigLocking = value; } } public bool IsAdmin => lockExempt || isSourceOfTruth; public bool IsSourceOfTruth { get { return isSourceOfTruth; } private set { if (value != isSourceOfTruth) { isSourceOfTruth = value; this.SourceOfTruthChanged?.Invoke(value); } } } public bool InitialSyncDone { get; private set; } = false; public event Action<bool>? SourceOfTruthChanged; private event Action? lockedConfigChanged; static ConfigSync() { ProcessingServerUpdate = false; configSyncs = new HashSet<ConfigSync>(); lockExempt = false; packageCounter = 0L; RuntimeHelpers.RunClassConstructor(typeof(VersionCheck).TypeHandle); } public ConfigSync(string name) { Name = name; configSyncs.Add(this); new VersionCheck(this); } public SyncedConfigEntry<T> AddConfigEntry<T>(ConfigEntry<T> configEntry) { ConfigEntry<T> configEntry2 = configEntry; OwnConfigEntryBase ownConfigEntryBase = configData((ConfigEntryBase)(object)configEntry2); SyncedConfigEntry<T> syncedEntry = ownConfigEntryBase as SyncedConfigEntry<T>; if (syncedEntry == null) { syncedEntry = new SyncedConfigEntry<T>(configEntry2); AccessTools.DeclaredField(typeof(ConfigDescription), "<Tags>k__BackingField").SetValue(((ConfigEntryBase)configEntry2).Description, new object[1] { new ConfigurationManagerAttributes() }.Concat(((ConfigEntryBase)configEntry2).Description.Tags ?? Array.Empty<object>()).Concat(new SyncedConfigEntry<T>[1] { syncedEntry }).ToArray()); configEntry2.SettingChanged += delegate { if (!ProcessingServerUpdate && syncedEntry.SynchronizedConfig) { Broadcast(ZRoutedRpc.Everybody, (ConfigEntryBase)configEntry2); } }; allConfigs.Add(syncedEntry); } return syncedEntry; } public SyncedConfigEntry<T> AddLockingConfigEntry<T>(ConfigEntry<T> lockingConfig) where T : IConvertible { if (lockedConfig != null) { throw new Exception("Cannot initialize locking ConfigEntry twice"); } lockedConfig = AddConfigEntry<T>(lockingConfig); lockingConfig.SettingChanged += delegate { this.lockedConfigChanged?.Invoke(); }; return (SyncedConfigEntry<T>)lockedConfig; } internal void AddCustomValue(CustomSyncedValueBase customValue) { CustomSyncedValueBase customValue2 = customValue; if (allCustomValues.Select((CustomSyncedValueBase v) => v.Identifier).Concat(new string[1] { "serverversion" }).Contains(customValue2.Identifier)) { throw new Exception("Cannot have multiple settings with the same name or with a reserved name (serverversion)"); } allCustomValues.Add(customValue2); allCustomValues = new HashSet<CustomSyncedValueBase>(allCustomValues.OrderByDescending((CustomSyncedValueBase v) => v.Priority)); customValue2.ValueChanged += delegate { if (!ProcessingServerUpdate) { Broadcast(ZRoutedRpc.Everybody, customValue2); } }; } private void RPC_FromServerConfigSync(ZRpc rpc, ZPackage package) { lockedConfigChanged += serverLockedSettingChanged; IsSourceOfTruth = false; if (HandleConfigSyncRPC(0L, package, clientUpdate: false)) { InitialSyncDone = true; } } private void RPC_FromOtherClientConfigSync(long sender, ZPackage package) { HandleConfigSyncRPC(sender, package, clientUpdate: true); } private bool HandleConfigSyncRPC(long sender, ZPackage package, bool clientUpdate) { //IL_0076: Unknown result type (might be due to invalid IL or missing references) //IL_007d: Expected O, but got Unknown //IL_0250: Unknown result type (might be due to invalid IL or missing references) //IL_0257: Expected O, but got Unknown //IL_01ea: Unknown result type (might be due to invalid IL or missing references) //IL_01f1: Expected O, but got Unknown try { if (isServer && IsLocked) { ZRpc? currentRpc = SnatchCurrentlyHandlingRPC.currentRpc; object obj; if (currentRpc == null) { obj = null; } else { ISocket socket = currentRpc.GetSocket(); obj = ((socket != null) ? socket.GetHostName() : null); } string text = (string)obj; if (text != null) { MethodInfo methodInfo = AccessTools.DeclaredMethod(typeof(ZNet), "ListContainsId", (Type[])null, (Type[])null); SyncedList val = (SyncedList)AccessTools.DeclaredField(typeof(ZNet), "m_adminList").GetValue(ZNet.instance); if (!(((object)methodInfo == null) ? val.Contains(text) : ((bool)methodInfo.Invoke(ZNet.instance, new object[2] { val, text })))) { return false; } } } cacheExpirations.RemoveAll(delegate(KeyValuePair<long, string> kv) { if (kv.Key < DateTimeOffset.Now.Ticks) { configValueCache.Remove(kv.Value); return true; } return false; }); byte b = package.ReadByte(); if ((b & 2u) != 0) { long num = package.ReadLong(); string text2 = sender.ToString() + num; if (!configValueCache.TryGetValue(text2, out SortedDictionary<int, byte[]> value)) { value = new SortedDictionary<int, byte[]>(); configValueCache[text2] = value; cacheExpirations.Add(new KeyValuePair<long, string>(DateTimeOffset.Now.AddSeconds(60.0).Ticks, text2)); } int key = package.ReadInt(); int num2 = package.ReadInt(); value.Add(key, package.ReadByteArray()); if (value.Count < num2) { return false; } configValueCache.Remove(text2); package = new ZPackage(value.Values.SelectMany((byte[] a) => a).ToArray()); b = package.ReadByte(); } ProcessingServerUpdate = true; if ((b & 4u) != 0) { byte[] buffer = package.ReadByteArray(); MemoryStream stream = new MemoryStream(buffer); MemoryStream memoryStream = new MemoryStream(); using (DeflateStream deflateStream = new DeflateStream(stream, CompressionMode.Decompress)) { deflateStream.CopyTo(memoryStream); } package = new ZPackage(memoryStream.ToArray()); b = package.ReadByte(); } if ((b & 1) == 0) { resetConfigsFromServer(); } ParsedConfigs parsedConfigs = ReadConfigsFromPackage(package); ConfigFile val2 = null; bool saveOnConfigSet = false; foreach (KeyValuePair<OwnConfigEntryBase, object> configValue in parsedConfigs.configValues) { if (!isServer && configValue.Key.LocalBaseValue == null) { configValue.Key.LocalBaseValue = configValue.Key.BaseConfig.BoxedValue; } if (val2 == null) { val2 = configValue.Key.BaseConfig.ConfigFile; saveOnConfigSet = val2.SaveOnConfigSet; val2.SaveOnConfigSet = false; } configValue.Key.BaseConfig.BoxedValue = configValue.Value; } if (val2 != null) { val2.SaveOnConfigSet = saveOnConfigSet; } foreach (KeyValuePair<CustomSyncedValueBase, object> customValue in parsedConfigs.customValues) { if (!isServer) { CustomSyncedValueBase key2 = customValue.Key; if (key2.LocalBaseValue == null) { key2.LocalBaseValue = customValue.Key.BoxedValue; } } customValue.Key.BoxedValue = customValue.Value; } Debug.Log((object)string.Format("Received {0} configs and {1} custom values from {2} for mod {3}", parsedConfigs.configValues.Count, parsedConfigs.customValues.Count, (isServer || clientUpdate) ? $"client {sender}" : "the server", DisplayName ?? Name)); if (!isServer) { serverLockedSettingChanged(); } return true; } finally { ProcessingServerUpdate = false; } } private ParsedConfigs ReadConfigsFromPackage(ZPackage package) { ParsedConfigs parsedConfigs = new ParsedConfigs(); Dictionary<string, OwnConfigEntryBase> dictionary = allConfigs.Where((OwnConfigEntryBase c) => c.SynchronizedConfig).ToDictionary((OwnConfigEntryBase c) => c.BaseConfig.Definition.Section + "_" + c.BaseConfig.Definition.Key, (OwnConfigEntryBase c) => c); Dictionary<string, CustomSyncedValueBase> dictionary2 = allCustomValues.ToDictionary((CustomSyncedValueBase c) => c.Identifier, (CustomSyncedValueBase c) => c); int num = package.ReadInt(); for (int i = 0; i < num; i++) { string text = package.ReadString(); string text2 = package.ReadString(); string text3 = package.ReadString(); Type type = Type.GetType(text3); if (text3 == "" || type != null) { object obj; try { obj = ((text3 == "") ? null : ReadValueWithTypeFromZPackage(package, type)); } catch (InvalidDeserializationTypeException ex) { Debug.LogWarning((object)("Got unexpected struct internal type " + ex.received + " for field " + ex.field + " struct " + text3 + " for " + text2 + " in section " + text + " for mod " + (DisplayName ?? Name) + ", expecting " + ex.expected)); continue; } OwnConfigEntryBase value2; if (text == "Internal") { CustomSyncedValueBase value; if (text2 == "serverversion") { if (obj?.ToString() != CurrentVersion) { Debug.LogWarning((object)("Received server version is not equal: server version = " + (obj?.ToString() ?? "null") + "; local version = " + (CurrentVersion ?? "unknown"))); } } else if (text2 == "lockexempt") { if (obj is bool flag) { lockExempt = flag; } } else if (dictionary2.TryGetValue(text2, out value)) { if ((text3 == "" && (!value.Type.IsValueType || Nullable.GetUnderlyingType(value.Type) != null)) || GetZPackageTypeString(value.Type) == text3) { parsedConfigs.customValues[value] = obj; continue; } Debug.LogWarning((object)("Got unexpected type " + text3 + " for internal value " + text2 + " for mod " + (DisplayName ?? Name) + ", expecting " + value.Type.AssemblyQualifiedName)); } } else if (dictionary.TryGetValue(text + "_" + text2, out value2)) { Type type2 = configType(value2.BaseConfig); if ((text3 == "" && (!type2.IsValueType || Nullable.GetUnderlyingType(type2) != null)) || GetZPackageTypeString(type2) == text3) { parsedConfigs.configValues[value2] = obj; continue; } Debug.LogWarning((object)("Got unexpected type " + text3 + " for " + text2 + " in section " + text + " for mod " + (DisplayName ?? Name) + ", expecting " + type2.AssemblyQualifiedName)); } else { Debug.LogWarning((object)("Received unknown config entry " + text2 + " in section " + text + " for mod " + (DisplayName ?? Name) + ". This may happen if client and server versions of the mod do not match.")); } continue; } Debug.LogWarning((object)("Got invalid type " + text3 + ", abort reading of received configs")); return new ParsedConfigs(); } return parsedConfigs; } private static bool isWritableConfig(OwnConfigEntryBase config) { OwnConfigEntryBase config2 = config; ConfigSync configSync = configSyncs.FirstOrDefault((ConfigSync cs) => cs.allConfigs.Contains(config2)); if (configSync == null) { return true; } return configSync.IsSourceOfTruth || !config2.SynchronizedConfig || config2.LocalBaseValue == null || (!configSync.IsLocked && (config2 != configSync.lockedConfig || lockExempt)); } private void serverLockedSettingChanged() { foreach (OwnConfigEntryBase allConfig in allConfigs) { configAttribute<ConfigurationManagerAttributes>(allConfig.BaseConfig).ReadOnly = !isWritableConfig(allConfig); } } private void resetConfigsFromServer() { ConfigFile val = null; bool saveOnConfigSet = false; foreach (OwnConfigEntryBase item in allConfigs.Where((OwnConfigEntryBase config) => config.LocalBaseValue != null)) { if (val == null) { val = item.BaseConfig.ConfigFile; saveOnConfigSet = val.SaveOnConfigSet; val.SaveOnConfigSet = false; } item.BaseConfig.BoxedValue = item.LocalBaseValue; item.LocalBaseValue = null; } if (val != null) { val.SaveOnConfigSet = saveOnConfigSet; } foreach (CustomSyncedValueBase item2 in allCustomValues.Where((CustomSyncedValueBase config) => config.LocalBaseValue != null)) { item2.BoxedValue = item2.LocalBaseValue; item2.LocalBaseValue = null; } lockedConfigChanged -= serverLockedSettingChanged; serverLockedSettingChanged(); } private IEnumerator<bool> distributeConfigToPeers(ZNetPeer peer, ZPackage package) { ZNetPeer peer2 = peer; ZRoutedRpc rpc = ZRoutedRpc.instance; if (rpc == null) { yield break; } byte[] data = package.GetArray(); if (data != null && data.LongLength > 250000) { int fragments = (int)(1 + (data.LongLength - 1) / 250000); long packageIdentifier = ++packageCounter; int fragment = 0; while (fragment < fragments) { foreach (bool item in waitForQueue()) { yield return item; } if (peer2.m_socket.IsConnected()) { ZPackage fragmentedPackage = new ZPackage(); fragmentedPackage.Write((byte)2); fragmentedPackage.Write(packageIdentifier); fragmentedPackage.Write(fragment); fragmentedPackage.Write(fragments); fragmentedPackage.Write(data.Skip(250000 * fragment).Take(250000).ToArray()); SendPackage(fragmentedPackage); if (fragment != fragments - 1) { yield return true; } int num = fragment + 1; fragment = num; continue; } break; } yield break; } foreach (bool item2 in waitForQueue()) { yield return item2; } SendPackage(package); void SendPackage(ZPackage pkg) { string text = Name + " ConfigSync"; if (isServer) { peer2.m_rpc.Invoke(text, new object[1] { pkg }); } else { rpc.InvokeRoutedRPC(peer2.m_server ? 0 : peer2.m_uid, text, new object[1] { pkg }); } } IEnumerable<bool> waitForQueue() { float timeout = Time.time + 30f; while (peer2.m_socket.GetSendQueueSize() > 20000) { if (Time.time > timeout) { Debug.Log((object)$"Disconnecting {peer2.m_uid} after 30 seconds config sending timeout"); peer2.m_rpc.Invoke("Error", new object[1] { (object)(ConnectionStatus)5 }); ZNet.instance.Disconnect(peer2); break; } yield return false; } } } private IEnumerator sendZPackage(long target, ZPackage package) { if (!Object.op_Implicit((Object)(object)ZNet.instance)) { return Enumerable.Empty<object>().GetEnumerator(); } List<ZNetPeer> list = (List<ZNetPeer>)AccessTools.DeclaredField(typeof(ZRoutedRpc), "m_peers").GetValue(ZRoutedRpc.instance); if (target != ZRoutedRpc.Everybody) { list = list.Where((ZNetPeer p) => p.m_uid == target).ToList(); } return sendZPackage(list, package); } private IEnumerator sendZPackage(List<ZNetPeer> peers, ZPackage package) { ZPackage package2 = package; if (!Object.op_Implicit((Object)(object)ZNet.instance)) { yield break; } byte[] rawData = package2.GetArray(); if (rawData != null && rawData.LongLength > 10000) { ZPackage compressedPackage = new ZPackage(); compressedPackage.Write((byte)4); MemoryStream output = new MemoryStream(); using (DeflateStream deflateStream = new DeflateStream(output, CompressionLevel.Optimal)) { deflateStream.Write(rawData, 0, rawData.Length); } compressedPackage.Write(output.ToArray()); package2 = compressedPackage; } List<IEnumerator<bool>> writers = (from peer in peers where peer.IsReady() select peer into p select distributeConfigToPeers(p, package2)).ToList(); writers.RemoveAll((IEnumerator<bool> writer) => !writer.MoveNext()); while (writers.Count > 0) { yield return null; writers.RemoveAll((IEnumerator<bool> writer) => !writer.MoveNext()); } } private void Broadcast(long target, params ConfigEntryBase[] configs) { if (!IsLocked || isServer) { ZPackage package = ConfigsToPackage(configs); ZNet instance = ZNet.instance; if (instance != null) { ((MonoBehaviour)instance).StartCoroutine(sendZPackage(target, package)); } } } private void Broadcast(long target, params CustomSyncedValueBase[] customValues) { if (!IsLocked || isServer) { ZPackage package = ConfigsToPackage(null, customValues); ZNet instance = ZNet.instance; if (instance != null) { ((MonoBehaviour)instance).StartCoroutine(sendZPackage(target, package)); } } } private static OwnConfigEntryBase? configData(ConfigEntryBase config) { return config.Description.Tags?.OfType<OwnConfigEntryBase>().SingleOrDefault(); } public static SyncedConfigEntry<T>? ConfigData<T>(ConfigEntry<T> config) { return ((ConfigEntryBase)config).Description.Tags?.OfType<SyncedConfigEntry<T>>().SingleOrDefault(); } private static T configAttribute<T>(ConfigEntryBase config) { return config.Description.Tags.OfType<T>().First(); } private static Type configType(ConfigEntryBase config) { return configType(config.SettingType); } private static Type configType(Type type) { return type.IsEnum ? Enum.GetUnderlyingType(type) : type; } private static ZPackage ConfigsToPackage(IEnumerable<ConfigEntryBase>? configs = null, IEnumerable<CustomSyncedValueBase>? customValues = null, IEnumerable<PackageEntry>? packageEntries = null, bool partial = true) { //IL_0051: Unknown result type (might be due to invalid IL or missing references) //IL_0057: Expected O, but got Unknown List<ConfigEntryBase> list = configs?.Where((ConfigEntryBase config) => configData(config).SynchronizedConfig).ToList() ?? new List<ConfigEntryBase>(); List<CustomSyncedValueBase> list2 = customValues?.ToList() ?? new List<CustomSyncedValueBase>(); ZPackage val = new ZPackage(); val.Write((byte)(partial ? 1 : 0)); val.Write(list.Count + list2.Count + (packageEntries?.Count() ?? 0)); foreach (PackageEntry item in packageEntries ?? Array.Empty<PackageEntry>()) { AddEntryToPackage(val, item); } foreach (CustomSyncedValueBase item2 in list2) { AddEntryToPackage(val, new PackageEntry { section = "Internal", key = item2.Identifier, type = item2.Type, value = item2.BoxedValue }); } foreach (ConfigEntryBase item3 in list) { AddEntryToPackage(val, new PackageEntry { section = item3.Definition.Section, key = item3.Definition.Key, type = configType(item3), value = item3.BoxedValue }); } return val; } private static void AddEntryToPackage(ZPackage package, PackageEntry entry) { package.Write(entry.section); package.Write(entry.key); package.Write((entry.value == null) ? "" : GetZPackageTypeString(entry.type)); AddValueToZPackage(package, entry.value); } private static string GetZPackageTypeString(Type type) { return type.AssemblyQualifiedName; } private static void AddValueToZPackage(ZPackage package, object? value) { Type type = value?.GetType(); if (value is Enum) { value = ((IConvertible)value).ToType(Enum.GetUnderlyingType(value.GetType()), CultureInfo.InvariantCulture); } else { if (value is ICollection collection) { package.Write(collection.Count); { foreach (object item in collection) { AddValueToZPackage(package, item); } return; } } if ((object)type != null && type.IsValueType && !type.IsPrimitive) { FieldInfo[] fields = type.GetFields(BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic); package.Write(fields.Length); FieldInfo[] array = fields; foreach (FieldInfo fieldInfo in array) { package.Write(GetZPackageTypeString(fieldInfo.FieldType)); AddValueToZPackage(package, fieldInfo.GetValue(value)); } return; } } ZRpc.Serialize(new object[1] { value }, ref package); } private static object ReadValueWithTypeFromZPackage(ZPackage package, Type type) { if ((object)type != null && type.IsValueType && !type.IsPrimitive && !type.IsEnum) { FieldInfo[] fields = type.GetFields(BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic); int num = package.ReadInt(); if (num != fields.Length) { throw new InvalidDeserializationTypeException { received = $"(field count: {num})", expected = $"(field count: {fields.Length})" }; } object uninitializedObject = FormatterServices.GetUninitializedObject(type); FieldInfo[] array = fields; foreach (FieldInfo fieldInfo in array) { string text = package.ReadString(); if (text != GetZPackageTypeString(fieldInfo.FieldType)) { throw new InvalidDeserializationTypeException { received = text, expected = GetZPackageTypeString(fieldInfo.FieldType), field = fieldInfo.Name }; } fieldInfo.SetValue(uninitializedObject, ReadValueWithTypeFromZPackage(package, fieldInfo.FieldType)); } return uninitializedObject; } if (type.IsGenericType && type.GetGenericTypeDefinition() == typeof(Dictionary<, >)) { int num2 = package.ReadInt(); IDictionary dictionary = (IDictionary)Activator.CreateInstance(type); Type type2 = typeof(KeyValuePair<, >).MakeGenericType(type.GenericTypeArguments); FieldInfo field = type2.GetField("key", BindingFlags.Instance | BindingFlags.NonPublic); FieldInfo field2 = type2.GetField("value", BindingFlags.Instance | BindingFlags.NonPublic); for (int j = 0; j < num2; j++) { object obj = ReadValueWithTypeFromZPackage(package, type2); dictionary.Add(field.GetValue(obj), field2.GetValue(obj)); } return dictionary; } if (type != typeof(List<string>) && type.IsGenericType) { Type type3 = typeof(ICollection<>).MakeGenericType(type.GenericTypeArguments[0]); if ((object)type3 != n