Decompiled source of BuddyClimb v0.1.11

plugins/com.github.LandmineHQ.BuddyClimb.dll

Decompiled 2 weeks ago
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Diagnostics;
using System.Diagnostics.CodeAnalysis;
using System.IO;
using System.Reflection;
using System.Runtime.CompilerServices;
using System.Runtime.Versioning;
using System.Security;
using System.Security.Permissions;
using System.Threading;
using BepInEx;
using BepInEx.Bootstrap;
using BepInEx.Configuration;
using BepInEx.Logging;
using BuddyClimb.Compatibility;
using BuddyClimb.Configuration;
using BuddyClimb.Gameplay;
using BuddyClimb.Localization;
using BuddyClimb.Patches;
using HarmonyLib;
using Microsoft.CodeAnalysis;
using Photon.Pun;
using UnityEngine;
using Zorro.Core;
using Zorro.Core.Serizalization;

[assembly: CompilationRelaxations(8)]
[assembly: RuntimeCompatibility(WrapNonExceptionThrows = true)]
[assembly: Debuggable(DebuggableAttribute.DebuggingModes.IgnoreSymbolStoreSequencePoints)]
[assembly: IgnoresAccessChecksTo("Assembly-CSharp")]
[assembly: TargetFramework(".NETStandard,Version=v2.1", FrameworkDisplayName = ".NET Standard 2.1")]
[assembly: AssemblyCompany("com.github.LandmineHQ.BuddyClimb")]
[assembly: AssemblyConfiguration("Release")]
[assembly: AssemblyFileVersion("0.1.11.0")]
[assembly: AssemblyInformationalVersion("0.1.11+f6b72a9e584034704f8ebe1b546b7638e440ac71")]
[assembly: AssemblyProduct("com.github.LandmineHQ.BuddyClimb")]
[assembly: AssemblyTitle("BuddyClimb")]
[assembly: AssemblyMetadata("RepositoryUrl", "https://github.com/LandmineHQ/PeakPunch")]
[assembly: SecurityPermission(SecurityAction.RequestMinimum, SkipVerification = true)]
[assembly: AssemblyVersion("0.1.11.0")]
[module: UnverifiableCode]
[module: RefSafetyRules(11)]
namespace System.Runtime.CompilerServices
{
	[CompilerGenerated]
	[Microsoft.CodeAnalysis.Embedded]
	[AttributeUsage(AttributeTargets.Class | AttributeTargets.Property | AttributeTargets.Field | AttributeTargets.Event | AttributeTargets.Parameter | AttributeTargets.ReturnValue | AttributeTargets.GenericParameter, AllowMultiple = false, Inherited = false)]
	internal sealed class NullableAttribute : Attribute
	{
		public readonly byte[] NullableFlags;

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

		public NullableAttribute(byte[] P_0)
		{
			NullableFlags = P_0;
		}
	}
	[CompilerGenerated]
	[Microsoft.CodeAnalysis.Embedded]
	[AttributeUsage(AttributeTargets.Class | AttributeTargets.Struct | AttributeTargets.Method | AttributeTargets.Interface | AttributeTargets.Delegate, AllowMultiple = false, Inherited = false)]
	internal sealed class NullableContextAttribute : Attribute
	{
		public readonly byte Flag;

		public NullableContextAttribute(byte P_0)
		{
			Flag = P_0;
		}
	}
	[CompilerGenerated]
	[Microsoft.CodeAnalysis.Embedded]
	[AttributeUsage(AttributeTargets.Module, AllowMultiple = false, Inherited = false)]
	internal sealed class RefSafetyRulesAttribute : Attribute
	{
		public readonly int Version;

		public RefSafetyRulesAttribute(int P_0)
		{
			Version = P_0;
		}
	}
}
namespace BepInEx
{
	[AttributeUsage(AttributeTargets.Class, Inherited = false, AllowMultiple = false)]
	[Conditional("CodeGeneration")]
	[Microsoft.CodeAnalysis.Embedded]
	internal sealed class BepInAutoPluginAttribute : Attribute
	{
		public BepInAutoPluginAttribute(string? id = null, string? name = null, string? version = null)
		{
		}
	}
}
namespace BepInEx.Preloader.Core.Patching
{
	[AttributeUsage(AttributeTargets.Class, Inherited = false, AllowMultiple = false)]
	[Conditional("CodeGeneration")]
	[Microsoft.CodeAnalysis.Embedded]
	internal sealed class PatcherAutoPluginAttribute : Attribute
	{
		public PatcherAutoPluginAttribute(string? id = null, string? name = null, string? version = null)
		{
		}
	}
}
namespace Microsoft.CodeAnalysis
{
	[Microsoft.CodeAnalysis.Embedded]
	internal sealed class EmbeddedAttribute : Attribute
	{
	}
}
namespace BuddyClimb
{
	[BepInDependency(/*Could not decode attribute arguments.*/)]
	[BepInPlugin("com.github.LandmineHQ.BuddyClimb", "BuddyClimb", "0.1.11")]
	public class Plugin : BaseUnityPlugin
	{
		public const string Id = "com.github.LandmineHQ.BuddyClimb";

		internal static ManualLogSource Log { get; private set; }

		public static string Name => "BuddyClimb";

		public static string Version => "0.1.11";

		private void Awake()
		{
			//IL_0026: Unknown result type (might be due to invalid IL or missing references)
			Log = ((BaseUnityPlugin)this).Logger;
			BuddyClimbConfig.Bind(((BaseUnityPlugin)this).Config);
			BuddyClimbConfig.EnableHotReload(((BaseUnityPlugin)this).Config);
			new Harmony("com.github.LandmineHQ.BuddyClimb").PatchAll(typeof(Plugin).Assembly);
			if (ModCompatibility.IsPiggybackLoaded)
			{
				Log.LogInfo((object)"Piggyback detected; BuddyClimb carry spectate patches are disabled.");
			}
			Log.LogInfo((object)("Plugin " + Name + " is loaded!"));
		}

		private void Update()
		{
			CarriedPlayerDropper.Update();
		}

		private void OnDestroy()
		{
			BuddyClimbConfig.DisableHotReload();
		}
	}
}
namespace BuddyClimb.Patches
{
	[HarmonyPatch]
	internal static class CarrySpectatePatch
	{
		[HarmonyPrepare]
		private static bool Prepare()
		{
			return !ModCompatibility.IsPiggybackLoaded;
		}

		[HarmonyPatch(typeof(MainCameraMovement), "LateUpdate")]
		[HarmonyPostfix]
		private static void LateUpdatePostfix(MainCameraMovement __instance)
		{
			if (TryGetBuddyClimbSpectateTarget(out Character spectateTarget) && !__instance.isGodCam && !__instance.isSpectating)
			{
				MainCamera cam = __instance.cam;
				if (!((Object)(object)cam == (Object)null) && !((Object)(object)cam.camOverride != (Object)null))
				{
					MainCameraMovement.specCharacter = spectateTarget;
					__instance.Spectate();
					__instance.isSpectating = true;
				}
			}
		}

		[HarmonyPatch(typeof(MainCameraMovement), "HandleSpecSelection")]
		[HarmonyPrefix]
		private static bool HandleSpecSelectionPrefix(MainCameraMovement __instance, ref bool __result)
		{
			if (!TryGetBuddyClimbSpectateTarget(out Character spectateTarget))
			{
				return true;
			}
			MainCameraMovement.specCharacter = spectateTarget;
			__result = true;
			return false;
		}

		private static bool TryGetBuddyClimbSpectateTarget(out Character spectateTarget)
		{
			spectateTarget = null;
			Character localCharacter = Character.localCharacter;
			if ((Object)(object)localCharacter == (Object)null || (Object)(object)localCharacter.data == (Object)null || !localCharacter.data.isCarried || (Object)(object)localCharacter.data.carrier == (Object)null || localCharacter.data.dead || localCharacter.data.fullyPassedOut || !CharacterCarryingPatch.IsBuddyClimbCarried(localCharacter))
			{
				return false;
			}
			spectateTarget = localCharacter;
			return true;
		}
	}
	[HarmonyPatch(typeof(CharacterBackpackHandler))]
	internal static class CharacterBackpackHandlerPatch
	{
		[HarmonyPatch("LateUpdate")]
		[HarmonyPostfix]
		private static void LateUpdatePostfix(CharacterBackpackHandler __instance)
		{
			CarriedBackpackVisuals.Update(__instance);
		}
	}
	[HarmonyPatch(typeof(CharacterCarrying))]
	internal static class CharacterCarryingPatch
	{
		private static readonly HashSet<int> BuddyClimbCarriedViewIds = new HashSet<int>();

		internal static bool IsBuddyClimbCarried(Character character)
		{
			if ((Object)(object)((character != null) ? ((MonoBehaviourPun)character).photonView : null) != (Object)null)
			{
				return BuddyClimbCarriedViewIds.Contains(((MonoBehaviourPun)character).photonView.ViewID);
			}
			return false;
		}

