Decompiled source of ShoppingList v0.2.0

BepInEx/plugins/ShoppingList.dll

Decompiled 2 months ago
using System;
using System.Collections;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Reflection;
using System.Runtime.CompilerServices;
using System.Runtime.Versioning;
using System.Security;
using System.Security.Permissions;
using System.Text;
using BepInEx;
using BepInEx.Logging;
using HarmonyLib;
using KaimiraGames;
using Microsoft.CodeAnalysis;
using StbRectPackSharp;
using Unity.Netcode;
using Unity.VisualScripting;
using UnityEngine;

[assembly: CompilationRelaxations(8)]
[assembly: RuntimeCompatibility(WrapNonExceptionThrows = true)]
[assembly: Debuggable(DebuggableAttribute.DebuggingModes.Default | DebuggableAttribute.DebuggingModes.DisableOptimizations | DebuggableAttribute.DebuggingModes.IgnoreSymbolStoreSequencePoints | DebuggableAttribute.DebuggingModes.EnableEditAndContinue)]
[assembly: TargetFramework(".NETFramework,Version=v4.8", FrameworkDisplayName = ".NET Framework 4.8")]
[assembly: AssemblyCompany("ShoppingList")]
[assembly: AssemblyConfiguration("Debug")]
[assembly: AssemblyFileVersion("0.1.0.0")]
[assembly: AssemblyInformationalVersion("0.1.0")]
[assembly: AssemblyProduct("Shopping List")]
[assembly: AssemblyTitle("ShoppingList")]
[assembly: SecurityPermission(SecurityAction.RequestMinimum, SkipVerification = true)]
[assembly: AssemblyVersion("0.1.0.0")]
[module: UnverifiableCode]
[module: RefSafetyRules(11)]
namespace Microsoft.CodeAnalysis
{
	[CompilerGenerated]
	[Microsoft.CodeAnalysis.Embedded]
	internal sealed class EmbeddedAttribute : Attribute
	{
	}
}
namespace System.Runtime.CompilerServices
{
	[CompilerGenerated]
	[Microsoft.CodeAnalysis.Embedded]
	[AttributeUsage(AttributeTargets.Module, AllowMultiple = false, Inherited = false)]
	internal sealed class RefSafetyRulesAttribute : Attribute
	{
		public readonly int Version;

		public RefSafetyRulesAttribute(int P_0)
		{
			Version = P_0;
		}
	}
}
namespace KaimiraGames
{
	public class WeightedList<T> : IEnumerable<T>, IEnumerable
	{
		private readonly List<T> _list = new List<T>();

		private readonly List<int> _weights = new List<int>();

		private readonly List<int> _probabilities = new List<int>();

		private readonly List<int> _alias = new List<int>();

		private readonly Random _rand;

		private int _totalWeight;

		private bool _areAllProbabilitiesIdentical = false;

		private int _minWeight;

		private int _maxWeight;

		public WeightErrorHandlingType BadWeightErrorHandling { get; set; } = WeightErrorHandlingType.SetWeightToOne;


		public int TotalWeight => _totalWeight;

		public int MinWeight => _minWeight;

		public int MaxWeight => _maxWeight;

		public IReadOnlyList<T> Items => _list.AsReadOnly();

		public T this[int index] => _list[index];

		public int Count => _list.Count;

		public WeightedList(Random rand = null)
		{
			_rand = rand ?? new Random();
		}

		public WeightedList(ICollection<WeightedListItem<T>> listItems, Random rand = null)
		{
			_rand = rand ?? new Random();
			foreach (WeightedListItem<T> listItem in listItems)
			{
				_list.Add(listItem._item);
				_weights.Add(listItem._weight);
			}
			Recalculate();
		}

		public T Next()
		{
			if (Count == 0)
			{
				return default(T);
			}
			int index = _rand.Next(Count);
			if (_areAllProbabilitiesIdentical)
			{
				return _list[index];
			}
			int num = _rand.Next(_totalWeight);
			return (num < _probabilities[index]) ? _list[index] : _list[_alias[index]];
		}

		public void AddWeightToAll(int weight)
		{
			if (weight + _minWeight <= 0 && BadWeightErrorHandling == WeightErrorHandlingType.ThrowExceptionOnAdd)
			{
				throw new ArgumentException($"Subtracting {-1 * weight} from all items would set weight to non-positive for at least one element.");
			}
			for (int i = 0; i < Count; i++)
			{
				_weights[i] = FixWeight(_weights[i] + weight);
			}
			Recalculate();
		}

