Decompiled source of BlacksmithingExpanded v1.0.0

BlacksmithingExpanded.dll

Decompiled 3 days 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.Reflection.Emit;
using System.Runtime.CompilerServices;
using System.Runtime.InteropServices;
using System.Runtime.Serialization;
using System.Runtime.Versioning;
using System.Security;
using System.Security.Permissions;
using System.Text;
using System.Text.RegularExpressions;
using BepInEx;
using BepInEx.Bootstrap;
using BepInEx.Configuration;
using HarmonyLib;
using ItemDataManager;
using JetBrains.Annotations;
using Microsoft.CodeAnalysis;
using ServerSync;
using SkillManager;
using TMPro;
using UnityEngine;

[assembly: TargetFramework(".NETFramework,Version=v4.8", FrameworkDisplayName = ".NET Framework 4.8")]
[assembly: RuntimeCompatibility(WrapNonExceptionThrows = true)]
[assembly: Debuggable(DebuggableAttribute.DebuggingModes.Default | DebuggableAttribute.DebuggingModes.DisableOptimizations | DebuggableAttribute.DebuggingModes.IgnoreSymbolStoreSequencePoints | DebuggableAttribute.DebuggingModes.EnableEditAndContinue)]
[assembly: AssemblyTitle("BlacksmithingExpanded")]
[assembly: AssemblyDescription("")]
[assembly: AssemblyConfiguration("")]
[assembly: AssemblyCompany("")]
[assembly: AssemblyProduct("BlacksmithingExpanded")]
[assembly: AssemblyCopyright("Copyright ©  2025")]
[assembly: AssemblyTrademark("")]
[assembly: ComVisible(false)]
[assembly: Guid("D181CDA7-EF07-4BBC-B975-2B80FC6BBFAE")]
[assembly: CompilationRelaxations(8)]
[assembly: AssemblyFileVersion("1.0.0.0")]
[assembly: SecurityPermission(SecurityAction.RequestMinimum, SkipVerification = true)]
[assembly: AssemblyVersion("1.0.0.0")]
[module: UnverifiableCode]
namespace BlacksmithingExpanded
{
	[BepInPlugin("org.bepinex.plugins.blacksmithingexpanded", "Blacksmithing Expanded", "1.0.0")]
	public class BlacksmithingExpanded : BaseUnityPlugin
	{
		public static class CoroutineRunner
		{
			private class CoroutineHost : MonoBehaviour
			{
			}

			[CompilerGenerated]
			private sealed class <RunDelayed>d__3 : IEnumerator<object>, IDisposable, IEnumerator
			{
				private int <>1__state;

				private object <>2__current;

				public Action action;

				public float delay;

				object IEnumerator<object>.Current
				{
					[DebuggerHidden]
					get
					{
						return <>2__current;
					}
				}

				object IEnumerator.Current
				{
					[DebuggerHidden]
					get
					{
						return <>2__current;
					}
				}

				[DebuggerHidden]
				public <RunDelayed>d__3(int <>1__state)
				{
					this.<>1__state = <>1__state;
				}

				[DebuggerHidden]
				void IDisposable.Dispose()
				{
					<>1__state = -2;
				}

				private bool MoveNext()
				{
					//IL_0027: Unknown result type (might be due to invalid IL or missing references)
					//IL_0031: Expected O, but got Unknown
					switch (<>1__state)
					{
					default:
						return false;
					case 0:
						<>1__state = -1;
						<>2__current = (object)new WaitForSeconds(delay);
						<>1__state = 1;
						return true;
					case 1:
						<>1__state = -1;
						action?.Invoke();
						return false;
					}
				}

				bool IEnumerator.MoveNext()
				{
					//ILSpy generated this explicit interface implementation from .override directive in MoveNext
					return this.MoveNext();
				}

				[DebuggerHidden]
				void IEnumerator.Reset()
				{
					throw new NotSupportedException();
				}
			}

			private static CoroutineHost host;

			public static void RunLater(Action action, float delaySeconds = 0.1f)
			{
				//IL_0016: Unknown result type (might be due to invalid IL or missing references)
				//IL_001c: Expected O, but got Unknown
				if ((Object)(object)host == (Object)null)
				{
					GameObject val = new GameObject("BlacksmithingCoroutineHost");
					Object.DontDestroyOnLoad((Object)(object)val);
					host = val.AddComponent<CoroutineHost>();
				}
				((MonoBehaviour)host).StartCoroutine(RunDelayed(action, delaySeconds));
			}

			[IteratorStateMachine(typeof(<RunDelayed>d__3))]
			private static IEnumerator RunDelayed(Action action, float delay)
			{
				//yield-return decompiler failed: Unexpected instruction in Iterator.Dispose()
				return new <RunDelayed>d__3(0)
				{
					action = action,
					delay = delay
				};
			}
		}

		public class BlacksmithingData : ItemData
		{
			public int blacksmithLevel = 0;

			public override void Save()
			{
				base.Value = blacksmithLevel.ToString();
			}

			public override void Load()
			{
				if (!string.IsNullOrEmpty(base.Value))
				{
					int.TryParse(base.Value, out blacksmithLevel);
				}
			}
		}

		[HarmonyPatch(typeof(InventoryGui), "DoCrafting")]
		public static class Patch_Blacksmithing_Crafting
		{
			private static void Postfix(InventoryGui __instance)
			{
				Player localPlayer = Player.m_localPlayer;
				if ((Object)(object)localPlayer == (Object)null)
				{
					return;
				}
				Inventory inventory = ((Humanoid)localPlayer).GetInventory();
				if (inventory == null)
				{
					return;
				}
				ItemData craftedItem = inventory.GetAllItems().LastOrDefault();
				if (craftedItem == null || craftedItem.m_shared == null)
				{
					return;
				}
				int level = GetPlayerBlacksmithingLevel(localPlayer);
				if (level > 0)
				{
					CoroutineRunner.RunLater(delegate
					{
						ApplyCraftedItemMultipliers(craftedItem, level);
						AttachBlacksmithingData(craftedItem, level);
					});
					GiveBlacksmithingXP(localPlayer, cfg_XPPerCraft.Value);
					float chanceScaledWithLevel = GetChanceScaledWithLevel(cfg_ChanceExtraItemAt100.Value, level);
					if (Random.value <= chanceScaledWithLevel)
					{
						TryGiveExtraItemToCrafter(craftedItem, localPlayer);
					}
					Debug.Log((object)("[BlacksmithingExpanded] Applied stats to crafted/upgraded item: " + craftedItem.m_shared.m_name));
				}
			}
		}

		[HarmonyPatch(typeof(ItemData), "GetTooltip", new Type[]
		{
			typeof(ItemData),
			typeof(int),
			typeof(bool),
			typeof(float),
			typeof(int)
		})]
		public static class Patch_Blacksmithing_Tooltip
		{
			public static void Postfix(ItemData item, bool crafting, ref string __result)
			{
				BlacksmithingData blacksmithingData = item.Data().Get<BlacksmithingData>();
				if (blacksmithingData == null || blacksmithingData.blacksmithLevel <= 0)
				{
					return;
				}
				if (cfg_ShowBlacksmithLevelInTooltip.Value)
				{
					__result += $"\n<color=orange>Forged at Blacksmithing {blacksmithingData.blacksmithLevel}</color>";
				}
				if (cfg_ShowInfusionInTooltip.Value && item.m_customData.TryGetValue("ElementalInfusion", out var value))
				{
					__result = __result + "\n<color=#87CEEB>Elemental Infusion: " + value + "</color>";
				}
				if (!cfg_ShowDurabilityBonusInTooltip.Value)
				{
					return;
				}
				float maxDurability = item.m_shared.m_maxDurability;
				if (maxDurability > 0f && blacksmithingData.blacksmithLevel >= cfg_DurabilityTierInterval.Value)
				{
					int num = blacksmithingData.blacksmithLevel / cfg_DurabilityTierInterval.Value;
					float num2 = (float)num * cfg_DurabilityBonusPerTier.Value + (float)item.m_quality * cfg_DurabilityBonusPerUpgrade.Value;
					if (num2 > 0f)
					{
						__result += $"\n<color=#90EE90>Durability Bonus: +{num2:F0}</color>";
					}
				}
			}
		}

		[HarmonyPatch(typeof(InventoryGui), "UpdateRecipe")]
		public static class Patch_Blacksmithing_RecipePreview
		{
			private static void Postfix(InventoryGui __instance)
			{
			}
		}

		[HarmonyPatch]
		public static class Patch_Smelter_OnAddOre_Postfix
		{
			private static MethodInfo TargetMethod()
			{
				MethodInfo methodInfo = AccessTools.Method(typeof(Smelter), "OnAddOre", new Type[3]
				{
					typeof(Switch),
					typeof(Humanoid),
					typeof(ItemData)
				}, (Type[])null);
				return methodInfo ?? AccessTools.Method(typeof(Smelter), "OnAddOre", (Type[])null, (Type[])null);
			}

			private static void Postfix(Smelter __instance, Switch sw, Humanoid user, ItemData item, bool __result)
			{
				//IL_00aa: Unknown result type (might be due to invalid IL or missing references)
				//IL_00af: Unknown result type (might be due to invalid IL or missing references)
				//IL_00be: Unknown result type (might be due to invalid IL or missing references)
				try
				{
					if (!__result || (Object)(object)user == (Object)null)
					{
						return;
					}
					Player val = (Player)(object)((user is Player) ? user : null);
					if ((Object)(object)val == (Object)null)
					{
						return;
					}
					GiveBlacksmithingXP(val, cfg_XPPerSmelt.Value);
					int playerBlacksmithingLevel = GetPlayerBlacksmithingLevel(val);
					if (playerBlacksmithingLevel > 0)
					{
						float chanceScaledWithLevel = GetChanceScaledWithLevel(cfg_SmelterSaveOreChanceAt100.Value, playerBlacksmithingLevel);
						if (Random.value <= chanceScaledWithLevel)
						{
							((Character)val).Message((MessageType)1, "Smelter efficiency saved some ore!", 0, (Sprite)null);
						}
					}
					ZNetView nview = __instance.m_nview;
					ZDO val2 = ((nview != null) ? nview.GetZDO() : null);
					if (val2 != null)
					{
						ZDOID uid = val2.m_uid;
						int playerBlacksmithingTier = GetPlayerBlacksmithingTier(val);
						smelterInfusions[uid] = (playerBlacksmithingTier, Time.time);
					}
				}
				catch (Exception arg)
				{
					Debug.LogError((object)$"[BlacksmithingExpanded] Smelter OnAddOre postfix error: {arg}");
				}
			}
		}