		[HarmonyPatch("Update")]
		[HarmonyPrefix]
		private static bool UpdatePrefix(CharacterCarrying __instance)
		{
			Character val = __instance.character ?? ((Component)__instance).GetComponent<Character>();
			if ((Object)(object)val == (Object)null || (Object)(object)((MonoBehaviourPun)val).photonView == (Object)null)
			{
				return true;
			}
			Character carriedPlayer = val.data.carriedPlayer;
			if ((Object)(object)carriedPlayer == (Object)null || !IsBuddyClimbCarried(carriedPlayer))
			{
				return true;
			}
			if ((carriedPlayer.data.dead || val.data.fullyPassedOut || val.data.dead) && val.refs.view.IsMine)
			{
				BuddyClimbDiagnostics.LogCarry("CharacterCarrying.Update dropping BuddyClimb carried player because carry state became invalid: " + BuddyClimbDiagnostics.DescribeViews(val, carriedPlayer));
				__instance.Drop(carriedPlayer);
			}
			return false;
		}

		[HarmonyPatch("RPCA_Drop")]
		[HarmonyPrefix]
		private static bool RPCA_DropPrefix(PhotonView targetView)
		{
			BuddyClimbDiagnostics.LogCarry($"RPCA_Drop prefix received targetView={(((Object)(object)targetView != (Object)null) ? targetView.ViewID : (-1))}");
			if ((Object)(object)targetView == (Object)null)
			{
				return true;
			}
			Character component = ((Component)targetView).GetComponent<Character>();
			if ((Object)(object)component != (Object)null && IsBuddyClimbCarried(component))
			{
				BuddyClimbRemotePassOutSync.RestoreRemoteCarryDrop(component, "incoming RPCA_Drop");
			}
			BuddyClimbCarriedViewIds.Remove(targetView.ViewID);
			return true;
		}

		[HarmonyPatch("RPCA_Drop")]
		[HarmonyPostfix]
		private static void RPCA_DropPostfix(PhotonView targetView)
		{
			if ((Object)(object)targetView != (Object)null)
			{
				Character component = ((Component)targetView).GetComponent<Character>();
				BuddyClimbDiagnostics.LogCarry("RPCA_Drop postfix target: " + BuddyClimbDiagnostics.Describe(component));
				CarryInteractionProxy.Disable(component);
				CarriedBackpackVisuals.HideIfForcedVisible(component);
			}
		}

		[HarmonyPatch("RPCA_StartCarry")]
		[HarmonyPrefix]
		private static bool RPCA_StartCarryPrefix(CharacterCarrying __instance, PhotonView targetView)
		{
			Character val = __instance.character ?? ((Component)__instance).GetComponent<Character>();
			Character val2 = (((Object)(object)targetView != (Object)null) ? ((Component)targetView).GetComponent<Character>() : null);
			BuddyClimbDiagnostics.LogCarry($"RPCA_StartCarry prefix received targetView={(((Object)(object)targetView != (Object)null) ? targetView.ViewID : (-1))}: {BuddyClimbDiagnostics.DescribeViews(val, val2)}");
			if ((Object)(object)targetView == (Object)null)
			{
				BuddyClimbDiagnostics.LogCarry("RPCA_StartCarry prefix returning false because targetView is null.");
				return false;
			}
			Character val3 = val;
			if ((Object)(object)val3 == (Object)null)
			{
				BuddyClimbDiagnostics.LogCarry("RPCA_StartCarry prefix returning false because carrierCharacter is null.");
				return false;
			}
			Character val4 = val2;
			if ((Object)(object)val4 == (Object)null)
			{
				BuddyClimbDiagnostics.LogCarry("RPCA_StartCarry prefix returning false because carriedCharacter is null.");
				return false;
			}
			if (val4.data.fullyPassedOut || val4.data.dead)
			{
				BuddyClimbDiagnostics.LogCarry("RPCA_StartCarry prefix allowing vanilla path for unconscious/dead carried character: " + BuddyClimbDiagnostics.Describe(val4));
				return true;
			}
			Character carriedPlayer = val3.data.carriedPlayer;
			if ((Object)(object)carriedPlayer == (Object)(object)val4)
			{
				BuddyClimbDiagnostics.LogCarry("RPCA_StartCarry prefix applying idempotent same-link state: " + BuddyClimbDiagnostics.DescribeViews(val3, val4));
				ApplyBuddyClimbCarryState(val3, val4);
				return false;
			}
			if ((Object)(object)carriedPlayer != (Object)null)
			{
				BuddyClimbDiagnostics.LogCarry("RPCA_StartCarry prefix found existing carriedPlayer=" + BuddyClimbDiagnostics.Describe(carriedPlayer) + " before applying new state.");
				if (!TryClearExistingBuddyClimbCarry(val3, carriedPlayer, val4))
				{
					BuddyClimbDiagnostics.LogCarry("RPCA_StartCarry prefix returning false because existing carried player is not BuddyClimb-owned stale state: " + BuddyClimbDiagnostics.DescribeViews(val3, val4));
					return false;
				}
			}
			BuddyClimbDiagnostics.LogCarry("RPCA_StartCarry prefix applying BuddyClimb carry state: " + BuddyClimbDiagnostics.DescribeViews(val3, val4));
			ApplyBuddyClimbCarryState(val3, val4);
			return false;
		}

		private static bool TryClearExistingBuddyClimbCarry(Character carrierCharacter, Character existingCarriedPlayer, Character nextCarriedCharacter)
		{
			if (!IsBuddyClimbCarried(existingCarriedPlayer) || (Object)(object)existingCarriedPlayer.data.carrier != (Object)(object)carrierCharacter)
			{
				Plugin.Log.LogDebug((object)("Ignoring BuddyClimb start carry for " + nextCarriedCharacter.characterName + " because " + carrierCharacter.characterName + " is already carrying " + existingCarriedPlayer.characterName + "."));
				return false;
			}
			Plugin.Log.LogDebug((object)("Clearing stale BuddyClimb carry state for " + existingCarriedPlayer.characterName + " before carrying " + nextCarriedCharacter.characterName + "."));
			existingCarriedPlayer.refs.carriying.ToggleCarryPhysics(false);
			existingCarriedPlayer.data.isCarried = false;
			existingCarriedPlayer.data.carrier = null;
			carrierCharacter.data.carriedPlayer = null;
			BuddyClimbRemotePassOutSync.RestoreRemoteCarryDrop(existingCarriedPlayer, "stale BuddyClimb carry cleanup");
			BuddyClimbCarriedViewIds.Remove(((MonoBehaviourPun)existingCarriedPlayer).photonView.ViewID);
			CarryInteractionProxy.Disable(existingCarriedPlayer);
			CarriedBackpackVisuals.HideIfForcedVisible(existingCarriedPlayer);
			foreach (Character allPlayerCharacter in PlayerHandler.GetAllPlayerCharacters())
			{
				allPlayerCharacter.refs.afflictions.UpdateWeight();
			}
			return true;
		}

		private static void ApplyBuddyClimbCarryState(Character carrierCharacter, Character carriedCharacter)
		{
			carriedCharacter.refs.carriying.ToggleCarryPhysics(true);
			carriedCharacter.data.isCarried = true;
			carrierCharacter.data.carriedPlayer = carriedCharacter;
			carriedCharacter.data.carrier = carrierCharacter;
			BuddyClimbCarriedViewIds.Add(((MonoBehaviourPun)carriedCharacter).photonView.ViewID);
			foreach (Character allPlayerCharacter in PlayerHandler.GetAllPlayerCharacters())
			{
				allPlayerCharacter.refs.afflictions.UpdateWeight();
			}
			BuddyClimbDiagnostics.LogCarry("Applied BuddyClimb carry state: " + BuddyClimbDiagnostics.DescribeViews(carrierCharacter, carriedCharacter));
		}

		[HarmonyPatch("RPCA_StartCarry")]
		[HarmonyPostfix]
		private static void RPCA_StartCarryPostfix(PhotonView targetView)
		{
			if (!((Object)(object)targetView != (Object)null))
			{
				return;
			}
			Character component = ((Component)targetView).GetComponent<Character>();
			if (component != null && component.data.isCarried)
			{
				BuddyClimbDiagnostics.LogCarry("RPCA_StartCarry postfix sees carried state true: " + BuddyClimbDiagnostics.Describe(component));
				CarryInteractionProxy.Enable(component);
				if (component.IsLocal)
				{
					CarriedBackpackVisuals.Update(((Component)component).GetComponent<CharacterBackpackHandler>());
				}
			}
		}
	}
	[HarmonyPatch(typeof(CharacterInteractible))]
	internal static class CharacterInteractiblePatch
	{
		[HarmonyPatch("GetInteractionText")]
		[HarmonyPostfix]
		private static void GetInteractionTextPatch(ref string __result, CharacterInteractible __instance)
		{
			Character localCharacter = Character.localCharacter;
			if (IsBuddyClimbDropInteraction(__instance.character, localCharacter))
			{
				__result = string.Empty;
			}
			else if (!(__result != string.Empty) && CanStartClimb(__instance.character, localCharacter))
			{
				BuddyClimbTextKey key = (BackpackCarryTransfer.WillDropCarriedBackpack(__instance.character, localCharacter) ? BuddyClimbTextKey.ClimbOnTeammateDropBackpack : BuddyClimbTextKey.ClimbOnTeammate);
				__result = BuddyClimbLocalization.Get(key);
			}
		}

		[HarmonyPatch("Interact")]
		[HarmonyPrefix]
		private static bool InteractPatch(CharacterInteractible __instance, ref Character interactor)
		{
			if (IsBuddyClimbDropInteraction(__instance.character, interactor))
			{
				return false;
			}
			if (__instance.CarriedByLocalCharacter() || __instance.IsCannibal() || __instance.CanBeCarried())
			{
				return true;
			}
			return !TryStartClimb(__instance.character, interactor);
		}