		public void SubtractWeightFromAll(int weight)
		{
			AddWeightToAll(weight * -1);
		}

		public void SetWeightOfAll(int weight)
		{
			if (weight <= 0 && BadWeightErrorHandling == WeightErrorHandlingType.ThrowExceptionOnAdd)
			{
				throw new ArgumentException("Weight cannot be non-positive.");
			}
			for (int i = 0; i < Count; i++)
			{
				_weights[i] = FixWeight(weight);
			}
			Recalculate();
		}

		public IEnumerator<T> GetEnumerator()
		{
			return _list.GetEnumerator();
		}

		IEnumerator IEnumerable.GetEnumerator()
		{
			return _list.GetEnumerator();
		}

		public void Add(T item, int weight)
		{
			_list.Add(item);
			_weights.Add(FixWeight(weight));
			Recalculate();
		}

		public void Add(ICollection<WeightedListItem<T>> listItems)
		{
			foreach (WeightedListItem<T> listItem in listItems)
			{
				_list.Add(listItem._item);
				_weights.Add(FixWeight(listItem._weight));
			}
			Recalculate();
		}

		public void Clear()
		{
			_list.Clear();
			_weights.Clear();
			Recalculate();
		}

		public bool Contains(T item)
		{
			return _list.Contains(item);
		}

		public int IndexOf(T item)
		{
			return _list.IndexOf(item);
		}

		public void Insert(int index, T item, int weight)
		{
			_list.Insert(index, item);
			_weights.Insert(index, FixWeight(weight));
			Recalculate();
		}

		public void Remove(T item)
		{
			int index = IndexOf(item);
			RemoveAt(index);
			Recalculate();
		}

		public void RemoveAt(int index)
		{
			_list.RemoveAt(index);
			_weights.RemoveAt(index);
			Recalculate();
		}

		public void SetWeight(T item, int newWeight)
		{
			SetWeightAtIndex(IndexOf(item), FixWeight(newWeight));
		}

		public int GetWeightOf(T item)
		{
			return GetWeightAtIndex(IndexOf(item));
		}

		public void SetWeightAtIndex(int index, int newWeight)
		{
			_weights[index] = FixWeight(newWeight);
			Recalculate();
		}

		public int GetWeightAtIndex(int index)
		{
			return _weights[index];
		}

		public override string ToString()
		{
			StringBuilder stringBuilder = new StringBuilder();
			stringBuilder.Append("WeightedList<");
			stringBuilder.Append(typeof(T).Name);
			stringBuilder.Append(">: TotalWeight:");
			stringBuilder.Append(TotalWeight);
			stringBuilder.Append(", Min:");
			stringBuilder.Append(_minWeight);
			stringBuilder.Append(", Max:");
			stringBuilder.Append(_maxWeight);
			stringBuilder.Append(", Count:");
			stringBuilder.Append(Count);
			stringBuilder.Append(", {");
			for (int i = 0; i < _list.Count; i++)
			{
				stringBuilder.Append(_list[i].ToString());
				stringBuilder.Append(":");
				stringBuilder.Append(_weights[i].ToString());
				if (i < _list.Count - 1)
				{
					stringBuilder.Append(", ");
				}
			}
			stringBuilder.Append("}");
			return stringBuilder.ToString();
		}

		private void Recalculate()
		{
			_totalWeight = 0;
			_areAllProbabilitiesIdentical = false;
			_minWeight = 0;
			_maxWeight = 0;
			bool flag = true;
			_alias.Clear();
			_probabilities.Clear();
			List<int> list = new List<int>(Count);
			List<int> list2 = new List<int>(Count);
			List<int> list3 = new List<int>(Count);
			foreach (int weight in _weights)
			{
				if (flag)
				{
					_minWeight = (_maxWeight = weight);
					flag = false;
				}
				_minWeight = ((weight < _minWeight) ? weight : _minWeight);
				_maxWeight = ((_maxWeight < weight) ? weight : _maxWeight);
				_totalWeight += weight;
				list.Add(weight * Count);
				_alias.Add(0);
				_probabilities.Add(0);
			}
			if (_minWeight == _maxWeight)
			{
				_areAllProbabilitiesIdentical = true;
				return;
			}
			for (int i = 0; i < Count; i++)
			{
				if (list[i] < _totalWeight)
				{
					list2.Add(i);
				}
				else
				{
					list3.Add(i);
				}
			}
			while (list2.Count > 0 && list3.Count > 0)
			{
				int index = list2[list2.Count - 1];
				list2.RemoveAt(list2.Count - 1);
				int num = list3[list3.Count - 1];
				list3.RemoveAt(list3.Count - 1);
				_probabilities[index] = list[index];
				_alias[index] = num;
				int num3 = (list[num] = list[num] + list[index] - _totalWeight);
				if (num3 < _totalWeight)
				{
					list2.Add(num);
				}
				else
				{
					list3.Add(num);
				}
			}
			while (list3.Count > 0)
			{
				int index2 = list3[list3.Count - 1];
				list3.RemoveAt(list3.Count - 1);
				_probabilities[index2] = _totalWeight;
			}
		}

