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

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 (266) hide show
  1. package/.gitattributes +63 -63
  2. package/.github/workflows/npm-publish.yml +60 -0
  3. package/.github/workflows/unity-package.yml +87 -0
  4. package/Editor/AnimationCopier.cs +158 -158
  5. package/Editor/AnimationCopier.cs.meta +2 -2
  6. package/Editor/AnimationCreator.cs +262 -262
  7. package/Editor/AnimationCreator.cs.meta +11 -11
  8. package/Editor/AnimationEventEditor.cs +887 -887
  9. package/Editor/AnimatorControllerCopier.cs +162 -162
  10. package/Editor/AnimatorControllerCopier.cs.meta +2 -2
  11. package/Editor/CustomEditors/MatchColliderToSpriteEditor.cs +34 -34
  12. package/Editor/CustomEditors/MatchColliderToSpriteEditor.cs.meta +2 -2
  13. package/Editor/CustomEditors.meta +2 -2
  14. package/Editor/EnsureTextureSizeWizard.cs +110 -110
  15. package/Editor/EnsureTextureSizeWizard.cs.meta +2 -2
  16. package/Editor/PrefabCheckWizard.cs +165 -165
  17. package/Editor/PrefabCheckWizard.cs.meta +11 -11
  18. package/Editor/SpriteSettingsApplier.cs +168 -168
  19. package/Editor/SpriteSettingsApplier.cs.meta +2 -2
  20. package/Editor/StringInListeDrawer.cs +56 -56
  21. package/Editor/TextureResizerWizard.cs +181 -181
  22. package/Editor/TextureResizerWizard.cs.meta +2 -2
  23. package/Editor/TextureSettingsApplier.cs +171 -171
  24. package/Editor/TextureSettingsApplier.cs.meta +2 -2
  25. package/Editor/Utils/EditorUtilities.cs +22 -22
  26. package/Editor/Utils/EditorUtilities.cs.meta +11 -11
  27. package/Editor/Utils/ReadOnlyPropertyDrawer.cs +26 -26
  28. package/Editor/Utils/ReadOnlyPropertyDrawer.cs.meta +11 -11
  29. package/Editor/Utils.meta +8 -8
  30. package/Editor/WallstopStudios.UnityHelpers.Editor.asmdef +17 -17
  31. package/Editor/WallstopStudios.UnityHelpers.Editor.asmdef.meta +7 -7
  32. package/LICENSE +21 -21
  33. package/LICENSE.md +6 -6
  34. package/LICENSE.meta +7 -7
  35. package/README.md +117 -117
  36. package/Runtime/Binaries/Microsoft.Bcl.AsyncInterfaces.dll.meta +33 -33
  37. package/Runtime/Binaries/Microsoft.Bcl.AsyncInterfaces.xml +223 -223
  38. package/Runtime/Binaries/Microsoft.Bcl.AsyncInterfaces.xml.meta +7 -7
  39. package/Runtime/Binaries/System.Text.Encodings.Web.dll.meta +33 -33
  40. package/Runtime/Binaries/System.Text.Encodings.Web.xml +935 -935
  41. package/Runtime/Binaries/System.Text.Encodings.Web.xml.meta +7 -7
  42. package/Runtime/Binaries/System.Text.Json.dll.meta +33 -33
  43. package/Runtime/Binaries/System.Text.Json.xml +4829 -4829
  44. package/Runtime/Binaries/System.Text.Json.xml.meta +7 -7
  45. package/Runtime/Binaries.meta +8 -8
  46. package/Runtime/Core/Attributes/AnimationEventAttribute.cs +131 -131
  47. package/Runtime/Core/Attributes/ChildComponentAttribute.cs +189 -189
  48. package/Runtime/Core/Attributes/KSerializableAttribute.cs +19 -19
  49. package/Runtime/Core/Attributes/NotNullAttribute.cs +32 -32
  50. package/Runtime/Core/Attributes/ParentComponent.cs +184 -184
  51. package/Runtime/Core/Attributes/ReadOnlyAttribute.cs +6 -6
  52. package/Runtime/Core/Attributes/RelationalComponentExtensions.cs +14 -14
  53. package/Runtime/Core/Attributes/SiblingComponentAttribute.cs +119 -119
  54. package/Runtime/Core/Attributes/SiblingComponentAttribute.cs.meta +11 -11
  55. package/Runtime/Core/Attributes/ValidateAssignmentAttribute.cs +101 -101
  56. package/Runtime/Core/Attributes/ValidateAssignmentAttribute.cs.meta +11 -11
  57. package/Runtime/Core/Attributes.meta +8 -8
  58. package/Runtime/Core/DataStructure/Adapters/FastVector2Int.cs +92 -92
  59. package/Runtime/Core/DataStructure/Adapters/FastVector3Int.cs +185 -185
  60. package/Runtime/Core/DataStructure/Adapters/KGuid.cs +305 -305
  61. package/Runtime/Core/DataStructure/Adapters/KVector2.cs +80 -80
  62. package/Runtime/Core/DataStructure/Circle.cs +50 -50
  63. package/Runtime/Core/DataStructure/CyclicBuffer.cs +155 -155
  64. package/Runtime/Core/DataStructure/ISpatialTree.cs +60 -60
  65. package/Runtime/Core/DataStructure/ISpatialTree.cs.meta +11 -11
  66. package/Runtime/Core/DataStructure/KDTree.cs +290 -290
  67. package/Runtime/Core/DataStructure/KDTree.cs.meta +11 -11
  68. package/Runtime/Core/DataStructure/QuadTree.cs +279 -279
  69. package/Runtime/Core/DataStructure/RTree.cs +336 -336
  70. package/Runtime/Core/DataStructure/RTree.cs.meta +11 -11
  71. package/Runtime/Core/DataStructure/StringWrapper.cs +91 -91
  72. package/Runtime/Core/DataStructure/TimedCache.cs +51 -51
  73. package/Runtime/Core/Extension/AnimatorExtensions.cs +25 -25
  74. package/Runtime/Core/Extension/CircleExtensions.cs +25 -25
  75. package/Runtime/Core/Extension/ColorExtensions.cs +338 -338
  76. package/Runtime/Core/Extension/DictionaryExtensions.cs +251 -251
  77. package/Runtime/Core/Extension/DirectionExtensions.cs +213 -213
  78. package/Runtime/Core/Extension/HashSetExtensions.cs +12 -12
  79. package/Runtime/Core/Extension/IEnumerableExtensions.cs +122 -122
  80. package/Runtime/Core/Extension/IListExtensions.cs +89 -89
  81. package/Runtime/Core/Extension/LoggingExtensions.cs +258 -258
  82. package/Runtime/Core/Extension/RandomExtensions.cs +109 -109
  83. package/Runtime/Core/Extension/StringExtensions.cs +151 -151
  84. package/Runtime/Core/Extension/UnityExtensions.cs +1607 -1607
  85. package/Runtime/Core/Helper/ArrayConverter.cs +39 -39
  86. package/Runtime/Core/Helper/ArrayConverter.cs.meta +2 -2
  87. package/Runtime/Core/Helper/AssignUtilities.cs +14 -14
  88. package/Runtime/Core/Helper/AssignUtilities.cs.meta +11 -11
  89. package/Runtime/Core/Helper/Enumerables.cs +17 -17
  90. package/Runtime/Core/Helper/Geometry.cs +43 -43
  91. package/Runtime/Core/Helper/Helpers.cs +722 -722
  92. package/Runtime/Core/Helper/Helpers.cs.meta +11 -11
  93. package/Runtime/Core/Helper/IterationHelpers.cs +32 -32
  94. package/Runtime/Core/Helper/IterationHelpers.cs.meta +11 -11
  95. package/Runtime/Core/Helper/LifetimeHelpers.cs +13 -13
  96. package/Runtime/Core/Helper/Objects.cs +767 -767
  97. package/Runtime/Core/Helper/Partials/LogHelpers.cs +13 -13
  98. package/Runtime/Core/Helper/Partials/LogHelpers.cs.meta +2 -2
  99. package/Runtime/Core/Helper/Partials/MathHelpers.cs +30 -30
  100. package/Runtime/Core/Helper/Partials/MathHelpers.cs.meta +2 -2
  101. package/Runtime/Core/Helper/Partials/ObjectHelpers.cs +388 -388
  102. package/Runtime/Core/Helper/Partials/ObjectHelpers.cs.meta +2 -2
  103. package/Runtime/Core/Helper/Partials/TransformHelpers.cs +167 -167
  104. package/Runtime/Core/Helper/Partials/TransformHelpers.cs.meta +2 -2
  105. package/Runtime/Core/Helper/Partials.meta +2 -2
  106. package/Runtime/Core/Helper/ReflectionHelpers.cs +152 -152
  107. package/Runtime/Core/Helper/ReflectionHelpers.cs.meta +2 -2
  108. package/Runtime/Core/Helper/SpriteHelpers.cs +86 -86
  109. package/Runtime/Core/Helper/SpriteHelpers.cs.meta +11 -11
  110. package/Runtime/Core/Helper/StringInList.cs +31 -31
  111. package/Runtime/Core/Helper/StringInList.cs.meta +11 -11
  112. package/Runtime/Core/Helper/WallMath.cs +166 -166
  113. package/Runtime/Core/Math/Line.cs +55 -55
  114. package/Runtime/Core/Math/Parabola.cs +47 -47
  115. package/Runtime/Core/Math/PointPolygonCheck.cs +36 -36
  116. package/Runtime/Core/Math/PointPolygonCheck.cs.meta +11 -11
  117. package/Runtime/Core/Math/Range.cs +92 -92
  118. package/Runtime/Core/Math/XXHash.cs +310 -310
  119. package/Runtime/Core/Math/XXHash.cs.meta +11 -11
  120. package/Runtime/Core/Model/Direction.cs +43 -43
  121. package/Runtime/Core/OneOf/FastOneOf.cs +152 -152
  122. package/Runtime/Core/OneOf/None.cs +4 -4
  123. package/Runtime/Core/Random/AbstractRandom.cs +561 -561
  124. package/Runtime/Core/Random/DotNetRandom.cs +52 -52
  125. package/Runtime/Core/Random/DotNetRandom.cs.meta +2 -2
  126. package/Runtime/Core/Random/IRandom.cs +160 -160
  127. package/Runtime/Core/Random/NativePcgRandom.cs +97 -97
  128. package/Runtime/Core/Random/PRNG.cs +7 -7
  129. package/Runtime/Core/Random/PRNG.cs.meta +2 -2
  130. package/Runtime/Core/Random/PcgRandom.cs +149 -149
  131. package/Runtime/Core/Random/PerlinNoise.cs +369 -369
  132. package/Runtime/Core/Random/PerlinNoise.cs.meta +2 -2
  133. package/Runtime/Core/Random/RandomState.cs +131 -131
  134. package/Runtime/Core/Random/RandomUtilities.cs +26 -26
  135. package/Runtime/Core/Random/RandomUtilities.cs.meta +11 -11
  136. package/Runtime/Core/Random/RomuDuo.cs +116 -116
  137. package/Runtime/Core/Random/RomuDuo.cs.meta +2 -2
  138. package/Runtime/Core/Random/SplitMix64.cs +94 -94
  139. package/Runtime/Core/Random/SplitMix64.cs.meta +2 -2
  140. package/Runtime/Core/Random/SquirrelRandom.cs +84 -84
  141. package/Runtime/Core/Random/SystemRandom.cs +162 -162
  142. package/Runtime/Core/Random/ThreadLocalRandom.cs +12 -12
  143. package/Runtime/Core/Random/UnityRandom.cs +57 -57
  144. package/Runtime/Core/Random/UnityRandom.cs.meta +11 -11
  145. package/Runtime/Core/Random/WyRandom.cs +121 -121
  146. package/Runtime/Core/Random/WyRandom.cs.meta +2 -2
  147. package/Runtime/Core/Random/XorShiftRandom.cs +47 -47
  148. package/Runtime/Core/Random/XorShiftRandom.cs.meta +11 -11
  149. package/Runtime/Core/Random/XorShiroRandom.cs +117 -117
  150. package/Runtime/Core/Random/XorShiroRandom.cs.meta +2 -2
  151. package/Runtime/Core/Serialization/JsonConverters/Vector2Converter.cs +74 -74
  152. package/Runtime/Core/Serialization/JsonConverters/Vector3Converter.cs +81 -81
  153. package/Runtime/Core/Serialization/Serializer.cs +184 -184
  154. package/Runtime/Core/Threading/SingleThreadedThreadPool.cs +107 -107
  155. package/Runtime/Protobuf-Net/System.Buffers.dll.meta +33 -33
  156. package/Runtime/Protobuf-Net/System.Collections.Immutable.dll.meta +33 -33
  157. package/Runtime/Protobuf-Net/System.Collections.Immutable.xml +5379 -5379
  158. package/Runtime/Protobuf-Net/System.Collections.Immutable.xml.meta +7 -7
  159. package/Runtime/Protobuf-Net/System.Numerics.Vectors.dll.meta +33 -33
  160. package/Runtime/Protobuf-Net/System.Runtime.CompilerServices.Unsafe.dll.meta +33 -33
  161. package/Runtime/Protobuf-Net/System.Runtime.CompilerServices.Unsafe.xml +290 -290
  162. package/Runtime/Protobuf-Net/System.Runtime.CompilerServices.Unsafe.xml.meta +7 -7
  163. package/Runtime/Protobuf-Net/protobuf-net.Core.dll.meta +33 -33
  164. package/Runtime/Protobuf-Net/protobuf-net.dll.meta +33 -33
  165. package/Runtime/UI/LayeredImage.cs +364 -364
  166. package/Runtime/UI/LayeredImage.cs.meta +2 -2
  167. package/Runtime/UI.meta +2 -2
  168. package/Runtime/Utils/AnimationEventEqualityComparer.cs +161 -161
  169. package/Runtime/Utils/AnimatorEnumStateMachine.cs +88 -88
  170. package/Runtime/Utils/Buffers.cs +32 -32
  171. package/Runtime/Utils/CenterPointOffset.cs +30 -30
  172. package/Runtime/Utils/CenterPointOffset.cs.meta +2 -2
  173. package/Runtime/Utils/CircleLineRenderer.cs +134 -134
  174. package/Runtime/Utils/CoroutineHandler.cs +4 -4
  175. package/Runtime/Utils/CoroutineHandler.cs.meta +2 -2
  176. package/Runtime/Utils/MatchColliderToSprite.cs +94 -94
  177. package/Runtime/Utils/MatchColliderToSprite.cs.meta +2 -2
  178. package/Runtime/Utils/Oscillator.cs +27 -27
  179. package/Runtime/Utils/RuntimeSingleton.cs +57 -57
  180. package/Runtime/Utils/RuntimeSingleton.cs.meta +11 -11
  181. package/Runtime/Utils/SetTextureImportData.cs +69 -69
  182. package/Runtime/Utils/SpriteRendererMetadata.cs +312 -312
  183. package/Runtime/Utils/SpriteRendererMetadata.cs.meta +2 -2
  184. package/Runtime/Utils/SpriteRendererSyncer.cs +100 -100
  185. package/Runtime/Utils/SpriteRendererSyncer.cs.meta +2 -2
  186. package/Runtime/Utils/TextureScale.cs +179 -179
  187. package/Runtime/Utils/TextureScale.cs.meta +2 -2
  188. package/Runtime/WallstopStudios.UnityHelpers.asmdef +13 -13
  189. package/Tests/Runtime/Attributes/ChildComponentTests.cs +81 -81
  190. package/Tests/Runtime/Attributes/Components/ExpectChildSpriteRenderers.cs +28 -28
  191. package/Tests/Runtime/Attributes/Components/ExpectParentSpriteRenderers.cs +28 -28
  192. package/Tests/Runtime/Attributes/ParentComponentTests.cs +68 -68
  193. package/Tests/Runtime/Components/RelationalComponentTester.cs +34 -34
  194. package/Tests/Runtime/Components/RelationalComponentTester.cs.meta +2 -2
  195. package/Tests/Runtime/Components.meta +2 -2
  196. package/Tests/Runtime/DataStructures/BalancedKDTreeTests.cs +14 -14
  197. package/Tests/Runtime/DataStructures/BalancedKDTreeTests.cs.meta +11 -11
  198. package/Tests/Runtime/DataStructures/CyclicBufferTests.cs +324 -324
  199. package/Tests/Runtime/DataStructures/QuadTreeTests.cs +14 -14
  200. package/Tests/Runtime/DataStructures/QuadTreeTests.cs.meta +11 -11
  201. package/Tests/Runtime/DataStructures/SpatialTreeTests.cs +130 -130
  202. package/Tests/Runtime/DataStructures/SpatialTreeTests.cs.meta +11 -11
  203. package/Tests/Runtime/DataStructures/UnbalancedKDTreeTests.cs +14 -14
  204. package/Tests/Runtime/DataStructures/UnbalancedKDTreeTests.cs.meta +11 -11
  205. package/Tests/Runtime/DataStructures.meta +8 -8
  206. package/Tests/Runtime/Extensions/DictionaryExtensionTests.cs +439 -439
  207. package/Tests/Runtime/Extensions/DictionaryExtensionTests.cs.meta +2 -2
  208. package/Tests/Runtime/Extensions/IListExtensionTests.cs +76 -76
  209. package/Tests/Runtime/Extensions/RandomExtensionTests.cs +27 -27
  210. package/Tests/Runtime/Extensions/RandomExtensionTests.cs.meta +2 -2
  211. package/Tests/Runtime/Extensions/StringExtensionTests.cs +31 -31
  212. package/Tests/Runtime/Extensions/StringExtensionTests.cs.meta +2 -2
  213. package/Tests/Runtime/Extensions.meta +2 -2
  214. package/Tests/Runtime/Helper/ArrayConverterTests.cs +19 -19
  215. package/Tests/Runtime/Helper/ArrayConverterTests.cs.meta +2 -2
  216. package/Tests/Runtime/Helper/ObjectHelperTests.cs +402 -402
  217. package/Tests/Runtime/Helper/ObjectHelperTests.cs.meta +2 -2
  218. package/Tests/Runtime/Helper/WallMathTests.cs +233 -233
  219. package/Tests/Runtime/Helper/WallMathTests.cs.meta +2 -2
  220. package/Tests/Runtime/Helper.meta +2 -2
  221. package/Tests/Runtime/Performance/KDTreePerformanceTests.cs +14 -14
  222. package/Tests/Runtime/Performance/KDTreePerformanceTests.cs.meta +11 -11
  223. package/Tests/Runtime/Performance/QuadTreePerformanceTests.cs +14 -14
  224. package/Tests/Runtime/Performance/QuadTreePerformanceTests.cs.meta +11 -11
  225. package/Tests/Runtime/Performance/RandomPerformanceTests.cs +139 -139
  226. package/Tests/Runtime/Performance/RandomPerformanceTests.cs.meta +11 -11
  227. package/Tests/Runtime/Performance/RelationComponentPerformanceTests.cs +37 -37
  228. package/Tests/Runtime/Performance/RelationComponentPerformanceTests.cs.meta +2 -2
  229. package/Tests/Runtime/Performance/SpatialTreePerformanceTest.cs +154 -154
  230. package/Tests/Runtime/Performance/SpatialTreePerformanceTest.cs.meta +11 -11
  231. package/Tests/Runtime/Performance/UnbalancedKDTreeTests.cs +14 -14
  232. package/Tests/Runtime/Performance/UnbalancedKDTreeTests.cs.meta +11 -11
  233. package/Tests/Runtime/Performance.meta +8 -8
  234. package/Tests/Runtime/Random/DotNetRandomTests.cs +9 -9
  235. package/Tests/Runtime/Random/DotNetRandomTests.cs.meta +2 -2
  236. package/Tests/Runtime/Random/PcgRandomTests.cs +9 -9
  237. package/Tests/Runtime/Random/PcgRandomTests.cs.meta +11 -11
  238. package/Tests/Runtime/Random/RandomTestBase.cs +787 -787
  239. package/Tests/Runtime/Random/RandomTestBase.cs.meta +11 -11
  240. package/Tests/Runtime/Random/RomuDuoRandomTests.cs +9 -9
  241. package/Tests/Runtime/Random/RomuDuoRandomTests.cs.meta +2 -2
  242. package/Tests/Runtime/Random/SplitMix64RandomTests.cs +9 -9
  243. package/Tests/Runtime/Random/SplitMix64RandomTests.cs.meta +2 -2
  244. package/Tests/Runtime/Random/SquirrelRandomTests.cs +14 -14
  245. package/Tests/Runtime/Random/SquirrelRandomTests.cs.meta +11 -11
  246. package/Tests/Runtime/Random/SystemRandomTests.cs +10 -10
  247. package/Tests/Runtime/Random/SystemRandomTests.cs.meta +11 -11
  248. package/Tests/Runtime/Random/UnityRandomTests.cs +9 -9
  249. package/Tests/Runtime/Random/UnityRandomTests.cs.meta +11 -11
  250. package/Tests/Runtime/Random/WyRandomTests.cs +9 -9
  251. package/Tests/Runtime/Random/WyRandomTests.cs.meta +2 -2
  252. package/Tests/Runtime/Random/XorShiftRandomTests.cs +9 -9
  253. package/Tests/Runtime/Random/XorShiftRandomTests.cs.meta +11 -11
  254. package/Tests/Runtime/Random/XorShiroRandomTests.cs +9 -9
  255. package/Tests/Runtime/Random/XorShiroRandomTests.cs.meta +2 -2
  256. package/Tests/Runtime/Random.meta +8 -8
  257. package/Tests/Runtime/Serialization/JsonSerializationTest.cs +84 -84
  258. package/Tests/Runtime/Serialization/JsonSerializationTest.cs.meta +2 -2
  259. package/Tests/Runtime/Serialization.meta +2 -2
  260. package/Tests/Runtime/Utils/SpriteRendererMetadataTests.cs +399 -399
  261. package/Tests/Runtime/Utils/SpriteRendererMetadataTests.cs.meta +2 -2
  262. package/Tests/Runtime/Utils.meta +2 -2
  263. package/Tests/Runtime/WallstopStudios.UnityHelpers.Tests.Runtime.asmdef +22 -22
  264. package/Tests/Runtime/WallstopStudios.UnityHelpers.Tests.Runtime.asmdef.meta +7 -7
  265. package/Tests/Runtime.meta +8 -8
  266. 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
+ }