		[HarmonyPatch(typeof(Smelter), "UpdateSmelter")]
		public static class Patch_Smelter_UpdateSmelter
		{
			private static void Prefix(Smelter __instance)
			{
				//IL_0024: Unknown result type (might be due to invalid IL or missing references)
				//IL_0029: Unknown result type (might be due to invalid IL or missing references)
				//IL_002f: Unknown result type (might be due to invalid IL or missing references)
				//IL_0080: Unknown result type (might be due to invalid IL or missing references)
				ZNetView nview = __instance.m_nview;
				ZDO val = ((nview != null) ? nview.GetZDO() : null);
				if (val == null)
				{
					return;
				}
				ZDOID uid = val.m_uid;
				if (smelterInfusions.TryGetValue(uid, out (int, float) value))
				{
					float value2 = cfg_InfusionExpireTime.Value;
					if (__instance.GetQueueSize() == 0 || __instance.GetFuel() <= 0f || Time.time - value.Item2 > value2)
					{
						smelterInfusions.Remove(uid);
						return;
					}
					float num = (float)value.Item1 * cfg_SmeltingSpeedBonusPerTier.Value;
					__instance.m_secPerProduct /= 1f + num;
				}
			}
		}

		[HarmonyPatch(typeof(Smelter), "OnAddOre")]
		public static class Patch_Kiln_OnAddWood
		{
			private static void Postfix(Smelter __instance, Humanoid user, ItemData item, bool __result)
			{
				//IL_0080: Unknown result type (might be due to invalid IL or missing references)
				//IL_0085: Unknown result type (might be due to invalid IL or missing references)
				//IL_0094: Unknown result type (might be due to invalid IL or missing references)
				try
				{
					if (!__result || (Object)(object)user == (Object)null || (Object)(object)__instance == (Object)null || __instance.m_name != "charcoal_kiln")
					{
						return;
					}
					Player val = (Player)(object)((user is Player) ? user : null);
					if (!((Object)(object)val == (Object)null))
					{
						GiveBlacksmithingXP(val, cfg_XPPerSmelt.Value);
						ZNetView nview = __instance.m_nview;
						ZDO val2 = ((nview != null) ? nview.GetZDO() : null);
						if (val2 != null)
						{
							ZDOID uid = val2.m_uid;
							int playerBlacksmithingTier = GetPlayerBlacksmithingTier(val);
							kilnInfusions[uid] = (playerBlacksmithingTier, Time.time);
						}
					}
				}
				catch (Exception arg)
				{
					Debug.LogError((object)$"[BlacksmithingExpanded] Kiln OnAddWood postfix error: {arg}");
				}
			}
		}

		[HarmonyPatch(typeof(Smelter), "UpdateSmelter")]
		public static class Patch_Kiln_UpdateKiln
		{
			private static void Prefix(Smelter __instance)
			{
				//IL_004c: Unknown result type (might be due to invalid IL or missing references)
				//IL_0051: Unknown result type (might be due to invalid IL or missing references)
				//IL_0057: Unknown result type (might be due to invalid IL or missing references)
				//IL_009b: Unknown result type (might be due to invalid IL or missing references)
				try
				{
					if ((Object)(object)__instance == (Object)null || __instance.m_name != "charcoal_kiln")
					{
						return;
					}
					ZNetView nview = __instance.m_nview;
					ZDO val = ((nview != null) ? nview.GetZDO() : null);
					if (val == null)
					{
						return;
					}
					ZDOID uid = val.m_uid;
					if (kilnInfusions.TryGetValue(uid, out (int, float) value))
					{
						float value2 = cfg_InfusionExpireTime.Value;
						if (__instance.GetQueueSize() == 0 || Time.time - value.Item2 > value2)
						{
							kilnInfusions.Remove(uid);
							return;
						}
						float num = (float)value.Item1 * cfg_KilnSpeedBonusPerTier.Value;
						__instance.m_secPerProduct /= 1f + num;
					}
				}
				catch (Exception arg)
				{
					Debug.LogError((object)$"[BlacksmithingExpanded] Kiln UpdateSmelter prefix error: {arg}");
				}
			}
		}

		[HarmonyPatch(typeof(InventoryGui), "OnRepairPressed")]
		public static class Patch_Blacksmithing_OnRepairPressed
		{
			private static bool Prefix(InventoryGui __instance)
			{
				//IL_0157: Unknown result type (might be due to invalid IL or missing references)
				//IL_015c: Unknown result type (might be due to invalid IL or missing references)
				Player localPlayer = Player.m_localPlayer;
				if ((Object)(object)localPlayer == (Object)null || !cfg_EnableInventoryRepair.Value)
				{
					return true;
				}
				Inventory inventory = ((Humanoid)localPlayer).GetInventory();
				if (inventory == null)
				{
					return true;
				}
				int playerBlacksmithingLevel = GetPlayerBlacksmithingLevel(localPlayer);
				if (playerBlacksmithingLevel < cfg_InventoryRepairUnlockLevel.Value)
				{
					return true;
				}
				foreach (ItemData allItem in inventory.GetAllItems())
				{
					if (allItem != null && allItem.m_shared?.m_maxDurability > 0f && allItem.m_durability < allItem.GetMaxDurability())
					{
						int num = playerBlacksmithingLevel / cfg_InfusionTierInterval.Value;
						float num2 = (float)num * cfg_DurabilityBonusPerTier.Value;
						allItem.m_durability = Mathf.Min(allItem.m_durability + num2, allItem.GetMaxDurability());
						GiveBlacksmithingXP(localPlayer, cfg_XPPerRepair.Value);
						GameObject prefab = ZNetScene.instance.GetPrefab("vfx_Smelter_add");
						if ((Object)(object)prefab != (Object)null)
						{
							Object.Instantiate<GameObject>(prefab, ((Component)localPlayer).transform.position, Quaternion.identity);
						}
						((Character)localPlayer).Message((MessageType)1, $"Repaired with masterwork precision (+{playerBlacksmithingLevel} skill)", 0, (Sprite)null);
						return false;
					}
				}
				return true;
			}
		}

		internal const string ModName = "Blacksmithing Expanded";

		internal const string ModVersion = "1.0.0";

		internal const string ModGUID = "org.bepinex.plugins.blacksmithingexpanded";

		private Harmony harmony;

		private static readonly ConfigSync configSync = new ConfigSync("org.bepinex.plugins.blacksmithingexpanded")
		{
			DisplayName = "Blacksmithing Expanded",
			CurrentVersion = "1.0.0",
			MinimumRequiredVersion = "1.0.0",
			ModRequired = true
		};

		internal static Skill blacksmithSkill;

		private static readonly Dictionary<string, float> baseArmorLookup = new Dictionary<string, float>();

		private static readonly Dictionary<string, DamageTypes> baseDamageLookup = new Dictionary<string, DamageTypes>();

		private static readonly Dictionary<string, float> baseDurabilityLookup = new Dictionary<string, float>();

		internal static ConfigEntry<float> cfg_SkillGainFactor;

		internal static ConfigEntry<float> cfg_SkillEffectFactor;

		internal static ConfigEntry<int> cfg_InfusionTierInterval;

		internal static ConfigEntry<int> cfg_MaxTierUnlockLevel;

		internal static ConfigEntry<float> cfg_ChanceExtraItemAt100;

		internal static ConfigEntry<float> cfg_SmelterSaveOreChanceAt100;

		internal static ConfigEntry<bool> cfg_EnableInventoryRepair;

		internal static ConfigEntry<int> cfg_InventoryRepairUnlockLevel;

		internal static ConfigEntry<float> cfg_SmeltingSpeedBonusPerTier;

		internal static ConfigEntry<float> cfg_KilnSpeedBonusPerTier;

		internal static ConfigEntry<float> cfg_InfusionExpireTime;

		internal static ConfigEntry<bool> cfg_ShowBlacksmithLevelInTooltip;

		internal static ConfigEntry<bool> cfg_ShowInfusionInTooltip;

		internal static ConfigEntry<int> cfg_DurabilityTierInterval;

		internal static ConfigEntry<float> cfg_DurabilityBonusPerTier;

		internal static ConfigEntry<float> cfg_DurabilityBonusPerUpgrade;

		internal static ConfigEntry<bool> cfg_RespectOriginalDurability;

		internal static ConfigEntry<float> cfg_MaxDurabilityCap;

		internal static ConfigEntry<bool> cfg_ShowDurabilityBonusInTooltip;

		internal static ConfigEntry<int> cfg_GearMilestoneInterval;

		internal static ConfigEntry<float> cfg_ArmorBonusPerTier;

		internal static ConfigEntry<float> cfg_ArmorBonusPerUpgrade;

		internal static ConfigEntry<float> cfg_ArmorCap;

		internal static ConfigEntry<bool> cfg_RespectOriginalStats;

		internal static ConfigEntry<int> cfg_StatTierInterval;

		internal static ConfigEntry<int> cfg_DamageBonusPerTier;

		internal static ConfigEntry<float> cfg_StatBonusPerUpgrade;

		internal static ConfigEntry<int> cfg_MaxStatTypesPerTier;

		internal static ConfigEntry<float> cfg_StatBonusMultiplierPerTier;

		internal static ConfigEntry<float> cfg_StatBonusCapPerType;

		internal static ConfigEntry<bool> cfg_AlwaysAddElementalAtMax;