		internal static int FixWeightSetToOne(int weight)
		{
			return (weight <= 0) ? 1 : weight;
		}

		internal static int FixWeightExceptionOnAdd(int weight)
		{
			if (weight > 0)
			{
				return weight;
			}
			throw new ArgumentException("Weight cannot be non-positive");
		}

		private int FixWeight(int weight)
		{
			return (BadWeightErrorHandling == WeightErrorHandlingType.ThrowExceptionOnAdd) ? FixWeightExceptionOnAdd(weight) : FixWeightSetToOne(weight);
		}
	}
	public readonly struct WeightedListItem<T>
	{
		internal readonly T _item;

		internal readonly int _weight;

		public WeightedListItem(T item, int weight)
		{
			_item = item;
			_weight = weight;
		}
	}
	public enum WeightErrorHandlingType
	{
		SetWeightToOne,
		ThrowExceptionOnAdd
	}
}
namespace System.Runtime.CompilerServices
{
	public class ExtensionAttribute : Attribute
	{
	}
}
namespace ShoppingList
{
	[BepInPlugin("yasenfire.shoppinglist", "ShoppingList", "0.1.0")]
	public class Plugin : BaseUnityPlugin
	{
		[HarmonyPatch(typeof(CustomerController), "SetupShoppingList")]
		private class SetupShoppingListPatch
		{
			private static bool Prefix(CustomerController __instance)
			{
				__instance.shoppingList.Clear();
				List<ProductSO> list = GameManager.Instance.itemDatabase.Where((ItemSO x) => x is ProductSO).Cast<ProductSO>().ToList();
				List<WeightedListItem<ProductSO>> list2 = new List<WeightedListItem<ProductSO>>();
				SeasonSO currentSeason = GameManager.Instance.GetCurrentSeason();
				EventSO currentEvent = GameManager.Instance.GetCurrentEvent();
				foreach (ProductSO item2 in list)
				{
					float num = 100f;
					num = ((!((Object)(object)currentSeason == (Object)(object)item2.season)) ? (num * 0.9f) : (num * 1.2f));
					float num2 = GameManager.Instance.GetSellingPriceServer(item2);
					float num3 = GameManager.Instance.GetRecommendedPrice(item2);
					float num4 = num3 / num2;
					num = ((!(num4 > 1f)) ? (num * Mathf.Lerp(0.1f, 1f, (Mathf.Clamp(num4, 0.8f, 1f) - 0.8f) * 10f)) : (num * Mathf.Lerp(1f, 8f, Mathf.Clamp(num4, 1f, 2f) - 1f)));
					num *= Random.Range(0.8f, 1.2f);
					if (currentEvent != null)
					{
						if (currentEvent.forbiddenProducts.Contains(item2))
						{
							num *= 0.1f;
						}
						if (currentEvent.products.Contains(item2))
						{
							num *= Random.Range(2f, 4f);
						}
					}
					list2.Add(new WeightedListItem<ProductSO>(item2, Mathf.RoundToInt(num)));
				}
				WeightedList<ProductSO> weightedList = new WeightedList<ProductSO>(list2);
				List<ProductSO> list3 = (from x in LinqUtility.DistinctBy<Item, long>(from x in Object.FindObjectsOfType<Item>()
						where x.itemSO is ProductSO && x.onStand.Value && x.amount.Value > 0
						select x, (Func<Item, long>)((Item x) => x.itemSO.id))
					select x.itemSO).Cast<ProductSO>().ToList();
				float num5 = (float)list3.Count / (float)list.Count;
				string text = weightedBuyerList.Next();
				if (text == "obsessive")
				{
					ProductSO val = weightedList.Next();
					int num6 = Random.Range(4, Mathf.Min(((ItemSO)val).amount, 24));
					for (int i = 0; i < num6; i++)
					{
						__instance.shoppingList.Add(val);
					}
					return false;
				}
				if (text == "loyal")
				{
					int num7 = Mathf.RoundToInt(Mathf.Lerp(0f, 18f, num5)) + 6;
					int num8 = Random.Range(1, num7);
					for (int j = 0; j < num8; j++)
					{
						ProductSO item;
						do
						{
							item = weightedList.Next();
						}
						while (!list3.Contains(item));
						__instance.shoppingList.Add(item);
					}
					return false;
				}
				int num9 = Random.Range(1, 24);
				int num10 = 0;
				List<ProductSO> list4 = new List<ProductSO>();
				for (int k = 0; k < num9; k++)
				{
					ProductSO val2 = weightedList.Next();
					list4.Add(val2);
					int num11 = 1;
					for (; k < num9; k++)
					{
						if (!(Random.Range(0f, 1f) < (float)(((ItemSO)val2).amount / 100 / num11)))
						{
							break;
						}
						list4.Add(val2);
						num11++;
					}
					if (list3.Contains(val2))
					{
						num10 += num11;
					}
				}
				if ((float)(num10 / num9) >= 0.5f)
				{
					for (int l = 0; l < list4.Count; l++)
					{
						if (list3.Contains(list4[l]))
						{
							__instance.shoppingList.Add(list4[l]);
						}
					}
				}
				return false;
			}
		}

