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
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"
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
{
2015-05-12 03:02:15 +02:00
typedef std : : vector < std : : pair < MWWorld : : Ptr , osg : : Vec3f > > PtrVelocityList ;
2015-05-10 02:08:25 +02:00
class HeightField ;
2015-05-12 03:02:15 +02:00
class Object ;
class Actor ;
2015-05-10 02:08:25 +02:00
2011-08-01 15:55:36 +02:00
class PhysicsSystem
{
public :
2015-05-12 16:24:53 +02:00
PhysicsSystem ( Resource : : ResourceSystem * resourceSystem , osg : : ref_ptr < osg : : Group > parentNode ) ;
2011-08-01 15:55:36 +02:00
~ 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
2014-06-23 20:43:24 +02:00
void stepSimulation ( float dt ) ;
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 ,
2017-02-02 16:20:34 +09:00
float queryDistance , std : : vector < MWWorld : : Ptr > targets = std : : vector < MWWorld : : Ptr > ( ) ) ;
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.
2015-12-18 17:36:14 +01:00
float getHitDistance ( const osg : : Vec3f & point , const MWWorld : : ConstPtr & target ) const ;
2015-11-18 19:00:43 +01:00
2015-06-01 01:57:15 +02:00
struct RayResult
{
bool mHit ;
osg : : Vec3f mHitPos ;
osg : : Vec3f mHitNormal ;
MWWorld : : Ptr mHitObject ;
} ;
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.
2017-04-20 20:36:14 +09:00
RayResult 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 > ( ) ,
int mask = CollisionType_World | CollisionType_HeightMap | CollisionType_Actor | CollisionType_Door , int group = 0xff ) const ;
2011-08-22 21:34:51 +02:00
2015-06-01 15:34:46 +02:00
RayResult castSphere ( const osg : : Vec3f & from , const osg : : Vec3f & to , float radius ) ;
2015-06-01 01:57:15 +02:00
/// Return true if actor1 can see actor2.
2015-12-18 18:02:57 +01:00
bool getLineOfSight ( const MWWorld : : ConstPtr & actor1 , const MWWorld : : ConstPtr & actor2 ) const ;
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.
2013-08-17 07:48:45 -07:00
const PtrVelocityList & applyQueuedMovement ( float dt ) ;
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 ( ) ;
2016-02-09 19:04:59 +01:00
osg : : ref_ptr < SceneUtil : : UnrefQueue > mUnrefQueue ;
2015-05-10 02:08:25 +02:00
btBroadphaseInterface * mBroadphase ;
btDefaultCollisionConfiguration * mCollisionConfiguration ;
btCollisionDispatcher * mDispatcher ;
2015-05-27 23:09:38 +02:00
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
2015-12-18 17:30:39 +01:00
typedef std : : map < MWWorld : : ConstPtr , Object * > ObjectMap ;
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
2015-12-18 17:30:39 +01:00
typedef std : : map < MWWorld : : ConstPtr , Actor * > ActorMap ;
2015-05-12 03:02:15 +02:00
ActorMap mActors ;
2015-05-10 02:08:25 +02:00
typedef std : : map < std : : pair < int , int > , HeightField * > HeightFieldMap ;
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
typedef std : : map < MWWorld : : Ptr , MWWorld : : Ptr > CollisionMap ;
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
2013-08-17 07:48:45 -07:00
PtrVelocityList mMovementQueue ;
PtrVelocityList mMovementResults ;
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