Decompiled source of ExtraSlotsCustomSlots v1.0.0

ExtraSlotsCustomSlots.dll

Decompiled 3 hours ago
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