		[HarmonyPatch("IsInteractible")]
		[HarmonyPostfix]
		private static void IsInteractiblePatch(CharacterInteractible __instance, ref bool __result, ref Character interactor)
		{
			if (IsBuddyClimbDropInteraction(__instance.character, interactor))
			{
				__result = false;
			}
			else if (!__result && CanStartClimb(__instance.character, interactor))
			{
				__result = true;
			}
		}

		[HarmonyPatch("IsPrimaryInteractible")]
		[HarmonyPostfix]
		private static void IsPrimaryInteractiblePatch(CharacterInteractible __instance, ref bool __result, ref Character interactor)
		{
			if (IsBuddyClimbDropInteraction(__instance.character, interactor))
			{
				__result = false;
			}
			else if (!__result && CanStartClimb(__instance.character, interactor))
			{
				__result = true;
			}
		}

		private static bool CanBeClimbed(Character character)
		{
			if ((Object)(object)character == (Object)null)
			{
				return false;
			}
			if (character.isBot)
			{
				return false;
			}
			if (character.IsLocal)
			{
				return false;
			}
			if (character.data.dead)
			{
				return false;
			}
			if (character.player.backpackSlot.hasBackpack && !BackpackCarryTransfer.AllowsCarrierBackpack)
			{
				return false;
			}
			if (IsCharacterDoingIllegalCarryActions(character))
			{
				return false;
			}
			if (character.data.IsCarryingCharacter)
			{
				return false;
			}
			if (character.refs.customization.isCannibalizable)
			{
				return false;
			}
			return true;
		}

		private static bool CanClimb(Character character)
		{
			if ((Object)(object)character == (Object)null)
			{
				return false;
			}
			if (character.refs.interactible.CanBeCarried())
			{
				return true;
			}
			if (!character.IsLocal)
			{
				return false;
			}
			if (character.data.dead)
			{
				return false;
			}
			if (Object.op_Implicit((Object)(object)character.data.currentItem))
			{
				return false;
			}
			if (character.data.isCarried)
			{
				return false;
			}
			if (character.refs.customization.isCannibalizable)
			{
				return false;
			}
			return true;
		}

		private static bool IsCharacterDoingIllegalCarryActions(Character character)
		{
			if (!character.data.isSprinting && !character.data.isJumping && !character.data.isClimbingAnything && !character.data.isCrouching)
			{
				return character.data.isReaching;
			}
			return true;
		}

		private static bool TryStartClimb(Character character, Character interactor)
		{
			BuddyClimbDiagnostics.LogCarry("TryStartClimb entered: " + BuddyClimbDiagnostics.DescribeViews(character, interactor));
			if (!CanStartClimb(character, interactor))
			{
				BuddyClimbDiagnostics.LogCarry("TryStartClimb blocked by CanStartClimb=false: " + BuddyClimbDiagnostics.DescribeViews(character, interactor));
				return false;
			}
			BackpackPreparationResult backpackPreparationResult = BackpackCarryTransfer.PrepareBackpacksForClimb(character, interactor);
			BuddyClimbDiagnostics.LogCarry($"Backpack preparation result={backpackPreparationResult}: {BuddyClimbDiagnostics.DescribeViews(character, interactor)}");
			switch (backpackPreparationResult)
			{
			case BackpackPreparationResult.Failed:
				BuddyClimbDiagnostics.LogCarry("TryStartClimb consumed interaction because backpack preparation failed.");
				return true;
			case BackpackPreparationResult.Ready:
			{
				bool flag = BuddyClimbCarryStarter.TryStartCarry(character, interactor);
				BuddyClimbDiagnostics.LogCarry($"TryStartClimb called TryStartCarry, sent={flag}: {BuddyClimbDiagnostics.DescribeViews(character, interactor)}");
				break;
			}
			}
			return true;
		}

		private static bool CanStartClimb(Character carrier, Character carried)
		{
			if (CanBeClimbed(carrier) && CanClimb(carried))
			{
				return BuddyClimbCarryStarter.CanCreateCarryLink(carrier, carried);
			}
			return false;
		}

		private static bool IsBuddyClimbDropInteraction(Character character, Character interactor)
		{
			if ((Object)(object)character != (Object)null && (Object)(object)interactor != (Object)null && (Object)(object)character.data.carrier == (Object)(object)interactor)
			{
				return CharacterCarryingPatch.IsBuddyClimbCarried(character);
			}
			return false;
		}
	}
	[HarmonyPatch(typeof(CharacterMovement))]
	internal static class CharacterMovementPatch
	{
		[HarmonyPatch("TryToJump")]
		[HarmonyPrefix]
		private static bool TryToJumpPrefix(CharacterMovement __instance)
		{
			return !CarriedPlayerDropper.HandleJumpAttempt(__instance.character);
		}
	}
	[HarmonyPatch(typeof(Item))]
	internal static class ItemBackpackVisualPatch
	{
		[HarmonyPatch("PutInBackpackRPC")]
		[HarmonyPostfix]
		private static void PutInBackpackRPCPostfix(Item __instance, BackpackReference backpackReference)
		{
			//IL_0001: Unknown result type (might be due to invalid IL or missing references)
			CarriedBackpackVisuals.ShowIfCarriedLocalBackpackItem(__instance, backpackReference);
		}
	}
	[HarmonyPatch(typeof(Player))]
	internal static class PlayerInventoryPatch
	{
		[HarmonyPatch("SyncInventoryRPC")]
		[HarmonyPrefix]
		private static bool SyncInventoryRPCPrefix(Player __instance, byte[] data, bool forceSync)
		{
			return !BackpackCarryTransfer.ShouldSuppressStaleBackpackEmptySync(__instance, data, forceSync);
		}
	}
}
namespace BuddyClimb.Localization
{
	internal enum BuddyClimbTextKey
	{
		ClimbOnTeammate,
		ClimbOnTeammateDropBackpack
	}
	internal static class BuddyClimbLocalization
	{
		private enum SupportedLanguage
		{
			English,
			Chinese
		}

		private static readonly IReadOnlyDictionary<BuddyClimbTextKey, string> EnglishText = new Dictionary<BuddyClimbTextKey, string>
		{
			[BuddyClimbTextKey.ClimbOnTeammate] = "Climb on!",
			[BuddyClimbTextKey.ClimbOnTeammateDropBackpack] = "Climb on! (drop backpack)"
		};

		private static readonly IReadOnlyDictionary<BuddyClimbTextKey, string> ChineseText = new Dictionary<BuddyClimbTextKey, string>
		{
			[BuddyClimbTextKey.ClimbOnTeammate] = "爬上去!",
			[BuddyClimbTextKey.ClimbOnTeammateDropBackpack] = "爬上去!(丢弃背包)"
		};

		internal static string Get(BuddyClimbTextKey key)
		{
			SupportedLanguage currentLanguage = GetCurrentLanguage();
			IReadOnlyDictionary<BuddyClimbTextKey, string> readOnlyDictionary = ((currentLanguage != SupportedLanguage.Chinese) ? EnglishText : ChineseText);
			IReadOnlyDictionary<BuddyClimbTextKey, string> readOnlyDictionary2 = readOnlyDictionary;
			if (readOnlyDictionary2.TryGetValue(key, out var value))
			{
				return value;
			}
			return EnglishText[key];
		}

		private static SupportedLanguage GetCurrentLanguage()
		{
			//IL_0000: Unknown result type (might be due to invalid IL or missing references)
			//IL_0005: Unknown result type (might be due to invalid IL or missing references)
			//IL_0006: Unknown result type (might be due to invalid IL or missing references)
			//IL_0009: Invalid comparison between Unknown and I4
			//IL_000b: Unknown result type (might be due to invalid IL or missing references)
			//IL_000e: Invalid comparison between Unknown and I4
			Language cURRENT_LANGUAGE = LocalizedText.CURRENT_LANGUAGE;
			if ((int)cURRENT_LANGUAGE != 9)
			{
				if ((int)cURRENT_LANGUAGE == 10)
				{
					return SupportedLanguage.Chinese;
				}
				return SupportedLanguage.English;
			}
			return SupportedLanguage.Chinese;
		}
	}
}
namespace BuddyClimb.Gameplay
{
	internal enum BackpackPreparationResult
	{
		Failed,
		Ready
	}
	internal static class BackpackCarryTransfer
	{
		private readonly struct BackpackSlotSnapshot
		{
			private readonly bool hasBackpack;

			private readonly Item? prefab;

			private readonly ItemInstanceData data;

			private BackpackSlotSnapshot(BackpackSlot backpackSlot)
			{
				hasBackpack = backpackSlot.hasBackpack;
				prefab = ((ItemSlot)backpackSlot).prefab;
				data = ((ItemSlot)backpackSlot).data;
			}

			internal static BackpackSlotSnapshot Capture(Player player)
			{
				return new BackpackSlotSnapshot(player.backpackSlot);
			}

			internal void Restore(Player player)
			{
				//IL_0005: Unknown result type (might be due to invalid IL or missing references)
				//IL_000a: Unknown result type (might be due to invalid IL or missing references)
				//IL_0016: Unknown result type (might be due to invalid IL or missing references)
				//IL_0022: Unknown result type (might be due to invalid IL or missing references)
				//IL_002f: Expected O, but got Unknown
				BackpackSlot backpackSlot = new BackpackSlot(BackpackSlotIndex)
				{
					hasBackpack = hasBackpack,
					prefab = prefab,
					data = data
				};
				player.backpackSlot = backpackSlot;
			}
		}

