Decompiled source of The Archive UNSTABLE TEST ONLY v0.0.6

BepInEx/plugins/TheArchive/TheArchive.Core.dll

Decompiled 2 weeks ago
using System;
using System.Collections;
using System.Collections.Generic;
using System.Diagnostics;
using System.Globalization;
using System.IO;
using System.Linq;
using System.Reflection;
using System.Runtime.CompilerServices;
using System.Runtime.InteropServices;
using System.Runtime.Versioning;
using System.Security;
using System.Security.Cryptography;
using System.Security.Permissions;
using System.Text;
using System.Text.RegularExpressions;
using System.Threading;
using BepInEx;
using BepInEx.Bootstrap;
using BepInEx.Core.Logging.Interpolation;
using BepInEx.Logging;
using BepInEx.Unity.IL2CPP;
using BepInEx.Unity.IL2CPP.Hook;
using BepInEx.Unity.IL2CPP.Utils;
using Clonesoft.Json;
using Clonesoft.Json.Converters;
using Clonesoft.Json.Serialization;
using HarmonyLib;
using Il2CppInterop.Runtime;
using Il2CppInterop.Runtime.Injection;
using Il2CppInterop.Runtime.InteropTypes;
using Il2CppInterop.Runtime.Runtime;
using Il2CppSystem;
using Il2CppSystem.Reflection;
using Microsoft.CodeAnalysis;
using Mono.Cecil;
using SemanticVersioning;
using TheArchive.Core;
using TheArchive.Core.Attributes;
using TheArchive.Core.Attributes.Feature;
using TheArchive.Core.Attributes.Feature.Settings;
using TheArchive.Core.Bootstrap;
using TheArchive.Core.Discord;
using TheArchive.Core.Exceptions;
using TheArchive.Core.FeaturesAPI;
using TheArchive.Core.FeaturesAPI.Components;
using TheArchive.Core.FeaturesAPI.Settings;
using TheArchive.Core.Localization;
using TheArchive.Core.Managers;
using TheArchive.Core.Models;
using TheArchive.Core.ModulesAPI;
using TheArchive.Core.Settings;
using TheArchive.Interfaces;
using TheArchive.Loader;
using TheArchive.Utilities;
using UnityEngine;

[assembly: CompilationRelaxations(8)]
[assembly: RuntimeCompatibility(WrapNonExceptionThrows = true)]
[assembly: Debuggable(DebuggableAttribute.DebuggingModes.IgnoreSymbolStoreSequencePoints)]
[assembly: InternalsVisibleTo("TheArchive.IL2CPP")]
[assembly: InternalsVisibleTo("TheArchive.MONO")]
[assembly: AssemblyDescription("")]
[assembly: AssemblyCopyright("Copyright ©  2021")]
[assembly: AssemblyTrademark("")]
[assembly: ComVisible(false)]
[assembly: Guid("7bb66dc5-d29d-4cdd-bb82-d228abe35bda")]
[assembly: AssemblyFileVersion("0.0.656")]
[assembly: AssemblyInformationalVersion("0.0.656-main+9e73217")]
[assembly: TargetFramework(".NETCoreApp,Version=v6.0", FrameworkDisplayName = ".NET 6.0")]
[assembly: AssemblyCompany("TheArchive.Core")]
[assembly: AssemblyConfiguration("R_BIE")]
[assembly: AssemblyProduct("TheArchive.Core")]
[assembly: AssemblyTitle("TheArchive.Core")]
[assembly: SecurityPermission(SecurityAction.RequestMinimum, SkipVerification = true)]
[assembly: AssemblyVersion("0.0.656.0")]
[module: UnverifiableCode]
namespace Microsoft.CodeAnalysis
{
	[CompilerGenerated]
	[Microsoft.CodeAnalysis.Embedded]
	internal sealed class EmbeddedAttribute : Attribute
	{
	}
}
namespace System.Runtime.CompilerServices
{
	[CompilerGenerated]
	[Microsoft.CodeAnalysis.Embedded]
	[AttributeUsage(AttributeTargets.Class | AttributeTargets.Property | AttributeTargets.Field | AttributeTargets.Event | AttributeTargets.Parameter | AttributeTargets.ReturnValue | AttributeTargets.GenericParameter, AllowMultiple = false, Inherited = false)]
	internal sealed class NullableAttribute : Attribute
	{
		public readonly byte[] NullableFlags;

		public NullableAttribute(byte P_0)
		{
			NullableFlags = new byte[1] { P_0 };
		}

		public NullableAttribute(byte[] P_0)
		{
			NullableFlags = P_0;
		}
	}
}
internal class ThisAssembly
{
	public class Git
	{
		public class BaseVersion
		{
			public const string Major = "0";

			public const string Minor = "0";

			public const string Patch = "0";
		}

		public class SemVer
		{
			public const string Major = "0";

			public const string Minor = "0";

			public const string Patch = "656";

			public const string Label = "";

			public const string DashLabel = "";

			public const string Source = "Default";
		}

		public const bool IsDirty = false;

		public const string IsDirtyString = "false";

		public const string RepositoryUrl = "https://github.com/Tuesday1028/GTFO_TheArchive";

		public const string Branch = "main";

		public const string Commit = "9e73217";

		public const string Sha = "9e732175a6a125ba6d9be550e3bfc3f775f3bf88";

		public const string CommitDate = "2025-02-12T10:45:59+08:00";

		public const string Commits = "656";

		public const string Tag = "";

		public const string BaseTag = "";
	}
}
namespace TheArchive
{
	public static class ArchiveMod
	{
		public const string GUID = "dev.AuriRex.gtfo.TheArchive";

		public const string MOD_NAME = "TheArchive";

		public const string ABBREVIATION = "Ar";

		public const string AUTHOR = "AuriRex";

		public const string VERSION_STRING = "0.0.656";

		public const string GITHUB_REPOSITORY_NAME = "GTFO_TheArchive";

		public const string GITHUB_OWNER_NAME = "AuriRex";

		public const string GITHUB_LINK = "https://github.com/AuriRex/GTFO_TheArchive";

		public static readonly bool GIT_IS_DIRTY = false;

		public const string GIT_COMMIT_SHORT_HASH = "9e73217";

		public const string GIT_COMMIT_DATE = "2025-02-12T10:45:59+08:00";

		public const string GIT_BASE_TAG = "";

		public const uint GTFO_STEAM_APPID = 493520u;

		public const string MTFO_GUID = "com.dak.MTFO";

		public const string ARCHIVE_CORE_FEATURE_GROUP = "Archive Core";

		public static readonly string CORE_PATH = Assembly.GetAssembly(typeof(ArchiveMod)).Location;

		private static JsonSerializerSettings _jsonSerializerSettings = null;

		private static string _currentlySelectedRundownKey = string.Empty;

		private static IArchiveModule _mainModule;

		private static readonly HashSet<Assembly> _inspectedAssemblies = new HashSet<Assembly>();

		private static readonly HashSet<Type> _typesToInitOnDataBlocksReady = new HashSet<Type>();

		private static readonly HashSet<Type> _typesToInitOnGameDataInit = new HashSet<Type>();

		private static readonly HashSet<IInitializable> _iinitializablesToInjectOnGameDataInit = new HashSet<IInitializable>();

		private static readonly HashSet<Assembly> _moduleAssemblies = new HashSet<Assembly>();

		private static readonly List<Type> _moduleTypes = new List<Type>();

		private const string kArchiveSettingsFile = "TheArchive_Settings.json";

		private static Harmony _harmonyInstance;

		public static ArchiveSettings Settings { get; private set; } = new ArchiveSettings();


		public static JsonSerializerSettings JsonSerializerSettings
		{
			get
			{
				//IL_0007: Unknown result type (might be due to invalid IL or missing references)
				//IL_000c: Unknown result type (might be due to invalid IL or missing references)
				//IL_0018: Expected O, but got Unknown
				//IL_0022: Unknown result type (might be due to invalid IL or missing references)
				//IL_002c: Expected O, but got Unknown
				if (_jsonSerializerSettings == null)
				{
					_jsonSerializerSettings = new JsonSerializerSettings
					{
						Formatting = (Formatting)1
					};
					_jsonSerializerSettings.Converters.Add((JsonConverter)new StringEnumConverter());
					_jsonSerializerSettings.ContractResolver = (IContractResolver)(object)ArchiveContractResolver.Instance;
				}
				return _jsonSerializerSettings;
			}
		}

		public static bool IsPlayingModded { get; private set; } = false;


		public static Utils.RundownID CurrentRundown { get; private set; } = Utils.RundownID.RundownUnitialized;


		public static GameBuildInfo CurrentBuildInfo { get; private set; }

		public static int CurrentGameState { get; private set; }

		public static bool IsOnALTBuild { get; private set; }

		public static bool IsInitialized { get; private set; } = false;


		public static string CurrentlySelectedRundownKey
		{
			get
			{
				return _currentlySelectedRundownKey;
			}
			internal set
			{
				_currentlySelectedRundownKey = value;
				ArchiveLogger.Debug($"Setting {"CurrentlySelectedRundownKey"} to \"{_currentlySelectedRundownKey}\".");
				if (string.IsNullOrEmpty(_currentlySelectedRundownKey))
				{
					CurrentlySelectedRundownPersistentID = 0u;
					return;
				}
				try
				{
					CurrentlySelectedRundownPersistentID = uint.Parse(_currentlySelectedRundownKey.Replace("Local_", ""));
				}
				catch (Exception ex)
				{
					ArchiveLogger.Error($"Failed to parse selected rundown persistentId from {"CurrentlySelectedRundownKey"} \"{CurrentlySelectedRundownKey}\"!");
					ArchiveLogger.Exception(ex);
				}
			}
		}

		public static uint CurrentlySelectedRundownPersistentID { get; set; } = 0u;


		public static HashSet<IArchiveModule> Modules { get; private set; } = new HashSet<IArchiveModule>();


		public static Type IL2CPP_BaseType { get; private set; } = null;


		public static event Action<Utils.RundownID> GameDataInitialized;

		public static event Action DataBlocksReady;

		public static event Action<int> GameStateChanged;

		public static event Action<bool> ApplicationFocusStateChanged;

		internal static event Action<IArchiveModule> OnNewModuleRegistered;

		internal static void OnApplicationStart(IArchiveLogger logger, Harmony harmonyInstance)
		{
			ArchiveLogger.logger = logger;
			_harmonyInstance = harmonyInstance;
			if (GIT_IS_DIRTY)
			{
				ArchiveLogger.Warning("Git is dirty, this is a development build!");
			}
			try
			{
				if (LoaderWrapper.IsGameIL2CPP())
				{
					IL2CPP_BaseType = ImplementationManager.FindTypeInCurrentAppDomain("Il2CppSystem.Object", exactMatch: true);
					ArchiveLogger.Debug("IL2CPP_BaseType: " + IL2CPP_BaseType?.FullName);
					if (IL2CPP_BaseType == null)
					{
						throw new Exception();
					}
				}
			}
			catch (Exception ex)
			{
				ArchiveLogger.Error("IL2CPP base type \"Il2CppSystem.Object\" could not be resolved!");
				ArchiveLogger.Exception(ex);
			}
			LoadConfig();
			if (LoaderWrapper.IsModInstalled("com.dak.MTFO"))
			{
				IsPlayingModded = true;
			}
			GTFOLogger.Logger = LoaderWrapper.CreateLoggerInstance("GTFO-Internals", ConsoleColor.DarkGray);
			CurrentRundown = BuildDB.GetCurrentRundownID(BuildDB.BuildNumber);
			ArchiveLogger.Msg(ConsoleColor.DarkMagenta, $"Current game revision determined to be {BuildDB.BuildNumber}! ({CurrentRundown})");
			GameBuildInfo currentBuildInfo = default(GameBuildInfo);
			currentBuildInfo.BuildNumber = BuildDB.BuildNumber;
			currentBuildInfo.Rundown = CurrentRundown;
			CurrentBuildInfo = currentBuildInfo;
			IsOnALTBuild = CurrentRundown.IsIncludedIn(Utils.RundownFlags.RundownAltOne.ToLatest());
			string path = Path.Combine(LoaderWrapper.GameDirectory, "steam_appid.txt");
			if (!File.Exists(path))
			{
				ArchiveLogger.Notice("Creating \"steam_appid.txt\" in GTFO folder ...");
				File.WriteAllText(path, $"{493520}");
			}
			FeatureManager.Internal_Init();
			try
			{
				_mainModule = CreateAndInitModule(LoadMainArchiveModule(LoaderWrapper.IsGameIL2CPP()).GetTypes().First((Type t) => typeof(IArchiveModule).IsAssignableFrom(t)));
			}
			catch (ReflectionTypeLoadException ex2)
			{
				ArchiveLogger.Error("Failed loading main module!!");
				ArchiveLogger.Exception(ex2);
				ArchiveLogger.Notice($"Loader Exceptions ({ex2.LoaderExceptions.Length}):");
				Exception[] loaderExceptions = ex2.LoaderExceptions;
				foreach (Exception obj in loaderExceptions)
				{
					ArchiveLogger.Warning(obj.Message);
					ArchiveLogger.Debug(obj.StackTrace);
				}
				ArchiveLogger.Info("-------------");
			}
			InitializeArchiveModuleChainloader();
			try
			{
				ApplyPatches(CurrentRundown);
			}
			catch (Exception ex3)
			{
				ArchiveLogger.Exception(ex3);
			}
		}

		internal static void OnApplicationQuit()
		{
			InitSingletonBase<FeatureManager>.Instance.OnApplicationQuit();
			CustomSettingsManager.OnApplicationQuit();
		}

		private static void LoadConfig()
		{
			LoadConfig(Path.Combine(LocalFiles.ModLocalLowPath, "TheArchive_Settings.json"));
		}

		private static void LoadConfig(string path)
		{
			try
			{
				ArchiveLogger.Info("Loading config file ... [" + path + "]");
				if (File.Exists(path))
				{
					Settings = JsonConvert.DeserializeObject<ArchiveSettings>(File.ReadAllText(path), JsonSerializerSettings);
				}
				SaveConfig(path);
			}
			catch (Exception ex)
			{
				ArchiveLogger.Exception(ex);
			}
		}

