Decompiled source of ItemStatistics v1.2.2
patchers/ItemStatisticsPatcher.dll
Decompiled a day agousing System; using System.Collections.Generic; using System.Diagnostics; using System.IO; using System.Linq; using System.Reflection; using System.Runtime.CompilerServices; using System.Runtime.Versioning; using BepInEx; using BepInEx.Logging; using Mono.Cecil; [assembly: CompilationRelaxations(8)] [assembly: RuntimeCompatibility(WrapNonExceptionThrows = true)] [assembly: Debuggable(DebuggableAttribute.DebuggingModes.IgnoreSymbolStoreSequencePoints)] [assembly: TargetFramework(".NETStandard,Version=v2.1", FrameworkDisplayName = ".NET Standard 2.1")] [assembly: AssemblyVersion("1.2.0.0")] public static class ItemStatisticsPatcher { private static ManualLogSource _log; private static ManualLogSource log { get { if (_log == null) { _log = Logger.CreateLogSource("ItemStatisticsPatcher"); } return _log; } } public static IEnumerable<string> TargetDLLs => new string[1] { "RoR2.dll" }; public static void Patch(AssemblyDefinition assembly) { //IL_0074: Unknown result type (might be due to invalid IL or missing references) //IL_007b: Expected O, but got Unknown ModuleDefinition mainModule = assembly.MainModule; string[] files = Directory.GetFiles(Paths.PluginPath, "ItemStatistics.dll", SearchOption.AllDirectories); if (files.Length == 0) { log.LogFatal((object)"Failed to find ItemStatistics.dll"); return; } AssemblyDefinition val = AssemblyDefinition.ReadAssembly(files[0]); TypeReference fieldType = mainModule.ImportReference((TypeReference)(object)val.MainModule.GetType("ItemStatistics.Components.TrackedInventory")); TypeReference val2 = mainModule.ImportReference((TypeReference)(object)val.MainModule.GetType("ItemStatistics.ProcChain")); GenericInstanceType val3 = new GenericInstanceType(mainModule.ImportReference(typeof(List<>))); val3.GenericArguments.Add(val2); TypeReference fieldType2 = mainModule.ImportReference((TypeReference)(object)val.MainModule.GetType("ItemStatistics.Index")); ModuleDefinition mainModule2 = AssemblyDefinition.ReadAssembly(Path.Combine(Paths.ManagedPath, "UnityEngine.CoreModule.dll")).MainModule; TypeReference fieldType3 = mainModule.ImportReference((TypeReference)(object)mainModule2.GetType("UnityEngine.GameObject")); AddField(mainModule.GetType("RoR2.CharacterMaster"), fieldType, "trackedInventory"); AddField(mainModule.GetType("RoR2.DotController/DotStack"), val2, "dotStackChain"); AddField(mainModule.GetType("RoR2.DotController/PendingDamage"), (TypeReference)(object)val3, "pendingDotChainList"); AddField(mainModule.GetType("RoR2.Orbs.GenericDamageOrb"), fieldType3, "inflictor"); AddField(mainModule.GetType("RoR2.Orbs.DevilOrb"), fieldType3, "inflictor"); AddField(mainModule.GetType("RoR2.Orbs.BounceOrb"), fieldType3, "inflictor"); AddField(mainModule.GetType("RoR2.Orbs.LunarDetonatorOrb"), fieldType3, "inflictor"); AddField(mainModule.GetType("RoR2.Orbs.HealOrb"), fieldType2, "inflictor"); AddField(mainModule.GetType("EntityStates.FrozenState"), fieldType2, "freezeInflictor"); AddField(mainModule.GetType("EntityStates.FrozenState"), (TypeReference)(object)mainModule.GetType("RoR2.CharacterMaster"), "freezeInflictorMaster"); AddField(mainModule.GetType("RoR2.Orbs.VineOrb/SplitDebuffInformation"), val2, "procChain"); } private static FieldDefinition AddField(TypeDefinition target, TypeReference fieldType, string fieldName) { //IL_0077: Unknown result type (might be due to invalid IL or missing references) //IL_007d: Expected O, but got Unknown if (target == null || fieldType == null) { log.LogError((object)("Null types in AddField: " + fieldName)); return null; } if (((IEnumerable<FieldDefinition>)target.Fields).FirstOrDefault((Func<FieldDefinition, bool>)((FieldDefinition a) => ((MemberReference)a).Name == fieldName)) != null) { log.LogWarning((object)(((MemberReference)target).Name + "." + fieldName + " already exists")); return null; } FieldDefinition val = new FieldDefinition(fieldName, (FieldAttributes)6, fieldType); target.Fields.Add(val); return val; } }
plugins/ItemStatistics.dll
Decompiled a day ago
The result has been truncated due to the large size, download it to view full contents!
using System; using System.Collections; using System.Collections.Generic; using System.Collections.ObjectModel; using System.Diagnostics; 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.Permissions; using System.Text; using System.Text.RegularExpressions; using System.Threading.Tasks; using BepInEx; using BepInEx.Bootstrap; using BepInEx.Configuration; using BepInEx.Logging; using Dolso; using EntityStates; using EntityStates.CaptainDefenseMatrixItem; using EntityStates.GlobalSkills.LunarDetonator; using EntityStates.GummyClone; using EntityStates.TeleporterHealNovaController; using EntityStates.Toolbot; using HG; using HG.GeneralSerializer; using HG.Reflection; using ItemStatistics; using ItemStatistics.Components; using ItemStatistics.Hooks; using ItemStatistics.Networking; using ItemStatistics.Trackers; using Mono.Cecil; using Mono.Cecil.Cil; using Mono.Collections.Generic; using MonoMod.Cil; using MonoMod.RuntimeDetour; using MonoMod.Utils; using Rewired.Integration.UnityUI; using RiskOfOptions; using RiskOfOptions.OptionConfigs; using RiskOfOptions.Options; using RoR2; using RoR2.Items; using RoR2.Networking; using RoR2.Orbs; using RoR2.Projectile; using RoR2.Skills; using RoR2.Stats; using RoR2.UI; using TMPro; using Unity; using UnityEngine; using UnityEngine.AddressableAssets; using UnityEngine.EventSystems; using UnityEngine.Networking; using UnityEngine.ResourceManagement.AsyncOperations; using UnityEngine.UI; [assembly: RuntimeCompatibility(WrapNonExceptionThrows = true)] [assembly: Debuggable(DebuggableAttribute.DebuggingModes.IgnoreSymbolStoreSequencePoints)] [assembly: CompilationRelaxations(8)] [assembly: TargetFramework(".NETStandard,Version=v2.1", FrameworkDisplayName = ".NET Standard 2.1")] [assembly: OptIn] [assembly: SecurityPermission(SecurityAction.RequestMinimum, SkipVerification = true)] [assembly: AssemblyVersion("1.2.2.0")] [module: UnverifiableCode] namespace Dolso { internal static class log { private static ManualLogSource logger; internal static void start(ManualLogSource logSource) { logger = logSource; } internal static void start(string name) { logger = Logger.CreateLogSource(name); } internal static void info(object data) { logger.LogInfo(data); } internal static void message(object data) { logger.LogMessage(data); } internal static void warning(object data) { logger.LogWarning(data); } internal static void error(object data) { logger.LogError(data); } internal static void fatal(object data) { logger.LogFatal(data); } internal static void LogError(this ILCursor c, object data) { logger.LogError((object)$"ILCursor failure, skipping: {data}\n{c}"); } internal static void LogErrorCaller(this ILCursor c, object data) { logger.LogError((object)$"ILCursor failed in {new StackFrame(1).GetMethod().Name}, skipping: {data}\n{c}"); } } internal static class HookManager { internal delegate bool ConfigEnabled<T>(T configValue); private class HookedConfig<T> { private readonly ConfigEnabled<T> enabled; private readonly IDetour detour; internal HookedConfig(ConfigEntry<T> configEntry, ConfigEnabled<T> enabled, IDetour detour) { this.enabled = enabled; this.detour = detour; configEntry.SettingChanged += ConfigChanged; ConfigChanged(configEntry, null); } private void ConfigChanged(object sender, EventArgs args) { if (enabled(((ConfigEntry<T>)sender).Value)) { if (!detour.IsApplied) { detour.Apply(); } } else if (detour.IsApplied) { detour.Undo(); } } } internal const BindingFlags allFlags = BindingFlags.DeclaredOnly | BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic; private static readonly ConfigEnabled<bool> boolConfigEnabled = BoolEnabled; private static ILHookConfig ilHookConfig = new ILHookConfig { ManualApply = true }; private static HookConfig onHookConfig = new HookConfig { ManualApply = true }; internal static void Hook(Type typeFrom, string methodFrom, Manipulator ilHook) { Hook(GetMethod(typeFrom, methodFrom), ilHook); } internal static void Hook(Delegate methodFrom, Manipulator ilHook) { Hook(methodFrom.Method, ilHook); } internal static void Hook(MethodBase methodFrom, Manipulator ilHook) { //IL_0007: Unknown result type (might be due to invalid IL or missing references) try { new ILHook(methodFrom, ilHook, ref ilHookConfig).Apply(); } catch (Exception e) { e.LogHookError(methodFrom, ((Delegate)(object)ilHook).Method); } } internal static void Hook(Type typeFrom, string methodFrom, Delegate onHook) { Hook(GetMethod(typeFrom, methodFrom), onHook.Method, onHook.Target); } internal static void Hook(Delegate methodFrom, Delegate onHook) { Hook(methodFrom.Method, onHook.Method, onHook.Target); } internal static void Hook(MethodBase methodFrom, Delegate onHook) { Hook(methodFrom, onHook.Method, onHook.Target); } internal static void Hook(MethodBase methodFrom, MethodInfo onHook, object target = null) { //IL_0008: Unknown result type (might be due to invalid IL or missing references) try { new Hook(methodFrom, onHook, target, ref onHookConfig).Apply(); } catch (Exception e) { e.LogHookError(methodFrom, onHook); } } internal static void HookConfig(this ConfigEntry<bool> configEntry, Type typeFrom, string methodFrom, Delegate hook) { configEntry.HookConfig(boolConfigEnabled, GetMethod(typeFrom, methodFrom), hook.Method, hook.Target); } internal static void HookConfig(this ConfigEntry<bool> configEntry, MethodBase methodFrom, Delegate hook) { configEntry.HookConfig(boolConfigEnabled, methodFrom, hook.Method, hook.Target); } internal static void HookConfig(this ConfigEntry<bool> configEntry, MethodBase methodFrom, MethodInfo hook) { configEntry.HookConfig(boolConfigEnabled, methodFrom, hook); } internal static void HookConfig<T>(this ConfigEntry<T> configEntry, ConfigEnabled<T> enabled, Type typeFrom, string methodFrom, Delegate hook) { configEntry.HookConfig(enabled, GetMethod(typeFrom, methodFrom), hook.Method, hook.Target); } internal static void HookConfig<T>(this ConfigEntry<T> configEntry, ConfigEnabled<T> enabled, MethodBase methodFrom, Delegate hook) { configEntry.HookConfig(enabled, methodFrom, hook.Method, hook.Target); } internal static void HookConfig<T>(this ConfigEntry<T> configEntry, ConfigEnabled<T> enabled, MethodBase methodFrom, MethodInfo hook, object target = null) { try { new HookedConfig<T>(configEntry, enabled, ManualDetour(methodFrom, hook, target)); } catch (Exception e) { e.LogHookError(methodFrom, hook); } } internal static IDetour ManualDetour(Type typeFrom, string methodFrom, Delegate hook) { return ManualDetour(GetMethod(typeFrom, methodFrom), hook.Method, hook.Target); } internal static IDetour ManualDetour(MethodBase methodFrom, Delegate hook) { return ManualDetour(methodFrom, hook.Method, hook.Target); } internal static IDetour ManualDetour(MethodBase methodFrom, MethodInfo hook, object target = null) { //IL_0051: Unknown result type (might be due to invalid IL or missing references) //IL_0057: Expected O, but got Unknown //IL_0037: Unknown result type (might be due to invalid IL or missing references) //IL_0046: Expected O, but got Unknown //IL_0041: Unknown result type (might be due to invalid IL or missing references) //IL_0047: Expected O, but got Unknown try { ParameterInfo[] parameters = hook.GetParameters(); if (parameters.Length == 1 && parameters[0].ParameterType == typeof(ILContext)) { return (IDetour)new ILHook(methodFrom, (Manipulator)hook.CreateDelegate(typeof(Manipulator)), ref ilHookConfig); } return (IDetour)new Hook(methodFrom, hook, target, ref onHookConfig); } catch (Exception e) { e.LogHookError(methodFrom, hook); return null; } } internal static MethodInfo GetMethod(Type typeFrom, string methodName) { if (typeFrom == null || methodName == null) { log.error($"Null argument in GetMethod: type={typeFrom}, name={methodName}"); return null; } MethodInfo[] array = (from predicate in typeFrom.GetMethods(BindingFlags.DeclaredOnly | BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic) where predicate.Name == methodName select predicate).ToArray(); switch (array.Length) { case 1: return array[0]; case 0: log.error($"Failed to find method: {typeFrom}::{methodName}"); return null; default: { log.error($"{array.Length} ambiguous matches found for: {typeFrom}::{methodName}"); MethodInfo[] array2 = array; for (int i = 0; i < array2.Length; i++) { log.error(array2[i]); } return null; } } } internal static MethodInfo GetMethod(Type typeFrom, string methodName, params Type[] parameters) { if (typeFrom == null || methodName == null) { log.error($"Null argument in GetMethod: type={typeFrom}, name={methodName}"); return null; } MethodInfo? method = typeFrom.GetMethod(methodName, BindingFlags.DeclaredOnly | BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic, null, parameters, null); if (method == null) { log.error($"Failed to find method: {typeFrom}::{methodName}_{parameters.Length}"); } return method; } internal static void SetPriority(string[] before = null, string[] after = null) { ilHookConfig.Before = before; onHookConfig.Before = before; ilHookConfig.After = after; onHookConfig.After = after; } internal static void LogHookError(this Exception e, MethodBase methodFrom, MethodInfo hook) { log.error((methodFrom == null) ? $"null methodFrom for hook: {hook.Name}\n{e}" : $"Failed to hook: {methodFrom.DeclaringType}::{methodFrom.Name} - {hook.Name}\n{e}"); } private static bool BoolEnabled(bool configValue) { return configValue; } } internal static class Utilities { private static GameObject _prefabParent; internal static GameObject CreatePrefab(GameObject gameObject, string name = null) { //IL_0011: Unknown result type (might be due to invalid IL or missing references) //IL_001b: Expected O, but got Unknown if (!Object.op_Implicit((Object)(object)_prefabParent)) { _prefabParent = new GameObject("DolsoPrefabs"); Object.DontDestroyOnLoad((Object)(object)_prefabParent); ((Object)_prefabParent).hideFlags = (HideFlags)61; _prefabParent.SetActive(false); } GameObject val = Object.Instantiate<GameObject>(gameObject, _prefabParent.transform); if (name != null) { ((Object)val).name = name; } return val; } internal static Task<TObj> GetAddressableAsync<TObj>(string addressable) where TObj : Object { //IL_0001: Unknown result type (might be due to invalid IL or missing references) //IL_0006: Unknown result type (might be due to invalid IL or missing references) return Addressables.LoadAssetAsync<TObj>((object)addressable).Task; } internal static Task<GameObject> GetAddressableAsync(string addressable) { //IL_0001: Unknown result type (might be due to invalid IL or missing references) //IL_0006: Unknown result type (might be due to invalid IL or missing references) return Addressables.LoadAssetAsync<GameObject>((object)addressable).Task; } internal static void DoAddressable<TObj>(string addressable, Action<TObj> callback) where TObj : Object { //IL_000e: Unknown result type (might be due to invalid IL or missing references) //IL_0013: Unknown result type (might be due to invalid IL or missing references) AsyncOperationHandle<TObj> val = Addressables.LoadAssetAsync<TObj>((object)addressable); val.Completed += delegate(AsyncOperationHandle<TObj> a) { callback(a.Result); }; } internal static void DoAddressable(string addressable, Action<GameObject> callback) { //IL_000e: Unknown result type (might be due to invalid IL or missing references) //IL_0013: Unknown result type (might be due to invalid IL or missing references) AsyncOperationHandle<GameObject> val = Addressables.LoadAssetAsync<GameObject>((object)addressable); val.Completed += delegate(AsyncOperationHandle<GameObject> a) { callback(a.Result); }; } internal static void AddressableAddComp<TComp>(string addressable, Action<TComp> callback = null) where TComp : Component { //IL_000e: Unknown result type (might be due to invalid IL or missing references) //IL_0013: Unknown result type (might be due to invalid IL or missing references) AsyncOperationHandle<GameObject> val = Addressables.LoadAssetAsync<GameObject>((object)addressable); val.Completed += delegate(AsyncOperationHandle<GameObject> a) { TComp obj = a.Result.AddComponent<TComp>(); callback?.Invoke(obj); }; } internal static void AddressableAddCompSingle<Comp>(string addressable) where Comp : Component { //IL_0001: Unknown result type (might be due to invalid IL or missing references) //IL_0006: Unknown result type (might be due to invalid IL or missing references) AsyncOperationHandle<GameObject> val = Addressables.LoadAssetAsync<GameObject>((object)addressable); val.Completed += delegate(AsyncOperationHandle<GameObject> a) { if (!Object.op_Implicit((Object)(object)a.Result.GetComponent<Comp>())) { a.Result.AddComponent<Comp>(); } }; } internal static void ModifyStateConfig(this EntityStateConfiguration stateConfig, string fieldName, object newValue) { SerializedField[] serializedFields = stateConfig.serializedFieldsCollection.serializedFields; for (int i = 0; i < serializedFields.Length; i++) { if (serializedFields[i].fieldName == fieldName) { Object val = (Object)((newValue is Object) ? newValue : null); if (val != null) { serializedFields[i].fieldValue.objectValue = val; } else if (newValue != null && StringSerializer.CanSerializeType(newValue.GetType())) { serializedFields[i].fieldValue.stringValue = StringSerializer.Serialize(newValue.GetType(), newValue); } else { log.error("Invalid value for SerializedField: " + newValue); } return; } } log.error("Failed to find " + fieldName + " for " + ((Object)stateConfig).name); } internal static bool IsKeyDown(this ConfigEntry<KeyboardShortcut> key, bool onlyPressed) { //IL_0004: Unknown result type (might be due to invalid IL or missing references) //IL_0009: 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_0033: Unknown result type (might be due to invalid IL or missing references) //IL_0038: Unknown result type (might be due to invalid IL or missing references) //IL_001c: Unknown result type (might be due to invalid IL or missing references) //IL_0021: Unknown result type (might be due to invalid IL or missing references) //IL_0024: Unknown result type (might be due to invalid IL or missing references) //IL_0049: Unknown result type (might be due to invalid IL or missing references) KeyboardShortcut value; if (onlyPressed) { value = key.Value; if (!Input.GetKeyDown(((KeyboardShortcut)(ref value)).MainKey)) { goto IL_0030; } } if (!onlyPressed) { value = key.Value; if (!Input.GetKey(((KeyboardShortcut)(ref value)).MainKey)) { goto IL_0030; } } value = key.Value; foreach (KeyCode modifier in ((KeyboardShortcut)(ref value)).Modifiers) { if (!Input.GetKey(modifier)) { return false; } } return true; IL_0030: return false; } } internal static class RiskofOptions { internal const string rooGuid = "com.rune580.riskofoptions"; internal static bool enabled => Chainloader.PluginInfos.ContainsKey("com.rune580.riskofoptions"); internal static void SetSprite(Sprite sprite) { ModSettingsManager.SetModIcon(sprite); } internal static void SetSpriteDefaultIcon() { //IL_0024: Unknown result type (might be due to invalid IL or missing references) //IL_002a: Expected O, but got Unknown //IL_005b: Unknown result type (might be due to invalid IL or missing references) //IL_006a: Unknown result type (might be due to invalid IL or missing references) try { string fullName = new DirectoryInfo(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location)).FullName; Texture2D val = new Texture2D(256, 256); if (ImageConversion.LoadImage(val, File.ReadAllBytes(Path.Combine(fullName, "icon.png")))) { ModSettingsManager.SetModIcon(Sprite.Create(val, new Rect(0f, 0f, (float)((Texture)val).width, (float)((Texture)val).height), new Vector2(0.5f, 0.5f))); } else { log.error("Failed to load icon.png"); } } catch (Exception ex) { log.error("Failed to load icon.png\n" + ex); } } internal static void AddOption<T>(ConfigEntry<T> entry) { AddOption<T>(entry, "", "", restartRequired: false); } internal static void AddOption<T>(ConfigEntry<T> entry, string categoryName = "", string name = "", bool restartRequired = false) { //IL_0146: Unknown result type (might be due to invalid IL or missing references) //IL_014b: Unknown result type (might be due to invalid IL or missing references) //IL_0156: Expected O, but got Unknown //IL_0156: Unknown result type (might be due to invalid IL or missing references) //IL_016c: Expected O, but got Unknown //IL_0167: Unknown result type (might be due to invalid IL or missing references) //IL_016d: Expected O, but got Unknown //IL_0126: Unknown result type (might be due to invalid IL or missing references) //IL_012b: Unknown result type (might be due to invalid IL or missing references) //IL_012d: Unknown result type (might be due to invalid IL or missing references) //IL_0132: Unknown result type (might be due to invalid IL or missing references) //IL_0134: Unknown result type (might be due to invalid IL or missing references) //IL_013e: Expected O, but got Unknown //IL_0139: Unknown result type (might be due to invalid IL or missing references) //IL_0114: Unknown result type (might be due to invalid IL or missing references) //IL_011e: Expected O, but got Unknown //IL_0119: Unknown result type (might be due to invalid IL or missing references) //IL_0102: Unknown result type (might be due to invalid IL or missing references) //IL_010c: Expected O, but got Unknown //IL_0107: Unknown result type (might be due to invalid IL or missing references) //IL_00f0: Unknown result type (might be due to invalid IL or missing references) //IL_00fa: Expected O, but got Unknown //IL_00f5: Unknown result type (might be due to invalid IL or missing references) //IL_00db: Unknown result type (might be due to invalid IL or missing references) //IL_00e5: Expected O, but got Unknown //IL_00e0: Unknown result type (might be due to invalid IL or missing references) //IL_00c6: Unknown result type (might be due to invalid IL or missing references) //IL_00d0: Expected O, but got Unknown //IL_00cb: Unknown result type (might be due to invalid IL or missing references) object obj; if (!(typeof(T) == typeof(float))) { obj = ((!(typeof(T) == typeof(string))) ? ((!(typeof(T) == typeof(bool))) ? ((!(typeof(T) == typeof(int))) ? ((!(typeof(T) == typeof(Color))) ? ((!(typeof(T) == typeof(KeyboardShortcut))) ? ((object)((!typeof(T).IsEnum) ? ((ChoiceOption)null) : new ChoiceOption((ConfigEntryBase)(object)entry, new ChoiceConfig()))) : ((object)new KeyBindOption(entry as ConfigEntry<KeyboardShortcut>, new KeyBindConfig()))) : ((object)new ColorOption(entry as ConfigEntry<Color>, new ColorOptionConfig()))) : ((object)new IntFieldOption(entry as ConfigEntry<int>, new IntFieldConfig()))) : ((object)new CheckBoxOption(entry as ConfigEntry<bool>, new CheckBoxConfig()))) : ((object)new StringInputFieldOption(entry as ConfigEntry<string>, new InputFieldConfig { submitOn = (SubmitEnum)6, lineType = (LineType)0 }))); } else { ConfigEntry<float> obj2 = entry as ConfigEntry<float>; FloatFieldConfig val = new FloatFieldConfig(); ((NumericFieldConfig<float>)val).FormatString = "{0:f2}"; ((BaseOptionConfig)val).description = ((ConfigEntryBase)(object)entry).DescWithDefault("{0:f2}"); obj = (object)new FloatFieldOption(obj2, val); } BaseOption val2 = (BaseOption)obj; if (val2 == null) { return; } BaseOptionConfig config = val2.GetConfig(); config.category = categoryName; config.name = name; config.restartRequired = restartRequired; if (config.description == "") { config.description = ((ConfigEntryBase)(object)entry).DescWithDefault(); } try { ModSettingsManager.AddOption(val2); } catch (Exception arg) { log.error($"AddOption {((ConfigEntryBase)entry).Definition} failed\n{arg}"); } } internal static void AddOption(ConfigEntry<float> entry, float min, float max, string format = "{0:f2}") { AddFloatSlider(entry, min, max, format); } internal static void AddFloatSlider(ConfigEntry<float> entry, float min, float max, string format = "{0:f2}", string categoryName = "") { //IL_0001: Unknown result type (might be due to invalid IL or missing references) //IL_0006: Unknown result type (might be due to invalid IL or missing references) //IL_000d: Unknown result type (might be due to invalid IL or missing references) //IL_0014: Unknown result type (might be due to invalid IL or missing references) //IL_001b: Unknown result type (might be due to invalid IL or missing references) //IL_0023: Unknown result type (might be due to invalid IL or missing references) //IL_0035: Expected O, but got Unknown //IL_0030: Unknown result type (might be due to invalid IL or missing references) //IL_003a: Expected O, but got Unknown ModSettingsManager.AddOption((BaseOption)new SliderOption(entry, new SliderConfig { min = min, max = max, FormatString = format, category = categoryName, description = ((ConfigEntryBase)(object)entry).DescWithDefault(format) })); } internal static void AddIntSlider(ConfigEntry<int> entry, int min, int max, string categoryName = "") { //IL_0001: Unknown result type (might be due to invalid IL or missing references) //IL_0006: Unknown result type (might be due to invalid IL or missing references) //IL_000d: Unknown result type (might be due to invalid IL or missing references) //IL_0014: Unknown result type (might be due to invalid IL or missing references) //IL_001b: Unknown result type (might be due to invalid IL or missing references) //IL_0031: Expected O, but got Unknown //IL_002c: Unknown result type (might be due to invalid IL or missing references) //IL_0036: Expected O, but got Unknown ModSettingsManager.AddOption((BaseOption)new IntSliderOption(entry, new IntSliderConfig { min = min, max = max, category = categoryName, description = ((ConfigEntryBase)(object)entry).DescWithDefault() })); } private static string DescWithDefault(this ConfigEntryBase entry, string format = "0") { return string.Format("{1}\n[Default: " + format + "]", entry.DefaultValue, entry.Description.Description); } } } namespace ItemStatistics { public readonly struct ChainContext : IDisposable { private readonly ProcChain prevProcChain; private static readonly List<ChainContext> chains; public static ConstructorInfo constructor0 => typeof(ChainContext).GetConstructor(Type.EmptyTypes); public static ConstructorInfo constructor1 => typeof(ChainContext).GetConstructor(new Type[1] { typeof(ProcChain) }); public ChainContext(ProcChain procChain) { prevProcChain = CurrentContext.procChain; chains.Add(this); CurrentContext.procChain = procChain; } public ChainContext() : this(null) { } public void Dispose() { CurrentContext.procChain = prevProcChain; chains.RemoveAt(chains.Count - 1); } static ChainContext() { chains = new List<ChainContext>(); Stage.onStageStartGlobal += SafetyCheck; } private static void SafetyCheck(Stage stage) { if (chains.Count != 0) { log.error("ChainContext has leaked chains: " + string.Join(", ", chains)); chains.Clear(); } } public override string ToString() { if (prevProcChain != null) { return prevProcChain.ToString(); } return "null chain"; } } internal static class Config { internal enum ResetMode { Never, WhenNotInInventory } private static ConfigFile configFile; internal static ConfigEntry<ResetMode> resetOnStackChanged; internal static ConfigEntry<KeyboardShortcut> resetSpamKey; internal static void DoConfig(ConfigFile bepConfigFile) { //IL_003e: Unknown result type (might be due to invalid IL or missing references) configFile = bepConfigFile; resetOnStackChanged = configFile.Bind<ResetMode>(string.Empty, "Reset on Stack Changed", ResetMode.Never, "When item stack changes, should its stats automatically be reset? Can right click on item to force a reset. Warning: having this enabled will cause stats to be reset during Mithrix's item steal phase"); resetSpamKey = configFile.Bind<KeyboardShortcut>(string.Empty, "Reset Spam Key", new KeyboardShortcut((KeyCode)306, Array.Empty<KeyCode>()), "Hold this key down to easily reset icons you hover over"); if (Chainloader.PluginInfos.ContainsKey("com.rune580.riskofoptions")) { DoRiskOfOptions(); } } internal static void DoRiskOfOptions() { RiskofOptions.SetSpriteDefaultIcon(); RiskofOptions.AddOption<ResetMode>(resetOnStackChanged); RiskofOptions.AddOption<KeyboardShortcut>(resetSpamKey); } [ConCommand(/*Could not decode attribute arguments.*/)] private static void ReloadConfig(ConCommandArgs args) { configFile.Reload(); Debug.Log((object)"ItemStatistics config reloaded"); } } public static class CurrentContext { public static TrackedInventory attackerTrackedInventory; public static Inventory attackerInventory; public static TrackedInventory victimTrackedInventory; public static Inventory victimInventory; public static ProcChain procChain; private static (DamageInfo damageInfo, ProcChain procChain) cachedInfoChain; internal static ProcChain FindProcChain(DamageInfo damageInfo) { //IL_0082: Unknown result type (might be due to invalid IL or missing references) if (damageInfo == cachedInfoChain.damageInfo) { return cachedInfoChain.procChain; } ProcComponent procComponent = default(ProcComponent); if (Object.op_Implicit((Object)(object)damageInfo.inflictor) && damageInfo.inflictor.TryGetComponent<ProcComponent>(ref procComponent)) { cachedInfoChain = (damageInfo, procComponent.procChain); return cachedInfoChain.procChain; } if (((DamageTypeCombo)(ref damageInfo.damageType)).IsDamageSourceSkillBased && (!Object.op_Implicit((Object)(object)damageInfo.attacker) || !Object.op_Implicit((Object)(object)damageInfo.attacker.GetComponent<MinionComponent>()))) { Index index = damageInfo.damageType.damageSource; cachedInfoChain = (damageInfo, index.isNotNone ? ProcChain.TryGetCachedSkillChain(index) : null); return cachedInfoChain.procChain; } cachedInfoChain = (damageInfo, null); return null; } } public readonly struct HitContext : IDisposable { private readonly ProcChain prevProcChain; private readonly TrackedInventory prevAttackerTrackedInventory; private readonly Inventory prevAttackerInventory; private readonly TrackedInventory prevVictimTrackedInventory; private readonly Inventory prevVictimInventory; private static readonly List<HitContext> hits; public HitContext(ProcChain procChain, TrackedInventory attackerTrackedInventory, Inventory attackerInventory, TrackedInventory victimTrackedInventory, Inventory victimInventory) { prevProcChain = CurrentContext.procChain; prevAttackerTrackedInventory = CurrentContext.attackerTrackedInventory; prevAttackerInventory = CurrentContext.attackerInventory; prevVictimTrackedInventory = CurrentContext.victimTrackedInventory; prevVictimInventory = CurrentContext.victimInventory; hits.Add(this); CurrentContext.procChain = procChain; CurrentContext.attackerTrackedInventory = attackerTrackedInventory; CurrentContext.attackerInventory = attackerInventory; CurrentContext.victimTrackedInventory = victimTrackedInventory; CurrentContext.victimInventory = victimInventory; } public HitContext() : this(null, null, null, null, null) { } public void Dispose() { CurrentContext.procChain = prevProcChain; CurrentContext.attackerTrackedInventory = prevAttackerTrackedInventory; CurrentContext.attackerInventory = prevAttackerInventory; CurrentContext.victimTrackedInventory = prevVictimTrackedInventory; CurrentContext.victimInventory = prevVictimInventory; hits.RemoveAt(hits.Count - 1); } static HitContext() { hits = new List<HitContext>(); Stage.onStageStartGlobal += SafetyCheck; } private static void SafetyCheck(Stage stage) { if (hits.Count != 0) { log.error("HitContext has leaked hits: " + string.Join(",\n", hits)); hits.Clear(); } } public override string ToString() { return string.Join(", ", prevProcChain, prevAttackerTrackedInventory, prevAttackerInventory, prevVictimTrackedInventory, prevVictimInventory); } } [Serializable] [StructLayout(LayoutKind.Explicit)] public struct Index : IEquatable<Index> { [FieldOffset(0)] [SerializeField] private short _index; [FieldOffset(2)] [SerializeField] private IndexType _type; public readonly short index => _index; public readonly IndexType type => _type; public readonly bool isNotNone => _type != IndexType.None; public Index(ItemIndex x) { //IL_0001: Unknown result type (might be due to invalid IL or missing references) _index = (short)x; _type = IndexType.Item; } public Index(EquipmentIndex x) { //IL_0001: Unknown result type (might be due to invalid IL or missing references) _index = (short)x; _type = IndexType.Equip; } public Index(SkillSlot x) { //IL_0001: Unknown result type (might be due to invalid IL or missing references) //IL_0007: Expected I4, but got Unknown _index = (short)(int)x; _type = IndexType.SkillSlot; } public Index(MiscIndex x) { _index = (short)x; _type = IndexType.Misc; } public Index(short index, IndexType type) { _index = index; _type = type; } public static implicit operator Index(ItemIndex index) { //IL_0000: Unknown result type (might be due to invalid IL or missing references) return new Index(index); } public static implicit operator Index(ItemDef itemDef) { //IL_0001: Unknown result type (might be due to invalid IL or missing references) return new Index(itemDef.itemIndex); } public static implicit operator Index(EquipmentIndex index) { //IL_0000: Unknown result type (might be due to invalid IL or missing references) return new Index(index); } public static implicit operator Index(EquipmentDef equipDef) { //IL_0001: Unknown result type (might be due to invalid IL or missing references) return new Index(equipDef.equipmentIndex); } public static implicit operator Index(SkillSlot index) { //IL_0000: Unknown result type (might be due to invalid IL or missing references) return new Index(index); } public static implicit operator Index(MiscIndex index) { return new Index(index); } public static implicit operator Index(DamageSource damageSource) { //IL_0000: Unknown result type (might be due to invalid IL or missing references) //IL_0002: Unknown result type (might be due to invalid IL or missing references) //IL_002c: Expected I4, but got Unknown switch (damageSource - 1) { case 0: return new Index((SkillSlot)0); case 1: case 2: return new Index((SkillSlot)1); case 3: case 4: return new Index((SkillSlot)2); case 7: case 8: return new Index((SkillSlot)3); default: return new Index(0, IndexType.None); } } public static bool operator ==(Index a, Index b) { if (a._index == b._index) { return a._type == b._type; } return false; } public static bool operator !=(Index a, Index b) { if (a._index == b._index) { return a._type != b._type; } return true; } public override readonly bool Equals(object obj) { if (obj is Index) { return (Index)obj == this; } return false; } public readonly bool Equals(Index index) { if (_index == index._index) { return _type == index._type; } return false; } public override readonly int GetHashCode() { return (_index & 0xFFFF) + ((int)_type << 16); } public override readonly string ToString() { //IL_0096: Unknown result type (might be due to invalid IL or missing references) switch (_type) { case IndexType.None: return "None" + ((_index == 0) ? "" : ("." + _index)); case IndexType.Item: return Language.GetString(ItemCatalog.GetItemDef((ItemIndex)_index).nameToken); case IndexType.Equip: return Language.GetString(EquipmentCatalog.GetEquipmentDef((EquipmentIndex)_index).nameToken); case IndexType.SkillSlot: { SkillSlot val = (SkillSlot)(sbyte)_index; return "Skill." + ((object)(SkillSlot)(ref val)).ToString(); } case IndexType.Misc: return "Misc." + (MiscIndex)_index; default: return $"Unknown Index {_type}.{_index}"; } } internal static void Serialize(NetworkWriter writer, Index value) { writer.Write(value._index); writer.Write((short)value._type); } internal static Index Deserialize(NetworkReader reader) { return new Index(reader.ReadInt16(), (IndexType)reader.ReadInt16()); } } public enum IndexType : short { None, Item, Equip, SkillSlot, Misc } public enum MiscIndex { NeverInInventory = -1, Ignore, Regen, Barrier, MinionDamage, Shield } public static class ItemStatisticsAPI { public static void AddTrackerToPrefab(GameObject prefab, Index indexToCredit) { prefab.AddComponent<ProcComponent>().index = indexToCredit; } public static void AddMinionTrackerToPrefab(GameObject prefab, Index indexToCredit) { prefab.AddComponent<MinionComponent>().index = indexToCredit; } public static void AddTrackedIndex(Index indexToAdd, TrackedDef trackedDef) { if (trackedDef != null) { ItemStatisticsPlugin.indexToDef.Add(indexToAdd, trackedDef); } else { log.error($"Attempted to give {indexToAdd} a null tracker type"); } } public static void AddCombatStackTracker(Index index, BuffIndex buffIndex) { //IL_0006: Unknown result type (might be due to invalid IL or missing references) ItemStatisticsPlugin.combatBuffsTrackers.Add((index, buffIndex)); } public static TrackedInventory GetTrackedInventory(CharacterMaster master, bool includeFromMinion) { return TrackedInventory.GetTrackedInventory(master, includeFromMinion); } public static bool TryGetTrackedInventory(CharacterMaster master, bool includeFromMinion, out TrackedInventory trackedInventory) { return TrackedInventory.TryGetTrackedInventory(master, includeFromMinion, out trackedInventory); } public static GameObject CreateNewInflictor(Index index, ProcChain parentChain = null) { return ProcComponent.CreateInflictor(index, parentChain); } public static void CreditNextHeal(Index indexToCredit) { HCHeal.healIndex = indexToCredit; } public static void CreditDamage(TrackedInventory trackedInventory, Index index, float damage) { trackedInventory.AddDamage(index, damage); } public static void CreditGold(TrackedInventory trackedInventory, Index index, float gold) { trackedInventory.AddGold(index, gold); } public static void CreditProcAttempt(TrackedInventory trackedInventory, Index index) { trackedInventory.AddAttempt(index); } public static void CreditProcSuccess(TrackedInventory trackedInventory, Index index) { trackedInventory.AddSuccess(index); } } [BepInDependency(/*Could not decode attribute arguments.*/)] [BepInPlugin("dolso.ItemStatistics", "ItemStatistics", "1.2.2")] [BepInDependency(/*Could not decode attribute arguments.*/)] public class ItemStatisticsPlugin : BaseUnityPlugin { public const string ModGuid = "dolso.ItemStatistics"; public const string Version = "1.2.2"; internal static readonly Dictionary<Index, TrackedDef> indexToDef = new Dictionary<Index, TrackedDef>(); internal static readonly List<(Index index, BuffIndex buffIndex)> combatBuffsTrackers = new List<(Index, BuffIndex)>(); internal static GameObject missileLauncher; internal static GameObject hhOverloading; internal static GameObject hhFireTrail; internal static GameObject hhGoldOuter; internal static GameObject hhGoldInner; internal static GameObject hhBead; internal static int crocoDiseaseProjectileIndex; internal static BodyIndex falseSonCharacterBodyIndex; private void Awake() { log.start(((BaseUnityPlugin)this).Logger); Config.DoConfig(((BaseUnityPlugin)this).Config); ModifyAddressablesAsync(); Stopwatch stopwatch = Stopwatch.StartNew(); GeneralHooks.Apply(); HCDamage.Apply(); HCHeal.Apply(); GEMHooks.Apply(); ItemHooks.Apply(); OrbHooks.Apply(); DotTracking.Apply(); EquipmentHooks.Apply(); SurvivorHooks.Apply(); NetworkHooks.Apply(); FreezeHooks.Apply(); Language.collectLanguageRootFolders += CollectLanguageRootFolders; log.info($"Hooks completed in {stopwatch.Elapsed.TotalSeconds:f2}s"); } private static void ModifyAddressablesAsync() { try { Utilities.AddressableAddComp<IconBehaviour>("RoR2/Base/UI/ItemIcon.prefab", (Action<IconBehaviour>)delegate(IconBehaviour a) { a.type = IndexType.Item; }); Utilities.AddressableAddComp<IconBehaviour>("RoR2/Base/UI/ItemIconScoreboard_InGame.prefab", (Action<IconBehaviour>)delegate(IconBehaviour a) { a.type = IndexType.Item; }); Utilities.AddressableAddComp<TrackedInventory>("RoR2/Base/Core/PlayerMaster.prefab", (Action<TrackedInventory>)delegate(TrackedInventory a) { ((Component)a).GetComponent<CharacterMaster>().trackedInventory = a; }); Utilities.DoAddressable("RoR2/Base/UI/HUDSimple.prefab", delegate(GameObject hudsimple) { EquipmentIcon[] componentsInChildren = hudsimple.GetComponentsInChildren<EquipmentIcon>(); for (int i = 0; i < componentsInChildren.Length; i++) { ((Component)componentsInChildren[i]).gameObject.AddComponent<IconBehaviour>().type = IndexType.Equip; } SkillIcon[] componentsInChildren2 = hudsimple.GetComponentsInChildren<SkillIcon>(); for (int i = 0; i < componentsInChildren2.Length; i++) { ((Component)componentsInChildren2[i]).gameObject.AddComponent<IconBehaviour>().type = IndexType.SkillSlot; } }); Utilities.DoAddressable("RoR2/Base/UI/ScoreboardStrip.prefab", delegate(GameObject scoreboardStrip) { ((Component)scoreboardStrip.GetComponentInChildren<EquipmentIcon>()).gameObject.AddComponent<IconBehaviour>().type = IndexType.Equip; PlayerTooltip.CreateTooltipPrefab(scoreboardStrip); }); ToolbotStance.DoSetup(); } catch (Exception ex) { log.error("Failed to do addressables\n" + ex); } } [SystemInitializer(new Type[] { typeof(PickupCatalog), typeof(SurvivorCatalog), typeof(ProjectileCatalog), typeof(BuffCatalog) })] private static void Init() { //IL_0bba: Unknown result type (might be due to invalid IL or missing references) //IL_0bd9: Unknown result type (might be due to invalid IL or missing references) //IL_0bf8: Unknown result type (might be due to invalid IL or missing references) //IL_0c17: Unknown result type (might be due to invalid IL or missing references) //IL_0c36: Unknown result type (might be due to invalid IL or missing references) //IL_0c55: Unknown result type (might be due to invalid IL or missing references) //IL_0c73: Unknown result type (might be due to invalid IL or missing references) //IL_0c9b: Unknown result type (might be due to invalid IL or missing references) //IL_0ca0: Unknown result type (might be due to invalid IL or missing references) Dictionary<Index, TrackedDef> dictionary = indexToDef; dictionary.Add((SkillSlot)0, TrackedDef.Skill); dictionary.Add((SkillSlot)1, TrackedDef.Skill); dictionary.Add((SkillSlot)2, TrackedDef.Skill); dictionary.Add((SkillSlot)3, TrackedDef.Skill); dictionary.Add(MiscIndex.Regen, new TrackedDef(typeof(HealTracker), TrackedDef.regenPortionTooltip)); dictionary.Add(MiscIndex.Barrier, new TrackedDef(typeof(ArmorTracker), TrackedDef.barrierBlockedTooltip)); dictionary.Add(MiscIndex.MinionDamage, new TrackedDef(typeof(DamageTracker), TrackedDef.minionPortionTooltip)); dictionary.Add(MiscIndex.Shield, new TrackedDef(typeof(ArmorTracker), TrackedDef.shieldBlockedTooltip)); dictionary.Add(Items.Bear, TrackedDef.ArmorCount); dictionary.Add(Items.BearVoid, TrackedDef.ArmorCount); dictionary.Add(Items.Crowbar, TrackedDef.Damage); dictionary.Add(Items.ExplodeOnDeathVoid, TrackedDef.Damage); dictionary.Add(Items.NearbyDamageBonus, TrackedDef.Damage); dictionary.Add(Items.FragileDamageBonus, TrackedDef.Damage); dictionary.Add(Items.FragileDamageBonusConsumed, new TrackedDef(typeof(WatchTracker), TrackedDef.damageTooltip)); dictionary.Add(Items.BossDamageBonus, TrackedDef.Damage); dictionary.Add(Items.CritGlasses, new TrackedDef(typeof(DamageTracker), TrackedDef.totalCritTooltip)); dictionary.Add(Items.CritDamage, TrackedDef.Damage); dictionary.Add(Items.DeathMark, TrackedDef.Damage); dictionary.Add(Items.CritGlassesVoid, TrackedDef.DamageChance); dictionary.Add(Items.ExecuteLowHealthElite, TrackedDef.Damage); dictionary.Add(Items.OutOfCombatArmor, TrackedDef.ArmorCount); dictionary.Add(Items.SprintArmor, TrackedDef.Armor); dictionary.Add(Items.ArmorPlate, TrackedDef.Armor); dictionary.Add(Items.ParentEgg, TrackedDef.Heal); dictionary.Add(Items.GoldOnHit, TrackedDef.Gold); dictionary.Add(Items.GoldOnHurt, TrackedDef.Gold); dictionary.Add(Items.Thorns, TrackedDef.Damage); dictionary.Add(Items.Seed, TrackedDef.Heal); dictionary.Add(Items.BleedOnHit, TrackedDef.DamageChance); dictionary.Add(Items.BleedOnHitVoid, TrackedDef.DamageChance); dictionary.Add(Items.SlowOnHitVoid, TrackedDef.Chance); dictionary.Add(Items.Missile, TrackedDef.DamageChance); dictionary.Add(Items.MissileVoid, TrackedDef.DamageChance); dictionary.Add(Items.ChainLightning, TrackedDef.DamageChance); dictionary.Add(Items.ChainLightningVoid, TrackedDef.DamageChance); dictionary.Add(Items.BounceNearby, TrackedDef.DamageChance); dictionary.Add(Items.StickyBomb, TrackedDef.DamageChance); dictionary.Add(Items.IceRing, TrackedDef.Damage); dictionary.Add(Items.FireRing, TrackedDef.Damage); dictionary.Add(Items.ElementalRingVoid, TrackedDef.Damage); dictionary.Add(Items.FireballsOnHit, TrackedDef.DamageChance); dictionary.Add(Items.LightningStrikeOnHit, TrackedDef.DamageChance); dictionary.Add(Items.HealOnCrit, TrackedDef.Heal); dictionary.Add(Items.Behemoth, TrackedDef.Damage); dictionary.Add(Items.IgniteOnKill, TrackedDef.DamageChance); dictionary.Add(Items.ExplodeOnDeath, TrackedDef.DamageChance); dictionary.Add(Items.Dagger, TrackedDef.Damage); dictionary.Add(Items.Tooth, TrackedDef.Heal); dictionary.Add(Items.Bandolier, TrackedDef.Chance); dictionary.Add(Items.BonusGoldPackOnKill, TrackedDef.Gold); dictionary.Add(Items.BleedOnHitAndExplode, new TrackedDef(typeof(DamageChanceTracker), TrackedDef.hit_damageTooltip)); dictionary.Add(Items.BarrierOnKill, new TrackedDef(typeof(HealTracker), TrackedDef.barrierGainedTooltip)); dictionary.Add(Items.Plant, TrackedDef.Heal); dictionary.Add(Items.Mushroom, TrackedDef.Heal); dictionary.Add(Items.MushroomVoid, TrackedDef.Heal); dictionary.Add(Items.Medkit, TrackedDef.Heal); dictionary.Add(Items.TPHealingNova, TrackedDef.Heal); dictionary.Add(Items.HealingPotionConsumed, TrackedDef.Heal); dictionary.Add(Items.BarrierOnOverHeal, new TrackedDef(typeof(HealTracker), TrackedDef.barrierGainedTooltip)); dictionary.Add(Items.ShockNearby, TrackedDef.DamagePortion); dictionary.Add(Items.PrimarySkillShuriken, new TrackedDef(typeof(DamageChanceTracker), TrackedDef.hit_damageTooltip)); dictionary.Add(Items.StrengthenBurn, TrackedDef.Damage); dictionary.Add(Items.LaserTurbine, TrackedDef.Damage); dictionary.Add(Items.CaptainDefenseMatrix, TrackedDef.Count); dictionary.Add(Items.Icicle, TrackedDef.Damage); dictionary.Add(Items.NovaOnLowHealth, TrackedDef.DamagePortion); dictionary.Add(Items.ImmuneToDebuff, new TrackedDef(typeof(HealChanceTracker), TrackedDef.chance_barrierTooltip)); dictionary.Add(Items.NovaOnHeal, TrackedDef.Damage); dictionary.Add(Items.SprintWisp, TrackedDef.DamagePortion); dictionary.Add(Items.RandomlyLunar, TrackedDef.Chance); dictionary.Add(Items.Firework, TrackedDef.Damage); dictionary.Add(Items.Phasing, TrackedDef.Count); dictionary.Add(Items.FallBoots, TrackedDef.DamagePortion); dictionary.Add(Items.SiphonOnLowHealth, TrackedDef.Heal); dictionary.Add(Items.LunarSun, TrackedDef.DamagePortion); dictionary.Add(Items.Infusion, TrackedDef.Count); dictionary.Add(Items.HeadHunter, TrackedDef.Damage); dictionary.Add(Items.RoboBallBuddy, TrackedDef.Damage); dictionary.Add(Items.VoidMegaCrabItem, TrackedDef.Damage); dictionary.Add(Items.BeetleGland, TrackedDef.Damage); dictionary.Add(Items.DroneWeapons, TrackedDef.Damage); dictionary.Add(Items.GhostOnKill, TrackedDef.Damage); dictionary.Add(Items.MinorConstructOnKill, TrackedDef.Damage); dictionary.Add(Items.Squid, TrackedDef.Damage); dictionary.Add(Items.TitanGoldDuringTP, TrackedDef.Damage); dictionary.Add(Items.ExtraShrineItem, TrackedDef.Chance); dictionary.Add(Items.IncreasePrimaryDamage, TrackedDef.Damage); dictionary.Add(Items.MeteorAttackOnHighDamage, TrackedDef.DamageChance); dictionary.Add(Items.KnockBackHitEnemies, TrackedDef.Damage); dictionary.Add(Items.StunAndPierce, TrackedDef.DamageChance); dictionary.Add(Items.LowerPricedChests, TrackedDef.Count); dictionary.Add(Items.LowerPricedChestsConsumed, TrackedDef.Count); dictionary.Add(Items.ResetChests, TrackedDef.Chance); dictionary.Add(Items.GoldOnStageStart, TrackedDef.Damage); dictionary.Add(Items.TeleportOnLowHealth, new TrackedDef(typeof(DamageChanceTracker), TrackedDef.count_damageTooltip)); dictionary.Add(Items.TriggerEnemyDebuffs, TrackedDef.Damage); dictionary.Add(Items.WardOnLevel, TrackedDef.StackCombat); dictionary.Add(Items.WarCryOnMultiKill, TrackedDef.StackCombat); dictionary.Add(Items.EnergizedOnEquipmentUse, TrackedDef.StackCombat); dictionary.Add(Items.KillEliteFrenzy, TrackedDef.StackCombat); dictionary.Add(Items.MoveSpeedOnKill, TrackedDef.StackCombat); dictionary.Add(Items.SpeedBoostPickup, TrackedDef.StackCombat); dictionary.Add(Items.BoostAllStats, TrackedDef.StackCombat); dictionary.Add(Items.SprintBonus, TrackedDef.StackCombat); dictionary.Add(Items.SprintOutOfCombat, TrackedDef.Stack); dictionary.Add(Items.ArmorReductionOnHit, TrackedDef.StackHit); dictionary.Add(Items.PermanentDebuffOnHit, TrackedDef.StackHit); dictionary.Add(Items.RandomDamageZone, TrackedDef.StackHit); dictionary.Add(Items.IncreaseDamageOnMultiKill, TrackedDef.StackHit); dictionary.Add(Items.LowerHealthHigherDamage, TrackedDef.StackHitSkill); dictionary.Add(Items.AttackSpeedOnCrit, TrackedDef.StackHitSkill); dictionary.Add(Equipment.Saw, TrackedDef.DamagePortion); dictionary.Add(Equipment.GoldGat, TrackedDef.Damage); dictionary.Add(Equipment.BFG, TrackedDef.Damage); dictionary.Add(Equipment.PassiveHealing, TrackedDef.Heal); dictionary.Add(Equipment.CommandMissile, TrackedDef.Damage); dictionary.Add(Equipment.Fruit, TrackedDef.Heal); dictionary.Add(Equipment.DroneBackup, TrackedDef.Damage); dictionary.Add(Equipment.Lightning, TrackedDef.Damage); dictionary.Add(Equipment.Recycle, new TrackedDef(typeof(CountTracker), TrackedDef.recycledTooltip)); dictionary.Add(Equipment.LifestealOnHit, TrackedDef.Heal); dictionary.Add(Equipment.Molotov, TrackedDef.Damage); dictionary.Add(Equipment.VendingMachine, TrackedDef.Heal); dictionary.Add(Equipment.GummyClone, TrackedDef.Damage); dictionary.Add(Equipment.MultiShopCard, TrackedDef.Gold); dictionary.Add(Equipment.GainArmor, TrackedDef.Armor); dictionary.Add(Equipment.Meteor, TrackedDef.Damage); dictionary.Add(Equipment.BurnNearby, TrackedDef.Damage); dictionary.Add(Equipment.AffixRed, TrackedDef.Damage); dictionary.Add(Equipment.AffixBlue, TrackedDef.Damage); dictionary.Add(Equipment.AffixWhite, TrackedDef.Damage); dictionary.Add(Equipment.EliteAurelioniteEquipment, TrackedDef.Damage); dictionary.Add(Equipment.EliteBeadEquipment, TrackedDef.Damage); List<(Index index, BuffIndex buffIndex)> list = combatBuffsTrackers; list.Add((Items.WardOnLevel, Buffs.Warbanner.buffIndex)); list.Add((Items.WarCryOnMultiKill, Buffs.WarCryBuff.buffIndex)); list.Add((Items.EnergizedOnEquipmentUse, Buffs.Energized.buffIndex)); list.Add((Items.KillEliteFrenzy, Buffs.NoCooldowns.buffIndex)); list.Add((Items.MoveSpeedOnKill, Buffs.KillMoveSpeed.buffIndex)); list.Add((Items.SpeedBoostPickup, Buffs.ElusiveAntlersBuff.buffIndex)); list.Add((Items.BoostAllStats, Buffs.BoostAllStatsBuff.buffIndex)); ModifyPrefabs(); crocoDiseaseProjectileIndex = ProjectileCatalog.FindProjectileIndex("CrocoDiseaseProjectile"); falseSonCharacterBodyIndex = BodyCatalog.FindBodyIndex("FalseSonBody"); } private static void ModifyPrefabs() { AddProcComponent("RoR2/Base/StickyBomb/StickyBomb.prefab", Items.StickyBomb); AddProcComponent("RoR2/Base/ElementalRings/FireTornado.prefab", Items.FireRing); AddProcComponent("RoR2/DLC1/ElementalRingVoid/ElementalRingVoidBlackHole.prefab", Items.ElementalRingVoid); AddProcComponent("RoR2/Base/FireballsOnHit/FireMeatBall.prefab", Items.FireballsOnHit); AddProcComponent("RoR2/DLC1/PrimarySkillShuriken/ShurikenProjectile.prefab", Items.PrimarySkillShuriken); AddProcComponent("RoR2/Base/LaserTurbine/LaserTurbineController.prefab", Items.LaserTurbine); AddProcComponent("RoR2/Base/LaserTurbine/LaserTurbineBomb.prefab", Items.LaserTurbine); AddProcComponent("RoR2/Base/Icicle/IcicleAura.prefab", Items.Icicle); AddProcComponent("RoR2/Base/NovaOnLowHealth/VagrantNovaItemBodyAttachment.prefab", Items.NovaOnLowHealth); AddProcComponent("RoR2/Base/Firework/FireworkProjectile.prefab", Items.Firework); AddProcComponent("RoR2/DLC1/LunarSun/LunarSunProjectile.prefab", Items.LunarSun); AddProcComponent("RoR2/Base/Dagger/DaggerProjectile.prefab", Items.Dagger); AddProcComponent("RoR2/Base/BleedOnHitAndExplode/BleedOnHitAndExplodeDelay.prefab", Items.BleedOnHitAndExplode); AddProcComponent("RoR2/DLC2/Items/StunAndPierce/StunAndPierceBoomerang.prefab", Items.StunAndPierce); AddProcComponent("RoR2/DLC2/Items/GoldOnStageStart/BossMissileProjectile.prefab", Items.GoldOnStageStart); AddProcComponent("RoR2/Base/BonusGoldPackOnKill/BonusMoneyPack.prefab", Items.BonusGoldPackOnKill); AddProcComponent("RoR2/Base/Mushroom/MushroomWard.prefab", Items.Mushroom); AddProcComponent("RoR2/Base/Plant/DeskplantWard.prefab", Items.Plant); AddMinionComponent("RoR2/Base/RoboBallBuddy/RoboBallGreenBuddyBody.prefab", Items.RoboBallBuddy); AddMinionComponent("RoR2/Base/RoboBallBuddy/RoboBallRedBuddyBody.prefab", Items.RoboBallBuddy); AddMinionComponent("RoR2/DLC1/VoidJailer/VoidJailerAllyBody.prefab", Items.VoidMegaCrabItem); AddMinionComponent("RoR2/Base/Nullifier/NullifierAllyBody.prefab", Items.VoidMegaCrabItem); AddMinionComponent("RoR2/DLC1/VoidMegaCrab/VoidMegaCrabAllyBody.prefab", Items.VoidMegaCrabItem); AddMinionComponent("RoR2/Base/BeetleGland/BeetleGuardAllyBody.prefab", Items.BeetleGland); AddMinionComponent("RoR2/Base/Squid/SquidTurretBody.prefab", Items.Squid); AddMinionComponent("RoR2/DLC1/MajorAndMinorConstruct/MinorConstructOnKillBody.prefab", Items.MinorConstructOnKill); AddMinionComponent("RoR2/Base/Titan/TitanGoldBody.prefab", Items.TitanGoldDuringTP); AddMinionComponent("RoR2/Base/Drones/BackupDroneBody.prefab", Equipment.DroneBackup); Utilities.AddressableAddComp<ProjectileBehaviour>("RoR2/Base/Nullifier/NullifierDeathBombProjectile.prefab", (Action<ProjectileBehaviour>)delegate(ProjectileBehaviour a) { a.projectile = ProjectileBehaviour.Projectile.ZoeaDeath; }); Utilities.AddressableAddComp<ProjectileBehaviour>("RoR2/DLC1/VoidJailer/VoidJailerDeathBombProjectile.prefab", (Action<ProjectileBehaviour>)delegate(ProjectileBehaviour a) { a.projectile = ProjectileBehaviour.Projectile.ZoeaDeath; }); Utilities.AddressableAddComp<ProjectileBehaviour>("RoR2/DLC1/VoidMegaCrab/VoidMegaCrabDeathBombProjectile.prefab", (Action<ProjectileBehaviour>)delegate(ProjectileBehaviour a) { a.projectile = ProjectileBehaviour.Projectile.ZoeaDeathChilder; }); Utilities.AddressableAddComp<ProjectileBehaviour>("RoR2/DLC1/VoidMegaCrab/VoidMegaCrabDeathBombletsProjectile.prefab", (Action<ProjectileBehaviour>)delegate(ProjectileBehaviour a) { a.projectile = ProjectileBehaviour.Projectile.ZoeaDeath; }); AddProcComponent("RoR2/Base/Saw/Sawmerang.prefab", Equipment.Saw); AddProcComponent("RoR2/Base/GoldGat/GoldGatController.prefab", Equipment.GoldGat); AddProcComponent("RoR2/Base/BFG/BeamSphere.prefab", Equipment.BFG); AddProcComponent("RoR2/Base/Meteor/MeteorStorm.prefab", Equipment.Meteor); AddProcComponent("RoR2/DLC1/Molotov/MolotovClusterProjectile.prefab", Equipment.Molotov); AddProcComponent("RoR2/DLC1/Molotov/MolotovSingleProjectile.prefab", Equipment.Molotov); AddProcComponent("RoR2/DLC1/Molotov/MolotovProjectileDotZone.prefab", Equipment.Molotov); AddProcComponent("RoR2/Base/Mage/MageFireboltBasic.prefab", (SkillSlot)0); AddProcComponent("RoR2/Base/Mage/MageLightningboltBasic.prefab", (SkillSlot)0); AddProcComponent("RoR2/Base/Mage/MageIceBombProjectile.prefab", (SkillSlot)1); AddProcComponent("RoR2/Base/Mage/MageLightningBombProjectile.prefab", (SkillSlot)1); AddProcComponent("RoR2/Base/Mage/MageIcewallPillarProjectile.prefab", (SkillSlot)2); AddProcComponent("RoR2/Base/Merc/EvisProjectile.prefab", (SkillSlot)3); AddProcComponent("RoR2/Base/Merc/EvisOverlapProjectile.prefab", (SkillSlot)3); AddProcComponent("RoR2/Base/Commando/FMJRamping.prefab", (SkillSlot)1); AddProcComponent("RoR2/Base/Commando/CommandoGrenadeProjectile.prefab", (SkillSlot)3); AddProcComponent("RoR2/Base/Captain/CaptainAirstrikeProjectile1.prefab", (SkillSlot)2); AddProcComponent("RoR2/Base/Captain/CaptainAirstrikeAltProjectile.prefab", (SkillSlot)2); AddProcComponent("RoR2/Base/Captain/CaptainSupplyDrop, Base.prefab", (SkillSlot)3); AddProcComponent("RoR2/Base/Captain/CaptainSupplyDrop, EquipmentRestock.prefab", (SkillSlot)3); AddProcComponent("RoR2/Base/Captain/CaptainSupplyDrop, Hacking.prefab", (SkillSlot)3); AddProcComponent("RoR2/Base/Captain/CaptainSupplyDrop, Healing.prefab", (SkillSlot)3); AddProcComponent("RoR2/Base/Captain/CaptainSupplyDrop, Shocking.prefab", (SkillSlot)3); AddProcComponent("RoR2/Base/Captain/CaptainTazer.prefab", (SkillSlot)1); AddProcComponent("RoR2/Base/Huntress/HuntressArrowRain.prefab", (SkillSlot)3); AddProcComponent("RoR2/Base/Loader/LoaderYankHook.prefab", (SkillSlot)1); AddProcComponent("RoR2/Base/Loader/LoaderZapCone.prefab", (SkillSlot)2); AddProcComponent("RoR2/Base/Loader/LoaderPylon.prefab", (SkillSlot)3); AddProcComponent("RoR2/Base/Engi/EngiGrenadeProjectile.prefab", (SkillSlot)0); AddProcComponent("RoR2/Base/Engi/EngiHarpoon.prefab", (SkillSlot)2); AddMinionComponent("RoR2/Base/Engi/EngiTurretBody.prefab", (SkillSlot)3); AddMinionComponent("RoR2/Base/Engi/EngiWalkerTurretBody.prefab", (SkillSlot)3); AddProcComponent("RoR2/Base/Croco/CrocoSpit.prefab", (SkillSlot)1); AddProcComponent("RoR2/Base/Croco/CrocoLeapAcid.prefab", (SkillSlot)2); AddProcComponent("RoR2/Base/Croco/CrocoDiseaseProjectile.prefab", (SkillSlot)3); AddProcComponent("RoR2/Base/Toolbot/CryoCanisterProjectile.prefab", (SkillSlot)1); AddProcComponent("RoR2/Base/Toolbot/CryoCanisterBombletsProjectile.prefab", (SkillSlot)1); AddProcComponent("RoR2/Base/Bandit2/Bandit2ShivProjectile.prefab", (SkillSlot)1); AddProcComponent("RoR2/DLC1/Railgunner/RailgunnerPistolProjectile.prefab", (SkillSlot)0); AddProcComponent("RoR2/Base/Treebot/SyringeProjectile.prefab", (SkillSlot)0); AddProcComponent("RoR2/Base/Treebot/SyringeProjectileHealing.prefab", (SkillSlot)0); AddProcComponent("RoR2/Base/Treebot/TreebotMortar2.prefab", (SkillSlot)1); AddProcComponent("RoR2/Base/Treebot/TreebotMortarRain.prefab", (SkillSlot)1); AddProcComponent("RoR2/Base/Treebot/TreebotFlowerSeed.prefab", (SkillSlot)3); AddProcComponent("RoR2/Base/Treebot/TreebotFruitSeedProjectile.prefab", (SkillSlot)3); Utilities.DoAddressable<GameObject>("RoR2/Base/Treebot/TreebotFruitPack.prefab", (Action<GameObject>)delegate(GameObject a) { ((Component)a.GetComponentInChildren<HealthPickup>()).gameObject.AddComponent<ProcComponent>().index = (SkillSlot)3; }); AddProcComponent("RoR2/DLC1/VoidSurvivor/VoidSurvivorMegaBlasterSmallProjectile.prefab", (SkillSlot)1); AddProcComponent("RoR2/DLC1/VoidSurvivor/VoidSurvivorMegaBlasterBigProjectile.prefab", (SkillSlot)1); AddProcComponent("RoR2/DLC1/VoidSurvivor/VoidSurvivorMegaBlasterBigProjectileCorrupted.prefab", (SkillSlot)1); AddProcComponent("RoR2/Base/LunarSkillReplacements/LunarNeedleProjectile.prefab", (SkillSlot)0); AddProcComponent("RoR2/Base/LunarSkillReplacements/LunarSecondaryProjectile.prefab", (SkillSlot)1); AddProcComponent("RoR2/DLC2/Seeker/SpiritPunchProjectile.prefab", (SkillSlot)0); AddProcComponent("RoR2/DLC2/Seeker/SpiritPunchFinisherProjectile.prefab", (SkillSlot)0); AddProcComponent("RoR2/DLC2/Seeker/SoulSpiralProjectile.prefab", (SkillSlot)1); AddProcComponent("RoR2/DLC2/Seeker/UnseenHandMovingProjectile.prefab", (SkillSlot)1); AddProcComponent("RoR2/DLC2/Seeker/SojournVehicle/SojournVehicle.prefab", (SkillSlot)2); AddProcComponent("RoR2/DLC2/FalseSon/LunarSpike.prefab", (SkillSlot)1); AddProcComponent("RoR2/DLC2/Chef/ChefCleaver.prefab", (SkillSlot)0); AddProcComponent("RoR2/DLC2/Chef/ChefDiceEnhanced.prefab", (SkillSlot)3); AddProcComponent("RoR2/DLC2/Chef/BoostedSearFireballProjectile.prefab", (SkillSlot)3); AddProcComponent("RoR2/DLC2/Chef/BoostedRolyPolyProjectile.prefab", (SkillSlot)3); AddProcComponent("RoR2/DLC2/Chef/ChefGlazeProjectile.prefab", (SkillSlot)3); Utilities.DoAddressable("RoR2/Base/Common/MissileProjectile.prefab", DoMissilePrefab); DoElites(); } private static void AddProcComponent(string addressable, Index index) { //IL_000e: Unknown result type (might be due to invalid IL or missing references) //IL_0013: Unknown result type (might be due to invalid IL or missing references) AsyncOperationHandle<GameObject> val = Addressables.LoadAssetAsync<GameObject>((object)addressable); val.Completed += delegate(AsyncOperationHandle<GameObject> a) { a.Result.AddComponent<ProcComponent>().index = index; }; } private static void AddMinionComponent(string addressable, Index index) { //IL_000e: Unknown result type (might be due to invalid IL or missing references) //IL_0013: Unknown result type (might be due to invalid IL or missing references) AsyncOperationHandle<GameObject> val = Addressables.LoadAssetAsync<GameObject>((object)addressable); val.Completed += delegate(AsyncOperationHandle<GameObject> a) { a.Result.AddComponent<MinionComponent>().index = index; }; } private static void DoMissilePrefab(GameObject missilePrefab) { missileLauncher = Utilities.CreatePrefab(missilePrefab, "DisposableMissileProjectile"); missileLauncher.AddComponent<ProcComponent>().index = Equipment.CommandMissile; CommonAssets.missilePrefab = Utilities.CreatePrefab(missilePrefab, "ATGMissileProjectile"); CommonAssets.missilePrefab.AddComponent<ProcComponent>().index = Items.Missile; } private static async void DoElites() { try { Task<GameObject>[] tasks = new Task<GameObject>[5] { Utilities.GetAddressableAsync("RoR2/Base/EliteLightning/LightningStake.prefab"), Utilities.GetAddressableAsync("RoR2/Base/Common/FireTrail.prefab"), Utilities.GetAddressableAsync("RoR2/DLC2/Elites/EliteAurelionite/AffixAurelionitePreStrikeProjectile.prefab"), Utilities.GetAddressableAsync("RoR2/DLC2/Elites/EliteAurelionite/AffixAurelioniteCenterProjectile.prefab"), Utilities.GetAddressableAsync("RoR2/DLC2/Elites/EliteBead/BeadProjectileTrackingBomb.prefab") }; await Task.WhenAll(tasks); DoDualElitePrefab(tasks[0].Result, Equipment.AffixBlue.equipmentIndex, out hhOverloading); DoDualElitePrefab(tasks[1].Result, Equipment.AffixRed.equipmentIndex, out hhFireTrail); DoDualElitePrefab(tasks[2].Result, Equipment.EliteAurelioniteEquipment.equipmentIndex, out hhGoldOuter); DoDualElitePrefab(tasks[3].Result, Equipment.EliteAurelioniteEquipment.equipmentIndex, out hhGoldInner); DoDualElitePrefab(tasks[4].Result, Equipment.EliteBeadEquipment.equipmentIndex, out hhBead); } catch (Exception data) { log.error(data); } } private static void DoDualElitePrefab(GameObject prefab, EquipmentIndex eliteIndex, out GameObject hhPrefab) { //IL_0006: Unknown result type (might be due to invalid IL or missing references) prefab.AddComponent<ProcComponent>().index = eliteIndex; hhPrefab = Utilities.CreatePrefab(prefab); hhPrefab.GetComponent<ProcComponent>().index = Items.HeadHunter; } private void CollectLanguageRootFolders(List<string> list) { list.Add(Path.Combine(Path.GetDirectoryName(((BaseUnityPlugin)this).Info.Location), "Language")); } } public class ProcChain { public readonly Index index; public readonly ProcChain parent; private static readonly ProcChain[] cachedSkillChains = new ProcChain[4] { new ProcChain((SkillSlot)0), new ProcChain((SkillSlot)1), new ProcChain((SkillSlot)2), new ProcChain((SkillSlot)3) }; public ProcChain(Index index, ProcChain parent = null) { this.index = index; this.parent = parent; } public static ProcChain TryGetCachedSkillChain(Index index) { int num = index.index; if (index.type == IndexType.SkillSlot && num >= 0 && num < cachedSkillChains.Length) { return cachedSkillChains[num]; } return new ProcChain(index); } public override string ToString() { string text = index.ToString(); for (ProcChain procChain = parent; procChain != null; procChain = procChain.parent) { text = procChain.index.ToString() + " -> " + text; } return text; } } [Serializable] public struct Totals { public static float allDamageDealt; public float damageDealt; public float damageTaken; public float healed; public uint goldEarned; } public class TrackedDef { private static readonly string tooltipPrefix = "ITEMSTATISTICS_TOOLTIPS_"; public static readonly string damageTooltip = tooltipPrefix + "DAMAGE"; public static readonly string chanceTooltip = tooltipPrefix + "CHANCE"; public static readonly string countTooltip = tooltipPrefix + "COUNT"; public static readonly string healTooltip = tooltipPrefix + "HEAL"; public static readonly string goldTooltip = tooltipPrefix + "GOLD"; public static readonly string reductionTooltip = tooltipPrefix + "REDUCTION"; public static readonly string portionTooltip = tooltipPrefix + "DAMAGEPORTION"; public static readonly string recycledTooltip = tooltipPrefix + "RECYCLED"; public static readonly string barrierGainedTooltip = tooltipPrefix + "BARRIERGAINED"; public static readonly string totalCritTooltip = tooltipPrefix + "CRITDAMAGE"; public static readonly string stackTooltip = tooltipPrefix + "STACK"; public static readonly string stackCombatTooltip = tooltipPrefix + "STACKCOMBAT"; public static readonly string stackHitTooltip = tooltipPrefix + "STACKHIT"; public static readonly string stackHitSkillTooltip = tooltipPrefix + "STACKHITSKILL"; public static readonly string globalPortionTooltip = tooltipPrefix + "GLOBALPORTION"; public static readonly string minionPortionTooltip = tooltipPrefix + "MINIONPORTION"; public static readonly string regenPortionTooltip = tooltipPrefix + "REGENPORTION"; public static readonly string barrierBlockedTooltip = tooltipPrefix + "BARRIERBLOCKED"; public static readonly string shieldBlockedTooltip = tooltipPrefix + "SHIELDBLOCKED"; public static readonly string chance_barrierTooltip = tooltipPrefix + "CHANCE_BARRIER"; public static readonly string chance_healedTooltip = tooltipPrefix + "CHANCE_HEALED"; public static readonly string chance_damageTooltip = tooltipPrefix + "CHANCE_DAMAGE"; public static readonly string hit_damageTooltip = tooltipPrefix + "HIT_DAMAGE"; public static readonly string count_chanceTooltip = tooltipPrefix + "COUNT_CHANCE"; public static readonly string count_reductionTooltip = tooltipPrefix + "COUNT_REDUCTION"; public static readonly string count_damageTooltip = tooltipPrefix + "COUNT_DAMAGE"; public static readonly TrackedDef ArmorCount = new TrackedDef(typeof(ArmorCountTracker), count_reductionTooltip); public static readonly TrackedDef Armor = new TrackedDef(typeof(ArmorTracker), reductionTooltip); public static readonly TrackedDef Chance = new TrackedDef(typeof(ChanceTracker), count_chanceTooltip); public static readonly TrackedDef Count = new TrackedDef(typeof(CountTracker), countTooltip); public static readonly TrackedDef DamageChance = new TrackedDef(typeof(DamageChanceTracker), chance_damageTooltip); public static readonly TrackedDef DamagePortion = new TrackedDef(typeof(DamageTracker), portionTooltip); public static readonly TrackedDef Damage = new TrackedDef(typeof(DamageTracker), damageTooltip); public static readonly TrackedDef Gold = new TrackedDef(typeof(GoldTracker), goldTooltip); public static readonly TrackedDef Heal = new TrackedDef(typeof(HealTracker), healTooltip); public static readonly TrackedDef Skill = new TrackedDef(typeof(SkillTracker), null); public static readonly TrackedDef Stack = new TrackedDef(typeof(ChanceTracker), stackTooltip); public static readonly TrackedDef StackCombat = new TrackedDef(typeof(ChanceTracker), stackCombatTooltip); public static readonly TrackedDef StackHit = new TrackedDef(typeof(ChanceTracker), stackHitTooltip); public static readonly TrackedDef StackHitSkill = new TrackedDef(typeof(ChanceTracker), stackHitSkillTooltip); private readonly Type trackerType; private readonly string tooltipToken; public TrackedDef(Type trackerType, string tooltipToken) { if (!typeof(ITracker).IsAssignableFrom(trackerType)) { throw new ArgumentException(trackerType?.ToString() + " is not a valid tracker type"); } this.trackerType = trackerType; this.tooltipToken = tooltipToken; } public string GetTooltip(ITracker tracker, in Totals totals) { string[] tooltipArgs = tracker.GetTooltipArgs(in totals); if (tooltipArgs == null) { return ""; } if (tooltipToken == null) { return string.Concat(tooltipArgs); } string text = tooltipToken; object[] array = tooltipArgs; return Language.GetStringFormatted(text, array); } internal ITracker CreateTracker() { return (ITracker)Activator.CreateInstance(trackerType); } } } namespace ItemStatistics.Trackers { public class ArmorCountTracker : ITracker, IReductionTracker, IChanceTracker { private float globalDamageTakenWhenSet; private float damageBlocked; private uint attempted; private uint succedded; public bool dirtyBit { get; set; } public void Attempted() { log.warning("Called Attempted on ArmorChanceTracker"); } public void Succedded() { succedded++; dirtyBit = true; } public void AddBlockedDamage(float blocked) { damageBlocked += blocked; dirtyBit = true; } string[] ITracker.GetTooltipArgs(in Totals totals) { return new string[2] { TrackerStatics.ReductionToString(damageBlocked, totals.damageTaken - globalDamageTakenWhenSet), succedded.ToString() }; } void ITracker.Reset(in Totals totals) { attempted = 0u; succedded = 0u; damageBlocked = 0f; globalDamageTakenWhenSet = totals.damageTaken; } void ITracker.Serialize(SerializationBuffer data) { data.Add(attempted); data.Add(succedded); data.Add(damageBlocked); } void ITracker.Deserialize(SerializationBuffer data) { attempted = data.nextInt; succedded = data.nextInt; damageBlocked = data.nextFloat; } } public class ArmorTracker : ITracker, IReductionTracker { private float globalDamageTakenWhenSet; private float damageBlocked; public bool dirtyBit { get; set; } public void AddBlockedDamage(float blocked) { damageBlocked += blocked; dirtyBit = true; } string[] ITracker.GetTooltipArgs(in Totals totals) { return new string[2] { TrackerStatics.ReductionToString(damageBlocked, totals.damageTaken - globalDamageTakenWhenSet), TrackerStatics.PortionToString(damageBlocked, totals.damageTaken - globalDamageTakenWhenSet) }; } void ITracker.Reset(in Totals totals) { damageBlocked = 0f; globalDamageTakenWhenSet = totals.damageTaken; } void ITracker.Serialize(SerializationBuffer data) { data.Add(damageBlocked); } void ITracker.Deserialize(SerializationBuffer data) { damageBlocked = data.nextFloat; } } public class ChanceTracker : ITracker, IChanceBatchTracker, IChanceTracker { private uint attempted; private uint succedded; public bool dirtyBit { get; set; } public void Attempted() { attempted++; dirtyBit = true; } public void Succedded() { succedded++; dirtyBit = true; } public void Succedded(uint count) { succedded += count; dirtyBit = true; } string[] ITracker.GetTooltipArgs(in Totals totals) { return new string[3] { succedded.ToString("n0"), TrackerStatics.ChanceToString(attempted, succedded), TrackerStatics.RateToString(attempted, succedded) }; } void ITracker.Reset(in Totals totals) { attempted = 0u; succedded = 0u; } void ITracker.Serialize(SerializationBuffer data) { data.Add(attempted); data.Add(succedded); } void ITracker.Deserialize(SerializationBuffer data) { attempted = data.nextInt; succedded = data.nextInt; } } public class CountTracker : ITracker, IChanceBatchTracker, IChanceTracker { private uint succedded; public bool dirtyBit { get; set; } public void Attempted() { log.warning("Called Attempted on CountTracker"); } public void Succedded() { succedded++; dirtyBit = true; } public void Succedded(uint count) { succedded += count; dirtyBit = true; } string[] ITracker.GetTooltipArgs(in Totals totals) { return new string[1] { succedded.ToString("n0") }; } void ITracker.Reset(in Totals totals) { succedded = 0u; } void ITracker.Serialize(SerializationBuffer data) { data.Add(succedded); } void ITracker.Deserialize(SerializationBuffer data) { succedded = data.nextInt; } } public class DamageChanceTracker : ITracker, IChanceTracker, IDamageTracker { private uint attempted; private uint succedded; private float globalDamageWhenSet; private float itemDamage; public bool dirtyBit { get; set; } public void Attempted() { attempted++; dirtyBit = true; } public void Succedded() { succedded++; dirtyBit = true; } public void AddDamage(float damage) { itemDamage += damage; dirtyBit = true; } string[] ITracker.GetTooltipArgs(in Totals totals) { return new string[3] { TrackerStatics.ChanceToString(attempted, succedded), TrackerStatics.IncreaseToString(itemDamage, totals.damageDealt - globalDamageWhenSet), succedded.ToString("n0") }; } void ITracker.Reset(in Totals totals) { attempted = 0u; succedded = 0u; itemDamage = 0f; globalDamageWhenSet = totals.damageDealt; } void ITracker.Serialize(SerializationBuffer data) { data.Add(attempted); data.Add(succedded); data.Add(itemDamage); } void ITracker.Deserialize(SerializationBuffer data) { attempted = data.nextInt; succedded = data.nextInt; itemDamage = data.nextFloat; } } public class DamageTracker : ITracker, IDamageTracker { internal float globalDamageWhenSet; internal float itemDamage; public bool dirtyBit { get; set; } public void AddDamage(float damage) { itemDamage += damage; dirtyBit = true; } string[] ITracker.GetTooltipArgs(in Totals totals) { return new string[2] { TrackerStatics.IncreaseToString(itemDamage, totals.damageDealt - globalDamageWhenSet), TrackerStatics.PortionToString(itemDamage, totals.damageDealt - globalDamageWhenSet) }; } public virtual void Reset(in Totals totals) { itemDamage = 0f; globalDamageWhenSet = totals.damageDealt; } public virtual void Serialize(SerializationBuffer data) { data.Add(itemDamage); } public virtual void Deserialize(SerializationBuffer data) { itemDamage = data.nextFloat; } } public class GoldTracker : ITracker, IGoldTracker { private float globalEarnedWhenSet; private float itemEarned; public bool dirtyBit { get; set; } public void AddGold(float gold) { itemEarned += gold; dirtyBit = true; } string[] ITracker.GetTooltipArgs(in Totals totals) { return new string[1] { TrackerStatics.IncreaseToString(itemEarned, (float)totals.goldEarned - globalEarnedWhenSet) }; } void ITracker.Reset(in Totals totals) { itemEarned = 0f; globalEarnedWhenSet = totals.goldEarned; } void ITracker.Serialize(SerializationBuffer data) { data.Add(itemEarned); } void ITracker.Deserialize(SerializationBuffer data) { itemEarned = data.nextFloat; } } public class HealChanceTracker : ITracker, IChanceTracker, IHealTracker { private uint attempted; private uint succedded; private float itemHealed; private float globalHealedWhenSet; public bool dirtyBit { get; set; } public void Attempted() { attempted++; dirtyBit = true; } public void Succedded() { succedded++; dirtyBit = true; } public void AddHealing(float heal) { itemHealed += heal; dirtyBit = true; } string[] ITracker.GetTooltipArgs(in Totals totals) { return new string[3] { TrackerStatics.ChanceToString(attempted, succedded), TrackerStatics.PortionToString(itemHealed, totals.healed - globalHealedWhenSet), itemHealed.ToString("n0") }; } void ITracker.Reset(in Totals totals) { attempted = 0u; succedded = 0u; itemHealed = 0f; globalHealedWhenSet = totals.healed; } void ITracker.Serialize(SerializationBuffer data) { data.Add(attempted); data.Add(succedded); data.Add(itemHealed); } void ITracker.Deserialize(SerializationBuffer data) { attempted = data.nextInt; succedded = data.nextInt; itemHealed = data.nextFloat; } } public class HealTracker : ITracker, IHealTracker { private float itemHealed; private float globalHealedWhenSet; public bool dirtyBit { get; set; } public void AddHealing(float heal) { itemHealed += heal; dirtyBit = true; } string[] ITracker.GetTooltipArgs(in Totals totals) { return new string[2] { TrackerStatics.PortionToString(itemHealed, totals.healed - globalHealedWhenSet), itemHealed.ToString("n0") }; } void ITracker.Reset(in Totals totals) { itemHealed = 0f; globalHealedWhenSet = totals.healed; } void ITracker.Serialize(SerializationBuffer data) { data.Add(itemHealed); } void ITracker.Deserialize(SerializationBuffer data) { itemHealed = data.nextFloat; } } public interface IChanceTracker { void Attempted(); void Succedded(); } public interface IChanceBatchTracker : IChanceTracker { void Succedded(uint count); } public interface IDamageTracker { void AddDamage(float damage); } public interface IHealTracker { void AddHealing(float heal); } public interface IReductionTracker { void AddBlockedDamage(float blocked); } public interface IGoldTracker { void AddGold(float gold); } public interface ITracker { bool dirtyBit { get; set; } string[] GetTooltipArgs(in Totals totals); void Reset(in Totals totals); void Serialize(SerializationBuffer data); void Deserialize(SerializationBuffer data); } public class SkillTracker : ITracker, IDamageTracker, IHealTracker { private float itemDamage; private float itemHealed; private float globalDamageWhenSet; private float globalHealedWhenSet; public bool dirtyBit { get; set; } public void AddDamage(float damage) { itemDamage += damage; dirtyBit = true; } public void AddHealing(float heal) { itemHealed += heal; dirtyBit = true; } string[] ITracker.GetTooltipArgs(in Totals totals) { bool flag = itemDamage != 0f; bool flag2 = itemHealed != 0f; string text = null; if (flag) { text = Language.GetStringFormatted(TrackedDef.portionTooltip, new object[2] { null, TrackerStatics.PortionToString(itemDamage, totals.damageDealt - globalDamageWhenSet) }); } if (flag2) { if (flag) { text += "\n"; } text += Language.GetStringFormatted(TrackedDef.healTooltip, new object[2] { TrackerStatics.PortionToString(itemHealed, totals.healed - globalHealedWhenSet), null }); } if (text != null) { return new string[1] { text }; } return null; } void ITracker.Reset(in Totals totals) { itemDamage = 0f; itemHealed = 0f; globalDamageWhenSet = totals.damageDealt; globalHealedWhenSet = totals.healed; } void ITracker.Serialize(SerializationBuffer data) { data.Add(itemDamage); data.Add(itemHealed); } void ITracker.Deserialize(SerializationBuffer data) { itemDamage = data.nextFloat; itemHealed = data.nextFloat; } } public static class TrackerStatics { public static string IncreaseToString(float localNum, float totalNum) { if (totalNum == 0f) { return "NaN"; } if (localNum >= 0.999999f * totalNum) { return "Max"; } if (localNum < 0f) { return (100f * localNum / totalNum).ToString("f0") + "%"; } return (100f / (1f - localNum / totalNum) - 100f).ToString("f0") + "%"; } public static string ChanceToString(uint attempted, uint succedded) { if (attempted == 0) { return "NaN"; } float num = 100f * (float)succedded / (float)attempted; return num.ToString((num < 10f) ? "f1" : "f0") + "%"; } public static string ReductionToString(float localNum, float totalNum) { if (totalNum == 0f) { if (!(localNum > 0f)) { return "NaN"; } return "100%"; } if (localNum <= -0.999999f * totalNum) { return "Min"; } return (100f - 100f / (1f + localNum / totalNum)).ToString("f0") + "%"; } public static string PortionToString(float localNum, float totalNum) { if (totalNum == 0f) { return "NaN"; } return (100f * localNum / totalNum).ToString("f0") + "%"; } public static string RateToString(uint attempted, uint succedded) { if (attempted == 0) { return "NaN"; } return ((float)succedded / (float)attempted).ToString("f2"); } internal static void LogInvalid(Index index, ITracker tracker, string requiredTracker, [CallerMemberName] string caller = "") { log.error($"Invalid call '{caller}' on {index} ({tracker}). Has to be a {requiredTracker} tracker"); } } public class WatchTracker : DamageTracker { private bool globalset; public void AddBrokenDamage(DamageTracker notBrokenWatchTracker) { AddDamage(notBrokenWatchTracker.itemDamage); if (!globalset) { globalDamageWhenSet = notBrokenWatchTracker.globalDamageWhenSet; globalset = true; } } public override void Reset(in Totals totals) { base.Reset(in totals); globalset = false; } public override void Serialize(SerializationBuffer data) { base.Serialize(data); data.Add(globalDamageWhenSet); } public override void Deserialize(SerializationBuffer data) { base.Deserialize(data); globalDamageWhenSet = data.nextFloat; } } } namespace ItemStatistics.Networking { public class BlastMessage { private readonly Index inflictorIndex; internal BlastAttackDamageInfo blastAttack; public BlastMessage(in BlastAttackDamageInfo blastAttack, Index inflictorIndex) { //IL_000f: Unknown result type (might be due to invalid IL or missing references) //IL_0014: Unknown result type (might be due to invalid IL or missing references) this.inflictorIndex = inflictorIndex; this.blastAttack = blastAttack; } public BlastMessage(NetworkReader reader) { inflictorIndex = Index.Deserialize(reader); ((BlastAttackDamageInfo)(ref blastAttack)).Read(reader); blastAttack.inflictor = ProcComponent.CreateInflictor(inflictorIndex); } internal static void Serialize(NetworkWriter writer, BlastMessage value) { Index.Serialize(writer, value.inflictorIndex); ((BlastAttackDamageInfo)(ref value.blastAttack)).Write(writer); } internal static BlastMessage Deserialize(NetworkReader reader) { return new BlastMessage(reader); } } internal static class NetworkHooks { private delegate void BlastAttack_ClientReportDamage(in BlastAttackDamageInfo blastAttack); private static Index inflictorIndex; private static bool shouldClientSendMessage { get { if (TrackModdedUsers.isModdedServer) { return inflictorIndex.isNotNone; } return false; } } internal static void Apply() { //IL_002c: Unknown result type (might be due to invalid IL or missing references) //IL_0036: Expected O, but got Unknown HookManager.Hook(MethodBase.GetMethodFromHandle((RuntimeMethodHandle)/*OpCode not supported: LdMemberToken*/), (Delegate)new <>A{00000018}<BlastAttack_ClientReportDamage, BlastAttackDamageInfo>(NetworkMessage_On_BlastAttack_ClientReportDamage)); HookManager.Hook(MethodBase.GetMethodFromHandle((RuntimeMethodHandle)/*OpCode not supported: LdMemberToken*/), new Manipulator(HeadStompers_IL_HeadstompersFall_DoStompExplosionAuthority)); } private static void NetworkMessage_On_BlastAttack_ClientReportDamage(BlastAttack_ClientReportDamage orig, in BlastAttackDamageInfo blastAttack) { if (!shouldClientSendMessage) { orig(in blastAttack); } else if (Object.op_Implicit((Object)(object)blastAttack.hitHealthComponent)) { TrackModdedUsers.localUser.CallCmdBlastAttack(new BlastMessage(in blastAttack, inflictorIndex)); } } private static void HeadStompers_IL_HeadstompersFall_DoStompExplosionAuthority(ILContext il) { //IL_0001: Unknown result type (might be due to invalid IL or missing references) //IL_0007: Expected O, but got Unknown //IL_003b: Unknown result type (might be due to invalid IL or missing references) //IL_009e: Unknown result type (might be due to invalid IL or missing references) //IL_00c0: Unknown result type (might be due to invalid IL or missing references) ILCursor val = new ILCursor(il); if (val.TryGotoNext(new Func<Instruction, bool>[1] { (Instruction a) => ILPatternMatchingExt.MatchStfld<BlastAttack>(a, "inflictor") })) { val.Emit(OpCodes.Pop); val.EmitDelegate<Func<GameObject>>((Func<GameObject>)delegate { //IL_001d: Unknown result type (might be due to invalid IL or missing references) if (NetworkServer.active) { return ProcComponent.CreateInflictor(Items.FallBoots); } inflictorIndex = Items.FallBoots.itemIndex; return null; }); if (val.TryGotoNext((MoveType)2, new Func<Instruction, bool>[1] { (Instruction a) => ILPatternMatchingExt.MatchCallOrCallvirt(a, typeof(BlastAttack), "Fire") })) { val.Emit(OpCodes.Ldsflda, typeof(NetworkHooks).GetField("inflictorIndex", BindingFlags.DeclaredOnly | BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic)); val.Emit(OpCodes.Initobj, typeof(Index)); return; } } val.LogErrorCaller("Headstompers"); } } public class SerializationBuffer : IEnumerable<Index>, IEnumerable { [StructLayout(LayoutKind.Explicit)] private struct IntFloat { [FieldOffset(0)] private float floatValue; [FieldOffset(0)] private uint intValue; internal static float IntToFloat(uint value) { IntFloat intFloat = default(IntFloat); intFloat.intValue = value; return intFloat.floatValue; } internal static uint FloatToInt(float value) { IntFloat intFloat = default(IntFloat); intFloat.floatValue = value; return intFloat.intValue; } } private static readonly SerializationBuffer staticBuffer = new SerializationBuffer(4u, 4u); private Index[] indexes; private uint[] bits; private uint indexSize; private uint bitsSize; private uint bitPos; public uint nextInt => bits[bitPos++]; public float nextFloat => IntFloat.IntToFloat(nextInt); public bool hasSomething => indexSize != 0; public void Add(float num) { Add(IntFloat.FloatToInt(num)); } public void Add(uint num) { uint num2 = bitsSize + 1; if (num2 >= bits.Length) { int num3 = ((bits.Length == 0) ? 4 : (2 * bits.Length)); if (num3 < num2) { num3 = (int)num2; } Array.Resize(ref bits, num3); } bits[bitsSize] = num; bitsSize = num2; } internal void Add(Index index) { uint num = indexSize + 1; if (num >= indexes.Length) { int num2 = ((indexes.Length == 0) ? 4 : (2 * indexes.Length)); if (num2 < num) { num2 = (int)num; } Array.Resize(ref indexes, num2); } indexes[indexSize] = index; indexSize = num; } internal static void Serialize(NetworkWriter writer, SerializationBuffer value) { writer.WritePackedUInt32(value.indexSize); writer.WritePackedUInt32(value.bitsSize); for (uint num = 0u; num < value.indexSize; num++) { Index.Serialize(writer, value.indexes[num]); } for (uint num2 = 0u; num2 < value.bitsSize; num2++) { writer.WritePackedUInt32(value.bits[num2]); } } internal static SerializationBuffer Deserialize(NetworkReader reader) { SerializationBuffer serializationBuffer = staticBuffer; serializationBuffer.Clear(); uint num = reader.ReadPackedUInt32(); uint num2 = reader.ReadPackedUInt32(); for (uint num3 = 0u; num3 < num; num3++) { serializationBuffer.Add(Index.Deserialize(reader)); } for (uint num4 = 0u; num4 < num2; num4++) { serializationBuffer.Add(reader.ReadPackedUInt32()); } return serializationBuffer; } internal void Clear() { indexSize = 0u; bitsSize = 0u; bitPos = 0u; } internal static SerializationBuffer GetBuffer() { return staticBuffer; } public IEnumerator<Index> GetEnumerator() { int i = 0; while (i < indexSize) { yield return indexes[i]; int num = i + 1; i = num; } } IEnumerator IEnumerable.GetEnumerator() { return GetEnumerator(); } private SerializationBuffer(uint initialIndexCapacity, uint initialBitsCapacity) { indexes = new Index[initialIndexCapacity]; bits = new uint[initialBitsCapacity]; } } } namespace ItemStatistics.Hooks { public static class DotTracking { internal static List<ProcChain> dotChainListParameter; private static ProcChain dotChainParameter; private static MethodInfo modifiedGetDotStackTotalDurationForIndex; internal static void Apply() { //IL_002c: Unknown result type (might be due to invalid IL or missing references) //IL_0036: Expected O, but got Unknown //IL_0047: Unknown result type (might be due to invalid IL or missing references) //IL_0051: Expected O, but got Unknown HookManager.Hook(MethodBase.GetMethodFromHandle((RuntimeMethodHandle)/*OpCode not supported: LdMemberToken*/), (Delegate)new Action<Action<DotController, DotStack>, DotController, DotStack>(AddTree_On_DotController_OnDotStackAddedServer)); HookManager.Hook(MethodBase.GetMethodFromHandle((RuntimeMethodHandle)/*OpCode not supported: LdMemberToken*/), new Manipulator(Passer_IL_DotController_EvaluateDotStacksForType)); HookManager.Hook(MethodBase.GetMethodFromHandle((RuntimeMethodHandle)/*OpCode not supported: LdMemberToken*/), new Manipulator(CollectProcChain_IL_DotController_AddPendingDamageEntry)); HookManager.Hook(MethodBase.GetMethodFromHandle((RuntimeMethodHandle)/*OpCode not supported: LdMemberToken*/), (Delegate)new Action<Action<DotStack>, DotStack>(On_DotStack_Reset)); HookManager.Hook(MethodBase.GetMethodFromHandle((RuntimeMethodHandle)/*OpCode not supported: LdMemberToken*/), (Delegate)new Action<Action<PendingDamage>, PendingDamage>(On_PendingDamage_Reset)); DoNoxious(); } private static void AddTree_On_DotController_OnDotStackAddedServer(Action<DotController, DotStack> orig, DotController self, DotStack dotStack) { orig(self, dotStack); if (CurrentContext.procChain != null) { dotStack.dotStackChain = CurrentContext.procChain; } } private static void Passer_IL_DotController_EvaluateDotStacksForType(ILContext il) { //IL_0001: Unknown result type (might be due to invalid IL or missing references) //IL_0007: Expected O, but got Unknown //IL_003b: 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) //IL_0067: Unknown result type (might be due to invalid IL or missing references) //IL_00bd: Unknown result type (might be due to invalid IL or missing references) //IL_00c9: Unknown result type (might be due to invalid IL or missing references) //IL_011e: Unknown result type (might be due to invalid IL or missing references) //IL_012a: Unknown result type (might be due to invalid IL or missing references) //IL_014a: Unknown result type (might be due to invalid IL or missing references) //IL_019d: Unknown result type (might be due to invalid IL or missing references) //IL_01a9: Unknown result type (might be due to invalid IL or missing references) ILCursor val = new ILCursor(il); if (val.TryGotoNext(new Func<Instruction, bool>[1] { (Instruction a) => ILPatternMatchingExt.MatchLdfld<DotStack>(a, "timer") })) { val.Emit(OpCodes.Dup); val.Emit(OpCodes.Ldfld, typeof(DotStack).GetField("dotStackChain")); val.Emit(OpCodes.Stsfld, typeof(DotTracking).GetField("dotChainParameter", BindingFlags.DeclaredOnly | BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic)); if (val.TryGotoNext((MoveType)2, new Func<Instruction, bool>[1] { (Instruction a) => ILPatternMatchingExt.MatchCallOrCallvirt<DotController>(a, "AddPendingDamageEntry") })) { val.Emit(OpCodes.Ldnull); val.Emit(OpCodes.Stsfld, typeof(DotTracking).GetField("dotChainParameter", BindingFlags.DeclaredOnly | BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic)); if (val.TryGotoNext(new Func<Instruction, bool>[1] { (Instruction a) => ILPatternMatchingExt.MatchLdfld<PendingDamage>(a, "totalDamage") })) { val.Emit(OpCodes.Dup); val.Emit(OpCodes.Ldfld, typeof(PendingDamage).GetField("pendingDotChainList")); val.Emit(OpCodes.Stsfld, typeof(DotTracking).GetField("dotChainListParameter", BindingFlags.DeclaredOnly | BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic)); if (val.TryGotoNext((MoveType)2, new Func<Instruction, bool>[1] { (Instruction a) => ILPatternMatchingExt.MatchCallOrCallvirt<HealthComponent>(a, "TakeDamage") })) { val.Emit(OpCodes.Ldnull); val.Emit(OpCodes.Stsfld, typeof(DotTracking).GetField("dotChainListParameter", BindingFlags.DeclaredOnly | BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic)); return; } } } } val.LogErrorCaller(""); } private static void CollectProcChain_IL_DotController_AddPendingDamageEntry(ILContext il) { //IL_0007: Unknown result type (might be due to invalid IL or missing references) //IL_000d: Expected O, but got Unknown //IL_0084: Unknown result type (might be due to invalid IL or missing references) //IL_0091: Unknown result type (might be due to invalid IL or missing references) //IL_00a3: Unknown result type (might be due to invalid IL or missing references) //IL_013a: Unknown result type (might be due to invalid IL or missing references) ILCursor val = new ILCursor(il); int iIndex = 1; ILCursor[] array = default(ILCursor[]); val.FindNext(ref array, new Func<Instruction, bool>[2] { (Instruction a) => ILPatternMatchingExt.MatchLdloc(a, ref iIndex), (Instruction a) => ILPatternMatchingExt.MatchLdcI4(a, 1) }); if (val.TryGotoNext(new Func<Instruction, bool>[1] { (Instruction a) => ILPatternMatchingExt.MatchStfld<PendingDamage>(a, "totalDamage") })) { val.Emit(OpCodes.Ldarg, 0); val.Emit(OpCodes.Ldloc, iIndex); val.Emit<List<PendingDamage>>(OpCodes.Callvirt, "get_Item"); val.EmitDelegate<Action<PendingDamage>>((Action<PendingDamage>)delegate(PendingDamage pending) { if (dotChainParameter != null) { pending.pendingDotChainList.Add(dotChainParameter); } }); } else { val.LogErrorCaller("1"); } MethodReference val2 = default(MethodReference); if (val.TryGotoNext((MoveType)2, new Func<Instruction, bool>[2] { (Instruction a) => ILPatternMatchingExt.MatchLdsfld<DotController>(a, "pendingDamagePool"), (Instruction a) => ILPatternMatchingExt.MatchCallOrCallvirt(a, ref val2) })) { val.Emit(OpCodes.Dup); val.EmitDelegate<Action<PendingDamage>>((Action<PendingDamage>)delegate(PendingDamage pending) { List<ProcChain> list = (pending.pendingDotChainList = CollectionPool<ProcChain, List<ProcChain>>.RentCollection()); if (dotChainParameter != null) { list.Add(dotChainParameter); } }); } else { val.LogErrorCaller("2"); } } private static void On_DotStack_Reset(Action<DotStack> orig, DotStack self) { orig(self); self.dotStackChain = null; } private static void On_PendingDamage_Reset(Action<PendingDamage> orig, PendingDamage self) { orig(self); List<ProcChain> pendingDotChainList = self.pendingDotChainList; if (pendingDotChainList != null) { CollectionPool<ProcChain, List<ProcChain>>.ReturnCollection(pendingDotChainList); } self.pendingDotChainList = null; } private static void DoNoxious() { //IL_0014: Unknown result type (might be due to invalid IL or missing references) //IL_001a: Expected O, but got Unknown //IL_0040: Unknown result type (might be due to invalid IL or missing references) //IL_004a: Expected O, but got Unknown //IL_0045: Unknown result type (might be due to invalid IL or missing references) //IL_004f: Expected O, but got Unknown //IL_0070: Unknown result type (might be due to invalid IL or missing references) //IL_0076: Expected O, but got Unknown //IL_0077: Unknown result type (might be due to invalid IL or missing references) //IL_007d: Expected O, but got Unknown //IL_007e: Unknown result type (might be due to invalid IL or missing references) //IL_008a: Unknown result type (might be due to invalid IL or missing references) //IL_0096: Unknown result type (might be due to invalid IL or missing references) //IL_00d8: Unknown result type (might be due to invalid IL or missing references) //IL_00e5: Unknown result type (might be due to invalid IL or missing references) //IL_00f1: Unknown result type (might be due to invalid IL or missing references) //IL_00fe: Unknown result type (might be due to invalid IL or missing references) //IL_010f: Unknown result type (might be due to invalid IL or missing references) //IL_011b: Unknown result type (might be due to invalid IL or missing references) //IL_0155: Unknown result type (might be due to invalid IL or missing references) //IL_015f: Expected O, but got Unknown //IL_0170: Unknown result type (might be due to invalid IL or missing references) //IL_017a: Expected O, but got Unknown try { DynamicMethodDefinition val = new DynamicMethodDefinition((MethodBase)HookManager.GetMethod(typeof(DotController), "GetDotStackTotalDurationForIndex")); try { ((MethodReference)val.Definition).Parameters.Add(new ParameterDefinition("procChain", (ParameterAttributes)2, (TypeReference)new ByReferenceType(val.Module.ImportReference(typeof(ProcChain))))); typeof(DynamicMethodDefinition).GetProperty("OriginalMethod").SetValue(val, null); ILContext val2 = new ILContext(val.Definition); ILCursor val3 = new ILCursor(val2); val3.Emit(OpCodes.Ldarg_3); val3.Emit(OpCodes.Ldnull); val3.Emit(OpCodes.Stind_Ref); VariableDefinition val4 = val3.AddNewVariable<DotStack>(); val3.GotoNext(new Func<Instruction, bool>[1] { (Instruction a) => ILPatternMatchingExt.MatchLdfld<DotStack>(a, "totalDuration") }); val3.Emit(OpCodes.Stloc, val4); val3.Emit(OpCodes.Ldarg_3); val3.Emit(OpCodes.Ldloc, val4); val3.Emit<DotStack>(OpCodes.Ldfld, "dotStackChain"); val3.Emit(OpCodes.Stind_Ref); val3.Emit(OpCodes.Ldloc, val4); val2.Dispose(); modifiedGetDotStackTotalDurationForIndex = val.Generate(); } finally { ((IDisposable)val)?.Dispose(); } HookManager.Hook(MethodBase.GetMethodFromHandle((RuntimeMethodHandle)/*OpCode not supported: LdMemberToken*/), new Manipulator(GetChain_IL_CharacterBody_TriggerEnemyDebuffs)); HookManager.Hook(MethodBase.GetMethodFromHandle((RuntimeMethodHandle)/*OpCode not supported: LdMemberToken*/), new Manipulator(ApplyChain_IL_VineOrb_OnArrival)); } catch (Exception ex) { log.error("Failed Noxious Thorn\n" + ex); } } private static void GetChain_IL_CharacterBody_TriggerEnemyDebuffs(ILContext il) { //IL_0001: Unknown result type (might be due to invalid IL or missing references) //IL_0007: Expected O, but got Unknown //IL_0042: Unknown result type (might be due to invalid IL or missing references) //IL_004e: Unknown result type (might be due to invalid IL or missing references) //IL_0094: Unknown result type (might be due to invalid IL or missing references) //IL_00a8: Unknown result type (might be due to invalid IL or missing references) //IL_010d: Unknown result type (might be due to invalid IL or missing references) //IL_011f: Unknown result type (might be due to invalid IL or missing references) //IL_012c: Unknown result type (might be due to invalid IL or missing references) ILCursor val = new ILCursor(il); VariableDefinition val2 = val.AddNewVariable<ProcChain>(); if (val.TryGotoNext(new Func<Instruction, bool>[1] { (Instruction a) => ILPatternMatchingExt.MatchLdfld<BuffDef>(a, "isDOT") })) { val.Emit(OpCodes.Ldnull); val.Emit(OpCodes.Stloc, val2); if (val.TryGotoNext(new Func<Instruction, bool>[1] { (Instruction a) => ILPatternMatchingExt.MatchCallOrCallvirt<DotController>(a, "GetDotStackTotalDurationForIndex") })) { val.Emit(OpCodes.Ldloca, val2); val.Remove(); val.Emit(OpCodes.Call, (MethodBase)modifiedGetDotStackTotalDurationForIndex); int infoIndex = -1; if (val.TryGotoNext(new Func<Instruction, bool>[1] { (Instruction a) => ILPatternMatchingExt.MatchStfld<SplitDebuffInformation>(a, "attacker") }) && val.TryGotoPrev(new Func<Instruction, bool>[1] { (Instruction a) => ILPatternMatchingExt.MatchLdloca(a, ref infoIndex) })) { val.Emit(OpCodes.Ldloca, infoIndex); val.Emit(OpCodes.Ldloc, val2); val.Emit<SplitDebuffInformation>(OpCodes.Stfld, "procChain"); return; } } } val.LogErrorCaller("Noxious Thorn get chain"); } private static void ApplyChain_IL_VineOrb_OnArrival(ILContext il) { //IL_0001: Unknown result type (might be due to invalid IL or missing references) //IL_0007: Expected O, but got Unknown //IL_0043: Unknown result type (might be due to invalid IL or missing references) //IL_0060: Unknown result type (might be due to invalid IL or missing references) //IL_009b: Unknown result type (might be due to invalid IL or missing references) //IL_00