com.wallstop-studios.unity-helpers 2.0.0-rc57 → 2.0.0-rc58
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.
- package/.config/dotnet-tools.json +9 -9
- package/.editorconfig +184 -184
- package/.gitattributes +63 -63
- package/.github/workflows/npm-publish.yml +66 -66
- package/.pre-commit-config.yaml +21 -21
- package/Editor/AnimationCopier.cs +181 -181
- package/Editor/AnimationCopier.cs.meta +2 -2
- package/Editor/AnimationCreator.cs +253 -253
- package/Editor/AnimationCreator.cs.meta +11 -11
- package/Editor/AnimationEventEditor.cs +887 -887
- package/Editor/AnimatorControllerCopier.cs +162 -162
- package/Editor/AnimatorControllerCopier.cs.meta +2 -2
- package/Editor/CustomEditors/MatchColliderToSpriteEditor.cs +34 -34
- package/Editor/CustomEditors/MatchColliderToSpriteEditor.cs.meta +2 -2
- package/Editor/CustomEditors.meta +2 -2
- package/Editor/FitTextureSizeWizard.cs +147 -147
- package/Editor/FitTextureSizeWizard.cs.meta +2 -2
- package/Editor/PrefabCheckWizard.cs +170 -170
- package/Editor/PrefabCheckWizard.cs.meta +11 -11
- package/Editor/SpriteSettingsApplier.cs +272 -272
- package/Editor/SpriteSettingsApplier.cs.meta +2 -2
- package/Editor/StringInListeDrawer.cs +56 -56
- package/Editor/TextureResizerWizard.cs +181 -181
- package/Editor/TextureResizerWizard.cs.meta +2 -2
- package/Editor/TextureSettingsApplier.cs +178 -178
- package/Editor/TextureSettingsApplier.cs.meta +2 -2
- package/Editor/Utils/DxReadOnlyPropertyDrawer.cs +26 -26
- package/Editor/Utils/DxReadOnlyPropertyDrawer.cs.meta +11 -11
- package/Editor/Utils/EditorUtilities.cs +22 -22
- package/Editor/Utils/EditorUtilities.cs.meta +11 -11
- package/Editor/Utils.meta +8 -8
- package/Editor/WShowIfPropertyDrawer.cs +63 -63
- package/Editor/WallstopStudios.UnityHelpers.Editor.asmdef +17 -17
- package/Editor/WallstopStudios.UnityHelpers.Editor.asmdef.meta +7 -7
- package/LICENSE +21 -21
- package/LICENSE.md +6 -6
- package/LICENSE.meta +7 -7
- package/README.md +177 -177
- package/Runtime/Binaries/Microsoft.Bcl.AsyncInterfaces.dll.meta +33 -33
- package/Runtime/Binaries/Microsoft.Bcl.AsyncInterfaces.xml +223 -223
- package/Runtime/Binaries/Microsoft.Bcl.AsyncInterfaces.xml.meta +7 -7
- package/Runtime/Binaries/System.Text.Encodings.Web.dll.meta +33 -33
- package/Runtime/Binaries/System.Text.Encodings.Web.xml +935 -935
- package/Runtime/Binaries/System.Text.Encodings.Web.xml.meta +7 -7
- package/Runtime/Binaries/System.Text.Json.dll.meta +33 -33
- package/Runtime/Binaries/System.Text.Json.xml +4829 -4829
- package/Runtime/Binaries/System.Text.Json.xml.meta +7 -7
- package/Runtime/Binaries.meta +8 -8
- package/Runtime/Core/Attributes/AnimationEventAttribute.cs +131 -131
- package/Runtime/Core/Attributes/ChildComponentAttribute.cs +209 -209
- package/Runtime/Core/Attributes/DxReadOnlyAttribute.cs +6 -6
- package/Runtime/Core/Attributes/KSerializableAttribute.cs +19 -19
- package/Runtime/Core/Attributes/NotNullAttribute.cs +32 -32
- package/Runtime/Core/Attributes/ParentComponent.cs +185 -185
- package/Runtime/Core/Attributes/RelationalComponentExtensions.cs +14 -14
- package/Runtime/Core/Attributes/SiblingComponentAttribute.cs +117 -117
- package/Runtime/Core/Attributes/SiblingComponentAttribute.cs.meta +11 -11
- package/Runtime/Core/Attributes/ValidateAssignmentAttribute.cs +101 -101
- package/Runtime/Core/Attributes/ValidateAssignmentAttribute.cs.meta +11 -11
- package/Runtime/Core/Attributes/WShowIfAttribute.cs +16 -16
- package/Runtime/Core/Attributes.meta +8 -8
- package/Runtime/Core/DataStructure/Adapters/FastVector2Int.cs +92 -92
- package/Runtime/Core/DataStructure/Adapters/FastVector3Int.cs +185 -185
- package/Runtime/Core/DataStructure/Adapters/KGuid.cs +305 -305
- package/Runtime/Core/DataStructure/Adapters/KVector2.cs +80 -80
- package/Runtime/Core/DataStructure/Circle.cs +50 -50
- package/Runtime/Core/DataStructure/CyclicBuffer.cs +153 -153
- package/Runtime/Core/DataStructure/ISpatialTree.cs +60 -60
- package/Runtime/Core/DataStructure/ISpatialTree.cs.meta +11 -11
- package/Runtime/Core/DataStructure/KDTree.cs +292 -292
- package/Runtime/Core/DataStructure/KDTree.cs.meta +11 -11
- package/Runtime/Core/DataStructure/QuadTree.cs +287 -287
- package/Runtime/Core/DataStructure/RTree.cs +346 -346
- package/Runtime/Core/DataStructure/RTree.cs.meta +11 -11
- package/Runtime/Core/DataStructure/StringWrapper.cs +91 -91
- package/Runtime/Core/DataStructure/TimedCache.cs +66 -66
- package/Runtime/Core/Extension/AnimatorExtensions.cs +25 -25
- package/Runtime/Core/Extension/AsyncOperationExtensions.cs +110 -110
- package/Runtime/Core/Extension/CircleExtensions.cs +25 -25
- package/Runtime/Core/Extension/ColorExtensions.cs +629 -629
- package/Runtime/Core/Extension/DictionaryExtensions.cs +279 -279
- package/Runtime/Core/Extension/DirectionExtensions.cs +213 -213
- package/Runtime/Core/Extension/EnumExtensions.cs +37 -37
- package/Runtime/Core/Extension/EnumExtensions.cs.meta +2 -2
- package/Runtime/Core/Extension/HashSetExtensions.cs +12 -12
- package/Runtime/Core/Extension/IEnumerableExtensions.cs +122 -122
- package/Runtime/Core/Extension/IListExtensions.cs +106 -106
- package/Runtime/Core/Extension/LoggingExtensions.cs +258 -258
- package/Runtime/Core/Extension/RandomExtensions.cs +109 -109
- package/Runtime/Core/Extension/SerializedPropertyExtensions.cs +157 -157
- package/Runtime/Core/Extension/StringExtensions.cs +151 -151
- package/Runtime/Core/Extension/UnityExtensions.cs +1608 -1608
- package/Runtime/Core/Helper/ArrayConverter.cs +39 -39
- package/Runtime/Core/Helper/ArrayConverter.cs.meta +2 -2
- package/Runtime/Core/Helper/AssignUtilities.cs +14 -14
- package/Runtime/Core/Helper/AssignUtilities.cs.meta +11 -11
- package/Runtime/Core/Helper/Enumerables.cs +17 -17
- package/Runtime/Core/Helper/FormattingHelpers.cs +32 -32
- package/Runtime/Core/Helper/Geometry.cs +43 -43
- package/Runtime/Core/Helper/Helpers.cs +722 -722
- package/Runtime/Core/Helper/Helpers.cs.meta +11 -11
- package/Runtime/Core/Helper/IterationHelpers.cs +32 -32
- package/Runtime/Core/Helper/IterationHelpers.cs.meta +11 -11
- package/Runtime/Core/Helper/LifetimeHelpers.cs +13 -13
- package/Runtime/Core/Helper/Objects.cs +769 -769
- package/Runtime/Core/Helper/Partials/LogHelpers.cs +13 -13
- package/Runtime/Core/Helper/Partials/LogHelpers.cs.meta +2 -2
- package/Runtime/Core/Helper/Partials/MathHelpers.cs +30 -30
- package/Runtime/Core/Helper/Partials/MathHelpers.cs.meta +2 -2
- package/Runtime/Core/Helper/Partials/ObjectHelpers.cs +388 -388
- package/Runtime/Core/Helper/Partials/ObjectHelpers.cs.meta +2 -2
- package/Runtime/Core/Helper/Partials/TransformHelpers.cs +189 -189
- package/Runtime/Core/Helper/Partials/TransformHelpers.cs.meta +2 -2
- package/Runtime/Core/Helper/Partials.meta +2 -2
- package/Runtime/Core/Helper/ReflectionHelpers.cs +452 -452
- package/Runtime/Core/Helper/ReflectionHelpers.cs.meta +2 -2
- package/Runtime/Core/Helper/SceneHelper.cs +209 -213
- package/Runtime/Core/Helper/SpriteHelpers.cs +41 -41
- package/Runtime/Core/Helper/SpriteHelpers.cs.meta +11 -11
- package/Runtime/Core/Helper/StringInList.cs +31 -31
- package/Runtime/Core/Helper/StringInList.cs.meta +11 -11
- package/Runtime/Core/Helper/UnityMainThreadDispatcher.cs +82 -82
- package/Runtime/Core/Helper/WallMath.cs +166 -166
- package/Runtime/Core/Math/Line.cs +55 -55
- package/Runtime/Core/Math/Parabola.cs +47 -47
- package/Runtime/Core/Math/PointPolygonCheck.cs +36 -36
- package/Runtime/Core/Math/PointPolygonCheck.cs.meta +11 -11
- package/Runtime/Core/Math/Range.cs +92 -92
- package/Runtime/Core/Math/XXHash.cs +310 -310
- package/Runtime/Core/Math/XXHash.cs.meta +11 -11
- package/Runtime/Core/Model/Direction.cs +43 -43
- package/Runtime/Core/OneOf/FastOneOf.cs +152 -152
- package/Runtime/Core/OneOf/None.cs +4 -4
- package/Runtime/Core/Random/AbstractRandom.cs +585 -585
- package/Runtime/Core/Random/DotNetRandom.cs +54 -54
- package/Runtime/Core/Random/DotNetRandom.cs.meta +2 -2
- package/Runtime/Core/Random/IRandom.cs +161 -161
- package/Runtime/Core/Random/LinearCongruentialGenerator.cs +49 -49
- package/Runtime/Core/Random/NativePcgRandom.cs +97 -97
- package/Runtime/Core/Random/PRNG.cs +7 -7
- package/Runtime/Core/Random/PRNG.cs.meta +2 -2
- package/Runtime/Core/Random/PcgRandom.cs +149 -149
- package/Runtime/Core/Random/PerlinNoise.cs +369 -369
- package/Runtime/Core/Random/PerlinNoise.cs.meta +2 -2
- package/Runtime/Core/Random/RandomState.cs +131 -131
- package/Runtime/Core/Random/RandomUtilities.cs +26 -26
- package/Runtime/Core/Random/RandomUtilities.cs.meta +11 -11
- package/Runtime/Core/Random/RomuDuo.cs +116 -116
- package/Runtime/Core/Random/RomuDuo.cs.meta +2 -2
- package/Runtime/Core/Random/SplitMix64.cs +94 -94
- package/Runtime/Core/Random/SplitMix64.cs.meta +2 -2
- package/Runtime/Core/Random/SquirrelRandom.cs +84 -84
- package/Runtime/Core/Random/SystemRandom.cs +162 -162
- package/Runtime/Core/Random/ThreadLocalRandom.cs +12 -12
- package/Runtime/Core/Random/UnityRandom.cs +57 -57
- package/Runtime/Core/Random/UnityRandom.cs.meta +11 -11
- package/Runtime/Core/Random/WyRandom.cs +121 -121
- package/Runtime/Core/Random/WyRandom.cs.meta +2 -2
- package/Runtime/Core/Random/XorShiftRandom.cs +52 -52
- package/Runtime/Core/Random/XorShiftRandom.cs.meta +11 -11
- package/Runtime/Core/Random/XorShiroRandom.cs +119 -119
- package/Runtime/Core/Random/XorShiroRandom.cs.meta +2 -2
- package/Runtime/Core/Serialization/JsonConverters/ColorConverter.cs +88 -88
- package/Runtime/Core/Serialization/JsonConverters/GameObjectConverter.cs +37 -37
- package/Runtime/Core/Serialization/JsonConverters/Matrix4x4Converter.cs +218 -218
- package/Runtime/Core/Serialization/JsonConverters/TypeConverter.cs +28 -28
- package/Runtime/Core/Serialization/JsonConverters/Vector2Converter.cs +74 -74
- package/Runtime/Core/Serialization/JsonConverters/Vector3Converter.cs +81 -81
- package/Runtime/Core/Serialization/JsonConverters/Vector4Converter.cs +88 -88
- package/Runtime/Core/Serialization/Serializer.cs +195 -195
- package/Runtime/Core/Threading/SingleThreadedThreadPool.cs +113 -113
- package/Runtime/Protobuf-Net/System.Buffers.dll.meta +33 -33
- package/Runtime/Protobuf-Net/System.Collections.Immutable.dll.meta +33 -33
- package/Runtime/Protobuf-Net/System.Collections.Immutable.xml +5379 -5379
- package/Runtime/Protobuf-Net/System.Collections.Immutable.xml.meta +7 -7
- package/Runtime/Protobuf-Net/System.Numerics.Vectors.dll.meta +33 -33
- package/Runtime/Protobuf-Net/System.Runtime.CompilerServices.Unsafe.dll.meta +33 -33
- package/Runtime/Protobuf-Net/System.Runtime.CompilerServices.Unsafe.xml +290 -290
- package/Runtime/Protobuf-Net/System.Runtime.CompilerServices.Unsafe.xml.meta +7 -7
- package/Runtime/Protobuf-Net/protobuf-net.Core.dll.meta +33 -33
- package/Runtime/Protobuf-Net/protobuf-net.dll.meta +33 -33
- package/Runtime/UI/LayeredImage.cs +364 -364
- package/Runtime/UI/LayeredImage.cs.meta +2 -2
- package/Runtime/UI.meta +2 -2
- package/Runtime/Utils/AnimationEventEqualityComparer.cs +161 -161
- package/Runtime/Utils/AnimatorEnumStateMachine.cs +88 -88
- package/Runtime/Utils/Buffers.cs +33 -33
- package/Runtime/Utils/CenterPointOffset.cs +30 -30
- package/Runtime/Utils/CenterPointOffset.cs.meta +2 -2
- package/Runtime/Utils/CircleLineRenderer.cs +134 -134
- package/Runtime/Utils/CoroutineHandler.cs +4 -4
- package/Runtime/Utils/CoroutineHandler.cs.meta +2 -2
- package/Runtime/Utils/DeferredDisposalResult.cs +23 -23
- package/Runtime/Utils/MatchColliderToSprite.cs +94 -94
- package/Runtime/Utils/MatchColliderToSprite.cs.meta +2 -2
- package/Runtime/Utils/Oscillator.cs +27 -27
- package/Runtime/Utils/RuntimeSingleton.cs +69 -69
- package/Runtime/Utils/RuntimeSingleton.cs.meta +11 -11
- package/Runtime/Utils/SetTextureImportData.cs +69 -69
- package/Runtime/Utils/SpriteRendererMetadata.cs +374 -374
- package/Runtime/Utils/SpriteRendererMetadata.cs.meta +2 -2
- package/Runtime/Utils/SpriteRendererSyncer.cs +100 -100
- package/Runtime/Utils/SpriteRendererSyncer.cs.meta +2 -2
- package/Runtime/Utils/TextureScale.cs +179 -179
- package/Runtime/Utils/TextureScale.cs.meta +2 -2
- package/Runtime/WallstopStudios.UnityHelpers.asmdef +13 -13
- package/Tests/Runtime/Attributes/ChildComponentTests.cs +81 -81
- package/Tests/Runtime/Attributes/Components/ExpectChildSpriteRenderers.cs +28 -28
- package/Tests/Runtime/Attributes/Components/ExpectParentSpriteRenderers.cs +28 -28
- package/Tests/Runtime/Attributes/ParentComponentTests.cs +68 -68
- package/Tests/Runtime/Components/RelationalComponentTesterComplex.cs +34 -34
- package/Tests/Runtime/Components/RelationalComponentTesterComplex.cs.meta +2 -2
- package/Tests/Runtime/Components/RelationalComponentsTesterSimple.cs +40 -40
- package/Tests/Runtime/Components.meta +2 -2
- package/Tests/Runtime/DataStructures/BalancedKDTreeTests.cs +14 -14
- package/Tests/Runtime/DataStructures/BalancedKDTreeTests.cs.meta +11 -11
- package/Tests/Runtime/DataStructures/CyclicBufferTests.cs +324 -324
- package/Tests/Runtime/DataStructures/QuadTreeTests.cs +14 -14
- package/Tests/Runtime/DataStructures/QuadTreeTests.cs.meta +11 -11
- package/Tests/Runtime/DataStructures/SpatialTreeTests.cs +130 -130
- package/Tests/Runtime/DataStructures/SpatialTreeTests.cs.meta +11 -11
- package/Tests/Runtime/DataStructures/UnbalancedKDTreeTests.cs +14 -14
- package/Tests/Runtime/DataStructures/UnbalancedKDTreeTests.cs.meta +11 -11
- package/Tests/Runtime/DataStructures.meta +8 -8
- package/Tests/Runtime/Extensions/DictionaryExtensionTests.cs +439 -439
- package/Tests/Runtime/Extensions/DictionaryExtensionTests.cs.meta +2 -2
- package/Tests/Runtime/Extensions/EnumExtensionTests.cs +128 -128
- package/Tests/Runtime/Extensions/EnumExtensionTests.cs.meta +2 -2
- package/Tests/Runtime/Extensions/IListExtensionTests.cs +104 -104
- package/Tests/Runtime/Extensions/RandomExtensionTests.cs +27 -27
- package/Tests/Runtime/Extensions/RandomExtensionTests.cs.meta +2 -2
- package/Tests/Runtime/Extensions/StringExtensionTests.cs +31 -31
- package/Tests/Runtime/Extensions/StringExtensionTests.cs.meta +2 -2
- package/Tests/Runtime/Extensions.meta +2 -2
- package/Tests/Runtime/Helper/ArrayConverterTests.cs +19 -19
- package/Tests/Runtime/Helper/ArrayConverterTests.cs.meta +2 -2
- package/Tests/Runtime/Helper/FormattingHelperTests.cs +129 -129
- package/Tests/Runtime/Helper/FormattingHelperTests.cs.meta +2 -2
- package/Tests/Runtime/Helper/ObjectHelperTests.cs +402 -402
- package/Tests/Runtime/Helper/ObjectHelperTests.cs.meta +2 -2
- package/Tests/Runtime/Helper/ReflectionHelperTests.cs +536 -536
- package/Tests/Runtime/Helper/SceneHelperTests.cs +94 -94
- package/Tests/Runtime/Helper/WallMathTests.cs +233 -233
- package/Tests/Runtime/Helper/WallMathTests.cs.meta +2 -2
- package/Tests/Runtime/Helper.meta +2 -2
- package/Tests/Runtime/Performance/KDTreePerformanceTests.cs +14 -14
- package/Tests/Runtime/Performance/KDTreePerformanceTests.cs.meta +11 -11
- package/Tests/Runtime/Performance/QuadTreePerformanceTests.cs +14 -14
- package/Tests/Runtime/Performance/QuadTreePerformanceTests.cs.meta +11 -11
- package/Tests/Runtime/Performance/RandomPerformanceTests.cs +157 -157
- package/Tests/Runtime/Performance/RandomPerformanceTests.cs.meta +11 -11
- package/Tests/Runtime/Performance/RelationComponentPerformanceTests.cs +61 -61
- package/Tests/Runtime/Performance/RelationComponentPerformanceTests.cs.meta +2 -2
- package/Tests/Runtime/Performance/SpatialTreePerformanceTest.cs +154 -154
- package/Tests/Runtime/Performance/SpatialTreePerformanceTest.cs.meta +11 -11
- package/Tests/Runtime/Performance/UnbalancedKDTreeTests.cs +14 -14
- package/Tests/Runtime/Performance/UnbalancedKDTreeTests.cs.meta +11 -11
- package/Tests/Runtime/Performance.meta +8 -8
- package/Tests/Runtime/Random/DotNetRandomTests.cs +9 -9
- package/Tests/Runtime/Random/DotNetRandomTests.cs.meta +2 -2
- package/Tests/Runtime/Random/LinearCongruentialGeneratorTests.cs +12 -12
- package/Tests/Runtime/Random/PcgRandomTests.cs +9 -9
- package/Tests/Runtime/Random/PcgRandomTests.cs.meta +11 -11
- package/Tests/Runtime/Random/RandomTestBase.cs +787 -787
- package/Tests/Runtime/Random/RandomTestBase.cs.meta +11 -11
- package/Tests/Runtime/Random/RomuDuoRandomTests.cs +9 -9
- package/Tests/Runtime/Random/RomuDuoRandomTests.cs.meta +2 -2
- package/Tests/Runtime/Random/SplitMix64RandomTests.cs +9 -9
- package/Tests/Runtime/Random/SplitMix64RandomTests.cs.meta +2 -2
- package/Tests/Runtime/Random/SquirrelRandomTests.cs +14 -14
- package/Tests/Runtime/Random/SquirrelRandomTests.cs.meta +11 -11
- package/Tests/Runtime/Random/SystemRandomTests.cs +10 -10
- package/Tests/Runtime/Random/SystemRandomTests.cs.meta +11 -11
- package/Tests/Runtime/Random/UnityRandomTests.cs +9 -9
- package/Tests/Runtime/Random/UnityRandomTests.cs.meta +11 -11
- package/Tests/Runtime/Random/WyRandomTests.cs +9 -9
- package/Tests/Runtime/Random/WyRandomTests.cs.meta +2 -2
- package/Tests/Runtime/Random/XorShiftRandomTests.cs +9 -9
- package/Tests/Runtime/Random/XorShiftRandomTests.cs.meta +11 -11
- package/Tests/Runtime/Random/XorShiroRandomTests.cs +9 -9
- package/Tests/Runtime/Random/XorShiroRandomTests.cs.meta +2 -2
- package/Tests/Runtime/Random.meta +8 -8
- package/Tests/Runtime/Serialization/JsonSerializationTest.cs +156 -156
- package/Tests/Runtime/Serialization/JsonSerializationTest.cs.meta +2 -2
- package/Tests/Runtime/Serialization.meta +2 -2
- package/Tests/Runtime/Utils/SpriteRendererMetadataTests.cs +399 -399
- package/Tests/Runtime/Utils/SpriteRendererMetadataTests.cs.meta +2 -2
- package/Tests/Runtime/Utils.meta +2 -2
- package/Tests/Runtime/WallstopStudios.UnityHelpers.Tests.Runtime.asmdef +22 -22
- package/Tests/Runtime/WallstopStudios.UnityHelpers.Tests.Runtime.asmdef.meta +7 -7
- package/Tests/Runtime.meta +8 -8
- package/package.json +38 -38
|
@@ -1,346 +1,346 @@
|
|
|
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
|
-
Comparison<RTreeNode<T>> comparison = Comparison;
|
|
292
|
-
while (!current.isTerminal)
|
|
293
|
-
{
|
|
294
|
-
childrenCopy.Clear();
|
|
295
|
-
foreach (RTreeNode<T> child in current.children)
|
|
296
|
-
{
|
|
297
|
-
childrenCopy.Add(child);
|
|
298
|
-
}
|
|
299
|
-
childrenCopy.Sort(comparison);
|
|
300
|
-
for (int i = childrenCopy.Count - 1; 0 <= i; --i)
|
|
301
|
-
{
|
|
302
|
-
stack.Push(childrenCopy[i]);
|
|
303
|
-
}
|
|
304
|
-
|
|
305
|
-
current = childrenCopy[0];
|
|
306
|
-
if (current.elements.Length <= count)
|
|
307
|
-
{
|
|
308
|
-
break;
|
|
309
|
-
}
|
|
310
|
-
}
|
|
311
|
-
|
|
312
|
-
while (nearestNeighborsSet.Count < count && stack.TryPop(out RTreeNode<T> selected))
|
|
313
|
-
{
|
|
314
|
-
foreach (T element in selected.elements)
|
|
315
|
-
{
|
|
316
|
-
_ = nearestNeighborsSet.Add(element);
|
|
317
|
-
}
|
|
318
|
-
}
|
|
319
|
-
|
|
320
|
-
foreach (T element in nearestNeighborsSet)
|
|
321
|
-
{
|
|
322
|
-
nearestNeighbors.Add(element);
|
|
323
|
-
}
|
|
324
|
-
if (count < nearestNeighbors.Count)
|
|
325
|
-
{
|
|
326
|
-
Vector2 localPosition = position;
|
|
327
|
-
nearestNeighbors.Sort(NearestComparison);
|
|
328
|
-
nearestNeighbors.RemoveRange(count, nearestNeighbors.Count - count);
|
|
329
|
-
|
|
330
|
-
int NearestComparison(T lhs, T rhs) =>
|
|
331
|
-
(
|
|
332
|
-
(Vector2)_elementTransformer(lhs).center - localPosition
|
|
333
|
-
).sqrMagnitude.CompareTo(
|
|
334
|
-
((Vector2)_elementTransformer(rhs).center - localPosition).sqrMagnitude
|
|
335
|
-
);
|
|
336
|
-
}
|
|
337
|
-
|
|
338
|
-
return;
|
|
339
|
-
|
|
340
|
-
int Comparison(RTreeNode<T> lhs, RTreeNode<T> rhs) =>
|
|
341
|
-
((Vector2)lhs.boundary.center - position).sqrMagnitude.CompareTo(
|
|
342
|
-
((Vector2)rhs.boundary.center - position).sqrMagnitude
|
|
343
|
-
);
|
|
344
|
-
}
|
|
345
|
-
}
|
|
346
|
-
}
|
|
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
|
+
Comparison<RTreeNode<T>> comparison = Comparison;
|
|
292
|
+
while (!current.isTerminal)
|
|
293
|
+
{
|
|
294
|
+
childrenCopy.Clear();
|
|
295
|
+
foreach (RTreeNode<T> child in current.children)
|
|
296
|
+
{
|
|
297
|
+
childrenCopy.Add(child);
|
|
298
|
+
}
|
|
299
|
+
childrenCopy.Sort(comparison);
|
|
300
|
+
for (int i = childrenCopy.Count - 1; 0 <= i; --i)
|
|
301
|
+
{
|
|
302
|
+
stack.Push(childrenCopy[i]);
|
|
303
|
+
}
|
|
304
|
+
|
|
305
|
+
current = childrenCopy[0];
|
|
306
|
+
if (current.elements.Length <= count)
|
|
307
|
+
{
|
|
308
|
+
break;
|
|
309
|
+
}
|
|
310
|
+
}
|
|
311
|
+
|
|
312
|
+
while (nearestNeighborsSet.Count < count && stack.TryPop(out RTreeNode<T> selected))
|
|
313
|
+
{
|
|
314
|
+
foreach (T element in selected.elements)
|
|
315
|
+
{
|
|
316
|
+
_ = nearestNeighborsSet.Add(element);
|
|
317
|
+
}
|
|
318
|
+
}
|
|
319
|
+
|
|
320
|
+
foreach (T element in nearestNeighborsSet)
|
|
321
|
+
{
|
|
322
|
+
nearestNeighbors.Add(element);
|
|
323
|
+
}
|
|
324
|
+
if (count < nearestNeighbors.Count)
|
|
325
|
+
{
|
|
326
|
+
Vector2 localPosition = position;
|
|
327
|
+
nearestNeighbors.Sort(NearestComparison);
|
|
328
|
+
nearestNeighbors.RemoveRange(count, nearestNeighbors.Count - count);
|
|
329
|
+
|
|
330
|
+
int NearestComparison(T lhs, T rhs) =>
|
|
331
|
+
(
|
|
332
|
+
(Vector2)_elementTransformer(lhs).center - localPosition
|
|
333
|
+
).sqrMagnitude.CompareTo(
|
|
334
|
+
((Vector2)_elementTransformer(rhs).center - localPosition).sqrMagnitude
|
|
335
|
+
);
|
|
336
|
+
}
|
|
337
|
+
|
|
338
|
+
return;
|
|
339
|
+
|
|
340
|
+
int Comparison(RTreeNode<T> lhs, RTreeNode<T> rhs) =>
|
|
341
|
+
((Vector2)lhs.boundary.center - position).sqrMagnitude.CompareTo(
|
|
342
|
+
((Vector2)rhs.boundary.center - position).sqrMagnitude
|
|
343
|
+
);
|
|
344
|
+
}
|
|
345
|
+
}
|
|
346
|
+
}
|