		private sealed class PendingBackpackTransfer
		{
			internal float ExpiresAt { get; }

			internal PendingBackpackTransfer(float expiresAt)
			{
				ExpiresAt = expiresAt;
			}
		}

		private static readonly byte BackpackSlotIndex = 3;

		private static readonly Dictionary<int, PendingBackpackTransfer> PendingBackpackTransferSyncSuppressions = new Dictionary<int, PendingBackpackTransfer>();

		private const float PendingBackpackTransferSyncSuppressionSeconds = 2f;

		internal static bool AllowsCarrierBackpack => BuddyClimbConfig.EnableBackpackTransfer.Value;

		internal static bool WillDropCarriedBackpack(Character carrier, Character carried)
		{
			if (AllowsCarrierBackpack && (Object)(object)carrier != (Object)null && (Object)(object)carried != (Object)null && HasBackpack(carrier))
			{
				return HasBackpack(carried);
			}
			return false;
		}

		internal static bool CanTransferCarrierBackpack(Character carrier, Character carried)
		{
			if (!AllowsCarrierBackpack || (Object)(object)carrier == (Object)null || (Object)(object)carried == (Object)null || !HasBackpack(carrier))
			{
				return true;
			}
			Player player = carrier.player;
			Player player2 = carried.player;
			if (CanSyncInventory(player) && CanSyncInventory(player2))
			{
				if (HasBackpack(carried))
				{
					return CanDropBackpackWithVanillaSlotDrop(carried);
				}
				return true;
			}
			return false;
		}

		internal static BackpackPreparationResult PrepareBackpacksForClimb(Character carrier, Character carried)
		{
			BuddyClimbDiagnostics.LogCarry("PrepareBackpacksForClimb entered: " + BuddyClimbDiagnostics.DescribeViews(carrier, carried));
			if (!AllowsCarrierBackpack || (Object)(object)carrier == (Object)null || (Object)(object)carried == (Object)null || !HasBackpack(carrier))
			{
				BuddyClimbDiagnostics.LogCarry($"PrepareBackpacksForClimb returning Ready without transfer: allows={AllowsCarrierBackpack} {BuddyClimbDiagnostics.DescribeViews(carrier, carried)}");
				return BackpackPreparationResult.Ready;
			}
			if (!CanTransferCarrierBackpack(carrier, carried))
			{
				BuddyClimbDiagnostics.LogCarry("PrepareBackpacksForClimb returning Failed because CanTransferCarrierBackpack=false: " + BuddyClimbDiagnostics.DescribeViews(carrier, carried));
				return BackpackPreparationResult.Failed;
			}
			if (HasBackpack(carried))
			{
				if (TryDropCarriedBackpackAndTransferOnMaster(carrier, carried))
				{
					BuddyClimbDiagnostics.LogCarry("PrepareBackpacksForClimb returning Ready after master drop+transfer: " + BuddyClimbDiagnostics.DescribeViews(carrier, carried));
					return BackpackPreparationResult.Ready;
				}
				bool flag = TryRequestMasterDropCarriedBackpackAndTransfer(carrier, carried);
				BuddyClimbDiagnostics.LogCarry($"PrepareBackpacksForClimb non-master drop+transfer requested={flag}: {BuddyClimbDiagnostics.DescribeViews(carrier, carried)}");
				if (!flag)
				{
					return BackpackPreparationResult.Failed;
				}
				return BackpackPreparationResult.Ready;
			}
			bool flag2 = TryTransferCarrierBackpack(carrier, carried, syncInventory: true);
			BuddyClimbDiagnostics.LogCarry($"PrepareBackpacksForClimb transfer without carried backpack result={flag2}: {BuddyClimbDiagnostics.DescribeViews(carrier, carried)}");
			if (!flag2)
			{
				return BackpackPreparationResult.Failed;
			}
			return BackpackPreparationResult.Ready;
		}

		internal static bool TryDropCarriedBackpackAndTransferOnMaster(Character carrier, Character carried)
		{
			if (!PhotonNetwork.IsMasterClient)
			{
				return false;
			}
			if (!CanTransferCarrierBackpack(carrier, carried))
			{
				return false;
			}
			BackpackSlotSnapshot carrierBackpackSnapshot = BackpackSlotSnapshot.Capture(carrier.player);
			BackpackSlotSnapshot carriedBackpackSnapshot = BackpackSlotSnapshot.Capture(carried.player);
			int droppedItemCount = GetDroppedItemCount(carried);
			try
			{
				if (HasBackpack(carried) && !TryDropCarriedBackpackWithVanillaSlotDrop(carried))
				{
					RollBackBackpackTransfer(carrier, carried, carrierBackpackSnapshot, carriedBackpackSnapshot, droppedItemCount, "the carried backpack could not be dropped");
					return false;
				}
				if (!TryTransferCarrierBackpack(carrier, carried, syncInventory: true))
				{
					RollBackBackpackTransfer(carrier, carried, carrierBackpackSnapshot, carriedBackpackSnapshot, droppedItemCount, "the carrier backpack could not be transferred");
					return false;
				}
				return true;
			}
			catch (Exception ex)
			{
				RollBackBackpackTransfer(carrier, carried, carrierBackpackSnapshot, carriedBackpackSnapshot, droppedItemCount, ex.ToString());
				return false;
			}
		}

		private static bool TryRequestMasterDropCarriedBackpackAndTransfer(Character carrier, Character carried)
		{
			BuddyClimbDiagnostics.LogCarry("Requesting MasterClient backpack drop before transfer: " + BuddyClimbDiagnostics.DescribeViews(carrier, carried));
			if (!TryRequestVanillaSlotDropOnMaster(carried))
			{
				BuddyClimbDiagnostics.LogCarry("TryRequestMasterDropCarriedBackpackAndTransfer failed because TryRequestVanillaSlotDropOnMaster=false.");
				return false;
			}
			TrackPendingBackpackTransfer(carrier, carried);
			ClearCarriedBackpackLocally(carried);
			if (!TryTransferCarrierBackpack(carrier, carried, syncInventory: true))
			{
				ClearPendingBackpackTransfer(carried);
				BuddyClimbDiagnostics.LogCarry("TryRequestMasterDropCarriedBackpackAndTransfer failed because TryTransferCarrierBackpack=false.");
				return false;
			}
			BuddyClimbDiagnostics.LogCarry("Requested MasterClient drop and transferred carrier backpack locally: " + BuddyClimbDiagnostics.DescribeViews(carrier, carried));
			return true;
		}

		internal static bool ShouldSuppressStaleBackpackEmptySync(Player player, byte[] data, bool forceSync)
		{
			//IL_0052: Unknown result type (might be due to invalid IL or missing references)
			//IL_0057: Unknown result type (might be due to invalid IL or missing references)
			//IL_0058: Unknown result type (might be due to invalid IL or missing references)
			if ((Object)(object)player == (Object)null || (Object)(object)((MonoBehaviourPun)player).photonView == (Object)null)
			{
				return false;
			}
			int viewID = ((MonoBehaviourPun)player).photonView.ViewID;
			if (!PendingBackpackTransferSyncSuppressions.TryGetValue(viewID, out PendingBackpackTransfer value))
			{
				return false;
			}
			if (Time.realtimeSinceStartup > value.ExpiresAt)
			{
				PendingBackpackTransferSyncSuppressions.Remove(viewID);
				return false;
			}
			InventorySyncData fromManagedArray = IBinarySerializable.GetFromManagedArray<InventorySyncData>(data);
			if (fromManagedArray.hasBackpack || forceSync)
			{
				return false;
			}
			PendingBackpackTransferSyncSuppressions.Remove(viewID);
			ManualLogSource log = Plugin.Log;
			Character character = player.character;
			log.LogDebug((object)("Suppressed stale empty backpack sync for " + (((character != null) ? character.characterName : null) ?? "unknown player") + " during BuddyClimb backpack transfer."));
			return true;
		}

		private static bool TryRequestVanillaSlotDropOnMaster(Character carried)
		{
			//IL_0056: Unknown result type (might be due to invalid IL or missing references)
			if (!CanDropBackpackWithVanillaSlotDrop(carried))
			{
				Plugin.Log.LogWarning((object)("Unable to request " + carried.characterName + "'s backpack drop because CharacterItems is unavailable."));
				return false;
			}
			try
			{
				carried.refs.items.photonView.RPC("DropItemFromSlotRPC", (RpcTarget)2, new object[2]
				{
					BackpackSlotIndex,
					GetBackpackDropPosition(carried)
				});
				return true;
			}
			catch (Exception arg)
			{
				Plugin.Log.LogWarning((object)$"Unable to request {carried.characterName}'s backpack drop through PEAK's slot drop RPC: {arg}");
				return false;
			}
		}

		private static bool TryDropCarriedBackpackWithVanillaSlotDrop(Character carried)
		{
			//IL_004e: Unknown result type (might be due to invalid IL or missing references)
			if ((Object)(object)carried == (Object)null || !HasBackpack(carried))
			{
				return true;
			}
			if (!CanDropBackpackWithVanillaSlotDrop(carried))
			{
				Plugin.Log.LogWarning((object)("Unable to drop " + carried.characterName + "'s backpack because CharacterItems is unavailable."));
				return false;
			}
			try
			{
				carried.refs.items.DropItemFromSlotRPC(BackpackSlotIndex, GetBackpackDropPosition(carried));
				return true;
			}
			catch (Exception arg)
			{
				Plugin.Log.LogWarning((object)$"Unable to drop {carried.characterName}'s backpack through PEAK's slot drop path: {arg}");
				return false;
			}
		}

