2021-01-07 10:02:53 +00:00
|
|
|
#include "actorconvexcallback.hpp"
|
2020-12-27 22:16:11 +00:00
|
|
|
#include "collisiontype.hpp"
|
|
|
|
#include "contacttestwrapper.h"
|
2020-03-30 21:25:33 +00:00
|
|
|
|
|
|
|
#include <BulletCollision/CollisionDispatch/btCollisionObject.h>
|
2020-12-27 22:16:11 +00:00
|
|
|
#include <components/misc/convert.hpp>
|
2020-03-30 21:25:33 +00:00
|
|
|
|
2019-02-13 07:30:16 +00:00
|
|
|
#include "projectile.hpp"
|
|
|
|
|
2020-03-30 21:25:33 +00:00
|
|
|
namespace MWPhysics
|
|
|
|
{
|
2024-02-02 21:16:49 +00:00
|
|
|
namespace
|
2020-12-27 22:16:11 +00:00
|
|
|
{
|
2024-02-02 21:16:49 +00:00
|
|
|
struct ActorOverlapTester : public btCollisionWorld::ContactResultCallback
|
2020-12-27 22:16:11 +00:00
|
|
|
{
|
2024-02-02 21:16:49 +00:00
|
|
|
bool mOverlapping = false;
|
2020-12-27 22:16:11 +00:00
|
|
|
|
2024-02-02 21:16:49 +00:00
|
|
|
btScalar addSingleResult(btManifoldPoint& cp, const btCollisionObjectWrapper* /*colObj0Wrap*/,
|
|
|
|
int /*partId0*/, int /*index0*/, const btCollisionObjectWrapper* /*colObj1Wrap*/, int /*partId1*/,
|
|
|
|
int /*index1*/) override
|
|
|
|
{
|
|
|
|
if (cp.getDistance() <= 0.0f)
|
|
|
|
mOverlapping = true;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
};
|
2020-03-30 21:25:33 +00:00
|
|
|
}
|
|
|
|
|
2021-01-07 10:02:53 +00:00
|
|
|
btScalar ActorConvexCallback::addSingleResult(
|
|
|
|
btCollisionWorld::LocalConvexResult& convexResult, bool normalInWorldSpace)
|
2020-03-30 21:25:33 +00:00
|
|
|
{
|
|
|
|
if (convexResult.m_hitCollisionObject == mMe)
|
2024-02-02 21:16:49 +00:00
|
|
|
return 1;
|
2020-03-30 21:25:33 +00:00
|
|
|
|
2020-12-27 22:16:11 +00:00
|
|
|
// override data for actor-actor collisions
|
|
|
|
// vanilla Morrowind seems to make overlapping actors collide as though they are both cylinders with a diameter
|
|
|
|
// of the distance between them For some reason this doesn't work as well as it should when using capsules, but
|
|
|
|
// it still helps a lot.
|
|
|
|
if (convexResult.m_hitCollisionObject->getBroadphaseHandle()->m_collisionFilterGroup == CollisionType_Actor)
|
|
|
|
{
|
|
|
|
ActorOverlapTester isOverlapping;
|
|
|
|
// FIXME: This is absolutely terrible and bullet should feel terrible for not making contactPairTest
|
|
|
|
// const-correct.
|
|
|
|
ContactTestWrapper::contactPairTest(const_cast<btCollisionWorld*>(mWorld),
|
|
|
|
const_cast<btCollisionObject*>(mMe), const_cast<btCollisionObject*>(convexResult.m_hitCollisionObject),
|
|
|
|
isOverlapping);
|
|
|
|
|
2024-02-02 21:16:49 +00:00
|
|
|
if (isOverlapping.mOverlapping)
|
2020-12-27 22:16:11 +00:00
|
|
|
{
|
|
|
|
auto originA = Misc::Convert::toOsg(mMe->getWorldTransform().getOrigin());
|
|
|
|
auto originB = Misc::Convert::toOsg(convexResult.m_hitCollisionObject->getWorldTransform().getOrigin());
|
|
|
|
osg::Vec3f motion = Misc::Convert::toOsg(mMotion);
|
|
|
|
osg::Vec3f normal = (originA - originB);
|
|
|
|
normal.z() = 0;
|
|
|
|
normal.normalize();
|
|
|
|
// only collide if horizontally moving towards the hit actor (note: the motion vector appears to be
|
|
|
|
// inverted)
|
|
|
|
// FIXME: This kinda screws with standing on actors that walk up slopes for some reason. Makes you fall
|
|
|
|
// through them. It happens in vanilla Morrowind too, but much less often. I tried hunting down why but
|
|
|
|
// couldn't figure it out. Possibly a stair stepping or ground ejection bug.
|
|
|
|
if (normal * motion > 0.0f)
|
|
|
|
{
|
|
|
|
convexResult.m_hitFraction = 0.0f;
|
|
|
|
convexResult.m_hitNormalLocal = Misc::Convert::toBullet(normal);
|
|
|
|
return ClosestConvexResultCallback::addSingleResult(convexResult, true);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-02-02 21:16:49 +00:00
|
|
|
return 1;
|
2020-12-27 22:16:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-02-13 07:30:16 +00:00
|
|
|
if (convexResult.m_hitCollisionObject->getBroadphaseHandle()->m_collisionFilterGroup
|
|
|
|
== CollisionType_Projectile)
|
|
|
|
{
|
2020-11-21 15:26:45 +00:00
|
|
|
auto* projectileHolder = static_cast<Projectile*>(convexResult.m_hitCollisionObject->getUserPointer());
|
2020-10-23 18:27:07 +00:00
|
|
|
if (!projectileHolder->isActive())
|
2024-02-02 21:16:49 +00:00
|
|
|
return 1;
|
2021-08-05 08:55:19 +00:00
|
|
|
if (projectileHolder->isValidTarget(mMe))
|
|
|
|
projectileHolder->hit(mMe, convexResult.m_hitPointLocal, convexResult.m_hitNormalLocal);
|
2024-02-02 21:16:49 +00:00
|
|
|
return 1;
|
2019-02-13 07:30:16 +00:00
|
|
|
}
|
|
|
|
|
2020-03-30 21:25:33 +00:00
|
|
|
btVector3 hitNormalWorld;
|
|
|
|
if (normalInWorldSpace)
|
|
|
|
hitNormalWorld = convexResult.m_hitNormalLocal;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/// need to transform normal into worldspace
|
|
|
|
hitNormalWorld
|
|
|
|
= convexResult.m_hitCollisionObject->getWorldTransform().getBasis() * convexResult.m_hitNormalLocal;
|
|
|
|
}
|
|
|
|
|
|
|
|
// dot product of the motion vector against the collision contact normal
|
|
|
|
btScalar dotCollision = mMotion.dot(hitNormalWorld);
|
|
|
|
if (dotCollision <= mMinCollisionDot)
|
2024-02-02 21:16:49 +00:00
|
|
|
return 1;
|
2020-03-30 21:25:33 +00:00
|
|
|
|
|
|
|
return ClosestConvexResultCallback::addSingleResult(convexResult, normalInWorldSpace);
|
|
|
|
}
|
|
|
|
}
|