		[HarmonyPatch(typeof(Checkout), "SpawnProducts")]
		private class CheckoutSpawnProductsPatch
		{
			private static bool Prefix(Checkout __instance)
			{
				//IL_0067: Unknown result type (might be due to invalid IL or missing references)
				//IL_0078: 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_00a2: Unknown result type (might be due to invalid IL or missing references)
				//IL_0103: Unknown result type (might be due to invalid IL or missing references)
				//IL_010a: Expected O, but got Unknown
				//IL_0138: Unknown result type (might be due to invalid IL or missing references)
				//IL_013f: Unknown result type (might be due to invalid IL or missing references)
				//IL_0162: Unknown result type (might be due to invalid IL or missing references)
				//IL_0167: Unknown result type (might be due to invalid IL or missing references)
				//IL_016b: Unknown result type (might be due to invalid IL or missing references)
				//IL_0170: Unknown result type (might be due to invalid IL or missing references)
				//IL_0174: Unknown result type (might be due to invalid IL or missing references)
				//IL_018d: 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_01b6: Unknown result type (might be due to invalid IL or missing references)
				//IL_0271: Unknown result type (might be due to invalid IL or missing references)
				//IL_02af: Unknown result type (might be due to invalid IL or missing references)
				//IL_02b4: Unknown result type (might be due to invalid IL or missing references)
				if (__instance.spawnedProducts)
				{
					return false;
				}
				object value = Traverse.Create((object)__instance).Field("checkoutEmployee").GetValue();
				CheckoutEmployee val = (CheckoutEmployee)((value is CheckoutEmployee) ? value : null);
				List<CustomerController> list = Traverse.Create((object)__instance).Field("customers").GetValue() as List<CustomerController>;
				Transform child = __instance.productSlots.GetChild(0);
				Transform child2 = __instance.productSlots.GetChild(8);
				float num = Mathf.Abs(child2.localPosition.x * 1000f - child.localPosition.x * 1000f);
				float num2 = Mathf.Abs(child2.localPosition.z * 1000f - child.localPosition.z * 1000f) * 2f;
				List<ProductSO> shoppingCart = list[0].shoppingCart;
				shoppingCart.Sort((ProductSO x, ProductSO y) => ((ItemSO)x).id.CompareTo(((ItemSO)y).id));
				Packer val2 = new Packer(Mathf.RoundToInt(num), Mathf.RoundToInt(num2));
				for (int i = 0; i < shoppingCart.Count; i++)
				{
					Transform child3 = __instance.productSlots.GetChild(0);
					ProductSO val3 = shoppingCart[i];
					GameObject item = Object.Instantiate<GameObject>(GameManager.Instance.checkoutItem, child3.position, child3.rotation);
					Mesh sharedMesh = val3.pickupPrefab.GetComponentsInChildren<MeshFilter>()[0].sharedMesh;
					Bounds bounds = sharedMesh.bounds;
					Vector3 size = ((Bounds)(ref bounds)).size;
					val2.PackRect(Mathf.RoundToInt(size.z * 1000f * val3.pickupPrefab.transform.localScale.z), Mathf.RoundToInt(size.x * 1000f * val3.pickupPrefab.transform.localScale.x), (object)(item, ((ItemSO)val3).id));
				}
				foreach (PackerRectangle packRectangle in val2.PackRectangles)
				{
					(GameObject, long) obj = ((GameObject, long))packRectangle.Data;
					GameObject item2 = obj.Item1;
					long item3 = obj.Item2;
					NetworkObject component = item2.GetComponent<NetworkObject>();
					component.Spawn(false);
					((Component)component).GetComponent<CheckoutItem>().ServerSetupItem(item3, __instance);
					component.TrySetParent(((Component)__instance).gameObject.transform, true);
					Transform transform = item2.transform;
					transform.localPosition += new Vector3((float)packRectangle.Rectangle.Y / 1000f, 0f, (float)packRectangle.Rectangle.X / 1000f * -1f);
				}
				__instance.spawnedProducts = true;
				if ((Object)(object)val != (Object)null)
				{
					val.Work();
				}
				return false;
			}
		}