		private static bool TryTransferCarrierBackpack(Character carrier, Character carried, bool syncInventory)
		{
			//IL_008f: Unknown result type (might be due to invalid IL or missing references)
			//IL_0099: Expected O, but got Unknown
			if (!AllowsCarrierBackpack || (Object)(object)carrier == (Object)null || (Object)(object)carried == (Object)null || !HasBackpack(carrier))
			{
				return true;
			}
			Player player = carrier.player;
			Player player2 = carried.player;
			if (!CanSyncInventory(player) || !CanSyncInventory(player2))
			{
				Plugin.Log.LogWarning((object)"Skipping backpack transfer because a Player inventory reference is unavailable.");
				return false;
			}
			if (HasBackpack(carried))
			{
				Plugin.Log.LogWarning((object)("Skipping backpack transfer because " + carried.characterName + " is still wearing a backpack."));
				return false;
			}
			BackpackSlot backpackSlot = player.backpackSlot;
			player2.backpackSlot = backpackSlot;
			player.backpackSlot = new BackpackSlot(BackpackSlotIndex);
			ClearCarrierHeldBackpack(carrier);
			carried.refs.afflictions.UpdateWeight();
			carrier.refs.afflictions.UpdateWeight();
			if (syncInventory)
			{
				SyncInventory(player2);
				SyncInventory(player);
			}
			return true;
		}

		internal static void ClearHeldBackpackAfterTransfer(Character carrier)
		{
			//IL_005c: Unknown result type (might be due to invalid IL or missing references)
			//IL_0061: Unknown result type (might be due to invalid IL or missing references)
			//IL_0051: Unknown result type (might be due to invalid IL or missing references)
			//IL_0056: Unknown result type (might be due to invalid IL or missing references)
			if ((Object)(object)carrier == (Object)null)
			{
				return;
			}
			CharacterItems items = carrier.refs.items;
			if ((Object)(object)items == (Object)null)
			{
				return;
			}
			bool flag = IsBackpackSlotSelected(items);
			bool flag2 = carrier.data.currentItem is Backpack;
			if (flag || flag2)
			{
				if (items.currentSelectedSlot.IsSome)
				{
					items.lastSelectedSlot = items.currentSelectedSlot;
				}
				items.currentSelectedSlot = Optionable<byte>.None;
				if (flag2)
				{
					items.DestroyHeldItemRpc();
				}
				carrier.player.itemsChangedAction?.Invoke(carrier.player.itemSlots);
				items.onSlotEquipped?.Invoke();
			}
		}

		private static void RollBackBackpackTransfer(Character carrier, Character carried, BackpackSlotSnapshot carrierBackpackSnapshot, BackpackSlotSnapshot carriedBackpackSnapshot, int droppedItemsStartCount, string reason)
		{
			Plugin.Log.LogWarning((object)("Rolling back BuddyClimb backpack transfer for " + carried.characterName + ": " + reason));
			DestroyDroppedItemsAddedSince(carried, droppedItemsStartCount);
			carrierBackpackSnapshot.Restore(carrier.player);
			carriedBackpackSnapshot.Restore(carried.player);
			carrier.refs.afflictions.UpdateWeight();
			carried.refs.afflictions.UpdateWeight();
			try
			{
				SyncInventory(carried.player);
				SyncInventory(carrier.player);
			}
			catch (Exception arg)
			{
				Plugin.Log.LogWarning((object)$"Unable to sync rollback state after BuddyClimb backpack transfer failed: {arg}");
			}
		}

		private static void DestroyDroppedItemsAddedSince(Character carried, int droppedItemsStartCount)
		{
			List<PhotonView> droppedItems = carried.refs.items.droppedItems;
			if (droppedItems == null)
			{
				return;
			}
			for (int num = droppedItems.Count - 1; num >= droppedItemsStartCount; num--)
			{
				PhotonView val = droppedItems[num];
				droppedItems.RemoveAt(num);
				if (!((Object)(object)val == (Object)null))
				{
					try
					{
						PhotonNetwork.Destroy(val);
					}
					catch (Exception arg)
					{
						Plugin.Log.LogWarning((object)$"Unable to destroy rolled-back BuddyClimb dropped backpack: {arg}");
					}
				}
			}
		}

		private static int GetDroppedItemCount(Character character)
		{
			return character.refs.items.droppedItems?.Count ?? 0;
		}

		private static bool HasBackpack(Character character)
		{
			return (character.player?.backpackSlot)?.hasBackpack ?? false;
		}

		private static void ClearCarrierHeldBackpack(Character carrier)
		{
			ClearHeldBackpackAfterTransfer(carrier);
			CharacterItems items = carrier.refs.items;
			if (!((Object)(object)items?.photonView == (Object)null))
			{
				items.photonView.RPC("DestroyHeldItemRpc", (RpcTarget)1, Array.Empty<object>());
				items.photonView.RPC("EquipSlotRpc", (RpcTarget)1, new object[2] { -1, -1 });
			}
		}

		private static bool IsBackpackSlotSelected(CharacterItems characterItems)
		{
			if (characterItems.currentSelectedSlot.IsSome)
			{
				return characterItems.currentSelectedSlot.Value == BackpackSlotIndex;
			}
			return false;
		}

		private static void ClearCarriedBackpackLocally(Character carried)
		{
			((ItemSlot)carried.player.backpackSlot).EmptyOut();
			carried.refs.afflictions.UpdateWeight();
		}

		private static void TrackPendingBackpackTransfer(Character carrier, Character carried)
		{
			Player player = carried.player;
			if (!((Object)(object)((player != null) ? ((MonoBehaviourPun)player).photonView : null) == (Object)null))
			{
				PendingBackpackTransferSyncSuppressions[((MonoBehaviourPun)carried.player).photonView.ViewID] = new PendingBackpackTransfer(Time.realtimeSinceStartup + 2f);
			}
		}

		private static void ClearPendingBackpackTransfer(Character carried)
		{
			Player player = carried.player;
			if (!((Object)(object)((player != null) ? ((MonoBehaviourPun)player).photonView : null) == (Object)null))
			{
				PendingBackpackTransferSyncSuppressions.Remove(((MonoBehaviourPun)carried.player).photonView.ViewID);
			}
		}

		private static bool CanDropBackpackWithVanillaSlotDrop(Character character)
		{
			if ((Object)(object)character.refs.items != (Object)null)
			{
				return (Object)(object)character.refs.items.photonView != (Object)null;
			}
			return false;
		}

		private static Vector3 GetBackpackDropPosition(Character character)
		{
			//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_0010: Unknown result type (might be due to invalid IL or missing references)
			//IL_0015: Unknown result type (might be due to invalid IL or missing references)
			//IL_001a: 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_0029: 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_003d: Unknown result type (might be due to invalid IL or missing references)
			//IL_0040: Unknown result type (might be due to invalid IL or missing references)
			try
			{
				return character.Center + Vector3.up * 0.5f;
			}
			catch
			{
				return ((Component)character).transform.position + Vector3.up * 0.5f;
			}
		}

		private static void SyncInventory(Player player)
		{
			//IL_0012: Unknown result type (might be due to invalid IL or missing references)
			byte[] array = IBinarySerializable.ToManagedArray<InventorySyncData>(new InventorySyncData(player.itemSlots, player.backpackSlot, player.tempFullSlot));
			((MonoBehaviourPun)player).photonView.RPC("SyncInventoryRPC", (RpcTarget)0, new object[2] { array, true });
		}

		private static bool CanSyncInventory(Player player)
		{
			if ((Object)(object)player != (Object)null && (Object)(object)((MonoBehaviourPun)player).photonView != (Object)null && player.itemSlots != null && player.backpackSlot != null)
			{
				return player.tempFullSlot != null;
			}
			return false;
		}
	}
	internal static class BuddyClimbCarryStarter
	{
		internal static bool TryStartCarry(Character carrier, Character carried)
		{
			BuddyClimbDiagnostics.LogCarry("TryStartCarry requested: " + BuddyClimbDiagnostics.DescribeViews(carrier, carried));
			if (!CanStartCarryRpc(carrier, carried))
			{
				BuddyClimbDiagnostics.LogCarry("TryStartCarry blocked by CanStartCarryRpc=false: " + BuddyClimbDiagnostics.DescribeViews(carrier, carried));
				return false;
			}
			BuddyClimbDiagnostics.LogCarry(string.Format("Sending Photon RPC {0} target=All carrierView={1} carriedView={2}: {3}", "RPCA_StartCarry", ((MonoBehaviourPun)carrier).photonView.ViewID, ((MonoBehaviourPun)carried).photonView.ViewID, BuddyClimbDiagnostics.DescribeViews(carrier, carried)));
			BuddyClimbRemotePassOutSync.PrepareRemoteCarryStart(carried);
			((MonoBehaviourPun)carrier).photonView.RPC("RPCA_StartCarry", (RpcTarget)0, new object[1] { ((MonoBehaviourPun)carried).photonView });
			BuddyClimbDiagnostics.LogCarry(string.Format("Sent Photon RPC {0} carrierView={1} carriedView={2}", "RPCA_StartCarry", ((MonoBehaviourPun)carrier).photonView.ViewID, ((MonoBehaviourPun)carried).photonView.ViewID));
			return true;
		}