		private static void SaveConfig(string path)
		{
			ArchiveLogger.Debug("Saving config file ... [" + path + "]");
			File.WriteAllText(path, JsonConvert.SerializeObject((object)Settings, JsonSerializerSettings));
		}

		public static bool RegisterArchiveModule(Assembly asm)
		{
			foreach (Type item in from t in asm.GetTypes()
				where typeof(IArchiveModule).IsAssignableFrom(t)
				select t)
			{
				if (RegisterArchiveModule(item))
				{
					return true;
				}
			}
			return false;
		}

		public static bool RegisterArchiveModule(Type moduleType)
		{
			if (moduleType == null)
			{
				throw new ArgumentException("Module can't be null!");
			}
			if (_moduleTypes.Contains(moduleType))
			{
				throw new ArgumentException("Module \"" + moduleType.Name + "\" is already registered!");
			}
			if (!typeof(IArchiveModule).IsAssignableFrom(moduleType))
			{
				throw new ArgumentException($"Type \"{moduleType.Name}\" does not implement {"IArchiveModule"}!");
			}
			IArchiveModule archiveModule = CreateAndInitModule(moduleType);
			ArchiveMod.OnNewModuleRegistered?.Invoke(archiveModule);
			if (CurrentRundown != Utils.RundownID.RundownUnitialized)
			{
				archiveModule.Patcher?.PatchRundownSpecificMethods(archiveModule.GetType().Assembly);
				return true;
			}
			return false;
		}

		internal static void InitializeArchiveModuleChainloader()
		{
			((BaseChainloader<BasePlugin>)(object)IL2CPPChainloader.Instance).Finished += delegate
			{
				ArchiveModuleChainloader.Initialize();
			};
		}

		internal static void InvokeGameDataInitialized()
		{
			if (IsInitialized)
			{
				ArchiveLogger.Info("Reload triggered, skipping init.");
				return;
			}
			IsInitialized = true;
			ArchiveLogger.Info("GameData has been initialized, invoking event.");
			try
			{
				PresenceFormatter.Setup();
			}
			catch (Exception ex)
			{
				ArchiveLogger.Exception(ex);
			}
			foreach (Type item in _typesToInitOnGameDataInit)
			{
				IInitializable initializable = null;
				try
				{
					InitInitializables(item, out initializable);
				}
				catch (Exception ex2)
				{
					ArchiveLogger.Error("Trying to Init \"" + item.FullName + "\" threw an exception:");
					ArchiveLogger.Exception(ex2);
				}
				try
				{
					InjectInstanceIntoModules(initializable);
				}
				catch (Exception ex3)
				{
					ArchiveLogger.Error("Trying to Inject \"" + item.FullName + "\" threw an exception:");
					ArchiveLogger.Exception(ex3);
				}
			}
			foreach (IInitializable item2 in _iinitializablesToInjectOnGameDataInit)
			{
				try
				{
					InjectInstanceIntoModules(item2);
				}
				catch (Exception ex4)
				{
					ArchiveLogger.Error("Trying to Inject \"" + item2.GetType().FullName + "\" threw an exception:");
					ArchiveLogger.Exception(ex4);
				}
			}
			InitSingletonBase<FeatureManager>.Instance.OnGameDataInitialized();
			ArchiveMod.GameDataInitialized?.Invoke(CurrentRundown);
		}

		internal static void InvokeDataBlocksReady()
		{
			try
			{
				DataBlockManager.Setup();
			}
			catch (Exception ex)
			{
				ArchiveLogger.Exception(ex);
			}
			ArchiveLogger.Info("DataBlocks should be ready to be interacted with, invoking event.");
			foreach (Type item in _typesToInitOnDataBlocksReady)
			{
				IInitializable initializable = null;
				try
				{
					InitInitializables(item, out initializable);
				}
				catch (Exception ex2)
				{
					ArchiveLogger.Error("Trying to Init \"" + item.FullName + "\" threw an exception:");
					ArchiveLogger.Exception(ex2);
				}
				try
				{
					InjectInstanceIntoModules(initializable);
				}
				catch (Exception ex3)
				{
					ArchiveLogger.Error("Trying to Inject \"" + item.FullName + "\" threw an exception:");
					ArchiveLogger.Exception(ex3);
				}
			}
			InitSingletonBase<FeatureManager>.Instance.OnDatablocksReady();
			CustomSettingsManager.OnGameDataInited();
			ArchiveMod.DataBlocksReady?.Invoke();
		}

		internal static void InvokeGameStateChanged(int eGameState_state)
		{
			CurrentGameState = eGameState_state;
			ArchiveMod.GameStateChanged?.Invoke(eGameState_state);
		}

		internal static void InvokeApplicationFocusChanged(bool focus)
		{
			ArchiveMod.ApplicationFocusStateChanged?.Invoke(focus);
		}

		public static void InjectInstanceIntoModules(object instance)
		{
			if (instance == null)
			{
				return;
			}
			foreach (IArchiveModule module in Modules)
			{
				InjectInstanceIntoModules(instance, module);
			}
		}

		private static void InjectInstanceIntoModules(object instance, IArchiveModule module)
		{
			foreach (PropertyInfo item in from p in module.GetType().GetProperties()
				where p.SetMethod != null && !p.SetMethod.IsStatic && p.PropertyType.IsAssignableFrom(instance.GetType())
				select p)
			{
				item.SetValue(module, instance);
			}
		}

		private static void InitInitializables(Type type, out IInitializable initializable)
		{
			ArchiveLogger.Debug("Creating instance of: \"" + type.FullName + "\".");
			IInitializable initializable2 = (initializable = (IInitializable)Activator.CreateInstance(type));
			bool flag = true;
			Type type2 = typeof(InitSingletonBase<>).MakeGenericType(type);
			bool flag2 = type2.IsAssignableFrom(type);
			if (flag2)
			{
				type2.GetProperty("Instance", BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic).SetValue(null, initializable2);
			}
			if (typeof(IInjectLogger).IsAssignableFrom(type))
			{
				((IInjectLogger)initializable2).Logger = LoaderWrapper.CreateArSubLoggerInstance(type.Name, ConsoleColor.Green);
			}
			if (typeof(IInitCondition).IsAssignableFrom(type))
			{
				IInitCondition initCondition = (IInitCondition)initializable2;
				try
				{
					flag = initCondition.InitCondition();
				}
				catch (Exception ex)
				{
					ArchiveLogger.Error("InitCondition method on Type \"" + type.FullName + "\" failed!");
					ArchiveLogger.Warning("This IInitializable won't be initialized.");
					ArchiveLogger.Exception(ex);
					flag = false;
				}
			}
			if (!flag)
			{
				return;
			}
			try
			{
				initializable2.Init();
				if (flag2)
				{
					type2.GetProperty("HasBeenInitialized", BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic).SetValue(null, true);
				}
			}
			catch (Exception ex2)
			{
				ArchiveLogger.Error("Init method on Type \"" + type.FullName + "\" failed!");
				ArchiveLogger.Exception(ex2);
			}
		}

		private static void InspectType(Type type, IArchiveModule module)
		{
			if (typeof(Feature).IsAssignableFrom(type) && type != typeof(Feature))
			{
				InitSingletonBase<FeatureManager>.Instance.InitFeature(type, module);
			}
			else if (typeof(IInitImmediately).IsAssignableFrom(type) && !type.IsInterface && !type.IsAbstract)
			{
				IInitializable initializable = null;
				try
				{
					InitInitializables(type, out initializable);
				}
				catch (Exception ex)
				{
					ArchiveLogger.Error("Trying to Init \"" + type.FullName + "\" (immediately) threw an exception:");
					ArchiveLogger.Exception(ex);
				}
				if (initializable != null)
				{
					_iinitializablesToInjectOnGameDataInit.Add(initializable);
				}
			}
			else if (typeof(IInitAfterGameDataInitialized).IsAssignableFrom(type) && !type.IsInterface && !type.IsAbstract)
			{
				_typesToInitOnGameDataInit.Add(type);
			}
			else if (typeof(IInitAfterDataBlocksReady).IsAssignableFrom(type) && !type.IsInterface && !type.IsAbstract)
			{
				_typesToInitOnDataBlocksReady.Add(type);
			}
		}

		internal static IArchiveModule CreateAndInitModule(Type moduleType)
		{
			if (moduleType == null)
			{
				throw new ArgumentException("Parameter moduleType can not be null!");
			}
			Stopwatch stopwatch = new Stopwatch();
			stopwatch.Start();
			_moduleTypes.Add(moduleType);
			ArchiveLogger.Info("Initializing module \"" + moduleType.FullName + "\" ...");
			IArchiveModule archiveModule = (IArchiveModule)Activator.CreateInstance(moduleType);
			if (string.IsNullOrWhiteSpace(archiveModule.ModuleGroup))
			{
				throw new Exception($"ArchiveModule: {archiveModule.GetType().FullName}, {"ModuleGroup"} can not be null!");
			}
			FeatureGroups.GetOrCreateModuleGroup(archiveModule.ModuleGroup);
			Type[] types = moduleType.Assembly.GetTypes();
			for (int i = 0; i < types.Length; i++)
			{
				InspectType(types[i], archiveModule);
			}
			_moduleAssemblies.Add(moduleType.Assembly);
			if (archiveModule.UsesLegacyPatches)
			{
				archiveModule.Patcher = new ArchiveLegacyPatcher(_harmonyInstance, moduleType.Assembly.GetName().Name + "_" + moduleType.FullName + "_ArchivePatcher");
			}
			try
			{
				archiveModule.Init();
				if (archiveModule.ApplyHarmonyPatches)
				{
					ArchiveLogger.Warning("Applying regular Harmony patches on module \"" + moduleType.FullName + "\" ...");
					_harmonyInstance.PatchAll(moduleType.Assembly);
				}
			}
			catch (Exception ex)
			{
				ArchiveLogger.Error("Error while trying to init \"" + moduleType.FullName + "\"!");
				ArchiveLogger.Exception(ex);
			}
			Modules.Add(archiveModule);
			stopwatch.Stop();
			ArchiveLogger.Debug($"Creation of \"{moduleType.FullName}\" took {stopwatch.Elapsed:ss\\.fff} seconds.");
			return archiveModule;
		}

		private static void ApplyPatches(Utils.RundownID rundownID)
		{
			if (rundownID == Utils.RundownID.RundownUnitialized)
			{
				return;
			}
			foreach (IArchiveModule module in Modules)
			{
				module.Patcher?.PatchRundownSpecificMethods(module.GetType().Assembly);
			}
		}

		internal static void UnpatchAll()
		{
			foreach (IArchiveModule module in Modules)
			{
				UnpatchModule(module);
			}
		}

		public static void UnpatchModule(Type moduleType)
		{
			if (!_moduleTypes.Contains(moduleType))
			{
				throw new ArgumentException("Can't unpatch non patched module \"" + moduleType.FullName + "\".");
			}
			foreach (IArchiveModule module in Modules)
			{
				if (module.GetType() == moduleType)
				{
					UnpatchModule(module);
					return;
				}
			}
			throw new ArgumentException("Can't unpatch module \"" + moduleType.FullName + "\", module not found.");
		}

		public static void UnpatchModule(IArchiveModule module)
		{
			try
			{
				module.Patcher?.Unpatch();
				module.OnExit();
			}
			catch (Exception ex)
			{
				ArchiveLogger.Error($"Error while trying to unpatch and/or run {"OnExit"} in module \"{module?.GetType()?.FullName ?? "Unknown"}\"!");
				ArchiveLogger.Exception(ex);
			}
			Modules.Remove(module);
			_moduleTypes.Remove(module.GetType());
		}

		public static void OnSceneWasLoaded(int buildIndex, string sceneName)
		{
			foreach (IArchiveModule module in Modules)
			{
				try
				{
					module?.OnSceneWasLoaded(buildIndex, sceneName);
				}
				catch (Exception ex)
				{
					ArchiveLogger.Error($"Error while trying to run {"OnSceneWasLoaded"} in module \"{module?.GetType()?.FullName ?? "Unknown"}\"!");
					ArchiveLogger.Exception(ex);
				}
			}
		}

		public static void OnUpdate()
		{
			InitSingletonBase<FeatureManager>.Instance.OnUpdate();
		}

		public static void OnLateUpdate()
		{
			foreach (IArchiveModule module in Modules)
			{
				try
				{
					module?.OnLateUpdate();
				}
				catch (Exception ex)
				{
					ArchiveLogger.Error($"Error while trying to run {"OnLateUpdate"} in module \"{module?.GetType()?.FullName ?? "Unknown"}\"!");
					ArchiveLogger.Exception(ex);
				}
			}
			InitSingletonBase<FeatureManager>.Instance.OnLateUpdate();
		}

		private static Assembly LoadMainArchiveModule(bool isIl2Cpp)
		{
			try
			{
				if (isIl2Cpp)
				{
					ArchiveLogger.Notice("Loading IL2CPP module ...");
					byte[] array = Utils.LoadFromResource("TheArchive.Resources.TheArchive.IL2CPP.dll");
					if (array.Length < 100)
					{
						throw new BadImageFormatException("IL2CPP Module is too small, this version might not contain the module build but a dummy dll!");
					}
					return Assembly.Load(array);
				}
				ArchiveLogger.Notice("Loading MONO module ...");
				byte[] array2 = Utils.LoadFromResource("TheArchive.Resources.TheArchive.MONO.dll");
				if (array2.Length < 100)
				{
					throw new BadImageFormatException("MONO Module is too small, this version might not contain the module build but a dummy dll!");
				}
				return Assembly.Load(array2);
			}
			catch (Exception ex)
			{
				ArchiveLogger.Error($"Could not load {(isIl2Cpp ? "IL2CPP" : "MONO")} module! {ex}: {ex.Message}");
				ArchiveLogger.Error(ex.StackTrace ?? "");
				return null;
			}
		}
	}
}
namespace TheArchive.Utilities
{
	public static class AccessorExtensions
	{
		public static IValueAccessor<T, MT> OrAlternative<T, MT>(this IValueAccessor<T, MT> self, Func<IValueAccessor<T, MT>> func)
		{
			if (self != null && self.HasMember)
			{
				return self;
			}
			if (func == null)
			{
				throw new NullReferenceException("Parameter func may not be null!");
			}
			return func();
		}
	}
	public interface IValueAccessor<T, MT>
	{
		bool CanGet { get; }

		bool CanSet { get; }

