1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-25 15:35:23 +00:00
OpenMW/apps/openmw/mwphysics/physicssystem.hpp

254 lines
10 KiB
C++
Raw Normal View History

#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>
#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>
#include <osg/BoundingBox>
2015-05-03 00:39:01 +02:00
#include <osg/ref_ptr>
#include "../mwworld/ptr.hpp"
2015-06-01 01:57:15 +02:00
#include "collisiontype.hpp"
2015-05-03 00:39:01 +02:00
namespace osg
{
class Group;
class Object;
class Stats;
2015-05-03 00:39:01 +02:00
}
2015-05-03 00:39:01 +02:00
namespace MWRender
{
class DebugDrawer;
}
namespace Resource
{
class BulletShapeManager;
class ResourceSystem;
}
namespace SceneUtil
{
class UnrefQueue;
}
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
namespace MWPhysics
2011-08-01 15:55:36 +02:00
{
typedef std::vector<std::pair<MWWorld::Ptr,osg::Vec3f> > PtrVelocityList;
2015-05-10 02:08:25 +02:00
class HeightField;
class Object;
class Actor;
2015-05-10 02:08:25 +02:00
2011-08-01 15:55:36 +02:00
class PhysicsSystem
{
public:
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
void setUnrefQueue(SceneUtil::UnrefQueue* unrefQueue);
Resource::BulletShapeManager* getShapeManager();
2014-10-05 22:24:11 +02:00
void enableWater(float height);
void setWaterHeight(float height);
void disableWater();
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);
const Actor* getActor(const MWWorld::ConstPtr& ptr) const;
const Object* getObject(const MWWorld::ConstPtr& ptr) const;
// 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);
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();
void stepSimulation(float dt);
void debugDraw();
std::vector<MWWorld::Ptr> getCollisions(const MWWorld::ConstPtr &ptr, int collisionGroup, int collisionMask) const; ///< get handles this object collides with
osg::Vec3f traceDown(const MWWorld::Ptr &ptr, const osg::Vec3f& position, float maxHeight);
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,
float queryDistance, std::vector<MWWorld::Ptr> targets = std::vector<MWWorld::Ptr>());
2012-03-25 20:52:56 +02: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.
float getHitDistance(const osg::Vec3f& point, const MWWorld::ConstPtr& target) const;
2015-06-01 01:57:15 +02:00
struct RayResult
{
bool mHit;
osg::Vec3f mHitPos;
osg::Vec3f mHitNormal;
MWWorld::Ptr mHitObject;
};
/// @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(),
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);
bool canMoveToWaterSurface (const MWWorld::ConstPtr &actor, const float waterlevel);
/// Get physical half extents (scaled) of the given actor.
osg::Vec3f getHalfExtents(const MWWorld::ConstPtr& actor) const;
2015-06-01 21:41:13 +02:00
/// Get physical half extents (not scaled) of the given actor.
osg::Vec3f getOriginalHalfExtents(const MWWorld::ConstPtr& actor) const;
/// @see MWPhysics::Actor::getRenderingHalfExtents
osg::Vec3f getRenderingHalfExtents(const MWWorld::ConstPtr& actor) const;
/// 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.
osg::Vec3f getCollisionObjectPosition(const MWWorld::ConstPtr& actor) const;
/// Get bounding box in world space of the given object.
osg::BoundingBox getBoundingBox(const MWWorld::ConstPtr &object) const;
/// Queues velocity movement for a Ptr. If a Ptr is already queued, its velocity will
/// be overwritten. Valid until the next call to applyQueuedMovement.
void queueObjectMovement(const MWWorld::Ptr &ptr, const osg::Vec3f &velocity);
/// Apply all queued movements, then clear the list.
const PtrVelocityList& applyQueuedMovement(float dt);
/// Clear the queued movements list without applying.
void clearQueuedMovement();
/// 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.
bool isActorStandingOn(const MWWorld::Ptr& actor, const MWWorld::ConstPtr& object) const;
/// Get the handle of all actors standing on \a object in this frame.
void getActorsStandingOn(const MWWorld::ConstPtr& object, std::vector<MWWorld::Ptr>& out) const;
/// 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.
bool isActorCollidingWith(const MWWorld::Ptr& actor, const MWWorld::ConstPtr& object) const;
/// Get the handle of all actors colliding with \a object in this frame.
void getActorsCollidingWith(const MWWorld::ConstPtr& object, std::vector<MWWorld::Ptr>& out) const;
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.
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
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);
}
bool isAreaOccupiedByOtherActor(const osg::Vec3f& position, const float radius, const MWWorld::ConstPtr& ignore) const;
void reportStats(unsigned int frameNumber, osg::Stats& stats) const;
2011-08-01 15:55:36 +02:00
private:
2014-10-05 22:24:11 +02:00
void updateWater();
osg::ref_ptr<SceneUtil::UnrefQueue> mUnrefQueue;
2015-05-10 02:08:25 +02:00
btBroadphaseInterface* mBroadphase;
btDefaultCollisionConfiguration* mCollisionConfiguration;
btCollisionDispatcher* mDispatcher;
btCollisionWorld* mCollisionWorld;
2015-05-10 02:08:25 +02:00
std::unique_ptr<Resource::BulletShapeManager> mShapeManager;
Resource::ResourceSystem* mResourceSystem;
typedef std::map<MWWorld::ConstPtr, Object*> ObjectMap;
ObjectMap mObjects;
std::set<Object*> mAnimatedObjects; // stores pointers to elements in mObjects
typedef std::map<MWWorld::ConstPtr, Actor*> ActorMap;
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;
// 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;
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);
PtrVelocityList mMovementQueue;
PtrVelocityList mMovementResults;
float mTimeAccum;
2014-10-05 22:24:11 +02:00
float mWaterHeight;
bool mWaterEnabled;
2014-10-05 22:24:11 +02:00
std::unique_ptr<btCollisionObject> mWaterCollisionObject;
std::unique_ptr<btCollisionShape> mWaterCollisionShape;
2014-10-05 22:24:11 +02:00
std::unique_ptr<MWRender::DebugDrawer> mDebugDrawer;
2015-05-03 00:39:01 +02:00
osg::ref_ptr<osg::Group> mParentNode;
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