		internal static bool CanCreateCarryLink(Character carrier, Character carried)
		{
			if ((Object)(object)carrier == (Object)null || (Object)(object)carried == (Object)null)
			{
				return false;
			}
			Character val = carrier;
			while (true)
			{
				if ((Object)(object)val == (Object)(object)carried)
				{
					return false;
				}
				Character val2 = val.data?.carrier;
				if (val2 == null)
				{
					break;
				}
				val = val2;
			}
			return true;
		}

		private static bool CanStartCarryRpc(Character carrier, Character carried)
		{
			if ((Object)(object)carrier == (Object)null || (Object)(object)carried == (Object)null || (Object)(object)carrier == (Object)(object)carried || (Object)(object)((MonoBehaviourPun)carrier).photonView == (Object)null || (Object)(object)((MonoBehaviourPun)carried).photonView == (Object)null)
			{
				BuddyClimbDiagnostics.LogCarry("CanStartCarryRpc basic validation failed: " + BuddyClimbDiagnostics.DescribeViews(carrier, carried));
				return false;
			}
			if (!CanCreateCarryLink(carrier, carried))
			{
				Plugin.Log.LogDebug((object)("Skipping BuddyClimb start carry because it would create a carry cycle between " + carrier.characterName + " and " + carried.characterName + "."));
				return false;
			}
			if (carrier.data.dead || carried.data.dead || carried.data.fullyPassedOut)
			{
				Plugin.Log.LogDebug((object)("Skipping BuddyClimb start carry for " + carried.characterName + " because the carry state is no longer valid."));
				return false;
			}
			return true;
		}
	}
	internal static class BuddyClimbDiagnostics
	{
		internal static void LogCarry(string message)
		{
			Plugin.Log.LogDebug((object)("[BuddyClimb carry] " + message));
		}

		internal static string Describe(Character? character)
		{
			if ((Object)(object)character == (Object)null)
			{
				return "<null character>";
			}
			Character character2 = character.data?.carrier;
			Character character3 = character.data?.carriedPlayer;
			CharacterData data = character.data;
			string text = (((Object)(object)((data != null) ? data.currentItem : null) != (Object)null) ? ((object)character.data.currentItem).GetType().Name : "none");
			string[] obj = new string[17]
			{
				GetName(character),
				$" charView={GetCharacterViewId(character)}",
				$" playerView={GetPlayerViewId(character)}",
				$" isLocal={character.IsLocal}",
				$" isBot={character.isBot}",
				null,
				null,
				null,
				null,
				null,
				null,
				null,
				null,
				null,
				null,
				null,
				null
			};
			CharacterData data2 = character.data;
			obj[5] = $" dead={((data2 != null) ? new bool?(data2.dead) : null)}";
			obj[6] = $" passedOut={character.data?.passedOut}";
			obj[7] = $" fullyPassedOut={character.data?.fullyPassedOut}";
			obj[8] = $" isCarried={character.data?.isCarried}";
			obj[9] = " carrier=";
			obj[10] = GetName(character2);
			obj[11] = " carriedPlayer=";
			obj[12] = GetName(character3);
			CharacterData data3 = character.data;
			obj[13] = $" isCarryingCharacter={((data3 != null) ? new bool?(data3.IsCarryingCharacter) : null)}";
			obj[14] = $" hasBackpack={HasBackpack(character)}";
			obj[15] = " currentItem=";
			obj[16] = text;
			return string.Concat(obj);
		}

		internal static string DescribeViews(Character? carrier, Character? carried)
		{
			return "carrier=[" + Describe(carrier) + "], carried=[" + Describe(carried) + "]";
		}

		private static string GetName(Character? character)
		{
			if ((Object)(object)character == (Object)null)
			{
				return "null";
			}
			return $"{character.characterName ?? ((Object)character).name}#{GetCharacterViewId(character)}";
		}

		private static int GetCharacterViewId(Character? character)
		{
			if (!((Object)(object)((character != null) ? ((MonoBehaviourPun)character).photonView : null) != (Object)null))
			{
				return -1;
			}
			return ((MonoBehaviourPun)character).photonView.ViewID;
		}

		private static int GetPlayerViewId(Character? character)
		{
			object obj;
			if (character == null)
			{
				obj = null;
			}
			else
			{
				Player player = character.player;
				obj = ((player != null) ? ((MonoBehaviourPun)player).photonView : null);
			}
			if (!((Object)obj != (Object)null))
			{
				return -1;
			}
			return ((MonoBehaviourPun)character.player).photonView.ViewID;
		}

		private static bool HasBackpack(Character? character)
		{
			return ((character == null) ? null : character.player?.backpackSlot)?.hasBackpack ?? false;
		}
	}
	internal static class BuddyClimbRemotePassOutSync
	{
		private static readonly HashSet<int> RemotePassOutViewIds = new HashSet<int>();

		internal static void PrepareRemoteCarryStart(Character carried)
		{
			if (!CanSyncLocalConsciousCharacter(carried))
			{
				BuddyClimbDiagnostics.LogCarry("Skipping remote-only pass-out sync before carry start: " + BuddyClimbDiagnostics.Describe(carried));
				return;
			}
			int viewID = ((MonoBehaviourPun)carried).photonView.ViewID;
			RemotePassOutViewIds.Add(viewID);
			BuddyClimbDiagnostics.LogCarry("Sending remote-only RPCA_PassOut before carry start: " + BuddyClimbDiagnostics.Describe(carried));
			((MonoBehaviourPun)carried).photonView.RPC("RPCA_PassOut", (RpcTarget)1, Array.Empty<object>());
		}

		internal static void RestoreRemoteCarryDrop(Character carried, string reason)
		{
			if ((Object)(object)((carried != null) ? ((MonoBehaviourPun)carried).photonView : null) == (Object)null)
			{
				return;
			}
			int viewID = ((MonoBehaviourPun)carried).photonView.ViewID;
			if (RemotePassOutViewIds.Contains(viewID))
			{
				if (!CanSyncLocalConsciousCharacter(carried))
				{
					BuddyClimbDiagnostics.LogCarry("Skipping remote-only un-pass-out sync for " + reason + ": " + BuddyClimbDiagnostics.Describe(carried));
					return;
				}
				BuddyClimbDiagnostics.LogCarry("Sending remote-only RPCA_UnPassOut for " + reason + ": " + BuddyClimbDiagnostics.Describe(carried));
				((MonoBehaviourPun)carried).photonView.RPC("RPCA_UnPassOut", (RpcTarget)1, Array.Empty<object>());
				RemotePassOutViewIds.Remove(viewID);
			}
		}

		private static bool CanSyncLocalConsciousCharacter(Character character)
		{
			if ((Object)(object)character != (Object)null && character.IsLocal && (Object)(object)((MonoBehaviourPun)character).photonView != (Object)null && (Object)(object)character.data != (Object)null && !character.data.dead && !character.data.passedOut)
			{
				return !character.data.fullyPassedOut;
			}
			return false;
		}
	}
	internal static class CarriedBackpackVisuals
	{
		private static readonly byte BackpackSlotIndex = 3;

		private static readonly MethodInfo HideRenderersMethod = AccessTools.Method(typeof(Item), "HideRenderers", (Type[])null, (Type[])null);

		private static readonly HashSet<int> ForcedVisibleItemCharacterViewIds = new HashSet<int>();

		internal static void Update(CharacterBackpackHandler handler)
		{
			if ((Object)(object)handler == (Object)null || (Object)(object)handler.character == (Object)null || (Object)(object)((MonoBehaviourPun)handler.character).photonView == (Object)null)
			{
				return;
			}
			Character character = handler.character;
			int viewID = ((MonoBehaviourPun)character).photonView.ViewID;
			if (ShouldForceBackpackItemRender(character))
			{
				bool flag = ForcedVisibleItemCharacterViewIds.Contains(viewID);
				if (ShowBackpackItems(handler, !flag))
				{
					ForcedVisibleItemCharacterViewIds.Add(viewID);
				}
			}
			else if (ForcedVisibleItemCharacterViewIds.Remove(viewID))
			{
				HideBackpackItems(handler);
			}
		}

		internal static void ShowIfCarriedLocalBackpackItem(Item item, BackpackReference backpackReference)
		{
			if (!((Object)(object)item == (Object)null) && ((BackpackReference)(ref backpackReference)).IsOnMyBack())
			{
				Character localCharacter = Character.localCharacter;
				if (ShouldForceBackpackItemRender(localCharacter))
				{
					ShowItem(item);
				}
			}
		}

		internal static void HideIfForcedVisible(Character character)
		{
			if (!((Object)(object)character == (Object)null) && !((Object)(object)((MonoBehaviourPun)character).photonView == (Object)null) && ForcedVisibleItemCharacterViewIds.Remove(((MonoBehaviourPun)character).photonView.ViewID))
			{
				CharacterBackpackHandler component = ((Component)character).GetComponent<CharacterBackpackHandler>();
				if ((Object)(object)component != (Object)null)
				{
					HideBackpackItems(component);
				}
			}
		}

		private static bool ShouldForceBackpackItemRender(Character character)
		{
			if ((Object)(object)character != (Object)null && character.IsLocal && (Object)(object)character.data != (Object)null && character.data.isCarried && !character.data.dead && !character.data.fullyPassedOut && MainCameraMovement.IsSpectating && (Object)(object)MainCameraMovement.specCharacter == (Object)(object)character && HasOnBackBackpack(character))
			{
				return CharacterCarryingPatch.IsBuddyClimbCarried(character);
			}
			return false;
		}

