2020-02-09 17:24:08 +00:00
|
|
|
#ifndef OPENMW_MWPHYSICS_HASSPHERECOLLISIONCALLBACK_H
|
|
|
|
#define OPENMW_MWPHYSICS_HASSPHERECOLLISIONCALLBACK_H
|
|
|
|
|
2022-08-17 16:44:04 +00:00
|
|
|
#include <BulletCollision/BroadphaseCollision/btBroadphaseInterface.h>
|
2020-02-09 17:24:08 +00:00
|
|
|
#include <BulletCollision/CollisionDispatch/btCollisionObject.h>
|
|
|
|
#include <LinearMath/btVector3.h>
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
namespace MWPhysics
|
|
|
|
{
|
|
|
|
// https://developer.mozilla.org/en-US/docs/Games/Techniques/3D_collision_detection
|
|
|
|
bool testAabbAgainstSphere(
|
|
|
|
const btVector3& aabbMin, const btVector3& aabbMax, const btVector3& position, const btScalar radius)
|
|
|
|
{
|
2021-11-06 04:30:28 +00:00
|
|
|
const btVector3 nearest(std::clamp(position.x(), aabbMin.x(), aabbMax.x()),
|
|
|
|
std::clamp(position.y(), aabbMin.y(), aabbMax.y()), std::clamp(position.z(), aabbMin.z(), aabbMax.z()));
|
2020-02-09 17:24:08 +00:00
|
|
|
return nearest.distance(position) < radius;
|
|
|
|
}
|
|
|
|
|
2022-04-11 17:30:54 +00:00
|
|
|
template <class Ignore, class OnCollision>
|
2020-02-09 17:24:08 +00:00
|
|
|
class HasSphereCollisionCallback final : public btBroadphaseAabbCallback
|
|
|
|
{
|
|
|
|
public:
|
2022-04-11 17:30:54 +00:00
|
|
|
HasSphereCollisionCallback(const btVector3& position, const btScalar radius, const int mask, const int group,
|
|
|
|
const Ignore& ignore, OnCollision* onCollision)
|
2020-02-09 17:24:08 +00:00
|
|
|
: mPosition(position)
|
|
|
|
, mRadius(radius)
|
2022-04-11 17:30:54 +00:00
|
|
|
, mIgnore(ignore)
|
2020-02-09 17:24:08 +00:00
|
|
|
, mCollisionFilterMask(mask)
|
2021-09-28 22:42:29 +00:00
|
|
|
, mCollisionFilterGroup(group)
|
|
|
|
, mOnCollision(onCollision)
|
2020-02-09 17:24:08 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2020-10-22 21:57:53 +00:00
|
|
|
bool process(const btBroadphaseProxy* proxy) override
|
2020-02-09 17:24:08 +00:00
|
|
|
{
|
2021-09-28 22:42:29 +00:00
|
|
|
if (mResult && mOnCollision == nullptr)
|
2020-02-09 17:24:08 +00:00
|
|
|
return false;
|
|
|
|
const auto collisionObject = static_cast<btCollisionObject*>(proxy->m_clientObject);
|
2022-04-11 17:30:54 +00:00
|
|
|
if (mIgnore(collisionObject) || !needsCollision(*proxy)
|
2021-09-28 22:42:29 +00:00
|
|
|
|| !testAabbAgainstSphere(proxy->m_aabbMin, proxy->m_aabbMax, mPosition, mRadius))
|
2020-02-09 17:24:08 +00:00
|
|
|
return true;
|
2021-09-28 22:42:29 +00:00
|
|
|
mResult = true;
|
|
|
|
if (mOnCollision != nullptr)
|
|
|
|
{
|
|
|
|
(*mOnCollision)(collisionObject);
|
|
|
|
return true;
|
|
|
|
}
|
2020-02-09 17:24:08 +00:00
|
|
|
return !mResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool getResult() const { return mResult; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
btVector3 mPosition;
|
|
|
|
btScalar mRadius;
|
2022-04-11 17:30:54 +00:00
|
|
|
Ignore mIgnore;
|
2020-02-09 17:24:08 +00:00
|
|
|
int mCollisionFilterMask;
|
|
|
|
int mCollisionFilterGroup;
|
2021-09-28 22:42:29 +00:00
|
|
|
OnCollision* mOnCollision;
|
2020-02-09 17:24:08 +00:00
|
|
|
bool mResult = false;
|
|
|
|
|
|
|
|
bool needsCollision(const btBroadphaseProxy& proxy) const
|
|
|
|
{
|
|
|
|
bool collides = (proxy.m_collisionFilterGroup & mCollisionFilterMask) != 0;
|
|
|
|
collides = collides && (mCollisionFilterGroup & proxy.m_collisionFilterMask);
|
|
|
|
return collides;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|