		bool HasMember { get; }

		MT Get(T instance);

		void Set(T instance, MT value);
	}
	public interface IStaticValueAccessor<T, MT> : IValueAccessor<T, MT>
	{
		bool IsStatic { get; }

		MT GetStaticValue();

		void SetStaticValue(MT value);
	}
	public abstract class AccessorBase
	{
		protected static readonly Dictionary<string, AccessorBase> Accessors = new Dictionary<string, AccessorBase>();

		public static object[] NoParams { get; private set; } = Array.Empty<object>();


		public static BindingFlags AnyBindingFlags => BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic;

		public string Identifier { get; private set; }

		public bool IgnoreErrors { get; private set; }

		public abstract bool HasMemberBeenFound { get; }

		protected AccessorBase(string identifier, bool ignoreErrors)
		{
			Identifier = identifier;
			IgnoreErrors = ignoreErrors;
		}

		public static IValueAccessor<T, MT> GetValueAccessor<T, MT>(string memberName, bool throwOnError = false)
		{
			if (LoaderWrapper.IsGameIL2CPP() && LoaderWrapper.IsIL2CPPType(typeof(T)))
			{
				return PropertyAccessor<T, MT>.GetAccessor(memberName, !throwOnError);
			}
			MemberInfo memberInfo = typeof(T).GetMember(memberName, BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic).FirstOrDefault();
			if (!(memberInfo is PropertyInfo))
			{
				if (memberInfo is FieldInfo)
				{
					return FieldAccessor<T, MT>.GetAccessor(memberName, !throwOnError);
				}
				if (throwOnError)
				{
					throw new ArgumentException($"Member with name \"{memberName}\" could not be found in type \"{typeof(T).Name}\" or isn't {"IValueAccessor"} compatible.", "memberName");
				}
				return null;
			}
			return PropertyAccessor<T, MT>.GetAccessor(memberName, !throwOnError);
		}

		public static IStaticValueAccessor<T, MT> GetStaticValueAccessor<T, MT>(string memberName, bool throwOnError = false)
		{
			IValueAccessor<T, MT> valueAccessor = GetValueAccessor<T, MT>(memberName, throwOnError);
			if (valueAccessor != null)
			{
				IStaticValueAccessor<T, MT> staticValueAccessor = valueAccessor as IStaticValueAccessor<T, MT>;
				if (throwOnError && !staticValueAccessor.IsStatic)
				{
					throw new ArgumentException("Member with name \"" + memberName + "\" is not static!", "memberName");
				}
				return staticValueAccessor;
			}
			return null;
		}
	}
	public class FieldAccessor<T, FT> : AccessorBase, IValueAccessor<T, FT>, IStaticValueAccessor<T, FT>
	{
		private readonly FieldInfo _field;

		public override bool HasMemberBeenFound => _field != null;

		public bool CanGet => true;

		public bool CanSet => true;

		public bool HasMember => HasMemberBeenFound;

		public bool IsStatic => _field?.IsStatic ?? false;

		public static FieldAccessor<T, FT> GetAccessor(string fieldName, bool ignoreErrors = false)
		{
			string text = "Field_" + typeof(T).FullName + "_" + fieldName;
			if (AccessorBase.Accessors.TryGetValue(text, out var value))
			{
				return (FieldAccessor<T, FT>)value;
			}
			value = new FieldAccessor<T, FT>(text, fieldName, ignoreErrors);
			AccessorBase.Accessors.Add(text, value);
			return (FieldAccessor<T, FT>)value;
		}

		private FieldAccessor(string identifier, string fieldName, bool ignoreErrors = false)
			: base(identifier, ignoreErrors)
		{
			_field = typeof(T).GetField(fieldName, AccessorBase.AnyBindingFlags);
		}

		public FT Get(T instance)
		{
			try
			{
				return (FT)_field.GetValue(instance);
			}
			catch (NullReferenceException ex)
			{
				if (!base.IgnoreErrors)
				{
					if (!HasMemberBeenFound)
					{
						ArchiveLogger.Warning($"NullReferenceException while getting {"FieldAccessor"} field \"{base.Identifier}\"! If this is intentional consider setting {"IgnoreErrors"} to true.");
						return default(FT);
					}
					ArchiveLogger.Error($"NullReferenceException while getting {"FieldAccessor"} field \"{base.Identifier}\"!");
					throw ex;
				}
			}
			catch (Exception ex2)
			{
				ArchiveLogger.Error($"Exception while getting {"FieldAccessor"} field \"{base.Identifier}\"!");
				ArchiveLogger.Exception(ex2);
			}
			return default(FT);
		}

		public void Set(T instance, FT value)
		{
			try
			{
				_field.SetValue(instance, value);
			}
			catch (NullReferenceException ex)
			{
				if (!base.IgnoreErrors)
				{
					if (HasMemberBeenFound)
					{
						ArchiveLogger.Error($"NullReferenceException while setting {"FieldAccessor"} field \"{base.Identifier}\"!");
						throw ex;
					}
					ArchiveLogger.Warning($"NullReferenceException while setting {"FieldAccessor"} field \"{base.Identifier}\"! If this is intentional consider setting {"IgnoreErrors"} to true.");
				}
			}
			catch (Exception ex2)
			{
				ArchiveLogger.Error($"Exception while setting {"FieldAccessor"} field \"{base.Identifier}\"!");
				ArchiveLogger.Exception(ex2);
			}
		}

		public FT GetStaticValue()
		{
			return Get(default(T));
		}

		public void SetStaticValue(FT value)
		{
			Set(default(T), value);
		}
	}
	public class PropertyAccessor<T, PT> : AccessorBase, IValueAccessor<T, PT>, IStaticValueAccessor<T, PT>
	{
		private readonly PropertyInfo _property;

		public override bool HasMemberBeenFound => _property != null;

		public bool CanGet => _property?.GetGetMethod(nonPublic: true) != null;

		public bool CanSet => _property?.GetSetMethod(nonPublic: true) != null;

		public bool HasMember => HasMemberBeenFound;

		public bool IsStatic => (_property?.GetGetMethod(nonPublic: true) ?? _property?.GetSetMethod(nonPublic: true))?.IsStatic ?? false;

		public static PropertyAccessor<T, PT> GetAccessor(string propertyName, bool ignoreErrors = false)
		{
			string text = "Property_" + typeof(T).FullName + "_" + propertyName;
			if (AccessorBase.Accessors.TryGetValue(text, out var value))
			{
				return (PropertyAccessor<T, PT>)value;
			}
			value = new PropertyAccessor<T, PT>(text, propertyName, ignoreErrors);
			AccessorBase.Accessors.Add(text, value);
			return (PropertyAccessor<T, PT>)value;
		}

		private PropertyAccessor(string identifier, string propertyName, bool ignoreErrors = false)
			: base(identifier, ignoreErrors)
		{
			_property = typeof(T).GetProperty(propertyName, AccessorBase.AnyBindingFlags);
		}

		public PT Get(T instance)
		{
			try
			{
				return (PT)_property.GetValue(instance);
			}
			catch (NullReferenceException ex)
			{
				if (!base.IgnoreErrors)
				{
					if (!HasMemberBeenFound)
					{
						ArchiveLogger.Warning($"NullReferenceException while getting {"PropertyAccessor"} property \"{base.Identifier}\"! If this is intentional consider setting {"IgnoreErrors"} to true.");
						return default(PT);
					}
					ArchiveLogger.Error($"NullReferenceException while getting {"PropertyAccessor"} property \"{base.Identifier}\"!");
					throw ex;
				}
			}
			catch (Exception ex2)
			{
				ArchiveLogger.Error($"Exception while getting {"PropertyAccessor"} property \"{base.Identifier}\"!");
				ArchiveLogger.Exception(ex2);
			}
			return default(PT);
		}

		public void Set(T instance, PT value)
		{
			try
			{
				_property.SetValue(instance, value);
			}
			catch (NullReferenceException ex)
			{
				if (!base.IgnoreErrors)
				{
					if (HasMemberBeenFound)
					{
						ArchiveLogger.Error($"NullReferenceException while setting {"PropertyAccessor"} property \"{base.Identifier}\"!");
						throw ex;
					}
					ArchiveLogger.Warning($"NullReferenceException while setting {"PropertyAccessor"} property \"{base.Identifier}\"! If this is intentional consider setting {"IgnoreErrors"} to true.");
				}
			}
			catch (Exception ex2)
			{
				ArchiveLogger.Error($"Exception while setting {"PropertyAccessor"} property \"{base.Identifier}\"!");
				ArchiveLogger.Exception(ex2);
			}
		}

		public PT GetStaticValue()
		{
			return Get(default(T));
		}

		public void SetStaticValue(PT value)
		{
			Set(default(T), value);
		}
	}
	public class MethodAccessor<T, RT> : AccessorBase
	{
		private readonly MethodInfo _method;

		public bool IsMethodStatic => _method.IsStatic;

		public override bool HasMemberBeenFound => _method != null;

		public static MethodAccessor<T, RT> GetAccessor(string methodName, Type[] parameterTypes = null, bool ignoreErrors = false)
		{
			string text = $"Method_{typeof(T).FullName}_{typeof(RT)}_{methodName}";
			if (parameterTypes != null)
			{
				text = text + "_" + string.Join("_", parameterTypes.Select((Type pt) => pt.Name));
			}
			if (AccessorBase.Accessors.TryGetValue(text, out var value))
			{
				return (MethodAccessor<T, RT>)value;
			}
			value = new MethodAccessor<T, RT>(text, methodName, parameterTypes, ignoreErrors);
			AccessorBase.Accessors.Add(text, value);
			return (MethodAccessor<T, RT>)value;
		}

		private MethodAccessor(string identifier, string methodName, Type[] parameterTypes, bool ignoreErrors = false)
			: base(identifier, ignoreErrors)
		{
			try
			{
				if (parameterTypes == null)
				{
					_method = typeof(T).GetMethod(methodName, AccessorBase.AnyBindingFlags);
				}
				else
				{
					_method = typeof(T).GetMethod(methodName, AccessorBase.AnyBindingFlags, null, parameterTypes, null);
				}
			}
			catch (Exception ex)
			{
				ArchiveLogger.Error($"Method \"{methodName}\" in Type {typeof(T).FullName} could not be resolved on {ex.Source}!");
				ArchiveLogger.Exception(ex);
			}
		}

		public RT Invoke(T instance, params object[] parameters)
		{
			try
			{
				object obj = _method.Invoke(instance, parameters ?? AccessorBase.NoParams);
				if (obj == null)
				{
					return default(RT);
				}
				return (RT)obj;
			}
			catch (NullReferenceException ex)
			{
				if (!base.IgnoreErrors)
				{
					if (!HasMemberBeenFound)
					{
						ArchiveLogger.Warning($"NullReferenceException while calling {"MethodAccessor"} method \"{base.Identifier}\"! If this is intentional consider setting {"IgnoreErrors"} to true.");
						return default(RT);
					}
					ArchiveLogger.Error($"NullReferenceException while calling {"MethodAccessor"} method \"{base.Identifier}\"!");
					throw ex;
				}
			}
			catch (Exception ex2)
			{
				ArchiveLogger.Error($"Exception while calling {"MethodAccessor"} method \"{base.Identifier}\"!");
				ArchiveLogger.Exception(ex2);
			}
			return default(RT);
		}

		public RT Invoke(T instance)
		{
			return Invoke(instance, null);
		}
	}
	public class MethodAccessor<T> : AccessorBase
	{
		private readonly MethodInfo _method;

		public bool IsMethodStatic => _method.IsStatic;

		public override bool HasMemberBeenFound => _method != null;

		public int ParameterCount { get; private set; }

		public static MethodAccessor<T> GetAccessor(string methodName, Type[] parameterTypes = null, bool ignoreErrors = false)
		{
			string text = "Method_" + typeof(T).FullName + "_void_" + methodName;
			if (parameterTypes != null && parameterTypes != Array.Empty<Type>())
			{
				text = text + "_" + string.Join("_", parameterTypes.Select((Type pt) => pt.Name));
			}
			if (AccessorBase.Accessors.TryGetValue(text, out var value))
			{
				return (MethodAccessor<T>)value;
			}
			value = new MethodAccessor<T>(text, methodName, parameterTypes, ignoreErrors);
			AccessorBase.Accessors.Add(text, value);
			return (MethodAccessor<T>)value;
		}

		private MethodAccessor(string identifier, string methodName, Type[] parameterTypes, bool ignoreErrors = false)
			: base(identifier, ignoreErrors)
		{
			try
			{
				if (parameterTypes == null)
				{
					_method = typeof(T).GetMethod(methodName, AccessorBase.AnyBindingFlags);
				}
				else
				{
					_method = typeof(T).GetMethod(methodName, AccessorBase.AnyBindingFlags, null, parameterTypes, null);
				}
				if (!ignoreErrors && _method == null)
				{
					throw new Exception("Method not found!");
				}
				if (_method != null)
				{
					ParameterCount = _method.GetParameters().Length;
				}
			}
			catch (Exception ex)
			{
				ArchiveLogger.Error($"Method \"{methodName}\" in Type {typeof(T).FullName} could not be resolved on {ex.Source}!");
				ArchiveLogger.Exception(ex);
				ArchiveLogger.Debug($"Constructor debug data:\nidentifier:{identifier}\nmethodName:{methodName}\nparameterTypes:{string.Join(", ", parameterTypes.Select((Type p) => p.FullName))}");
				StackFrame frame = new StackTrace().GetFrame(2);
				ArchiveLogger.Debug($"FileName:{frame.GetFileName()} {frame.GetFileLineNumber()}\nMethod:{frame.GetMethod().DeclaringType.FullName}:{frame.GetMethod().Name}");
				PrintDebug();
			}
		}

		private void PrintDebug()
		{
			if (!(_method == null))
			{
				ArchiveLogger.Debug($"Method debug data:\nName:{_method.Name}\nDeclaringType:{_method.DeclaringType.FullName}\nReturnType:{_method.ReturnType}\nParameter Count:{_method.GetParameters()?.Count() ?? 0}\nParameters:{string.Join(", ", from p in _method.GetParameters()
					select p.ParameterType.FullName)}");
			}
		}

