2015-05-12 01:02:15 +00:00
|
|
|
#ifndef OPENMW_MWPHYSICS_ACTOR_H
|
|
|
|
#define OPENMW_MWPHYSICS_ACTOR_H
|
|
|
|
|
|
|
|
#include <memory>
|
Process movement queue in one or several background threads
Before movement calculation, the main thread prepare a
vector of ActorFrameData, which contains all data necessary to perform
the simulation, and feed it to the solver. At the same time it fetches
the result from the previous background simulation, which in turn is
used by the game mechanics.
Other functions of the physics system (weapon hit for instance)
interrupt the background simulation, with some exceptions described
below.
The number of threads is controlled by the numeric setting
[Physics]
async num threads
In case 'async num threads' > 1 and Bullet doesn't support multiple threads,
1 async thread will be used. 0 means synchronous solver.
Additional settings (will be silently switched off if async num threads = 0)
[Physics]
defer aabb update
Update AABBs of actors and objects in the background thread(s). It is not an especially
costly operation, but it needs exclusive access to the collision world, which blocks
other operations. Since AABB needs to be updated for collision detection, one can queue
them to defer update before start of the movement solver. Extensive tests on as much
as one installation (mine) show no drawback having that switched on.
[Physics]
lineofsight keep inactive cache
Control for how long (how many frames) the line of sight (LOS) request will be kept updated.
When a request for LOS is made for the first time, the background threads are stopped to
service it. From now on, the LOS will be refreshed preemptively as part of the background
routine until it is not required for lineofsight keep inactive cache frames. This mean
that subsequent request will not interrupt the background computation.
2020-10-15 04:11:44 +00:00
|
|
|
#include <mutex>
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2018-03-18 11:32:45 +00:00
|
|
|
#include "ptrholder.hpp"
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2022-06-16 22:28:44 +00:00
|
|
|
#include <components/detournavigator/collisionshapetype.hpp>
|
|
|
|
|
2015-05-12 14:49:21 +00:00
|
|
|
#include <osg/Quat>
|
2015-05-12 01:02:15 +00:00
|
|
|
#include <osg/Vec3f>
|
|
|
|
|
|
|
|
class btCollisionShape;
|
2021-10-07 19:47:05 +00:00
|
|
|
class btCollisionWorld;
|
2017-02-10 00:58:27 +00:00
|
|
|
class btConvexShape;
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2015-11-16 22:30:10 +00:00
|
|
|
namespace Resource
|
2015-05-12 01:02:15 +00:00
|
|
|
{
|
2022-08-24 22:14:17 +00:00
|
|
|
struct BulletShape;
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace MWPhysics
|
|
|
|
{
|
2020-10-14 09:32:12 +00:00
|
|
|
class PhysicsTaskScheduler;
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2020-10-14 09:32:12 +00:00
|
|
|
class Actor final : public PtrHolder
|
2015-05-12 01:02:15 +00:00
|
|
|
{
|
|
|
|
public:
|
2022-06-03 22:44:42 +00:00
|
|
|
Actor(const MWWorld::Ptr& ptr, const Resource::BulletShape* shape, PhysicsTaskScheduler* scheduler,
|
|
|
|
bool canWaterWalk, DetourNavigator::CollisionShapeType collisionShapeType);
|
2020-10-14 09:32:12 +00:00
|
|
|
~Actor() override;
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2022-09-16 21:48:42 +00:00
|
|
|
Actor(const Actor&) = delete;
|
|
|
|
|
|
|
|
Actor& operator=(const Actor&) = delete;
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
/**
|
|
|
|
* Sets the collisionMode for this actor. If disabled, the actor can fly and clip geometry.
|
|
|
|
*/
|
|
|
|
void enableCollisionMode(bool collision);
|
|
|
|
|
2021-07-21 17:04:36 +00:00
|
|
|
bool getCollisionMode() const { return mInternalCollisionMode; }
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2017-02-10 00:58:27 +00:00
|
|
|
btConvexShape* getConvexShape() const { return mConvexShape; }
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
/**
|
|
|
|
* Enables or disables the *external* collision body. If disabled, other actors will not collide with this
|
|
|
|
* actor.
|
|
|
|
*/
|
|
|
|
void enableCollisionBody(bool collision);
|
|
|
|
|
|
|
|
void updateScale();
|
2021-02-05 13:55:42 +00:00
|
|
|
void setRotation(osg::Quat quat);
|
2016-02-13 01:56:41 +00:00
|
|
|
|
2017-02-23 21:34:42 +00:00
|
|
|
/**
|
|
|
|
* Return true if the collision shape looks the same no matter how its Z rotated.
|
|
|
|
*/
|
|
|
|
bool isRotationallyInvariant() const;
|
|
|
|
|
2020-11-01 16:14:59 +00:00
|
|
|
/**
|
|
|
|
* Used by the physics simulation to store the simulation result. Used in conjunction with mWorldPosition
|
|
|
|
* to account for e.g. scripted movements
|
|
|
|
*/
|
2020-11-28 19:45:23 +00:00
|
|
|
void setSimulationPosition(const osg::Vec3f& position);
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2016-02-13 01:56:41 +00:00
|
|
|
void updateCollisionObjectPosition();
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
/**
|
2015-11-01 20:45:58 +00:00
|
|
|
* Returns the half extents of the collision body (scaled according to collision scale)
|
2015-05-12 01:02:15 +00:00
|
|
|
*/
|
|
|
|
osg::Vec3f getHalfExtents() const;
|
|
|
|
|
2019-03-03 11:45:36 +00:00
|
|
|
/**
|
|
|
|
* Returns the half extents of the collision body (not scaled)
|
|
|
|
*/
|
|
|
|
osg::Vec3f getOriginalHalfExtents() const;
|
|
|
|
|
2015-11-03 17:15:47 +00:00
|
|
|
/**
|
|
|
|
* Returns the position of the collision body
|
|
|
|
* @note The collision shape's origin is in its center, so the position returned can be described as center of
|
|
|
|
* the actor collision box in world space.
|
|
|
|
*/
|
2016-02-13 01:56:41 +00:00
|
|
|
osg::Vec3f getCollisionObjectPosition() const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Store the current position into mPreviousPosition, then move to this position.
|
2020-12-18 07:18:07 +00:00
|
|
|
* Returns true if the new position is different.
|
2016-02-13 01:56:41 +00:00
|
|
|
*/
|
2020-12-18 07:18:07 +00:00
|
|
|
bool setPosition(const osg::Vec3f& position);
|
2021-03-13 08:52:05 +00:00
|
|
|
|
|
|
|
// force set actor position to be as in Ptr::RefData
|
2020-12-19 19:25:46 +00:00
|
|
|
void updatePosition();
|
2021-03-13 08:52:05 +00:00
|
|
|
|
|
|
|
// register a position offset that will be applied during simulation.
|
2021-10-19 16:36:30 +00:00
|
|
|
void adjustPosition(const osg::Vec3f& offset);
|
2016-02-13 01:56:41 +00:00
|
|
|
|
2021-03-13 08:52:05 +00:00
|
|
|
// apply position offset. Can't be called during simulation
|
In 0.46, SetPos was setting position of actors before physics simulation, and from this position movement was simulated. This changed with async physics merging, and at the same time problems started, mostly with abot's scenic travel.
Skipping the simulation, switching off collisions, and other approaches were not correct as they either broke some mods, or some core mechanics of the engine such as teleportation or waterwalking. As it turns out, the way to go is to simply do _nothing_ (modulo some gymnastics to account for the 1 frame difference in case of async).
Scripted movement and the unstucking logic tends to collide. Early out of unstuck in case the actor doesn't attempt to move. This means there is no AI package for NPC, which are the case for some boats and striders, or the player is content with their position.
2023-03-16 20:33:36 +00:00
|
|
|
osg::Vec3f applyOffsetChange();
|
2021-03-13 08:52:05 +00:00
|
|
|
|
2015-11-01 20:45:58 +00:00
|
|
|
/**
|
|
|
|
* Returns the half extents of the collision body (scaled according to rendering scale)
|
|
|
|
* @note The reason we need this extra method is because of an inconsistency in MW - NPC race scales aren't
|
|
|
|
* applied to the collision shape, most likely to make environment collision testing easier. However in some
|
|
|
|
* cases (swimming level) we want the actual scale.
|
|
|
|
*/
|
|
|
|
osg::Vec3f getRenderingHalfExtents() const;
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
/**
|
|
|
|
* Sets the current amount of inertial force (incl. gravity) affecting this physic actor
|
|
|
|
*/
|
|
|
|
void setInertialForce(const osg::Vec3f& force);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the current amount of inertial force (incl. gravity) affecting this physic actor
|
|
|
|
*/
|
|
|
|
const osg::Vec3f& getInertialForce() const { return mForce; }
|
|
|
|
|
|
|
|
void setOnGround(bool grounded);
|
|
|
|
|
2022-05-24 21:24:52 +00:00
|
|
|
bool getOnGround() const { return mOnGround; }
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2017-02-06 03:46:44 +00:00
|
|
|
void setOnSlope(bool slope);
|
|
|
|
|
2022-05-24 21:24:52 +00:00
|
|
|
bool getOnSlope() const { return mOnSlope; }
|
2017-02-06 03:46:44 +00:00
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
/// Sets whether this actor should be able to collide with the water surface
|
|
|
|
void setCanWaterWalk(bool waterWalk);
|
|
|
|
|
|
|
|
/// Sets whether this actor has been walking on the water surface in the last frame
|
|
|
|
void setWalkingOnWater(bool walkingOnWater);
|
|
|
|
bool isWalkingOnWater() const;
|
|
|
|
|
2020-11-07 17:30:52 +00:00
|
|
|
MWWorld::Ptr getStandingOnPtr() const;
|
|
|
|
void setStandingOnPtr(const MWWorld::Ptr& ptr);
|
|
|
|
|
2021-03-21 01:14:56 +00:00
|
|
|
unsigned int getStuckFrames() const { return mStuckFrames; }
|
|
|
|
void setStuckFrames(unsigned int frames) { mStuckFrames = frames; }
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2021-03-21 01:14:56 +00:00
|
|
|
const osg::Vec3f& getLastStuckPosition() const { return mLastStuckPosition; }
|
|
|
|
void setLastStuckPosition(osg::Vec3f position) { mLastStuckPosition = position; }
|
|
|
|
|
2021-10-07 19:47:05 +00:00
|
|
|
bool canMoveToWaterSurface(float waterlevel, const btCollisionWorld* world) const;
|
|
|
|
|
2022-05-25 18:41:11 +00:00
|
|
|
bool isActive() const { return mActive; }
|
|
|
|
|
|
|
|
void setActive(bool value) { mActive = value; }
|
|
|
|
|
2022-06-16 22:28:44 +00:00
|
|
|
DetourNavigator::CollisionShapeType getCollisionShapeType() const { return mCollisionShapeType; }
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
private:
|
2020-11-07 17:30:52 +00:00
|
|
|
MWWorld::Ptr mStandingOnPtr;
|
2015-05-12 01:02:15 +00:00
|
|
|
/// Removes then re-adds the collision object to the dynamics world
|
|
|
|
void updateCollisionMask();
|
2016-12-16 19:22:07 +00:00
|
|
|
void addCollisionMask(int collisionMask);
|
2020-10-14 09:32:12 +00:00
|
|
|
int getCollisionMask() const;
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2022-02-22 04:04:08 +00:00
|
|
|
/// Returns the mesh translation, scaled and rotated as necessary
|
|
|
|
osg::Vec3f getScaledMeshTranslation() const;
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
bool mCanWaterWalk;
|
2021-07-21 17:04:36 +00:00
|
|
|
bool mWalkingOnWater;
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2017-02-23 21:34:42 +00:00
|
|
|
bool mRotationallyInvariant;
|
|
|
|
|
2022-06-16 22:28:44 +00:00
|
|
|
DetourNavigator::CollisionShapeType mCollisionShapeType;
|
|
|
|
|
2017-04-28 15:30:26 +00:00
|
|
|
std::unique_ptr<btCollisionShape> mShape;
|
2017-02-10 00:58:27 +00:00
|
|
|
btConvexShape* mConvexShape;
|
2015-05-12 01:02:15 +00:00
|
|
|
|
|
|
|
osg::Vec3f mMeshTranslation;
|
2021-07-24 14:44:21 +00:00
|
|
|
osg::Vec3f mOriginalHalfExtents;
|
2015-05-12 01:02:15 +00:00
|
|
|
osg::Vec3f mHalfExtents;
|
2021-07-24 14:44:21 +00:00
|
|
|
osg::Vec3f mRenderingHalfExtents;
|
2015-05-12 14:49:21 +00:00
|
|
|
osg::Quat mRotation;
|
2015-05-12 01:02:15 +00:00
|
|
|
|
|
|
|
osg::Vec3f mScale;
|
2020-12-18 07:18:07 +00:00
|
|
|
osg::Vec3f mPositionOffset;
|
2022-09-16 21:43:02 +00:00
|
|
|
bool mSkipSimulation = true;
|
Process movement queue in one or several background threads
Before movement calculation, the main thread prepare a
vector of ActorFrameData, which contains all data necessary to perform
the simulation, and feed it to the solver. At the same time it fetches
the result from the previous background simulation, which in turn is
used by the game mechanics.
Other functions of the physics system (weapon hit for instance)
interrupt the background simulation, with some exceptions described
below.
The number of threads is controlled by the numeric setting
[Physics]
async num threads
In case 'async num threads' > 1 and Bullet doesn't support multiple threads,
1 async thread will be used. 0 means synchronous solver.
Additional settings (will be silently switched off if async num threads = 0)
[Physics]
defer aabb update
Update AABBs of actors and objects in the background thread(s). It is not an especially
costly operation, but it needs exclusive access to the collision world, which blocks
other operations. Since AABB needs to be updated for collision detection, one can queue
them to defer update before start of the movement solver. Extensive tests on as much
as one installation (mine) show no drawback having that switched on.
[Physics]
lineofsight keep inactive cache
Control for how long (how many frames) the line of sight (LOS) request will be kept updated.
When a request for LOS is made for the first time, the background threads are stopped to
service it. From now on, the LOS will be refreshed preemptively as part of the background
routine until it is not required for lineofsight keep inactive cache frames. This mean
that subsequent request will not interrupt the background computation.
2020-10-15 04:11:44 +00:00
|
|
|
mutable std::mutex mPositionMutex;
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2021-03-21 01:14:56 +00:00
|
|
|
unsigned int mStuckFrames;
|
|
|
|
osg::Vec3f mLastStuckPosition;
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
osg::Vec3f mForce;
|
2021-07-21 17:04:36 +00:00
|
|
|
bool mOnGround;
|
|
|
|
bool mOnSlope;
|
|
|
|
bool mInternalCollisionMode;
|
2015-05-12 01:02:15 +00:00
|
|
|
bool mExternalCollisionMode;
|
2022-05-25 18:41:11 +00:00
|
|
|
bool mActive;
|
2015-05-12 01:02:15 +00:00
|
|
|
|
Process movement queue in one or several background threads
Before movement calculation, the main thread prepare a
vector of ActorFrameData, which contains all data necessary to perform
the simulation, and feed it to the solver. At the same time it fetches
the result from the previous background simulation, which in turn is
used by the game mechanics.
Other functions of the physics system (weapon hit for instance)
interrupt the background simulation, with some exceptions described
below.
The number of threads is controlled by the numeric setting
[Physics]
async num threads
In case 'async num threads' > 1 and Bullet doesn't support multiple threads,
1 async thread will be used. 0 means synchronous solver.
Additional settings (will be silently switched off if async num threads = 0)
[Physics]
defer aabb update
Update AABBs of actors and objects in the background thread(s). It is not an especially
costly operation, but it needs exclusive access to the collision world, which blocks
other operations. Since AABB needs to be updated for collision detection, one can queue
them to defer update before start of the movement solver. Extensive tests on as much
as one installation (mine) show no drawback having that switched on.
[Physics]
lineofsight keep inactive cache
Control for how long (how many frames) the line of sight (LOS) request will be kept updated.
When a request for LOS is made for the first time, the background threads are stopped to
service it. From now on, the LOS will be refreshed preemptively as part of the background
routine until it is not required for lineofsight keep inactive cache frames. This mean
that subsequent request will not interrupt the background computation.
2020-10-15 04:11:44 +00:00
|
|
|
PhysicsTaskScheduler* mTaskScheduler;
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2022-09-16 21:47:39 +00:00
|
|
|
inline void updateScaleUnsafe();
|
|
|
|
|
|
|
|
inline void updateCollisionObjectPositionUnsafe();
|
2015-05-12 01:02:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|