		internal static ConfigEntry<int> cfg_ElementalUnlockLevel;

		internal static ConfigEntry<float> cfg_FireBonusAtMax;

		internal static ConfigEntry<float> cfg_FrostBonusAtMax;

		internal static ConfigEntry<float> cfg_LightningBonusAtMax;

		internal static ConfigEntry<float> cfg_PoisonBonusAtMax;

		internal static ConfigEntry<float> cfg_SpiritBonusAtMax;

		internal static ConfigEntry<float> cfg_ElementalBonusPerTier;

		internal static ConfigEntry<float> cfg_TimedBlockBonusPerTier;

		internal static ConfigEntry<float> cfg_TimedBlockBonusPerUpgrade;

		internal static ConfigEntry<float> cfg_BlockPowerBonusPerTier;

		internal static ConfigEntry<float> cfg_BlockPowerBonusPerUpgrade;

		internal static ConfigEntry<bool> cfg_ApplyUpgradeBonusAtTierZero;

		internal static ConfigEntry<float> cfg_GearBonusPerMilestone;

		internal static ConfigEntry<float> cfg_GearUpgradeBonusPerMilestone;

		internal static ConfigEntry<float> cfg_XPPerCraft;

		internal static ConfigEntry<float> cfg_XPPerSmelt;

		internal static ConfigEntry<float> cfg_XPPerRepair;

		internal static Dictionary<ZDOID, (int tier, float timestamp)> smelterInfusions = new Dictionary<ZDOID, (int, float)>();

		internal static Dictionary<ZDOID, (int tier, float timestamp)> kilnInfusions = new Dictionary<ZDOID, (int, float)>();

		private static Sprite s_skillIcon;

		private ConfigEntry<T> AddConfig<T>(string group, string name, T value, ConfigDescription description, bool synchronizedSetting = true)
		{
			ConfigEntry<T> val = ((BaseUnityPlugin)this).Config.Bind<T>(group, name, value, description);
			SyncedConfigEntry<T> syncedConfigEntry = configSync.AddConfigEntry<T>(val);
			syncedConfigEntry.SynchronizedConfig = synchronizedSetting;
			return val;
		}

		private ConfigEntry<T> AddConfig<T>(string group, string name, T value, 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 AddConfig(group, name, value, new ConfigDescription(description, (AcceptableValueBase)null, Array.Empty<object>()), synchronizedSetting);
		}

		private void Awake()
		{
			//IL_0007: Unknown result type (might be due to invalid IL or missing references)
			//IL_0011: Expected O, but got Unknown
			harmony = new Harmony("org.bepinex.plugins.blacksmithingexpanded");
			try
			{
				s_skillIcon = LoadEmbeddedSprite("smithing.png", 64, 64);
			}
			catch (Exception arg)
			{
				Debug.LogError((object)$"[BlacksmithingExpanded] Failed to load embedded sprite: {arg}");
				s_skillIcon = null;
			}
			try
			{
				if ((Object)(object)s_skillIcon != (Object)null)
				{
					blacksmithSkill = new Skill("Blacksmithing", s_skillIcon)
					{
						Configurable = true
					};
				}
				else
				{
					blacksmithSkill = new Skill("Blacksmithing", "smithing.png")
					{
						Configurable = true
					};
				}
				blacksmithSkill.Name.English("Blacksmithing");
				blacksmithSkill.Description.English("Craft better, last longer. Improves durability, damage, and armor of crafted items. Grants smelting and repair bonuses.");
			}
			catch (Exception arg2)
			{
				Debug.LogError((object)$"[BlacksmithingExpanded] Failed to construct SkillManager skill: {arg2}");
			}
			cfg_SkillGainFactor = AddConfig("General", "Skill gain factor", 1f, "Rate at which you gain Blacksmithing XP.");
			cfg_SkillEffectFactor = AddConfig("General", "Skill effect factor", 1f, "Multiplier applied to all skill effects.");
			cfg_XPPerCraft = AddConfig("General", "XP per craft", 0.5f, "Base XP granted when crafting an item.");
			cfg_XPPerSmelt = AddConfig("General", "XP per smelt", 0.75f, "Base XP granted when adding ore to smelter.");
			cfg_XPPerRepair = AddConfig("General", "XP per repair", 0.05f, "Base XP granted when repairing an item.");
			cfg_InfusionTierInterval = AddConfig("General", "Workstation infusion milestone interval", 10, "Levels per infusion tier (smelter, kiln, repair).");
			cfg_KilnSpeedBonusPerTier = AddConfig("General", "Kiln infusion speed bonus per milestone", 0.05f, "Speed bonus per infusion tier.");
			cfg_SmeltingSpeedBonusPerTier = AddConfig("General", "Smelting infusion speed bonus per milestone", 0.05f, "Speed bonus per infusion tier.");
			cfg_InfusionExpireTime = AddConfig("General", "Infusion expire time (seconds)", 60f, "How long an infusion lasts.");
			cfg_ChanceExtraItemAt100 = AddConfig("General", "Chance to craft extra item at 100", 0.05f, "Chance to produce an extra copy when crafting at level 100.");
			cfg_SmelterSaveOreChanceAt100 = AddConfig("General", "Chance smelter saves ore at 100", 0.2f, "Chance ore is saved at level 100.");
			cfg_EnableInventoryRepair = AddConfig("General", "Inventory repair enabled", value: true, "Enable repairing from inventory.");
			cfg_InventoryRepairUnlockLevel = AddConfig("General", "Inventory repair unlock level", 70, "Level required for inventory repairs.");
			cfg_ShowBlacksmithLevelInTooltip = AddConfig("Tooltip", "Show blacksmith level in tooltip", value: true, "If true, shows the blacksmithing level used to forge the item in its tooltip.");
			cfg_ShowInfusionInTooltip = AddConfig("Tooltip", "Show elemental infusion in tooltip", value: false, "If true, shows the elemental infusion type applied to the item in its tooltip.");
			cfg_ShowDurabilityBonusInTooltip = AddConfig("Tooltip", "Show durability bonus in tooltip", value: false, "If true, shows bonus durability next to blacksmith level in item tooltip.");
			cfg_DurabilityTierInterval = AddConfig("Durability", "Durability milestone interval", 10, "Levels required per durability milestone.");
			cfg_DurabilityBonusPerTier = AddConfig("Durability", "Durability bonus per milestone", 50f, "Flat durability bonus per milestone.");
			cfg_DurabilityBonusPerUpgrade = AddConfig("Durability", "Durability bonus per milestone per upgrade", 50f, "Durability bonus applied per item upgrade.");
			cfg_RespectOriginalDurability = AddConfig("Durability", "Respect original durability", value: true, "Only boost if base durability > 0.");
			cfg_MaxDurabilityCap = AddConfig("Durability", "Max durability cap", 2000f, "Maximum durability after all bonuses.");
			cfg_GearMilestoneInterval = AddConfig("Armor", "Armor bonus milestone interval", 20, "Levels required per armor milestone.");
			cfg_ArmorBonusPerTier = AddConfig("Armor", "Armor bonus per milestone", 5f, "Flat armor per milestone.");
			cfg_ArmorBonusPerUpgrade = AddConfig("Armor", "Armor bonus per milestone per upgrade", 2f, "Extra armor per upgrade per milestone.");
			cfg_ArmorCap = AddConfig("Armor", "Armor cap", 300f, "Maximum armor value allowed after all bonuses. Set to 0 to disable.");
			cfg_RespectOriginalStats = AddConfig("Weapons", "Respect original weapon stats", value: true, "Only scale stats that exist on base item.");
			cfg_StatTierInterval = AddConfig("Weapons", "Weapon bonus milestone interval", 20, "Levels required per weapon milestone.");
			cfg_DamageBonusPerTier = AddConfig("Weapons", "Weapon bonus increase per milestone", 10, "Flat base damage bonus per milestone.");
			cfg_StatBonusPerUpgrade = AddConfig("Weapons", "Weapon bonus increase per milestone per upgrade", 8f, "Flat damage bonus per upgrade per milestone.");
			cfg_StatBonusCapPerType = AddConfig("Weapons", "Stat bonus cap per damage type", 100f, "Maximum allowed bonus per damage type.");
			cfg_MaxStatTypesPerTier = AddConfig("Weapons", "Max stat types per tier", 3, "Number of damage types boosted per milestone.");
			cfg_AlwaysAddElementalAtMax = AddConfig("Elemental", "Add elemental bonus at milestone", value: true, "Adds elemental bonus when milestone is reached.");
			cfg_ElementalUnlockLevel = AddConfig("Elemental", "Elemental unlock milestone", 100, "Milestone at which elemental bonuses are enabled.");
			cfg_FireBonusAtMax = AddConfig("Elemental", "Fire bonus at level 100", 20f, "Fire damage added at milestone 100.");
			cfg_FrostBonusAtMax = AddConfig("Elemental", "Frost bonus at level 100", 20f, "Frost damage added at milestone 100.");
			cfg_LightningBonusAtMax = AddConfig("Elemental", "Lightning bonus at level 100", 20f, "Lightning damage added at milestone 100.");
			cfg_PoisonBonusAtMax = AddConfig("Elemental", "Poison bonus at level 100", 20f, "Poison damage added at milestone 100.");
			cfg_SpiritBonusAtMax = AddConfig("Elemental", "Spirit bonus at level 100", 20f, "Spirit damage added at milestone 100.");
			cfg_ElementalBonusPerTier = AddConfig("Elemental", "Elemental bonus per weapon tier", 5f, "Elemental bonus applied per weapon milestone.");
			cfg_TimedBlockBonusPerTier = AddConfig("Shields", "Timed block bonus per milestone", 0.05f, "Parry bonus per shield milestone.");
			cfg_TimedBlockBonusPerUpgrade = AddConfig("Shields", "Timed block bonus per milestone per upgrade", 0.05f, "Parry bonus per upgrade per shield milestone.");
			cfg_BlockPowerBonusPerTier = AddConfig("Shields", "Block power bonus per milestone", 2f, "Block power bonus per shield milestone.");
			cfg_BlockPowerBonusPerUpgrade = AddConfig("Shields", "Block power bonus per milestone per upgrade", 1f, "Block power bonus per upgrade per shield milestone.");
			cfg_StatBonusMultiplierPerTier = AddConfig("Advanced", "Stat bonus multiplier per tier", 1f, "Multiplier for advanced scaling.");
			cfg_ApplyUpgradeBonusAtTierZero = AddConfig("Advanced", "Apply upgrade bonus at tier 0", value: false, "If false, upgrade bonuses only start at first milestone.");
			cfg_GearBonusPerMilestone = AddConfig("Advanced", "Gear bonus per milestone", 5f, "Shared bonus applied to gear per milestone.");
			cfg_GearUpgradeBonusPerMilestone = AddConfig("Advanced", "Gear upgrade bonus per milestone", 2f, "Shared upgrade bonus applied to gear per milestone.");
			cfg_MaxTierUnlockLevel = AddConfig("Advanced", "Max tier unlock level", 100, "Maximum level used for stat scaling.");
			harmony.PatchAll();
			((BaseUnityPlugin)this).Logger.LogInfo((object)"Blacksmithing Expanded v1.0.0 loaded.");
		}

