1503 lines
74 KiB
C++
1503 lines
74 KiB
C++
/********************************************************************************
|
|
* ReactPhysics3D physics library, http://www.reactphysics3d.com *
|
|
* Copyright (c) 2010-2018 Daniel Chappuis *
|
|
*********************************************************************************
|
|
* *
|
|
* 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. *
|
|
* *
|
|
********************************************************************************/
|
|
|
|
// Libraries
|
|
#include "systems/CollisionDetectionSystem.h"
|
|
#include "engine/CollisionWorld.h"
|
|
#include "collision/OverlapCallback.h"
|
|
#include "collision/shapes/BoxShape.h"
|
|
#include "collision/shapes/ConcaveShape.h"
|
|
#include "collision/ContactManifoldInfo.h"
|
|
#include "constraint/ContactPoint.h"
|
|
#include "body/RigidBody.h"
|
|
#include "configuration.h"
|
|
#include "collision/CollisionCallback.h"
|
|
#include "collision/OverlapCallback.h"
|
|
#include "collision/narrowphase/NarrowPhaseInfoBatch.h"
|
|
#include "collision/ContactManifold.h"
|
|
#include "utils/Profiler.h"
|
|
#include "engine/EventListener.h"
|
|
#include "collision/RaycastInfo.h"
|
|
#include "engine/Islands.h"
|
|
#include "containers/Pair.h"
|
|
#include <cassert>
|
|
#include <iostream>
|
|
|
|
// We want to use the ReactPhysics3D namespace
|
|
using namespace reactphysics3d;
|
|
using namespace std;
|
|
|
|
|
|
// Constructor
|
|
CollisionDetectionSystem::CollisionDetectionSystem(CollisionWorld* world, ColliderComponents& collidersComponents, TransformComponents& transformComponents,
|
|
CollisionBodyComponents& collisionBodyComponents, RigidBodyComponents& rigidBodyComponents, MemoryManager& memoryManager)
|
|
: mMemoryManager(memoryManager), mCollidersComponents(collidersComponents),
|
|
mCollisionDispatch(mMemoryManager.getPoolAllocator()), mWorld(world),
|
|
mNoCollisionPairs(mMemoryManager.getPoolAllocator()),
|
|
mOverlappingPairs(mMemoryManager.getPoolAllocator(), mMemoryManager.getSingleFrameAllocator(), mCollidersComponents,
|
|
collisionBodyComponents, rigidBodyComponents, mNoCollisionPairs, mCollisionDispatch),
|
|
mBroadPhaseSystem(*this, mCollidersComponents, transformComponents, rigidBodyComponents),
|
|
mMapBroadPhaseIdToColliderEntity(memoryManager.getPoolAllocator()),
|
|
mNarrowPhaseInput(mMemoryManager.getSingleFrameAllocator(), mOverlappingPairs), mPotentialContactPoints(mMemoryManager.getSingleFrameAllocator()),
|
|
mPotentialContactManifolds(mMemoryManager.getSingleFrameAllocator()), mContactPairs1(mMemoryManager.getPoolAllocator()),
|
|
mContactPairs2(mMemoryManager.getPoolAllocator()), mPreviousContactPairs(&mContactPairs1), mCurrentContactPairs(&mContactPairs2),
|
|
mMapPairIdToContactPairIndex1(mMemoryManager.getPoolAllocator()), mMapPairIdToContactPairIndex2(mMemoryManager.getPoolAllocator()),
|
|
mPreviousMapPairIdToContactPairIndex(&mMapPairIdToContactPairIndex1), mCurrentMapPairIdToContactPairIndex(&mMapPairIdToContactPairIndex2),
|
|
mContactManifolds1(mMemoryManager.getPoolAllocator()), mContactManifolds2(mMemoryManager.getPoolAllocator()),
|
|
mPreviousContactManifolds(&mContactManifolds1), mCurrentContactManifolds(&mContactManifolds2),
|
|
mContactPoints1(mMemoryManager.getPoolAllocator()),
|
|
mContactPoints2(mMemoryManager.getPoolAllocator()), mPreviousContactPoints(&mContactPoints1),
|
|
mCurrentContactPoints(&mContactPoints2), mMapBodyToContactPairs(mMemoryManager.getSingleFrameAllocator()) {
|
|
|
|
#ifdef IS_PROFILING_ACTIVE
|
|
|
|
mProfiler = nullptr;
|
|
mCollisionDispatch.setProfiler(mProfiler);
|
|
mOverlappingPairs.setProfiler(mProfiler);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
// Compute the collision detection
|
|
void CollisionDetectionSystem::computeCollisionDetection() {
|
|
|
|
RP3D_PROFILE("CollisionDetectionSystem::computeCollisionDetection()", mProfiler);
|
|
|
|
// Compute the broad-phase collision detection
|
|
computeBroadPhase();
|
|
|
|
// Compute the middle-phase collision detection
|
|
computeMiddlePhase(mNarrowPhaseInput);
|
|
|
|
// Compute the narrow-phase collision detection
|
|
computeNarrowPhase();
|
|
}
|
|
|
|
// Compute the broad-phase collision detection
|
|
void CollisionDetectionSystem::computeBroadPhase() {
|
|
|
|
RP3D_PROFILE("CollisionDetectionSystem::computeBroadPhase()", mProfiler);
|
|
|
|
// Ask the broad-phase to compute all the shapes overlapping the shapes that
|
|
// have moved or have been added in the last frame. This call can only add new
|
|
// overlapping pairs in the collision detection.
|
|
List<Pair<int32, int32>> overlappingNodes(mMemoryManager.getPoolAllocator(), 32);
|
|
mBroadPhaseSystem.computeOverlappingPairs(mMemoryManager, overlappingNodes);
|
|
|
|
// Create new overlapping pairs if necessary
|
|
updateOverlappingPairs(overlappingNodes);
|
|
|
|
// Remove non overlapping pairs
|
|
removeNonOverlappingPairs();
|
|
}
|
|
|
|
// Remove pairs that are not overlapping anymore
|
|
void CollisionDetectionSystem::removeNonOverlappingPairs() {
|
|
|
|
RP3D_PROFILE("CollisionDetectionSystem::removeNonOverlappingPairs()", mProfiler);
|
|
|
|
// For each possible convex vs convex pair of bodies
|
|
for (uint64 i=0; i < mOverlappingPairs.getNbPairs(); i++) {
|
|
|
|
// Check if we need to test overlap. If so, test if the two shapes are still overlapping.
|
|
// Otherwise, we destroy the overlapping pair
|
|
if (mOverlappingPairs.mNeedToTestOverlap[i]) {
|
|
|
|
if(mBroadPhaseSystem.testOverlappingShapes(mOverlappingPairs.mPairBroadPhaseId1[i], mOverlappingPairs.mPairBroadPhaseId2[i])) {
|
|
mOverlappingPairs.mNeedToTestOverlap[i] = false;
|
|
}
|
|
else {
|
|
mOverlappingPairs.removePair(mOverlappingPairs.mPairIds[i]);
|
|
i--;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Take a list of overlapping nodes in the broad-phase and create new overlapping pairs if necessary
|
|
void CollisionDetectionSystem::updateOverlappingPairs(const List<Pair<int32, int32>>& overlappingNodes) {
|
|
|
|
RP3D_PROFILE("CollisionDetectionSystem::updateOverlappingPairs()", mProfiler);
|
|
|
|
// For each overlapping pair of nodes
|
|
for (uint i=0; i < overlappingNodes.size(); i++) {
|
|
|
|
Pair<int32, int32> nodePair = overlappingNodes[i];
|
|
|
|
assert(nodePair.first != -1);
|
|
assert(nodePair.second != -1);
|
|
|
|
// Skip pairs with same overlapping nodes
|
|
if (nodePair.first != nodePair.second) {
|
|
|
|
// Get the two colliders
|
|
const Entity collider1Entity = mMapBroadPhaseIdToColliderEntity[nodePair.first];
|
|
const Entity collider2Entity = mMapBroadPhaseIdToColliderEntity[nodePair.second];
|
|
|
|
const uint collider1Index = mCollidersComponents.getEntityIndex(collider1Entity);
|
|
const uint collider2Index = mCollidersComponents.getEntityIndex(collider2Entity);
|
|
|
|
// Get the two bodies
|
|
const Entity body1Entity = mCollidersComponents.mBodiesEntities[collider1Index];
|
|
const Entity body2Entity = mCollidersComponents.mBodiesEntities[collider2Index];
|
|
|
|
// If the two colliders are from the same body, skip it
|
|
if (body1Entity != body2Entity) {
|
|
|
|
// Compute the overlapping pair ID
|
|
const uint64 pairId = pairNumbers(std::max(nodePair.first, nodePair.second), std::min(nodePair.first, nodePair.second));
|
|
|
|
// Check if the overlapping pair already exists
|
|
auto it = mOverlappingPairs.mMapPairIdToPairIndex.find(pairId);
|
|
if (it == mOverlappingPairs.mMapPairIdToPairIndex.end()) {
|
|
|
|
const unsigned short shape1CollideWithMaskBits = mCollidersComponents.mCollideWithMaskBits[collider1Index];
|
|
const unsigned short shape2CollideWithMaskBits = mCollidersComponents.mCollideWithMaskBits[collider2Index];
|
|
|
|
const unsigned short shape1CollisionCategoryBits = mCollidersComponents.mCollisionCategoryBits[collider1Index];
|
|
const unsigned short shape2CollisionCategoryBits = mCollidersComponents.mCollisionCategoryBits[collider2Index];
|
|
|
|
// Check if the collision filtering allows collision between the two shapes
|
|
if ((shape1CollideWithMaskBits & shape2CollisionCategoryBits) != 0 &&
|
|
(shape1CollisionCategoryBits & shape2CollideWithMaskBits) != 0) {
|
|
|
|
Collider* shape1 = mCollidersComponents.mColliders[collider1Index];
|
|
Collider* shape2 = mCollidersComponents.mColliders[collider2Index];
|
|
|
|
// Check that at least one collision shape is convex
|
|
if (shape1->getCollisionShape()->isConvex() || shape2->getCollisionShape()->isConvex()) {
|
|
|
|
// Add the new overlapping pair
|
|
mOverlappingPairs.addPair(shape1, shape2);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
|
|
// We do not need to test the pair for overlap because it has just been reported that they still overlap
|
|
mOverlappingPairs.mNeedToTestOverlap[it->second] = false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Compute the middle-phase collision detection
|
|
void CollisionDetectionSystem::computeMiddlePhase(NarrowPhaseInput& narrowPhaseInput) {
|
|
|
|
RP3D_PROFILE("CollisionDetectionSystem::computeMiddlePhase()", mProfiler);
|
|
|
|
// Reserve memory for the narrow-phase input using cached capacity from previous frame
|
|
narrowPhaseInput.reserveMemory();
|
|
|
|
// Remove the obsolete last frame collision infos and mark all the others as obsolete
|
|
mOverlappingPairs.clearObsoleteLastFrameCollisionInfos();
|
|
|
|
// For each possible convex vs convex pair of bodies
|
|
for (uint64 i=0; i < mOverlappingPairs.getNbConvexVsConvexPairs(); i++) {
|
|
|
|
assert(mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders1[i]) != -1);
|
|
assert(mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders2[i]) != -1);
|
|
assert(mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders1[i]) != mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders2[i]));
|
|
|
|
// Check that at least one body is enabled (active and awake) and not static
|
|
if (mOverlappingPairs.mIsActive[i]) {
|
|
|
|
const Entity collider1Entity = mOverlappingPairs.mColliders1[i];
|
|
const Entity collider2Entity = mOverlappingPairs.mColliders2[i];
|
|
|
|
const uint collider1Index = mCollidersComponents.getEntityIndex(collider1Entity);
|
|
const uint collider2Index = mCollidersComponents.getEntityIndex(collider2Entity);
|
|
|
|
CollisionShape* collisionShape1 = mCollidersComponents.mCollisionShapes[collider1Index];
|
|
CollisionShape* collisionShape2 = mCollidersComponents.mCollisionShapes[collider2Index];
|
|
|
|
NarrowPhaseAlgorithmType algorithmType = mOverlappingPairs.mNarrowPhaseAlgorithmType[i];
|
|
|
|
// No middle-phase is necessary, simply create a narrow phase info
|
|
// for the narrow-phase collision detection
|
|
narrowPhaseInput.addNarrowPhaseTest(mOverlappingPairs.mPairIds[i], i, collider1Entity, collider2Entity, collisionShape1, collisionShape2,
|
|
mCollidersComponents.mLocalToWorldTransforms[collider1Index],
|
|
mCollidersComponents.mLocalToWorldTransforms[collider2Index],
|
|
algorithmType, mMemoryManager.getSingleFrameAllocator());
|
|
}
|
|
}
|
|
|
|
// For each possible convex vs concave pair of bodies
|
|
const uint64 convexVsConcaveStartIndex = mOverlappingPairs.getConvexVsConcavePairsStartIndex();
|
|
for (uint64 i=convexVsConcaveStartIndex; i < convexVsConcaveStartIndex + mOverlappingPairs.getNbConvexVsConcavePairs(); i++) {
|
|
|
|
assert(mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders1[i]) != -1);
|
|
assert(mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders2[i]) != -1);
|
|
assert(mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders1[i]) != mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders2[i]));
|
|
|
|
// Check that at least one body is enabled (active and awake) and not static
|
|
if (mOverlappingPairs.mIsActive[i]) {
|
|
|
|
computeConvexVsConcaveMiddlePhase(i, mMemoryManager.getSingleFrameAllocator(), narrowPhaseInput);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Compute the middle-phase collision detection
|
|
void CollisionDetectionSystem::computeMiddlePhaseCollisionSnapshot(List<uint64>& convexPairs, List<uint64>& concavePairs, NarrowPhaseInput& narrowPhaseInput) {
|
|
|
|
RP3D_PROFILE("CollisionDetectionSystem::computeMiddlePhase()", mProfiler);
|
|
|
|
// Reserve memory for the narrow-phase input using cached capacity from previous frame
|
|
narrowPhaseInput.reserveMemory();
|
|
|
|
// Remove the obsolete last frame collision infos and mark all the others as obsolete
|
|
mOverlappingPairs.clearObsoleteLastFrameCollisionInfos();
|
|
|
|
// For each possible convex vs convex pair of bodies
|
|
for (uint64 p=0; p < convexPairs.size(); p++) {
|
|
|
|
const uint64 pairId = convexPairs[p];
|
|
|
|
const uint64 pairIndex = mOverlappingPairs.mMapPairIdToPairIndex[pairId];
|
|
assert(pairIndex < mOverlappingPairs.getNbPairs());
|
|
|
|
const Entity collider1Entity = mOverlappingPairs.mColliders1[pairIndex];
|
|
const Entity collider2Entity = mOverlappingPairs.mColliders2[pairIndex];
|
|
|
|
const uint collider1Index = mCollidersComponents.getEntityIndex(collider1Entity);
|
|
const uint collider2Index = mCollidersComponents.getEntityIndex(collider2Entity);
|
|
|
|
assert(mCollidersComponents.getBroadPhaseId(collider1Entity) != -1);
|
|
assert(mCollidersComponents.getBroadPhaseId(collider2Entity) != -1);
|
|
assert(mCollidersComponents.getBroadPhaseId(collider1Entity) != mCollidersComponents.getBroadPhaseId(collider2Entity));
|
|
|
|
CollisionShape* collisionShape1 = mCollidersComponents.mCollisionShapes[collider1Index];
|
|
CollisionShape* collisionShape2 = mCollidersComponents.mCollisionShapes[collider2Index];
|
|
|
|
NarrowPhaseAlgorithmType algorithmType = mOverlappingPairs.mNarrowPhaseAlgorithmType[pairIndex];
|
|
|
|
// No middle-phase is necessary, simply create a narrow phase info
|
|
// for the narrow-phase collision detection
|
|
narrowPhaseInput.addNarrowPhaseTest(pairId, pairIndex, collider1Entity, collider2Entity, collisionShape1, collisionShape2,
|
|
mCollidersComponents.mLocalToWorldTransforms[collider1Index],
|
|
mCollidersComponents.mLocalToWorldTransforms[collider2Index],
|
|
algorithmType, mMemoryManager.getSingleFrameAllocator());
|
|
|
|
}
|
|
|
|
// For each possible convex vs concave pair of bodies
|
|
for (uint p=0; p < concavePairs.size(); p++) {
|
|
|
|
const uint64 pairId = concavePairs[p];
|
|
const uint64 pairIndex = mOverlappingPairs.mMapPairIdToPairIndex[pairId];
|
|
|
|
assert(mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders1[pairIndex]) != -1);
|
|
assert(mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders2[pairIndex]) != -1);
|
|
assert(mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders1[pairIndex]) != mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders2[pairIndex]));
|
|
|
|
computeConvexVsConcaveMiddlePhase(pairIndex, mMemoryManager.getSingleFrameAllocator(), narrowPhaseInput);
|
|
}
|
|
}
|
|
|
|
// Compute the concave vs convex middle-phase algorithm for a given pair of bodies
|
|
void CollisionDetectionSystem::computeConvexVsConcaveMiddlePhase(uint64 pairIndex, MemoryAllocator& allocator, NarrowPhaseInput& narrowPhaseInput) {
|
|
|
|
RP3D_PROFILE("CollisionDetectionSystem::computeConvexVsConcaveMiddlePhase()", mProfiler);
|
|
|
|
const Entity collider1 = mOverlappingPairs.mColliders1[pairIndex];
|
|
const Entity collider2 = mOverlappingPairs.mColliders2[pairIndex];
|
|
|
|
const uint collider1Index = mCollidersComponents.getEntityIndex(collider1);
|
|
const uint collider2Index = mCollidersComponents.getEntityIndex(collider2);
|
|
|
|
Transform& shape1LocalToWorldTransform = mCollidersComponents.mLocalToWorldTransforms[collider1Index];
|
|
Transform& shape2LocalToWorldTransform = mCollidersComponents.mLocalToWorldTransforms[collider2Index];
|
|
|
|
Transform convexToConcaveTransform;
|
|
|
|
// Collision shape 1 is convex, collision shape 2 is concave
|
|
ConvexShape* convexShape;
|
|
ConcaveShape* concaveShape;
|
|
const bool isShape1Convex = mOverlappingPairs.mIsShape1Convex[pairIndex];
|
|
if (isShape1Convex) {
|
|
convexShape = static_cast<ConvexShape*>(mCollidersComponents.mCollisionShapes[collider1Index]);
|
|
concaveShape = static_cast<ConcaveShape*>(mCollidersComponents.mCollisionShapes[collider2Index]);
|
|
convexToConcaveTransform = shape2LocalToWorldTransform.getInverse() * shape1LocalToWorldTransform;
|
|
}
|
|
else { // Collision shape 2 is convex, collision shape 1 is concave
|
|
convexShape = static_cast<ConvexShape*>(mCollidersComponents.mCollisionShapes[collider1Index]);
|
|
concaveShape = static_cast<ConcaveShape*>(mCollidersComponents.mCollisionShapes[collider2Index]);
|
|
convexToConcaveTransform = shape1LocalToWorldTransform.getInverse() * shape2LocalToWorldTransform;
|
|
}
|
|
|
|
assert(mOverlappingPairs.mNarrowPhaseAlgorithmType[pairIndex] != NarrowPhaseAlgorithmType::None);
|
|
|
|
// Compute the convex shape AABB in the local-space of the convex shape
|
|
AABB aabb;
|
|
convexShape->computeAABB(aabb, convexToConcaveTransform);
|
|
|
|
// Compute the concave shape triangles that are overlapping with the convex mesh AABB
|
|
List<Vector3> triangleVertices(allocator);
|
|
List<Vector3> triangleVerticesNormals(allocator);
|
|
List<uint> shapeIds(allocator);
|
|
concaveShape->computeOverlappingTriangles(aabb, triangleVertices, triangleVerticesNormals, shapeIds, allocator);
|
|
|
|
assert(triangleVertices.size() == triangleVerticesNormals.size());
|
|
assert(shapeIds.size() == triangleVertices.size() / 3);
|
|
assert(triangleVertices.size() % 3 == 0);
|
|
assert(triangleVerticesNormals.size() % 3 == 0);
|
|
|
|
// For each overlapping triangle
|
|
for (uint i=0; i < shapeIds.size(); i++)
|
|
{
|
|
// Create a triangle collision shape (the allocated memory for the TriangleShape will be released in the
|
|
// destructor of the corresponding NarrowPhaseInfo.
|
|
TriangleShape* triangleShape = new (allocator.allocate(sizeof(TriangleShape)))
|
|
TriangleShape(&(triangleVertices[i * 3]), &(triangleVerticesNormals[i * 3]), shapeIds[i], allocator);
|
|
|
|
#ifdef IS_PROFILING_ACTIVE
|
|
|
|
// Set the profiler to the triangle shape
|
|
triangleShape->setProfiler(mProfiler);
|
|
|
|
#endif
|
|
|
|
// Create a narrow phase info for the narrow-phase collision detection
|
|
narrowPhaseInput.addNarrowPhaseTest(mOverlappingPairs.mPairIds[pairIndex], pairIndex, collider1, collider2, isShape1Convex ? convexShape : triangleShape,
|
|
isShape1Convex ? triangleShape : convexShape,
|
|
shape1LocalToWorldTransform, shape2LocalToWorldTransform,
|
|
mOverlappingPairs.mNarrowPhaseAlgorithmType[pairIndex], allocator);
|
|
}
|
|
}
|
|
|
|
// Execute the narrow-phase collision detection algorithm on batches
|
|
bool CollisionDetectionSystem::testNarrowPhaseCollision(NarrowPhaseInput& narrowPhaseInput, bool reportContacts,
|
|
bool clipWithPreviousAxisIfStillColliding, MemoryAllocator& allocator) {
|
|
|
|
bool contactFound = false;
|
|
|
|
// Get the narrow-phase collision detection algorithms for each kind of collision shapes
|
|
SphereVsSphereAlgorithm* sphereVsSphereAlgo = mCollisionDispatch.getSphereVsSphereAlgorithm();
|
|
SphereVsCapsuleAlgorithm* sphereVsCapsuleAlgo = mCollisionDispatch.getSphereVsCapsuleAlgorithm();
|
|
CapsuleVsCapsuleAlgorithm* capsuleVsCapsuleAlgo = mCollisionDispatch.getCapsuleVsCapsuleAlgorithm();
|
|
SphereVsConvexPolyhedronAlgorithm* sphereVsConvexPolyAlgo = mCollisionDispatch.getSphereVsConvexPolyhedronAlgorithm();
|
|
CapsuleVsConvexPolyhedronAlgorithm* capsuleVsConvexPolyAlgo = mCollisionDispatch.getCapsuleVsConvexPolyhedronAlgorithm();
|
|
ConvexPolyhedronVsConvexPolyhedronAlgorithm* convexPolyVsConvexPolyAlgo = mCollisionDispatch.getConvexPolyhedronVsConvexPolyhedronAlgorithm();
|
|
|
|
// get the narrow-phase batches to test for collision
|
|
SphereVsSphereNarrowPhaseInfoBatch& sphereVsSphereBatch = narrowPhaseInput.getSphereVsSphereBatch();
|
|
SphereVsCapsuleNarrowPhaseInfoBatch& sphereVsCapsuleBatch = narrowPhaseInput.getSphereVsCapsuleBatch();
|
|
CapsuleVsCapsuleNarrowPhaseInfoBatch& capsuleVsCapsuleBatch = narrowPhaseInput.getCapsuleVsCapsuleBatch();
|
|
NarrowPhaseInfoBatch& sphereVsConvexPolyhedronBatch = narrowPhaseInput.getSphereVsConvexPolyhedronBatch();
|
|
NarrowPhaseInfoBatch& capsuleVsConvexPolyhedronBatch = narrowPhaseInput.getCapsuleVsConvexPolyhedronBatch();
|
|
NarrowPhaseInfoBatch& convexPolyhedronVsConvexPolyhedronBatch = narrowPhaseInput.getConvexPolyhedronVsConvexPolyhedronBatch();
|
|
|
|
// Compute the narrow-phase collision detection for each kind of collision shapes
|
|
if (sphereVsSphereBatch.getNbObjects() > 0) {
|
|
contactFound |= sphereVsSphereAlgo->testCollision(sphereVsSphereBatch, 0, sphereVsSphereBatch.getNbObjects(), reportContacts, allocator);
|
|
}
|
|
if (sphereVsCapsuleBatch.getNbObjects() > 0) {
|
|
contactFound |= sphereVsCapsuleAlgo->testCollision(sphereVsCapsuleBatch, 0, sphereVsCapsuleBatch.getNbObjects(), reportContacts, allocator);
|
|
}
|
|
if (capsuleVsCapsuleBatch.getNbObjects() > 0) {
|
|
contactFound |= capsuleVsCapsuleAlgo->testCollision(capsuleVsCapsuleBatch, 0, capsuleVsCapsuleBatch.getNbObjects(), reportContacts, allocator);
|
|
}
|
|
if (sphereVsConvexPolyhedronBatch.getNbObjects() > 0) {
|
|
contactFound |= sphereVsConvexPolyAlgo->testCollision(sphereVsConvexPolyhedronBatch, 0, sphereVsConvexPolyhedronBatch.getNbObjects(), reportContacts, clipWithPreviousAxisIfStillColliding, allocator);
|
|
}
|
|
if (capsuleVsConvexPolyhedronBatch.getNbObjects() > 0) {
|
|
contactFound |= capsuleVsConvexPolyAlgo->testCollision(capsuleVsConvexPolyhedronBatch, 0, capsuleVsConvexPolyhedronBatch.getNbObjects(), reportContacts, clipWithPreviousAxisIfStillColliding, allocator);
|
|
}
|
|
if (convexPolyhedronVsConvexPolyhedronBatch.getNbObjects() > 0) {
|
|
contactFound |= convexPolyVsConvexPolyAlgo->testCollision(convexPolyhedronVsConvexPolyhedronBatch, 0, convexPolyhedronVsConvexPolyhedronBatch.getNbObjects(), reportContacts, clipWithPreviousAxisIfStillColliding, allocator);
|
|
}
|
|
|
|
return contactFound;
|
|
}
|
|
|
|
// Process the potential contacts after narrow-phase collision detection
|
|
void CollisionDetectionSystem::processAllPotentialContacts(NarrowPhaseInput& narrowPhaseInput, bool updateLastFrameInfo,
|
|
List<ContactPointInfo>& potentialContactPoints,
|
|
Map<uint64, uint>* mapPairIdToContactPairIndex,
|
|
List<ContactManifoldInfo>& potentialContactManifolds,
|
|
List<ContactPair>* contactPairs,
|
|
Map<Entity, List<uint>>& mapBodyToContactPairs) {
|
|
|
|
assert(contactPairs->size() == 0);
|
|
assert(mapPairIdToContactPairIndex->size() == 0);
|
|
|
|
// get the narrow-phase batches to test for collision
|
|
NarrowPhaseInfoBatch& sphereVsSphereBatch = narrowPhaseInput.getSphereVsSphereBatch();
|
|
NarrowPhaseInfoBatch& sphereVsCapsuleBatch = narrowPhaseInput.getSphereVsCapsuleBatch();
|
|
NarrowPhaseInfoBatch& capsuleVsCapsuleBatch = narrowPhaseInput.getCapsuleVsCapsuleBatch();
|
|
NarrowPhaseInfoBatch& sphereVsConvexPolyhedronBatch = narrowPhaseInput.getSphereVsConvexPolyhedronBatch();
|
|
NarrowPhaseInfoBatch& capsuleVsConvexPolyhedronBatch = narrowPhaseInput.getCapsuleVsConvexPolyhedronBatch();
|
|
NarrowPhaseInfoBatch& convexPolyhedronVsConvexPolyhedronBatch = narrowPhaseInput.getConvexPolyhedronVsConvexPolyhedronBatch();
|
|
|
|
// Process the potential contacts
|
|
processPotentialContacts(sphereVsSphereBatch, updateLastFrameInfo, potentialContactPoints, mapPairIdToContactPairIndex,
|
|
potentialContactManifolds, contactPairs, mapBodyToContactPairs);
|
|
processPotentialContacts(sphereVsCapsuleBatch, updateLastFrameInfo, potentialContactPoints, mapPairIdToContactPairIndex,
|
|
potentialContactManifolds, contactPairs, mapBodyToContactPairs);
|
|
processPotentialContacts(capsuleVsCapsuleBatch, updateLastFrameInfo, potentialContactPoints, mapPairIdToContactPairIndex,
|
|
potentialContactManifolds, contactPairs, mapBodyToContactPairs);
|
|
processPotentialContacts(sphereVsConvexPolyhedronBatch, updateLastFrameInfo, potentialContactPoints, mapPairIdToContactPairIndex,
|
|
potentialContactManifolds, contactPairs, mapBodyToContactPairs);
|
|
processPotentialContacts(capsuleVsConvexPolyhedronBatch, updateLastFrameInfo, potentialContactPoints, mapPairIdToContactPairIndex,
|
|
potentialContactManifolds, contactPairs, mapBodyToContactPairs);
|
|
processPotentialContacts(convexPolyhedronVsConvexPolyhedronBatch, updateLastFrameInfo, potentialContactPoints, mapPairIdToContactPairIndex,
|
|
potentialContactManifolds, contactPairs, mapBodyToContactPairs);
|
|
}
|
|
|
|
// Compute the narrow-phase collision detection
|
|
void CollisionDetectionSystem::computeNarrowPhase() {
|
|
|
|
RP3D_PROFILE("CollisionDetectionSystem::computeNarrowPhase()", mProfiler);
|
|
|
|
MemoryAllocator& allocator = mMemoryManager.getSingleFrameAllocator();
|
|
|
|
// Swap the previous and current contacts lists
|
|
swapPreviousAndCurrentContacts();
|
|
|
|
// Test the narrow-phase collision detection on the batches to be tested
|
|
testNarrowPhaseCollision(mNarrowPhaseInput, true, true, allocator);
|
|
|
|
// Process all the potential contacts after narrow-phase collision
|
|
processAllPotentialContacts(mNarrowPhaseInput, true, mPotentialContactPoints, mCurrentMapPairIdToContactPairIndex,
|
|
mPotentialContactManifolds, mCurrentContactPairs, mMapBodyToContactPairs);
|
|
|
|
// Reduce the number of contact points in the manifolds
|
|
reducePotentialContactManifolds(mCurrentContactPairs, mPotentialContactManifolds, mPotentialContactPoints);
|
|
|
|
assert(mCurrentContactManifolds->size() == 0);
|
|
assert(mCurrentContactPoints->size() == 0);
|
|
|
|
// Create the actual narrow-phase contacts
|
|
createContacts();
|
|
|
|
mNarrowPhaseInput.clear();
|
|
}
|
|
|
|
// Compute the narrow-phase collision detection for the testOverlap() methods.
|
|
/// This method returns true if contacts are found.
|
|
bool CollisionDetectionSystem::computeNarrowPhaseOverlapSnapshot(NarrowPhaseInput& narrowPhaseInput, OverlapCallback* callback) {
|
|
|
|
RP3D_PROFILE("CollisionDetectionSystem::computeNarrowPhaseOverlapSnapshot()", mProfiler);
|
|
|
|
MemoryAllocator& allocator = mMemoryManager.getPoolAllocator();
|
|
|
|
// Test the narrow-phase collision detection on the batches to be tested
|
|
bool collisionFound = testNarrowPhaseCollision(narrowPhaseInput, false, false, allocator);
|
|
if (collisionFound && callback != nullptr) {
|
|
|
|
// Compute the overlapping bodies
|
|
List<Pair<Entity, Entity>> overlapBodies(allocator);
|
|
computeSnapshotContactPairs(narrowPhaseInput, overlapBodies);
|
|
|
|
// Report overlapping bodies
|
|
OverlapCallback::CallbackData callbackData(overlapBodies, *mWorld);
|
|
(*callback).onOverlap(callbackData);
|
|
}
|
|
|
|
return collisionFound;
|
|
}
|
|
|
|
// Process the potential overlapping bodies for the testOverlap() methods
|
|
void CollisionDetectionSystem::computeSnapshotContactPairs(NarrowPhaseInput& narrowPhaseInput, List<Pair<Entity, Entity>>& overlapPairs) const {
|
|
|
|
// get the narrow-phase batches to test for collision
|
|
NarrowPhaseInfoBatch& sphereVsSphereBatch = narrowPhaseInput.getSphereVsSphereBatch();
|
|
NarrowPhaseInfoBatch& sphereVsCapsuleBatch = narrowPhaseInput.getSphereVsCapsuleBatch();
|
|
NarrowPhaseInfoBatch& capsuleVsCapsuleBatch = narrowPhaseInput.getCapsuleVsCapsuleBatch();
|
|
NarrowPhaseInfoBatch& sphereVsConvexPolyhedronBatch = narrowPhaseInput.getSphereVsConvexPolyhedronBatch();
|
|
NarrowPhaseInfoBatch& capsuleVsConvexPolyhedronBatch = narrowPhaseInput.getCapsuleVsConvexPolyhedronBatch();
|
|
NarrowPhaseInfoBatch& convexPolyhedronVsConvexPolyhedronBatch = narrowPhaseInput.getConvexPolyhedronVsConvexPolyhedronBatch();
|
|
|
|
// Process the potential contacts
|
|
computeSnapshotContactPairs(sphereVsSphereBatch, overlapPairs);
|
|
computeSnapshotContactPairs(sphereVsCapsuleBatch, overlapPairs);
|
|
computeSnapshotContactPairs(capsuleVsCapsuleBatch, overlapPairs);
|
|
computeSnapshotContactPairs(sphereVsConvexPolyhedronBatch, overlapPairs);
|
|
computeSnapshotContactPairs(capsuleVsConvexPolyhedronBatch, overlapPairs);
|
|
computeSnapshotContactPairs(convexPolyhedronVsConvexPolyhedronBatch, overlapPairs);
|
|
}
|
|
|
|
// Notify that the overlapping pairs where a given collider is involved need to be tested for overlap
|
|
void CollisionDetectionSystem::notifyOverlappingPairsToTestOverlap(Collider* collider) {
|
|
|
|
// Get the overlapping pairs involved with this collider
|
|
List<uint64>& overlappingPairs = mCollidersComponents.getOverlappingPairs(collider->getEntity());
|
|
|
|
for (uint i=0; i < overlappingPairs.size(); i++) {
|
|
|
|
// Notify that the overlapping pair needs to be testbed for overlap
|
|
mOverlappingPairs.setNeedToTestOverlap(overlappingPairs[i], true);
|
|
}
|
|
}
|
|
|
|
// Convert the potential overlapping bodies for the testOverlap() methods
|
|
void CollisionDetectionSystem::computeSnapshotContactPairs(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, List<Pair<Entity, Entity>>& overlapPairs) const {
|
|
|
|
RP3D_PROFILE("CollisionDetectionSystem::computeSnapshotContactPairs()", mProfiler);
|
|
|
|
// For each narrow phase info object
|
|
for(uint i=0; i < narrowPhaseInfoBatch.getNbObjects(); i++) {
|
|
|
|
// If there is a contact
|
|
if (narrowPhaseInfoBatch.isColliding[i]) {
|
|
|
|
// Add the pair of bodies in contact into the list
|
|
overlapPairs.add(Pair<Entity, Entity>(mCollidersComponents.getBody(narrowPhaseInfoBatch.colliderEntities1[i]),
|
|
mCollidersComponents.getBody(narrowPhaseInfoBatch.colliderEntities2[i])));
|
|
}
|
|
|
|
narrowPhaseInfoBatch.resetContactPoints(i);
|
|
}
|
|
}
|
|
|
|
// Compute the narrow-phase collision detection for the testCollision() methods.
|
|
// This method returns true if contacts are found.
|
|
bool CollisionDetectionSystem::computeNarrowPhaseCollisionSnapshot(NarrowPhaseInput& narrowPhaseInput, CollisionCallback& callback) {
|
|
|
|
RP3D_PROFILE("CollisionDetectionSystem::computeNarrowPhaseCollisionSnapshot()", mProfiler);
|
|
|
|
MemoryAllocator& allocator = mMemoryManager.getHeapAllocator();
|
|
|
|
// Test the narrow-phase collision detection on the batches to be tested
|
|
bool collisionFound = testNarrowPhaseCollision(narrowPhaseInput, true, false, allocator);
|
|
|
|
// If collision has been found, create contacts
|
|
if (collisionFound) {
|
|
|
|
List<ContactPointInfo> potentialContactPoints(allocator);
|
|
List<ContactManifoldInfo> potentialContactManifolds(allocator);
|
|
Map<uint64, uint> mapPairIdToContactPairIndex(allocator);
|
|
List<ContactPair> contactPairs(allocator);
|
|
List<ContactManifold> contactManifolds(allocator);
|
|
List<ContactPoint> contactPoints(allocator);
|
|
Map<Entity, List<uint>> mapBodyToContactPairs(allocator);
|
|
|
|
// Process all the potential contacts after narrow-phase collision
|
|
processAllPotentialContacts(narrowPhaseInput, true, potentialContactPoints, &mapPairIdToContactPairIndex, potentialContactManifolds,
|
|
&contactPairs, mapBodyToContactPairs);
|
|
|
|
// Reduce the number of contact points in the manifolds
|
|
reducePotentialContactManifolds(&contactPairs, potentialContactManifolds, potentialContactPoints);
|
|
|
|
// Create the actual contact manifolds and contact points
|
|
createSnapshotContacts(contactPairs, contactManifolds, contactPoints, potentialContactManifolds,
|
|
potentialContactPoints);
|
|
|
|
// Report the contacts to the user
|
|
reportContacts(callback, &contactPairs, &contactManifolds, &contactPoints);
|
|
}
|
|
|
|
return collisionFound;
|
|
}
|
|
|
|
// Swap the previous and current contacts lists
|
|
void CollisionDetectionSystem::swapPreviousAndCurrentContacts() {
|
|
|
|
if (mPreviousContactPairs == &mContactPairs1) {
|
|
|
|
mPreviousContactPairs = &mContactPairs2;
|
|
mPreviousContactManifolds = &mContactManifolds2;
|
|
mPreviousContactPoints = &mContactPoints2;
|
|
mPreviousMapPairIdToContactPairIndex = &mMapPairIdToContactPairIndex2;
|
|
|
|
mCurrentContactPairs = &mContactPairs1;
|
|
mCurrentContactManifolds = &mContactManifolds1;
|
|
mCurrentContactPoints = &mContactPoints1;
|
|
mCurrentMapPairIdToContactPairIndex = &mMapPairIdToContactPairIndex1;
|
|
}
|
|
else {
|
|
|
|
mPreviousContactPairs = &mContactPairs1;
|
|
mPreviousContactManifolds = &mContactManifolds1;
|
|
mPreviousContactPoints = &mContactPoints1;
|
|
mPreviousMapPairIdToContactPairIndex = &mMapPairIdToContactPairIndex1;
|
|
|
|
mCurrentContactPairs = &mContactPairs2;
|
|
mCurrentContactManifolds = &mContactManifolds2;
|
|
mCurrentContactPoints = &mContactPoints2;
|
|
mCurrentMapPairIdToContactPairIndex = &mMapPairIdToContactPairIndex2;
|
|
}
|
|
}
|
|
|
|
// Create the actual contact manifolds and contacts points
|
|
void CollisionDetectionSystem::createContacts() {
|
|
|
|
RP3D_PROFILE("CollisionDetectionSystem::createContacts()", mProfiler);
|
|
|
|
mCurrentContactManifolds->reserve(mCurrentContactPairs->size());
|
|
mCurrentContactPoints->reserve(mCurrentContactManifolds->size());
|
|
|
|
// For each contact pair
|
|
for (uint p=0; p < (*mCurrentContactPairs).size(); p++) {
|
|
|
|
ContactPair& contactPair = (*mCurrentContactPairs)[p];
|
|
assert(contactPair.potentialContactManifoldsIndices.size() > 0);
|
|
|
|
contactPair.contactManifoldsIndex = mCurrentContactManifolds->size();
|
|
contactPair.nbContactManifolds = contactPair.potentialContactManifoldsIndices.size();
|
|
contactPair.contactPointsIndex = mCurrentContactPoints->size();
|
|
|
|
// For each potential contact manifold of the pair
|
|
for (uint m=0; m < contactPair.potentialContactManifoldsIndices.size(); m++) {
|
|
|
|
ContactManifoldInfo& potentialManifold = mPotentialContactManifolds[contactPair.potentialContactManifoldsIndices[m]];
|
|
|
|
// Start index and number of contact points for this manifold
|
|
const uint contactPointsIndex = mCurrentContactPoints->size();
|
|
const int8 nbContactPoints = static_cast<int8>(potentialManifold.potentialContactPointsIndices.size());
|
|
contactPair.nbToTalContactPoints += nbContactPoints;
|
|
|
|
// We create a new contact manifold
|
|
ContactManifold contactManifold(contactPair.body1Entity, contactPair.body2Entity, contactPair.collider1Entity,
|
|
contactPair.collider2Entity, contactPointsIndex, nbContactPoints);
|
|
|
|
// Add the contact manifold
|
|
mCurrentContactManifolds->add(contactManifold);
|
|
|
|
assert(potentialManifold.potentialContactPointsIndices.size() > 0);
|
|
|
|
// For each contact point of the manifold
|
|
for (uint c=0; c < potentialManifold.potentialContactPointsIndices.size(); c++) {
|
|
|
|
ContactPointInfo& potentialContactPoint = mPotentialContactPoints[potentialManifold.potentialContactPointsIndices[c]];
|
|
|
|
// Create a new contact point
|
|
ContactPoint contactPoint(potentialContactPoint, mWorld->mConfig);
|
|
|
|
// Add the contact point
|
|
mCurrentContactPoints->add(contactPoint);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Initialize the current contacts with the contacts from the previous frame (for warmstarting)
|
|
initContactsWithPreviousOnes();
|
|
|
|
// Report contacts to the user
|
|
if (mWorld->mEventListener != nullptr) {
|
|
reportContacts(*(mWorld->mEventListener), mCurrentContactPairs, mCurrentContactManifolds, mCurrentContactPoints);
|
|
}
|
|
|
|
// Reset the potential contacts
|
|
mPotentialContactPoints.clear(true);
|
|
mPotentialContactManifolds.clear(true);
|
|
}
|
|
|
|
// Create the actual contact manifolds and contacts points for testCollision() methods
|
|
void CollisionDetectionSystem::createSnapshotContacts(List<ContactPair>& contactPairs,
|
|
List<ContactManifold>& contactManifolds,
|
|
List<ContactPoint>& contactPoints,
|
|
List<ContactManifoldInfo>& potentialContactManifolds,
|
|
List<ContactPointInfo>& potentialContactPoints) {
|
|
|
|
RP3D_PROFILE("CollisionDetectionSystem::createSnapshotContacts()", mProfiler);
|
|
|
|
contactManifolds.reserve(contactPairs.size());
|
|
contactPoints.reserve(contactManifolds.size());
|
|
|
|
// For each contact pair
|
|
for (uint p=0; p < contactPairs.size(); p++) {
|
|
|
|
ContactPair& contactPair = contactPairs[p];
|
|
assert(contactPair.potentialContactManifoldsIndices.size() > 0);
|
|
|
|
contactPair.contactManifoldsIndex = contactManifolds.size();
|
|
contactPair.nbContactManifolds = contactPair.potentialContactManifoldsIndices.size();
|
|
contactPair.contactPointsIndex = contactPoints.size();
|
|
|
|
// For each potential contact manifold of the pair
|
|
for (uint m=0; m < contactPair.potentialContactManifoldsIndices.size(); m++) {
|
|
|
|
ContactManifoldInfo& potentialManifold = potentialContactManifolds[contactPair.potentialContactManifoldsIndices[m]];
|
|
|
|
// Start index and number of contact points for this manifold
|
|
const uint contactPointsIndex = contactPoints.size();
|
|
const int8 nbContactPoints = static_cast<int8>(potentialManifold.potentialContactPointsIndices.size());
|
|
contactPair.nbToTalContactPoints += nbContactPoints;
|
|
|
|
// We create a new contact manifold
|
|
ContactManifold contactManifold(contactPair.body1Entity, contactPair.body2Entity, contactPair.collider1Entity,
|
|
contactPair.collider2Entity, contactPointsIndex, nbContactPoints);
|
|
|
|
// Add the contact manifold
|
|
contactManifolds.add(contactManifold);
|
|
|
|
assert(potentialManifold.potentialContactPointsIndices.size() > 0);
|
|
|
|
// For each contact point of the manifold
|
|
for (uint c=0; c < potentialManifold.potentialContactPointsIndices.size(); c++) {
|
|
|
|
ContactPointInfo& potentialContactPoint = potentialContactPoints[potentialManifold.potentialContactPointsIndices[c]];
|
|
|
|
// Create a new contact point
|
|
ContactPoint contactPoint(potentialContactPoint, mWorld->mConfig);
|
|
|
|
// Add the contact point
|
|
contactPoints.add(contactPoint);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Initialize the current contacts with the contacts from the previous frame (for warmstarting)
|
|
void CollisionDetectionSystem::initContactsWithPreviousOnes() {
|
|
|
|
// For each contact pair of the current frame
|
|
for (uint i=0; i < mCurrentContactPairs->size(); i++) {
|
|
|
|
ContactPair& currentContactPair = (*mCurrentContactPairs)[i];
|
|
|
|
// Find the corresponding contact pair in the previous frame (if any)
|
|
auto itPrevContactPair = mPreviousMapPairIdToContactPairIndex->find(currentContactPair.pairId);
|
|
|
|
// If we have found a corresponding contact pair in the previous frame
|
|
if (itPrevContactPair != mPreviousMapPairIdToContactPairIndex->end()) {
|
|
|
|
const uint previousContactPairIndex = itPrevContactPair->second;
|
|
ContactPair& previousContactPair = (*mPreviousContactPairs)[previousContactPairIndex];
|
|
|
|
// --------------------- Contact Manifolds --------------------- //
|
|
|
|
const uint contactManifoldsIndex = currentContactPair.contactManifoldsIndex;
|
|
const uint nbContactManifolds = currentContactPair.nbContactManifolds;
|
|
|
|
// For each current contact manifold of the contact pair
|
|
for (uint m=contactManifoldsIndex; m < contactManifoldsIndex + nbContactManifolds; m++) {
|
|
|
|
assert(m < mCurrentContactManifolds->size());
|
|
ContactManifold& currentContactManifold = (*mCurrentContactManifolds)[m];
|
|
assert(currentContactManifold.nbContactPoints > 0);
|
|
ContactPoint& currentContactPoint = (*mCurrentContactPoints)[currentContactManifold.contactPointsIndex];
|
|
const Vector3& currentContactPointNormal = currentContactPoint.getNormal();
|
|
|
|
// Find a similar contact manifold among the contact manifolds from the previous frame (for warmstarting)
|
|
const uint previousContactManifoldIndex = previousContactPair.contactManifoldsIndex;
|
|
const uint previousNbContactManifolds = previousContactPair.nbContactManifolds;
|
|
for (uint p=previousContactManifoldIndex; p < previousContactManifoldIndex + previousNbContactManifolds; p++) {
|
|
|
|
ContactManifold& previousContactManifold = (*mPreviousContactManifolds)[p];
|
|
assert(previousContactManifold.nbContactPoints > 0);
|
|
ContactPoint& previousContactPoint = (*mPreviousContactPoints)[previousContactManifold.contactPointsIndex];
|
|
|
|
// If the previous contact manifold has a similar contact normal with the current manifold
|
|
if (previousContactPoint.getNormal().dot(currentContactPointNormal) >= mWorld->mConfig.cosAngleSimilarContactManifold) {
|
|
|
|
// Transfer data from the previous contact manifold to the current one
|
|
currentContactManifold.frictionVector1 = previousContactManifold.frictionVector1;
|
|
currentContactManifold.frictionVector2 = previousContactManifold.frictionVector2;
|
|
currentContactManifold.frictionImpulse1 = previousContactManifold.frictionImpulse1;
|
|
currentContactManifold.frictionImpulse2 = previousContactManifold.frictionImpulse2;
|
|
currentContactManifold.frictionTwistImpulse = previousContactManifold.frictionTwistImpulse;
|
|
currentContactManifold.rollingResistanceImpulse = previousContactManifold.rollingResistanceImpulse;
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// --------------------- Contact Points --------------------- //
|
|
|
|
const uint contactPointsIndex = currentContactPair.contactPointsIndex;
|
|
const uint nbTotalContactPoints = currentContactPair.nbToTalContactPoints;
|
|
|
|
// For each current contact point of the contact pair
|
|
for (uint c=contactPointsIndex; c < contactPointsIndex + nbTotalContactPoints; c++) {
|
|
|
|
assert(c < mCurrentContactPoints->size());
|
|
ContactPoint& currentContactPoint = (*mCurrentContactPoints)[c];
|
|
|
|
// Find a similar contact point among the contact points from the previous frame (for warmstarting)
|
|
const uint previousContactPointsIndex = previousContactPair.contactPointsIndex;
|
|
const uint previousNbContactPoints = previousContactPair.nbToTalContactPoints;
|
|
for (uint p=previousContactPointsIndex; p < previousContactPointsIndex + previousNbContactPoints; p++) {
|
|
|
|
ContactPoint& previousContactPoint = (*mPreviousContactPoints)[p];
|
|
|
|
// If the previous contact point is very close to th current one
|
|
const decimal distSquare = (currentContactPoint.getLocalPointOnShape1() - previousContactPoint.getLocalPointOnShape1()).lengthSquare();
|
|
if (distSquare <= mWorld->mConfig.persistentContactDistanceThreshold * mWorld->mConfig.persistentContactDistanceThreshold) {
|
|
|
|
// Transfer data from the previous contact point to the current one
|
|
currentContactPoint.setPenetrationImpulse(previousContactPoint.getPenetrationImpulse());
|
|
currentContactPoint.setIsRestingContact(previousContactPoint.getIsRestingContact());
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
mPreviousContactPoints->clear();
|
|
mPreviousContactManifolds->clear();
|
|
mPreviousContactPairs->clear();
|
|
mPreviousMapPairIdToContactPairIndex->clear();
|
|
}
|
|
|
|
// Remove a body from the collision detection
|
|
void CollisionDetectionSystem::removeCollider(Collider* collider) {
|
|
|
|
const int colliderBroadPhaseId = collider->getBroadPhaseId();
|
|
|
|
assert(colliderBroadPhaseId != -1);
|
|
assert(mMapBroadPhaseIdToColliderEntity.containsKey(colliderBroadPhaseId));
|
|
|
|
// Remove all the overlapping pairs involving this collider
|
|
List<uint64>& overlappingPairs = mCollidersComponents.getOverlappingPairs(collider->getEntity());
|
|
while(overlappingPairs.size() > 0) {
|
|
|
|
// TODO : Remove all the contact manifold of the overlapping pair from the contact manifolds list of the two bodies involved
|
|
|
|
// Remove the overlapping pair
|
|
mOverlappingPairs.removePair(overlappingPairs[0]);
|
|
}
|
|
|
|
mMapBroadPhaseIdToColliderEntity.remove(colliderBroadPhaseId);
|
|
|
|
// Remove the body from the broad-phase
|
|
mBroadPhaseSystem.removeCollider(collider);
|
|
}
|
|
|
|
// Ray casting method
|
|
void CollisionDetectionSystem::raycast(RaycastCallback* raycastCallback,
|
|
const Ray& ray,
|
|
unsigned short raycastWithCategoryMaskBits) const {
|
|
|
|
RP3D_PROFILE("CollisionDetectionSystem::raycast()", mProfiler);
|
|
|
|
RaycastTest rayCastTest(raycastCallback);
|
|
|
|
// Ask the broad-phase algorithm to call the testRaycastAgainstShape()
|
|
// callback method for each collider hit by the ray in the broad-phase
|
|
mBroadPhaseSystem.raycast(ray, rayCastTest, raycastWithCategoryMaskBits);
|
|
}
|
|
|
|
// Convert the potential contact into actual contacts
|
|
void CollisionDetectionSystem::processPotentialContacts(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, bool updateLastFrameInfo,
|
|
List<ContactPointInfo>& potentialContactPoints,
|
|
Map<uint64, uint>* mapPairIdToContactPairIndex,
|
|
List<ContactManifoldInfo>& potentialContactManifolds,
|
|
List<ContactPair>* contactPairs,
|
|
Map<Entity, List<uint>>& mapBodyToContactPairs) {
|
|
|
|
RP3D_PROFILE("CollisionDetectionSystem::processPotentialContacts()", mProfiler);
|
|
|
|
// For each narrow phase info object
|
|
for(uint i=0; i < narrowPhaseInfoBatch.getNbObjects(); i++) {
|
|
|
|
if (updateLastFrameInfo) {
|
|
narrowPhaseInfoBatch.lastFrameCollisionInfos[i]->wasColliding = narrowPhaseInfoBatch.isColliding[i];
|
|
|
|
// The previous frame collision info is now valid
|
|
narrowPhaseInfoBatch.lastFrameCollisionInfos[i]->isValid = true;
|
|
}
|
|
|
|
// Add the potential contacts
|
|
for (uint j=0; j < narrowPhaseInfoBatch.contactPoints[i].size(); j++) {
|
|
|
|
assert(narrowPhaseInfoBatch.isColliding[i]);
|
|
|
|
const ContactPointInfo& contactPoint = *(narrowPhaseInfoBatch.contactPoints[i][j]);
|
|
|
|
// Add the contact point to the list of potential contact points
|
|
const uint contactPointIndex = static_cast<uint>(potentialContactPoints.size());
|
|
|
|
potentialContactPoints.add(contactPoint);
|
|
|
|
bool similarManifoldFound = false;
|
|
|
|
// If there is already a contact pair for this overlapping pair
|
|
const uint64 pairId = narrowPhaseInfoBatch.overlappingPairIds[i];
|
|
auto it = mapPairIdToContactPairIndex->find(pairId);
|
|
ContactPair* pairContact = nullptr;
|
|
if (it != mapPairIdToContactPairIndex->end()) {
|
|
|
|
assert(it->first == pairId);
|
|
|
|
const uint pairContactIndex = it->second;
|
|
pairContact = &((*contactPairs)[pairContactIndex]);
|
|
|
|
assert(pairContact->potentialContactManifoldsIndices.size() > 0);
|
|
|
|
// For each contact manifold of the overlapping pair
|
|
for (uint m=0; m < pairContact->potentialContactManifoldsIndices.size(); m++) {
|
|
|
|
uint contactManifoldIndex = pairContact->potentialContactManifoldsIndices[m];
|
|
|
|
// Get the first contact point of the current manifold
|
|
assert(potentialContactManifolds[contactManifoldIndex].potentialContactPointsIndices.size() > 0);
|
|
const uint manifoldContactPointIndex = potentialContactManifolds[contactManifoldIndex].potentialContactPointsIndices[0];
|
|
const ContactPointInfo& manifoldContactPoint = potentialContactPoints[manifoldContactPointIndex];
|
|
|
|
// If we have found a corresponding manifold for the new contact point
|
|
// (a manifold with a similar contact normal direction)
|
|
if (manifoldContactPoint.normal.dot(contactPoint.normal) >= mWorld->mConfig.cosAngleSimilarContactManifold) {
|
|
|
|
// Add the contact point to the manifold
|
|
potentialContactManifolds[contactManifoldIndex].potentialContactPointsIndices.add(contactPointIndex);
|
|
|
|
similarManifoldFound = true;
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// If we have not found a manifold with a similar contact normal for the contact point
|
|
if (!similarManifoldFound) {
|
|
|
|
// Create a new contact manifold for the overlapping pair
|
|
ContactManifoldInfo contactManifoldInfo(pairId, mMemoryManager.getPoolAllocator());
|
|
|
|
// Add the contact point to the manifold
|
|
contactManifoldInfo.potentialContactPointsIndices.add(contactPointIndex);
|
|
|
|
// If the overlapping pair contact does not exists yet
|
|
if (pairContact == nullptr) {
|
|
|
|
const Entity collider1Entity = narrowPhaseInfoBatch.colliderEntities1[i];
|
|
const Entity collider2Entity = narrowPhaseInfoBatch.colliderEntities2[i];
|
|
|
|
const Entity body1Entity = mCollidersComponents.getBody(collider1Entity);
|
|
const Entity body2Entity = mCollidersComponents.getBody(collider2Entity);
|
|
|
|
assert(!mWorld->mCollisionBodyComponents.getIsEntityDisabled(body1Entity) || !mWorld->mCollisionBodyComponents.getIsEntityDisabled(body2Entity));
|
|
|
|
const uint newContactPairIndex = contactPairs->size();
|
|
ContactPair overlappingPairContact(pairId, body1Entity, body2Entity,
|
|
collider1Entity, collider2Entity,
|
|
newContactPairIndex, mMemoryManager.getHeapAllocator());
|
|
contactPairs->add(overlappingPairContact);
|
|
pairContact = &((*contactPairs)[newContactPairIndex]);
|
|
mapPairIdToContactPairIndex->add(Pair<uint64, uint>(pairId, newContactPairIndex));
|
|
|
|
auto itbodyContactPairs = mapBodyToContactPairs.find(body1Entity);
|
|
if (itbodyContactPairs != mapBodyToContactPairs.end()) {
|
|
itbodyContactPairs->second.add(newContactPairIndex);
|
|
}
|
|
else {
|
|
List<uint> contactPairs(mMemoryManager.getPoolAllocator(), 1);
|
|
contactPairs.add(newContactPairIndex);
|
|
mapBodyToContactPairs.add(Pair<Entity, List<uint>>(body1Entity, contactPairs));
|
|
}
|
|
itbodyContactPairs = mapBodyToContactPairs.find(body2Entity);
|
|
if (itbodyContactPairs != mapBodyToContactPairs.end()) {
|
|
itbodyContactPairs->second.add(newContactPairIndex);
|
|
}
|
|
else {
|
|
List<uint> contactPairs(mMemoryManager.getPoolAllocator(), 1);
|
|
contactPairs.add(newContactPairIndex);
|
|
mapBodyToContactPairs.add(Pair<Entity, List<uint>>(body2Entity, contactPairs));
|
|
}
|
|
}
|
|
|
|
assert(pairContact != nullptr);
|
|
|
|
// Add the potential contact manifold
|
|
uint contactManifoldIndex = static_cast<uint>(potentialContactManifolds.size());
|
|
potentialContactManifolds.add(contactManifoldInfo);
|
|
|
|
// Add the contact manifold to the overlapping pair contact
|
|
assert(pairContact->pairId == contactManifoldInfo.pairId);
|
|
pairContact->potentialContactManifoldsIndices.add(contactManifoldIndex);
|
|
}
|
|
|
|
assert(pairContact->potentialContactManifoldsIndices.size() > 0);
|
|
}
|
|
|
|
narrowPhaseInfoBatch.resetContactPoints(i);
|
|
}
|
|
}
|
|
|
|
// Clear the obsolete manifolds and contact points and reduce the number of contacts points of the remaining manifolds
|
|
void CollisionDetectionSystem::reducePotentialContactManifolds(List<ContactPair>* contactPairs,
|
|
List<ContactManifoldInfo>& potentialContactManifolds,
|
|
const List<ContactPointInfo>& potentialContactPoints) const {
|
|
|
|
RP3D_PROFILE("CollisionDetectionSystem::reducePotentialContactManifolds()", mProfiler);
|
|
|
|
// Reduce the number of potential contact manifolds in a contact pair
|
|
for (uint i=0; i < contactPairs->size(); i++) {
|
|
|
|
ContactPair& contactPair = (*contactPairs)[i];
|
|
|
|
assert(contactPair.potentialContactManifoldsIndices.size() > 0);
|
|
|
|
// While there are too many manifolds in the contact pair
|
|
while(contactPair.potentialContactManifoldsIndices.size() > mWorld->mConfig.nbMaxContactManifolds) {
|
|
|
|
// Look for a manifold with the smallest contact penetration depth.
|
|
decimal minDepth = DECIMAL_LARGEST;
|
|
int minDepthManifoldIndex = -1;
|
|
for (uint j=0; j < contactPair.potentialContactManifoldsIndices.size(); j++) {
|
|
|
|
ContactManifoldInfo& manifold = potentialContactManifolds[contactPair.potentialContactManifoldsIndices[j]];
|
|
|
|
// Get the largest contact point penetration depth of the manifold
|
|
const decimal depth = computePotentialManifoldLargestContactDepth(manifold, potentialContactPoints);
|
|
|
|
if (depth < minDepth) {
|
|
minDepth = depth;
|
|
minDepthManifoldIndex = static_cast<int>(j);
|
|
}
|
|
}
|
|
|
|
// Remove the non optimal manifold
|
|
assert(minDepthManifoldIndex >= 0);
|
|
contactPair.potentialContactManifoldsIndices.removeAt(minDepthManifoldIndex);
|
|
}
|
|
}
|
|
|
|
// Reduce the number of potential contact points in the manifolds
|
|
for (uint i=0; i < contactPairs->size(); i++) {
|
|
|
|
const ContactPair& pairContact = (*contactPairs)[i];
|
|
|
|
// For each potential contact manifold
|
|
for (uint j=0; j < pairContact.potentialContactManifoldsIndices.size(); j++) {
|
|
|
|
ContactManifoldInfo& manifold = potentialContactManifolds[pairContact.potentialContactManifoldsIndices[j]];
|
|
|
|
// If there are two many contact points in the manifold
|
|
if (manifold.potentialContactPointsIndices.size() > MAX_CONTACT_POINTS_IN_MANIFOLD) {
|
|
|
|
Entity collider1 = mOverlappingPairs.mColliders1[mOverlappingPairs.mMapPairIdToPairIndex[manifold.pairId]];
|
|
|
|
Transform shape1LocalToWorldTransoform = mCollidersComponents.getLocalToWorldTransform(collider1);
|
|
|
|
// Reduce the number of contact points in the manifold
|
|
reduceContactPoints(manifold, shape1LocalToWorldTransoform, potentialContactPoints);
|
|
}
|
|
|
|
assert(manifold.potentialContactPointsIndices.size() <= MAX_CONTACT_POINTS_IN_MANIFOLD);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Return the largest depth of all the contact points of a potential manifold
|
|
decimal CollisionDetectionSystem::computePotentialManifoldLargestContactDepth(const ContactManifoldInfo& manifold,
|
|
const List<ContactPointInfo>& potentialContactPoints) const {
|
|
|
|
decimal largestDepth = 0.0f;
|
|
|
|
assert(manifold.potentialContactPointsIndices.size() > 0);
|
|
|
|
for (uint i=0; i < manifold.potentialContactPointsIndices.size(); i++) {
|
|
decimal depth = potentialContactPoints[manifold.potentialContactPointsIndices[i]].penetrationDepth;
|
|
|
|
if (depth > largestDepth) {
|
|
largestDepth = depth;
|
|
}
|
|
}
|
|
|
|
return largestDepth;
|
|
}
|
|
|
|
// Reduce the number of contact points of a potential contact manifold
|
|
// This is based on the technique described by Dirk Gregorius in his
|
|
// "Contacts Creation" GDC presentation. This method will reduce the number of
|
|
// contact points to a maximum of 4 points (but it can be less).
|
|
void CollisionDetectionSystem::reduceContactPoints(ContactManifoldInfo& manifold, const Transform& shape1ToWorldTransform,
|
|
const List<ContactPointInfo>& potentialContactPoints) const {
|
|
|
|
assert(manifold.potentialContactPointsIndices.size() > MAX_CONTACT_POINTS_IN_MANIFOLD);
|
|
|
|
// The following algorithm only works to reduce to a maximum of 4 contact points
|
|
assert(MAX_CONTACT_POINTS_IN_MANIFOLD == 4);
|
|
|
|
// List of the candidate contact points indices in the manifold. Every time that we have found a
|
|
// point we want to keep, we will remove it from this list
|
|
List<uint> candidatePointsIndices(manifold.potentialContactPointsIndices);
|
|
|
|
int8 nbReducedPoints = 0;
|
|
|
|
uint pointsToKeepIndices[MAX_CONTACT_POINTS_IN_MANIFOLD];
|
|
for (int8 i=0; i<MAX_CONTACT_POINTS_IN_MANIFOLD; i++) {
|
|
pointsToKeepIndices[i] = 0;
|
|
}
|
|
|
|
// Compute the initial contact point we need to keep.
|
|
// The first point we keep is always the point in a given
|
|
// constant direction (in order to always have same contact points
|
|
// between frames for better stability)
|
|
|
|
const Transform worldToShape1Transform = shape1ToWorldTransform.getInverse();
|
|
|
|
// Compute the contact normal of the manifold (we use the first contact point)
|
|
// in the local-space of the first collision shape
|
|
const Vector3 contactNormalShape1Space = worldToShape1Transform.getOrientation() * potentialContactPoints[candidatePointsIndices[0]].normal;
|
|
|
|
// Compute a search direction
|
|
const Vector3 searchDirection(1, 1, 1);
|
|
decimal maxDotProduct = DECIMAL_SMALLEST;
|
|
uint elementIndexToKeep = 0;
|
|
for (uint i=0; i < candidatePointsIndices.size(); i++) {
|
|
|
|
const ContactPointInfo& element = potentialContactPoints[candidatePointsIndices[i]];
|
|
decimal dotProduct = searchDirection.dot(element.localPoint1);
|
|
if (dotProduct > maxDotProduct) {
|
|
maxDotProduct = dotProduct;
|
|
elementIndexToKeep = i;
|
|
nbReducedPoints = 1;
|
|
}
|
|
}
|
|
pointsToKeepIndices[0] = candidatePointsIndices[elementIndexToKeep];
|
|
candidatePointsIndices.removeAt(elementIndexToKeep);
|
|
assert(nbReducedPoints == 1);
|
|
|
|
// Compute the second contact point we need to keep.
|
|
// The second point we keep is the one farthest away from the first point.
|
|
|
|
decimal maxDistance = decimal(0.0);
|
|
elementIndexToKeep = 0;
|
|
for (uint i=0; i < candidatePointsIndices.size(); i++) {
|
|
|
|
const ContactPointInfo& element = potentialContactPoints[candidatePointsIndices[i]];
|
|
const ContactPointInfo& pointToKeep0 = potentialContactPoints[pointsToKeepIndices[0]];
|
|
|
|
assert(candidatePointsIndices[i] != pointsToKeepIndices[0]);
|
|
|
|
const decimal distance = (pointToKeep0.localPoint1 - element.localPoint1).lengthSquare();
|
|
if (distance >= maxDistance) {
|
|
maxDistance = distance;
|
|
elementIndexToKeep = i;
|
|
nbReducedPoints = 2;
|
|
}
|
|
|
|
}
|
|
pointsToKeepIndices[1] = candidatePointsIndices[elementIndexToKeep];
|
|
candidatePointsIndices.removeAt(elementIndexToKeep);
|
|
assert(nbReducedPoints == 2);
|
|
|
|
// Compute the third contact point we need to keep.
|
|
// The third point is the one producing the triangle with the larger area
|
|
// with first and second point.
|
|
|
|
// We compute the most positive or most negative triangle area (depending on winding)
|
|
uint thirdPointMaxAreaIndex = 0;
|
|
uint thirdPointMinAreaIndex = 0;
|
|
decimal minArea = decimal(0.0);
|
|
decimal maxArea = decimal(0.0);
|
|
bool isPreviousAreaPositive = true;
|
|
for (uint i=0; i < candidatePointsIndices.size(); i++) {
|
|
|
|
const ContactPointInfo& element = potentialContactPoints[candidatePointsIndices[i]];
|
|
const ContactPointInfo& pointToKeep0 = potentialContactPoints[pointsToKeepIndices[0]];
|
|
const ContactPointInfo& pointToKeep1 = potentialContactPoints[pointsToKeepIndices[1]];
|
|
|
|
assert(candidatePointsIndices[i] != pointsToKeepIndices[0]);
|
|
assert(candidatePointsIndices[i] != pointsToKeepIndices[1]);
|
|
|
|
const Vector3 newToFirst = pointToKeep0.localPoint1 - element.localPoint1;
|
|
const Vector3 newToSecond = pointToKeep1.localPoint1 - element.localPoint1;
|
|
|
|
// Compute the triangle area
|
|
decimal area = newToFirst.cross(newToSecond).dot(contactNormalShape1Space);
|
|
|
|
if (area >= maxArea) {
|
|
maxArea = area;
|
|
thirdPointMaxAreaIndex = i;
|
|
}
|
|
if (area <= minArea) {
|
|
minArea = area;
|
|
thirdPointMinAreaIndex = i;
|
|
}
|
|
}
|
|
assert(minArea <= decimal(0.0));
|
|
assert(maxArea >= decimal(0.0));
|
|
if (maxArea > (-minArea)) {
|
|
isPreviousAreaPositive = true;
|
|
pointsToKeepIndices[2] = candidatePointsIndices[thirdPointMaxAreaIndex];
|
|
candidatePointsIndices.removeAt(thirdPointMaxAreaIndex);
|
|
}
|
|
else {
|
|
isPreviousAreaPositive = false;
|
|
pointsToKeepIndices[2] = candidatePointsIndices[thirdPointMinAreaIndex];
|
|
candidatePointsIndices.removeAt(thirdPointMinAreaIndex);
|
|
}
|
|
nbReducedPoints = 3;
|
|
|
|
// Compute the 4th point by choosing the triangle that add the most
|
|
// triangle area to the previous triangle and has opposite sign area (opposite winding)
|
|
|
|
decimal largestArea = decimal(0.0); // Largest area (positive or negative)
|
|
elementIndexToKeep = 0;
|
|
nbReducedPoints = 4;
|
|
decimal area;
|
|
|
|
// For each remaining candidate points
|
|
for (uint i=0; i < candidatePointsIndices.size(); i++) {
|
|
|
|
const ContactPointInfo& element = potentialContactPoints[candidatePointsIndices[i]];
|
|
|
|
assert(candidatePointsIndices[i] != pointsToKeepIndices[0]);
|
|
assert(candidatePointsIndices[i] != pointsToKeepIndices[1]);
|
|
assert(candidatePointsIndices[i] != pointsToKeepIndices[2]);
|
|
|
|
// For each edge of the triangle made by the first three points
|
|
for (uint j=0; j<3; j++) {
|
|
|
|
uint edgeVertex1Index = j;
|
|
uint edgeVertex2Index = j < 2 ? j + 1 : 0;
|
|
|
|
const ContactPointInfo& pointToKeepEdgeV1 = potentialContactPoints[pointsToKeepIndices[edgeVertex1Index]];
|
|
const ContactPointInfo& pointToKeepEdgeV2 = potentialContactPoints[pointsToKeepIndices[edgeVertex2Index]];
|
|
|
|
const Vector3 newToFirst = pointToKeepEdgeV1.localPoint1 - element.localPoint1;
|
|
const Vector3 newToSecond = pointToKeepEdgeV2.localPoint1 - element.localPoint1;
|
|
|
|
// Compute the triangle area
|
|
area = newToFirst.cross(newToSecond).dot(contactNormalShape1Space);
|
|
|
|
// We are looking at the triangle with maximal area (positive or negative).
|
|
// If the previous area is positive, we are looking at negative area now.
|
|
// If the previous area is negative, we are looking at the positive area now.
|
|
if (isPreviousAreaPositive && area <= largestArea) {
|
|
largestArea = area;
|
|
elementIndexToKeep = i;
|
|
}
|
|
else if (!isPreviousAreaPositive && area >= largestArea) {
|
|
largestArea = area;
|
|
elementIndexToKeep = i;
|
|
}
|
|
}
|
|
}
|
|
pointsToKeepIndices[3] = candidatePointsIndices[elementIndexToKeep];
|
|
candidatePointsIndices.removeAt(elementIndexToKeep);
|
|
|
|
// Only keep the four selected contact points in the manifold
|
|
manifold.potentialContactPointsIndices.clear();
|
|
manifold.potentialContactPointsIndices.add(pointsToKeepIndices[0]);
|
|
manifold.potentialContactPointsIndices.add(pointsToKeepIndices[1]);
|
|
manifold.potentialContactPointsIndices.add(pointsToKeepIndices[2]);
|
|
manifold.potentialContactPointsIndices.add(pointsToKeepIndices[3]);
|
|
}
|
|
|
|
// Report contacts
|
|
void CollisionDetectionSystem::reportContacts() {
|
|
|
|
if (mWorld->mEventListener != nullptr) {
|
|
reportContacts(*(mWorld->mEventListener), mCurrentContactPairs, mCurrentContactManifolds,
|
|
mCurrentContactPoints);
|
|
}
|
|
}
|
|
|
|
// Report all contacts
|
|
void CollisionDetectionSystem::reportContacts(CollisionCallback& callback, List<ContactPair>* contactPairs,
|
|
List<ContactManifold>* manifolds, List<ContactPoint>* contactPoints) {
|
|
|
|
RP3D_PROFILE("CollisionDetectionSystem::reportContacts()", mProfiler);
|
|
|
|
// If there are contacts
|
|
if (contactPairs->size() > 0) {
|
|
|
|
CollisionCallback::CallbackData callbackData(contactPairs, manifolds, contactPoints, *mWorld);
|
|
|
|
// Call the callback method to report the contacts
|
|
callback.onContact(callbackData);
|
|
}
|
|
}
|
|
|
|
// Return true if two bodies overlap (collide)
|
|
bool CollisionDetectionSystem::testOverlap(CollisionBody* body1, CollisionBody* body2) {
|
|
|
|
NarrowPhaseInput narrowPhaseInput(mMemoryManager.getPoolAllocator(), mOverlappingPairs);
|
|
|
|
// Compute the broad-phase collision detection
|
|
computeBroadPhase();
|
|
|
|
// Filter the overlapping pairs to get only the ones with the selected body involved
|
|
List<uint64> convexPairs(mMemoryManager.getPoolAllocator());
|
|
List<uint64> concavePairs(mMemoryManager.getPoolAllocator());
|
|
filterOverlappingPairs(body1->getEntity(), body2->getEntity(), convexPairs, concavePairs);
|
|
|
|
if (convexPairs.size() > 0 || concavePairs.size() > 0) {
|
|
|
|
// Compute the middle-phase collision detection
|
|
computeMiddlePhaseCollisionSnapshot(convexPairs, concavePairs, narrowPhaseInput);
|
|
|
|
// Compute the narrow-phase collision detection
|
|
return computeNarrowPhaseOverlapSnapshot(narrowPhaseInput, nullptr);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
// Report all the bodies that overlap (collide) in the world
|
|
void CollisionDetectionSystem::testOverlap(OverlapCallback& callback) {
|
|
|
|
NarrowPhaseInput narrowPhaseInput(mMemoryManager.getPoolAllocator(), mOverlappingPairs);
|
|
|
|
// Compute the broad-phase collision detection
|
|
computeBroadPhase();
|
|
|
|
// Compute the middle-phase collision detection
|
|
computeMiddlePhase(narrowPhaseInput);
|
|
|
|
// Compute the narrow-phase collision detection and report overlapping shapes
|
|
computeNarrowPhaseOverlapSnapshot(narrowPhaseInput, &callback);
|
|
}
|
|
|
|
// Report all the bodies that overlap (collide) with the body in parameter
|
|
void CollisionDetectionSystem::testOverlap(CollisionBody* body, OverlapCallback& callback) {
|
|
|
|
NarrowPhaseInput narrowPhaseInput(mMemoryManager.getPoolAllocator(), mOverlappingPairs);
|
|
|
|
// Compute the broad-phase collision detection
|
|
computeBroadPhase();
|
|
|
|
// Filter the overlapping pairs to get only the ones with the selected body involved
|
|
List<uint64> convexPairs(mMemoryManager.getPoolAllocator());
|
|
List<uint64> concavePairs(mMemoryManager.getPoolAllocator());
|
|
filterOverlappingPairs(body->getEntity(), convexPairs, concavePairs);
|
|
|
|
if (convexPairs.size() > 0 || concavePairs.size() > 0) {
|
|
|
|
// Compute the middle-phase collision detection
|
|
computeMiddlePhaseCollisionSnapshot(convexPairs, concavePairs, narrowPhaseInput);
|
|
|
|
// Compute the narrow-phase collision detection
|
|
computeNarrowPhaseOverlapSnapshot(narrowPhaseInput, &callback);
|
|
}
|
|
}
|
|
|
|
// Test collision and report contacts between two bodies.
|
|
void CollisionDetectionSystem::testCollision(CollisionBody* body1, CollisionBody* body2, CollisionCallback& callback) {
|
|
|
|
NarrowPhaseInput narrowPhaseInput(mMemoryManager.getPoolAllocator(), mOverlappingPairs);
|
|
|
|
// Compute the broad-phase collision detection
|
|
computeBroadPhase();
|
|
|
|
// Filter the overlapping pairs to get only the ones with the selected body involved
|
|
List<uint64> convexPairs(mMemoryManager.getPoolAllocator());
|
|
List<uint64> concavePairs(mMemoryManager.getPoolAllocator());
|
|
filterOverlappingPairs(body1->getEntity(), body2->getEntity(), convexPairs, concavePairs);
|
|
|
|
if (convexPairs.size() > 0 || concavePairs.size() > 0) {
|
|
|
|
// Compute the middle-phase collision detection
|
|
computeMiddlePhaseCollisionSnapshot(convexPairs, concavePairs, narrowPhaseInput);
|
|
|
|
// Compute the narrow-phase collision detection and report contacts
|
|
computeNarrowPhaseCollisionSnapshot(narrowPhaseInput, callback);
|
|
}
|
|
}
|
|
|
|
// Test collision and report all the contacts involving the body in parameter
|
|
void CollisionDetectionSystem::testCollision(CollisionBody* body, CollisionCallback& callback) {
|
|
|
|
NarrowPhaseInput narrowPhaseInput(mMemoryManager.getPoolAllocator(), mOverlappingPairs);
|
|
|
|
// Compute the broad-phase collision detection
|
|
computeBroadPhase();
|
|
|
|
// Filter the overlapping pairs to get only the ones with the selected body involved
|
|
List<uint64> convexPairs(mMemoryManager.getPoolAllocator());
|
|
List<uint64> concavePairs(mMemoryManager.getPoolAllocator());
|
|
filterOverlappingPairs(body->getEntity(), convexPairs, concavePairs);
|
|
|
|
if (convexPairs.size() > 0 || concavePairs.size() > 0) {
|
|
|
|
// Compute the middle-phase collision detection
|
|
computeMiddlePhaseCollisionSnapshot(convexPairs, concavePairs, narrowPhaseInput);
|
|
|
|
// Compute the narrow-phase collision detection and report contacts
|
|
computeNarrowPhaseCollisionSnapshot(narrowPhaseInput, callback);
|
|
}
|
|
}
|
|
|
|
// Test collision and report contacts between each colliding bodies in the world
|
|
void CollisionDetectionSystem::testCollision(CollisionCallback& callback) {
|
|
|
|
NarrowPhaseInput narrowPhaseInput(mMemoryManager.getPoolAllocator(), mOverlappingPairs);
|
|
|
|
// Compute the broad-phase collision detection
|
|
computeBroadPhase();
|
|
|
|
// Compute the middle-phase collision detection
|
|
computeMiddlePhase(narrowPhaseInput);
|
|
|
|
// Compute the narrow-phase collision detection and report contacts
|
|
computeNarrowPhaseCollisionSnapshot(narrowPhaseInput, callback);
|
|
}
|
|
|
|
// Filter the overlapping pairs to keep only the pairs where a given body is involved
|
|
void CollisionDetectionSystem::filterOverlappingPairs(Entity bodyEntity, List<uint64>& convexPairs, List<uint64>& concavePairs) const {
|
|
|
|
// For each possible collision pair of bodies
|
|
for (uint i=0; i < mOverlappingPairs.getNbPairs(); i++) {
|
|
|
|
if (mCollidersComponents.getBody(mOverlappingPairs.mColliders1[i]) == bodyEntity ||
|
|
mCollidersComponents.getBody(mOverlappingPairs.mColliders2[i]) == bodyEntity) {
|
|
|
|
if (i < mOverlappingPairs.getNbConvexVsConvexPairs()) {
|
|
convexPairs.add(mOverlappingPairs.mPairIds[i]);
|
|
}
|
|
else {
|
|
concavePairs.add(mOverlappingPairs.mPairIds[i]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Filter the overlapping pairs to keep only the pairs where two given bodies are involved
|
|
void CollisionDetectionSystem::filterOverlappingPairs(Entity body1Entity, Entity body2Entity, List<uint64>& convexPairs, List<uint64>& concavePairs) const {
|
|
|
|
// TODO : Do not go through all the overlapping pairs here but get all the involded overlapping pairs directly from the bodies
|
|
|
|
// For each possible collision pair of bodies
|
|
for (uint i=0; i < mOverlappingPairs.getNbPairs(); i++) {
|
|
|
|
const Entity collider1Body = mCollidersComponents.getBody(mOverlappingPairs.mColliders1[i]);
|
|
const Entity collider2Body = mCollidersComponents.getBody(mOverlappingPairs.mColliders2[i]);
|
|
|
|
if ((collider1Body == body1Entity && collider2Body == body2Entity) ||
|
|
(collider1Body == body2Entity && collider2Body == body1Entity)) {
|
|
|
|
if (i < mOverlappingPairs.getNbConvexVsConvexPairs()) {
|
|
convexPairs.add(mOverlappingPairs.mPairIds[i]);
|
|
}
|
|
else {
|
|
concavePairs.add(mOverlappingPairs.mPairIds[i]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Return the world event listener
|
|
EventListener* CollisionDetectionSystem::getWorldEventListener() {
|
|
return mWorld->mEventListener;
|
|
}
|
|
|
|
// Return the world-space AABB of a given collider
|
|
const AABB CollisionDetectionSystem::getWorldAABB(const Collider* collider) const {
|
|
assert(collider->getBroadPhaseId() > -1);
|
|
return mBroadPhaseSystem.getFatAABB(collider->getBroadPhaseId());
|
|
}
|