		internal static ManualLogSource Logger;

		private Harmony m_harmony = new Harmony("yasenfire.shoppinglist");

		public static WeightedList<string> weightedBuyerList;

		private void Awake()
		{
			Logger = ((BaseUnityPlugin)this).Logger;
			Logger.LogInfo((object)"Plugin ShoppingList is loaded!");
			m_harmony.PatchAll();
			List<WeightedListItem<string>> listItems = new List<WeightedListItem<string>>
			{
				new WeightedListItem<string>("obsessive", 1),
				new WeightedListItem<string>("loyal", 20),
				new WeightedListItem<string>("standard", 79)
			};
			weightedBuyerList = new WeightedList<string>(listItems);
		}
	}
}

BepInEx/plugins/StbRectPackSharp.dll

Decompiled 2 months ago
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Drawing;
using System.Reflection;
using System.Runtime.CompilerServices;
using System.Runtime.InteropServices;
using System.Runtime.Versioning;
using System.Security;
using System.Security.Permissions;

[assembly: CompilationRelaxations(8)]
[assembly: RuntimeCompatibility(WrapNonExceptionThrows = true)]
[assembly: Debuggable(DebuggableAttribute.DebuggingModes.IgnoreSymbolStoreSequencePoints)]
[assembly: TargetFramework(".NETFramework,Version=v4.5", FrameworkDisplayName = ".NET Framework 4.5")]
[assembly: AssemblyCompany("StbRectPackSharpTeam")]
[assembly: AssemblyConfiguration("Release")]
[assembly: AssemblyDescription("C# port of the stb_rect_pack.h")]
[assembly: AssemblyFileVersion("1.0.4.0")]
[assembly: AssemblyInformationalVersion("1.00.4")]
[assembly: AssemblyProduct("StbRectPackSharp")]
[assembly: AssemblyTitle("StbRectPackSharp")]
[assembly: SecurityPermission(SecurityAction.RequestMinimum, SkipVerification = true)]
[assembly: AssemblyVersion("1.0.4.0")]
[module: UnverifiableCode]
namespace StbRectPackSharp;

internal static class CRuntime
{
	public unsafe delegate int QSortComparer(void* a, void* b);

	public unsafe static void* malloc(ulong size)
	{
		return malloc((long)size);
	}

	public unsafe static void* malloc(long size)
	{
		return Marshal.AllocHGlobal((int)size).ToPointer();
	}

	public unsafe static void free(void* a)
	{
		Marshal.FreeHGlobal(new IntPtr(a));
	}

	private unsafe static void qsortSwap(byte* data, long size, long pos1, long pos2)
	{
		byte* ptr = data + size * pos1;
		byte* ptr2 = data + size * pos2;
		for (long num = 0L; num < size; num++)
		{
			byte b = *ptr;
			*ptr = *ptr2;
			*ptr2 = b;
			ptr++;
			ptr2++;
		}
	}