		internal static int GetPlayerBlacksmithingLevel(Player player)
		{
			//IL_0030: Unknown result type (might be due to invalid IL or missing references)
			//IL_0035: Unknown result type (might be due to invalid IL or missing references)
			//IL_0037: Unknown result type (might be due to invalid IL or missing references)
			try
			{
				if ((Object)(object)player == (Object)null)
				{
					return 0;
				}
				Skills component = ((Component)player).GetComponent<Skills>();
				if ((Object)(object)component == (Object)null)
				{
					return 0;
				}
				SkillType val = Skill.fromName("Blacksmithing");
				float skillLevel = component.GetSkillLevel(val);
				return Mathf.FloorToInt(skillLevel);
			}
			catch (Exception arg)
			{
				Debug.LogError((object)$"[BlacksmithingExpanded] GetPlayerBlacksmithingLevel error: {arg}");
				return 0;
			}
		}

		internal static void GiveBlacksmithingXP(Player player, float amount)
		{
			if ((Object)(object)player == (Object)null || amount <= 0f)
			{
				return;
			}
			try
			{
				float num = amount * cfg_SkillGainFactor.Value;
				((Character)(object)player).RaiseSkill("Blacksmithing", num);
				Debug.Log((object)$"[BlacksmithingExpanded] Gave {num:F2} XP to {player.GetPlayerName()}");
			}
			catch (Exception arg)
			{
				Debug.LogError((object)$"[BlacksmithingExpanded] Failed to raise skill XP for {((player != null) ? player.GetPlayerName() : null)}: {arg}");
			}
		}

		internal static float GetChanceScaledWithLevel(float maxChanceAt100, int level)
		{
			return Mathf.Clamp01(maxChanceAt100 * ((float)level / 100f));
		}

		internal static void ApplyCraftedItemMultipliers(ItemData item, int level)
		{
			//IL_0067: Unknown result type (might be due to invalid IL or missing references)
			//IL_006c: Unknown result type (might be due to invalid IL or missing references)
			//IL_0070: Unknown result type (might be due to invalid IL or missing references)
			//IL_0075: Unknown result type (might be due to invalid IL or missing references)
			//IL_01dd: Unknown result type (might be due to invalid IL or missing references)
			//IL_01e2: Unknown result type (might be due to invalid IL or missing references)
			//IL_0212: Unknown result type (might be due to invalid IL or missing references)
			//IL_0234: Unknown result type (might be due to invalid IL or missing references)
			//IL_0256: Unknown result type (might be due to invalid IL or missing references)
			//IL_0278: Unknown result type (might be due to invalid IL or missing references)
			//IL_029a: Unknown result type (might be due to invalid IL or missing references)
			//IL_02bc: Unknown result type (might be due to invalid IL or missing references)
			//IL_02de: Unknown result type (might be due to invalid IL or missing references)
			//IL_0300: Unknown result type (might be due to invalid IL or missing references)
			//IL_038a: Unknown result type (might be due to invalid IL or missing references)
			//IL_0390: Invalid comparison between Unknown and I4
			//IL_0a53: Unknown result type (might be due to invalid IL or missing references)
			//IL_0a59: Invalid comparison between Unknown and I4
			if (item == null || item.m_shared == null || level <= 0)
			{
				return;
			}
			CacheBaseStats(item);
			string name = item.m_shared.m_name;
			float num = baseArmorLookup[name];
			DamageTypes val = baseDamageLookup[name];
			DamageTypes val2 = ((DamageTypes)(ref val)).Clone();
			float num2 = baseDurabilityLookup[name];
			int num3 = Mathf.Min(level, cfg_MaxTierUnlockLevel.Value);
			int num4 = num3 / cfg_StatTierInterval.Value;
			int num5 = num3 / cfg_DurabilityTierInterval.Value;
			int num6 = num3 / cfg_GearMilestoneInterval.Value;
			float damageBonus = (float)(num4 * cfg_DamageBonusPerTier.Value) * cfg_StatBonusMultiplierPerTier.Value + (float)num6 * cfg_GearBonusPerMilestone.Value;
			float upgradeDamageBonus = 0f;
			if (cfg_ApplyUpgradeBonusAtTierZero.Value || num4 > 0)
			{
				upgradeDamageBonus = ((float)item.m_quality * cfg_StatBonusPerUpgrade.Value + (float)(item.m_quality * num6) * cfg_GearUpgradeBonusPerMilestone.Value) * cfg_StatBonusMultiplierPerTier.Value;
			}
			float num7 = (float)num4 * cfg_ArmorBonusPerTier.Value + (float)num6 * cfg_GearBonusPerMilestone.Value;
			float num8 = 0f;
			if (cfg_ApplyUpgradeBonusAtTierZero.Value || num4 > 0)
			{
				num8 = (float)item.m_quality * cfg_ArmorBonusPerUpgrade.Value + (float)(item.m_quality * num6) * cfg_GearUpgradeBonusPerMilestone.Value;
			}
			item.m_shared.m_damages = ((DamageTypes)(ref val2)).Clone();
			if (cfg_RespectOriginalStats.Value)
			{
				ApplyStat(ref item.m_shared.m_damages.m_blunt, val2.m_blunt);
				ApplyStat(ref item.m_shared.m_damages.m_slash, val2.m_slash);
				ApplyStat(ref item.m_shared.m_damages.m_pierce, val2.m_pierce);
				ApplyStat(ref item.m_shared.m_damages.m_fire, val2.m_fire);
				ApplyStat(ref item.m_shared.m_damages.m_frost, val2.m_frost);
				ApplyStat(ref item.m_shared.m_damages.m_lightning, val2.m_lightning);
				ApplyStat(ref item.m_shared.m_damages.m_poison, val2.m_poison);
				ApplyStat(ref item.m_shared.m_damages.m_spirit, val2.m_spirit);
				if (num > 0f)
				{
					item.m_shared.m_armor = Mathf.RoundToInt(num + num7 + num8);
					if (cfg_ArmorCap.Value > 0f)
					{
						item.m_shared.m_armor = Mathf.Min(item.m_shared.m_armor, cfg_ArmorCap.Value);
					}
				}
				if ((int)item.m_shared.m_itemType == 5)
				{
					if (item.m_shared.m_blockPower > 0f)
					{
						SharedData shared = item.m_shared;
						shared.m_blockPower += (float)num4 * cfg_BlockPowerBonusPerTier.Value + (float)item.m_quality * cfg_BlockPowerBonusPerUpgrade.Value;
					}
					if (item.m_shared.m_deflectionForce > 0f)
					{
						SharedData shared2 = item.m_shared;
						shared2.m_deflectionForce += num7 + num8;
					}
					if (item.m_shared.m_timedBlockBonus > 0f)
					{
						SharedData shared3 = item.m_shared;
						shared3.m_timedBlockBonus += (float)num4 * cfg_TimedBlockBonusPerTier.Value + (float)item.m_quality * cfg_TimedBlockBonusPerUpgrade.Value;
					}
				}
				if (num3 >= cfg_ElementalUnlockLevel.Value && cfg_AlwaysAddElementalAtMax.Value)
				{
					float infusionBonus = (float)num4 * cfg_ElementalBonusPerTier.Value + damageBonus + upgradeDamageBonus;
					if (num3 >= 100)
					{
						item.m_shared.m_damages.m_fire += cfg_FireBonusAtMax.Value;
						item.m_shared.m_damages.m_frost += cfg_FrostBonusAtMax.Value;
						item.m_shared.m_damages.m_lightning += cfg_LightningBonusAtMax.Value;
						item.m_shared.m_damages.m_poison += cfg_PoisonBonusAtMax.Value;
						item.m_shared.m_damages.m_spirit += cfg_SpiritBonusAtMax.Value;
					}
					if (!item.m_customData.ContainsKey("ElementalInfusion"))
					{
						List<(string, Action)> list = new List<(string, Action)>
						{
							("Fire", delegate
							{
								item.m_shared.m_damages.m_fire += infusionBonus;
							}),
							("Frost", delegate
							{
								item.m_shared.m_damages.m_frost += infusionBonus;
							}),
							("Lightning", delegate
							{
								item.m_shared.m_damages.m_lightning += infusionBonus;
							}),
							("Poison", delegate
							{
								item.m_shared.m_damages.m_poison += infusionBonus;
							}),
							("Spirit", delegate
							{
								item.m_shared.m_damages.m_spirit += infusionBonus;
							})
						};
						Random random = new Random();
						(string, Action) tuple = list[random.Next(list.Count)];
						tuple.Item2();
						item.m_customData["ElementalInfusion"] = tuple.Item1;
						Debug.Log((object)$"[BlacksmithingExpanded] Infused {item.m_shared.m_name} with {tuple.Item1} (bonus={infusionBonus})");
					}
					else
					{
						switch (item.m_customData["ElementalInfusion"])
						{
						case "Fire":
							item.m_shared.m_damages.m_fire += infusionBonus;
							break;
						case "Frost":
							item.m_shared.m_damages.m_frost += infusionBonus;
							break;
						case "Lightning":
							item.m_shared.m_damages.m_lightning += infusionBonus;
							break;
						case "Poison":
							item.m_shared.m_damages.m_poison += infusionBonus;
							break;
						case "Spirit":
							item.m_shared.m_damages.m_spirit += infusionBonus;
							break;
						}
					}
				}
			}
			else
			{
				List<(string, Action)> list2 = new List<(string, Action)>
				{
					("Blunt", delegate
					{
						item.m_shared.m_damages.m_blunt += damageBonus + upgradeDamageBonus;
					}),
					("Slash", delegate
					{
						item.m_shared.m_damages.m_slash += damageBonus + upgradeDamageBonus;
					}),
					("Pierce", delegate
					{
						item.m_shared.m_damages.m_pierce += damageBonus + upgradeDamageBonus;
					}),
					("Fire", delegate
					{
						item.m_shared.m_damages.m_fire += damageBonus + upgradeDamageBonus;
					}),
					("Frost", delegate
					{
						item.m_shared.m_damages.m_frost += damageBonus + upgradeDamageBonus;
					}),
					("Lightning", delegate
					{
						item.m_shared.m_damages.m_lightning += damageBonus + upgradeDamageBonus;
					}),
					("Poison", delegate
					{
						item.m_shared.m_damages.m_poison += damageBonus + upgradeDamageBonus;
					}),
					("Spirit", delegate
					{
						item.m_shared.m_damages.m_spirit += damageBonus + upgradeDamageBonus;
					})
				};
				Random rng = new Random();
				List<(string, Action)> list3 = list2.OrderBy<(string, Action), int>(((string name, Action) _) => rng.Next()).Take(Math.Min(cfg_MaxStatTypesPerTier.Value, list2.Count)).ToList();
				foreach (var item2 in list3)
				{
					item2.Item2();
				}
				item.m_shared.m_armor = Mathf.RoundToInt(num + num7 + num8);
				if (cfg_ArmorCap.Value > 0f)
				{
					item.m_shared.m_armor = Mathf.Min(item.m_shared.m_armor, cfg_ArmorCap.Value);
				}
				if ((int)item.m_shared.m_itemType == 5)
				{
					SharedData shared4 = item.m_shared;
					shared4.m_blockPower += (float)num4 * cfg_BlockPowerBonusPerTier.Value + (float)item.m_quality * cfg_BlockPowerBonusPerUpgrade.Value;
					SharedData shared5 = item.m_shared;
					shared5.m_deflectionForce += num7 + num8;
					SharedData shared6 = item.m_shared;
					shared6.m_timedBlockBonus += (float)num4 * cfg_TimedBlockBonusPerTier.Value + (float)item.m_quality * cfg_TimedBlockBonusPerUpgrade.Value;
				}
				Debug.Log((object)("[BlacksmithingExpanded] Randomized boost: " + string.Join(", ", list3.Select<(string, Action), string>(((string name, Action) s) => s.name))));
			}
			if (!cfg_RespectOriginalDurability.Value || num2 > 0f)
			{
				float num9 = (float)num5 * cfg_DurabilityBonusPerTier.Value + (float)item.m_quality * cfg_DurabilityBonusPerUpgrade.Value;
				float num10 = num2 + num9;
				if (cfg_MaxDurabilityCap.Value > 0f)
				{
					num10 = Mathf.Min(num10, cfg_MaxDurabilityCap.Value);
				}
				item.m_shared.m_maxDurability = num10;
				item.m_durability = item.GetMaxDurability();
				Debug.Log((object)$"[BlacksmithingExpanded] Durability: base={num2}, tierBonus={(float)num5 * cfg_DurabilityBonusPerTier.Value}, upgradeBonus={(float)item.m_quality * cfg_DurabilityBonusPerUpgrade.Value}, final={num10}");
			}
			else
			{
				item.m_shared.m_maxDurability = num2;
				item.m_durability = item.GetMaxDurability();
			}
			Debug.Log((object)$"[BlacksmithingExpanded] {item.m_shared.m_name}: level={level} (capped {num3}), statTier={num4}, durabilityTier={num5}, milestones={num6}, damageBonus={damageBonus}, upgradeDamageBonus={upgradeDamageBonus}, armorBonus={num7}, upgradeArmorBonus={num8}");
			void ApplyStat(ref float stat, float baseValue)
			{
				if (baseValue > 0f)
				{
					stat += damageBonus + upgradeDamageBonus;
					stat = Mathf.Min(stat, cfg_StatBonusCapPerType.Value);
				}
			}
		}