		public void Invoke(T instance, params object[] parameters)
		{
			try
			{
				if (parameters != null && parameters.Length > ParameterCount)
				{
					parameters = parameters.Take(ParameterCount).ToArray();
				}
				_method.Invoke(instance, parameters ?? AccessorBase.NoParams);
			}
			catch (NullReferenceException ex)
			{
				if (!base.IgnoreErrors)
				{
					if (HasMemberBeenFound)
					{
						ArchiveLogger.Error($"NullReferenceException while calling {"MethodAccessor"} method \"{base.Identifier}\"!");
						throw ex;
					}
					ArchiveLogger.Warning($"NullReferenceException while calling {"MethodAccessor"} method \"{base.Identifier}\"! If this is intentional consider setting {"IgnoreErrors"} to true.");
				}
			}
			catch (Exception ex2)
			{
				ArchiveLogger.Error($"Exception while calling {"MethodAccessor"} method \"{base.Identifier}\"!");
				ArchiveLogger.Exception(ex2);
				PrintDebug();
			}
		}

		public void Invoke(T instance)
		{
			Invoke(instance, null);
		}
	}
	public class ArchiveLogger
	{
		internal static IArchiveLogger logger;

		public static void Success(string msg)
		{
			logger.Msg(ConsoleColor.Green, msg);
		}

		public static void Notice(string msg)
		{
			logger.Msg(ConsoleColor.Cyan, msg);
		}

		public static void Msg(ConsoleColor col, string msg)
		{
			logger.Msg(col, msg);
		}

		public static void Debug(string msg)
		{
			logger.Msg(ConsoleColor.DarkGray, msg);
		}

		public static void Info(string msg)
		{
			logger.Info(msg);
		}

		public static void Warning(string msg)
		{
			logger.Msg(ConsoleColor.DarkYellow, msg);
		}

		public static void Error(string msg)
		{
			logger.Error(msg);
		}

		public static void Msg(string v)
		{
			Info(v);
		}

		public static void Exception(Exception ex)
		{
			Error($"{ex}: {ex.Message}\n{ex.StackTrace}");
		}
	}
	public static class GTFOLogger
	{
		private static readonly HashSet<string> _ignoreListExact = new HashSet<string> { "show crosshair", "Setting and getting Body Position/Rotation, IK Goals, Lookat and BoneLocalRotation should only be done in OnAnimatorIK or OnStateIK" };

		private static readonly HashSet<string> _ignoreListStartsWith = new HashSet<string> { "Wielding new item in slot", "Backend.", "BE.OnGameEvent" };

		internal static IArchiveLogger Logger { private get; set; }

		public static void Ignore(string str)
		{
			_ignoreListExact.Add(str);
		}

		public static void Log(string message)
		{
			if (!_ignoreListExact.Contains(message) && !_ignoreListStartsWith.Any((string s) => message.StartsWith(s)))
			{
				Logger.Info(message);
			}
		}

		public static void Warn(string message)
		{
			if (!_ignoreListExact.Contains(message) && !_ignoreListStartsWith.Any((string s) => message.StartsWith(s)))
			{
				Logger.Warning(message);
			}
		}

		public static void Error(string message)
		{
			if (!_ignoreListExact.Contains(message) && !_ignoreListStartsWith.Any((string s) => message.StartsWith(s)))
			{
				Logger.Error(message);
			}
		}
	}
	public static class ImplementationManagerExtensions
	{
		public static void RegisterForIdentifier(this Type type, string identifier)
		{
			ImplementationManager.RegisterGameType(identifier, type);
		}

		public static void RegisterSelf<T>(this T type) where T : Type
		{
			if (type.IsGenericTypeDefinition)
			{
				type.RegisterForIdentifier(type.Name.Split('`')[0] + "<" + ((type.GenericTypeArguments.Length > 1) ? string.Join(",", new string[type.GenericTypeArguments.Length - 1]) : string.Empty) + ">");
			}
			else
			{
				type.RegisterForIdentifier(type.Name);
			}
		}
	}
	public class LocalFiles
	{
		public const string GTFO_SETTINGS_JSON = "GTFO_Settings.json";

		public const string GTFO_FAVORITES_JSON = "GTFO_Favorites.json";

		public const string GTFO_BOT_FAVORITES_JSON = "GTFO_BotFavorites.json";

		private static string _modLocalLowPath;

		private static string _modDefaultGameLogsAndCachePath;

		private static string _modDefaultSaveDataPath;

		private static string _dataBlockDumpPath;

		private static string _savePath;

		private static string _gameLogsAndCacheSavePath;

		private static string _versionSpecificLogsAndCachePath;

		private static string _versionSpecificSavePath;

		private static string _otherConfigsPath;

		private static string _featureConfigsPath;

		private static string _filesPath;

		private static string _settingsPath;

		private static string _favoritesPath;

		private static string _botFavoritesPath;

		private static string _boostersPath;

		private static string _vanityItemsPath;

		private static string _vanityItemsLayerDropsPath;

		private static string _progressionPath;

		public static string ModLocalLowPath
		{
			get
			{
				if (_modLocalLowPath == null)
				{
					_modLocalLowPath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.UserProfile), "AppData", "LocalLow", "GTFO_TheArchive");
					if (!Directory.Exists(_modLocalLowPath))
					{
						Directory.CreateDirectory(_modLocalLowPath);
					}
				}
				return _modLocalLowPath;
			}
		}

		public static string ModDefaultGameLogsAndCachePath
		{
			get
			{
				if (_modDefaultGameLogsAndCachePath == null)
				{
					_modDefaultGameLogsAndCachePath = Path.Combine(ModLocalLowPath, "GameLogsAndCache");
					if (!Directory.Exists(_modDefaultGameLogsAndCachePath))
					{
						Directory.CreateDirectory(_modDefaultGameLogsAndCachePath);
					}
				}
				return _modDefaultGameLogsAndCachePath;
			}
		}

		public static string ModDefaultSaveDataPath
		{
			get
			{
				if (_modDefaultSaveDataPath == null)
				{
					_modDefaultSaveDataPath = Path.Combine(ModLocalLowPath, "SaveData");
					if (!Directory.Exists(_modLocalLowPath))
					{
						Directory.CreateDirectory(_modLocalLowPath);
					}
				}
				return _modDefaultSaveDataPath;
			}
		}

		public static string DataBlockDumpPath
		{
			get
			{
				if (string.IsNullOrEmpty(_dataBlockDumpPath))
				{
					_dataBlockDumpPath = Path.Combine(SaveDirectoryPath, "DataBlocks", $"Build_{BuildDB.BuildNumber}_{ArchiveMod.CurrentRundown}");
					if (!Directory.Exists(_dataBlockDumpPath))
					{
						Directory.CreateDirectory(_dataBlockDumpPath);
					}
				}
				return _dataBlockDumpPath;
			}
		}

		public static string SaveDirectoryPath
		{
			get
			{
				if (string.IsNullOrEmpty(_savePath))
				{
					_savePath = (string.IsNullOrWhiteSpace(ArchiveMod.Settings.CustomFileSaveLocation) ? ModDefaultSaveDataPath : ArchiveMod.Settings.CustomFileSaveLocation);
					if (!Directory.Exists(_savePath))
					{
						Directory.CreateDirectory(_savePath);
					}
				}
				return _savePath;
			}
		}

		public static string GameLogsAndCachePath
		{
			get
			{
				if (_gameLogsAndCacheSavePath == null)
				{
					_gameLogsAndCacheSavePath = (string.IsNullOrWhiteSpace(ArchiveMod.Settings.CustomLogsAndCacheLocation) ? ModDefaultGameLogsAndCachePath : ArchiveMod.Settings.CustomLogsAndCacheLocation);
					if (!Directory.Exists(_gameLogsAndCacheSavePath))
					{
						Directory.CreateDirectory(_gameLogsAndCacheSavePath);
					}
				}
				return _gameLogsAndCacheSavePath;
			}
		}

		public static string VersionSpecificLogsAndCachePath
		{
			get
			{
				if (string.IsNullOrEmpty(_versionSpecificLogsAndCachePath))
				{
					_versionSpecificLogsAndCachePath = Path.Combine(GameLogsAndCachePath, $"{((int)ArchiveMod.CurrentRundown).ToString().PadLeft(2, '0')}_{ArchiveMod.CurrentRundown}_Data", "appdata");
					if (!Directory.Exists(_versionSpecificLogsAndCachePath))
					{
						Directory.CreateDirectory(_versionSpecificLogsAndCachePath);
					}
				}
				return _versionSpecificLogsAndCachePath;
			}
		}

		public static string VersionSpecificSaveDirectoryPath
		{
			get
			{
				if (string.IsNullOrEmpty(_versionSpecificSavePath))
				{
					if (LoaderWrapper.IsModInstalled("com.dak.MTFO"))
					{
						_versionSpecificSavePath = Path.Combine(SaveDirectoryPath, "Modded", "TODO_USE_MTFO_FOLDER_HERE_Data");
					}
					else
					{
						_versionSpecificSavePath = GetVersionSpecificSaveDirectoryPath(ArchiveMod.CurrentRundown);
					}
					if (!Directory.Exists(_versionSpecificSavePath))
					{
						Directory.CreateDirectory(_versionSpecificSavePath);
						try
						{
							if (!CopyMostRecentSaveFiles(ArchiveMod.CurrentRundown - 1, ArchiveMod.CurrentRundown))
							{
								ArchiveLogger.Notice("Creating new game settings file(s)!");
							}
						}
						catch (Exception ex)
						{
							ArchiveLogger.Warning($"Caught an exception while trying to copy over older settings files: {ex}: {ex.Message}");
							ArchiveLogger.Debug(ex.StackTrace);
						}
					}
				}
				return _versionSpecificSavePath;
			}
		}

		public static string OtherConfigsDirectoryPath
		{
			get
			{
				if (string.IsNullOrEmpty(_otherConfigsPath))
				{
					_otherConfigsPath = Path.Combine(SaveDirectoryPath, "OtherConfigs");
					if (!Directory.Exists(_otherConfigsPath))
					{
						Directory.CreateDirectory(_otherConfigsPath);
					}
				}
				return _otherConfigsPath;
			}
		}

		public static string FeatureConfigsDirectoryPath
		{
			get
			{
				if (string.IsNullOrEmpty(_featureConfigsPath))
				{
					_featureConfigsPath = Path.Combine(SaveDirectoryPath, "FeatureSettings");
					if (!Directory.Exists(_featureConfigsPath))
					{
						Directory.CreateDirectory(_featureConfigsPath);
					}
				}
				return _featureConfigsPath;
			}
		}

		public static string FilesDirectoryPath
		{
			get
			{
				if (string.IsNullOrEmpty(_filesPath))
				{
					_filesPath = Path.Combine(VersionSpecificSaveDirectoryPath, "Files");
					if (!Directory.Exists(_filesPath))
					{
						Directory.CreateDirectory(_filesPath);
					}
				}
				return _filesPath;
			}
		}

		public static string SettingsPath
		{
			get
			{
				if (string.IsNullOrEmpty(_settingsPath))
				{
					_settingsPath = Path.Combine(VersionSpecificSaveDirectoryPath, "GTFO_Settings.json");
				}
				return _settingsPath;
			}
		}

		public static string FavoritesPath
		{
			get
			{
				if (string.IsNullOrEmpty(_favoritesPath))
				{
					_favoritesPath = Path.Combine(VersionSpecificSaveDirectoryPath, "GTFO_Favorites.json");
				}
				return _favoritesPath;
			}
		}

		public static string BotFavoritesPath
		{
			get
			{
				if (string.IsNullOrEmpty(_botFavoritesPath))
				{
					_botFavoritesPath = Path.Combine(VersionSpecificSaveDirectoryPath, "GTFO_BotFavorites.json");
				}
				return _botFavoritesPath;
			}
		}

		public static string BoostersPath
		{
			get
			{
				if (string.IsNullOrEmpty(_boostersPath))
				{
					_boostersPath = Path.Combine(VersionSpecificSaveDirectoryPath, "Booster_Data.json");
				}
				return _boostersPath;
			}
		}

		public static string VanityItemsPath
		{
			get
			{
				if (string.IsNullOrEmpty(_vanityItemsPath))
				{
					_vanityItemsPath = Path.Combine(VersionSpecificSaveDirectoryPath, "VanityItems_Data.json");
				}
				return _vanityItemsPath;
			}
		}

		public static string VanityItemsLayerDropsPath
		{
			get
			{
				if (string.IsNullOrEmpty(_vanityItemsLayerDropsPath))
				{
					_vanityItemsLayerDropsPath = Path.Combine(VersionSpecificSaveDirectoryPath, "VanityItemsLayerDrops_Data.json");
				}
				return _vanityItemsLayerDropsPath;
			}
		}

		private static string LocalProgressionBasePathNoExtension
		{
			get
			{
				if (string.IsNullOrEmpty(_progressionPath))
				{
					_progressionPath = Path.Combine(VersionSpecificSaveDirectoryPath, "RundownProgression_Data");
				}
				return _progressionPath;
			}
		}

		internal static bool CopyFromBaseGameLocation(Utils.RundownID copyTo)
		{
			string path = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.UserProfile), "AppData", "LocalLow", "10 Chambers Collective", "GTFO");
			string text = Path.Combine(path, "GTFO_Settings.txt");
			string text2 = Path.Combine(path, "GTFO_Favorites.txt");
			string text3 = Path.Combine(path, "GTFO_BotFavorites.txt");
			if (File.Exists(text))
			{
				string settingsPath = GetSettingsPath(copyTo);
				ArchiveLogger.Debug($"Copying vanilla game settings file! (\"{text}\" -> \"{settingsPath}\")");
				File.Copy(text, settingsPath);
				string favoritesPath = GetFavoritesPath(copyTo);
				string botFavoritesPath = GetBotFavoritesPath(copyTo);
				if (File.Exists(text2))
				{
					ArchiveLogger.Debug($"Copying vanilla game favorites file! (\"{text2}\" -> \"{favoritesPath}\")");
					File.Copy(text2, favoritesPath);
				}
				if (File.Exists(text3))
				{
					ArchiveLogger.Debug($"Copying vanilla game bot favorites file! (\"{text3}\" -> \"{botFavoritesPath}\")");
					File.Copy(text3, botFavoritesPath);
				}
				return true;
			}
			return false;
		}

		internal static bool CopyMostRecentSaveFiles(Utils.RundownID copyFrom, Utils.RundownID copyTo, int maxStep = 3)
		{
			if (copyFrom < Utils.RundownID.RundownOne)
			{
				return CopyFromBaseGameLocation(copyTo);
			}
			string settingsPath = GetSettingsPath(copyFrom);
			if (!File.Exists(settingsPath))
			{
				if (maxStep <= 1)
				{
					return CopyFromBaseGameLocation(copyTo);
				}
				return CopyMostRecentSaveFiles(copyFrom - 1, copyTo, maxStep - 1);
			}
			string settingsPath2 = GetSettingsPath(copyTo);
			ArchiveLogger.Debug($"Copying most recent settings file! (\"{settingsPath}\" -> \"{settingsPath2}\")");
			File.Copy(settingsPath, settingsPath2);
			if (!ArchiveMod.IsPlayingModded)
			{
				string favoritesPath = GetFavoritesPath(copyTo);
				string favoritesPath2 = GetFavoritesPath(copyFrom);
				if (File.Exists(favoritesPath2))
				{
					ArchiveLogger.Debug($"Copying most recent favorites file! (\"{favoritesPath2}\" -> \"{favoritesPath}\")");
					File.Copy(favoritesPath2, favoritesPath);
				}
				string botFavoritesPath = GetBotFavoritesPath(copyTo);
				string botFavoritesPath2 = GetBotFavoritesPath(copyFrom);
				if (File.Exists(botFavoritesPath2))
				{
					ArchiveLogger.Debug($"Copying most recent bot favorites file! (\"{botFavoritesPath2}\" -> \"{botFavoritesPath}\")");
					File.Copy(botFavoritesPath2, botFavoritesPath);
				}
			}
			return true;
		}

		public static string GetVersionSpecificSaveDirectoryPath(Utils.RundownID rundown)
		{
			string saveDirectoryPath = SaveDirectoryPath;
			DefaultInterpolatedStringHandler defaultInterpolatedStringHandler = new DefaultInterpolatedStringHandler(6, 2);
			int num = (int)rundown;
			defaultInterpolatedStringHandler.AppendFormatted(num.ToString().PadLeft(2, '0'));
			defaultInterpolatedStringHandler.AppendLiteral("_");
			defaultInterpolatedStringHandler.AppendFormatted(rundown);
			defaultInterpolatedStringHandler.AppendLiteral("_Data");
			return Path.Combine(saveDirectoryPath, defaultInterpolatedStringHandler.ToStringAndClear());
		}

		public static string GetSettingsPath(Utils.RundownID rundown)
		{
			return Path.Combine(GetVersionSpecificSaveDirectoryPath(rundown), "GTFO_Settings.json");
		}

		public static string GetFavoritesPath(Utils.RundownID rundown)
		{
			return Path.Combine(GetVersionSpecificSaveDirectoryPath(rundown), "GTFO_Favorites.json");
		}

		public static string GetBotFavoritesPath(Utils.RundownID rundown)
		{
			return Path.Combine(GetVersionSpecificSaveDirectoryPath(rundown), "GTFO_BotFavorites.json");
		}

		public static string GetLocalProgressionPathForKey(string rundownKey)
		{
			return LocalProgressionBasePathNoExtension + "_" + rundownKey + ".json";
		}

		public static void SaveToFilesDir(string filename, string jsonOrSomething)
		{
			string text = Path.Combine(FilesDirectoryPath, filename);
			if (string.IsNullOrEmpty(jsonOrSomething))
			{
				ArchiveLogger.Msg(ConsoleColor.DarkRed, "Saving \"" + filename + "\" to disk failed because the data is null or empty! Path: " + text);
				return;
			}
			ArchiveLogger.Msg(ConsoleColor.Blue, "Saving \"" + filename + "\" to disk at: " + text);
			File.WriteAllText(text, jsonOrSomething);
		}

		public static string LoadFromFilesDir(string filename, bool isJson = true)
		{
			string text = Path.Combine(FilesDirectoryPath, filename);
			ArchiveLogger.Msg(ConsoleColor.Green, "Loading \"" + filename + "\" from disk at: " + text);
			if (!File.Exists(text))
			{
				if (!isJson)
				{
					return string.Empty;
				}
				return "{}";
			}
			return File.ReadAllText(text);
		}

		public static T LoadConfig<T>(out bool fileExists, bool saveIfNonExistent = true) where T : new()
		{
			string path = Path.Combine(OtherConfigsDirectoryPath, typeof(T).Name + ".json");
			if (!File.Exists(path))
			{
				T val = new T();
				if (saveIfNonExistent)
				{
					SaveConfig(val);
				}
				fileExists = false;
				return val;
			}
			fileExists = true;
			return JsonConvert.DeserializeObject<T>(File.ReadAllText(path), ArchiveMod.JsonSerializerSettings);
		}

		public static T LoadConfig<T>(bool saveIfNonExistent = true) where T : new()
		{
			bool fileExists;
			return LoadConfig<T>(out fileExists, saveIfNonExistent);
		}

		public static void SaveConfig<T>(T config)
		{
			File.WriteAllText(Path.Combine(OtherConfigsDirectoryPath, typeof(T).Name + ".json"), JsonConvert.SerializeObject((object)config, ArchiveMod.JsonSerializerSettings));
		}

		internal static object LoadFeatureConfig(string moduleIdentifier, string featureIdentifier, Type configType, out bool fileExists, bool saveIfNonExistent = true)
		{
			if (string.IsNullOrWhiteSpace(featureIdentifier))
			{
				throw new ArgumentException("Parameter featureIdentifier may not be null or whitespace.");
			}
			if (configType == null)
			{
				throw new ArgumentNullException("Parameter configType may not be null.");
			}
			string text = Path.Combine(FeatureConfigsDirectoryPath, moduleIdentifier);
			if (!Directory.Exists(text))
			{
				Directory.CreateDirectory(text);
			}
			string path = Path.Combine(text, featureIdentifier + "_" + configType.Name + ".json");
			if (!File.Exists(path))
			{
				object obj = Activator.CreateInstance(configType);
				if (saveIfNonExistent)
				{
					SaveFeatureConfig(moduleIdentifier, featureIdentifier, configType, obj);
				}
				fileExists = false;
				return obj;
			}
			fileExists = true;
			try
			{
				return JsonConvert.DeserializeObject(File.ReadAllText(path), configType, ArchiveMod.JsonSerializerSettings);
			}
			catch
			{
				object obj2 = Activator.CreateInstance(configType);
				if (saveIfNonExistent)
				{
					SaveFeatureConfig(moduleIdentifier, featureIdentifier, configType, obj2);
				}
				fileExists = false;
				return obj2;
			}
		}

		internal static FeatureLocalizationData LoadFeatureLocalizationText(Feature feature)
		{
			string text = Path.Combine(Path.GetDirectoryName((feature.ModuleGroup == "Archive Core") ? ArchiveMod.CORE_PATH : feature.FeatureInternal.OriginAssembly.Location), "Localization");
			if (!Directory.Exists(text))
			{
				Directory.CreateDirectory(text);
			}
			string path = Path.Combine(text, feature.Identifier + "_Localization.json");
			if (!File.Exists(path))
			{
				FeatureLocalizationData featureLocalizationData = FeatureInternal.GenerateFeatureLocalization(feature);
				File.WriteAllText(path, JsonConvert.SerializeObject((object)featureLocalizationData, ArchiveMod.JsonSerializerSettings));
				return featureLocalizationData;
			}
			FeatureLocalizationData featureLocalizationData2 = JsonConvert.DeserializeObject<FeatureLocalizationData>(File.ReadAllText(path), ArchiveMod.JsonSerializerSettings);
			string input = JsonConvert.SerializeObject((object)featureLocalizationData2, ArchiveMod.JsonSerializerSettings);
			FeatureLocalizationData featureLocalizationData3 = FeatureInternal.GenerateFeatureLocalization(feature, featureLocalizationData2);
			string text2 = JsonConvert.SerializeObject((object)featureLocalizationData3, ArchiveMod.JsonSerializerSettings);
			if (text2.HashString() != input.HashString())
			{
				File.WriteAllText(path, text2);
			}
			return featureLocalizationData3;
		}

		internal static object LoadFeatureConfig(string moduleIdentifier, string featureIdentifier, Type configType, bool saveIfNonExistent = true)
		{
			bool fileExists;
			return LoadFeatureConfig(moduleIdentifier, featureIdentifier, configType, out fileExists, saveIfNonExistent);
		}

		internal static void SaveFeatureConfig(string moduleIdentifier, string featureIdentifier, Type configType, object configInstance)
		{
			if (string.IsNullOrWhiteSpace(featureIdentifier))
			{
				throw new ArgumentException("Parameter featureIdentifier may not be null or whitespace.");
			}
			if (configType == null)
			{
				throw new ArgumentNullException("Parameter configType may not be null.");
			}
			if (configInstance == null)
			{
				throw new ArgumentNullException("Parameter configInstance may not be null.");
			}
			string text = Path.Combine(FeatureConfigsDirectoryPath, moduleIdentifier);
			if (!Directory.Exists(text))
			{
				Directory.CreateDirectory(text);
			}
			string text2 = Path.Combine(text, featureIdentifier + "_" + configType.Name + ".json");
			ArchiveLogger.Debug("Saving Feature Setting to: " + text2);
			File.WriteAllText(text2, JsonConvert.SerializeObject(configInstance, ArchiveMod.JsonSerializerSettings));
		}
	}
	public static class MetadataHelper
	{
		internal static IEnumerable<CustomAttribute> GetCustomAttributes<T>(TypeDefinition td, bool inherit) where T : Attribute
		{
			List<CustomAttribute> list = new List<CustomAttribute>();
			Type type = typeof(T);
			TypeDefinition val = td;
			do
			{
				list.AddRange(((IEnumerable<CustomAttribute>)val.CustomAttributes).Where((CustomAttribute ca) => ((MemberReference)ca.AttributeType).FullName == type.FullName));
				TypeReference baseType = val.BaseType;
				val = ((baseType != null) ? baseType.Resolve() : null);
			}
			while (inherit && ((val != null) ? ((MemberReference)val).FullName : null) != typeof(object).FullName);
			return list;
		}

		public static ArchiveModule GetMetadata(Type pluginType)
		{
			object[] customAttributes = pluginType.GetCustomAttributes(typeof(ArchiveModule), inherit: false);
			if (customAttributes.Length == 0)
			{
				return null;
			}
			return (ArchiveModule)customAttributes[0];
		}

		public static ArchiveModule GetMetadata(object plugin)
		{
			return GetMetadata(plugin.GetType());
		}

		public static T[] GetAttributes<T>(Type pluginType) where T : Attribute
		{
			return (T[])pluginType.GetCustomAttributes(typeof(T), inherit: true);
		}

		public static T[] GetAttributes<T>(Assembly assembly) where T : Attribute
		{
			return (T[])assembly.GetCustomAttributes(typeof(T), inherit: true);
		}

		public static IEnumerable<T> GetAttributes<T>(object plugin) where T : Attribute
		{
			return GetAttributes<T>(plugin.GetType());
		}

		public static T[] GetAttributes<T>(MemberInfo member) where T : Attribute
		{
			return (T[])member.GetCustomAttributes(typeof(T), inherit: true);
		}

		public static IEnumerable<ArchiveDependency> GetDependencies(Type plugin)
		{
			return plugin.GetCustomAttributes(typeof(ArchiveDependency), inherit: true).Cast<ArchiveDependency>();
		}
	}
	public static class PresenceFormatter
	{
		[AttributeUsage(AttributeTargets.Property)]
		public class PresenceFormatProvider : Attribute
		{
			private PropertyInfo _propertyInfo;

			public string Identifier { get; private set; }

			public bool IsValid => _propertyInfo != null;

			public Type PropertyType => PropertyInfo.PropertyType;

			public string DebugIdentifier => $"{PropertyInfo.DeclaringType.Name}.{PropertyInfo.Name} (ASM:{PropertyInfo.DeclaringType.Assembly.GetName().Name})";

			internal PropertyInfo PropertyInfo => _propertyInfo ?? throw new Exception($"PropertyInfo not set on {"PresenceFormatProvider"} with ID \"{Identifier}\"!");

			public PresenceFormatProvider(string identifier)
			{
				Identifier = identifier;
			}

			internal void SetPropertyInfo(PropertyInfo propertyInfo)
			{
				if (propertyInfo == null)
				{
					throw new ArgumentException($"Provided {"PresenceFormatProvider"} {"PropertyInfo"} may not be null! (ID:{Identifier})");
				}
				MethodInfo? getMethod = propertyInfo.GetGetMethod();
				if ((object)getMethod == null || !getMethod.IsStatic)
				{
					throw new ArgumentException($"Provided {"PresenceFormatProvider"} {"PropertyInfo"} has to implement a static get method! (ID:{Identifier})");
				}
				_propertyInfo = propertyInfo;
			}

			public object GetValue()
			{
				return PropertyInfo.GetValue(null);
			}
		}

		[AttributeUsage(AttributeTargets.Property)]
		internal class FallbackPresenceFormatProvider : PresenceFormatProvider
		{
			public bool NoNotImplementedWarning { get; private set; }

			public FallbackPresenceFormatProvider(string identifier, bool noNotImplementedWarning = false)
				: base(identifier)
			{
				NoNotImplementedWarning = noNotImplementedWarning;
			}
		}

		private static readonly Dictionary<string, PresenceFormatProvider> _formatters = new Dictionary<string, PresenceFormatProvider>();

		private static readonly List<Type> _typesToCheckForProviders = new List<Type>();

		private static IArchiveLogger _logger;

		private static IArchiveLogger Logger => _logger ?? (_logger = LoaderWrapper.CreateArSubLoggerInstance("PresenceFormatter", ConsoleColor.DarkMagenta));

		internal static void Setup()
		{
			Logger.Debug("Setting up providers ...");
			typeof(PresenceManager).RegisterAllPresenceFormatProviders(throwOnDuplicate: false);
			foreach (Type typesToCheckForProvider in _typesToCheckForProviders)
			{
				PropertyInfo[] properties = typesToCheckForProvider.GetProperties();
				foreach (PropertyInfo propertyInfo in properties)
				{
					try
					{
						PresenceFormatProvider customAttribute = propertyInfo.GetCustomAttribute<PresenceFormatProvider>();
						if (customAttribute != null)
						{
							customAttribute.SetPropertyInfo(propertyInfo);
							RegisterFormatter(customAttribute);
						}
					}
					catch (Exception ex)
					{
						Logger.Exception(ex);
					}
				}
			}
			foreach (KeyValuePair<string, PresenceFormatProvider> item in _formatters.Where((KeyValuePair<string, PresenceFormatProvider> kvp) => kvp.Value is FallbackPresenceFormatProvider && !((FallbackPresenceFormatProvider)kvp.Value).NoNotImplementedWarning))
			{
				Logger.Warning("Identifier \"" + item.Key + "\" has not been implemented! Using Fallback default values!");
			}
		}

		public static void RegisterAllPresenceFormatProviders(this Type type, bool throwOnDuplicate = true)
		{
			if (type == null)
			{
				throw new ArgumentException("Type must not be null!");
			}
			if (_typesToCheckForProviders.Contains(type))
			{
				if (throwOnDuplicate)
				{
					throw new ArgumentException("Duplicate Type registered: \"" + type?.FullName + "\"");
				}
			}
			else
			{
				_typesToCheckForProviders.Add(type);
			}
		}

		public static void RegisterFormatter(PresenceFormatProvider pfp)
		{
			if (!pfp.IsValid)
			{
				return;
			}
			bool flag = false;
			if (_formatters.TryGetValue(pfp.Identifier, out var value))
			{
				if (pfp is FallbackPresenceFormatProvider)
				{
					return;
				}
				if (!(value is FallbackPresenceFormatProvider))
				{
					throw new ArgumentException($"Duplicate formatter identifier: \"{pfp.Identifier}\" (\"{pfp.DebugIdentifier}\")");
				}
				_formatters.Remove(pfp.Identifier);
				flag = true;
			}
			_formatters.Add(pfp.Identifier, pfp);
			Logger.Debug($"{(flag ? " (Fallback Overridden)" : ((pfp is FallbackPresenceFormatProvider) ? " (Fallback)" : string.Empty))} Registered: \"{pfp.Identifier}\" => {pfp.DebugIdentifier}");
		}

		internal static object Get(string identifier)
		{
			_formatters.TryGetValue(identifier, out var value);
			return value?.GetValue();
		}

		internal static T Get<T>(string identifier)
		{
			_formatters.TryGetValue(identifier, out var value);
			if (value == null)
			{
				return default(T);
			}
			if (!value.PropertyType.IsAssignableFrom(typeof(T)) && typeof(T) != typeof(string))
			{
				throw new ArgumentException($"The property at identifier \"{identifier}\" is not declared as Type \"{typeof(T).Name}\"!");
			}
			return (T)value.GetValue();
		}

		public static string Format(this string formatString, params (string search, string replace)[] extraFormatters)
		{
			return FormatPresenceString(formatString, extraFormatters);
		}

		public static string FormatPresenceString(string formatString)
		{
			return FormatPresenceString(formatString, null);
		}

		public static string FormatPresenceString(string formatString, params (string search, string replace)[] extraFormatters)
		{
			return FormatPresenceString(formatString, stripAllTMPTags: true, extraFormatters);
		}

		public static string FormatPresenceString(string formatString, bool stripAllTMPTags = true, params (string search, string replace)[] extraFormatters)
		{
			string text = formatString;
			foreach (KeyValuePair<string, PresenceFormatProvider> formatter in _formatters)
			{
				if (text.Contains("%" + formatter.Key + "%"))
				{
					text = text.ReplaceCaseInsensitive("%" + formatter.Key + "%", formatter.Value.GetValue()?.ToString() ?? "null");
				}
			}
			if (extraFormatters != null)
			{
				for (int i = 0; i < extraFormatters.Length; i++)
				{
					(string, string) tuple = extraFormatters[i];
					if (text.Contains("%" + tuple.Item1 + "%"))
					{
						text = text.ReplaceCaseInsensitive("%" + tuple.Item1 + "%", tuple.Item2);
					}
				}
			}
			if (stripAllTMPTags)
			{
				return Utils.StripTMPTagsRegex(text.Trim());
			}
			return text.Trim();
		}
	}
	public static class RundownFlagsExtensions
	{
		private static IEnumerable<Utils.RundownFlags> _allFlagsOrdered;

		public static IEnumerable<Utils.RundownFlags> AllFlagsOrdered
		{
			get
			{
				if (_allFlagsOrdered == null)
				{
					_allFlagsOrdered = (from Utils.RundownFlags x in Enum.GetValues(typeof(Utils.RundownFlags))
						orderby x
						select x).Skip(2);
				}
				return _allFlagsOrdered;
			}
		}

		public static bool IsIncludedIn(this Utils.RundownID rundownID, Utils.RundownFlags flags)
		{
			return Utils.FlagsContain(flags, rundownID);
		}

		public static Utils.RundownFlags To(this Utils.RundownFlags flags, Utils.RundownFlags to)
		{
			if (to == Utils.RundownFlags.Latest)
			{
				return flags.ToLatest();
			}
			if (flags > to)
			{
				return Utils.FlagsFromTo(to, flags);
			}
			return Utils.FlagsFromTo(flags, to);
		}

		public static Utils.RundownFlags ToLatest(this Utils.RundownFlags flags)
		{
			return Utils.FlagsFromTo(flags, Utils.RundownFlags.Latest);
		}

		public static Utils.RundownFlags LowestRundownFlag(this Utils.RundownFlags flags)
		{
			return AllFlagsOrdered.FirstOrDefault((Utils.RundownFlags x) => flags.HasFlag(x));
		}

		public static Utils.RundownFlags HighestRundownFlag(this Utils.RundownFlags flags)
		{
			return AllFlagsOrdered.LastOrDefault((Utils.RundownFlags x) => flags.HasFlag(x));
		}

		[Obsolete]
		public static int GetIntValue<T>(this T thisEnum) where T : IConvertible
		{
			if (thisEnum is Enum)
			{
				Type type = thisEnum.GetType();
				foreach (int value in Enum.GetValues(type))
				{
					ref T reference = ref thisEnum;
					T val = default(T);
					if (val == null)
					{
						val = reference;
						reference = ref val;
					}
					if (value == reference.ToInt32(CultureInfo.InvariantCulture) && type.GetField(type.GetEnumName(value)).GetCustomAttributes(typeof(Utils.ValueAttribute), inherit: false).FirstOrDefault() is Utils.ValueAttribute valueAttribute && valueAttribute.Type == typeof(int))
					{
						return (int)valueAttribute.Value;
					}
				}
			}
			return -1;
		}
	}
	public class UnityMessages
	{
		public const string Awake = "Awake";

		public const string Start = "Start";

		public const string Update = "Update";

		public const string FixedUpdate = "FixedUpdate";

		public const string LateUpdate = "LateUpdate";

		public const string OnEnable = "OnEnable";

		public const string OnDisable = "OnDisable";

		public const string OnDestroy = "OnDestroy";

		public const string OnApplicationFocus = "OnApplicationFocus";

		public const string OnApplicationQuit = "OnApplicationQuit";
	}
	public static class Utils
	{
		[Obsolete]
		public class ValueAttribute : Attribute
		{
			public object Value { get; private set; }

			public Type Type { get; private set; }

			public ValueAttribute(object value)
			{
				Value = value;
				Type = value.GetType();
			}
		}

		public enum RundownID
		{
			Latest = -2,
			RundownUnitialized,
			RundownUnknown,
			RundownOne,
			RundownTwo,
			RundownThree,
			RundownFour,
			RundownFive,
			RundownSix,
			RundownSeven,
			RundownAltOne,
			RundownAltTwo,
			RundownAltThree,
			RundownAltFour,
			RundownAltFive,
			RundownAltSix,
			RundownEight
		}

		[Flags]
		public enum RundownFlags
		{
			Latest = -2,
			None = 0,
			RundownOne = 1,
			RundownTwo = 2,
			RundownThree = 4,
			RundownFour = 8,
			RundownFive = 0x10,
			RundownSix = 0x20,
			RundownSeven = 0x40,
			RundownAltOne = 0x80,
			RundownAltTwo = 0x100,
			RundownAltThree = 0x200,
			RundownAltFour = 0x400,
			RundownAltFive = 0x800,
			RundownAltSix = 0x1000,
			RundownEight = 0x2000,
			All = 0x3FFF
		}

		public const BindingFlags AnyBindingFlagss = BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic;

		private static Type _UnityEngine_Random;

		private static MethodInfo _UnityEngine_Random_RandomRangeInt;

		private static MethodInfo _UnityEngine_Random_Range;

		private static Type _UnityEngine_Time;

		private static PropertyInfo _UnityEngine_time_PI;

		public const string EXTENDED_WITHOUT_TMP_TAGS = "://EXTENDED";

		public const string EXTENDED = "<color=orange><size=80%>://EXTENDED</size></color>";

		private static RundownID? _latestRundownID;

		private const RundownFlags LatestRundownFlags = RundownFlags.RundownEight;

		internal static float Unity_Time => (float)_UnityEngine_time_PI.GetValue(null);

		static Utils()
		{
			_UnityEngine_Random = Type.GetType("UnityEngine.Random, UnityEngine.CoreModule, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null");
			_UnityEngine_Time = Type.GetType("UnityEngine.Time, UnityEngine.CoreModule, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null");
			_UnityEngine_time_PI = _UnityEngine_Time.GetProperty("time");
			_latestRundownID = null;
			try
			{
				_UnityEngine_Random_RandomRangeInt = _UnityEngine_Random.GetMethod("RandomRangeInt");
			}
			catch (Exception)
			{
			}
			try
			{
				_UnityEngine_Random_Range = _UnityEngine_Random.GetMethod("Range", new Type[2]
				{
					typeof(int),
					typeof(int)
				});
			}
			catch (Exception)
			{
			}
		}

		internal static int RandomRangeInt(int min, int max)
		{
			if (_UnityEngine_Random_RandomRangeInt != null)
			{
				return (int)_UnityEngine_Random_RandomRangeInt.Invoke(null, new object[2] { min, max });
			}
			return (int)_UnityEngine_Random_Range.Invoke(null, new object[2] { min, max });
		}

		public static T PickRandom<T>(this T[] array)
		{
			if (array.Length == 0)
			{
				return default(T);
			}
			return (T)array.GetValue(RandomRangeInt(0, array.Length - 1));
		}

		public static string StripTMPTagsRegex(string input)
		{
			return Regex.Replace(input, "<.*?>", string.Empty);
		}

		public static T PickRandom<T>(this List<T> list)
		{
			return list.ToArray().PickRandom();
		}

		public static T PickRandomExcept<T>(this T[] array, T except)
		{
			if (array.Length == 1)
			{
				return array[0];
			}
			int num = 0;
			T result;
			do
			{
				result = array.PickRandom();
				num++;
			}
			while (result.Equals(except) && num < 20);
			return result;
		}

		public static T PickRandomExcept<T>(this List<T> list, T except)
		{
			if (list != null)
			{
				return list.ToArray().PickRandomExcept(except);
			}
			return default(T);
		}

		public static T PickRandomExcept<T>(this T[] array, Func<T, bool> selectFunction)
		{
			if (array.Length == 1)
			{
				return array[0];
			}
			int num = 0;
			T val;
			do
			{
				val = array.PickRandom();
				num++;
			}
			while (!selectFunction(val) && num < 20);
			return val;
		}

		public static T PickRandomExcept<T>(this List<T> list, Func<T, bool> selectFunction)
		{
			if (list != null)
			{
				return list.ToArray().PickRandomExcept(selectFunction);
			}
			return default(T);
		}

		public static bool TryPickRandom<T>(this T[] array, out T value)
		{
			if (array.Length == 0)
			{
				value = default(T);
				return false;
			}
			value = array[RandomRangeInt(0, array.Length - 1)];
			return true;
		}

		public static bool TryPickRandom<T>(this List<T> list, out T value)
		{
			if (list.Count == 0)
			{
				value = default(T);
				return false;
			}
			value = list[RandomRangeInt(0, list.Count - 1)];
			return true;
		}

		public static T GetEnumFromName<T>(string name) where T : struct
		{
			if (Enum.TryParse<T>(name, out var result))
			{
				return result;
			}
			ArchiveLogger.Warning($"{"GetEnumFromName"} couldn't resolve enum \"{name}\" from type \"{typeof(T).FullName}\"!");
			return default(T);
		}

		public static bool TryGetEnumFromName<T>(string name, out T value) where T : struct
		{
			if (Enum.TryParse<T>(name, out value))
			{
				return true;
			}
			return false;
		}

		public static object StartCoroutine(IEnumerator routine)
		{
			return LoaderWrapper.StartCoroutine(routine);
		}

		public static void StopCoroutine(object coroutineToken)
		{
			LoaderWrapper.StopCoroutine(coroutineToken);
		}

		public static string GetRundownTag(RundownFlags rundowns, bool generalizeLatest = false)
		{
			Enum.TryParse<RundownID>(rundowns.LowestRundownFlag().ToString(), out var result);
			Enum.TryParse<RundownID>(rundowns.HighestRundownFlag().ToString(), out var result2);
			if (result2 == RundownID.RundownUnknown)
			{
				result2 = GetLatestRundownID();
			}
			bool flag = result2 == GetLatestRundownID();
			if (result == result2)
			{
				string value = "R";
				if (result >= RundownID.RundownAltOne && result < RundownID.RundownEight)
				{
					value = "A";
					result = result - 8 + 1;
				}
				if (result >= RundownID.RundownEight)
				{
					result -= 6;
				}
				return $"{value}{result}";
			}
			string value2 = "R";
			if (result >= RundownID.RundownAltOne && result < RundownID.RundownEight)
			{
				value2 = "A";
				result = result - 8 + 1;
			}
			else if (result >= RundownID.RundownEight)
			{
				result -= 6;
			}
			string text = "R";
			if (result2 >= RundownID.RundownAltOne && result2 < RundownID.RundownEight)
			{
				text = "A";
				result2 = result2 - 8 + 1;
			}
			else if (result2 >= RundownID.RundownEight)
			{
				result2 -= 6;
			}
			DefaultInterpolatedStringHandler defaultInterpolatedStringHandler = new DefaultInterpolatedStringHandler(1, 3);
			defaultInterpolatedStringHandler.AppendFormatted(value2);
			defaultInterpolatedStringHandler.AppendFormatted((int)result);
			defaultInterpolatedStringHandler.AppendLiteral("-");
			object value3;
			if (!(flag && generalizeLatest))
			{
				string text2 = text;
				int num = (int)result2;
				value3 = text2 + num;
			}
			else
			{
				value3 = "RL";
			}
			defaultInterpolatedStringHandler.AppendFormatted((string?)value3);
			return defaultInterpolatedStringHandler.ToStringAndClear();
		}

		public static string GetRundownTitle()
		{
			return GetRundownTitle(ArchiveMod.CurrentRundown);
		}

		public static string GetRundownTitle(RundownID rundown)
		{
			return rundown switch
			{
				RundownID.RundownOne => "Rundown Protocol #001", 
				RundownID.RundownTwo => "Infection", 
				RundownID.RundownThree => "The Vessel", 
				RundownID.RundownFour => "Contact", 
				RundownID.RundownFive => "Rebirth", 
				RundownID.RundownSix => "Destination", 
				RundownID.RundownSeven => "Rise", 
				_ => "Unknown", 
			};
		}

		public static string GetHash(byte[] bytes)
		{
			using SHA256 sHA = SHA256.Create();
			byte[] array = sHA.ComputeHash(bytes);
			StringBuilder stringBuilder = new StringBuilder();
			byte[] array2 = array;
			foreach (byte b in array2)
			{
				stringBuilder.Append(b.ToString("x2"));
			}
			return stringBuilder.ToString();
		}

		public static string UsersafeFormat(string format, params string[] replacementData)
		{
			for (int i = 0; i < replacementData.Length; i++)
			{
				string text = $"{{{i}}}";
				if (format.Contains(text))
				{
					format = format.ReplaceCaseInsensitive(text, replacementData[i]);
				}
			}
			return format;
		}

		public static IList ToSystemListSlow(object allBlocks, Type type)
		{
			if (LoaderWrapper.IsGameIL2CPP())
			{
				Type type2 = ImplementationManager.GameTypeByIdentifier("GenericList").MakeGenericType(type);
				IList list = (IList)Activator.CreateInstance(typeof(List<>).MakeGenericType(type));
				IEnumerator enumerator = ((IEnumerable)type2.GetMethod("ToArray", BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic).Invoke(allBlocks, Array.Empty<object>())).GetEnumerator();
				while (enumerator.MoveNext())
				{
					list.Add(enumerator.Current);
				}
				return list;
			}
			return (IList)allBlocks;
		}

		public static string ToRoman(int number)
		{
			if (number < 0 || number > 3999)
			{
				throw new ArgumentOutOfRangeException("insert value betwheen 1 and 3999");
			}
			if (number < 1)
			{
				return string.Empty;
			}
			if (number >= 1000)
			{
				return "M" + ToRoman(number - 1000);
			}
			if (number >= 900)
			{
				return "CM" + ToRoman(number - 900);
			}
			if (number >= 500)
			{
				return "D" + ToRoman(number - 500);
			}
			if (number >= 400)
			{
				return "CD" + ToRoman(number - 400);
			}
			if (number >= 100)
			{
				return "C" + ToRoman(number - 100);
			}
			if (number >= 90)
			{
				return "XC" + ToRoman(number - 90);
			}
			if (number >= 50)
			{
				return "L" + ToRoman(number - 50);
			}
			if (number >= 40)
			{
				return "XL" + ToRoman(number - 40);
			}
			if (number >= 10)
			{
				return "X" + ToRoman(number - 10);
			}
			if (number >= 9)
			{
				return "IX" + ToRoman(number - 9);
			}
			if (number >= 5)
			{
				return "V" + ToRoman(number - 5);
			}
			if (number >= 4)
			{
				return "IV" + ToRoman(number - 4);
			}
			if (number >= 1)
			{
				return "I" + ToRoman(number - 1);
			}
			throw new ArgumentOutOfRangeException("something bad happened");
		}

		public static bool IsPowerOfTwo(ulong x)
		{
			if (x != 0L)
			{
				return (x & (x - 1)) == 0;
			}
			return false;
		}

		public static string ReplaceCaseInsensitive(this string input, string search, string replacement)
		{
			return Regex.Replace(input, Regex.Escape(search), replacement.Replace("$", "$$"), RegexOptions.IgnoreCase);
		}

		public static byte[] LoadFromResource(string resourcePath)
		{
			return GetResource(Assembly.GetCallingAssembly(), resourcePath);
		}

		public static byte[] GetResource(Assembly assembly, string resourcePath)
		{
			Stream manifestResourceStream = assembly.GetManifestResourceStream(resourcePath);
			byte[] array = new byte[manifestResourceStream.Length];
			manifestResourceStream.Read(array, 0, (int)manifestResourceStream.Length);
			return array;
		}

		public static string GetStartupTextForRundown(RundownID currentRundownID)
		{
			StringBuilder stringBuilder = new StringBuilder();
			switch (currentRundownID)
			{
			case RundownID.RundownOne:
				stringBuilder.Append("<color=red>Rundown #001</color>");
				break;
			case RundownID.RundownTwo:
				stringBuilder.Append("<color=red>Rundown #002 Infection</color>");
				break;
			case RundownID.RundownThree:
				stringBuilder.Append("<color=red>Rundown #003 The Vessel</color>");
				break;
			case RundownID.RundownFour:
				stringBuilder.Append("<color=red>Rundown #004 Contact</color>");
				break;
			case RundownID.RundownFive:
				stringBuilder.Append("<color=red>Rundown #005 Rebirth</color>");
				break;
			default:
				return "<color=red>Rundown #??? Yo Waddup?!</color>\n\nThis shouldn't happen unless you somehow modified the datablocks in R1 to R5 builds ...\nAnyways, things are probably gonna break :)";
			}
			stringBuilder.Append("\n");
			stringBuilder.Append("<size=80%><color=#8211b2>The Archive active.</color></size>\n\n");
			return stringBuilder.ToString();
		}

		[MethodImpl(MethodImplOptions.NoInlining)]
		public static RundownID GetLatestRundownID()
		{
			RundownID valueOrDefault = _latestRundownID.GetValueOrDefault();
			if (!_latestRundownID.HasValue)
			{
				valueOrDefault = GetEnumFromName<RundownID>(GetLatestRundownFlags().ToString());
				_latestRundownID = valueOrDefault;
				return valueOrDefault;
			}
			return valueOrDefault;
		}

		[MethodImpl(MethodImplOptions.NoInlining)]
		public static RundownFlags GetLatestRundownFlags()
		{
			return RundownFlags.RundownEight;
		}

		public static bool FlagsContain(RundownFlags flags, RundownID id)
		{
			if (flags == RundownFlags.None)
			{
				return false;
			}
			if (id == RundownID.RundownUnknown)
			{
				return false;
			}
			RundownFlags latestRundownFlags = GetLatestRundownFlags();
			if (flags == RundownFlags.Latest)
			{
				flags = latestRundownFlags;
			}
			if (id == RundownID.Latest)
			{
				id = GetLatestRundownID();
			}
			if (!Enum.TryParse<RundownFlags>(id.ToString(), out var result))
			{
				return false;
			}
			return (flags & result) == result;
		}

		public static RundownFlags FlagsFromTo(RundownFlags from, RundownFlags to)
		{
			if (from == RundownFlags.Latest)
			{
				from = GetLatestRundownFlags();
			}
			if (to == RundownFlags.Latest)
			{
				to = GetLatestRundownFlags();
			}
			if (from == to)
			{
				return from;
			}
			if (from > to)
			{
				throw new ArgumentException($"{"from"} ({from}) may not be larger than {"to"} ({to})!");
			}
			if (!IsPowerOfTwo((ulong)from) || !IsPowerOfTwo((ulong)to) || from > to)
			{
				return RundownFlags.None;
			}
			RundownFlags? rundownFlags = null;
			for (int num = (int)from; num <= (int)to; num *= 2)
			{
				rundownFlags = (RundownFlags?)((!rundownFlags.HasValue) ? ((ValueType)new RundownFlags?((RundownFlags)num)) : ((ValueType)((uint?)rundownFlags | (uint)num)));
			}
			return rundownFlags.Value;
		}

		public static bool AnyRundownConstraintMatches(MemberInfo memberInfo)
		{
			IEnumerable<RundownConstraint> customAttributes = memberInfo.GetCustomAttributes<RundownConstraint>();
			if (customAttributes.Count() == 0)
			{
				return true;
			}
			RundownID rundown = ArchiveMod.CurrentBuildInfo.Rundown;
			foreach (RundownConstraint item in customAttributes)
			{
				if (item.Matches(rundown))
				{
					return true;
				}
			}
			return false;
		}

		public static bool AnyBuildConstraintMatches(MemberInfo memberInfo)
		{
			IEnumerable<BuildConstraint> customAttributes = memberInfo.GetCustomAttributes<BuildConstraint>();
			if (customAttributes.Count() == 0)
			{
				return true;
			}
			int buildNumber = ArchiveMod.CurrentBuildInfo.BuildNumber;
			foreach (BuildConstraint item in customAttributes)
			{
				if (item.Matches(buildNumber))
				{
					return true;
				}
			}
			return false;
		}

		public static bool TryGetMethodByName(Type type, string name, out MethodInfo methodInfo)
		{
			if (type.GetMethods(BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic).Any((MethodInfo x) => x.Name.Equals(name)))
			{
				methodInfo = type.GetMethod(name, BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic);
				return true;
			}
			methodInfo = null;
			return false;
		}

		public static HashSet<Type> GetNestedClasses(Type type)
		{
			List<Type> list = new List<Type> { type };
			Type[] nestedTypes = type.GetNestedTypes();
			foreach (Type type2 in nestedTypes)
			{
				if (type2.IsClass)
				{
					list.AddRange(GetNestedClasses(type2));
				}
			}
			return list.ToHashSet();
		}

		public static string ByteArrayToString(byte[] data)
		{
			StringBuilder stringBuilder = new StringBuilder(data.Length * 2);
			foreach (byte b in data)
			{
				stringBuilder.AppendFormat("{0:x2}", b);
			}
			return stringBuilder.ToString();
		}

		public static string HashString(this string input)
		{
			using SHA256 sHA = SHA256.Create();
			byte[] bytes = Encoding.UTF8.GetBytes(input);
			return BitConverter.ToString(sHA.ComputeHash(bytes)).Replace("-", "").ToLower();
		}

		public static string HashStream(Stream stream)
		{
			using SHA256 sHA = SHA256.Create();
			byte[] array = new byte[4096];
			int inputCount;
			while ((inputCount = stream.Read(array, 0, array.Length)) > 0)
			{
				sHA.TransformBlock(array, 0, inputCount, array, 0);
			}
			sHA.TransformFinalBlock(new byte[0], 0, 0);
			return ByteArrayToString(sHA.Hash);
		}
	}
}
namespace TheArchive.Loader
{
	public static class LoaderWrapper
	{
		public static class ClassInjector
		{
			public static IntPtr DerivedConstructorPointer<T>()
			{
				return ClassInjector.DerivedConstructorPointer<T>();
			}

