|
|
@ -21,6 +21,7 @@
|
|
|
|
#include <components/misc/constants.hpp>
|
|
|
|
#include <components/misc/constants.hpp>
|
|
|
|
#include <components/sceneutil/positionattitudetransform.hpp>
|
|
|
|
#include <components/sceneutil/positionattitudetransform.hpp>
|
|
|
|
#include <components/sceneutil/unrefqueue.hpp>
|
|
|
|
#include <components/sceneutil/unrefqueue.hpp>
|
|
|
|
|
|
|
|
#include <components/misc/convert.hpp>
|
|
|
|
|
|
|
|
|
|
|
|
#include <components/nifosg/particle.hpp> // FindRecIndexVisitor
|
|
|
|
#include <components/nifosg/particle.hpp> // FindRecIndexVisitor
|
|
|
|
|
|
|
|
|
|
|
@ -40,7 +41,6 @@
|
|
|
|
|
|
|
|
|
|
|
|
#include "collisiontype.hpp"
|
|
|
|
#include "collisiontype.hpp"
|
|
|
|
#include "actor.hpp"
|
|
|
|
#include "actor.hpp"
|
|
|
|
#include "convert.hpp"
|
|
|
|
|
|
|
|
#include "trace.h"
|
|
|
|
#include "trace.h"
|
|
|
|
#include "object.hpp"
|
|
|
|
#include "object.hpp"
|
|
|
|
#include "heightfield.hpp"
|
|
|
|
#include "heightfield.hpp"
|
|
|
@ -243,7 +243,7 @@ namespace MWPhysics
|
|
|
|
// Check if we actually found a valid spawn point (use an infinitely thin ray this time).
|
|
|
|
// Check if we actually found a valid spawn point (use an infinitely thin ray this time).
|
|
|
|
// Required for some broken door destinations in Morrowind.esm, where the spawn point
|
|
|
|
// Required for some broken door destinations in Morrowind.esm, where the spawn point
|
|
|
|
// intersects with other geometry if the actor's base is taken into account
|
|
|
|
// intersects with other geometry if the actor's base is taken into account
|
|
|
|
btVector3 from = toBullet(position);
|
|
|
|
btVector3 from = Misc::Convert::toBullet(position);
|
|
|
|
btVector3 to = from - btVector3(0,0,maxHeight);
|
|
|
|
btVector3 to = from - btVector3(0,0,maxHeight);
|
|
|
|
|
|
|
|
|
|
|
|
btCollisionWorld::ClosestRayResultCallback resultCallback1(from, to);
|
|
|
|
btCollisionWorld::ClosestRayResultCallback resultCallback1(from, to);
|
|
|
@ -253,11 +253,11 @@ namespace MWPhysics
|
|
|
|
collisionWorld->rayTest(from, to, resultCallback1);
|
|
|
|
collisionWorld->rayTest(from, to, resultCallback1);
|
|
|
|
|
|
|
|
|
|
|
|
if (resultCallback1.hasHit() &&
|
|
|
|
if (resultCallback1.hasHit() &&
|
|
|
|
( (toOsg(resultCallback1.m_hitPointWorld) - (tracer.mEndPos-offset)).length2() > 35*35
|
|
|
|
( (Misc::Convert::toOsg(resultCallback1.m_hitPointWorld) - (tracer.mEndPos-offset)).length2() > 35*35
|
|
|
|
|| !isWalkableSlope(tracer.mPlaneNormal)))
|
|
|
|
|| !isWalkableSlope(tracer.mPlaneNormal)))
|
|
|
|
{
|
|
|
|
{
|
|
|
|
actor->setOnSlope(!isWalkableSlope(resultCallback1.m_hitNormalWorld));
|
|
|
|
actor->setOnSlope(!isWalkableSlope(resultCallback1.m_hitNormalWorld));
|
|
|
|
return toOsg(resultCallback1.m_hitPointWorld) + osg::Vec3f(0.f, 0.f, sGroundOffset);
|
|
|
|
return Misc::Convert::toOsg(resultCallback1.m_hitPointWorld) + osg::Vec3f(0.f, 0.f, sGroundOffset);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
else
|
|
|
|
{
|
|
|
|
{
|
|
|
@ -696,7 +696,7 @@ namespace MWPhysics
|
|
|
|
|
|
|
|
|
|
|
|
btCollisionObject object;
|
|
|
|
btCollisionObject object;
|
|
|
|
object.setCollisionShape(&shape);
|
|
|
|
object.setCollisionShape(&shape);
|
|
|
|
object.setWorldTransform(btTransform(toBullet(orient), toBullet(center)));
|
|
|
|
object.setWorldTransform(btTransform(Misc::Convert::toBullet(orient), Misc::Convert::toBullet(center)));
|
|
|
|
|
|
|
|
|
|
|
|
const btCollisionObject* me = nullptr;
|
|
|
|
const btCollisionObject* me = nullptr;
|
|
|
|
std::vector<const btCollisionObject*> targetCollisionObjects;
|
|
|
|
std::vector<const btCollisionObject*> targetCollisionObjects;
|
|
|
@ -715,7 +715,7 @@ namespace MWPhysics
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DeepestNotMeContactTestResultCallback resultCallback(me, targetCollisionObjects, toBullet(origin));
|
|
|
|
DeepestNotMeContactTestResultCallback resultCallback(me, targetCollisionObjects, Misc::Convert::toBullet(origin));
|
|
|
|
resultCallback.m_collisionFilterGroup = CollisionType_Actor;
|
|
|
|
resultCallback.m_collisionFilterGroup = CollisionType_Actor;
|
|
|
|
resultCallback.m_collisionFilterMask = CollisionType_World | CollisionType_Door | CollisionType_HeightMap | CollisionType_Actor;
|
|
|
|
resultCallback.m_collisionFilterMask = CollisionType_World | CollisionType_Door | CollisionType_HeightMap | CollisionType_Actor;
|
|
|
|
mCollisionWorld->contactTest(&object, resultCallback);
|
|
|
|
mCollisionWorld->contactTest(&object, resultCallback);
|
|
|
@ -724,7 +724,7 @@ namespace MWPhysics
|
|
|
|
{
|
|
|
|
{
|
|
|
|
PtrHolder* holder = static_cast<PtrHolder*>(resultCallback.mObject->getUserPointer());
|
|
|
|
PtrHolder* holder = static_cast<PtrHolder*>(resultCallback.mObject->getUserPointer());
|
|
|
|
if (holder)
|
|
|
|
if (holder)
|
|
|
|
return std::make_pair(holder->getPtr(), toOsg(resultCallback.mContactPoint));
|
|
|
|
return std::make_pair(holder->getPtr(), Misc::Convert::toOsg(resultCallback.mContactPoint));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return std::make_pair(MWWorld::Ptr(), osg::Vec3f());
|
|
|
|
return std::make_pair(MWWorld::Ptr(), osg::Vec3f());
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -740,7 +740,7 @@ namespace MWPhysics
|
|
|
|
|
|
|
|
|
|
|
|
btTransform rayFrom;
|
|
|
|
btTransform rayFrom;
|
|
|
|
rayFrom.setIdentity();
|
|
|
|
rayFrom.setIdentity();
|
|
|
|
rayFrom.setOrigin(toBullet(point));
|
|
|
|
rayFrom.setOrigin(Misc::Convert::toBullet(point));
|
|
|
|
|
|
|
|
|
|
|
|
// target the collision object's world origin, this should be the center of the collision object
|
|
|
|
// target the collision object's world origin, this should be the center of the collision object
|
|
|
|
btTransform rayTo;
|
|
|
|
btTransform rayTo;
|
|
|
@ -756,7 +756,7 @@ namespace MWPhysics
|
|
|
|
return 0.f;
|
|
|
|
return 0.f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
else
|
|
|
|
return (point - toOsg(cb.m_hitPointWorld)).length();
|
|
|
|
return (point - Misc::Convert::toOsg(cb.m_hitPointWorld)).length();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class ClosestNotMeRayResultCallback : public btCollisionWorld::ClosestRayResultCallback
|
|
|
|
class ClosestNotMeRayResultCallback : public btCollisionWorld::ClosestRayResultCallback
|
|
|
@ -790,8 +790,8 @@ namespace MWPhysics
|
|
|
|
|
|
|
|
|
|
|
|
PhysicsSystem::RayResult PhysicsSystem::castRay(const osg::Vec3f &from, const osg::Vec3f &to, const MWWorld::ConstPtr& ignore, std::vector<MWWorld::Ptr> targets, int mask, int group) const
|
|
|
|
PhysicsSystem::RayResult PhysicsSystem::castRay(const osg::Vec3f &from, const osg::Vec3f &to, const MWWorld::ConstPtr& ignore, std::vector<MWWorld::Ptr> targets, int mask, int group) const
|
|
|
|
{
|
|
|
|
{
|
|
|
|
btVector3 btFrom = toBullet(from);
|
|
|
|
btVector3 btFrom = Misc::Convert::toBullet(from);
|
|
|
|
btVector3 btTo = toBullet(to);
|
|
|
|
btVector3 btTo = Misc::Convert::toBullet(to);
|
|
|
|
|
|
|
|
|
|
|
|
const btCollisionObject* me = nullptr;
|
|
|
|
const btCollisionObject* me = nullptr;
|
|
|
|
std::vector<const btCollisionObject*> targetCollisionObjects;
|
|
|
|
std::vector<const btCollisionObject*> targetCollisionObjects;
|
|
|
@ -829,8 +829,8 @@ namespace MWPhysics
|
|
|
|
result.mHit = resultCallback.hasHit();
|
|
|
|
result.mHit = resultCallback.hasHit();
|
|
|
|
if (resultCallback.hasHit())
|
|
|
|
if (resultCallback.hasHit())
|
|
|
|
{
|
|
|
|
{
|
|
|
|
result.mHitPos = toOsg(resultCallback.m_hitPointWorld);
|
|
|
|
result.mHitPos = Misc::Convert::toOsg(resultCallback.m_hitPointWorld);
|
|
|
|
result.mHitNormal = toOsg(resultCallback.m_hitNormalWorld);
|
|
|
|
result.mHitNormal = Misc::Convert::toOsg(resultCallback.m_hitNormalWorld);
|
|
|
|
if (PtrHolder* ptrHolder = static_cast<PtrHolder*>(resultCallback.m_collisionObject->getUserPointer()))
|
|
|
|
if (PtrHolder* ptrHolder = static_cast<PtrHolder*>(resultCallback.m_collisionObject->getUserPointer()))
|
|
|
|
result.mHitObject = ptrHolder->getPtr();
|
|
|
|
result.mHitObject = ptrHolder->getPtr();
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -839,15 +839,15 @@ namespace MWPhysics
|
|
|
|
|
|
|
|
|
|
|
|
PhysicsSystem::RayResult PhysicsSystem::castSphere(const osg::Vec3f &from, const osg::Vec3f &to, float radius)
|
|
|
|
PhysicsSystem::RayResult PhysicsSystem::castSphere(const osg::Vec3f &from, const osg::Vec3f &to, float radius)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
btCollisionWorld::ClosestConvexResultCallback callback(toBullet(from), toBullet(to));
|
|
|
|
btCollisionWorld::ClosestConvexResultCallback callback(Misc::Convert::toBullet(from), Misc::Convert::toBullet(to));
|
|
|
|
callback.m_collisionFilterGroup = 0xff;
|
|
|
|
callback.m_collisionFilterGroup = 0xff;
|
|
|
|
callback.m_collisionFilterMask = CollisionType_World|CollisionType_HeightMap|CollisionType_Door;
|
|
|
|
callback.m_collisionFilterMask = CollisionType_World|CollisionType_HeightMap|CollisionType_Door;
|
|
|
|
|
|
|
|
|
|
|
|
btSphereShape shape(radius);
|
|
|
|
btSphereShape shape(radius);
|
|
|
|
const btQuaternion btrot = btQuaternion::getIdentity();
|
|
|
|
const btQuaternion btrot = btQuaternion::getIdentity();
|
|
|
|
|
|
|
|
|
|
|
|
btTransform from_ (btrot, toBullet(from));
|
|
|
|
btTransform from_ (btrot, Misc::Convert::toBullet(from));
|
|
|
|
btTransform to_ (btrot, toBullet(to));
|
|
|
|
btTransform to_ (btrot, Misc::Convert::toBullet(to));
|
|
|
|
|
|
|
|
|
|
|
|
mCollisionWorld->convexSweepTest(&shape, from_, to_, callback);
|
|
|
|
mCollisionWorld->convexSweepTest(&shape, from_, to_, callback);
|
|
|
|
|
|
|
|
|
|
|
@ -855,8 +855,8 @@ namespace MWPhysics
|
|
|
|
result.mHit = callback.hasHit();
|
|
|
|
result.mHit = callback.hasHit();
|
|
|
|
if (result.mHit)
|
|
|
|
if (result.mHit)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
result.mHitPos = toOsg(callback.m_hitPointWorld);
|
|
|
|
result.mHitPos = Misc::Convert::toOsg(callback.m_hitPointWorld);
|
|
|
|
result.mHitNormal = toOsg(callback.m_hitNormalWorld);
|
|
|
|
result.mHitNormal = Misc::Convert::toOsg(callback.m_hitNormalWorld);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -1131,7 +1131,7 @@ namespace MWPhysics
|
|
|
|
ObjectMap::iterator found = mObjects.find(ptr);
|
|
|
|
ObjectMap::iterator found = mObjects.find(ptr);
|
|
|
|
if (found != mObjects.end())
|
|
|
|
if (found != mObjects.end())
|
|
|
|
{
|
|
|
|
{
|
|
|
|
found->second->setRotation(toBullet(ptr.getRefData().getBaseNode()->getAttitude()));
|
|
|
|
found->second->setRotation(Misc::Convert::toBullet(ptr.getRefData().getBaseNode()->getAttitude()));
|
|
|
|
mCollisionWorld->updateSingleAabb(found->second->getCollisionObject());
|
|
|
|
mCollisionWorld->updateSingleAabb(found->second->getCollisionObject());
|
|
|
|
return;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -1152,7 +1152,7 @@ namespace MWPhysics
|
|
|
|
ObjectMap::iterator found = mObjects.find(ptr);
|
|
|
|
ObjectMap::iterator found = mObjects.find(ptr);
|
|
|
|
if (found != mObjects.end())
|
|
|
|
if (found != mObjects.end())
|
|
|
|
{
|
|
|
|
{
|
|
|
|
found->second->setOrigin(toBullet(ptr.getRefData().getPosition().asVec3()));
|
|
|
|
found->second->setOrigin(Misc::Convert::toBullet(ptr.getRefData().getPosition().asVec3()));
|
|
|
|
mCollisionWorld->updateSingleAabb(found->second->getCollisionObject());
|
|
|
|
mCollisionWorld->updateSingleAabb(found->second->getCollisionObject());
|
|
|
|
return;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|