2015-05-12 01:02:15 +00:00
|
|
|
#include "actor.hpp"
|
|
|
|
|
2017-02-06 04:39:56 +00:00
|
|
|
#include <BulletCollision/CollisionShapes/btCapsuleShape.h>
|
2015-05-27 20:32:11 +00:00
|
|
|
#include <BulletCollision/CollisionShapes/btBoxShape.h>
|
2015-05-27 21:09:38 +00:00
|
|
|
#include <BulletCollision/CollisionDispatch/btCollisionWorld.h>
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2015-11-20 20:57:04 +00:00
|
|
|
#include <components/sceneutil/positionattitudetransform.hpp>
|
2015-11-16 22:30:10 +00:00
|
|
|
#include <components/resource/bulletshape.hpp>
|
2018-10-22 11:14:25 +00:00
|
|
|
#include <components/debug/debuglog.hpp>
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2015-05-27 20:32:11 +00:00
|
|
|
#include "../mwworld/class.hpp"
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
#include "convert.hpp"
|
|
|
|
#include "collisiontype.hpp"
|
|
|
|
|
|
|
|
namespace MWPhysics
|
|
|
|
{
|
|
|
|
|
|
|
|
|
2016-02-09 17:51:17 +00:00
|
|
|
Actor::Actor(const MWWorld::Ptr& ptr, osg::ref_ptr<const Resource::BulletShape> shape, btCollisionWorld* world)
|
2015-05-12 01:02:15 +00:00
|
|
|
: mCanWaterWalk(false), mWalkingOnWater(false)
|
2018-05-26 11:35:48 +00:00
|
|
|
, mCollisionObject(nullptr), mForce(0.f, 0.f, 0.f), mOnGround(true), mOnSlope(false)
|
2015-05-12 01:02:15 +00:00
|
|
|
, mInternalCollisionMode(true)
|
|
|
|
, mExternalCollisionMode(true)
|
2015-05-27 21:09:38 +00:00
|
|
|
, mCollisionWorld(world)
|
2015-05-12 01:02:15 +00:00
|
|
|
{
|
|
|
|
mPtr = ptr;
|
|
|
|
|
|
|
|
mHalfExtents = shape->mCollisionBoxHalfExtents;
|
|
|
|
mMeshTranslation = shape->mCollisionBoxTranslate;
|
|
|
|
|
2018-10-22 11:14:25 +00:00
|
|
|
// We can not create actor without collisions - he will fall through the ground.
|
|
|
|
// In this case we should autogenerate collision box based on mesh shape
|
|
|
|
// (NPCs have bodyparts and use a different approach)
|
|
|
|
if (!ptr.getClass().isNpc() && mHalfExtents.length2() == 0.f)
|
|
|
|
{
|
|
|
|
const Resource::BulletShape* collisionShape = shape.get();
|
|
|
|
if (collisionShape && collisionShape->mCollisionShape)
|
|
|
|
{
|
|
|
|
btTransform transform;
|
|
|
|
transform.setIdentity();
|
|
|
|
btVector3 min;
|
|
|
|
btVector3 max;
|
|
|
|
|
|
|
|
collisionShape->mCollisionShape->getAabb(transform, min, max);
|
|
|
|
mHalfExtents.x() = (max[0] - min[0])/2.f;
|
|
|
|
mHalfExtents.y() = (max[1] - min[1])/2.f;
|
|
|
|
mHalfExtents.z() = (max[2] - min[2])/2.f;
|
|
|
|
|
|
|
|
mMeshTranslation = osg::Vec3f(0.f, 0.f, mHalfExtents.z());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mHalfExtents.length2() == 0.f)
|
|
|
|
Log(Debug::Error) << "Error: Failed to calculate bounding box for actor \"" << ptr.getCellRef().getRefId() << "\".";
|
|
|
|
}
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
// Use capsule shape only if base is square (nonuniform scaling apparently doesn't work on it)
|
|
|
|
if (std::abs(mHalfExtents.x()-mHalfExtents.y())<mHalfExtents.x()*0.05 && mHalfExtents.z() >= mHalfExtents.x())
|
|
|
|
{
|
2017-02-06 04:39:56 +00:00
|
|
|
mShape.reset(new btCapsuleShapeZ(mHalfExtents.x(), 2*mHalfExtents.z() - 2*mHalfExtents.x()));
|
2017-02-23 21:34:42 +00:00
|
|
|
mRotationallyInvariant = true;
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
else
|
2017-02-23 21:34:42 +00:00
|
|
|
{
|
2015-05-12 01:02:15 +00:00
|
|
|
mShape.reset(new btBoxShape(toBullet(mHalfExtents)));
|
2017-02-23 21:34:42 +00:00
|
|
|
mRotationallyInvariant = false;
|
|
|
|
}
|
|
|
|
|
2017-02-10 00:58:27 +00:00
|
|
|
mConvexShape = static_cast<btConvexShape*>(mShape.get());
|
2015-05-12 01:02:15 +00:00
|
|
|
|
|
|
|
mCollisionObject.reset(new btCollisionObject);
|
|
|
|
mCollisionObject->setCollisionFlags(btCollisionObject::CF_KINEMATIC_OBJECT);
|
|
|
|
mCollisionObject->setActivationState(DISABLE_DEACTIVATION);
|
|
|
|
mCollisionObject->setCollisionShape(mShape.get());
|
|
|
|
mCollisionObject->setUserPointer(static_cast<PtrHolder*>(this));
|
|
|
|
|
|
|
|
updateRotation();
|
|
|
|
updateScale();
|
2016-02-13 01:56:41 +00:00
|
|
|
updatePosition();
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2016-12-16 19:22:07 +00:00
|
|
|
addCollisionMask(getCollisionMask());
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Actor::~Actor()
|
|
|
|
{
|
|
|
|
if (mCollisionObject.get())
|
2015-05-27 21:09:38 +00:00
|
|
|
mCollisionWorld->removeCollisionObject(mCollisionObject.get());
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::enableCollisionMode(bool collision)
|
|
|
|
{
|
|
|
|
mInternalCollisionMode = collision;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::enableCollisionBody(bool collision)
|
|
|
|
{
|
|
|
|
if (mExternalCollisionMode != collision)
|
|
|
|
{
|
|
|
|
mExternalCollisionMode = collision;
|
|
|
|
updateCollisionMask();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-16 19:22:07 +00:00
|
|
|
void Actor::addCollisionMask(int collisionMask)
|
|
|
|
{
|
|
|
|
mCollisionWorld->addCollisionObject(mCollisionObject.get(), CollisionType_Actor, collisionMask);
|
|
|
|
}
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
void Actor::updateCollisionMask()
|
|
|
|
{
|
2016-12-16 19:22:07 +00:00
|
|
|
mCollisionWorld->removeCollisionObject(mCollisionObject.get());
|
|
|
|
addCollisionMask(getCollisionMask());
|
|
|
|
}
|
|
|
|
|
|
|
|
int Actor::getCollisionMask()
|
|
|
|
{
|
2015-05-12 01:02:15 +00:00
|
|
|
int collisionMask = CollisionType_World | CollisionType_HeightMap;
|
|
|
|
if (mExternalCollisionMode)
|
2015-12-18 17:32:42 +00:00
|
|
|
collisionMask |= CollisionType_Actor | CollisionType_Projectile | CollisionType_Door;
|
2015-05-12 01:02:15 +00:00
|
|
|
if (mCanWaterWalk)
|
|
|
|
collisionMask |= CollisionType_Water;
|
2016-12-16 19:22:07 +00:00
|
|
|
return collisionMask;
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::updatePosition()
|
|
|
|
{
|
|
|
|
osg::Vec3f position = mPtr.getRefData().getPosition().asVec3();
|
|
|
|
|
2016-02-13 01:56:41 +00:00
|
|
|
mPosition = position;
|
|
|
|
mPreviousPosition = position;
|
|
|
|
|
|
|
|
updateCollisionObjectPosition();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::updateCollisionObjectPosition()
|
|
|
|
{
|
2015-05-12 01:02:15 +00:00
|
|
|
btTransform tr = mCollisionObject->getWorldTransform();
|
2015-05-12 14:49:21 +00:00
|
|
|
osg::Vec3f scaledTranslation = mRotation * osg::componentMultiply(mMeshTranslation, mScale);
|
2016-02-13 01:56:41 +00:00
|
|
|
osg::Vec3f newPosition = scaledTranslation + mPosition;
|
2015-05-12 01:02:15 +00:00
|
|
|
tr.setOrigin(toBullet(newPosition));
|
|
|
|
mCollisionObject->setWorldTransform(tr);
|
|
|
|
}
|
|
|
|
|
2016-02-13 01:56:41 +00:00
|
|
|
osg::Vec3f Actor::getCollisionObjectPosition() const
|
2015-11-03 17:15:47 +00:00
|
|
|
{
|
|
|
|
return toOsg(mCollisionObject->getWorldTransform().getOrigin());
|
|
|
|
}
|
|
|
|
|
2016-02-13 01:56:41 +00:00
|
|
|
void Actor::setPosition(const osg::Vec3f &position)
|
|
|
|
{
|
|
|
|
mPreviousPosition = mPosition;
|
|
|
|
|
|
|
|
mPosition = position;
|
|
|
|
updateCollisionObjectPosition();
|
|
|
|
}
|
|
|
|
|
|
|
|
osg::Vec3f Actor::getPosition() const
|
|
|
|
{
|
|
|
|
return mPosition;
|
|
|
|
}
|
|
|
|
|
|
|
|
osg::Vec3f Actor::getPreviousPosition() const
|
|
|
|
{
|
|
|
|
return mPreviousPosition;
|
|
|
|
}
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
void Actor::updateRotation ()
|
|
|
|
{
|
|
|
|
btTransform tr = mCollisionObject->getWorldTransform();
|
2015-05-12 14:49:21 +00:00
|
|
|
mRotation = mPtr.getRefData().getBaseNode()->getAttitude();
|
|
|
|
tr.setRotation(toBullet(mRotation));
|
2015-05-12 01:02:15 +00:00
|
|
|
mCollisionObject->setWorldTransform(tr);
|
2015-05-12 14:49:21 +00:00
|
|
|
|
2016-02-13 01:56:41 +00:00
|
|
|
updateCollisionObjectPosition();
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
|
2017-02-23 21:34:42 +00:00
|
|
|
bool Actor::isRotationallyInvariant() const
|
|
|
|
{
|
|
|
|
return mRotationallyInvariant;
|
|
|
|
}
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
void Actor::updateScale()
|
|
|
|
{
|
|
|
|
float scale = mPtr.getCellRef().getScale();
|
|
|
|
osg::Vec3f scaleVec(scale,scale,scale);
|
|
|
|
|
2015-11-01 20:45:58 +00:00
|
|
|
mPtr.getClass().adjustScale(mPtr, scaleVec, false);
|
2015-05-12 01:02:15 +00:00
|
|
|
mScale = scaleVec;
|
|
|
|
mShape->setLocalScaling(toBullet(mScale));
|
|
|
|
|
2015-11-01 20:45:58 +00:00
|
|
|
scaleVec = osg::Vec3f(scale,scale,scale);
|
|
|
|
mPtr.getClass().adjustScale(mPtr, scaleVec, true);
|
|
|
|
mRenderingScale = scaleVec;
|
|
|
|
|
2016-02-13 01:56:41 +00:00
|
|
|
updateCollisionObjectPosition();
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
osg::Vec3f Actor::getHalfExtents() const
|
|
|
|
{
|
|
|
|
return osg::componentMultiply(mHalfExtents, mScale);
|
|
|
|
}
|
|
|
|
|
2015-11-01 20:45:58 +00:00
|
|
|
osg::Vec3f Actor::getRenderingHalfExtents() const
|
|
|
|
{
|
|
|
|
return osg::componentMultiply(mHalfExtents, mRenderingScale);
|
|
|
|
}
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
void Actor::setInertialForce(const osg::Vec3f &force)
|
|
|
|
{
|
|
|
|
mForce = force;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::setOnGround(bool grounded)
|
|
|
|
{
|
|
|
|
mOnGround = grounded;
|
|
|
|
}
|
|
|
|
|
2017-02-06 03:46:44 +00:00
|
|
|
void Actor::setOnSlope(bool slope)
|
|
|
|
{
|
|
|
|
mOnSlope = slope;
|
|
|
|
}
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
bool Actor::isWalkingOnWater() const
|
|
|
|
{
|
|
|
|
return mWalkingOnWater;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::setWalkingOnWater(bool walkingOnWater)
|
|
|
|
{
|
|
|
|
mWalkingOnWater = walkingOnWater;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::setCanWaterWalk(bool waterWalk)
|
|
|
|
{
|
|
|
|
if (waterWalk != mCanWaterWalk)
|
|
|
|
{
|
|
|
|
mCanWaterWalk = waterWalk;
|
|
|
|
updateCollisionMask();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|