			public static void DerivedConstructorBody(Il2CppObjectBase objectBase)
			{
				ClassInjector.DerivedConstructorBody(objectBase);
			}

			public static void RegisterTypeInIl2Cpp<T>(bool logSuccess = false) where T : class
			{
				//IL_0000: Unknown result type (might be due to invalid IL or missing references)
				//IL_0005: Unknown result type (might be due to invalid IL or missing references)
				//IL_0011: Expected O, but got Unknown
				RegisterTypeOptions val = new RegisterTypeOptions();
				val.set_LogSuccess(logSuccess);
				ClassInjector.RegisterTypeInIl2Cpp<T>(val);
			}

			public static void RegisterTypeInIl2CppWithInterfaces<T>(bool logSuccess = false, params Type[] interfaces) where T : class
			{
				//IL_0000: Unknown result type (might be due to invalid IL or missing references)
				//IL_0005: Unknown result type (might be due to invalid IL or missing references)
				//IL_0011: Unknown result type (might be due to invalid IL or missing references)
				//IL_001d: Expected O, but got Unknown
				RegisterTypeOptions val = new RegisterTypeOptions();
				val.set_Interfaces(Il2CppInterfaceCollection.op_Implicit(interfaces));
				val.set_LogSuccess(logSuccess);
				ClassInjector.RegisterTypeInIl2Cpp<T>(val);
			}
		}