		private static bool ShowBackpackItems(CharacterBackpackHandler handler, bool refreshItems)
		{
			if (!HasOnBackBackpack(handler.character))
			{
				return false;
			}
			if (!refreshItems)
			{
				return true;
			}
			BackpackOnBackVisuals backpackVisuals = handler.backpackVisuals;
			if ((Object)(object)backpackVisuals == (Object)null)
			{
				return false;
			}
			((BackpackVisuals)backpackVisuals).RefreshVisuals();
			SetSpawnedItemsVisible(backpackVisuals, visible: true);
			return true;
		}

		private static void HideBackpackItems(CharacterBackpackHandler handler)
		{
			SetSpawnedItemsVisible(handler.backpackVisuals, visible: false);
		}

		private static void SetSpawnedItemsVisible(BackpackOnBackVisuals backpackVisuals, bool visible)
		{
			if ((Object)(object)backpackVisuals == (Object)null)
			{
				return;
			}
			BackpackData backpackData = ((BackpackVisuals)backpackVisuals).GetBackpackData();
			if (backpackData?.itemSlots == null)
			{
				return;
			}
			Item val2 = default(Item);
			for (byte b = 0; b < backpackData.itemSlots.Length; b++)
			{
				ItemSlot val = backpackData.itemSlots[b];
				if (val != null && !val.IsEmpty() && ((BackpackVisuals)backpackVisuals).TryGetSpawnedItem(b, ref val2) && !((Object)(object)val2 == (Object)null))
				{
					if (visible)
					{
						ShowItem(val2);
					}
					else
					{
						HideItem(val2);
					}
				}
			}
		}

		private static bool HasOnBackBackpack(Character character)
		{
			BackpackSlot val = ((character == null) ? null : character.player?.backpackSlot);
			if (val != null && val.hasBackpack)
			{
				return !IsBackpackSlotSelected(character);
			}
			return false;
		}

		private static bool IsBackpackSlotSelected(Character character)
		{
			CharacterItems items = character.refs.items;
			if ((Object)(object)items != (Object)null && items.currentSelectedSlot.IsSome)
			{
				return items.currentSelectedSlot.Value == BackpackSlotIndex;
			}
			return false;
		}

		private static void ShowItem(Item item)
		{
			Renderer[] componentsInChildren = ((Component)item).GetComponentsInChildren<Renderer>(true);
			foreach (Renderer val in componentsInChildren)
			{
				val.enabled = true;
			}
		}

		private static void HideItem(Item item)
		{
			HideRenderersMethod.Invoke(item, null);
		}
	}
	internal static class CarriedPlayerDropper
	{
		private static int dropInputConsumedFrame = -1;

		internal static void Update()
		{
			if (dropInputConsumedFrame != Time.frameCount && Input.GetKeyDown((KeyCode)32))
			{
				TryDropLocalPlayer(Character.localCharacter);
			}
		}

		internal static bool HandleJumpAttempt(Character character)
		{
			if ((Object)(object)character == (Object)null || !character.IsLocal)
			{
				return false;
			}
			if (dropInputConsumedFrame == Time.frameCount)
			{
				ClearJumpInput(character);
				return true;
			}
			return TryDropLocalPlayer(character);
		}

		private static bool TryDropLocalPlayer(Character localCharacter)
		{
			if (!CanRequestDrop(localCharacter))
			{
				return false;
			}
			CharacterCarrying carriying = localCharacter.data.carrier.refs.carriying;
			if ((Object)(object)carriying == (Object)null)
			{
				return false;
			}
			ClearJumpInput(localCharacter);
			BuddyClimbRemotePassOutSync.RestoreRemoteCarryDrop(localCharacter, "local Space drop request");
			carriying.Drop(localCharacter);
			ClearJumpInput(localCharacter);
			dropInputConsumedFrame = Time.frameCount;
			return true;
		}

		private static bool CanRequestDrop(Character character)
		{
			if ((Object)(object)character != (Object)null && character.data.isCarried && (Object)(object)character.data.carrier != (Object)null && !character.data.dead && !character.data.passedOut && !character.data.fullyPassedOut)
			{
				return CharacterCarryingPatch.IsBuddyClimbCarried(character);
			}
			return false;
		}

		private static void ClearJumpInput(Character character)
		{
			if (!((Object)(object)character.input == (Object)null))
			{
				character.input.jumpWasPressed = false;
				character.input.jumpIsPressed = false;
			}
		}
	}
	internal static class CarryInteractionProxy
	{
		private const string ProxyName = "BuddyClimbInteractionProxy";

		private const float ProxyRadius = 0.75f;

		internal static void Enable(Character character)
		{
			//IL_0042: Unknown result type (might be due to invalid IL or missing references)
			//IL_0048: Expected O, but got Unknown
			//IL_005b: Unknown result type (might be due to invalid IL or missing references)
			//IL_006b: Unknown result type (might be due to invalid IL or missing references)
			//IL_007b: Unknown result type (might be due to invalid IL or missing references)
			if ((Object)(object)character == (Object)null)
			{
				return;
			}
			Transform proxyParent = GetProxyParent(character);
			if (!((Object)(object)proxyParent == (Object)null))
			{
				Transform val = proxyParent.Find("BuddyClimbInteractionProxy");
				if ((Object)(object)val != (Object)null)
				{
					((Component)val).gameObject.SetActive(true);
					return;
				}
				GameObject val2 = new GameObject("BuddyClimbInteractionProxy");
				val2.transform.SetParent(proxyParent, false);
				val2.transform.localPosition = Vector3.zero;
				val2.transform.localRotation = Quaternion.identity;
				val2.transform.localScale = Vector3.one;
				int num = LayerMask.NameToLayer("Character");
				val2.layer = ((num >= 0) ? num : ((Component)character).gameObject.layer);
				SphereCollider val3 = val2.AddComponent<SphereCollider>();
				((Collider)val3).isTrigger = true;
				val3.radius = 0.75f;
			}
		}

		internal static void Disable(Character character)
		{
			if (!((Object)(object)character == (Object)null))
			{
				Transform proxyParent = GetProxyParent(character);
				Transform val = proxyParent.Find("BuddyClimbInteractionProxy");
				if (val != null)
				{
					Object.Destroy((Object)(object)((Component)val).gameObject);
				}
			}
		}

		private static Transform GetProxyParent(Character character)
		{
			Bodypart bodypart = character.GetBodypart((BodypartType)2);
			if ((Object)(object)bodypart != (Object)null)
			{
				return ((Component)bodypart).transform;
			}
			return ((Component)character).transform;
		}
	}
}
namespace BuddyClimb.Configuration
{
	internal static class BuddyClimbConfig
	{
		private const int HotReloadDebounceMilliseconds = 250;

		private static readonly object HotReloadLock = new object();

		private static FileSystemWatcher? configWatcher;

		private static ConfigFile? configFile;

		private static Timer? reloadTimer;

		internal static ConfigEntry<bool> EnableBackpackTransfer { get; private set; } = null;


		internal static void Bind(ConfigFile config)
		{
			EnableBackpackTransfer = config.Bind<bool>("Backpack", "EnableBackpackTransfer", false, "Allow climbing onto players who are wearing a backpack. When enabled, the carrier's backpack is moved to the carried player; if the carried player already has a backpack, their old backpack is dropped first.");
		}

		internal static void EnableHotReload(ConfigFile config)
		{
			DisableHotReload();
			configFile = config;
			string directoryName = Path.GetDirectoryName(config.ConfigFilePath);
			string fileName = Path.GetFileName(config.ConfigFilePath);
			if (string.IsNullOrEmpty(directoryName) || string.IsNullOrEmpty(fileName) || !Directory.Exists(directoryName))
			{
				Plugin.Log.LogWarning((object)("Config hot reload is disabled because the config directory is unavailable: " + config.ConfigFilePath));
				return;
			}
			reloadTimer = new Timer(ReloadConfigFromTimer);
			configWatcher = new FileSystemWatcher(directoryName, fileName)
			{
				IncludeSubdirectories = false,
				NotifyFilter = (NotifyFilters.FileName | NotifyFilters.Size | NotifyFilters.LastWrite | NotifyFilters.CreationTime)
			};
			configWatcher.Changed += OnConfigFileChanged;
			configWatcher.Created += OnConfigFileChanged;
			configWatcher.Renamed += OnConfigFileChanged;
			configWatcher.EnableRaisingEvents = true;
		}

		internal static void DisableHotReload()
		{
			if (configWatcher != null)
			{
				configWatcher.EnableRaisingEvents = false;
				configWatcher.Changed -= OnConfigFileChanged;
				configWatcher.Created -= OnConfigFileChanged;
				configWatcher.Renamed -= OnConfigFileChanged;
				configWatcher.Dispose();
				configWatcher = null;
			}
			lock (HotReloadLock)
			{
				reloadTimer?.Dispose();
				reloadTimer = null;
				configFile = null;
			}
		}

		private static void OnConfigFileChanged(object sender, FileSystemEventArgs args)
		{
			lock (HotReloadLock)
			{
				reloadTimer?.Change(250, -1);
			}
		}

		private static void ReloadConfigFromTimer(object? state)
		{
			ConfigFile val;
			lock (HotReloadLock)
			{
				val = configFile;
			}
			if (val == null)
			{
				return;
			}
			try
			{
				val.Reload();
				Plugin.Log.LogInfo((object)"Reloaded BuddyClimb config from disk.");
			}
			catch (Exception ex)
			{
				Plugin.Log.LogWarning((object)("Failed to reload BuddyClimb config: " + ex.Message));
			}
		}
	}
}
namespace BuddyClimb.Compatibility
{
	internal static class ModCompatibility
	{
		internal const string PiggybackPluginId = "nakazora.peak.piggyback";

