481 lines
16 KiB
C++
481 lines
16 KiB
C++
//========= Copyright Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose: Provides structures and classes necessary to simulate a portal.
|
|
//
|
|
// $NoKeywords: $
|
|
//=====================================================================================//
|
|
|
|
#ifndef PORTALSIMULATION_H
|
|
#define PORTALSIMULATION_H
|
|
|
|
#ifdef _WIN32
|
|
#pragma once
|
|
#endif
|
|
|
|
#include "mathlib/polyhedron.h"
|
|
#include "const.h"
|
|
#include "tier1/utlmap.h"
|
|
#include "tier1/utlvector.h"
|
|
|
|
#define PORTAL_SIMULATORS_EMBED_GUID //define this to embed a unique integer with each portal simulator for debugging purposes
|
|
|
|
struct StaticPropPolyhedronGroups_t //each static prop is made up of a group of polyhedrons, these help us pull those groups from an array
|
|
{
|
|
int iStartIndex;
|
|
int iNumPolyhedrons;
|
|
};
|
|
|
|
enum PortalSimulationEntityFlags_t
|
|
{
|
|
PSEF_OWNS_ENTITY = (1 << 0), //this environment is responsible for the entity's physics objects
|
|
PSEF_OWNS_PHYSICS = (1 << 1),
|
|
PSEF_IS_IN_PORTAL_HOLE = (1 << 2), //updated per-phyframe
|
|
PSEF_CLONES_ENTITY_FROM_MAIN = (1 << 3), //entity is close enough to the portal to affect objects intersecting the portal
|
|
//PSEF_HAS_LINKED_CLONE = (1 << 1), //this environment has a clone of the entity which is transformed from its linked portal
|
|
};
|
|
|
|
enum PS_PhysicsObjectSourceType_t
|
|
{
|
|
PSPOST_LOCAL_BRUSHES,
|
|
PSPOST_REMOTE_BRUSHES,
|
|
PSPOST_LOCAL_STATICPROPS,
|
|
PSPOST_REMOTE_STATICPROPS,
|
|
PSPOST_HOLYWALL_TUBE
|
|
};
|
|
|
|
struct PortalTransformAsAngledPosition_t //a matrix transformation from this portal to the linked portal, stored as vector and angle transforms
|
|
{
|
|
Vector ptOriginTransform;
|
|
QAngle qAngleTransform;
|
|
};
|
|
|
|
inline bool LessFunc_Integer( const int &a, const int &b ) { return a < b; };
|
|
|
|
|
|
class CPortalSimulatorEventCallbacks //sends out notifications of events to game specific code
|
|
{
|
|
public:
|
|
virtual void PortalSimulator_TookOwnershipOfEntity( CBaseEntity *pEntity ) { };
|
|
virtual void PortalSimulator_ReleasedOwnershipOfEntity( CBaseEntity *pEntity ) { };
|
|
|
|
virtual void PortalSimulator_TookPhysicsOwnershipOfEntity( CBaseEntity *pEntity ) { };
|
|
virtual void PortalSimulator_ReleasedPhysicsOwnershipOfEntity( CBaseEntity *pEntity ) { };
|
|
};
|
|
|
|
//====================================================================================
|
|
// To any coder trying to understand the following nested structures....
|
|
//
|
|
// You may be wondering... why? wtf?
|
|
//
|
|
// The answer. The previous incarnation of server side portal simulation suffered
|
|
// terribly from evolving variables with increasingly cryptic names with no clear
|
|
// definition of what part of the system the variable was involved with.
|
|
//
|
|
// It's my hope that a nested structure with clear boundaries will eliminate that
|
|
// horrible, awful, nasty, frustrating confusion. (It was really really bad). This
|
|
// system has the added benefit of pseudo-forcing a naming structure.
|
|
//
|
|
// Lastly, if it all roots in one struct, we can const reference it out to allow
|
|
// easy reads without writes
|
|
//
|
|
// It's broken out like this to solve a few problems....
|
|
// 1. It cleans up intellisense when you don't actually define a structure
|
|
// within a structure.
|
|
// 2. Shorter typenames when you want to have a pointer/reference deep within
|
|
// the nested structure.
|
|
// 3. Needed at least one level removed from CPortalSimulator so
|
|
// pointers/references could be made while the primary instance of the
|
|
// data was private/protected.
|
|
//
|
|
// It may be slightly difficult to understand in it's broken out structure, but
|
|
// intellisense brings all the data together in a very cohesive manner for
|
|
// working with.
|
|
//====================================================================================
|
|
|
|
struct PS_PlacementData_t //stuff useful for geometric operations
|
|
{
|
|
Vector ptCenter;
|
|
QAngle qAngles;
|
|
Vector vForward;
|
|
Vector vUp;
|
|
Vector vRight;
|
|
VPlane PortalPlane;
|
|
VMatrix matThisToLinked;
|
|
VMatrix matLinkedToThis;
|
|
PortalTransformAsAngledPosition_t ptaap_ThisToLinked;
|
|
PortalTransformAsAngledPosition_t ptaap_LinkedToThis;
|
|
CPhysCollide *pHoleShapeCollideable; //used to test if a collideable is in the hole, should NOT be collided against in general
|
|
PS_PlacementData_t( void )
|
|
{
|
|
memset( this, 0, sizeof( PS_PlacementData_t ) );
|
|
}
|
|
};
|
|
|
|
struct PS_SD_Static_World_Brushes_t
|
|
{
|
|
CUtlVector<CPolyhedron *> Polyhedrons; //the building blocks of more complex collision
|
|
CPhysCollide *pCollideable;
|
|
#ifndef CLIENT_DLL
|
|
IPhysicsObject *pPhysicsObject;
|
|
PS_SD_Static_World_Brushes_t() : pCollideable(NULL), pPhysicsObject(NULL) {};
|
|
#else
|
|
PS_SD_Static_World_Brushes_t() : pCollideable(NULL) {};
|
|
#endif
|
|
|
|
};
|
|
|
|
|
|
struct PS_SD_Static_World_StaticProps_ClippedProp_t
|
|
{
|
|
StaticPropPolyhedronGroups_t PolyhedronGroup;
|
|
CPhysCollide * pCollide;
|
|
#ifndef CLIENT_DLL
|
|
IPhysicsObject * pPhysicsObject;
|
|
#endif
|
|
IHandleEntity * pSourceProp;
|
|
|
|
int iTraceContents;
|
|
short iTraceSurfaceProps;
|
|
static CBaseEntity * pTraceEntity;
|
|
static const char * szTraceSurfaceName; //same for all static props, here just for easy reference
|
|
static const int iTraceSurfaceFlags; //same for all static props, here just for easy reference
|
|
};
|
|
|
|
struct PS_SD_Static_World_StaticProps_t
|
|
{
|
|
CUtlVector<CPolyhedron *> Polyhedrons; //the building blocks of more complex collision
|
|
CUtlVector<PS_SD_Static_World_StaticProps_ClippedProp_t> ClippedRepresentations;
|
|
bool bCollisionExists; //the shortcut to know if collideables exist for each prop
|
|
bool bPhysicsExists; //the shortcut to know if physics obects exist for each prop
|
|
PS_SD_Static_World_StaticProps_t( void ) : bCollisionExists( false ), bPhysicsExists( false ) { };
|
|
};
|
|
|
|
struct PS_SD_Static_World_t //stuff in front of the portal
|
|
{
|
|
PS_SD_Static_World_Brushes_t Brushes;
|
|
PS_SD_Static_World_StaticProps_t StaticProps;
|
|
};
|
|
|
|
struct PS_SD_Static_Wall_Local_Tube_t //a minimal tube, an object must fit inside this to be eligible for portaling
|
|
{
|
|
CUtlVector<CPolyhedron *> Polyhedrons; //the building blocks of more complex collision
|
|
CPhysCollide *pCollideable;
|
|
|
|
#ifndef CLIENT_DLL
|
|
IPhysicsObject *pPhysicsObject;
|
|
PS_SD_Static_Wall_Local_Tube_t() : pCollideable(NULL), pPhysicsObject(NULL) {};
|
|
#else
|
|
PS_SD_Static_Wall_Local_Tube_t() : pCollideable(NULL) {};
|
|
#endif
|
|
};
|
|
|
|
struct PS_SD_Static_Wall_Local_Brushes_t
|
|
{
|
|
CUtlVector<CPolyhedron *> Polyhedrons; //the building blocks of more complex collision
|
|
CPhysCollide *pCollideable;
|
|
|
|
#ifndef CLIENT_DLL
|
|
IPhysicsObject *pPhysicsObject;
|
|
PS_SD_Static_Wall_Local_Brushes_t() : pCollideable(NULL), pPhysicsObject(NULL) {};
|
|
#else
|
|
PS_SD_Static_Wall_Local_Brushes_t() : pCollideable(NULL) {};
|
|
#endif
|
|
};
|
|
|
|
struct PS_SD_Static_Wall_Local_t //things in the wall that are completely independant of having a linked portal
|
|
{
|
|
PS_SD_Static_Wall_Local_Tube_t Tube;
|
|
PS_SD_Static_Wall_Local_Brushes_t Brushes;
|
|
};
|
|
|
|
struct PS_SD_Static_Wall_RemoteTransformedToLocal_Brushes_t
|
|
{
|
|
IPhysicsObject *pPhysicsObject;
|
|
PS_SD_Static_Wall_RemoteTransformedToLocal_Brushes_t() : pPhysicsObject(NULL) {};
|
|
};
|
|
|
|
struct PS_SD_Static_Wall_RemoteTransformedToLocal_StaticProps_t
|
|
{
|
|
CUtlVector<IPhysicsObject *> PhysicsObjects;
|
|
};
|
|
|
|
struct PS_SD_Static_Wall_RemoteTransformedToLocal_t //things taken from the linked portal's "World" collision and transformed into local space
|
|
{
|
|
PS_SD_Static_Wall_RemoteTransformedToLocal_Brushes_t Brushes;
|
|
PS_SD_Static_Wall_RemoteTransformedToLocal_StaticProps_t StaticProps;
|
|
};
|
|
|
|
struct PS_SD_Static_Wall_t //stuff behind the portal
|
|
{
|
|
PS_SD_Static_Wall_Local_t Local;
|
|
#ifndef CLIENT_DLL
|
|
PS_SD_Static_Wall_RemoteTransformedToLocal_t RemoteTransformedToLocal;
|
|
#endif
|
|
};
|
|
|
|
struct PS_SD_Static_SurfaceProperties_t //surface properties to pretend every collideable here is using
|
|
{
|
|
int contents;
|
|
csurface_t surface;
|
|
CBaseEntity *pEntity;
|
|
};
|
|
|
|
struct PS_SD_Static_t //stuff that doesn't move around
|
|
{
|
|
PS_SD_Static_World_t World;
|
|
PS_SD_Static_Wall_t Wall;
|
|
PS_SD_Static_SurfaceProperties_t SurfaceProperties;
|
|
};
|
|
|
|
class CPhysicsShadowClone;
|
|
|
|
struct PS_SD_Dynamic_PhysicsShadowClones_t
|
|
{
|
|
CUtlVector<CBaseEntity *> ShouldCloneFromMain; //a list of entities that should be cloned from main if physics simulation is enabled
|
|
//in single-environment mode, this helps us track who should collide with who
|
|
|
|
CUtlVector<CPhysicsShadowClone *> FromLinkedPortal;
|
|
};
|
|
|
|
struct PS_SD_Dynamic_t //stuff that moves around
|
|
{
|
|
unsigned int EntFlags[MAX_EDICTS]; //flags maintained for every entity in the world based on its index
|
|
|
|
PS_SD_Dynamic_PhysicsShadowClones_t ShadowClones;
|
|
|
|
CUtlVector<CBaseEntity *> OwnedEntities;
|
|
|
|
PS_SD_Dynamic_t()
|
|
{
|
|
memset( EntFlags, 0, sizeof( EntFlags ) );
|
|
}
|
|
};
|
|
|
|
class CPSCollisionEntity;
|
|
|
|
struct PS_SimulationData_t //compartmentalized data for coherent management
|
|
{
|
|
PS_SD_Static_t Static;
|
|
|
|
#ifndef CLIENT_DLL
|
|
PS_SD_Dynamic_t Dynamic;
|
|
|
|
IPhysicsEnvironment *pPhysicsEnvironment;
|
|
CPSCollisionEntity *pCollisionEntity; //the entity we'll be tying physics objects to for collision
|
|
|
|
PS_SimulationData_t() : pPhysicsEnvironment(NULL), pCollisionEntity(NULL) {};
|
|
#endif
|
|
};
|
|
|
|
struct PS_InternalData_t
|
|
{
|
|
PS_PlacementData_t Placement;
|
|
PS_SimulationData_t Simulation;
|
|
};
|
|
|
|
|
|
class CPortalSimulator
|
|
{
|
|
public:
|
|
CPortalSimulator( void );
|
|
~CPortalSimulator( void );
|
|
|
|
void MoveTo( const Vector &ptCenter, const QAngle &angles );
|
|
void ClearEverything( void );
|
|
|
|
void AttachTo( CPortalSimulator *pLinkedPortalSimulator );
|
|
void DetachFromLinked( void ); //detach portals to sever the connection, saves work when planning on moving both portals
|
|
CPortalSimulator *GetLinkedPortalSimulator( void ) const;
|
|
|
|
void SetPortalSimulatorCallbacks( CPortalSimulatorEventCallbacks *pCallbacks );
|
|
|
|
bool IsReadyToSimulate( void ) const; //is active and linked to another portal
|
|
|
|
void SetCollisionGenerationEnabled( bool bEnabled ); //enable/disable collision generation for the hole in the wall, needed for proper vphysics simulation
|
|
bool IsCollisionGenerationEnabled( void ) const;
|
|
|
|
void SetVPhysicsSimulationEnabled( bool bEnabled ); //enable/disable vphysics simulation. Will automatically update the linked portal to be the same
|
|
bool IsSimulatingVPhysics( void ) const; //this portal is setup to handle any physically simulated object, false means the portal is handling player movement only
|
|
|
|
bool EntityIsInPortalHole( CBaseEntity *pEntity ) const; //true if the entity is within the portal cutout bounds and crossing the plane. Not just *near* the portal
|
|
bool EntityHitBoxExtentIsInPortalHole( CBaseAnimating *pBaseAnimating ) const; //true if the entity is within the portal cutout bounds and crossing the plane. Not just *near* the portal
|
|
void RemoveEntityFromPortalHole( CBaseEntity *pEntity ); //if the entity is in the portal hole, this forcibly moves it out by any means possible
|
|
|
|
bool RayIsInPortalHole( const Ray_t &ray ) const; //traces a ray against the same detector for EntityIsInPortalHole(), bias is towards false positives
|
|
|
|
#ifndef CLIENT_DLL
|
|
int GetMoveableOwnedEntities( CBaseEntity **pEntsOut, int iEntOutLimit ); //gets owned entities that aren't either world or static props. Excludes fake portal ents such as physics clones
|
|
|
|
static CPortalSimulator *GetSimulatorThatOwnsEntity( const CBaseEntity *pEntity ); //fairly cheap to call
|
|
static CPortalSimulator *GetSimulatorThatCreatedPhysicsObject( const IPhysicsObject *pObject, PS_PhysicsObjectSourceType_t *pOut_SourceType = NULL );
|
|
static void Pre_UTIL_Remove( CBaseEntity *pEntity );
|
|
static void Post_UTIL_Remove( CBaseEntity *pEntity );
|
|
|
|
//these three really should be made internal and the public interface changed to a "watch this entity" setup
|
|
void TakeOwnershipOfEntity( CBaseEntity *pEntity ); //general ownership, not necessarily physics ownership
|
|
void ReleaseOwnershipOfEntity( CBaseEntity *pEntity, bool bMovingToLinkedSimulator = false ); //if bMovingToLinkedSimulator is true, the code skips some steps that are going to be repeated when the entity is added to the other simulator
|
|
void ReleaseAllEntityOwnership( void ); //go back to not owning any entities
|
|
|
|
//void TeleportEntityToLinkedPortal( CBaseEntity *pEntity );
|
|
void StartCloningEntity( CBaseEntity *pEntity );
|
|
void StopCloningEntity( CBaseEntity *pEntity );
|
|
|
|
bool OwnsEntity( const CBaseEntity *pEntity ) const;
|
|
bool OwnsPhysicsForEntity( const CBaseEntity *pEntity ) const;
|
|
|
|
bool CreatedPhysicsObject( const IPhysicsObject *pObject, PS_PhysicsObjectSourceType_t *pOut_SourceType = NULL ) const; //true if the physics object was generated by this portal simulator
|
|
|
|
static void PrePhysFrame( void );
|
|
static void PostPhysFrame( void );
|
|
|
|
#endif //#ifndef CLIENT_DLL
|
|
|
|
#ifdef PORTAL_SIMULATORS_EMBED_GUID
|
|
int GetPortalSimulatorGUID( void ) const { return m_iPortalSimulatorGUID; };
|
|
#endif
|
|
|
|
protected:
|
|
bool m_bLocalDataIsReady; //this side of the portal is properly setup, no guarantees as to linkage to another portal
|
|
bool m_bSimulateVPhysics;
|
|
bool m_bGenerateCollision;
|
|
bool m_bSharedCollisionConfiguration; //when portals are in certain configurations, they need to cross-clip and share some collision data and things get nasty. For the love of all that is holy, pray that this is false.
|
|
CPortalSimulator *m_pLinkedPortal;
|
|
bool m_bInCrossLinkedFunction; //A flag to mark that we're already in a linked function and that the linked portal shouldn't call our side
|
|
CPortalSimulatorEventCallbacks *m_pCallbacks;
|
|
#ifdef PORTAL_SIMULATORS_EMBED_GUID
|
|
int m_iPortalSimulatorGUID;
|
|
#endif
|
|
|
|
struct
|
|
{
|
|
bool bPolyhedronsGenerated;
|
|
bool bLocalCollisionGenerated;
|
|
bool bLinkedCollisionGenerated;
|
|
bool bLocalPhysicsGenerated;
|
|
bool bLinkedPhysicsGenerated;
|
|
} m_CreationChecklist;
|
|
|
|
friend class CPSCollisionEntity;
|
|
|
|
#ifndef CLIENT_DLL //physics handled purely by server side
|
|
void TakePhysicsOwnership( CBaseEntity *pEntity );
|
|
void ReleasePhysicsOwnership( CBaseEntity *pEntity, bool bContinuePhysicsCloning = true, bool bMovingToLinkedSimulator = false );
|
|
|
|
void CreateAllPhysics( void );
|
|
void CreateMinimumPhysics( void ); //stuff needed by any part of physics simulations
|
|
void CreateLocalPhysics( void );
|
|
void CreateLinkedPhysics( void );
|
|
|
|
void ClearAllPhysics( void );
|
|
void ClearMinimumPhysics( void );
|
|
void ClearLocalPhysics( void );
|
|
void ClearLinkedPhysics( void );
|
|
|
|
void ClearLinkedEntities( void ); //gets rid of transformed shadow clones
|
|
#endif
|
|
|
|
void CreateAllCollision( void );
|
|
void CreateLocalCollision( void );
|
|
void CreateLinkedCollision( void );
|
|
|
|
void ClearAllCollision( void );
|
|
void ClearLinkedCollision( void );
|
|
void ClearLocalCollision( void );
|
|
|
|
void CreatePolyhedrons( void ); //carves up the world around the portal's position into sets of polyhedrons
|
|
void ClearPolyhedrons( void );
|
|
|
|
void UpdateLinkMatrix( void );
|
|
|
|
void MarkAsOwned( CBaseEntity *pEntity );
|
|
void MarkAsReleased( CBaseEntity *pEntity );
|
|
|
|
PS_InternalData_t m_InternalData;
|
|
|
|
public:
|
|
const PS_InternalData_t &m_DataAccess;
|
|
|
|
friend class CPS_AutoGameSys_EntityListener;
|
|
};
|
|
|
|
extern CUtlVector<CPortalSimulator *> const &g_PortalSimulators;
|
|
|
|
|
|
#ifndef CLIENT_DLL
|
|
class CPSCollisionEntity : public CBaseEntity
|
|
{
|
|
DECLARE_CLASS( CPSCollisionEntity, CBaseEntity );
|
|
private:
|
|
CPortalSimulator *m_pOwningSimulator;
|
|
|
|
public:
|
|
CPSCollisionEntity( void );
|
|
virtual ~CPSCollisionEntity( void );
|
|
|
|
virtual void Spawn( void );
|
|
virtual void Activate( void );
|
|
virtual int ObjectCaps( void );
|
|
virtual IPhysicsObject *VPhysicsGetObject( void );
|
|
virtual int VPhysicsGetObjectList( IPhysicsObject **pList, int listMax );
|
|
virtual void UpdateOnRemove( void );
|
|
virtual bool ShouldCollide( int collisionGroup, int contentsMask ) const;
|
|
virtual void VPhysicsCollision( int index, gamevcollisionevent_t *pEvent ) {}
|
|
virtual void VPhysicsFriction( IPhysicsObject *pObject, float energy, int surfaceProps, int surfacePropsHit ) {}
|
|
|
|
static bool IsPortalSimulatorCollisionEntity( const CBaseEntity *pEntity );
|
|
friend class CPortalSimulator;
|
|
};
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef CLIENT_DLL
|
|
inline bool CPortalSimulator::OwnsEntity( const CBaseEntity *pEntity ) const
|
|
{
|
|
return ((m_InternalData.Simulation.Dynamic.EntFlags[pEntity->entindex()] & PSEF_OWNS_ENTITY) != 0);
|
|
}
|
|
|
|
inline bool CPortalSimulator::OwnsPhysicsForEntity( const CBaseEntity *pEntity ) const
|
|
{
|
|
return ((m_InternalData.Simulation.Dynamic.EntFlags[pEntity->entindex()] & PSEF_OWNS_PHYSICS) != 0);
|
|
}
|
|
#endif
|
|
|
|
inline bool CPortalSimulator::IsReadyToSimulate( void ) const
|
|
{
|
|
return m_bLocalDataIsReady && m_pLinkedPortal && m_pLinkedPortal->m_bLocalDataIsReady;
|
|
}
|
|
|
|
inline bool CPortalSimulator::IsSimulatingVPhysics( void ) const
|
|
{
|
|
return m_bSimulateVPhysics;
|
|
}
|
|
|
|
inline bool CPortalSimulator::IsCollisionGenerationEnabled( void ) const
|
|
{
|
|
return m_bGenerateCollision;
|
|
}
|
|
|
|
inline CPortalSimulator *CPortalSimulator::GetLinkedPortalSimulator( void ) const
|
|
{
|
|
return m_pLinkedPortal;
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif //#ifndef PORTALSIMULATION_H
|
|
|