		internal static void AttachBlacksmithingData(ItemData item, int level)
		{
			if (item == null)
			{
				return;
			}
			ItemInfo itemInfo = item.Data();
			if (itemInfo == null)
			{
				Debug.LogWarning((object)"[BlacksmithingExpanded] item.Data() returned null.");
				return;
			}
			BlacksmithingData blacksmithingData = itemInfo.Add<BlacksmithingData>();
			if (blacksmithingData != null)
			{
				blacksmithingData.blacksmithLevel = level;
				blacksmithingData.Save();
			}
		}

		internal static int GetPlayerBlacksmithingTier(Player player)
		{
			int playerBlacksmithingLevel = GetPlayerBlacksmithingLevel(player);
			return playerBlacksmithingLevel / cfg_InfusionTierInterval.Value;
		}

		private static void CacheBaseStats(ItemData item)
		{
			//IL_0048: Unknown result type (might be due to invalid IL or missing references)
			string name = item.m_shared.m_name;
			if (!baseArmorLookup.ContainsKey(name))
			{
				baseArmorLookup[name] = item.m_shared.m_armor;
				baseDamageLookup[name] = ((DamageTypes)(ref item.m_shared.m_damages)).Clone();
				baseDurabilityLookup[name] = item.m_shared.m_maxDurability;
			}
		}

		internal static void TryGiveExtraItemToCrafter(ItemData item, Player crafter)
		{
			try
			{
				if (item == null || (Object)(object)crafter == (Object)null)
				{
					return;
				}
				SharedData shared = item.m_shared;
				if (shared != null)
				{
					Inventory inventory = ((Humanoid)crafter).GetInventory();
					if (inventory != null)
					{
						inventory.AddItem(shared.m_name, 1, 1, 0, crafter.GetPlayerID(), crafter.GetPlayerName(), false);
					}
				}
			}
			catch (Exception arg)
			{
				Debug.LogError((object)$"[BlacksmithingExpanded] TryGiveExtraItemToCrafter error: {arg}");
			}
		}

		private static Sprite LoadEmbeddedSprite(string resourceName, int width, int height)
		{
			//IL_005a: Unknown result type (might be due to invalid IL or missing references)
			//IL_0060: Expected O, but got Unknown
			//IL_0077: Unknown result type (might be due to invalid IL or missing references)
			//IL_007c: Unknown result type (might be due to invalid IL or missing references)
			byte[] array = null;
			using (MemoryStream memoryStream = new MemoryStream())
			{
				Stream manifestResourceStream = Assembly.GetExecutingAssembly().GetManifestResourceStream("BlacksmithingExpanded.icons." + resourceName);
				if (manifestResourceStream == null)
				{
					throw new FileNotFoundException("Embedded resource not found: " + resourceName);
				}
				manifestResourceStream.CopyTo(memoryStream);
				array = memoryStream.ToArray();
			}
			Texture2D val = new Texture2D(0, 0);
			ImageConversion.LoadImage(val, array);
			return Sprite.Create(val, new Rect(0f, 0f, (float)width, (float)height), Vector2.zero);
		}

		private void OnDestroy()
		{
			Harmony obj = harmony;
			if (obj != null)
			{
				obj.UnpatchSelf();
			}
		}
	}
}
namespace Microsoft.CodeAnalysis
{
	[<7b904072-6faf-4917-a632-ca9c3f6c0e61>Embedded]
	[CompilerGenerated]
	internal sealed class <7b904072-6faf-4917-a632-ca9c3f6c0e61>EmbeddedAttribute : Attribute
	{
	}
}
namespace System.Runtime.CompilerServices
{
	[AttributeUsage(AttributeTargets.Class | AttributeTargets.Property | AttributeTargets.Field | AttributeTargets.Event | AttributeTargets.Parameter | AttributeTargets.ReturnValue | AttributeTargets.GenericParameter, AllowMultiple = false, Inherited = false)]
	[<7b904072-6faf-4917-a632-ca9c3f6c0e61>Embedded]
	[CompilerGenerated]
	internal sealed class <c5aeeba3-342a-4dc3-8fb3-b3799674ceea>NullableAttribute : Attribute
	{
		public readonly byte[] NullableFlags;

		public <c5aeeba3-342a-4dc3-8fb3-b3799674ceea>NullableAttribute(byte P_0)
		{
			NullableFlags = new byte[1] { P_0 };
		}

		public <c5aeeba3-342a-4dc3-8fb3-b3799674ceea>NullableAttribute(byte[] P_0)
		{
			NullableFlags = P_0;
		}
	}
	[CompilerGenerated]
	[<7b904072-6faf-4917-a632-ca9c3f6c0e61>Embedded]
	[AttributeUsage(AttributeTargets.Class | AttributeTargets.Struct | AttributeTargets.Method | AttributeTargets.Interface | AttributeTargets.Delegate, AllowMultiple = false, Inherited = false)]
	internal sealed class <2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContextAttribute : Attribute
	{
		public readonly byte Flag;

		public <2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContextAttribute(byte P_0)
		{
			Flag = P_0;
		}
	}
	[CompilerGenerated]
	[<7b904072-6faf-4917-a632-ca9c3f6c0e61>Embedded]
	[AttributeUsage(AttributeTargets.Module, AllowMultiple = false, Inherited = false)]
	internal sealed class <f86a9b4b-6174-4e5e-b849-c3a03cae91f4>RefSafetyRulesAttribute : Attribute
	{
		public readonly int Version;

