2016-02-14 19:55:40 +00:00
|
|
|
/********************************************************************************
|
|
|
|
* ReactPhysics3D physics library, http://www.reactphysics3d.com *
|
2016-04-11 18:15:20 +00:00
|
|
|
* Copyright (c) 2010-2016 Daniel Chappuis *
|
2016-02-14 19:55:40 +00:00
|
|
|
*********************************************************************************
|
|
|
|
* *
|
|
|
|
* This software is provided 'as-is', without any express or implied warranty. *
|
|
|
|
* In no event will the authors be held liable for any damages arising from the *
|
|
|
|
* use of this software. *
|
|
|
|
* *
|
|
|
|
* Permission is granted to anyone to use this software for any purpose, *
|
|
|
|
* including commercial applications, and to alter it and redistribute it *
|
|
|
|
* freely, subject to the following restrictions: *
|
|
|
|
* *
|
|
|
|
* 1. The origin of this software must not be misrepresented; you must not claim *
|
|
|
|
* that you wrote the original software. If you use this software in a *
|
|
|
|
* product, an acknowledgment in the product documentation would be *
|
|
|
|
* appreciated but is not required. *
|
|
|
|
* *
|
|
|
|
* 2. Altered source versions must be plainly marked as such, and must not be *
|
|
|
|
* misrepresented as being the original software. *
|
|
|
|
* *
|
|
|
|
* 3. This notice may not be removed or altered from any source distribution. *
|
|
|
|
* *
|
|
|
|
********************************************************************************/
|
|
|
|
|
|
|
|
#ifndef TEST_DYNAMIC_AABB_TREE_H
|
|
|
|
#define TEST_DYNAMIC_AABB_TREE_H
|
|
|
|
|
|
|
|
// Libraries
|
|
|
|
#include "Test.h"
|
2020-03-18 06:28:34 +00:00
|
|
|
#include <reactphysics3d/collision/broadphase/DynamicAABBTree.h>
|
|
|
|
#include <reactphysics3d/memory/MemoryManager.h>
|
|
|
|
#include <reactphysics3d/engine/PhysicsCommon.h>
|
|
|
|
#include <reactphysics3d/utils/Profiler.h>
|
2019-08-04 21:24:48 +00:00
|
|
|
#include <vector>
|
2016-02-14 19:55:40 +00:00
|
|
|
|
|
|
|
/// Reactphysics3D namespace
|
|
|
|
namespace reactphysics3d {
|
|
|
|
|
2016-03-30 05:10:15 +00:00
|
|
|
class DynamicTreeRaycastCallback : public DynamicAABBTreeRaycastCallback {
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
std::vector<int> mHitNodes;
|
|
|
|
|
|
|
|
// Called when the AABB of a leaf node is hit by a ray
|
2016-07-11 06:33:24 +00:00
|
|
|
virtual decimal raycastBroadPhaseShape(int32 nodeId, const Ray& ray) override {
|
2016-03-30 05:10:15 +00:00
|
|
|
mHitNodes.push_back(nodeId);
|
|
|
|
return 1.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void reset() {
|
|
|
|
mHitNodes.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isHit(int nodeId) const {
|
|
|
|
return std::find(mHitNodes.begin(), mHitNodes.end(), nodeId) != mHitNodes.end();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-12-12 06:13:27 +00:00
|
|
|
class DefaultTestTreeAllocator : public MemoryAllocator {
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
/// Destructor
|
|
|
|
virtual ~DefaultTestTreeAllocator() override = default;
|
|
|
|
|
|
|
|
/// Assignment operator
|
|
|
|
DefaultTestTreeAllocator& operator=(DefaultTestTreeAllocator& allocator) = default;
|
|
|
|
|
|
|
|
/// Allocate memory of a given size (in bytes) and return a pointer to the
|
|
|
|
/// allocated memory.
|
|
|
|
virtual void* allocate(size_t size) override {
|
|
|
|
|
|
|
|
return malloc(size);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Release previously allocated memory.
|
|
|
|
virtual void release(void* pointer, size_t size) override {
|
|
|
|
free(pointer);
|
|
|
|
}
|
|
|
|
};
|
2019-08-04 21:24:48 +00:00
|
|
|
|
2016-02-14 19:55:40 +00:00
|
|
|
// Class TestDynamicAABBTree
|
|
|
|
/**
|
|
|
|
* Unit test for the dynamic AABB tree
|
|
|
|
*/
|
|
|
|
class TestDynamicAABBTree : public Test {
|
|
|
|
|
|
|
|
private :
|
|
|
|
|
|
|
|
// ---------- Atributes ---------- //
|
|
|
|
|
2019-12-12 06:13:27 +00:00
|
|
|
DefaultTestTreeAllocator mAllocator;
|
|
|
|
|
2016-03-30 05:10:15 +00:00
|
|
|
DynamicTreeRaycastCallback mRaycastCallback;
|
|
|
|
|
2020-04-27 18:38:12 +00:00
|
|
|
PhysicsCommon mPhysicsCommon;
|
|
|
|
|
2020-05-11 10:38:59 +00:00
|
|
|
#ifdef IS_RP3D_PROFILING_ENABLED
|
|
|
|
|
2020-04-27 18:38:12 +00:00
|
|
|
Profiler* mProfiler;
|
|
|
|
#endif
|
|
|
|
|
2016-02-14 19:55:40 +00:00
|
|
|
public :
|
|
|
|
|
|
|
|
// ---------- Methods ---------- //
|
|
|
|
|
|
|
|
/// Constructor
|
2016-03-30 05:10:15 +00:00
|
|
|
TestDynamicAABBTree(const std::string& name): Test(name) {
|
|
|
|
|
2020-05-11 10:38:59 +00:00
|
|
|
#ifdef IS_RP3D_PROFILING_ENABLED
|
|
|
|
|
2020-05-11 09:13:26 +00:00
|
|
|
mProfiler = new Profiler();
|
2020-04-27 18:38:12 +00:00
|
|
|
#endif
|
2020-05-11 09:13:26 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Constructor
|
|
|
|
~TestDynamicAABBTree() {
|
|
|
|
|
2020-05-11 10:38:59 +00:00
|
|
|
#ifdef IS_RP3D_PROFILING_ENABLED
|
|
|
|
|
2020-05-11 09:13:26 +00:00
|
|
|
delete mProfiler;
|
|
|
|
#endif
|
|
|
|
|
2016-03-30 05:10:15 +00:00
|
|
|
}
|
2016-02-14 19:55:40 +00:00
|
|
|
|
2019-08-04 21:24:48 +00:00
|
|
|
bool isOverlapping(int nodeId, const List<int>& overlappingNodes) const {
|
|
|
|
return std::find(overlappingNodes.begin(), overlappingNodes.end(), nodeId) != overlappingNodes.end();
|
|
|
|
}
|
|
|
|
|
2016-02-14 19:55:40 +00:00
|
|
|
/// Run the tests
|
|
|
|
void run() {
|
|
|
|
|
2016-03-30 05:10:15 +00:00
|
|
|
testBasicsMethods();
|
|
|
|
testOverlapping();
|
|
|
|
testRaycast();
|
|
|
|
|
2016-02-14 19:55:40 +00:00
|
|
|
}
|
|
|
|
|
2016-03-30 05:10:15 +00:00
|
|
|
void testBasicsMethods() {
|
|
|
|
|
2016-04-01 23:51:41 +00:00
|
|
|
// ------------ Create tree ---------- //
|
|
|
|
|
|
|
|
// Dynamic AABB Tree
|
2019-12-12 06:13:27 +00:00
|
|
|
DynamicAABBTree tree(mAllocator);
|
2020-05-11 10:38:59 +00:00
|
|
|
#ifdef IS_RP3D_PROFILING_ENABLED
|
|
|
|
|
2020-04-27 18:38:12 +00:00
|
|
|
tree.setProfiler(mProfiler);
|
|
|
|
#endif
|
2017-12-27 19:53:09 +00:00
|
|
|
|
2016-04-01 23:51:41 +00:00
|
|
|
int object1Data = 56;
|
|
|
|
int object2Data = 23;
|
|
|
|
int object3Data = 13;
|
|
|
|
int object4Data = 7;
|
|
|
|
|
|
|
|
// First object
|
|
|
|
AABB aabb1 = AABB(Vector3(-6, 4, -3), Vector3(4, 8, 3));
|
|
|
|
int object1Id = tree.addObject(aabb1, &object1Data);
|
|
|
|
|
|
|
|
// Second object
|
|
|
|
AABB aabb2 = AABB(Vector3(5, 2, -3), Vector3(10, 7, 3));
|
|
|
|
int object2Id = tree.addObject(aabb2, &object2Data);
|
|
|
|
|
|
|
|
// Third object
|
|
|
|
AABB aabb3 = AABB(Vector3(-5, 1, -3), Vector3(-2, 3, 3));
|
|
|
|
int object3Id = tree.addObject(aabb3, &object3Data);
|
|
|
|
|
|
|
|
// Fourth object
|
|
|
|
AABB aabb4 = AABB(Vector3(0, -4, -3), Vector3(3, -2, 3));
|
|
|
|
int object4Id = tree.addObject(aabb4, &object4Data);
|
|
|
|
|
|
|
|
// ----------- Tests ----------- //
|
|
|
|
|
|
|
|
// Test root AABB
|
|
|
|
AABB rootAABB = tree.getRootAABB();
|
2018-08-05 14:10:13 +00:00
|
|
|
rp3d_test(rootAABB.getMin().x == -6);
|
|
|
|
rp3d_test(rootAABB.getMin().y == -4);
|
|
|
|
rp3d_test(rootAABB.getMin().z == -3);
|
|
|
|
rp3d_test(rootAABB.getMax().x == 10);
|
|
|
|
rp3d_test(rootAABB.getMax().y == 8);
|
|
|
|
rp3d_test(rootAABB.getMax().z == 3);
|
2016-04-01 23:51:41 +00:00
|
|
|
|
2016-03-30 05:10:15 +00:00
|
|
|
// Test data stored at the nodes of the tree
|
2018-08-05 14:10:13 +00:00
|
|
|
rp3d_test(*(int*)(tree.getNodeDataPointer(object1Id)) == object1Data);
|
|
|
|
rp3d_test(*(int*)(tree.getNodeDataPointer(object2Id)) == object2Data);
|
|
|
|
rp3d_test(*(int*)(tree.getNodeDataPointer(object3Id)) == object3Data);
|
|
|
|
rp3d_test(*(int*)(tree.getNodeDataPointer(object4Id)) == object4Data);
|
2016-03-30 05:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void testOverlapping() {
|
|
|
|
|
2016-04-01 23:51:41 +00:00
|
|
|
// ------------- Create tree ----------- //
|
|
|
|
|
|
|
|
// Dynamic AABB Tree
|
2019-12-12 06:13:27 +00:00
|
|
|
DynamicAABBTree tree(mAllocator);
|
2020-05-11 10:38:59 +00:00
|
|
|
#ifdef IS_RP3D_PROFILING_ENABLED
|
|
|
|
|
2020-04-27 18:38:12 +00:00
|
|
|
tree.setProfiler(mProfiler);
|
|
|
|
#endif
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
int object1Data = 56;
|
|
|
|
int object2Data = 23;
|
|
|
|
int object3Data = 13;
|
|
|
|
int object4Data = 7;
|
|
|
|
|
|
|
|
// First object
|
|
|
|
AABB aabb1 = AABB(Vector3(-6, 4, -3), Vector3(4, 8, 3));
|
|
|
|
int object1Id = tree.addObject(aabb1, &object1Data);
|
|
|
|
|
|
|
|
// Second object
|
|
|
|
AABB aabb2 = AABB(Vector3(5, 2, -3), Vector3(10, 7, 3));
|
|
|
|
int object2Id = tree.addObject(aabb2, &object2Data);
|
|
|
|
|
|
|
|
// Third object
|
|
|
|
AABB aabb3 = AABB(Vector3(-5, 1, -3), Vector3(-2, 3, 3));
|
|
|
|
int object3Id = tree.addObject(aabb3, &object3Data);
|
|
|
|
|
|
|
|
// Fourth object
|
|
|
|
AABB aabb4 = AABB(Vector3(0, -4, -3), Vector3(3, -2, 3));
|
|
|
|
int object4Id = tree.addObject(aabb4, &object4Data);
|
|
|
|
|
|
|
|
// ---------- Tests ---------- //
|
|
|
|
|
2019-08-04 21:24:48 +00:00
|
|
|
List<int> overlappingNodes(mAllocator);
|
|
|
|
|
2016-03-30 05:10:15 +00:00
|
|
|
// AABB overlapping nothing
|
2019-08-04 21:24:48 +00:00
|
|
|
overlappingNodes.clear();
|
|
|
|
tree.reportAllShapesOverlappingWithAABB(AABB(Vector3(-10, 12, -4), Vector3(10, 50, 4)), overlappingNodes);
|
|
|
|
rp3d_test(!isOverlapping(object1Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object2Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object3Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object4Id, overlappingNodes));
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// AABB overlapping everything
|
2019-08-04 21:24:48 +00:00
|
|
|
overlappingNodes.clear();
|
|
|
|
tree.reportAllShapesOverlappingWithAABB(AABB(Vector3(-15, -15, -4), Vector3(15, 15, 4)), overlappingNodes);
|
|
|
|
rp3d_test(isOverlapping(object1Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object2Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object3Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object4Id, overlappingNodes));
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// AABB overlapping object 1 and 3
|
2019-08-04 21:24:48 +00:00
|
|
|
overlappingNodes.clear();
|
|
|
|
tree.reportAllShapesOverlappingWithAABB(AABB(Vector3(-4, 2, -4), Vector3(-1, 7, 4)), overlappingNodes);
|
|
|
|
rp3d_test(isOverlapping(object1Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object2Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object3Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object4Id, overlappingNodes));
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// AABB overlapping object 3 and 4
|
2019-08-04 21:24:48 +00:00
|
|
|
overlappingNodes.clear();
|
|
|
|
tree.reportAllShapesOverlappingWithAABB(AABB(Vector3(-6, -5, -2), Vector3(2, 2, 0)), overlappingNodes);
|
|
|
|
rp3d_test(!isOverlapping(object1Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object2Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object3Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object4Id, overlappingNodes));
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// AABB overlapping object 2
|
2019-08-04 21:24:48 +00:00
|
|
|
overlappingNodes.clear();
|
|
|
|
tree.reportAllShapesOverlappingWithAABB(AABB(Vector3(5, -10, -2), Vector3(7, 10, 9)), overlappingNodes);
|
|
|
|
rp3d_test(!isOverlapping(object1Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object2Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object3Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object4Id, overlappingNodes));
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// ---- Update the object AABBs with the initial AABBs (no reinsertion) ----- //
|
|
|
|
|
2020-02-14 16:08:02 +00:00
|
|
|
tree.updateObject(object1Id, aabb1);
|
|
|
|
tree.updateObject(object2Id, aabb2);
|
|
|
|
tree.updateObject(object3Id, aabb3);
|
|
|
|
tree.updateObject(object4Id, aabb4);
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// AABB overlapping nothing
|
2019-08-04 21:24:48 +00:00
|
|
|
overlappingNodes.clear();
|
|
|
|
tree.reportAllShapesOverlappingWithAABB(AABB(Vector3(-10, 12, -4), Vector3(10, 50, 4)), overlappingNodes);
|
|
|
|
rp3d_test(!isOverlapping(object1Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object2Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object3Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object4Id, overlappingNodes));
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// AABB overlapping everything
|
2019-08-04 21:24:48 +00:00
|
|
|
overlappingNodes.clear();
|
|
|
|
tree.reportAllShapesOverlappingWithAABB(AABB(Vector3(-15, -15, -4), Vector3(15, 15, 4)), overlappingNodes);
|
|
|
|
rp3d_test(isOverlapping(object1Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object2Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object3Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object4Id, overlappingNodes));
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// AABB overlapping object 1 and 3
|
2019-08-04 21:24:48 +00:00
|
|
|
overlappingNodes.clear();
|
|
|
|
tree.reportAllShapesOverlappingWithAABB(AABB(Vector3(-4, 2, -4), Vector3(-1, 7, 4)), overlappingNodes);
|
|
|
|
rp3d_test(isOverlapping(object1Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object2Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object3Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object4Id, overlappingNodes));
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// AABB overlapping object 3 and 4
|
2019-08-04 21:24:48 +00:00
|
|
|
overlappingNodes.clear();
|
|
|
|
tree.reportAllShapesOverlappingWithAABB(AABB(Vector3(-6, -5, -2), Vector3(2, 2, 0)), overlappingNodes);
|
|
|
|
rp3d_test(!isOverlapping(object1Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object2Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object3Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object4Id, overlappingNodes));
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// AABB overlapping object 2
|
2019-08-04 21:24:48 +00:00
|
|
|
overlappingNodes.clear();
|
|
|
|
tree.reportAllShapesOverlappingWithAABB(AABB(Vector3(5, -10, -2), Vector3(7, 10, 9)), overlappingNodes);
|
|
|
|
rp3d_test(!isOverlapping(object1Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object2Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object3Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object4Id, overlappingNodes));
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// ---- Update the object AABBs with the initial AABBs (with reinsertion) ----- //
|
|
|
|
|
2020-02-14 16:08:02 +00:00
|
|
|
tree.updateObject(object1Id, aabb1);
|
|
|
|
tree.updateObject(object2Id, aabb2);
|
|
|
|
tree.updateObject(object3Id, aabb3);
|
|
|
|
tree.updateObject(object4Id, aabb4);
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// AABB overlapping nothing
|
2019-08-04 21:24:48 +00:00
|
|
|
overlappingNodes.clear();
|
|
|
|
tree.reportAllShapesOverlappingWithAABB(AABB(Vector3(-10, 12, -4), Vector3(10, 50, 4)), overlappingNodes);
|
|
|
|
rp3d_test(!isOverlapping(object1Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object2Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object3Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object4Id, overlappingNodes));
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// AABB overlapping everything
|
2019-08-04 21:24:48 +00:00
|
|
|
overlappingNodes.clear();
|
|
|
|
tree.reportAllShapesOverlappingWithAABB(AABB(Vector3(-15, -15, -4), Vector3(15, 15, 4)), overlappingNodes);
|
|
|
|
rp3d_test(isOverlapping(object1Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object2Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object3Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object4Id, overlappingNodes));
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// AABB overlapping object 1 and 3
|
2019-08-04 21:24:48 +00:00
|
|
|
overlappingNodes.clear();
|
|
|
|
tree.reportAllShapesOverlappingWithAABB(AABB(Vector3(-4, 2, -4), Vector3(-1, 7, 4)), overlappingNodes);
|
|
|
|
rp3d_test(isOverlapping(object1Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object2Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object3Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object4Id, overlappingNodes));
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// AABB overlapping object 3 and 4
|
2019-08-04 21:24:48 +00:00
|
|
|
overlappingNodes.clear();
|
|
|
|
tree.reportAllShapesOverlappingWithAABB(AABB(Vector3(-6, -5, -2), Vector3(2, 2, 0)), overlappingNodes);
|
|
|
|
rp3d_test(!isOverlapping(object1Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object2Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object3Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object4Id, overlappingNodes));
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// AABB overlapping object 2
|
2019-08-04 21:24:48 +00:00
|
|
|
overlappingNodes.clear();
|
|
|
|
tree.reportAllShapesOverlappingWithAABB(AABB(Vector3(5, -10, -2), Vector3(7, 10, 9)), overlappingNodes);
|
|
|
|
rp3d_test(!isOverlapping(object1Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object2Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object3Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object4Id, overlappingNodes));
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
// ---- Move objects 2 and 3 ----- //
|
|
|
|
|
|
|
|
AABB newAABB2(Vector3(-7, 10, -3), Vector3(1, 13, 3));
|
2020-02-14 16:08:02 +00:00
|
|
|
tree.updateObject(object2Id, newAABB2);
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
AABB newAABB3(Vector3(7, -6, -3), Vector3(9, 1, 3));
|
2020-02-14 16:08:02 +00:00
|
|
|
tree.updateObject(object3Id, newAABB3);
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
// AABB overlapping object 3
|
2019-08-04 21:24:48 +00:00
|
|
|
overlappingNodes.clear();
|
|
|
|
tree.reportAllShapesOverlappingWithAABB(AABB(Vector3(6, -10, -2), Vector3(8, 5, 3)), overlappingNodes);
|
|
|
|
rp3d_test(!isOverlapping(object1Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object2Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object3Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object4Id, overlappingNodes));
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
// AABB overlapping objects 1, 2
|
2019-08-04 21:24:48 +00:00
|
|
|
overlappingNodes.clear();
|
|
|
|
tree.reportAllShapesOverlappingWithAABB(AABB(Vector3(-8, 5, -3), Vector3(-2, 11, 3)), overlappingNodes);
|
|
|
|
rp3d_test(isOverlapping(object1Id, overlappingNodes));
|
|
|
|
rp3d_test(isOverlapping(object2Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object3Id, overlappingNodes));
|
|
|
|
rp3d_test(!isOverlapping(object4Id, overlappingNodes));
|
2016-04-01 23:51:41 +00:00
|
|
|
|
2016-03-30 05:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void testRaycast() {
|
|
|
|
|
2016-04-01 23:51:41 +00:00
|
|
|
// ------------- Create tree ----------- //
|
|
|
|
|
|
|
|
// Dynamic AABB Tree
|
2019-12-12 06:13:27 +00:00
|
|
|
DynamicAABBTree tree(mAllocator);
|
2020-05-11 10:38:59 +00:00
|
|
|
#ifdef IS_RP3D_PROFILING_ENABLED
|
|
|
|
|
2020-04-27 18:38:12 +00:00
|
|
|
tree.setProfiler(mProfiler);
|
|
|
|
#endif
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
int object1Data = 56;
|
|
|
|
int object2Data = 23;
|
|
|
|
int object3Data = 13;
|
|
|
|
int object4Data = 7;
|
|
|
|
|
|
|
|
// First object
|
|
|
|
AABB aabb1 = AABB(Vector3(-6, 4, -3), Vector3(4, 8, 3));
|
|
|
|
int object1Id = tree.addObject(aabb1, &object1Data);
|
|
|
|
|
|
|
|
// Second object
|
|
|
|
AABB aabb2 = AABB(Vector3(5, 2, -3), Vector3(10, 7, 3));
|
|
|
|
int object2Id = tree.addObject(aabb2, &object2Data);
|
|
|
|
|
|
|
|
// Third object
|
|
|
|
AABB aabb3 = AABB(Vector3(-5, 1, -3), Vector3(-2, 3, 3));
|
|
|
|
int object3Id = tree.addObject(aabb3, &object3Data);
|
|
|
|
|
|
|
|
// Fourth object
|
|
|
|
AABB aabb4 = AABB(Vector3(0, -4, -3), Vector3(3, -2, 3));
|
|
|
|
int object4Id = tree.addObject(aabb4, &object4Data);
|
|
|
|
|
|
|
|
// ---------- Tests ---------- //
|
|
|
|
|
2016-03-30 05:10:15 +00:00
|
|
|
// Ray with no hits
|
|
|
|
mRaycastCallback.reset();
|
|
|
|
Ray ray1(Vector3(4.5, -10, -5), Vector3(4.5, 10, -5));
|
2016-04-01 23:51:41 +00:00
|
|
|
tree.raycast(ray1, mRaycastCallback);
|
2018-08-05 14:10:13 +00:00
|
|
|
rp3d_test(!mRaycastCallback.isHit(object1Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object2Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object3Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object4Id));
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// Ray that hits object 1
|
|
|
|
mRaycastCallback.reset();
|
|
|
|
Ray ray2(Vector3(-1, -20, -2), Vector3(-1, 20, -2));
|
2016-04-01 23:51:41 +00:00
|
|
|
tree.raycast(ray2, mRaycastCallback);
|
2018-08-05 14:10:13 +00:00
|
|
|
rp3d_test(mRaycastCallback.isHit(object1Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object2Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object3Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object4Id));
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// Ray that hits object 1 and 2
|
|
|
|
mRaycastCallback.reset();
|
|
|
|
Ray ray3(Vector3(-7, 6, -2), Vector3(8, 6, -2));
|
2016-04-01 23:51:41 +00:00
|
|
|
tree.raycast(ray3, mRaycastCallback);
|
2018-08-05 14:10:13 +00:00
|
|
|
rp3d_test(mRaycastCallback.isHit(object1Id));
|
|
|
|
rp3d_test(mRaycastCallback.isHit(object2Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object3Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object4Id));
|
2016-03-30 05:10:15 +00:00
|
|
|
|
|
|
|
// Ray that hits object 3
|
|
|
|
mRaycastCallback.reset();
|
|
|
|
Ray ray4(Vector3(-7, 2, 0), Vector3(-1, 2, 0));
|
2016-04-01 23:51:41 +00:00
|
|
|
tree.raycast(ray4, mRaycastCallback);
|
2018-08-05 14:10:13 +00:00
|
|
|
rp3d_test(!mRaycastCallback.isHit(object1Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object2Id));
|
|
|
|
rp3d_test(mRaycastCallback.isHit(object3Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object4Id));
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
// ---- Update the object AABBs with the initial AABBs (no reinsertion) ----- //
|
|
|
|
|
2020-02-14 16:08:02 +00:00
|
|
|
tree.updateObject(object1Id, aabb1);
|
|
|
|
tree.updateObject(object2Id, aabb2);
|
|
|
|
tree.updateObject(object3Id, aabb3);
|
|
|
|
tree.updateObject(object4Id, aabb4);
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
// Ray with no hits
|
|
|
|
mRaycastCallback.reset();
|
|
|
|
tree.raycast(ray1, mRaycastCallback);
|
2018-08-05 14:10:13 +00:00
|
|
|
rp3d_test(!mRaycastCallback.isHit(object1Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object2Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object3Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object4Id));
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
// Ray that hits object 1
|
|
|
|
mRaycastCallback.reset();
|
|
|
|
tree.raycast(ray2, mRaycastCallback);
|
2018-08-05 14:10:13 +00:00
|
|
|
rp3d_test(mRaycastCallback.isHit(object1Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object2Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object3Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object4Id));
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
// Ray that hits object 1 and 2
|
|
|
|
mRaycastCallback.reset();
|
|
|
|
tree.raycast(ray3, mRaycastCallback);
|
2018-08-05 14:10:13 +00:00
|
|
|
rp3d_test(mRaycastCallback.isHit(object1Id));
|
|
|
|
rp3d_test(mRaycastCallback.isHit(object2Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object3Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object4Id));
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
// Ray that hits object 3
|
|
|
|
mRaycastCallback.reset();
|
|
|
|
tree.raycast(ray4, mRaycastCallback);
|
2018-08-05 14:10:13 +00:00
|
|
|
rp3d_test(!mRaycastCallback.isHit(object1Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object2Id));
|
|
|
|
rp3d_test(mRaycastCallback.isHit(object3Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object4Id));
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
// ---- Update the object AABBs with the initial AABBs (with reinsertion) ----- //
|
|
|
|
|
2020-02-14 16:08:02 +00:00
|
|
|
tree.updateObject(object1Id, aabb1);
|
|
|
|
tree.updateObject(object2Id, aabb2);
|
|
|
|
tree.updateObject(object3Id, aabb3);
|
|
|
|
tree.updateObject(object4Id, aabb4);
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
// Ray with no hits
|
|
|
|
mRaycastCallback.reset();
|
|
|
|
tree.raycast(ray1, mRaycastCallback);
|
2018-08-05 14:10:13 +00:00
|
|
|
rp3d_test(!mRaycastCallback.isHit(object1Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object2Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object3Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object4Id));
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
// Ray that hits object 1
|
|
|
|
mRaycastCallback.reset();
|
|
|
|
tree.raycast(ray2, mRaycastCallback);
|
2018-08-05 14:10:13 +00:00
|
|
|
rp3d_test(mRaycastCallback.isHit(object1Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object2Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object3Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object4Id));
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
// Ray that hits object 1 and 2
|
|
|
|
mRaycastCallback.reset();
|
|
|
|
tree.raycast(ray3, mRaycastCallback);
|
2018-08-05 14:10:13 +00:00
|
|
|
rp3d_test(mRaycastCallback.isHit(object1Id));
|
|
|
|
rp3d_test(mRaycastCallback.isHit(object2Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object3Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object4Id));
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
// Ray that hits object 3
|
|
|
|
mRaycastCallback.reset();
|
|
|
|
tree.raycast(ray4, mRaycastCallback);
|
2018-08-05 14:10:13 +00:00
|
|
|
rp3d_test(!mRaycastCallback.isHit(object1Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object2Id));
|
|
|
|
rp3d_test(mRaycastCallback.isHit(object3Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object4Id));
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
// ---- Move objects 2 and 3 ----- //
|
|
|
|
|
|
|
|
AABB newAABB2(Vector3(-7, 10, -3), Vector3(1, 13, 3));
|
2020-02-14 16:08:02 +00:00
|
|
|
tree.updateObject(object2Id, newAABB2);
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
AABB newAABB3(Vector3(7, -6, -3), Vector3(9, 1, 3));
|
2020-02-14 16:08:02 +00:00
|
|
|
tree.updateObject(object3Id, newAABB3);
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
// Ray that hits object 1, 2
|
|
|
|
Ray ray5(Vector3(-4, -5, 0), Vector3(-4, 12, 0));
|
|
|
|
mRaycastCallback.reset();
|
|
|
|
tree.raycast(ray5, mRaycastCallback);
|
2018-08-05 14:10:13 +00:00
|
|
|
rp3d_test(mRaycastCallback.isHit(object1Id));
|
|
|
|
rp3d_test(mRaycastCallback.isHit(object2Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object3Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object4Id));
|
2016-04-01 23:51:41 +00:00
|
|
|
|
|
|
|
// Ray that hits object 3 and 4
|
|
|
|
Ray ray6(Vector3(11, -3, 1), Vector3(-2, -3, 1));
|
|
|
|
mRaycastCallback.reset();
|
|
|
|
tree.raycast(ray6, mRaycastCallback);
|
2018-08-05 14:10:13 +00:00
|
|
|
rp3d_test(!mRaycastCallback.isHit(object1Id));
|
|
|
|
rp3d_test(!mRaycastCallback.isHit(object2Id));
|
|
|
|
rp3d_test(mRaycastCallback.isHit(object3Id));
|
|
|
|
rp3d_test(mRaycastCallback.isHit(object4Id));
|
2017-12-27 19:53:09 +00:00
|
|
|
|
2016-03-30 05:10:15 +00:00
|
|
|
}
|
2016-02-14 19:55:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|