com.wallstop-studios.unity-helpers 2.0.0-rc27 → 2.0.0-rc31

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 (267) hide show
  1. package/.gitattributes +63 -63
  2. package/.github/workflows/npm-publish.yml +60 -0
  3. package/.github/workflows/unity-package.yml +101 -0
  4. package/.github/workflows/unity-tests.yml +38 -0
  5. package/Editor/AnimationCopier.cs +158 -158
  6. package/Editor/AnimationCopier.cs.meta +2 -2
  7. package/Editor/AnimationCreator.cs +262 -262
  8. package/Editor/AnimationCreator.cs.meta +11 -11
  9. package/Editor/AnimationEventEditor.cs +887 -887
  10. package/Editor/AnimatorControllerCopier.cs +162 -162
  11. package/Editor/AnimatorControllerCopier.cs.meta +2 -2
  12. package/Editor/CustomEditors/MatchColliderToSpriteEditor.cs +34 -34
  13. package/Editor/CustomEditors/MatchColliderToSpriteEditor.cs.meta +2 -2
  14. package/Editor/CustomEditors.meta +2 -2
  15. package/Editor/EnsureTextureSizeWizard.cs +110 -110
  16. package/Editor/EnsureTextureSizeWizard.cs.meta +2 -2
  17. package/Editor/PrefabCheckWizard.cs +165 -165
  18. package/Editor/PrefabCheckWizard.cs.meta +11 -11
  19. package/Editor/SpriteSettingsApplier.cs +168 -168
  20. package/Editor/SpriteSettingsApplier.cs.meta +2 -2
  21. package/Editor/StringInListeDrawer.cs +56 -56
  22. package/Editor/TextureResizerWizard.cs +181 -181
  23. package/Editor/TextureResizerWizard.cs.meta +2 -2
  24. package/Editor/TextureSettingsApplier.cs +171 -171
  25. package/Editor/TextureSettingsApplier.cs.meta +2 -2
  26. package/Editor/Utils/EditorUtilities.cs +22 -22
  27. package/Editor/Utils/EditorUtilities.cs.meta +11 -11
  28. package/Editor/Utils/ReadOnlyPropertyDrawer.cs +26 -26
  29. package/Editor/Utils/ReadOnlyPropertyDrawer.cs.meta +11 -11
  30. package/Editor/Utils.meta +8 -8
  31. package/Editor/WallstopStudios.UnityHelpers.Editor.asmdef +17 -17
  32. package/Editor/WallstopStudios.UnityHelpers.Editor.asmdef.meta +7 -7
  33. package/LICENSE +21 -21
  34. package/LICENSE.md +6 -6
  35. package/LICENSE.meta +7 -7
  36. package/README.md +117 -117
  37. package/Runtime/Binaries/Microsoft.Bcl.AsyncInterfaces.dll.meta +33 -33
  38. package/Runtime/Binaries/Microsoft.Bcl.AsyncInterfaces.xml +223 -223
  39. package/Runtime/Binaries/Microsoft.Bcl.AsyncInterfaces.xml.meta +7 -7
  40. package/Runtime/Binaries/System.Text.Encodings.Web.dll.meta +33 -33
  41. package/Runtime/Binaries/System.Text.Encodings.Web.xml +935 -935
  42. package/Runtime/Binaries/System.Text.Encodings.Web.xml.meta +7 -7
  43. package/Runtime/Binaries/System.Text.Json.dll.meta +33 -33
  44. package/Runtime/Binaries/System.Text.Json.xml +4829 -4829
  45. package/Runtime/Binaries/System.Text.Json.xml.meta +7 -7
  46. package/Runtime/Binaries.meta +8 -8
  47. package/Runtime/Core/Attributes/AnimationEventAttribute.cs +131 -131
  48. package/Runtime/Core/Attributes/ChildComponentAttribute.cs +189 -189
  49. package/Runtime/Core/Attributes/KSerializableAttribute.cs +19 -19
  50. package/Runtime/Core/Attributes/NotNullAttribute.cs +32 -32
  51. package/Runtime/Core/Attributes/ParentComponent.cs +184 -184
  52. package/Runtime/Core/Attributes/ReadOnlyAttribute.cs +6 -6
  53. package/Runtime/Core/Attributes/RelationalComponentExtensions.cs +14 -14
  54. package/Runtime/Core/Attributes/SiblingComponentAttribute.cs +119 -119
  55. package/Runtime/Core/Attributes/SiblingComponentAttribute.cs.meta +11 -11
  56. package/Runtime/Core/Attributes/ValidateAssignmentAttribute.cs +101 -101
  57. package/Runtime/Core/Attributes/ValidateAssignmentAttribute.cs.meta +11 -11
  58. package/Runtime/Core/Attributes.meta +8 -8
  59. package/Runtime/Core/DataStructure/Adapters/FastVector2Int.cs +92 -92
  60. package/Runtime/Core/DataStructure/Adapters/FastVector3Int.cs +185 -185
  61. package/Runtime/Core/DataStructure/Adapters/KGuid.cs +305 -305
  62. package/Runtime/Core/DataStructure/Adapters/KVector2.cs +80 -80
  63. package/Runtime/Core/DataStructure/Circle.cs +50 -50
  64. package/Runtime/Core/DataStructure/CyclicBuffer.cs +155 -155
  65. package/Runtime/Core/DataStructure/ISpatialTree.cs +60 -60
  66. package/Runtime/Core/DataStructure/ISpatialTree.cs.meta +11 -11
  67. package/Runtime/Core/DataStructure/KDTree.cs +290 -290
  68. package/Runtime/Core/DataStructure/KDTree.cs.meta +11 -11
  69. package/Runtime/Core/DataStructure/QuadTree.cs +279 -279
  70. package/Runtime/Core/DataStructure/RTree.cs +336 -336
  71. package/Runtime/Core/DataStructure/RTree.cs.meta +11 -11
  72. package/Runtime/Core/DataStructure/StringWrapper.cs +91 -91
  73. package/Runtime/Core/DataStructure/TimedCache.cs +51 -51
  74. package/Runtime/Core/Extension/AnimatorExtensions.cs +25 -25
  75. package/Runtime/Core/Extension/CircleExtensions.cs +25 -25
  76. package/Runtime/Core/Extension/ColorExtensions.cs +338 -338
  77. package/Runtime/Core/Extension/DictionaryExtensions.cs +251 -251
  78. package/Runtime/Core/Extension/DirectionExtensions.cs +213 -213
  79. package/Runtime/Core/Extension/HashSetExtensions.cs +12 -12
  80. package/Runtime/Core/Extension/IEnumerableExtensions.cs +122 -122
  81. package/Runtime/Core/Extension/IListExtensions.cs +89 -89
  82. package/Runtime/Core/Extension/LoggingExtensions.cs +258 -258
  83. package/Runtime/Core/Extension/RandomExtensions.cs +109 -109
  84. package/Runtime/Core/Extension/StringExtensions.cs +151 -151
  85. package/Runtime/Core/Extension/UnityExtensions.cs +1607 -1607
  86. package/Runtime/Core/Helper/ArrayConverter.cs +39 -39
  87. package/Runtime/Core/Helper/ArrayConverter.cs.meta +2 -2
  88. package/Runtime/Core/Helper/AssignUtilities.cs +14 -14
  89. package/Runtime/Core/Helper/AssignUtilities.cs.meta +11 -11
  90. package/Runtime/Core/Helper/Enumerables.cs +17 -17
  91. package/Runtime/Core/Helper/Geometry.cs +43 -43
  92. package/Runtime/Core/Helper/Helpers.cs +722 -722
  93. package/Runtime/Core/Helper/Helpers.cs.meta +11 -11
  94. package/Runtime/Core/Helper/IterationHelpers.cs +32 -32
  95. package/Runtime/Core/Helper/IterationHelpers.cs.meta +11 -11
  96. package/Runtime/Core/Helper/LifetimeHelpers.cs +13 -13
  97. package/Runtime/Core/Helper/Objects.cs +767 -767
  98. package/Runtime/Core/Helper/Partials/LogHelpers.cs +13 -13
  99. package/Runtime/Core/Helper/Partials/LogHelpers.cs.meta +2 -2
  100. package/Runtime/Core/Helper/Partials/MathHelpers.cs +30 -30
  101. package/Runtime/Core/Helper/Partials/MathHelpers.cs.meta +2 -2
  102. package/Runtime/Core/Helper/Partials/ObjectHelpers.cs +388 -388
  103. package/Runtime/Core/Helper/Partials/ObjectHelpers.cs.meta +2 -2
  104. package/Runtime/Core/Helper/Partials/TransformHelpers.cs +167 -167
  105. package/Runtime/Core/Helper/Partials/TransformHelpers.cs.meta +2 -2
  106. package/Runtime/Core/Helper/Partials.meta +2 -2
  107. package/Runtime/Core/Helper/ReflectionHelpers.cs +152 -152
  108. package/Runtime/Core/Helper/ReflectionHelpers.cs.meta +2 -2
  109. package/Runtime/Core/Helper/SpriteHelpers.cs +86 -86
  110. package/Runtime/Core/Helper/SpriteHelpers.cs.meta +11 -11
  111. package/Runtime/Core/Helper/StringInList.cs +31 -31
  112. package/Runtime/Core/Helper/StringInList.cs.meta +11 -11
  113. package/Runtime/Core/Helper/WallMath.cs +166 -166
  114. package/Runtime/Core/Math/Line.cs +55 -55
  115. package/Runtime/Core/Math/Parabola.cs +47 -47
  116. package/Runtime/Core/Math/PointPolygonCheck.cs +36 -36
  117. package/Runtime/Core/Math/PointPolygonCheck.cs.meta +11 -11
  118. package/Runtime/Core/Math/Range.cs +92 -92
  119. package/Runtime/Core/Math/XXHash.cs +310 -310
  120. package/Runtime/Core/Math/XXHash.cs.meta +11 -11
  121. package/Runtime/Core/Model/Direction.cs +43 -43
  122. package/Runtime/Core/OneOf/FastOneOf.cs +152 -152
  123. package/Runtime/Core/OneOf/None.cs +4 -4
  124. package/Runtime/Core/Random/AbstractRandom.cs +561 -561
  125. package/Runtime/Core/Random/DotNetRandom.cs +52 -52
  126. package/Runtime/Core/Random/DotNetRandom.cs.meta +2 -2
  127. package/Runtime/Core/Random/IRandom.cs +160 -160
  128. package/Runtime/Core/Random/NativePcgRandom.cs +97 -97
  129. package/Runtime/Core/Random/PRNG.cs +7 -7
  130. package/Runtime/Core/Random/PRNG.cs.meta +2 -2
  131. package/Runtime/Core/Random/PcgRandom.cs +149 -149
  132. package/Runtime/Core/Random/PerlinNoise.cs +369 -369
  133. package/Runtime/Core/Random/PerlinNoise.cs.meta +2 -2
  134. package/Runtime/Core/Random/RandomState.cs +131 -131
  135. package/Runtime/Core/Random/RandomUtilities.cs +26 -26
  136. package/Runtime/Core/Random/RandomUtilities.cs.meta +11 -11
  137. package/Runtime/Core/Random/RomuDuo.cs +116 -116
  138. package/Runtime/Core/Random/RomuDuo.cs.meta +2 -2
  139. package/Runtime/Core/Random/SplitMix64.cs +94 -94
  140. package/Runtime/Core/Random/SplitMix64.cs.meta +2 -2
  141. package/Runtime/Core/Random/SquirrelRandom.cs +84 -84
  142. package/Runtime/Core/Random/SystemRandom.cs +162 -162
  143. package/Runtime/Core/Random/ThreadLocalRandom.cs +12 -12
  144. package/Runtime/Core/Random/UnityRandom.cs +57 -57
  145. package/Runtime/Core/Random/UnityRandom.cs.meta +11 -11
  146. package/Runtime/Core/Random/WyRandom.cs +121 -121
  147. package/Runtime/Core/Random/WyRandom.cs.meta +2 -2
  148. package/Runtime/Core/Random/XorShiftRandom.cs +47 -47
  149. package/Runtime/Core/Random/XorShiftRandom.cs.meta +11 -11
  150. package/Runtime/Core/Random/XorShiroRandom.cs +117 -117
  151. package/Runtime/Core/Random/XorShiroRandom.cs.meta +2 -2
  152. package/Runtime/Core/Serialization/JsonConverters/Vector2Converter.cs +74 -74
  153. package/Runtime/Core/Serialization/JsonConverters/Vector3Converter.cs +81 -81
  154. package/Runtime/Core/Serialization/Serializer.cs +184 -184
  155. package/Runtime/Core/Threading/SingleThreadedThreadPool.cs +107 -107
  156. package/Runtime/Protobuf-Net/System.Buffers.dll.meta +33 -33
  157. package/Runtime/Protobuf-Net/System.Collections.Immutable.dll.meta +33 -33
  158. package/Runtime/Protobuf-Net/System.Collections.Immutable.xml +5379 -5379
  159. package/Runtime/Protobuf-Net/System.Collections.Immutable.xml.meta +7 -7
  160. package/Runtime/Protobuf-Net/System.Numerics.Vectors.dll.meta +33 -33
  161. package/Runtime/Protobuf-Net/System.Runtime.CompilerServices.Unsafe.dll.meta +33 -33
  162. package/Runtime/Protobuf-Net/System.Runtime.CompilerServices.Unsafe.xml +290 -290
  163. package/Runtime/Protobuf-Net/System.Runtime.CompilerServices.Unsafe.xml.meta +7 -7
  164. package/Runtime/Protobuf-Net/protobuf-net.Core.dll.meta +33 -33
  165. package/Runtime/Protobuf-Net/protobuf-net.dll.meta +33 -33
  166. package/Runtime/UI/LayeredImage.cs +364 -364
  167. package/Runtime/UI/LayeredImage.cs.meta +2 -2
  168. package/Runtime/UI.meta +2 -2
  169. package/Runtime/Utils/AnimationEventEqualityComparer.cs +161 -161
  170. package/Runtime/Utils/AnimatorEnumStateMachine.cs +88 -88
  171. package/Runtime/Utils/Buffers.cs +32 -32
  172. package/Runtime/Utils/CenterPointOffset.cs +30 -30
  173. package/Runtime/Utils/CenterPointOffset.cs.meta +2 -2
  174. package/Runtime/Utils/CircleLineRenderer.cs +134 -134
  175. package/Runtime/Utils/CoroutineHandler.cs +4 -4
  176. package/Runtime/Utils/CoroutineHandler.cs.meta +2 -2
  177. package/Runtime/Utils/MatchColliderToSprite.cs +94 -94
  178. package/Runtime/Utils/MatchColliderToSprite.cs.meta +2 -2
  179. package/Runtime/Utils/Oscillator.cs +27 -27
  180. package/Runtime/Utils/RuntimeSingleton.cs +57 -57
  181. package/Runtime/Utils/RuntimeSingleton.cs.meta +11 -11
  182. package/Runtime/Utils/SetTextureImportData.cs +69 -69
  183. package/Runtime/Utils/SpriteRendererMetadata.cs +312 -312
  184. package/Runtime/Utils/SpriteRendererMetadata.cs.meta +2 -2
  185. package/Runtime/Utils/SpriteRendererSyncer.cs +100 -100
  186. package/Runtime/Utils/SpriteRendererSyncer.cs.meta +2 -2
  187. package/Runtime/Utils/TextureScale.cs +179 -179
  188. package/Runtime/Utils/TextureScale.cs.meta +2 -2
  189. package/Runtime/WallstopStudios.UnityHelpers.asmdef +13 -13
  190. package/Tests/Runtime/Attributes/ChildComponentTests.cs +81 -81
  191. package/Tests/Runtime/Attributes/Components/ExpectChildSpriteRenderers.cs +28 -28
  192. package/Tests/Runtime/Attributes/Components/ExpectParentSpriteRenderers.cs +28 -28
  193. package/Tests/Runtime/Attributes/ParentComponentTests.cs +68 -68
  194. package/Tests/Runtime/Components/RelationalComponentTester.cs +34 -34
  195. package/Tests/Runtime/Components/RelationalComponentTester.cs.meta +2 -2
  196. package/Tests/Runtime/Components.meta +2 -2
  197. package/Tests/Runtime/DataStructures/BalancedKDTreeTests.cs +14 -14
  198. package/Tests/Runtime/DataStructures/BalancedKDTreeTests.cs.meta +11 -11
  199. package/Tests/Runtime/DataStructures/CyclicBufferTests.cs +324 -324
  200. package/Tests/Runtime/DataStructures/QuadTreeTests.cs +14 -14
  201. package/Tests/Runtime/DataStructures/QuadTreeTests.cs.meta +11 -11
  202. package/Tests/Runtime/DataStructures/SpatialTreeTests.cs +130 -130
  203. package/Tests/Runtime/DataStructures/SpatialTreeTests.cs.meta +11 -11
  204. package/Tests/Runtime/DataStructures/UnbalancedKDTreeTests.cs +14 -14
  205. package/Tests/Runtime/DataStructures/UnbalancedKDTreeTests.cs.meta +11 -11
  206. package/Tests/Runtime/DataStructures.meta +8 -8
  207. package/Tests/Runtime/Extensions/DictionaryExtensionTests.cs +439 -439
  208. package/Tests/Runtime/Extensions/DictionaryExtensionTests.cs.meta +2 -2
  209. package/Tests/Runtime/Extensions/IListExtensionTests.cs +76 -76
  210. package/Tests/Runtime/Extensions/RandomExtensionTests.cs +27 -27
  211. package/Tests/Runtime/Extensions/RandomExtensionTests.cs.meta +2 -2
  212. package/Tests/Runtime/Extensions/StringExtensionTests.cs +31 -31
  213. package/Tests/Runtime/Extensions/StringExtensionTests.cs.meta +2 -2
  214. package/Tests/Runtime/Extensions.meta +2 -2
  215. package/Tests/Runtime/Helper/ArrayConverterTests.cs +19 -19
  216. package/Tests/Runtime/Helper/ArrayConverterTests.cs.meta +2 -2
  217. package/Tests/Runtime/Helper/ObjectHelperTests.cs +402 -402
  218. package/Tests/Runtime/Helper/ObjectHelperTests.cs.meta +2 -2
  219. package/Tests/Runtime/Helper/WallMathTests.cs +233 -233
  220. package/Tests/Runtime/Helper/WallMathTests.cs.meta +2 -2
  221. package/Tests/Runtime/Helper.meta +2 -2
  222. package/Tests/Runtime/Performance/KDTreePerformanceTests.cs +14 -14
  223. package/Tests/Runtime/Performance/KDTreePerformanceTests.cs.meta +11 -11
  224. package/Tests/Runtime/Performance/QuadTreePerformanceTests.cs +14 -14
  225. package/Tests/Runtime/Performance/QuadTreePerformanceTests.cs.meta +11 -11
  226. package/Tests/Runtime/Performance/RandomPerformanceTests.cs +139 -139
  227. package/Tests/Runtime/Performance/RandomPerformanceTests.cs.meta +11 -11
  228. package/Tests/Runtime/Performance/RelationComponentPerformanceTests.cs +37 -37
  229. package/Tests/Runtime/Performance/RelationComponentPerformanceTests.cs.meta +2 -2
  230. package/Tests/Runtime/Performance/SpatialTreePerformanceTest.cs +154 -154
  231. package/Tests/Runtime/Performance/SpatialTreePerformanceTest.cs.meta +11 -11
  232. package/Tests/Runtime/Performance/UnbalancedKDTreeTests.cs +14 -14
  233. package/Tests/Runtime/Performance/UnbalancedKDTreeTests.cs.meta +11 -11
  234. package/Tests/Runtime/Performance.meta +8 -8
  235. package/Tests/Runtime/Random/DotNetRandomTests.cs +9 -9
  236. package/Tests/Runtime/Random/DotNetRandomTests.cs.meta +2 -2
  237. package/Tests/Runtime/Random/PcgRandomTests.cs +9 -9
  238. package/Tests/Runtime/Random/PcgRandomTests.cs.meta +11 -11
  239. package/Tests/Runtime/Random/RandomTestBase.cs +787 -787
  240. package/Tests/Runtime/Random/RandomTestBase.cs.meta +11 -11
  241. package/Tests/Runtime/Random/RomuDuoRandomTests.cs +9 -9
  242. package/Tests/Runtime/Random/RomuDuoRandomTests.cs.meta +2 -2
  243. package/Tests/Runtime/Random/SplitMix64RandomTests.cs +9 -9
  244. package/Tests/Runtime/Random/SplitMix64RandomTests.cs.meta +2 -2
  245. package/Tests/Runtime/Random/SquirrelRandomTests.cs +14 -14
  246. package/Tests/Runtime/Random/SquirrelRandomTests.cs.meta +11 -11
  247. package/Tests/Runtime/Random/SystemRandomTests.cs +10 -10
  248. package/Tests/Runtime/Random/SystemRandomTests.cs.meta +11 -11
  249. package/Tests/Runtime/Random/UnityRandomTests.cs +9 -9
  250. package/Tests/Runtime/Random/UnityRandomTests.cs.meta +11 -11
  251. package/Tests/Runtime/Random/WyRandomTests.cs +9 -9
  252. package/Tests/Runtime/Random/WyRandomTests.cs.meta +2 -2
  253. package/Tests/Runtime/Random/XorShiftRandomTests.cs +9 -9
  254. package/Tests/Runtime/Random/XorShiftRandomTests.cs.meta +11 -11
  255. package/Tests/Runtime/Random/XorShiroRandomTests.cs +9 -9
  256. package/Tests/Runtime/Random/XorShiroRandomTests.cs.meta +2 -2
  257. package/Tests/Runtime/Random.meta +8 -8
  258. package/Tests/Runtime/Serialization/JsonSerializationTest.cs +84 -84
  259. package/Tests/Runtime/Serialization/JsonSerializationTest.cs.meta +2 -2
  260. package/Tests/Runtime/Serialization.meta +2 -2
  261. package/Tests/Runtime/Utils/SpriteRendererMetadataTests.cs +399 -399
  262. package/Tests/Runtime/Utils/SpriteRendererMetadataTests.cs.meta +2 -2
  263. package/Tests/Runtime/Utils.meta +2 -2
  264. package/Tests/Runtime/WallstopStudios.UnityHelpers.Tests.Runtime.asmdef +22 -22
  265. package/Tests/Runtime/WallstopStudios.UnityHelpers.Tests.Runtime.asmdef.meta +7 -7
  266. package/Tests/Runtime.meta +8 -8
  267. package/package.json +38 -38
