2015-05-12 03:02:15 +02:00
# ifndef OPENMW_MWPHYSICS_PHYSICSSYSTEM_H
# define OPENMW_MWPHYSICS_PHYSICSSYSTEM_H
2011-08-01 15:55:36 +02:00
2020-10-15 06:11:00 +02:00
# include <array>
2014-10-05 22:24:11 +02:00
# include <memory>
2015-06-03 23:04:35 +02:00
# include <map>
2015-11-19 23:33:08 +01:00
# include <set>
2018-05-26 17:44:25 +03:00
# include <algorithm>
2014-10-05 22:24:11 +02:00
2015-06-03 23:04:35 +02:00
# include <osg/Quat>
2020-05-11 13:37:00 +00:00
# include <osg/BoundingBox>
2015-05-03 00:39:01 +02:00
# include <osg/ref_ptr>
2015-05-10 01:09:00 +02:00
# include "../mwworld/ptr.hpp"
2013-08-17 07:48:45 -07:00
2015-06-01 01:57:15 +02:00
# include "collisiontype.hpp"
2020-08-03 22:44:16 +02:00
# include "raycasting.hpp"
2015-06-01 01:57:15 +02:00
2015-05-03 00:39:01 +02:00
namespace osg
{
class Group ;
2017-03-06 19:04:17 +01:00
class Object ;
2020-05-22 00:11:23 +02:00
class Stats ;
2015-05-03 00:39:01 +02:00
}
2013-02-07 12:11:10 -08:00
2015-05-03 00:39:01 +02:00
namespace MWRender
{
class DebugDrawer ;
}
2015-11-16 23:30:10 +01:00
namespace Resource
2015-05-12 03:02:15 +02:00
{
class BulletShapeManager ;
2016-02-09 19:04:59 +01:00
class ResourceSystem ;
2015-05-12 03:02:15 +02:00
}
2016-02-09 19:04:59 +01:00
namespace SceneUtil
2015-05-12 03:02:15 +02:00
{
2016-02-09 19:04:59 +01:00
class UnrefQueue ;
2015-05-12 03:02:15 +02:00
}
2015-05-27 23:09:38 +02:00
class btCollisionWorld ;
2015-05-27 22:32:11 +02:00
class btBroadphaseInterface ;
class btDefaultCollisionConfiguration ;
class btCollisionDispatcher ;
class btCollisionObject ;
class btCollisionShape ;
2015-05-10 02:08:25 +02:00
2015-05-10 01:09:00 +02:00
namespace MWPhysics
2011-08-01 15:55:36 +02:00
{
2020-10-14 11:32:12 +02:00
using PtrPositionList = std : : map < MWWorld : : Ptr , osg : : Vec3f > ;
using CollisionMap = std : : map < MWWorld : : Ptr , MWWorld : : Ptr > ;
2015-05-10 02:08:25 +02:00
class HeightField ;
2015-05-12 03:02:15 +02:00
class Object ;
class Actor ;
2020-10-15 06:11:00 +02:00
class PhysicsTaskScheduler ;
struct LOSRequest
{
LOSRequest ( const std : : weak_ptr < Actor > & a1 , const std : : weak_ptr < Actor > & a2 ) ;
std : : array < std : : weak_ptr < Actor > , 2 > mActors ;
std : : array < const Actor * , 2 > mRawActors ;
bool mResult ;
bool mStale ;
int mAge ;
} ;
bool operator = = ( const LOSRequest & lhs , const LOSRequest & rhs ) noexcept ;
struct ActorFrameData
{
ActorFrameData ( const std : : shared_ptr < Actor > & actor , const MWWorld : : Ptr character , const MWWorld : : Ptr standingOn , bool moveToWaterSurface , osg : : Vec3f movement , float slowFall , float waterlevel ) ;
void updatePosition ( ) ;
std : : weak_ptr < Actor > mActor ;
Actor * mActorRaw ;
MWWorld : : Ptr mPtr ;
MWWorld : : Ptr mStandingOn ;
bool mFlying ;
bool mSwimming ;
bool mPositionChanged ;
bool mWasOnGround ;
bool mWantJump ;
bool mDidJump ;
bool mIsDead ;
bool mNeedLand ;
bool mMoveToWaterSurface ;
float mWaterlevel ;
float mSlowFall ;
float mOldHeight ;
float mFallHeight ;
osg : : Vec3f mMovement ;
osg : : Vec3f mPosition ;
ESM : : Position mRefpos ;
} ;
struct WorldFrameData
{
WorldFrameData ( ) ;
bool mIsInStorm ;
osg : : Vec3f mStormDirection ;
} ;
2015-05-10 02:08:25 +02:00
2020-08-03 22:44:16 +02:00
class PhysicsSystem : public RayCastingInterface
2011-08-01 15:55:36 +02:00
{
public :
2015-05-12 16:24:53 +02:00
PhysicsSystem ( Resource : : ResourceSystem * resourceSystem , osg : : ref_ptr < osg : : Group > parentNode ) ;
2020-08-26 12:34:27 +04:00
virtual ~ PhysicsSystem ( ) ;
2011-08-22 21:34:51 +02:00
2016-02-09 19:04:59 +01:00
void setUnrefQueue ( SceneUtil : : UnrefQueue * unrefQueue ) ;
2016-02-07 00:07:02 +01:00
Resource : : BulletShapeManager * getShapeManager ( ) ;
2014-10-05 22:24:11 +02:00
void enableWater ( float height ) ;
void setWaterHeight ( float height ) ;
void disableWater ( ) ;
2015-12-18 18:32:42 +01:00
void addObject ( const MWWorld : : Ptr & ptr , const std : : string & mesh , int collisionType = CollisionType_World ) ;
2015-05-12 19:02:56 +02:00
void addActor ( const MWWorld : : Ptr & ptr , const std : : string & mesh ) ;
void updatePtr ( const MWWorld : : Ptr & old , const MWWorld : : Ptr & updated ) ;
Actor * getActor ( const MWWorld : : Ptr & ptr ) ;
2015-12-18 17:36:14 +01:00
const Actor * getActor ( const MWWorld : : ConstPtr & ptr ) const ;
2016-03-05 15:56:19 +01:00
const Object * getObject ( const MWWorld : : ConstPtr & ptr ) const ;
2015-05-12 03:02:15 +02:00
// Object or Actor
void remove ( const MWWorld : : Ptr & ptr ) ;
void updateScale ( const MWWorld : : Ptr & ptr ) ;
void updateRotation ( const MWWorld : : Ptr & ptr ) ;
void updatePosition ( const MWWorld : : Ptr & ptr ) ;
2011-08-22 21:34:51 +02:00
2017-03-07 15:00:16 +01:00
void addHeightField ( const float * heights , int x , int y , float triSize , float sqrtVerts , float minH , float maxH , const osg : : Object * holdObject ) ;
2012-03-13 17:09:50 +01:00
void removeHeightField ( int x , int y ) ;
2018-03-14 01:49:08 +03:00
const HeightField * getHeightField ( int x , int y ) const ;
2011-08-01 15:55:36 +02:00
bool toggleCollisionMode ( ) ;
2013-08-20 11:31:49 -07:00
2020-10-14 11:32:12 +02:00
void stepSimulation ( ) ;
2015-06-07 17:00:00 +02:00
void debugDraw ( ) ;
2014-06-23 20:43:24 +02:00
2015-12-18 17:56:48 +01:00
std : : vector < MWWorld : : Ptr > getCollisions ( const MWWorld : : ConstPtr & ptr , int collisionGroup , int collisionMask ) const ; ///< get handles this object collides with
2017-02-10 02:43:49 +01:00
osg : : Vec3f traceDown ( const MWWorld : : Ptr & ptr , const osg : : Vec3f & position , float maxHeight ) ;
2013-02-05 12:45:10 -08:00
2015-12-18 17:38:21 +01:00
std : : pair < MWWorld : : Ptr , osg : : Vec3f > getHitContact ( const MWWorld : : ConstPtr & actor ,
2015-05-22 04:36:17 +02:00
const osg : : Vec3f & origin ,
const osg : : Quat & orientation ,
2020-10-14 11:32:12 +02:00
float queryDistance , std : : vector < MWWorld : : Ptr > & targets ) ;
2012-03-25 20:52:56 +02:00
2015-11-18 19:00:43 +01:00
/// Get distance from \a point to the collision shape of \a target. Uses a raycast to find where the
/// target vector hits the collision shape and then calculates distance from the intersection point.
/// This can be used to find out how much nearer we need to move to the target for a "getHitContact" to be successful.
/// \note Only Actor targets are supported at the moment.
2020-08-26 12:34:27 +04:00
float getHitDistance ( const osg : : Vec3f & point , const MWWorld : : ConstPtr & target ) const override final ;
2015-06-01 01:57:15 +02:00
2017-02-02 16:20:34 +09:00
/// @param me Optional, a Ptr to ignore in the list of results. targets are actors to filter for, ignoring all other actors.
2020-08-03 22:44:16 +02:00
RayCastingResult castRay ( const osg : : Vec3f & from , const osg : : Vec3f & to , const MWWorld : : ConstPtr & ignore = MWWorld : : ConstPtr ( ) ,
2017-02-02 16:20:34 +09:00
std : : vector < MWWorld : : Ptr > targets = std : : vector < MWWorld : : Ptr > ( ) ,
2020-08-26 12:34:27 +04:00
int mask = CollisionType_World | CollisionType_HeightMap | CollisionType_Actor | CollisionType_Door , int group = 0xff ) const override final ;
2011-08-22 21:34:51 +02:00
2020-08-26 12:34:27 +04:00
RayCastingResult castSphere ( const osg : : Vec3f & from , const osg : : Vec3f & to , float radius ) const override final ;
2015-06-01 15:34:46 +02:00
2015-06-01 01:57:15 +02:00
/// Return true if actor1 can see actor2.
2020-08-26 12:34:27 +04:00
bool getLineOfSight ( const MWWorld : : ConstPtr & actor1 , const MWWorld : : ConstPtr & actor2 ) const override final ;
2012-07-25 20:25:53 +04:00
2015-06-01 02:40:42 +02:00
bool isOnGround ( const MWWorld : : Ptr & actor ) ;
2016-11-15 00:20:17 +09:00
bool canMoveToWaterSurface ( const MWWorld : : ConstPtr & actor , const float waterlevel ) ;
2015-09-17 01:08:16 +02:00
/// Get physical half extents (scaled) of the given actor.
2015-12-18 17:36:14 +01:00
osg : : Vec3f getHalfExtents ( const MWWorld : : ConstPtr & actor ) const ;
2015-06-01 21:41:13 +02:00
2019-03-03 14:45:36 +03:00
/// Get physical half extents (not scaled) of the given actor.
osg : : Vec3f getOriginalHalfExtents ( const MWWorld : : ConstPtr & actor ) const ;
2015-11-01 21:45:58 +01:00
/// @see MWPhysics::Actor::getRenderingHalfExtents
2015-12-18 17:36:14 +01:00
osg : : Vec3f getRenderingHalfExtents ( const MWWorld : : ConstPtr & actor ) const ;
2015-11-03 18:15:47 +01:00
/// Get the position of the collision shape for the actor. Use together with getHalfExtents() to get the collision bounds in world space.
/// @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 02:56:41 +01:00
osg : : Vec3f getCollisionObjectPosition ( const MWWorld : : ConstPtr & actor ) const ;
2015-11-01 21:45:58 +01:00
2020-05-11 13:37:00 +00:00
/// Get bounding box in world space of the given object.
osg : : BoundingBox getBoundingBox ( const MWWorld : : ConstPtr & object ) const ;
2013-08-17 07:48:45 -07:00
/// Queues velocity movement for a Ptr. If a Ptr is already queued, its velocity will
/// be overwritten. Valid until the next call to applyQueuedMovement.
2015-05-12 03:02:15 +02:00
void queueObjectMovement ( const MWWorld : : Ptr & ptr , const osg : : Vec3f & velocity ) ;
2013-08-17 07:48:45 -07:00
2014-08-13 16:23:34 +02:00
/// Apply all queued movements, then clear the list.
2020-10-14 11:32:12 +02:00
const PtrPositionList & applyQueuedMovement ( float dt ) ;
2013-08-17 07:48:45 -07:00
2014-08-13 16:23:34 +02:00
/// Clear the queued movements list without applying.
void clearQueuedMovement ( ) ;
2014-07-29 19:01:40 +02:00
/// Return true if \a actor has been standing on \a object in this frame
/// This will trigger whenever the object is directly below the actor.
/// It doesn't matter if the actor is stationary or moving.
2015-12-18 17:56:48 +01:00
bool isActorStandingOn ( const MWWorld : : Ptr & actor , const MWWorld : : ConstPtr & object ) const ;
2014-07-29 19:01:40 +02:00
/// Get the handle of all actors standing on \a object in this frame.
2015-12-18 17:56:48 +01:00
void getActorsStandingOn ( const MWWorld : : ConstPtr & object , std : : vector < MWWorld : : Ptr > & out ) const ;
2014-07-29 19:01:40 +02:00
/// Return true if \a actor has collided with \a object in this frame.
/// This will detect running into objects, but will not detect climbing stairs, stepping up a small object, etc.
2015-12-18 17:56:48 +01:00
bool isActorCollidingWith ( const MWWorld : : Ptr & actor , const MWWorld : : ConstPtr & object ) const ;
2014-07-29 19:01:40 +02:00
/// Get the handle of all actors colliding with \a object in this frame.
2015-12-18 17:56:48 +01:00
void getActorsCollidingWith ( const MWWorld : : ConstPtr & object , std : : vector < MWWorld : : Ptr > & out ) const ;
2014-07-29 19:01:40 +02:00
2015-05-03 00:39:01 +02:00
bool toggleDebugRendering ( ) ;
2015-11-20 19:22:31 +01:00
/// Mark the given object as a 'non-solid' object. A non-solid object means that
/// \a isOnSolidGround will return false for actors standing on that object.
2015-12-18 17:56:48 +01:00
void markAsNonSolid ( const MWWorld : : ConstPtr & ptr ) ;
2015-11-20 19:22:31 +01:00
bool isOnSolidGround ( const MWWorld : : Ptr & actor ) const ;
2018-05-26 17:44:25 +03:00
2018-09-22 12:57:50 +04:00
void updateAnimatedCollisionShape ( const MWWorld : : Ptr & object ) ;
2018-05-26 17:44:25 +03:00
template < class Function >
void forEachAnimatedObject ( Function & & function ) const
{
std : : for_each ( mAnimatedObjects . begin ( ) , mAnimatedObjects . end ( ) , function ) ;
}
2020-02-09 18:24:08 +01:00
bool isAreaOccupiedByOtherActor ( const osg : : Vec3f & position , const float radius , const MWWorld : : ConstPtr & ignore ) const ;
2020-05-22 00:11:23 +02:00
void reportStats ( unsigned int frameNumber , osg : : Stats & stats ) const ;
2011-08-01 15:55:36 +02:00
private :
2012-08-17 10:10:37 +04:00
2014-10-05 22:24:11 +02:00
void updateWater ( ) ;
2020-10-15 06:11:00 +02:00
std : : vector < ActorFrameData > prepareFrameData ( ) ;
2016-02-09 19:04:59 +01:00
osg : : ref_ptr < SceneUtil : : UnrefQueue > mUnrefQueue ;
2020-10-14 11:32:12 +02:00
std : : unique_ptr < btBroadphaseInterface > mBroadphase ;
std : : unique_ptr < btDefaultCollisionConfiguration > mCollisionConfiguration ;
std : : unique_ptr < btCollisionDispatcher > mDispatcher ;
std : : shared_ptr < btCollisionWorld > mCollisionWorld ;
2015-05-10 02:08:25 +02:00
2017-04-28 17:30:26 +02:00
std : : unique_ptr < Resource : : BulletShapeManager > mShapeManager ;
2016-02-06 16:57:54 +01:00
Resource : : ResourceSystem * mResourceSystem ;
2015-05-12 03:02:15 +02:00
2020-10-14 11:32:12 +02:00
using ObjectMap = std : : map < MWWorld : : ConstPtr , std : : shared_ptr < Object > > ;
2015-05-12 03:02:15 +02:00
ObjectMap mObjects ;
2015-11-19 23:33:08 +01:00
std : : set < Object * > mAnimatedObjects ; // stores pointers to elements in mObjects
2020-10-14 11:32:12 +02:00
using ActorMap = std : : map < MWWorld : : ConstPtr , std : : shared_ptr < Actor > > ;
2015-05-12 03:02:15 +02:00
ActorMap mActors ;
2020-10-14 11:32:12 +02:00
using HeightFieldMap = std : : map < std : : pair < int , int > , HeightField * > ;
2015-05-10 02:08:25 +02:00
HeightFieldMap mHeightFields ;
2015-05-03 00:39:01 +02:00
bool mDebugDrawEnabled ;
2015-11-18 21:20:12 +01:00
// Tracks standing collisions happening during a single frame. <actor handle, collided handle>
// This will detect standing on an object, but won't detect running e.g. against a wall.
2015-05-30 01:32:00 +02:00
CollisionMap mStandingCollisions ;
2014-07-29 19:01:40 +02:00
2016-12-14 22:11:22 +01:00
// replaces all occurrences of 'old' in the map by 'updated', no matter if it's a key or value
2015-05-30 01:32:00 +02:00
void updateCollisionMapPtr ( CollisionMap & map , const MWWorld : : Ptr & old , const MWWorld : : Ptr & updated ) ;
2014-07-29 19:01:40 +02:00
2020-10-14 11:32:12 +02:00
using PtrVelocityList = std : : vector < std : : pair < MWWorld : : Ptr , osg : : Vec3f > > ;
2013-08-17 07:48:45 -07:00
PtrVelocityList mMovementQueue ;
2020-10-14 11:32:12 +02:00
PtrPositionList mMovementResults ;
2020-10-15 06:11:00 +02:00
std : : unique_ptr < WorldFrameData > mWorldFrameData ;
std : : vector < ActorFrameData > mActorsFrameData ;
2013-08-17 07:48:45 -07:00
2013-08-20 11:31:49 -07:00
float mTimeAccum ;
2014-10-05 22:24:11 +02:00
float mWaterHeight ;
2017-04-20 00:28:27 +09:00
bool mWaterEnabled ;
2014-10-05 22:24:11 +02:00
2017-04-28 17:30:26 +02:00
std : : unique_ptr < btCollisionObject > mWaterCollisionObject ;
std : : unique_ptr < btCollisionShape > mWaterCollisionShape ;
2014-10-05 22:24:11 +02:00
2017-04-28 17:30:26 +02:00
std : : unique_ptr < MWRender : : DebugDrawer > mDebugDrawer ;
2015-05-03 00:39:01 +02:00
osg : : ref_ptr < osg : : Group > mParentNode ;
2017-09-26 21:19:53 +04:00
float mPhysicsDt ;
2011-08-22 21:34:51 +02:00
PhysicsSystem ( const PhysicsSystem & ) ;
PhysicsSystem & operator = ( const PhysicsSystem & ) ;
2011-08-01 15:55:36 +02:00
} ;
}
# endif