	private unsafe static long qsortPartition(byte* data, long size, QSortComparer comparer, long left, long right)
	{
		void* b = data + size * left;
		long num = left - 1;
		long num2 = right + 1;
		while (true)
		{
			num++;
			if (comparer(data + size * num, b) >= 0)
			{
				do
				{
					num2--;
				}
				while (comparer(data + size * num2, b) > 0);
				if (num >= num2)
				{
					break;
				}
				qsortSwap(data, size, num, num2);
			}
		}
		return num2;
	}

	private unsafe static void qsortInternal(byte* data, long size, QSortComparer comparer, long left, long right)
	{
		if (left < right)
		{
			long num = qsortPartition(data, size, comparer, left, right);
			qsortInternal(data, size, comparer, left, num);
			qsortInternal(data, size, comparer, num + 1, right);
		}
	}

	public unsafe static void qsort(void* data, ulong count, ulong size, QSortComparer comparer)
	{
		qsortInternal((byte*)data, (long)size, comparer, 0L, (long)(count - 1));
	}
}
public class PackerRectangle
{
	public Rectangle Rectangle { get; private set; }

	public int X => Rectangle.X;

	public int Y => Rectangle.Y;

	public int Width => Rectangle.Width;

	public int Height => Rectangle.Height;

	public object Data { get; private set; }

	public PackerRectangle(Rectangle rect, object data)
	{
		Rectangle = rect;
		Data = data;
	}
}
public class Packer : IDisposable
{
	private StbRectPack.stbrp_context _context;

	private List<PackerRectangle> _rectangles = new List<PackerRectangle>();

	public int Width => _context.width;

	public int Height => _context.height;

	public List<PackerRectangle> PackRectangles => _rectangles;

	public Packer(int initialWidth = 256, int initialHeight = 256)
	{
		InitContext(initialWidth, initialHeight);
	}

	public void Dispose()
	{
		_context.Dispose();
	}

	private unsafe void InitContext(int width, int height)
	{
		if (width <= 0)
		{
			throw new ArgumentOutOfRangeException("width");
		}
		if (height <= 0)
		{
			throw new ArgumentOutOfRangeException("height");
		}
		_context.Dispose();
		StbRectPack.stbrp_context context = new StbRectPack.stbrp_context(width);
		StbRectPack.stbrp_init_target(&context, width, height, context.all_nodes, width);
		_context = context;
		_rectangles = new List<PackerRectangle>();
	}

	public unsafe PackerRectangle PackRect(int width, int height, object userData)
	{
		StbRectPack.stbrp_rect stbrp_rect = default(StbRectPack.stbrp_rect);
		stbrp_rect.id = _rectangles.Count;
		stbrp_rect.w = width;
		stbrp_rect.h = height;
		StbRectPack.stbrp_rect stbrp_rect2 = stbrp_rect;
		int num;
		fixed (StbRectPack.stbrp_context* context = &_context)
		{
			num = StbRectPack.stbrp_pack_rects(context, &stbrp_rect2, 1);
		}
		if (num == 0)
		{
			return null;
		}
		PackerRectangle packerRectangle = new PackerRectangle(new Rectangle(stbrp_rect2.x, stbrp_rect2.y, stbrp_rect2.w, stbrp_rect2.h), userData);
		_rectangles.Add(packerRectangle);
		return packerRectangle;
	}
}
public static class StbRectPack
{
	public struct stbrp_context : IDisposable
	{
		public int width;

		public int height;

		public int align;

		public int init_mode;

		public int heuristic;

		public int num_nodes;

		public unsafe stbrp_node* active_head;

		public unsafe stbrp_node* free_head;

		public unsafe stbrp_node* extra;

		public unsafe stbrp_node* all_nodes;

		public unsafe stbrp_context(int nodesCount)
		{
			if (nodesCount <= 0)
			{
				throw new ArgumentOutOfRangeException("nodesCount");
			}
			width = (height = (align = (init_mode = (heuristic = (num_nodes = 0)))));
			active_head = (free_head = null);
			all_nodes = (stbrp_node*)CRuntime.malloc(sizeof(stbrp_node) * nodesCount);
			extra = (stbrp_node*)CRuntime.malloc(sizeof(stbrp_node) * 2);
		}

		public unsafe void Dispose()
		{
			if (all_nodes != null)
			{
				CRuntime.free(all_nodes);
				all_nodes = null;
			}
			if (extra != null)
			{
				CRuntime.free(extra);
				extra = null;
			}
		}
	}

	public struct stbrp_rect
	{
		public int id;

		public int w;

		public int h;

		public int x;

		public int y;

		public int was_packed;
	}

	public struct stbrp_node
	{
		public int x;

		public int y;

		public unsafe stbrp_node* next;
	}

	public struct stbrp__findresult
	{
		public int x;

		public int y;

		public unsafe stbrp_node** prev_link;
	}

	public const int STBRP_HEURISTIC_Skyline_default = 0;

	public const int STBRP_HEURISTIC_Skyline_BL_sortHeight = 0;

	public const int STBRP_HEURISTIC_Skyline_BF_sortHeight = 2;

	public const int STBRP__INIT_skyline = 1;

	public unsafe static void stbrp_setup_heuristic(stbrp_context* context, int heuristic)
	{
		if (context->init_mode == 1)
		{
			context->heuristic = heuristic;
			return;
		}
		throw new Exception("Mode " + context->init_mode + " is not supported.");
	}

	public unsafe static void stbrp_setup_allow_out_of_mem(stbrp_context* context, int allow_out_of_mem)
	{
		if (allow_out_of_mem != 0)
		{
			context->align = 1;
		}
		else
		{
			context->align = (context->width + context->num_nodes - 1) / context->num_nodes;
		}
	}

	public unsafe static void stbrp_init_target(stbrp_context* context, int width, int height, stbrp_node* nodes, int num_nodes)
	{
		int num = 0;
		for (num = 0; num < num_nodes - 1; num++)
		{
			nodes[num].next = nodes + (num + 1);
		}
		nodes[num].next = null;
		context->init_mode = 1;
		context->heuristic = 0;
		context->free_head = nodes;
		context->active_head = context->extra;
		context->width = width;
		context->height = height;
		context->num_nodes = num_nodes;
		stbrp_setup_allow_out_of_mem(context, 0);
		context->extra->x = 0;
		context->extra->y = 0;
		context->extra->next = context->extra + 1;
		context->extra[1].x = width;
		context->extra[1].y = 65535;
		context->extra[1].next = null;
	}

	public unsafe static int stbrp__skyline_find_min_y(stbrp_context* c, stbrp_node* first, int x0, int width, int* pwaste)
	{
		stbrp_node* ptr = first;
		int num = x0 + width;
		int num2 = 0;
		int num3 = 0;
		int num4 = 0;
		num2 = 0;
		num4 = 0;
		num3 = 0;
		while (ptr->x < num)
		{
			if (ptr->y > num2)
			{
				num4 += num3 * (ptr->y - num2);
				num2 = ptr->y;
				num3 = ((ptr->x >= x0) ? (num3 + (ptr->next->x - ptr->x)) : (num3 + (ptr->next->x - x0)));
			}
			else
			{
				int num5 = ptr->next->x - ptr->x;
				if (num5 + num3 > width)
				{
					num5 = width - num3;
				}
				num4 += num5 * (num2 - ptr->y);
				num3 += num5;
			}
			ptr = ptr->next;
		}
		*pwaste = num4;
		return num2;
	}

	public unsafe static stbrp__findresult stbrp__skyline_find_best_pos(stbrp_context* c, int width, int height)
	{
		int num = 1073741824;
		int num2 = 0;
		int num3 = 1073741824;
		stbrp__findresult result = default(stbrp__findresult);
		stbrp_node** ptr = null;
		width = width + c->align - 1;
		width -= width % c->align;
		if (width > c->width || height > c->height)
		{
			result.prev_link = null;
			result.x = (result.y = 0);
			return result;
		}
		stbrp_node* ptr2 = c->active_head;
		stbrp_node** ptr3 = &c->active_head;
		while (ptr2->x + width <= c->width)
		{
			int num4 = 0;
			int num5 = 0;
			num4 = stbrp__skyline_find_min_y(c, ptr2, ptr2->x, width, &num5);
			if (c->heuristic == 0)
			{
				if (num4 < num3)
				{
					num3 = num4;
					ptr = ptr3;
				}
			}
			else if (num4 + height <= c->height && (num4 < num3 || (num4 == num3 && num5 < num)))
			{
				num3 = num4;
				num = num5;
				ptr = ptr3;
			}
			ptr3 = &ptr2->next;
			ptr2 = ptr2->next;
		}
		num2 = ((ptr != null) ? (*ptr)->x : 0);
		if (c->heuristic == 2)
		{
			stbrp_node* ptr4 = c->active_head;
			ptr2 = c->active_head;
			ptr3 = &c->active_head;
			while (ptr4->x < width)
			{
				ptr4 = ptr4->next;
			}
			while (ptr4 != null)
			{
				int num6 = ptr4->x - width;
				int num7 = 0;
				int num8 = 0;
				while (ptr2->next->x <= num6)
				{
					ptr3 = &ptr2->next;
					ptr2 = ptr2->next;
				}
				num7 = stbrp__skyline_find_min_y(c, ptr2, num6, width, &num8);
				if (num7 + height <= c->height && num7 <= num3 && (num7 < num3 || num8 < num || (num8 == num && num6 < num2)))
				{
					num2 = num6;
					num3 = num7;
					num = num8;
					ptr = ptr3;
				}
				ptr4 = ptr4->next;
			}
		}
		result.prev_link = ptr;
		result.x = num2;
		result.y = num3;
		return result;
	}