		public static string GameDirectory => Paths.GameRootPath;

		public static string UserDataDirectory => Path.Combine(GameDirectory, "UserData");

		public static bool IsIL2CPPType(Type type)
		{
			if (!IsGameIL2CPP())
			{
				return false;
			}
			return ArchiveMod.IL2CPP_BaseType.IsAssignableFrom(type);
		}

		public static bool IsGameIL2CPP()
		{
			return true;
		}

		public static IArchiveLogger CreateLoggerInstance(string name, ConsoleColor col = ConsoleColor.White)
		{
			//IL_0001: Unknown result type (might be due to invalid IL or missing references)
			//IL_000b: Expected O, but got Unknown
			return new BIE_LogWrapper(new ManualLogSource(name));
		}

		public static IArchiveLogger CreateArSubLoggerInstance(string name, ConsoleColor col = ConsoleColor.White)
		{
			return CreateLoggerInstance("Ar::" + name, col);
		}

		public static IArchiveLogger WrapLogger(ManualLogSource loggerInstance)
		{
			return new BIE_LogWrapper(loggerInstance);
		}

		public static object StartCoroutine(IEnumerator routine)
		{
			return MonoBehaviourExtensions.StartCoroutine(BIE_ArchiveMod.MainComponent, routine);
		}

		public static void StopCoroutine(object coroutineToken)
		{
			//IL_0006: Unknown result type (might be due to invalid IL or missing references)
			//IL_0010: Expected O, but got Unknown
			BIE_ArchiveMod.MainComponent.StopCoroutine((Coroutine)coroutineToken);
		}