		public <f86a9b4b-6174-4e5e-b849-c3a03cae91f4>RefSafetyRulesAttribute(int P_0)
		{
			Version = P_0;
		}
	}
}
namespace ServerSync
{
	[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(0)]
	[PublicAPI]
	[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(1)]
	internal abstract class OwnConfigEntryBase
	{
		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)]
		public object LocalBaseValue;

		public bool SynchronizedConfig = true;

		public abstract ConfigEntryBase BaseConfig { get; }
	}
	[PublicAPI]
	[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(0)]
	[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(1)]
	internal class SyncedConfigEntry<[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)] 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;
			base..ctor();
		}

		public void AssignLocalValue(T value)
		{
			if (LocalBaseValue == null)
			{
				Value = value;
			}
			else
			{
				LocalBaseValue = value;
			}
		}
	}
	[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(0)]
	[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(2)]
	internal abstract class CustomSyncedValueBase
	{
		public object LocalBaseValue;

		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(1)]
		public readonly string Identifier;

		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(1)]
		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;

		[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(1)]
		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;
			};
		}
	}
	[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(0)]
	[PublicAPI]
	[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(1)]
	internal sealed class CustomSyncedValue<[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)] 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]
	[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(0)]
	[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(1)]
	internal class ConfigSync
	{
		[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)]
		[HarmonyPatch(typeof(ZRpc), "HandlePackage")]
		private static class SnatchCurrentlyHandlingRPC
		{
			[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)]
			public static ZRpc currentRpc;

			[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(1)]
			[HarmonyPrefix]
			private static void Prefix(ZRpc __instance)
			{
				currentRpc = __instance;
			}
		}

		[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)]
		[HarmonyPatch(typeof(ZNet), "Awake")]
		internal static class RegisterRPCPatch
		{
			[HarmonyPostfix]
			[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(1)]
			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());
				}
				[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(1)]
				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));
					}
				}
				[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(1)]
				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);
						}
					}
				}
			}
		}

		[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)]
		[HarmonyPatch(typeof(ZNet), "OnNewConnection")]
		private static class RegisterClientRPCPatch
		{
			[HarmonyPostfix]
			[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(1)]
			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);
				}
			}
		}

		[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)]
		private class ParsedConfigs
		{
			[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(new byte[] { 1, 1, 2 })]
			public readonly Dictionary<OwnConfigEntryBase, object> configValues = new Dictionary<OwnConfigEntryBase, object>();

			[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(new byte[] { 1, 1, 2 })]
			public readonly Dictionary<CustomSyncedValueBase, object> customValues = new Dictionary<CustomSyncedValueBase, object>();
		}

		[HarmonyPatch(typeof(ZNet), "Shutdown")]
		[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)]
		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")]
		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(0)]
		private class SendConfigsAfterLogin
		{
			[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(0)]
			private class BufferingSocket : ZPlayFabSocket, 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;
					((ZPlayFabSocket)this)..ctor();
				}

				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([<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(new byte[] { 2, 1, 1 })] 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())
				{
					return;
				}
				BufferingSocket bufferingSocket = new BufferingSocket(rpc.GetSocket());
				AccessTools.DeclaredField(typeof(ZRpc), "m_socket").SetValue(rpc, bufferingSocket);
				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)
				{
					FieldInfo fieldInfo = AccessTools.DeclaredField(typeof(ZNetPeer), "m_socket");
					object? value = fieldInfo.GetValue(val);
					ZPlayFabSocket val2 = (ZPlayFabSocket)((value is ZPlayFabSocket) ? value : null);
					if (val2 != null)
					{
						typeof(ZPlayFabSocket).GetField("m_remotePlayerId").SetValue(bufferingSocket, val2.m_remotePlayerId);
					}
					fieldInfo.SetValue(val, bufferingSocket);
				}
				if (__state == null)
				{
					__state = new Dictionary<Assembly, BufferingSocket>();
				}
				__state[Assembly.GetExecutingAssembly()] = bufferingSocket;
			}

			[HarmonyPostfix]
			private static void Postfix(Dictionary<Assembly, BufferingSocket> __state, ZNet __instance, ZRpc rpc)
			{
				ZNetPeer peer;
				if (__instance.IsServer())
				{
					object obj = AccessTools.DeclaredMethod(typeof(ZNet), "GetPeer", new Type[1] { typeof(ZRpc) }, (Type[])null).Invoke(__instance, new object[1] { rpc });
					peer = (ZNetPeer)((obj is ZNetPeer) ? obj : null);
					if (peer == null)
					{
						SendBufferedData();
					}
					else
					{
						((MonoBehaviour)__instance).StartCoroutine(sendAsync());
					}
				}
				void SendBufferedData()
				{
					if (rpc.GetSocket() is BufferingSocket bufferingSocket)
					{
						AccessTools.DeclaredField(typeof(ZRpc), "m_socket").SetValue(rpc, bufferingSocket.Original);
						object? obj2 = AccessTools.DeclaredMethod(typeof(ZNet), "GetPeer", new Type[1] { typeof(ZRpc) }, (Type[])null).Invoke(__instance, new object[1] { rpc });
						ZNetPeer val = (ZNetPeer)((obj2 is ZNetPeer) ? obj2 : null);
						if (val != null)
						{
							AccessTools.DeclaredField(typeof(ZNetPeer), "m_socket").SetValue(val, bufferingSocket.Original);
						}
					}
					BufferingSocket bufferingSocket2 = __state[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(rpc.GetSocket().GetHostName())) : listContainsId.Invoke(ZNet.instance, new object[2]
							{
								adminList,
								rpc.GetSocket().GetHostName()
							}))
						});
						ZPackage package = ConfigsToPackage(configSync.allConfigs.Select([<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (OwnConfigEntryBase c) => c.BaseConfig), configSync.allCustomValues, entries, partial: false);
						yield return ((MonoBehaviour)__instance).StartCoroutine(configSync.sendZPackage(new List<ZNetPeer> { peer }, package));
					}
					SendBufferedData();
				}
			}
		}

		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(0)]
		private class PackageEntry
		{
			public string section = null;

			public string key = null;

			public Type type = null;

			[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)]
			public object value;
		}

		[HarmonyPatch(typeof(ConfigEntryBase), "GetSerializedValue")]
		[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)]
		private static class PreventSavingServerInfo
		{
			[HarmonyPrefix]
			[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(1)]
			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")]
		[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)]
		private static class PreventConfigRereadChangingValues
		{
			[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(1)]
			[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;
			}
		}

		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(0)]
		private class InvalidDeserializationTypeException : Exception
		{
			public string expected = null;

			public string received = null;

			public string field = "";
		}

		public static bool ProcessingServerUpdate;

		public readonly string Name;

		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)]
		public string DisplayName;

		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)]
		public string CurrentVersion;

		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)]
		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;

		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)]
		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[]>>();

		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(new byte[] { 1, 0, 1 })]
		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;


		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)]
		[method: <2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(2)]
		[field: <c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)]
		public event Action<bool> SourceOfTruthChanged;

		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)]
		[method: <2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(2)]
		[field: <c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)]
		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<[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)] T>(ConfigEntry<T> configEntry)
		{
			OwnConfigEntryBase ownConfigEntryBase = configData((ConfigEntryBase)(object)configEntry);
			SyncedConfigEntry<T> syncedEntry = ownConfigEntryBase as SyncedConfigEntry<T>;
			if (syncedEntry == null)
			{
				syncedEntry = new SyncedConfigEntry<T>(configEntry);
				AccessTools.DeclaredField(typeof(ConfigDescription), "<Tags>k__BackingField").SetValue(((ConfigEntryBase)configEntry).Description, new object[1]
				{
					new ConfigurationManagerAttributes()
				}.Concat(((ConfigEntryBase)configEntry).Description.Tags ?? Array.Empty<object>()).Concat(new SyncedConfigEntry<T>[1] { syncedEntry }).ToArray());
				configEntry.SettingChanged += [<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (object _, EventArgs _) =>
				{
					if (!ProcessingServerUpdate && syncedEntry.SynchronizedConfig)
					{
						Broadcast(ZRoutedRpc.Everybody, (ConfigEntryBase)configEntry);
					}
				};
				allConfigs.Add(syncedEntry);
			}
			return syncedEntry;
		}

		public SyncedConfigEntry<T> AddLockingConfigEntry<[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(0)] T>(ConfigEntry<T> lockingConfig) where T : IConvertible
		{
			if (lockedConfig != null)
			{
				throw new Exception("Cannot initialize locking ConfigEntry twice");
			}
			lockedConfig = AddConfigEntry<T>(lockingConfig);
			lockingConfig.SettingChanged += [<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (object _, EventArgs _) =>
			{
				this.lockedConfigChanged?.Invoke();
			};
			return (SyncedConfigEntry<T>)lockedConfig;
		}

		internal void AddCustomValue(CustomSyncedValueBase customValue)
		{
			if (allCustomValues.Select([<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (CustomSyncedValueBase v) => v.Identifier).Concat(new string[1] { "serverversion" }).Contains(customValue.Identifier))
			{
				throw new Exception("Cannot have multiple settings with the same name or with a reserved name (serverversion)");
			}
			allCustomValues.Add(customValue);
			allCustomValues = new HashSet<CustomSyncedValueBase>(allCustomValues.OrderByDescending([<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (CustomSyncedValueBase v) => v.Priority));
			customValue.ValueChanged += delegate
			{
				if (!ProcessingServerUpdate)
				{
					Broadcast(ZRoutedRpc.Everybody, customValue);
				}
			};
		}

		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(([<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(new byte[] { 0, 1 })] 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 var 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([<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (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;
					val2.Save();
				}
				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([<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (OwnConfigEntryBase c) => c.SynchronizedConfig).ToDictionary([<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (OwnConfigEntryBase c) => c.BaseConfig.Definition.Section + "_" + c.BaseConfig.Definition.Key, [<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (OwnConfigEntryBase c) => c);
			Dictionary<string, CustomSyncedValueBase> dictionary2 = allCustomValues.ToDictionary([<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (CustomSyncedValueBase c) => c.Identifier, [<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (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)
		{
			ConfigSync configSync = configSyncs.FirstOrDefault([<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (ConfigSync cs) => cs.allConfigs.Contains(config));
			if (configSync == null)
			{
				return true;
			}
			return configSync.IsSourceOfTruth || !config.SynchronizedConfig || config.LocalBaseValue == null || (!configSync.IsLocked && (config != 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([<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (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([<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (CustomSyncedValueBase config) => config.LocalBaseValue != null))
			{
				item2.BoxedValue = item2.LocalBaseValue;
				item2.LocalBaseValue = null;
			}
			lockedConfigChanged -= serverLockedSettingChanged;
			serverLockedSettingChanged();
		}

		private IEnumerator<bool> distributeConfigToPeers(ZNetPeer peer, ZPackage package)
		{
			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 (peer.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)
				{
					peer.m_rpc.Invoke(text, new object[1] { pkg });
				}
				else
				{
					rpc.InvokeRoutedRPC(peer.m_server ? 0 : peer.m_uid, text, new object[1] { pkg });
				}
			}
			IEnumerable<bool> waitForQueue()
			{
				float timeout = Time.time + 30f;
				while (peer.m_socket.GetSendQueueSize() > 20000)
				{
					if (Time.time > timeout)
					{
						Debug.Log((object)$"Disconnecting {peer.m_uid} after 30 seconds config sending timeout");
						peer.m_rpc.Invoke("Error", new object[1] { (object)(ConnectionStatus)5 });
						ZNet.instance.Disconnect(peer);
						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([<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (ZNetPeer p) => p.m_uid == target).ToList();
			}
			return sendZPackage(list, package);
		}

		private IEnumerator sendZPackage(List<ZNetPeer> peers, ZPackage package)
		{
			if (!Object.op_Implicit((Object)(object)ZNet.instance))
			{
				yield break;
			}
			byte[] rawData = package.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());
				package = compressedPackage;
			}
			List<IEnumerator<bool>> writers = (from peer in peers
				where peer.IsReady()
				select peer into p
				select distributeConfigToPeers(p, package)).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));
				}
			}
		}

		[return: <c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)]
		private static OwnConfigEntryBase configData(ConfigEntryBase config)
		{
			return config.Description.Tags?.OfType<OwnConfigEntryBase>().SingleOrDefault();
		}

		[return: <c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(new byte[] { 2, 1 })]
		public static SyncedConfigEntry<T> ConfigData<[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)] T>(ConfigEntry<T> config)
		{
			return ((ConfigEntryBase)config).Description.Tags?.OfType<SyncedConfigEntry<T>>().SingleOrDefault();
		}

		private static T configAttribute<[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)] 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([<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(new byte[] { 2, 1 })] IEnumerable<ConfigEntryBase> configs = null, [<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(new byte[] { 2, 1 })] IEnumerable<CustomSyncedValueBase> customValues = null, [<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(new byte[] { 2, 1 })] 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([<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (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, [<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)] 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 != null && type3.IsAssignableFrom(type))
				{
					int num3 = package.ReadInt();
					object obj2 = Activator.CreateInstance(type);
					MethodInfo method = type3.GetMethod("Add");
					for (int k = 0; k < num3; k++)
					{
						method.Invoke(obj2, new object[1] { ReadValueWithTypeFromZPackage(package, type.GenericTypeArguments[0]) });
					}
					return obj2;
				}
			}
			ParameterInfo parameterInfo = (ParameterInfo)FormatterServices.GetUninitializedObject(typeof(ParameterInfo));
			AccessTools.DeclaredField(typeof(ParameterInfo), "ClassImpl").SetValue(parameterInfo, type);
			List<object> source = new List<object>();
			ZRpc.Deserialize(new ParameterInfo[2] { null, parameterInfo }, package, ref source);
			return source.First();
		}
	}
	[HarmonyPatch]
	[PublicAPI]
	[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(0)]
	[<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(1)]
	internal class VersionCheck
	{
		private static readonly HashSet<VersionCheck> versionChecks;

		private static readonly Dictionary<string, string> notProcessedNames;

		public string Name;

		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)]
		private string displayName;

		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)]
		private string currentVersion;

		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)]
		private string minimumRequiredVersion;

		public bool ModRequired = true;

		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)]
		private string ReceivedCurrentVersion;

		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)]
		private string ReceivedMinimumRequiredVersion;

		private readonly List<ZRpc> ValidatedClients = new List<ZRpc>();

		[<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)]
		private ConfigSync ConfigSync;

		public string DisplayName
		{
			get
			{
				return displayName ?? Name;
			}
			set
			{
				displayName = value;
			}
		}

		public string CurrentVersion
		{
			get
			{
				return currentVersion ?? "0.0.0";
			}
			set
			{
				currentVersion = value;
			}
		}

		public string MinimumRequiredVersion
		{
			get
			{
				return minimumRequiredVersion ?? (ModRequired ? CurrentVersion : "0.0.0");
			}
			set
			{
				minimumRequiredVersion = value;
			}
		}

		private static void PatchServerSync()
		{
			//IL_005e: Unknown result type (might be due to invalid IL or missing references)
			//IL_0064: Expected O, but got Unknown
			Patches patchInfo = PatchProcessor.GetPatchInfo((MethodBase)AccessTools.DeclaredMethod(typeof(ZNet), "Awake", (Type[])null, (Type[])null));
			if (patchInfo != null && patchInfo.Postfixes.Count([<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (Patch p) => p.PatchMethod.DeclaringType == typeof(ConfigSync.RegisterRPCPatch)) > 0)
			{
				return;
			}
			Harmony val = new Harmony("org.bepinex.helpers.ServerSync");
			foreach (Type item in from t in typeof(ConfigSync).GetNestedTypes(BindingFlags.NonPublic).Concat(new Type[1] { typeof(VersionCheck) })
				where t.IsClass
				select t)
			{
				val.PatchAll(item);
			}
		}

		static VersionCheck()
		{
			versionChecks = new HashSet<VersionCheck>();
			notProcessedNames = new Dictionary<string, string>();
			typeof(ThreadingHelper).GetMethod("StartSyncInvoke").Invoke(ThreadingHelper.Instance, new object[1]
			{
				new Action(PatchServerSync)
			});
		}

		public VersionCheck(string name)
		{
			Name = name;
			ModRequired = true;
			versionChecks.Add(this);
		}

		public VersionCheck(ConfigSync configSync)
		{
			ConfigSync = configSync;
			Name = ConfigSync.Name;
			versionChecks.Add(this);
		}

		public void Initialize()
		{
			ReceivedCurrentVersion = null;
			ReceivedMinimumRequiredVersion = null;
			if (ConfigSync != null)
			{
				Name = ConfigSync.Name;
				DisplayName = ConfigSync.DisplayName;
				CurrentVersion = ConfigSync.CurrentVersion;
				MinimumRequiredVersion = ConfigSync.MinimumRequiredVersion;
				ModRequired = ConfigSync.ModRequired;
			}
		}

		private bool IsVersionOk()
		{
			if (ReceivedMinimumRequiredVersion == null || ReceivedCurrentVersion == null)
			{
				return !ModRequired;
			}
			bool flag = new Version(CurrentVersion) >= new Version(ReceivedMinimumRequiredVersion);
			bool flag2 = new Version(ReceivedCurrentVersion) >= new Version(MinimumRequiredVersion);
			return flag && flag2;
		}

		private string ErrorClient()
		{
			if (ReceivedMinimumRequiredVersion == null)
			{
				return DisplayName + " is not installed on the server.";
			}
			return (new Version(CurrentVersion) >= new Version(ReceivedMinimumRequiredVersion)) ? (DisplayName + " may not be higher than version " + ReceivedCurrentVersion + ". You have version " + CurrentVersion + ".") : (DisplayName + " needs to be at least version " + ReceivedMinimumRequiredVersion + ". You have version " + CurrentVersion + ".");
		}

		private string ErrorServer(ZRpc rpc)
		{
			return "Disconnect: The client (" + rpc.GetSocket().GetHostName() + ") doesn't have the correct " + DisplayName + " version " + MinimumRequiredVersion;
		}

		private string Error([<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(2)] ZRpc rpc = null)
		{
			return (rpc == null) ? ErrorClient() : ErrorServer(rpc);
		}

		private static VersionCheck[] GetFailedClient()
		{
			return versionChecks.Where([<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (VersionCheck check) => !check.IsVersionOk()).ToArray();
		}

		private static VersionCheck[] GetFailedServer(ZRpc rpc)
		{
			return versionChecks.Where([<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (VersionCheck check) => check.ModRequired && !check.ValidatedClients.Contains(rpc)).ToArray();
		}

		private static void Logout()
		{
			Game.instance.Logout(true, true);
			AccessTools.DeclaredField(typeof(ZNet), "m_connectionStatus").SetValue(null, (object)(ConnectionStatus)3);
		}

		private static void DisconnectClient(ZRpc rpc)
		{
			rpc.Invoke("Error", new object[1] { 3 });
		}

		private static void CheckVersion(ZRpc rpc, ZPackage pkg)
		{
			CheckVersion(rpc, pkg, null);
		}

		private static void CheckVersion(ZRpc rpc, ZPackage pkg, [<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(new byte[] { 2, 1, 1 })] Action<ZRpc, ZPackage> original)
		{
			string text = pkg.ReadString();
			string text2 = pkg.ReadString();
			string text3 = pkg.ReadString();
			bool flag = false;
			foreach (VersionCheck versionCheck in versionChecks)
			{
				if (!(text != versionCheck.Name))
				{
					Debug.Log((object)("Received " + versionCheck.DisplayName + " version " + text3 + " and minimum version " + text2 + " from the " + (ZNet.instance.IsServer() ? "client" : "server") + "."));
					versionCheck.ReceivedMinimumRequiredVersion = text2;
					versionCheck.ReceivedCurrentVersion = text3;
					if (ZNet.instance.IsServer() && versionCheck.IsVersionOk())
					{
						versionCheck.ValidatedClients.Add(rpc);
					}
					flag = true;
				}
			}
			if (flag)
			{
				return;
			}
			pkg.SetPos(0);
			if (original != null)
			{
				original(rpc, pkg);
				if (pkg.GetPos() == 0)
				{
					notProcessedNames.Add(text, text3);
				}
			}
		}

		[HarmonyPrefix]
		[HarmonyPatch(typeof(ZNet), "RPC_PeerInfo")]
		private static bool RPC_PeerInfo(ZRpc rpc, ZNet __instance)
		{
			VersionCheck[] array = (__instance.IsServer() ? GetFailedServer(rpc) : GetFailedClient());
			if (array.Length == 0)
			{
				return true;
			}
			VersionCheck[] array2 = array;
			foreach (VersionCheck versionCheck in array2)
			{
				Debug.LogWarning((object)versionCheck.Error(rpc));
			}
			if (__instance.IsServer())
			{
				DisconnectClient(rpc);
			}
			else
			{
				Logout();
			}
			return false;
		}

		[HarmonyPatch(typeof(ZNet), "OnNewConnection")]
		[HarmonyPrefix]
		private static void RegisterAndCheckVersion(ZNetPeer peer, ZNet __instance)
		{
			//IL_018e: Unknown result type (might be due to invalid IL or missing references)
			//IL_0195: Expected O, but got Unknown
			notProcessedNames.Clear();
			IDictionary dictionary = (IDictionary)typeof(ZRpc).GetField("m_functions", BindingFlags.Instance | BindingFlags.NonPublic).GetValue(peer.m_rpc);
			if (dictionary.Contains(StringExtensionMethods.GetStableHashCode("ServerSync VersionCheck")))
			{
				object obj = dictionary[StringExtensionMethods.GetStableHashCode("ServerSync VersionCheck")];
				Action<ZRpc, ZPackage> action = (Action<ZRpc, ZPackage>)obj.GetType().GetField("m_action", BindingFlags.Instance | BindingFlags.NonPublic).GetValue(obj);
				peer.m_rpc.Register<ZPackage>("ServerSync VersionCheck", (Action<ZRpc, ZPackage>)([<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (ZRpc rpc, [<c5aeeba3-342a-4dc3-8fb3-b3799674ceea>Nullable(1)] ZPackage pkg) =>
				{
					CheckVersion(rpc, pkg, action);
				}));
			}
			else
			{
				peer.m_rpc.Register<ZPackage>("ServerSync VersionCheck", (Action<ZRpc, ZPackage>)CheckVersion);
			}
			foreach (VersionCheck versionCheck in versionChecks)
			{
				versionCheck.Initialize();
				if (versionCheck.ModRequired || __instance.IsServer())
				{
					Debug.Log((object)("Sending " + versionCheck.DisplayName + " version " + versionCheck.CurrentVersion + " and minimum version " + versionCheck.MinimumRequiredVersion + " to the " + (__instance.IsServer() ? "client" : "server") + "."));
					ZPackage val = new ZPackage();
					val.Write(versionCheck.Name);
					val.Write(versionCheck.MinimumRequiredVersion);
					val.Write(versionCheck.CurrentVersion);
					peer.m_rpc.Invoke("ServerSync VersionCheck", new object[1] { val });
				}
			}
		}

		[HarmonyPatch(typeof(ZNet), "Disconnect")]
		[HarmonyPrefix]
		private static void RemoveDisconnected(ZNetPeer peer, ZNet __instance)
		{
			if (!__instance.IsServer())
			{
				return;
			}
			foreach (VersionCheck versionCheck in versionChecks)
			{
				versionCheck.ValidatedClients.Remove(peer.m_rpc);
			}
		}

		[HarmonyPatch(typeof(FejdStartup), "ShowConnectError")]
		[HarmonyPostfix]
		private static void ShowConnectionError(FejdStartup __instance)
		{
			//IL_000e: Unknown result type (might be due to invalid IL or missing references)
			//IL_0014: Invalid comparison between Unknown and I4
			//IL_0186: Unknown result type (might be due to invalid IL or missing references)
			//IL_018b: Unknown result type (might be due to invalid IL or missing references)
			//IL_0199: Unknown result type (might be due to invalid IL or missing references)
			//IL_01de: Unknown result type (might be due to invalid IL or missing references)
			//IL_01ea: Unknown result type (might be due to invalid IL or missing references)
			//IL_01f8: Unknown result type (might be due to invalid IL or missing references)
			//IL_020a: Unknown result type (might be due to invalid IL or missing references)
			//IL_0219: Unknown result type (might be due to invalid IL or missing references)
			//IL_021e: Unknown result type (might be due to invalid IL or missing references)
			//IL_0229: Unknown result type (might be due to invalid IL or missing references)
			if (!__instance.m_connectionFailedPanel.activeSelf || (int)ZNet.GetConnectionStatus() != 3)
			{
				return;
			}
			bool flag = false;
			VersionCheck[] failedClient = GetFailedClient();
			if (failedClient.Length != 0)
			{
				string text = string.Join("\n", failedClient.Select([<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (VersionCheck check) => check.Error()));
				TMP_Text connectionFailedError = __instance.m_connectionFailedError;
				connectionFailedError.text = connectionFailedError.text + "\n" + text;
				flag = true;
			}
			foreach (KeyValuePair<string, string> item in notProcessedNames.OrderBy([<2cf012d7-e5d1-47f1-a0ca-f11e3aae9cbb>NullableContext(0)] (KeyValuePair<string, string> kv) => kv.Key))
			{
				if (!__instance.m_connectionFailedError.text.Contains(item.Key))
				{
					TMP_Text connectionFailedError2 = __instance.m_connectionFailedError;
					connectionFailedError2.text = connectionFailedError2.text + "\nServer expects you to have " + item.Key + " (Version: " + item.Value + ") installed.";
					flag = true;
				}
			}
			if (flag)
			{
				RectTransform component = ((Component)__instance.m_connectionFailedPanel.transform.Find("Image")).GetComponent<RectTransform>();
				Vector2 sizeDelta = component.sizeDelta;
				sizeDelta.x = 675f;
				component.sizeDelta = sizeDelta;
				__instance.m_connectionFailedError.ForceMeshUpdate(false, false);
				float num = __instance.m_connectionFailedError.renderedHeight + 105f;
				RectTransform component2 = ((Component)((Component)component).transform.Find("ButtonOk")).GetComponent<RectTransform>();
				component2.anchoredPosition = new Vector2(component2.anchoredPosition.x, component2.anchoredPosition.y - (num - component.sizeDelta.y) / 2f);
				sizeDelta = component.sizeDelta;
				sizeDelta.y = num;
				component.sizeDelta = sizeDelta;
			}
		}
	}
}
namespace Microsoft.CodeAnalysis
{
	[CompilerGenerated]
	[<2ea98159-4873-4e85-beec-f3cb332fe4a3>Embedded]
	internal sealed class <2ea98159-4873-4e85-beec-f3cb332fe4a3>EmbeddedAttribute : Attribute
	{
	}
}
namespace System.Runtime.CompilerServices
{
	[CompilerGenerated]
	[<2ea98159-4873-4e85-beec-f3cb332fe4a3>Embedded]
	[AttributeUsage(AttributeTargets.Class | AttributeTargets.Property | AttributeTargets.Field | AttributeTargets.Event | AttributeTargets.Parameter | AttributeTargets.ReturnValue | AttributeTargets.GenericParameter, AllowMultiple = false, Inherited = false)]
	internal sealed class <47474f70-c3fb-4d7b-9263-9a9abcfb2515>NullableAttribute : Attribute
	{
		public readonly byte[] NullableFlags;

		public <47474f70-c3fb-4d7b-9263-9a9abcfb2515>NullableAttribute(byte P_0)
		{
			NullableFlags = new byte[1] { P_0 };
		}

		public <47474f70-c3fb-4d7b-9263-9a9abcfb2515>NullableAttribute(byte[] P_0)
		{
			NullableFlags = P_0;
		}
	}
	[CompilerGenerated]
	[<2ea98159-4873-4e85-beec-f3cb332fe4a3>Embedded]
	[AttributeUsage(AttributeTargets.Class | AttributeTargets.Struct | AttributeTargets.Method | AttributeTargets.Interface | AttributeTargets.Delegate, AllowMultiple = false, Inherited = false)]
	internal sealed class <69686d9f-7b13-4d56-a555-17491987f043>NullableContextAttribute : Attribute
	{
		public readonly byte Flag;

		public <69686d9f-7b13-4d56-a555-17491987f043>NullableContextAttribute(byte P_0)
		{
			Flag = P_0;
		}
	}
}
namespace SkillManager
{
	[<69686d9f-7b13-4d56-a555-17491987f043>NullableContext(1)]
	[<47474f70-c3fb-4d7b-9263-9a9abcfb2515>Nullable(0)]
	[PublicAPI]
	internal class Skill
	{
		[<47474f70-c3fb-4d7b-9263-9a9abcfb2515>Nullable(0)]
		public static class LocalizationCache
		{
			private static readonly Dictionary<string, Localization> localizations = new Dictionary<string, Localization>();

			internal static void LocalizationPostfix(Localization __instance, string language)
			{
				string key = localizations.FirstOrDefault([<69686d9f-7b13-4d56-a555-17491987f043>NullableContext(0)] (KeyValuePair<string, Localization> l) => l.Value == __instance).Key;
				if (key != null)
				{
					localizations.Remove(key);
				}
				if (!localizations.ContainsKey(language))
				{
					localizations.Add(language, __instance);
				}
			}

			public static Localization ForLanguage([<47474f70-c3fb-4d7b-9263-9a9abcfb2515>Nullable(2)] string language = null)
			{
				//IL_0024: Unknown result type (might be due to invalid IL or missing references)
				//IL_002a