	public unsafe static stbrp__findresult stbrp__skyline_pack_rectangle(stbrp_context* context, int width, int height)
	{
		stbrp__findresult result = stbrp__skyline_find_best_pos(context, width, height);
		if (result.prev_link == null || result.y + height > context->height || context->free_head == null)
		{
			result.prev_link = null;
			return result;
		}
		stbrp_node* free_head = context->free_head;
		free_head->x = result.x;
		free_head->y = result.y + height;
		context->free_head = free_head->next;
		stbrp_node* ptr = *result.prev_link;
		if (ptr->x < result.x)
		{
			stbrp_node* next = ptr->next;
			ptr->next = free_head;
			ptr = next;
		}
		else
		{
			*result.prev_link = free_head;
		}
		while (ptr->next != null && ptr->next->x <= result.x + width)
		{
			stbrp_node* next2 = ptr->next;
			ptr->next = context->free_head;
			context->free_head = ptr;
			ptr = next2;
		}
		free_head->next = ptr;
		if (ptr->x < result.x + width)
		{
			ptr->x = result.x + width;
		}
		return result;
	}

	public unsafe static int rect_height_compare(void* a, void* b)
	{
		if (((stbrp_rect*)a)->h > ((stbrp_rect*)b)->h)
		{
			return -1;
		}
		if (((stbrp_rect*)a)->h < ((stbrp_rect*)b)->h)
		{
			return 1;
		}
		if (((stbrp_rect*)a)->w <= ((stbrp_rect*)b)->w)
		{
			if (((stbrp_rect*)a)->w >= ((stbrp_rect*)b)->w)
			{
				return 0;
			}
			return 1;
		}
		return -1;
	}

	public unsafe static int rect_original_order(void* a, void* b)
	{
		if (((stbrp_rect*)a)->was_packed >= ((stbrp_rect*)b)->was_packed)
		{
			if (((stbrp_rect*)a)->was_packed <= ((stbrp_rect*)b)->was_packed)
			{
				return 0;
			}
			return 1;
		}
		return -1;
	}

	public unsafe static int stbrp_pack_rects(stbrp_context* context, stbrp_rect* rects, int num_rects)
	{
		int num = 0;
		int result = 1;
		for (num = 0; num < num_rects; num++)
		{
			rects[num].was_packed = num;
		}
		CRuntime.qsort(rects, (ulong)num_rects, (ulong)sizeof(stbrp_rect), rect_height_compare);
		for (num = 0; num < num_rects; num++)
		{
			if (rects[num].w == 0 || rects[num].h == 0)
			{
				rects[num].x = (rects[num].y = 0);
				continue;
			}
			stbrp__findresult stbrp__findresult = stbrp__skyline_pack_rectangle(context, rects[num].w, rects[num].h);
			if (stbrp__findresult.prev_link != null)
			{
				rects[num].x = stbrp__findresult.x;
				rects[num].y = stbrp__findresult.y;
			}
			else
			{
				rects[num].x = (rects[num].y = 65535);
			}
		}
		CRuntime.qsort(rects, (ulong)num_rects, (ulong)sizeof(stbrp_rect), rect_original_order);
		for (num = 0; num < num_rects; num++)
		{
			rects[num].was_packed = ((rects[num].x != 65535 || rects[num].y != 65535) ? 1 : 0);
			if (rects[num].was_packed == 0)
			{
				result = 0;
			}
		}
		return result;
	}
}