		internal static bool IsPiggybackLoaded => Chainloader.PluginInfos.ContainsKey("nakazora.peak.piggyback");
	}
}
namespace System.Diagnostics.CodeAnalysis
{
	[AttributeUsage(AttributeTargets.Parameter, Inherited = false)]
	[ExcludeFromCodeCoverage]
	internal sealed class ConstantExpectedAttribute : Attribute
	{
		public object? Min { get; set; }

		public object? Max { get; set; }
	}
	[AttributeUsage(AttributeTargets.Assembly | AttributeTargets.Module | AttributeTargets.Class | AttributeTargets.Struct | AttributeTargets.Enum | AttributeTargets.Constructor | AttributeTargets.Method | AttributeTargets.Property | AttributeTargets.Field | AttributeTargets.Event | AttributeTargets.Interface | AttributeTargets.Delegate, Inherited = false)]
	[ExcludeFromCodeCoverage]
	internal sealed class ExperimentalAttribute : Attribute
	{
		public string DiagnosticId { get; }

		public string? UrlFormat { get; set; }

		public ExperimentalAttribute(string diagnosticId)
		{
			DiagnosticId = diagnosticId;
		}
	}
	[AttributeUsage(AttributeTargets.Method | AttributeTargets.Property, Inherited = false, AllowMultiple = true)]
	[ExcludeFromCodeCoverage]
	internal sealed class MemberNotNullAttribute : Attribute
	{
		public string[] Members { get; }

		public MemberNotNullAttribute(string member)
		{
			Members = new string[1] { member };
		}

		public MemberNotNullAttribute(params string[] members)
		{
			Members = members;
		}
	}
	[AttributeUsage(AttributeTargets.Method | AttributeTargets.Property, Inherited = false, AllowMultiple = true)]
	[ExcludeFromCodeCoverage]
	internal sealed class MemberNotNullWhenAttribute : Attribute
	{
		public bool ReturnValue { get; }

		public string[] Members { get; }

		public MemberNotNullWhenAttribute(bool returnValue, string member)
		{
			ReturnValue = returnValue;
			Members = new string[1] { member };
		}

		public MemberNotNullWhenAttribute(bool returnValue, params string[] members)
		{
			ReturnValue = returnValue;
			Members = members;
		}
	}
	[AttributeUsage(AttributeTargets.Constructor, AllowMultiple = false, Inherited = false)]
	[ExcludeFromCodeCoverage]
	internal sealed class SetsRequiredMembersAttribute : Attribute
	{
	}
	[AttributeUsage(AttributeTargets.Property | AttributeTargets.Field | AttributeTargets.Parameter, AllowMultiple = false, Inherited = false)]
	[ExcludeFromCodeCoverage]
	internal sealed class StringSyntaxAttribute : Attribute
	{
		public const string CompositeFormat = "CompositeFormat";

		public const string DateOnlyFormat = "DateOnlyFormat";

		public const string DateTimeFormat = "DateTimeFormat";

		public const string EnumFormat = "EnumFormat";

		public const string GuidFormat = "GuidFormat";

		public const string Json = "Json";

		public const string NumericFormat = "NumericFormat";

		public const string Regex = "Regex";

		public const string TimeOnlyFormat = "TimeOnlyFormat";

		public const string TimeSpanFormat = "TimeSpanFormat";

		public const string Uri = "Uri";

		public const string Xml = "Xml";

		public string Syntax { get; }

		public object?[] Arguments { get; }

		public StringSyntaxAttribute(string syntax)
		{
			Syntax = syntax;
			Arguments = new object[0];
		}

		public StringSyntaxAttribute(string syntax, params object?[] arguments)
		{
			Syntax = syntax;
			Arguments = arguments;
		}
	}
	[AttributeUsage(AttributeTargets.Method | AttributeTargets.Property | AttributeTargets.Parameter, AllowMultiple = false, Inherited = false)]
	[ExcludeFromCodeCoverage]
	internal sealed class UnscopedRefAttribute : Attribute
	{
	}
}
namespace System.Runtime.Versioning
{
	[AttributeUsage(AttributeTargets.Assembly | AttributeTargets.Module | AttributeTargets.Class | AttributeTargets.Struct | AttributeTargets.Enum | AttributeTargets.Constructor | AttributeTargets.Method | AttributeTargets.Property | AttributeTargets.Field | AttributeTargets.Event | AttributeTargets.Interface | AttributeTargets.Delegate, Inherited = false)]
	[ExcludeFromCodeCoverage]
	internal sealed class RequiresPreviewFeaturesAttribute : Attribute
	{
		public string? Message { get; }

		public string? Url { get; set; }

		public RequiresPreviewFeaturesAttribute()
		{
		}

		public RequiresPreviewFeaturesAttribute(string? message)
		{
			Message = message;
		}
	}
}
namespace System.Runtime.CompilerServices
{
	[AttributeUsage(AttributeTargets.Assembly, AllowMultiple = true)]
	internal sealed class IgnoresAccessChecksToAttribute : Attribute
	{
		public IgnoresAccessChecksToAttribute(string assemblyName)
		{
		}
	}
	[AttributeUsage(AttributeTargets.Parameter, AllowMultiple = false, Inherited = false)]
	[ExcludeFromCodeCoverage]
	internal sealed class CallerArgumentExpressionAttribute : Attribute
	{
		public string ParameterName { get; }

		public CallerArgumentExpressionAttribute(string parameterName)
		{
			ParameterName = parameterName;
		}
	}
	[AttributeUsage(AttributeTargets.Class | AttributeTargets.Struct | AttributeTargets.Interface, Inherited = false)]
	[ExcludeFromCodeCoverage]
	internal sealed class CollectionBuilderAttribute : Attribute
	{
		public Type BuilderType { get; }

		public string MethodName { get; }

		public CollectionBuilderAttribute(Type builderType, string methodName)
		{
			BuilderType = builderType;
			MethodName = methodName;
		}
	}
	[AttributeUsage(AttributeTargets.All, AllowMultiple = true, Inherited = false)]
	[ExcludeFromCodeCoverage]
	internal sealed class CompilerFeatureRequiredAttribute : Attribute
	{
		public const string RefStructs = "RefStructs";

		public const string RequiredMembers = "RequiredMembers";

		public string FeatureName { get; }

		public bool IsOptional { get; set; }

		public CompilerFeatureRequiredAttribute(string featureName)
		{
			FeatureName = featureName;
		}
	}
	[AttributeUsage(AttributeTargets.Parameter, AllowMultiple = false, Inherited = false)]
	[ExcludeFromCodeCoverage]
	internal sealed class InterpolatedStringHandlerArgumentAttribute : Attribute
	{
		public string[] Arguments { get; }

		public InterpolatedStringHandlerArgumentAttribute(string argument)
		{
			Arguments = new string[1] { argument };
		}

		public InterpolatedStringHandlerArgumentAttribute(params string[] arguments)
		{
			Arguments = arguments;
		}
	}
	[AttributeUsage(AttributeTargets.Class | AttributeTargets.Struct, AllowMultiple = false, Inherited = false)]
	[ExcludeFromCodeCoverage]
	internal sealed class InterpolatedStringHandlerAttribute : Attribute
	{
	}
	[EditorBrowsable(EditorBrowsableState.Never)]
	[ExcludeFromCodeCoverage]
	internal static class IsExternalInit
	{
	}
	[AttributeUsage(AttributeTargets.Method, Inherited = false)]
	[ExcludeFromCodeCoverage]
	internal sealed class ModuleInitializerAttribute : Attribute
	{
	}
	[AttributeUsage(AttributeTargets.Constructor | AttributeTargets.Method | AttributeTargets.Property, AllowMultiple = false, Inherited = false)]
	[ExcludeFromCodeCoverage]
	internal sealed class OverloadResolutionPriorityAttribute : Attribute
	{
		public int Priority { get; }

		public OverloadResolutionPriorityAttribute(int priority)
		{
			Priority = priority;
		}
	}
	[AttributeUsage(AttributeTargets.Parameter, Inherited = true, AllowMultiple = false)]
	[ExcludeFromCodeCoverage]
	internal sealed class ParamCollectionAttribute : Attribute
	{
	}
	[AttributeUsage(AttributeTargets.Class | AttributeTargets.Struct | AttributeTargets.Property | AttributeTargets.Field, AllowMultiple = false, Inherited = false)]
	[ExcludeFromCodeCoverage]
	internal sealed class RequiredMemberAttribute : Attribute
	{
	}
	[AttributeUsage(AttributeTargets.Parameter, Inherited = false)]
	[EditorBrowsable(EditorBrowsableState.Never)]
	[ExcludeFromCodeCoverage]
	internal sealed class RequiresLocationAttribute : Attribute
	{
	}
	[AttributeUsage(AttributeTargets.Module | AttributeTargets.Class | AttributeTargets.Struct | AttributeTargets.Constructor | AttributeTargets.Method | AttributeTargets.Property | AttributeTargets.Event | AttributeTargets.Interface, Inherited = false)]
	[ExcludeFromCodeCoverage]
	internal sealed class SkipLocalsInitAttribute : Attribute
	{
	}
}