@@ -1,336 +1,336 @@
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
- using Utils;
10
-
11
- [Serializable]
12
- public sealed class RTree<T>
13
- {
14
- [Serializable]
15
- public sealed class RTreeNode<V>
16
- {
17
- public readonly Bounds boundary;
18
- internal readonly RTreeNode<V>[] children;
19
- public readonly V[] elements;
20
- public readonly bool isTerminal;
21
-
22
- public RTreeNode(
23
- List<V> elements,
24
- Func<V, Bounds> elementTransformer,
25
- int bucketSize,
26
- int branchFactor
27
- )
28
- {
29
- float minX = float.MaxValue;
30
- float minY = float.MaxValue;
31
- float maxX = float.MinValue;
32
- float maxY = float.MinValue;
33
- foreach (V element in elements)
34
- {
35
- Bounds rectangle = elementTransformer(element);
36
- Vector3 min = rectangle.min;
37
- Vector3 max = rectangle.max;
38
- minX = Math.Min(minX, min.x);
39
- maxX = Math.Max(maxX, max.x);
40
- minY = Math.Min(minY, min.y);
41
- maxY = Math.Max(maxY, max.y);
42
- }
43
-
44
- boundary =
45
- elements.Count <= 0
46
- ? new Bounds()
47
- : new Bounds(
48
- new Vector3(minX + (maxX - minX) / 2, minY + (maxY - minY) / 2),
49
- new Vector3(maxX - minX, maxY - minY)
50
- );
51
- this.elements = elements.ToArray();
52
- isTerminal = elements.Count <= bucketSize;
53
- if (isTerminal)
54
- {
55
- children = Array.Empty<RTreeNode<V>>();
56
- return;
57
- }
58
-
59
- /*
60
- http://www.dtic.mil/get-tr-doc/pdf?AD=ADA324493
61
- var targetSize = rectangles.Count / (double) branchFactor;
62
- P = branchFactor;
63
- S = Math.sqrt(P);
64
- N = targetSize;
65
-
66
- Ugh.
67
- */
68
- double targetSize = elements.Count / (double)branchFactor;
69
- int intTargetSize = (int)Math.Ceiling(targetSize);
70
-
71
- List<RTreeNode<V>> tempChildren = new(intTargetSize);
72
-
73
- double slicesPerAxis = Math.Sqrt(branchFactor);
74
- int rectanglesPerPagePerAxis = (int)(slicesPerAxis * targetSize);
75
-
76
- elements.Sort(XAxis);
77
- foreach (
78
- List<V> xSlice in elements
79
- .Partition(rectanglesPerPagePerAxis)
80
- .Select(enumerable => enumerable as List<V> ?? enumerable.ToList())
81
- )
82
- {
83
- xSlice.Sort(YAxis);
84
- foreach (
85
- List<V> ySlice in xSlice
86
- .Partition(intTargetSize)
87
- .Select(enumerable => enumerable as List<V> ?? enumerable.ToList())
88
- )
89
- {
90
- RTreeNode<V> node = new(
91
- ySlice,
92
- elementTransformer,
93
- bucketSize,
94
- branchFactor
95
- );
96
- tempChildren.Add(node);
97
- }
98
- }
99
-
100
- children = tempChildren.ToArray();
101
- return;
102
-
103
- int XAxis(V lhs, V rhs)
104
- {
105
- return elementTransformer(lhs)
106
- .center.x.CompareTo(elementTransformer(rhs).center.x);
107
- }
108
-
109
- int YAxis(V lhs, V rhs)
110
- {
111
- return elementTransformer(lhs)
112
- .center.y.CompareTo(elementTransformer(rhs).center.y);
113
- }
114
- }
115
- }
116
-
117
- public const int DefaultBucketSize = 10;
118
- public const int DefaultBranchFactor = 4;
119
-
120
- public readonly ImmutableArray<T> elements;
121
- public Bounds Boundary => _bounds;
122
-
123
- private readonly Bounds _bounds;
124
- private readonly Func<T, Bounds> _elementTransformer;
125
- private readonly RTreeNode<T> _head;
126
-
127
- public RTree(
128
- IEnumerable<T> points,
129
- Func<T, Bounds> elementTransformer,
130
- int bucketSize = DefaultBucketSize,
131
- int branchFactor = DefaultBranchFactor
132
- )
133
- {
134
- _elementTransformer =
135
- elementTransformer ?? throw new ArgumentNullException(nameof(elementTransformer));
136
- elements =
137
- points?.ToImmutableArray() ?? throw new ArgumentNullException(nameof(points));
138
- _bounds = elements.Select(elementTransformer).GetBounds() ?? new Bounds();
139
- _head = new RTreeNode<T>(
140
- elements.ToList(),
141
- elementTransformer,
142
- bucketSize,
143
- branchFactor
144
- );
145
- }
146
-
147
- public IEnumerable<T> GetElementsInRange(
148
- Vector2 position,
149
- float range,
150
- float minimumRange = 0f
151
- )
152
- {
153
- Circle area = new(position, range);
154
- if (0 < minimumRange)
155
- {
156
- Circle minimumArea = new(position, minimumRange);
157
- return GetElementsInBounds(
158
- new Bounds(
159
- new Vector3(position.x, position.y, 0f),
160
- new Vector3(range * 2f, range * 2f, 1f)
161
- )
162
- )
163
- .Where(element =>
164
- {
165
- Bounds elementBoundary = _elementTransformer(element);
166
- if (!area.Intersects(elementBoundary))
167
- {
168
- return false;
169
- }
170
-
171
- return !minimumArea.Intersects(elementBoundary);
172
- });
173
- }
174
- return GetElementsInBounds(
175
- new Bounds(
176
- new Vector3(position.x, position.y, 0f),
177
- new Vector3(range * 2f, range * 2f, 1f)
178
- )
179
- )
180
- .Where(element =>
181
- {
182
- Bounds elementBoundary = _elementTransformer(element);
183
- if (!area.Intersects(elementBoundary))
184
- {
185
- return false;
186
- }
187
-
188
- return true;
189
- });
190
- }
191
-
192
- public IEnumerable<T> GetElementsInBounds(Bounds bounds)
193
- {
194
- Stack<RTreeNode<T>> nodeBuffer = Buffers<RTreeNode<T>>.Stack;
195
- return GetElementsInBounds(bounds, nodeBuffer);
196
- }
197
-
198
- public IEnumerable<T> GetElementsInBounds(Bounds bounds, Stack<RTreeNode<T>> nodeBuffer)
199
- {
200
- if (!bounds.FastIntersects2D(_bounds))
201
- {
202
- yield break;
203
- }
204
-
205
- Stack<RTreeNode<T>> nodesToVisit = nodeBuffer ?? new Stack<RTreeNode<T>>();
206
- nodeBuffer.Clear();
207
- nodesToVisit.Push(_head);
208
-
209
- while (nodesToVisit.TryPop(out RTreeNode<T> currentNode))
210
- {
211
- if (currentNode.isTerminal)
212
- {
213
- foreach (T element in currentNode.elements)
214
- {
215
- if (bounds.FastIntersects2D(_elementTransformer(element)))
216
- {
217
- yield return element;
218
- }
219
- }
220
-
221
- continue;
222
- }
223
-
224
- if (bounds.Overlaps2D(currentNode.boundary))
225
- {
226
- foreach (T element in currentNode.elements)
227
- {
228
- yield return element;
229
- }
230
-
231
- continue;
232
- }
233
-
234
- foreach (RTreeNode<T> child in currentNode.children)
235
- {
236
- if (child.elements.Length <= 0)
237
- {
238
- continue;
239
- }
240
-
241
- if (!bounds.FastIntersects2D(child.boundary))
242
- {
243
- continue;
244
- }
245
-
246
- nodesToVisit.Push(child);
247
- }
248
- }
249
- }
250
-
251
- public void GetApproximateNearestNeighbors(
252
- Vector2 position,
253
- int count,
254
- List<T> nearestNeighbors
255
- )
256
- {
257
- Stack<RTreeNode<T>> nodeBuffer = Buffers<RTreeNode<T>>.Stack;
258
- List<RTreeNode<T>> childrenBuffer = Buffers<RTreeNode<T>>.List;
259
- HashSet<T> nearestNeighborBuffer = Buffers<T>.HashSet;
260
- GetApproximateNearestNeighbors(
261
- position,
262
- count,
263
- nearestNeighbors,
264
- nodeBuffer,
265
- childrenBuffer,
266
- nearestNeighborBuffer
267
- );
268
- }
269
-
270
- // Heavily adapted http://homepage.divms.uiowa.edu/%7Ekvaradar/sp2012/daa/ann.pdf
271
- public void GetApproximateNearestNeighbors(
272
- Vector2 position,
273
- int count,
274
- List<T> nearestNeighbors,
275
- Stack<RTreeNode<T>> nodeBuffer,
276
- List<RTreeNode<T>> childrenBuffer,
277
- HashSet<T> nearestNeighborsBuffer
278
- )
279
- {
280
- nearestNeighbors.Clear();
281
-
282
- RTreeNode<T> current = _head;
283
- Stack<RTreeNode<T>> stack = nodeBuffer ?? new Stack<RTreeNode<T>>();
284
- stack.Clear();
285
- stack.Push(_head);
286
- List<RTreeNode<T>> childrenCopy = childrenBuffer ?? new List<RTreeNode<T>>();
287
- childrenCopy.Clear();
288
- HashSet<T> nearestNeighborsSet = nearestNeighborsBuffer ?? new HashSet<T>(count);
289
- nearestNeighborsSet.Clear();
290
-
291
- while (!current.isTerminal)
292
- {
293
- childrenCopy.Clear();
294
- childrenCopy.AddRange(current.children);
295
- childrenCopy.Sort(Comparison);
296
- for (int i = childrenCopy.Count - 1; 0 <= i; --i)
297
- {
298
- stack.Push(childrenCopy[i]);
299
- }
300
-
301
- current = childrenCopy[0];
302
- if (current.elements.Length <= count)
303
- {
304
- break;
305
- }
306
- }
307
-
308
- while (nearestNeighborsSet.Count < count && stack.TryPop(out RTreeNode<T> selected))
309
- {
310
- foreach (T element in selected.elements)
311
- {
312
- _ = nearestNeighborsSet.Add(element);
313
- }
314
- }
315
-
316
- nearestNeighbors.AddRange(nearestNeighborsSet);
317
- if (count < nearestNeighbors.Count)
318
- {
319
- nearestNeighbors.Sort(NearestComparison);
320
- nearestNeighbors.RemoveRange(count, nearestNeighbors.Count - count);
321
- }
322
-
323
- return;
324
-
325
- int Comparison(RTreeNode<T> lhs, RTreeNode<T> rhs) =>
326
- ((Vector2)lhs.boundary.center - position).sqrMagnitude.CompareTo(
327
- ((Vector2)rhs.boundary.center - position).sqrMagnitude
328
- );
329
-
330
- int NearestComparison(T lhs, T rhs) =>
331
- ((Vector2)_elementTransformer(lhs).center - position).sqrMagnitude.CompareTo(
332
- ((Vector2)_elementTransformer(rhs).center - position).sqrMagnitude
333
- );
334
- }
335
- }
336
- }
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
+ using Utils;
10
+
11
+ [Serializable]
12
+ public sealed class RTree<T>
13
+ {
14
+ [Serializable]
15
+ public sealed class RTreeNode<V>
16
+ {
17
+ public readonly Bounds boundary;
18
+ internal readonly RTreeNode<V>[] children;
19
+ public readonly V[] elements;
20
+ public readonly bool isTerminal;
21
+
22
+ public RTreeNode(
23
+ List<V> elements,
24
+ Func<V, Bounds> elementTransformer,
25
+ int bucketSize,
26
+ int branchFactor
27
+ )
28
+ {
29
+ float minX = float.MaxValue;
30
+ float minY = float.MaxValue;
31
+ float maxX = float.MinValue;
32
+ float maxY = float.MinValue;
33
+ foreach (V element in elements)
34
+ {
35
+ Bounds rectangle = elementTransformer(element);
36
+ Vector3 min = rectangle.min;
37
+ Vector3 max = rectangle.max;
38
+ minX = Math.Min(minX, min.x);
39
+ maxX = Math.Max(maxX, max.x);
40
+ minY = Math.Min(minY, min.y);
41
+ maxY = Math.Max(maxY, max.y);
42
+ }
43
+
44
+ boundary =
45
+ elements.Count <= 0
46
+ ? new Bounds()
47
+ : new Bounds(
48
+ new Vector3(minX + (maxX - minX) / 2, minY + (maxY - minY) / 2),
49
+ new Vector3(maxX - minX, maxY - minY)
50
+ );
51
+ this.elements = elements.ToArray();
52
+ isTerminal = elements.Count <= bucketSize;
53
+ if (isTerminal)
54
+ {
55
+ children = Array.Empty<RTreeNode<V>>();
56
+ return;
57
+ }
58
+
59
+ /*
60
+ http://www.dtic.mil/get-tr-doc/pdf?AD=ADA324493
61
+ var targetSize = rectangles.Count / (double) branchFactor;
62
+ P = branchFactor;
63
+ S = Math.sqrt(P);
64
+ N = targetSize;
65
+
66
+ Ugh.
67
+ */
68
+ double targetSize = elements.Count / (double)branchFactor;
69
+ int intTargetSize = (int)Math.Ceiling(targetSize);
70
+
71
+ List<RTreeNode<V>> tempChildren = new(intTargetSize);
72
+
73
+ double slicesPerAxis = Math.Sqrt(branchFactor);
74
+ int rectanglesPerPagePerAxis = (int)(slicesPerAxis * targetSize);
75
+
76
+ elements.Sort(XAxis);
77
+ foreach (
78
+ List<V> xSlice in elements
79
+ .Partition(rectanglesPerPagePerAxis)
80
+ .Select(enumerable => enumerable as List<V> ?? enumerable.ToList())
81
+ )
82
+ {
83
+ xSlice.Sort(YAxis);
84
+ foreach (
85
+ List<V> ySlice in xSlice
86
+ .Partition(intTargetSize)
87
+ .Select(enumerable => enumerable as List<V> ?? enumerable.ToList())
88
+ )
89
+ {
90
+ RTreeNode<V> node = new(
91
+ ySlice,
92
+ elementTransformer,
93
+ bucketSize,
94
+ branchFactor
95
+ );
96
+ tempChildren.Add(node);
97
+ }
98
+ }
99
+
100
+ children = tempChildren.ToArray();
101
+ return;
102
+
103
+ int XAxis(V lhs, V rhs)
104
+ {
105
+ return elementTransformer(lhs)
106
+ .center.x.CompareTo(elementTransformer(rhs).center.x);
107
+ }
108
+
109
+ int YAxis(V lhs, V rhs)
110
+ {
111
+ return elementTransformer(lhs)
112
+ .center.y.CompareTo(elementTransformer(rhs).center.y);
113
+ }
114
+ }
115
+ }
116
+
117
+ public const int DefaultBucketSize = 10;
118
+ public const int DefaultBranchFactor = 4;
119
+
120
+ public readonly ImmutableArray<T> elements;
121
+ public Bounds Boundary => _bounds;
122
+
123
+ private readonly Bounds _bounds;
124
+ private readonly Func<T, Bounds> _elementTransformer;
125
+ private readonly RTreeNode<T> _head;
126
+
127
+ public RTree(
128
+ IEnumerable<T> points,
129
+ Func<T, Bounds> elementTransformer,
130
+ int bucketSize = DefaultBucketSize,
131
+ int branchFactor = DefaultBranchFactor
132
+ )
133
+ {
134
+ _elementTransformer =
135
+ elementTransformer ?? throw new ArgumentNullException(nameof(elementTransformer));
136
+ elements =
137
+ points?.ToImmutableArray() ?? throw new ArgumentNullException(nameof(points));
138
+ _bounds = elements.Select(elementTransformer).GetBounds() ?? new Bounds();
139
+ _head = new RTreeNode<T>(
140
+ elements.ToList(),
141
+ elementTransformer,
142
+ bucketSize,
143
+ branchFactor
144
+ );
145
+ }
146
+
147
+ public IEnumerable<T> GetElementsInRange(
148
+ Vector2 position,
149
+ float range,
150
+ float minimumRange = 0f
151
+ )
152
+ {
153
+ Circle area = new(position, range);
154
+ if (0 < minimumRange)
155
+ {
156
+ Circle minimumArea = new(position, minimumRange);
157
+ return GetElementsInBounds(
158
+ new Bounds(
159
+ new Vector3(position.x, position.y, 0f),
160
+ new Vector3(range * 2f, range * 2f, 1f)
161
+ )
162
+ )
163
+ .Where(element =>
164
+ {
165
+ Bounds elementBoundary = _elementTransformer(element);
166
+ if (!area.Intersects(elementBoundary))
167
+ {
168
+ return false;
169
+ }
170
+
171
+ return !minimumArea.Intersects(elementBoundary);
172
+ });
173
+ }
174
+ return GetElementsInBounds(
175
+ new Bounds(
176
+ new Vector3(position.x, position.y, 0f),
177
+ new Vector3(range * 2f, range * 2f, 1f)
178
+ )
179
+ )
180
+ .Where(element =>
181
+ {
182
+ Bounds elementBoundary = _elementTransformer(element);
183
+ if (!area.Intersects(elementBoundary))
184
+ {
185
+ return false;
186
+ }
187
+
188
+ return true;
189
+ });
190
+ }
191
+
192
+ public IEnumerable<T> GetElementsInBounds(Bounds bounds)
193
+ {
194
+ Stack<RTreeNode<T>> nodeBuffer = Buffers<RTreeNode<T>>.Stack;
195
+ return GetElementsInBounds(bounds, nodeBuffer);
196
+ }
197
+
198
+ public IEnumerable<T> GetElementsInBounds(Bounds bounds, Stack<RTreeNode<T>> nodeBuffer)
199
+ {
200
+ if (!bounds.FastIntersects2D(_bounds))
201
+ {
202
+ yield break;
203
+ }
204
+
205
+ Stack<RTreeNode<T>> nodesToVisit = nodeBuffer ?? new Stack<RTreeNode<T>>();
206
+ nodeBuffer.Clear();
207
+ nodesToVisit.Push(_head);
208
+
209
+ while (nodesToVisit.TryPop(out RTreeNode<T> currentNode))
210
+ {
211
+ if (currentNode.isTerminal)
212
+ {
213
+ foreach (T element in currentNode.elements)
214
+ {
215
+ if (bounds.FastIntersects2D(_elementTransformer(element)))
216
+ {
217
+ yield return element;
218
+ }
219
+ }
220
+
221
+ continue;
222
+ }
223
+
224
+ if (bounds.Overlaps2D(currentNode.boundary))
225
+ {
226
+ foreach (T element in currentNode.elements)
227
+ {
228
+ yield return element;
229
+ }
230
+
231
+ continue;
232
+ }
233
+
234
+ foreach (RTreeNode<T> child in currentNode.children)
235
+ {
236
+ if (child.elements.Length <= 0)
237
+ {
238
+ continue;
239
+ }
240
+
241
+ if (!bounds.FastIntersects2D(child.boundary))
242
+ {
243
+ continue;
244
+ }
245
+
246
+ nodesToVisit.Push(child);
247
+ }
248
+ }
249
+ }
250
+
251
+ public void GetApproximateNearestNeighbors(
252
+ Vector2 position,
253
+ int count,
254
+ List<T> nearestNeighbors
255
+ )
256
+ {
257
+ Stack<RTreeNode<T>> nodeBuffer = Buffers<RTreeNode<T>>.Stack;
258
+ List<RTreeNode<T>> childrenBuffer = Buffers<RTreeNode<T>>.List;
259
+ HashSet<T> nearestNeighborBuffer = Buffers<T>.HashSet;
260
+ GetApproximateNearestNeighbors(
261
+ position,
262
+ count,
263
+ nearestNeighbors,
264
+ nodeBuffer,
265
+ childrenBuffer,
266
+ nearestNeighborBuffer
267
+ );
268
+ }
269
+
270
+ // Heavily adapted http://homepage.divms.uiowa.edu/%7Ekvaradar/sp2012/daa/ann.pdf
271
+ public void GetApproximateNearestNeighbors(
272
+ Vector2 position,
273
+ int count,
274
+ List<T> nearestNeighbors,
275
+ Stack<RTreeNode<T>> nodeBuffer,
276
+ List<RTreeNode<T>> childrenBuffer,
277
+ HashSet<T> nearestNeighborsBuffer
278
+ )
279
+ {
280
+ nearestNeighbors.Clear();
281
+
282
+ RTreeNode<T> current = _head;
283
+ Stack<RTreeNode<T>> stack = nodeBuffer ?? new Stack<RTreeNode<T>>();
284
+ stack.Clear();
285
+ stack.Push(_head);
286
+ List<RTreeNode<T>> childrenCopy = childrenBuffer ?? new List<RTreeNode<T>>();
287
+ childrenCopy.Clear();
288
+ HashSet<T> nearestNeighborsSet = nearestNeighborsBuffer ?? new HashSet<T>(count);
289
+ nearestNeighborsSet.Clear();
290
+
291
+ while (!current.isTerminal)
292
+ {
293
+ childrenCopy.Clear();
294
+ childrenCopy.AddRange(current.children);
295
+ childrenCopy.Sort(Comparison);
296
+ for (int i = childrenCopy.Count - 1; 0 <= i; --i)
297
+ {
298
+ stack.Push(childrenCopy[i]);
299
+ }
300
+
301
+ current = childrenCopy[0];
302
+ if (current.elements.Length <= count)
303
+ {
304
+ break;
305
+ }
306
+ }
307
+
308
+ while (nearestNeighborsSet.Count < count && stack.TryPop(out RTreeNode<T> selected))
309
+ {
310
+ foreach (T element in selected.elements)
311
+ {
312
+ _ = nearestNeighborsSet.Add(element);
313
+ }
314
+ }
315
+
316
+ nearestNeighbors.AddRange(nearestNeighborsSet);
317
+ if (count < nearestNeighbors.Count)
318
+ {
319
+ nearestNeighbors.Sort(NearestComparison);
320
+ nearestNeighbors.RemoveRange(count, nearestNeighbors.Count - count);
321
+ }
322
+
323
+ return;
324
+
325
+ int Comparison(RTreeNode<T> lhs, RTreeNode<T> rhs) =>
326
+ ((Vector2)lhs.boundary.center - position).sqrMagnitude.CompareTo(
327
+ ((Vector2)rhs.boundary.center - position).sqrMagnitude
328
+ );
329
+
330
+ int NearestComparison(T lhs, T rhs) =>
331
+ ((Vector2)_elementTransformer(lhs).center - position).sqrMagnitude.CompareTo(
332
+ ((Vector2)_elementTransformer(rhs).center - position).sqrMagnitude
333
+ );
334
+ }
335
+ }
336
+ }