com.wallstop-studios.unity-helpers 1.0.0-rc4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (238) hide show
  1. package/.gitattributes +63 -0
  2. package/CHANGELOG.md +0 -0
  3. package/CHANGELOG.md.meta +7 -0
  4. package/Editor/AnimationCreator.cs +218 -0
  5. package/Editor/AnimationCreator.cs.meta +11 -0
  6. package/Editor/AnimationEventEditor.cs +742 -0
  7. package/Editor/AnimationEventEditor.cs.meta +11 -0
  8. package/Editor/PrefabCheckWizard.cs +140 -0
  9. package/Editor/PrefabCheckWizard.cs.meta +11 -0
  10. package/Editor/WallstopStudios.UnityHelpers.Editor.asmdef +16 -0
  11. package/Editor/WallstopStudios.UnityHelpers.Editor.asmdef.meta +7 -0
  12. package/Editor.meta +8 -0
  13. package/LICENSE +21 -0
  14. package/LICENSE.md +7 -0
  15. package/LICENSE.md.meta +7 -0
  16. package/LICENSE.meta +7 -0
  17. package/README.md +2 -0
  18. package/README.md.meta +7 -0
  19. package/Runtime/Core/Attributes/AnimationEventAttribute.cs +102 -0
  20. package/Runtime/Core/Attributes/AnimationEventAttribute.cs.meta +11 -0
  21. package/Runtime/Core/Attributes/AutomaticallyFindAttribute.cs +43 -0
  22. package/Runtime/Core/Attributes/AutomaticallyFindAttribute.cs.meta +11 -0
  23. package/Runtime/Core/Attributes/ChildComponentAttribute.cs +85 -0
  24. package/Runtime/Core/Attributes/ChildComponentAttribute.cs.meta +11 -0
  25. package/Runtime/Core/Attributes/KSerializableAttribute.cs +23 -0
  26. package/Runtime/Core/Attributes/KSerializableAttribute.cs.meta +11 -0
  27. package/Runtime/Core/Attributes/NotNullAttribute.cs +33 -0
  28. package/Runtime/Core/Attributes/NotNullAttribute.cs.meta +11 -0
  29. package/Runtime/Core/Attributes/ParentComponent.cs +85 -0
  30. package/Runtime/Core/Attributes/ParentComponent.cs.meta +11 -0
  31. package/Runtime/Core/Attributes/ReadOnlyAttribute.cs +8 -0
  32. package/Runtime/Core/Attributes/ReadOnlyAttribute.cs.meta +11 -0
  33. package/Runtime/Core/Attributes/RelationalComponentExtensions.cs +14 -0
  34. package/Runtime/Core/Attributes/RelationalComponentExtensions.cs.meta +11 -0
  35. package/Runtime/Core/Attributes/SiblingComponentAttribute.cs +89 -0
  36. package/Runtime/Core/Attributes/SiblingComponentAttribute.cs.meta +11 -0
  37. package/Runtime/Core/Attributes/ValidateAssignmentAttribute.cs +66 -0
  38. package/Runtime/Core/Attributes/ValidateAssignmentAttribute.cs.meta +11 -0
  39. package/Runtime/Core/Attributes.meta +8 -0
  40. package/Runtime/Core/DataStructure/Adapters/FastVector2Int.cs +91 -0
  41. package/Runtime/Core/DataStructure/Adapters/FastVector2Int.cs.meta +11 -0
  42. package/Runtime/Core/DataStructure/Adapters/FastVector3Int.cs +180 -0
  43. package/Runtime/Core/DataStructure/Adapters/FastVector3Int.cs.meta +11 -0
  44. package/Runtime/Core/DataStructure/Adapters/KGuid.cs +274 -0
  45. package/Runtime/Core/DataStructure/Adapters/KGuid.cs.meta +11 -0
  46. package/Runtime/Core/DataStructure/Adapters/KVector2.cs +79 -0
  47. package/Runtime/Core/DataStructure/Adapters/KVector2.cs.meta +11 -0
  48. package/Runtime/Core/DataStructure/Adapters.meta +8 -0
  49. package/Runtime/Core/DataStructure/Circle.cs +50 -0
  50. package/Runtime/Core/DataStructure/Circle.cs.meta +11 -0
  51. package/Runtime/Core/DataStructure/CyclicBuffer.cs +130 -0
  52. package/Runtime/Core/DataStructure/CyclicBuffer.cs.meta +11 -0
  53. package/Runtime/Core/DataStructure/ISpatialTree.cs +58 -0
  54. package/Runtime/Core/DataStructure/ISpatialTree.cs.meta +11 -0
  55. package/Runtime/Core/DataStructure/KDTree.cs +186 -0
  56. package/Runtime/Core/DataStructure/KDTree.cs.meta +11 -0
  57. package/Runtime/Core/DataStructure/QuadTree.cs +184 -0
  58. package/Runtime/Core/DataStructure/QuadTree.cs.meta +11 -0
  59. package/Runtime/Core/DataStructure/RTree.cs +247 -0
  60. package/Runtime/Core/DataStructure/RTree.cs.meta +11 -0
  61. package/Runtime/Core/DataStructure/StringWrapper.cs +90 -0
  62. package/Runtime/Core/DataStructure/StringWrapper.cs.meta +11 -0
  63. package/Runtime/Core/DataStructure/TimedCache.cs +50 -0
  64. package/Runtime/Core/DataStructure/TimedCache.cs.meta +11 -0
  65. package/Runtime/Core/DataStructure.meta +8 -0
  66. package/Runtime/Core/Extension/AnimatorExtensions.cs +25 -0
  67. package/Runtime/Core/Extension/AnimatorExtensions.cs.meta +11 -0
  68. package/Runtime/Core/Extension/CircleExtensions.cs +25 -0
  69. package/Runtime/Core/Extension/CircleExtensions.cs.meta +11 -0
  70. package/Runtime/Core/Extension/ColorExtensions.cs +72 -0
  71. package/Runtime/Core/Extension/ColorExtensions.cs.meta +11 -0
  72. package/Runtime/Core/Extension/DictionaryExtensions.cs +173 -0
  73. package/Runtime/Core/Extension/DictionaryExtensions.cs.meta +11 -0
  74. package/Runtime/Core/Extension/DirectionExtensions.cs +210 -0
  75. package/Runtime/Core/Extension/DirectionExtensions.cs.meta +11 -0
  76. package/Runtime/Core/Extension/HashSetExtensions.cs +12 -0
  77. package/Runtime/Core/Extension/HashSetExtensions.cs.meta +11 -0
  78. package/Runtime/Core/Extension/IEnumerableExtensions.cs +109 -0
  79. package/Runtime/Core/Extension/IEnumerableExtensions.cs.meta +11 -0
  80. package/Runtime/Core/Extension/IListExtensions.cs +49 -0
  81. package/Runtime/Core/Extension/IListExtensions.cs.meta +11 -0
  82. package/Runtime/Core/Extension/LoggingExtensions.cs +196 -0
  83. package/Runtime/Core/Extension/LoggingExtensions.cs.meta +11 -0
  84. package/Runtime/Core/Extension/RandomExtensions.cs +110 -0
  85. package/Runtime/Core/Extension/RandomExtensions.cs.meta +11 -0
  86. package/Runtime/Core/Extension/StringExtensions.cs +76 -0
  87. package/Runtime/Core/Extension/StringExtensions.cs.meta +11 -0
  88. package/Runtime/Core/Extension/UnityExtensions.cs +1409 -0
  89. package/Runtime/Core/Extension/UnityExtensions.cs.meta +11 -0
  90. package/Runtime/Core/Extension.meta +8 -0
  91. package/Runtime/Core/Helper/AssignUtilities.cs +14 -0
  92. package/Runtime/Core/Helper/AssignUtilities.cs.meta +11 -0
  93. package/Runtime/Core/Helper/Enumerables.cs +17 -0
  94. package/Runtime/Core/Helper/Enumerables.cs.meta +11 -0
  95. package/Runtime/Core/Helper/Geometry.cs +26 -0
  96. package/Runtime/Core/Helper/Geometry.cs.meta +11 -0
  97. package/Runtime/Core/Helper/Helpers.cs +1092 -0
  98. package/Runtime/Core/Helper/Helpers.cs.meta +11 -0
  99. package/Runtime/Core/Helper/IterationHelpers.cs +32 -0
  100. package/Runtime/Core/Helper/IterationHelpers.cs.meta +11 -0
  101. package/Runtime/Core/Helper/LifetimeHelpers.cs +12 -0
  102. package/Runtime/Core/Helper/LifetimeHelpers.cs.meta +11 -0
  103. package/Runtime/Core/Helper/Objects.cs +447 -0
  104. package/Runtime/Core/Helper/Objects.cs.meta +11 -0
  105. package/Runtime/Core/Helper/SpriteHelpers.cs +53 -0
  106. package/Runtime/Core/Helper/SpriteHelpers.cs.meta +11 -0
  107. package/Runtime/Core/Helper/StringInList.cs +31 -0
  108. package/Runtime/Core/Helper/StringInList.cs.meta +11 -0
  109. package/Runtime/Core/Helper/WallMath.cs +75 -0
  110. package/Runtime/Core/Helper/WallMath.cs.meta +11 -0
  111. package/Runtime/Core/Helper.meta +8 -0
  112. package/Runtime/Core/Math/Line.cs +51 -0
  113. package/Runtime/Core/Math/Line.cs.meta +11 -0
  114. package/Runtime/Core/Math/Parabola.cs +44 -0
  115. package/Runtime/Core/Math/Parabola.cs.meta +11 -0
  116. package/Runtime/Core/Math/PointPolygonCheck.cs +25 -0
  117. package/Runtime/Core/Math/PointPolygonCheck.cs.meta +11 -0
  118. package/Runtime/Core/Math/Range.cs +56 -0
  119. package/Runtime/Core/Math/Range.cs.meta +11 -0
  120. package/Runtime/Core/Math/XXHash.cs +308 -0
  121. package/Runtime/Core/Math/XXHash.cs.meta +11 -0
  122. package/Runtime/Core/Math.meta +8 -0
  123. package/Runtime/Core/Model/Direction.cs +26 -0
  124. package/Runtime/Core/Model/Direction.cs.meta +11 -0
  125. package/Runtime/Core/Model.meta +8 -0
  126. package/Runtime/Core/OneOf/FastOneOf.cs +145 -0
  127. package/Runtime/Core/OneOf/FastOneOf.cs.meta +11 -0
  128. package/Runtime/Core/OneOf/None.cs +6 -0
  129. package/Runtime/Core/OneOf/None.cs.meta +11 -0
  130. package/Runtime/Core/OneOf.meta +8 -0
  131. package/Runtime/Core/Random/AbstractRandom.cs +537 -0
  132. package/Runtime/Core/Random/AbstractRandom.cs.meta +11 -0
  133. package/Runtime/Core/Random/IRandom.cs +141 -0
  134. package/Runtime/Core/Random/IRandom.cs.meta +11 -0
  135. package/Runtime/Core/Random/NativePcgRandom.cs +97 -0
  136. package/Runtime/Core/Random/NativePcgRandom.cs.meta +11 -0
  137. package/Runtime/Core/Random/PcgRandom.cs +142 -0
  138. package/Runtime/Core/Random/PcgRandom.cs.meta +11 -0
  139. package/Runtime/Core/Random/RandomState.cs +92 -0
  140. package/Runtime/Core/Random/RandomState.cs.meta +11 -0
  141. package/Runtime/Core/Random/RandomUtilities.cs +26 -0
  142. package/Runtime/Core/Random/RandomUtilities.cs.meta +11 -0
  143. package/Runtime/Core/Random/SquirrelRandom.cs +82 -0
  144. package/Runtime/Core/Random/SquirrelRandom.cs.meta +11 -0
  145. package/Runtime/Core/Random/SystemRandom.cs +110 -0
  146. package/Runtime/Core/Random/SystemRandom.cs.meta +11 -0
  147. package/Runtime/Core/Random/ThreadLocalRandom.cs +11 -0
  148. package/Runtime/Core/Random/ThreadLocalRandom.cs.meta +11 -0
  149. package/Runtime/Core/Random/UnityRandom.cs +24 -0
  150. package/Runtime/Core/Random/UnityRandom.cs.meta +11 -0
  151. package/Runtime/Core/Random/XorShiftRandom.cs +45 -0
  152. package/Runtime/Core/Random/XorShiftRandom.cs.meta +11 -0
  153. package/Runtime/Core/Random.meta +8 -0
  154. package/Runtime/Core/Serialization/JsonConverters/Vector2Converter.cs +32 -0
  155. package/Runtime/Core/Serialization/JsonConverters/Vector2Converter.cs.meta +11 -0
  156. package/Runtime/Core/Serialization/JsonConverters/Vector3Converter.cs +32 -0
  157. package/Runtime/Core/Serialization/JsonConverters/Vector3Converter.cs.meta +11 -0
  158. package/Runtime/Core/Serialization/JsonConverters.meta +8 -0
  159. package/Runtime/Core/Serialization/Serializer.cs +125 -0
  160. package/Runtime/Core/Serialization/Serializer.cs.meta +11 -0
  161. package/Runtime/Core/Serialization.meta +8 -0
  162. package/Runtime/Core/Threading/SingleThreadedThreadPool.cs +102 -0
  163. package/Runtime/Core/Threading/SingleThreadedThreadPool.cs.meta +11 -0
  164. package/Runtime/Core/Threading.meta +8 -0
  165. package/Runtime/Core.meta +8 -0
  166. package/Runtime/Protobuf-Net/System.Buffers.dll +0 -0
  167. package/Runtime/Protobuf-Net/System.Buffers.dll.meta +33 -0
  168. package/Runtime/Protobuf-Net/System.Collections.Immutable.dll +0 -0
  169. package/Runtime/Protobuf-Net/System.Collections.Immutable.dll.meta +33 -0
  170. package/Runtime/Protobuf-Net/System.Numerics.Vectors.dll +0 -0
  171. package/Runtime/Protobuf-Net/System.Numerics.Vectors.dll.meta +33 -0
  172. package/Runtime/Protobuf-Net/System.Runtime.CompilerServices.Unsafe.dll +0 -0
  173. package/Runtime/Protobuf-Net/System.Runtime.CompilerServices.Unsafe.dll.meta +33 -0
  174. package/Runtime/Protobuf-Net/protobuf-net.Core.dll +0 -0
  175. package/Runtime/Protobuf-Net/protobuf-net.Core.dll.meta +33 -0
  176. package/Runtime/Protobuf-Net/protobuf-net.dll +0 -0
  177. package/Runtime/Protobuf-Net/protobuf-net.dll.meta +33 -0
  178. package/Runtime/Protobuf-Net.meta +8 -0
  179. package/Runtime/Utils/AnimationEventEqualityComparer.cs +149 -0
  180. package/Runtime/Utils/AnimationEventEqualityComparer.cs.meta +11 -0
  181. package/Runtime/Utils/AnimatorEnumStateMachine.cs +80 -0
  182. package/Runtime/Utils/AnimatorEnumStateMachine.cs.meta +11 -0
  183. package/Runtime/Utils/Buffers.cs +32 -0
  184. package/Runtime/Utils/Buffers.cs.meta +11 -0
  185. package/Runtime/Utils/CircleLineRenderer.cs +122 -0
  186. package/Runtime/Utils/CircleLineRenderer.cs.meta +11 -0
  187. package/Runtime/Utils/Oscillator.cs +25 -0
  188. package/Runtime/Utils/Oscillator.cs.meta +11 -0
  189. package/Runtime/Utils/SetTextureImportData.cs +67 -0
  190. package/Runtime/Utils/SetTextureImportData.cs.meta +11 -0
  191. package/Runtime/Utils.meta +8 -0
  192. package/Runtime/WallstopStudios.UnityHelpers.asmdef +14 -0
  193. package/Runtime/WallstopStudios.UnityHelpers.asmdef.meta +7 -0
  194. package/Runtime.meta +8 -0
  195. package/Tests/Editor/WallstopStudios.UnityHelpers.Tests.Editor.asmdef +17 -0
  196. package/Tests/Editor/WallstopStudios.UnityHelpers.Tests.Editor.asmdef.meta +7 -0
  197. package/Tests/Editor.meta +8 -0
  198. package/Tests/Runtime/DataStructures/BalancedKDTreeTests.cs +14 -0
  199. package/Tests/Runtime/DataStructures/BalancedKDTreeTests.cs.meta +11 -0
  200. package/Tests/Runtime/DataStructures/QuadTreeTests.cs +14 -0
  201. package/Tests/Runtime/DataStructures/QuadTreeTests.cs.meta +11 -0
  202. package/Tests/Runtime/DataStructures/SpatialTreeTests.cs +106 -0
  203. package/Tests/Runtime/DataStructures/SpatialTreeTests.cs.meta +11 -0
  204. package/Tests/Runtime/DataStructures/UnbalancedKDTreeTests.cs +14 -0
  205. package/Tests/Runtime/DataStructures/UnbalancedKDTreeTests.cs.meta +11 -0
  206. package/Tests/Runtime/DataStructures.meta +8 -0
  207. package/Tests/Runtime/Performance/KDTreePerformanceTests.cs +14 -0
  208. package/Tests/Runtime/Performance/KDTreePerformanceTests.cs.meta +11 -0
  209. package/Tests/Runtime/Performance/QuadTreePerformanceTests.cs +14 -0
  210. package/Tests/Runtime/Performance/QuadTreePerformanceTests.cs.meta +11 -0
  211. package/Tests/Runtime/Performance/RandomPerformanceTests.cs +76 -0
  212. package/Tests/Runtime/Performance/RandomPerformanceTests.cs.meta +11 -0
  213. package/Tests/Runtime/Performance/SpatialTreePerformanceTest.cs +141 -0
  214. package/Tests/Runtime/Performance/SpatialTreePerformanceTest.cs.meta +11 -0
  215. package/Tests/Runtime/Performance/UnbalancedKDTreeTests.cs +14 -0
  216. package/Tests/Runtime/Performance/UnbalancedKDTreeTests.cs.meta +11 -0
  217. package/Tests/Runtime/Performance.meta +8 -0
  218. package/Tests/Runtime/Random/PcgRandomTests.cs +9 -0
  219. package/Tests/Runtime/Random/PcgRandomTests.cs.meta +11 -0
  220. package/Tests/Runtime/Random/RandomTestBase.cs +117 -0
  221. package/Tests/Runtime/Random/RandomTestBase.cs.meta +11 -0
  222. package/Tests/Runtime/Random/SquirrelRandomTests.cs +9 -0
  223. package/Tests/Runtime/Random/SquirrelRandomTests.cs.meta +11 -0
  224. package/Tests/Runtime/Random/SystemRandomTests.cs +10 -0
  225. package/Tests/Runtime/Random/SystemRandomTests.cs.meta +11 -0
  226. package/Tests/Runtime/Random/UnityRandomTests.cs +9 -0
  227. package/Tests/Runtime/Random/UnityRandomTests.cs.meta +11 -0
  228. package/Tests/Runtime/Random/XorShiftRandomTests.cs +9 -0
  229. package/Tests/Runtime/Random/XorShiftRandomTests.cs.meta +11 -0
  230. package/Tests/Runtime/Random.meta +8 -0
  231. package/Tests/Runtime/WallstopStudios.UnityHelpers.Tests.Runtime.asmdef +22 -0
  232. package/Tests/Runtime/WallstopStudios.UnityHelpers.Tests.Runtime.asmdef.meta +7 -0
  233. package/Tests/Runtime.meta +8 -0
  234. package/Tests.meta +8 -0
  235. package/Third Party Notices.md +1 -0
  236. package/Third Party Notices.md.meta +7 -0
  237. package/package.json +35 -0
  238. package/package.json.meta +7 -0