		public unsafe static void* GetIl2CppMethod<T>(string methodName, string returnTypeName, bool isGeneric, params string[] argTypes) where T : Il2CppObjectBase
		{
			void** ptr = (void**)IL2CPP.GetIl2CppMethod(Il2CppClassPointerStore<T>.NativeClassPtr, isGeneric, methodName, returnTypeName, argTypes).ToPointer();
			if (ptr == null)
			{
				return ptr;
			}
			return *ptr;
		}

		public unsafe static TDelegate GetIl2CppMethod<T, TDelegate>(string methodName, string returnTypeName, bool isGeneric, params string[] argTypes) where T : Il2CppObjectBase where TDelegate : Delegate
		{
			void* il2CppMethod = GetIl2CppMethod<T>(methodName, returnTypeName, isGeneric, argTypes);
			if (il2CppMethod == null)
			{
				return null;
			}
			return Marshal.GetDelegateForFunctionPointer<TDelegate>((IntPtr)il2CppMethod);
		}

		public unsafe static INativeDetour ApplyNativeHook<TClass, TDelegate>(string methodName, string returnType, string[] paramTypes, TDelegate to, out TDelegate original) where TClass : Object where TDelegate : Delegate
		{
			//IL_0042: Unknown result type (might be due to invalid IL or missing references)
			IntPtr nativeClassPtr = Il2CppClassPointerStore<TClass>.NativeClassPtr;
			if (nativeClassPtr == IntPtr.Zero)
			{
				throw new ArgumentException(typeof(TClass).Name + " does not exist in il2cpp domain");
			}
			return INativeDetour.CreateAndApply<TDelegate>(UnityVersionHandler.Wrap((Il2CppMethodInfo*)(void*)IL2CPP.il2cpp_method_get_from_reflection(((Il2CppObjectBase)new MethodInfo(IL2CPP.il2cpp_method_get_object(IL2CPP.GetIl2CppMethod(nativeClassPtr, false, methodName, returnType, paramTypes), nativeClassPtr))).Pointer)).MethodPointer, to, ref original);
		}

		public unsafe static INativeDetour ApplyNativeHook<TClass, TDelegate>(string methodName, string returnType, string[] paramTypes, Type[] genericArguments, TDelegate to, out TDelegate original) where TClass : Object where TDelegate : Delegate
		{
			//IL_0042: Unknown result type (might be due to invalid IL or missing references)
			//IL_0047: Unknown result type (might be due to invalid IL or missing references)
			IntPtr nativeClassPtr = Il2CppClassPointerStore<TClass>.NativeClassPtr;
			if (nativeClassPtr == IntPtr.Zero)
			{
				throw new ArgumentException(typeof(TClass).Name + " does not exist in il2cpp domain");
			}
			MethodInfo val = new MethodInfo(IL2CPP.il2cpp_method_get_object(IL2CPP.GetIl2CppMethod(nativeClassPtr, true, methodName, returnType, paramTypes), nativeClassPtr));
			val.MakeGenericMethod(((IEnumerable<Type>)genericArguments).Select((Func<Type, Type>)Il2CppType.From).ToArray());
			return INativeDetour.CreateAndApply<TDelegate>(UnityVersionHandler.Wrap((Il2CppMethodInfo*)(void*)IL2CPP.il2cpp_method_get_from_reflection(((Il2CppObjectBase)val).Pointer)).MethodPointer, to, ref original);
		}

		public static bool IsModInstalled(string guid)
		{
			PluginInfo value;
			ModuleInfo value2;
			return ((BaseChainloader<BasePlugin>)(object)IL2CPPChainloader.Instance).Plugins.TryGetValue(guid, out value) | (ArchiveModuleChainloader.Instance?.Modules.TryGetValue(guid, out value2) ?? false);
		}
	}
	[BepInPlugin("dev.AuriRex.gtfo.TheArchive", "TheArchive", "0.0.656")]
	[BepInDependency(/*Could not decode attribute arguments.*/)]
	[BepInDependency(/*Could not decode attribute arguments.*/)]
	public class BIE_ArchiveMod : BasePlugin
	{
		public class TheArchive_BIE_Controller : MonoBehaviour
		{
			public TheArchive_BIE_Controller(IntPtr ptr)
				: base(ptr)
			{
			}

			public void Awake()
			{
				Object.DontDestroyOnLoad((Object)(object)this);
				((Object)this).hideFlags = (HideFlags)61;
			}

			public void Update()
			{
				ArchiveMod.OnUpdate();
			}

			public void LateUpdate()
			{
				ArchiveMod.OnLateUpdate();
			}
		}

		public static MonoBehaviour MainComponent { get; private set; }

		public override void Load()
		{
			//IL_0005: Unknown result type (might be due to invalid IL or missing references)
			//IL_000b: Expected O, but got Unknown
			Harmony harmonyInstance = new Harmony("dev.AuriRex.gtfo.TheArchive");
			ArchiveMod.OnApplicationStart(LoaderWrapper.WrapLogger(((BasePlugin)this).Log), harmonyInstance);
			Application.quitting += Action.op_Implicit((Action)delegate
			{
				ArchiveMod.OnApplicationQuit();
			});
			MainComponent = (MonoBehaviour)(object)((BasePlugin)this).AddComponent<TheArchive_BIE_Controller>();
		}

		public override bool Unload()
		{
			ArchiveMod.OnApplicationQuit();
			return ((BasePlugin)this).Unload();
		}
	}
	internal class BIE_LogWrapper : IArchiveLogger
	{
		private readonly ManualLogSource _logger;

		public BIE_LogWrapper(ManualLogSource logger)
		{
			_logger = logger;
			if (!Logger.Sources.Contains((ILogSource)(object)_logger))
			{
				Logger.Sources.Add((ILogSource)(object)_logger);
			}
		}

		public void Success(string msg)
		{
			_logger.LogMessage((object)msg);
		}

		public void Notice(string msg)
		{
			_logger.LogMessage((object)msg);
		}

		public void Info(string msg)
		{
			_logger.LogInfo((object)msg);
		}

		public void Fail(string msg)
		{
			_logger.LogInfo((object)msg);
		}

		public void Msg(ConsoleColor col, string msg)
		{
			_logger.LogMessage((object)msg);
		}

		public void Msg(string msg)
		{
			_logger.LogMessage((object)msg);
		}

		public void Debug(string msg)
		{
			_logger.LogDebug((object)msg);
		}

		public void Warning(string msg)
		{
			_logger.LogWarning((object)msg);
		}

		public void Error(string msg)
		{
			_logger.LogError((object)msg);
		}

		public void Exception(Exception ex)
		{
			//IL_000a: Unknown result type (might be due to invalid IL or missing references)
			//IL_0010: Expected O, but got Unknown
			ManualLogSource logger = _logger;
			bool flag = default(bool);
			BepInExErrorLogInterpolatedStringHandler val = new BepInExErrorLogInterpolatedStringHandler(3, 3, ref flag);
			if (flag)
			{
				((BepInExLogInterpolatedStringHandler)val).AppendFormatted<Exception>(ex);
				((BepInExLogInterpolatedStringHandler)val).AppendLiteral(": ");
				((BepInExLogInterpolatedStringHandler)val).AppendFormatted<string>(ex.Message);
				((BepInExLogInterpolatedStringHandler)val).AppendLiteral("\n");
				((BepInExLogInterpolatedStringHandler)val).AppendFormatted<string>(ex.StackTrace);
			}
			logger.LogError(val);
		}
	}
}
namespace TheArchive.Interfaces
{
	public interface IArchiveLogger
	{
		void Success(string msg);

		void Notice(string msg);

		void Msg(ConsoleColor col, string msg);

		void Info(string msg);

		void Fail(string msg);

		void Debug(string msg);

		void Warning(string msg);

		void Error(string msg);

		void Exception(Exception ex);
	}
	public interface IBaseGameConverter<CT> where CT : class, new()
	{
		CT FromBaseGame(object baseGame, CT existingCT = null);

		object ToBaseGame(CT customType, object existingBaseGame = null);

		Type GetBaseGameType();

		Type GetCustomType();
	}
	public interface IInitAfterDataBlocksReady : IInitializable
	{
	}
	public interface IInitAfterGameDataInitialized : IInitializable
	{
	}
	public interface IInitCondition
	{
		bool InitCondition();
	}
	public interface IInitializable
	{
		void Init();
	}
	public interface IInitImmediately : IInitializable
	{
	}
	public interface IInjectLogger
	{
		IArchiveLogger Logger { get; set; }
	}
}
namespace TheArchive.Core
{
	internal class ArchiveContractResolver : DefaultContractResolver
	{
		public static readonly ArchiveContractResolver Instance = new ArchiveContractResolver();

		protected override JsonProperty CreateProperty(MemberInfo member, MemberSerialization memberSerialization)
		{
			//IL_0002: Unknown result type (might be due to invalid IL or missing references)
			JsonProperty val = ((DefaultContractResolver)this).CreateProperty(member, memberSerialization);
			if (typeof(ISettingsComponent).IsAssignableFrom(val.PropertyType))
			{
				val.Ignored = true;
			}
			return val;
		}
	}
	public class ArchiveLegacyNativePatcher
	{
		[Obsolete("Just a janky mess, wouldn't recommend using")]
		public class ArchiveLegacyNativePatch : Attribute
		{
			public bool HasType => Type != null;

			public int ParameterCount
			{
				get
				{
					Type[] parameterTypes = ParameterTypes;
					if (parameterTypes == null)
					{
						return 0;
					}
					return parameterTypes.Length;
				}
			}

			public string ReturnTypeString => ReturnType?.Name ?? "void";

			public Type Type { get; internal set; }

			public Type ReturnType { get; set; }

			public string MethodName { get; private set; }

			public Utils.RundownFlags RundownsToPatch { get; private set; }

			public bool GeneralPurposePatch { get; private set; }

			public Type[] ParameterTypes { get; internal set; }

			public ArchiveLegacyNativePatch(Type type, string methodName, Utils.RundownFlags rundowns, Type[] parameterTypes = null, Type returnType = null)
			{
				Type = type;
				MethodName = methodName;
				RundownsToPatch = rundowns;
				ParameterTypes = parameterTypes;
				ReturnType = returnType;
			}

			public ArchiveLegacyNativePatch(Type type, string methodName, Utils.RundownFlags from, Utils.RundownFlags to, Type[] parameterTypes = null, Type returnType = null)
			{
				Type = type;
				MethodName = methodName;
				RundownsToPatch = from.To(to);
				ParameterTypes = parameterTypes;
				ReturnType = returnType;
			}

			public ArchiveLegacyNativePatch(Type type, string methodName, Type[] parameterTypes = null, Type returnType = null)
			{
				Type = type;
				MethodName = methodName;
				ParameterTypes = parameterTypes;
				ReturnType = returnType;
				GeneralPurposePatch = true;
			}
		}

		public class LegacyNativePatchInstance
		{
			public const string kReplacementMethodName = "Replacement";

			public Type DelegateType { get; }

			public Delegate OriginalMethod { get; }

			public bool WasNotAbleToSetProperty { get; }

			private LegacyNativePatchInstance(ArchiveLegacyNativePatch nativePatchInfo, Type patchContainingType)
			{
				if (!ArchiveLegacyPatcher.TryGetMethodByName(nativePatchInfo.Type, nativePatchInfo.MethodName, out var _))
				{
					throw new ArgumentException($"Could not find the original method \"{nativePatchInfo.Type.FullName}.{nativePatchInfo.MethodName}\" targeted by patch \"{patchContainingType.FullName}\"");
				}
				if (!ArchiveLegacyPatcher.TryGetMethodByName(patchContainingType, "Replacement", out var _))
				{
					throw new ArgumentException($"Could not find \"{"Replacement"}\" method for patch \"{patchContainingType.FullName}\"!");
				}
				DelegateType = NativeDelegates.Get(nativePatchInfo);
			}

			internal static LegacyNativePatchInstance CreatePatch(ArchiveLegacyNativePatch nativePatchInfo, Type patchContainingType)
			{
				return new LegacyNativePatchInstance(nativePatchInfo, patchContainingType);
			}
		}

		public static class NativeDelegates
		{
			public delegate void void_Param_0(IntPtr s, IntPtr n);

			public delegate void void_Param_1(IntPtr s, IntPtr a, IntPtr n);

			public delegate void void_Param_2(IntPtr s, IntPtr a, IntPtr b, IntPtr n);

			public delegate void void_Param_3(IntPtr s, IntPtr a, IntPtr b, IntPtr c, IntPtr n);

			internal static Type Get(ArchiveLegacyNativePatch nativePatchInfo)
			{
				string text = $"{nativePatchInfo.ReturnTypeString}_Param_{nativePatchInfo.ParameterCount}";
				Type? nestedType = typeof(NativeDelegates).GetNestedType(text, BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic);
				if (nestedType == null)
				{
					throw new NotImplementedException(text);
				}
				return nestedType;
			}
		}

		private Utils.RundownID _currentRundown;

		private Type[] _patchTypes;

		private List<LegacyNativePatchInstance> _patchInstances = new List<LegacyNativePatchInstance>();

		public void ApplyNativePatches(Utils.RundownID currentRundown, Assembly assembly)
		{
			_currentRundown = currentRundown;
			ArchiveLogger.Msg(ConsoleColor.Magenta, "Applying Native Patches for assembly \"" + assembly.GetName().Name + "\" ...");
			if (_patchTypes == null)
			{
				_patchTypes = ArchiveLegacyPatcher.GetAllTypesWithPatchAttribute(typeof(ArchiveLegacyNativePatch), assembly);
			}
			Type[] patchTypes = _patchTypes;
			foreach (Type type in patchTypes)
			{
				try
				{
					ArchiveLegacyNativePatch customAttribute = type.GetCustomAttribute<ArchiveLegacyNativePatch>();
					ArchiveLegacyPatcher.TryGetBindToSettingsAttribute(type, out var bindPatchToSettingsInfo);
					if (!ArchiveLegacyPatcher.IsPatchEnabledInSettings(ArchiveMod.Settings, bindPatchToSettingsInfo, type.FullName))
					{
						ArchiveLogger.Msg(ConsoleColor.DarkMagenta, $"[{bindPatchToSettingsInfo.BindToSetting}==false] Skipped native patch: \"{type.FullName}\". ({customAttribute.RundownsToPatch})");
						continue;
					}
					if (!customAttribute.GeneralPurposePatch && !Utils.FlagsContain(customAttribute.RundownsToPatch, currentRundown))
					{
						ArchiveLogger.Warning($"Not native patching meth