@@ -0,0 +1,186 @@
1
+ namespace UnityHelpers.Core.DataStructure
2
+ {
3
+ using System;
4
+ using System.Collections.Generic;
5
+ using System.Collections.Immutable;
6
+ using System.Linq;
7
+ using Extension;
8
+ using UnityEngine;
9
+
10
+ [Serializable]
11
+ public sealed class KDTree<T> : ISpatialTree<T>
12
+ {
13
+ public delegate float Axis<in V>(V element);
14
+
15
+ [Serializable]
16
+ private sealed class KDTreeNode<V>
17
+ {
18
+ public readonly Bounds boundary;
19
+ public readonly KDTreeNode<V> left;
20
+ public readonly KDTreeNode<V> right;
21
+ public readonly V[] elements;
22
+ public readonly bool isTerminal;
23
+
24
+ public KDTreeNode(List<V> elements, Func<V, Vector2> elementTransformer, int bucketSize, bool isXAxis, bool balanced)
25
+ {
26
+ boundary = elements.Select(elementTransformer).GetBounds() ?? new Bounds();
27
+ this.elements = elements.ToArray();
28
+ isTerminal = elements.Count <= bucketSize;
29
+ if (isTerminal)
30
+ {
31
+ return;
32
+ }
33
+
34
+ if (balanced)
35
+ {
36
+ Axis<V> axisFunction = isXAxis
37
+ ? element => elementTransformer(element).x
38
+ : element => elementTransformer(element).y;
39
+
40
+ int Comparison(V lhs, V rhs)
41
+ {
42
+ return axisFunction(lhs).CompareTo(axisFunction(rhs));
43
+ }
44
+ elements.Sort(Comparison);
45
+
46
+ int cutoff = elements.Count / 2;
47
+ left = new KDTreeNode<V>(elements.Take(cutoff).ToList(), elementTransformer, bucketSize, !isXAxis, true);
48
+ right = new KDTreeNode<V>(elements.Skip(cutoff).ToList(), elementTransformer, bucketSize, !isXAxis, true);
49
+ }
50
+ else
51
+ {
52
+ Vector2 cutoff = boundary.center;
53
+ if (isXAxis)
54
+ {
55
+ left = new KDTreeNode<V>(elements.Where(element => elementTransformer(element).x <= cutoff.x).ToList(), elementTransformer, bucketSize, false, false);
56
+ right = new KDTreeNode<V>(elements.Where(element => cutoff.x < elementTransformer(element).x).ToList(), elementTransformer, bucketSize, false, false);
57
+ }
58
+ else
59
+ {
60
+ left = new KDTreeNode<V>(elements.Where(element => elementTransformer(element).y <= cutoff.y).ToList(), elementTransformer, bucketSize, true, false);
61
+ right = new KDTreeNode<V>(elements.Where(element => cutoff.y < elementTransformer(element).y).ToList(), elementTransformer, bucketSize, true, false);
62
+ }
63
+ }
64
+ }
65
+ }
66
+
67
+ public const int DefaultBucketSize = 12;
68
+
69
+ public readonly ImmutableArray<T> elements;
70
+ public Bounds Boundary => _bounds;
71
+ public Func<T, Vector2> ElementTransformer => _elementTransformer;
72
+
73
+ private readonly Bounds _bounds;
74
+ private readonly Func<T, Vector2> _elementTransformer;
75
+ private readonly KDTreeNode<T> _head;
76
+
77
+ public KDTree(IEnumerable<T> points, Func<T, Vector2> elementTransformer, int bucketSize = DefaultBucketSize, bool balanced = true)
78
+ {
79
+ _elementTransformer = elementTransformer ?? throw new ArgumentNullException(nameof(elementTransformer));
80
+ elements = points?.ToImmutableArray() ?? throw new ArgumentNullException(nameof(points));
81
+ _bounds = elements.Select(elementTransformer).GetBounds() ?? new Bounds();
82
+ _head = new KDTreeNode<T>(elements.ToList(), elementTransformer, bucketSize: bucketSize, isXAxis:true, balanced: balanced);
83
+ }
84
+
85
+ public IEnumerable<T> GetElementsInBounds(Bounds bounds)
86
+ {
87
+ if (!bounds.FastIntersects2D(_bounds))
88
+ {
89
+ yield break;
90
+ }
91
+
92
+ Stack<KDTreeNode<T>> nodesToVisit = new();
93
+ nodesToVisit.Push(_head);
94
+
95
+ while (nodesToVisit.TryPop(out KDTreeNode<T> currentNode))
96
+ {
97
+ if (currentNode.isTerminal)
98
+ {
99
+ foreach (T element in currentNode.elements)
100
+ {
101
+ if (bounds.FastContains2D(_elementTransformer(element)))
102
+ {
103
+ yield return element;
104
+ }
105
+ }
106
+
107
+ continue;
108
+ }
109
+
110
+ if (bounds.Overlaps2D(currentNode.boundary))
111
+ {
112
+ foreach (T element in currentNode.elements)
113
+ {
114
+ yield return element;
115
+ }
116
+
117
+ continue;
118
+ }
119
+
120
+ KDTreeNode<T> leftNode = currentNode.left;
121
+ if (0 < leftNode.elements.Length && bounds.FastIntersects2D(leftNode.boundary))
122
+ {
123
+ nodesToVisit.Push(leftNode);
124
+ }
125
+
126
+ KDTreeNode<T> rightNode = currentNode.right;
127
+ if (0 < rightNode.elements.Length && bounds.FastIntersects2D(rightNode.boundary))
128
+ {
129
+ nodesToVisit.Push(rightNode);
130
+ }
131
+ }
132
+ }
133
+
134
+ // Heavily adapted http://homepage.divms.uiowa.edu/%7Ekvaradar/sp2012/daa/ann.pdf
135
+ public void GetApproximateNearestNeighbors(Vector2 position, int count, List<T> nearestNeighbors)
136
+ {
137
+ nearestNeighbors.Clear();
138
+
139
+ KDTreeNode<T> current = _head;
140
+ Stack<KDTreeNode<T>> stack = new();
141
+ stack.Push(_head);
142
+ HashSet<T> nearestNeighborsSet = new(count);
143
+
144
+ while (!current.isTerminal)
145
+ {
146
+ KDTreeNode<T> left = current.left;
147
+ KDTreeNode<T> right = current.right;
148
+ if (((Vector2)left.boundary.center - position).sqrMagnitude <
149
+ ((Vector2)right.boundary.center - position).sqrMagnitude)
150
+ {
151
+ stack.Push(left);
152
+ current = left;
153
+ if (left.elements.Length <= count)
154
+ {
155
+ break;
156
+ }
157
+ }
158
+ else
159
+ {
160
+ stack.Push(right);
161
+ current = right;
162
+ if (right.elements.Length <= count)
163
+ {
164
+ break;
165
+ }
166
+ }
167
+ }
168
+
169
+ while (nearestNeighborsSet.Count < count && stack.TryPop(out KDTreeNode<T> selected))
170
+ {
171
+ foreach (T element in selected.elements)
172
+ {
173
+ _ = nearestNeighborsSet.Add(element);
174
+ }
175
+ }
176
+
177
+ nearestNeighbors.AddRange(nearestNeighborsSet);
178
+ if (count < nearestNeighbors.Count)
179
+ {
180
+ int NearestComparison(T lhs, T rhs) => (_elementTransformer(lhs) - position).sqrMagnitude.CompareTo((_elementTransformer(rhs) - position).sqrMagnitude);
181
+ nearestNeighbors.Sort(NearestComparison);
182
+ nearestNeighbors.RemoveRange(count, nearestNeighbors.Count - count);
183
+ }
184
+ }
185
+ }
186
+ }
@@ -0,0 +1,11 @@
1
+ fileFormatVersion: 2
2
+ guid: 2ebf309d2de45e944946aa5cccd9754a
3
+ MonoImporter:
4
+ externalObjects: {}
5
+ serializedVersion: 2
6
+ defaultReferences: []
7
+ executionOrder: 0
8
+ icon: {instanceID: 0}
9
+ userData:
10
+ assetBundleName:
11
+ assetBundleVariant:
@@ -0,0 +1,184 @@
1
+ namespace UnityHelpers.Core.DataStructure
2
+ {
3
+ using System;
4
+ using System.Collections.Generic;
5
+ using System.Collections.Immutable;
6
+ using System.Linq;
7
+ using Extension;
8
+ using UnityEngine;
9
+
10
+ [Serializable]
11
+ public sealed class QuadTree<T> : ISpatialTree<T>
12
+ {
13
+ private const int NumChildren = 4;
14
+
15
+ [Serializable]
16
+ private sealed class QuadTreeNode<V>
17
+ {
18
+ private static readonly List<V> Buffer = new();
19
+
20
+ public readonly Bounds boundary;
21
+ public readonly QuadTreeNode<V>[] children;
22
+ public readonly V[] elements;
23
+ public readonly bool isTerminal;
24
+
25
+ public QuadTreeNode(V[] elements, Func<V, Vector2> elementTransformer, Bounds boundary,
26
+ int bucketSize)
27
+ {
28
+ this.boundary = boundary;
29
+ this.elements = elements;
30
+ isTerminal = elements.Length <= bucketSize;
31
+ if (isTerminal)
32
+ {
33
+ children = Array.Empty<QuadTreeNode<V>>();
34
+ return;
35
+ }
36
+ children = new QuadTreeNode<V>[NumChildren];
37
+
38
+ Vector3 quadrantSize = boundary.size / 2f;
39
+ Vector2 halfQuadrantSize = quadrantSize / 2f;
40
+
41
+ Bounds[] quadrants =
42
+ {
43
+ new Bounds(new Vector3(boundary.center.x - halfQuadrantSize.x, boundary.center.y + halfQuadrantSize.y, boundary.center.z), quadrantSize),
44
+ new Bounds(new Vector3(boundary.center.x + halfQuadrantSize.x, boundary.center.y + halfQuadrantSize.y, boundary.center.z), quadrantSize),
45
+ new Bounds(new Vector3(boundary.center.x + halfQuadrantSize.x, boundary.center.y - halfQuadrantSize.y, boundary.center.z), quadrantSize),
46
+ new Bounds(new Vector3(boundary.center.x - halfQuadrantSize.x, boundary.center.y - halfQuadrantSize.y, boundary.center.z), quadrantSize),
47
+ };
48
+
49
+ for (int i = 0; i < quadrants.Length; ++i)
50
+ {
51
+ Bounds quadrant = quadrants[i];
52
+ Buffer.Clear();
53
+ foreach (V element in elements)
54
+ {
55
+ if (quadrant.FastContains2D(elementTransformer(element)))
56
+ {
57
+ Buffer.Add(element);
58
+ }
59
+ }
60
+
61
+ children[i] = new QuadTreeNode<V>(Buffer.ToArray(), elementTransformer, quadrant, bucketSize);
62
+ }
63
+ }
64
+ }
65
+
66
+ public const int DefaultBucketSize = 12;
67
+
68
+ public readonly ImmutableArray<T> elements;
69
+ public Bounds Boundary => _bounds;
70
+ public Func<T, Vector2> ElementTransformer => _elementTransformer;
71
+
72
+ private readonly Bounds _bounds;
73
+ private readonly Func<T, Vector2> _elementTransformer;
74
+ private readonly QuadTreeNode<T> _head;
75
+
76
+ public QuadTree(IEnumerable<T> points, Func<T, Vector2> elementTransformer, Bounds? boundary = null,
77
+ int bucketSize = DefaultBucketSize)
78
+ {
79
+ _elementTransformer = elementTransformer ?? throw new ArgumentNullException(nameof(elementTransformer));
80
+ elements = points?.ToImmutableArray() ?? throw new ArgumentNullException(nameof(points));
81
+ _bounds = boundary ?? elements.Select(elementTransformer).GetBounds() ?? new Bounds();
82
+ _head = new QuadTreeNode<T>(elements.ToArray(), elementTransformer, _bounds, bucketSize);
83
+ }
84
+
85
+ public IEnumerable<T> GetElementsInBounds(Bounds bounds)
86
+ {
87
+ if (!bounds.FastIntersects2D(_bounds))
88
+ {
89
+ yield break;
90
+ }
91
+
92
+ Stack<QuadTreeNode<T>> nodesToVisit = new();
93
+ nodesToVisit.Push(_head);
94
+
95
+ while (nodesToVisit.TryPop(out QuadTreeNode<T> currentNode))
96
+ {
97
+ if (currentNode.isTerminal)
98
+ {
99
+ foreach (T element in currentNode.elements)
100
+ {
101
+ if (bounds.FastContains2D(_elementTransformer(element)))
102
+ {
103
+ yield return element;
104
+ }
105
+ }
106
+
107
+ continue;
108
+ }
109
+
110
+ if (bounds.Overlaps2D(currentNode.boundary))
111
+ {
112
+ foreach (T element in currentNode.elements)
113
+ {
114
+ yield return element;
115
+ }
116
+
117
+ continue;
118
+ }
119
+
120
+ foreach (QuadTreeNode<T> child in currentNode.children)
121
+ {
122
+ if (child.elements.Length <= 0)
123
+ {
124
+ continue;
125
+ }
126
+
127
+ if (!bounds.FastIntersects2D(child.boundary))
128
+ {
129
+ continue;
130
+ }
131
+
132
+ nodesToVisit.Push(child);
133
+ }
134
+ }
135
+ }
136
+
137
+ // Heavily adapted http://homepage.divms.uiowa.edu/%7Ekvaradar/sp2012/daa/ann.pdf
138
+ public void GetApproximateNearestNeighbors(Vector2 position, int count, List<T> nearestNeighbors)
139
+ {
140
+ nearestNeighbors.Clear();
141
+
142
+ QuadTreeNode<T> current = _head;
143
+ Stack<QuadTreeNode<T>> stack = new();
144
+ stack.Push(_head);
145
+ List<QuadTreeNode<T>> childrenCopy = new(NumChildren);
146
+ HashSet<T> nearestNeighborsSet = new(count);
147
+
148
+ int Comparison(QuadTreeNode<T> lhs, QuadTreeNode<T> rhs) => ((Vector2)lhs.boundary.center - position).sqrMagnitude.CompareTo(((Vector2)rhs.boundary.center - position).sqrMagnitude);
149
+
150
+ while (!current.isTerminal)
151
+ {
152
+ childrenCopy.Clear();
153
+ childrenCopy.AddRange(current.children);
154
+ childrenCopy.Sort(Comparison);
155
+ for (int i = childrenCopy.Count - 1; 0 <= i; --i)
156
+ {
157
+ stack.Push(childrenCopy[i]);
158
+ }
159
+
160
+ current = childrenCopy[0];
161
+ if (current.elements.Length <= count)
162
+ {
163
+ break;
164
+ }
165
+ }
166
+
167
+ while (nearestNeighborsSet.Count < count && stack.TryPop(out QuadTreeNode<T> selected))
168
+ {
169
+ foreach (T element in selected.elements)
170
+ {
171
+ _ = nearestNeighborsSet.Add(element);
172
+ }
173
+ }
174
+
175
+ nearestNeighbors.AddRange(nearestNeighborsSet);
176
+ if (count < nearestNeighbors.Count)
177
+ {
178
+ int NearestComparison(T lhs, T rhs) => (_elementTransformer(lhs) - position).sqrMagnitude.CompareTo((_elementTransformer(rhs) - position).sqrMagnitude);
179
+ nearestNeighbors.Sort(NearestComparison);
180
+ nearestNeighbors.RemoveRange(count, nearestNeighbors.Count - count);
181
+ }
182
+ }
183
+ }
184
+ }
@@ -0,0 +1,11 @@
1
+ fileFormatVersion: 2
2
+ guid: bd888839de97c284e9a7153ab84c74df
3
+ MonoImporter:
4
+ externalObjects: {}
5
+ serializedVersion: 2
6
+ defaultReferences: []
7
+ executionOrder: 0
8
+ icon: {instanceID: 0}
9
+ userData:
10
+ assetBundleName:
11
+ assetBundleVariant:
@@ -0,0 +1,247 @@
1
+ namespace UnityHelpers.Core.DataStructure
2
+ {
3
+ using System;
4
+ using System.Collections.Generic;
5
+ using System.Collections.Immutable;
6
+ using System.Linq;
7
+ using Extension;
8
+ using UnityEngine;
9
+
10
+ [Serializable]
11
+ public sealed class RTree<T>
12
+ {
13
+ [Serializable]
14
+ private sealed class RTreeNode<V>
15
+ {
16
+ public readonly Bounds boundary;
17
+ public readonly RTreeNode<V>[] children;
18
+ public readonly V[] elements;
19
+ public readonly bool isTerminal;
20
+
21
+ public RTreeNode(List<V> elements, Func<V, Bounds> elementTransformer, int bucketSize, int branchFactor)
22
+ {
23
+ float minX = float.MaxValue;
24
+ float minY = float.MaxValue;
25
+ float maxX = float.MinValue;
26
+ float maxY = float.MinValue;
27
+ foreach (V element in elements)
28
+ {
29
+ Bounds rectangle = elementTransformer(element);
30
+ Vector3 min = rectangle.min;
31
+ Vector3 max = rectangle.max;
32
+ minX = Math.Min(minX, min.x);
33
+ maxX = Math.Max(maxX, max.x);
34
+ minY = Math.Min(minY, min.y);
35
+ maxY = Math.Max(maxY, max.y);
36
+ }
37
+
38
+ boundary = elements.Count <= 0 ? new Bounds() : new Bounds(new Vector3(minX + (maxX - minX) / 2, minY + (maxY - minY) / 2), new Vector3(maxX - minX, maxY - minY));
39
+ this.elements = elements.ToArray();
40
+ isTerminal = elements.Count <= bucketSize;
41
+ if (isTerminal)
42
+ {
43
+ children = Array.Empty<RTreeNode<V>>();
44
+ return;
45
+ }
46
+
47
+ /*
48
+ http://www.dtic.mil/get-tr-doc/pdf?AD=ADA324493
49
+ var targetSize = rectangles.Count / (double) branchFactor;
50
+ P = branchFactor;
51
+ S = Math.sqrt(P);
52
+ N = targetSize;
53
+
54
+ Ugh.
55
+ */
56
+ double targetSize = elements.Count / (double)branchFactor;
57
+ int intTargetSize = (int)Math.Ceiling(targetSize);
58
+
59
+ List<RTreeNode<V>> tempChildren = new(intTargetSize);
60
+
61
+ double slicesPerAxis = Math.Sqrt(branchFactor);
62
+ int rectanglesPerPagePerAxis = (int)(slicesPerAxis * targetSize);
63
+
64
+ int XAxis(V lhs, V rhs)
65
+ {
66
+ return elementTransformer(lhs).center.x.CompareTo(elementTransformer(rhs).center.x);
67
+ }
68
+
69
+ int YAxis(V lhs, V rhs)
70
+ {
71
+ return elementTransformer(lhs).center.y.CompareTo(elementTransformer(rhs).center.y);
72
+ }
73
+
74
+ elements.Sort(XAxis);
75
+ foreach (List<V> xSlice in elements.Partition(rectanglesPerPagePerAxis).Select(enumerable => enumerable as List<V> ?? enumerable.ToList()))
76
+ {
77
+ xSlice.Sort(YAxis);
78
+ foreach (List<V> ySlice in xSlice.Partition(intTargetSize).Select(enumerable => enumerable as List<V> ?? enumerable.ToList()))
79
+ {
80
+ RTreeNode<V> node = new(ySlice, elementTransformer, bucketSize, branchFactor);
81
+ tempChildren.Add(node);
82
+ }
83
+ }
84
+
85
+ children = tempChildren.ToArray();
86
+ }
87
+ }
88
+
89
+ public const int DefaultBucketSize = 10;
90
+ public const int DefaultBranchFactor = 4;
91
+
92
+ public readonly ImmutableArray<T> elements;
93
+ public Bounds Boundary => _bounds;
94
+
95
+ private readonly Bounds _bounds;
96
+ private readonly Func<T, Bounds> _elementTransformer;
97
+ private readonly RTreeNode<T> _head;
98
+
99
+ public RTree(
100
+ IEnumerable<T> points, Func<T, Bounds> elementTransformer, int bucketSize = DefaultBucketSize,
101
+ int branchFactor = DefaultBranchFactor)
102
+ {
103
+ _elementTransformer = elementTransformer ?? throw new ArgumentNullException(nameof(elementTransformer));
104
+ elements = points?.ToImmutableArray() ?? throw new ArgumentNullException(nameof(points));
105
+ _bounds = elements.Select(elementTransformer).GetBounds() ?? new Bounds();
106
+ _head = new RTreeNode<T>(elements.ToList(), elementTransformer, bucketSize, branchFactor);
107
+ }
108
+
109
+ public IEnumerable<T> GetElementsInRange(Vector2 position, float range, float minimumRange = 0f)
110
+ {
111
+ Circle area = new(position, range);
112
+ if (0 < minimumRange)
113
+ {
114
+ Circle minimumArea = new(position, minimumRange);
115
+ return GetElementsInBounds(
116
+ new Bounds(
117
+ new Vector3(position.x, position.y, 0f),
118
+ new Vector3(range * 2f, range * 2f, 1f)))
119
+ .Where(
120
+ element =>
121
+ {
122
+ Bounds elementBoundary = _elementTransformer(element);
123
+ if (!area.Intersects(elementBoundary))
124
+ {
125
+ return false;
126
+ }
127
+
128
+ return !minimumArea.Intersects(elementBoundary);
129
+ });
130
+ }
131
+ return GetElementsInBounds(
132
+ new Bounds(
133
+ new Vector3(position.x, position.y, 0f),
134
+ new Vector3(range * 2f, range * 2f, 1f)))
135
+ .Where(
136
+ element =>
137
+ {
138
+ Bounds elementBoundary = _elementTransformer(element);
139
+ if (!area.Intersects(elementBoundary))
140
+ {
141
+ return false;
142
+ }
143
+
144
+ return true;
145
+ });
146
+ }
147
+
148
+ public IEnumerable<T> GetElementsInBounds(Bounds bounds)
149
+ {
150
+ if (!bounds.FastIntersects2D(_bounds))
151
+ {
152
+ yield break;
153
+ }
154
+
155
+ Stack<RTreeNode<T>> nodesToVisit = new();
156
+ nodesToVisit.Push(_head);
157
+
158
+ while (nodesToVisit.TryPop(out RTreeNode<T> currentNode))
159
+ {
160
+ if (currentNode.isTerminal)
161
+ {
162
+ foreach (T element in currentNode.elements)
163
+ {
164
+ if (bounds.FastIntersects2D(_elementTransformer(element)))
165
+ {
166
+ yield return element;
167
+ }
168
+ }
169
+
170
+ continue;
171
+ }
172
+
173
+ if (bounds.Overlaps2D(currentNode.boundary))
174
+ {
175
+ foreach (T element in currentNode.elements)
176
+ {
177
+ yield return element;
178
+ }
179
+
180
+ continue;
181
+ }
182
+
183
+ foreach (RTreeNode<T> child in currentNode.children)
184
+ {
185
+ if (child.elements.Length <= 0)
186
+ {
187
+ continue;
188
+ }
189
+
190
+ if (!bounds.FastIntersects2D(child.boundary))
191
+ {
192
+ continue;
193
+ }
194
+
195
+ nodesToVisit.Push(child);
196
+ }
197
+ }
198
+ }
199
+
200
+ // Heavily adapted http://homepage.divms.uiowa.edu/%7Ekvaradar/sp2012/daa/ann.pdf
201
+ public void GetApproximateNearestNeighbors(Vector2 position, int count, List<T> nearestNeighbors)
202
+ {
203
+ nearestNeighbors.Clear();
204
+
205
+ RTreeNode<T> current = _head;
206
+ Stack<RTreeNode<T>> stack = new();
207
+ stack.Push(_head);
208
+ List<RTreeNode<T>> childrenCopy = new();
209
+ HashSet<T> nearestNeighborsSet = new(count);
210
+
211
+ int Comparison(RTreeNode<T> lhs, RTreeNode<T> rhs) => ((Vector2)lhs.boundary.center - position).sqrMagnitude.CompareTo(((Vector2)rhs.boundary.center - position).sqrMagnitude);
212
+
213
+ while (!current.isTerminal)
214
+ {
215
+ childrenCopy.Clear();
216
+ childrenCopy.AddRange(current.children);
217
+ childrenCopy.Sort(Comparison);
218
+ for (int i = childrenCopy.Count - 1; 0 <= i; --i)
219
+ {
220
+ stack.Push(childrenCopy[i]);
221
+ }
222
+
223
+ current = childrenCopy[0];
224
+ if (current.elements.Length <= count)
225
+ {
226
+ break;
227
+ }
228
+ }
229
+
230
+ while (nearestNeighborsSet.Count < count && stack.TryPop(out RTreeNode<T> selected))
231
+ {
232
+ foreach (T element in selected.elements)
233
+ {
234
+ _ = nearestNeighborsSet.Add(element);
235
+ }
236
+ }
237
+
238
+ nearestNeighbors.AddRange(nearestNeighborsSet);
239
+ if (count < nearestNeighbors.Count)
240
+ {
241
+ int NearestComparison(T lhs, T rhs) => ((Vector2)_elementTransformer(lhs).center - position).sqrMagnitude.CompareTo(((Vector2)_elementTransformer(rhs).center - position).sqrMagnitude);
242
+ nearestNeighbors.Sort(NearestComparison);
243
+ nearestNeighbors.RemoveRange(count, nearestNeighbors.Count - count);
244
+ }
245
+ }
246
+ }
247
+ }
@@ -0,0 +1,11 @@
1
+ fileFormatVersion: 2
2
+ guid: 6c6e3230b807b894cbccf3dc0281058b
3
+ MonoImporter:
4
+ externalObjects: {}
5
+ serializedVersion: 2
6
+ defaultReferences: []
7
+ executionOrder: 0
8
+ icon: {instanceID: 0}
9
+ userData:
10
+ assetBundleName:
11
+ assetBundleVariant: