|
|
//========= Copyright � 1996-2005, Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//=============================================================================//
#ifndef COLLISIONPROPERTY_H
#define COLLISIONPROPERTY_H
#ifdef _WIN32
#pragma once
#endif
#include "networkvar.h"
#include "engine/ICollideable.h"
#include "mathlib/vector.h"
#include "ispatialpartition.h"
//-----------------------------------------------------------------------------
// Forward declarations
//-----------------------------------------------------------------------------
class CBaseEntity; class IHandleEntity; class QAngle; class Vector; struct Ray_t; class IPhysicsObject;
//-----------------------------------------------------------------------------
// Force spatial partition updates (to avoid threading problems caused by lazy update)
//-----------------------------------------------------------------------------
void UpdateDirtySpatialPartitionEntities();
//-----------------------------------------------------------------------------
// Specifies how to compute the surrounding box
//-----------------------------------------------------------------------------
enum SurroundingBoundsType_t { USE_OBB_COLLISION_BOUNDS = 0, USE_BEST_COLLISION_BOUNDS, // Always use the best bounds (most expensive)
USE_HITBOXES, USE_SPECIFIED_BOUNDS, USE_GAME_CODE, USE_ROTATION_EXPANDED_BOUNDS, USE_COLLISION_BOUNDS_NEVER_VPHYSICS, USE_ROTATION_EXPANDED_SEQUENCE_BOUNDS,
SURROUNDING_TYPE_BIT_COUNT = 3 };
//-----------------------------------------------------------------------------
// Encapsulates collision representation for an entity
//-----------------------------------------------------------------------------
class CCollisionProperty : public ICollideable { DECLARE_CLASS_NOBASE( CCollisionProperty ); DECLARE_EMBEDDED_NETWORKVAR(); DECLARE_PREDICTABLE();
#ifdef GAME_DLL
DECLARE_DATADESC(); #endif
public: CCollisionProperty(); ~CCollisionProperty();
void Init( CBaseEntity *pEntity );
// Methods of ICollideable
virtual IHandleEntity *GetEntityHandle(); virtual const Vector& OBBMins( ) const; virtual const Vector& OBBMaxs( ) const; virtual void WorldSpaceTriggerBounds( Vector *pVecWorldMins, Vector *pVecWorldMaxs ) const; virtual bool TestCollision( const Ray_t &ray, unsigned int fContentsMask, trace_t& tr ); virtual bool TestHitboxes( const Ray_t &ray, unsigned int fContentsMask, trace_t& tr ); virtual int GetCollisionModelIndex(); virtual const model_t* GetCollisionModel(); virtual const Vector& GetCollisionOrigin() const; virtual const QAngle& GetCollisionAngles() const; virtual const matrix3x4_t& CollisionToWorldTransform() const; virtual SolidType_t GetSolid() const; virtual int GetSolidFlags() const; virtual IClientUnknown* GetIClientUnknown(); virtual int GetCollisionGroup() const; virtual void WorldSpaceSurroundingBounds( Vector *pVecMins, Vector *pVecMaxs ); virtual uint GetRequiredTriggerFlags() const; virtual const matrix3x4_t *GetRootParentToWorldTransform() const; virtual IPhysicsObject *GetVPhysicsObject() const;
public: // Spatial partition management
void CreatePartitionHandle(); void DestroyPartitionHandle(); SpatialPartitionHandle_t GetPartitionHandle() const;
// Marks the spatial partition dirty
void MarkPartitionHandleDirty();
// Sets the collision bounds + the size (OBB)
void SetCollisionBounds( const Vector& mins, const Vector &maxs );
// Sets special trigger bounds. The bloat amount indicates how much bigger the
// trigger bounds should be beyond the bounds set in SetCollisionBounds
// This method will also set the FSOLID flag FSOLID_USE_TRIGGER_BOUNDS
void UseTriggerBounds( bool bEnable, float flBloat = 0.0f );
// Sets the method by which the surrounding collision bounds is set
// You must pass in values for mins + maxs if you select the USE_SPECIFIED_BOUNDS type.
void SetSurroundingBoundsType( SurroundingBoundsType_t type, const Vector *pMins = NULL, const Vector *pMaxs = NULL ); SurroundingBoundsType_t GetSurroundingBoundsType() const;
// Sets the solid type (which type of collision representation)
void SetSolid( SolidType_t val );
// Methods related to size. The OBB here is measured in CollisionSpace
// (specified by GetCollisionToWorld)
const Vector& OBBSize( ) const;
// Returns a radius (or the square of the radius) of a sphere
// *centered at the world space center* bounding the collision representation
// of the entity. NOTE: The world space center *may* move when the entity rotates.
float BoundingRadius() const; float BoundingRadius2D() const;
// Returns the center of the OBB in collision space
const Vector & OBBCenter( ) const;
// center point of entity measured in world space
// NOTE: This point *may* move when the entity moves depending on
// which solid type is being used.
const Vector & WorldSpaceCenter( ) const;
// Methods related to solid flags
void ClearSolidFlags( void ); void RemoveSolidFlags( int flags ); void AddSolidFlags( int flags ); bool IsSolidFlagSet( int flagMask ) const; void SetSolidFlags( int flags ); bool IsSolid() const;
// Updates the spatial partition
void UpdatePartition( );
// Are the bounds defined in entity space?
bool IsBoundsDefinedInEntitySpace() const;
// Transforms a point in OBB space to world space
const Vector & CollisionToWorldSpace( const Vector &in, Vector *pResult ) const;
// Transforms a point in world space to OBB space
const Vector & WorldToCollisionSpace( const Vector &in, Vector *pResult ) const;
// Transforms a direction in world space to OBB space
const Vector & WorldDirectionToCollisionSpace( const Vector &in, Vector *pResult ) const;
// Selects a random point in the bounds given the normalized 0-1 bounds
void RandomPointInBounds( const Vector &vecNormalizedMins, const Vector &vecNormalizedMaxs, Vector *pPoint) const;
// Is a worldspace point within the bounds of the OBB?
bool IsPointInBounds( const Vector &vecWorldPt ) const;
// Computes a bounding box in world space surrounding the collision bounds
void WorldSpaceAABB( Vector *pWorldMins, Vector *pWorldMaxs ) const;
// Computes a "normalized" point (range 0,0,0 - 1,1,1) in collision space
// Useful for things like getting a point 75% of the way along z on the OBB, for example
const Vector & NormalizedToCollisionSpace( const Vector &in, Vector *pResult ) const;
// Computes a "normalized" point (range 0,0,0 - 1,1,1) in world space
const Vector & NormalizedToWorldSpace( const Vector &in, Vector *pResult ) const;
// Transforms a point in world space to normalized space
const Vector & WorldToNormalizedSpace( const Vector &in, Vector *pResult ) const;
// Transforms a point in collision space to normalized space
const Vector & CollisionToNormalizedSpace( const Vector &in, Vector *pResult ) const;
// Computes the nearest point in the OBB to a point specified in world space
void CalcNearestPoint( const Vector &vecWorldPt, Vector *pVecNearestWorldPt ) const;
// Computes the distance from a point in world space to the OBB
float CalcDistanceFromPoint( const Vector &vecWorldPt ) const; float CalcSqrDistanceFromPoint( const Vector &vecWorldPt ) const;
// Does a rotation make us need to recompute the surrounding box?
bool DoesRotationInvalidateSurroundingBox( ) const;
// Does VPhysicsUpdate make us need to recompute the surrounding box?
bool DoesVPhysicsInvalidateSurroundingBox( ) const;
// Does a sequence change make us need to recompute the surrounding box?
bool DoesSequenceChangeInvalidateSurroundingBox( ) const;
// Marks the entity has having a dirty surrounding box
void MarkSurroundingBoundsDirty();
// Compute the largest dot product of the OBB and the specified direction vector
float ComputeSupportMap( const Vector &vecDirection ) const;
private: // Transforms an AABB measured in collision space to a box that surrounds it in world space
void CollisionAABBToWorldAABB( const Vector &entityMins, const Vector &entityMaxs, Vector *pWorldMins, Vector *pWorldMaxs ) const;
// Expand trigger bounds..
void ComputeVPhysicsSurroundingBox( Vector *pVecWorldMins, Vector *pVecWorldMaxs );
// Expand trigger bounds..
bool ComputeHitboxSurroundingBox( Vector *pVecWorldMins, Vector *pVecWorldMaxs ); bool ComputeEntitySpaceHitboxSurroundingBox( Vector *pVecWorldMins, Vector *pVecWorldMaxs );
// Computes the surrounding collision bounds based on the current sequence box
void ComputeOBBBounds( Vector *pVecWorldMins, Vector *pVecWorldMaxs );
// Computes the surrounding collision bounds from the current sequence box
void ComputeRotationExpandedSequenceBounds( Vector *pVecWorldMins, Vector *pVecWorldMaxs );
// Computes the surrounding collision bounds based on whatever algorithm we want...
void ComputeCollisionSurroundingBox( bool bUseVPhysics, Vector *pVecWorldMins, Vector *pVecWorldMaxs );
// Computes the surrounding collision bounds from the OBB (not vphysics)
void ComputeRotationExpandedBounds( Vector *pVecWorldMins, Vector *pVecWorldMaxs );
// Computes the surrounding collision bounds based on whatever algorithm we want...
void ComputeSurroundingBox( Vector *pVecWorldMins, Vector *pVecWorldMaxs );
// Check for untouch
void CheckForUntouch();
// Updates the spatial partition
void UpdateServerPartitionMask( ); uint ComputeServerPartitionMask( ); inline const Vector& GetCollisionOrigin_Inline() const;
// Outer
CBaseEntity *GetOuter(); const CBaseEntity *GetOuter() const;
private: CBaseEntity *m_pOuter;
// BEGIN PREDICTION DATA COMPACTION (these fields are together to allow for faster copying in prediction system)
CNetworkVector( m_vecMins ); CNetworkVector( m_vecMaxs ); CNetworkVar( unsigned short, m_usSolidFlags ); // One of the SOLID_ defines. Use GetSolid/SetSolid.
CNetworkVar( unsigned char, m_nSolidType ); CNetworkVar( unsigned char , m_triggerBloat ); // END PREDICTION DATA COMPACTION
float m_flRadius;
// Spatial partition
SpatialPartitionHandle_t m_Partition; CNetworkVar( unsigned char, m_nSurroundType );
// SUCKY: We didn't use to have to store this previously
// but storing it here means that we can network it + avoid a ton of
// client-side mismatch problems
CNetworkVector( m_vecSpecifiedSurroundingMins ); CNetworkVector( m_vecSpecifiedSurroundingMaxs );
// Cached off world-aligned surrounding bounds
#if 0
short m_surroundingMins[3]; short m_surroundingMaxs[3]; #else
Vector m_vecSurroundingMins; Vector m_vecSurroundingMaxs; #endif
// pointer to the entity's physics object (vphysics.dll)
//IPhysicsObject *m_pPhysicsObject;
friend class CBaseEntity; };
//-----------------------------------------------------------------------------
// For networking this bad boy
//-----------------------------------------------------------------------------
#ifdef CLIENT_DLL
EXTERN_RECV_TABLE( DT_CollisionProperty ); #else
EXTERN_SEND_TABLE( DT_CollisionProperty ); #endif
//-----------------------------------------------------------------------------
// Inline methods
//-----------------------------------------------------------------------------
inline CBaseEntity *CCollisionProperty::GetOuter() { return m_pOuter; }
inline const CBaseEntity *CCollisionProperty::GetOuter() const { return m_pOuter; }
//-----------------------------------------------------------------------------
// Spatial partition
//-----------------------------------------------------------------------------
inline SpatialPartitionHandle_t CCollisionProperty::GetPartitionHandle() const { return m_Partition; }
inline SurroundingBoundsType_t CCollisionProperty::GetSurroundingBoundsType() const { return (SurroundingBoundsType_t)m_nSurroundType.Get(); }
//-----------------------------------------------------------------------------
// Methods related to size
//-----------------------------------------------------------------------------
inline const Vector& CCollisionProperty::OBBSize( ) const { // NOTE: Could precache this, but it's not used that often..
Vector &temp = AllocTempVector(); VectorSubtract( m_vecMaxs, m_vecMins, temp ); return temp; }
//-----------------------------------------------------------------------------
// Bounding radius size
//-----------------------------------------------------------------------------
inline float CCollisionProperty::BoundingRadius() const { return m_flRadius; }
//-----------------------------------------------------------------------------
// Methods relating to solid flags
//-----------------------------------------------------------------------------
inline bool CCollisionProperty::IsBoundsDefinedInEntitySpace() const { return (( m_usSolidFlags & FSOLID_FORCE_WORLD_ALIGNED ) == 0 ) && ( m_nSolidType != SOLID_BBOX ) && ( m_nSolidType != SOLID_NONE ); }
inline void CCollisionProperty::ClearSolidFlags( void ) { SetSolidFlags( 0 ); }
inline void CCollisionProperty::RemoveSolidFlags( int flags ) { SetSolidFlags( m_usSolidFlags & ~flags ); }
inline void CCollisionProperty::AddSolidFlags( int flags ) { SetSolidFlags( m_usSolidFlags | flags ); }
inline int CCollisionProperty::GetSolidFlags( void ) const { return m_usSolidFlags; }
inline bool CCollisionProperty::IsSolidFlagSet( int flagMask ) const { return (m_usSolidFlags & flagMask) != 0; }
inline bool CCollisionProperty::IsSolid() const { return ::IsSolid( (SolidType_t)(unsigned char)m_nSolidType, m_usSolidFlags ); }
//-----------------------------------------------------------------------------
// Returns the center in OBB space
//-----------------------------------------------------------------------------
inline const Vector& CCollisionProperty::OBBCenter( ) const { Vector &vecResult = AllocTempVector(); VectorLerp( m_vecMins, m_vecMaxs, 0.5f, vecResult ); return vecResult; }
//-----------------------------------------------------------------------------
// center point of entity
//-----------------------------------------------------------------------------
inline const Vector &CCollisionProperty::WorldSpaceCenter( ) const { Vector &vecResult = AllocTempVector(); CollisionToWorldSpace( OBBCenter(), &vecResult ); return vecResult; }
//-----------------------------------------------------------------------------
// Transforms a point in OBB space to world space
//-----------------------------------------------------------------------------
inline const Vector &CCollisionProperty::CollisionToWorldSpace( const Vector &in, Vector *pResult ) const { // Makes sure we don't re-use the same temp twice
if ( !IsBoundsDefinedInEntitySpace() || ( GetCollisionAngles() == vec3_angle ) ) { VectorAdd( in, GetCollisionOrigin(), *pResult ); } else { VectorTransform( in, CollisionToWorldTransform(), *pResult ); } return *pResult; }
//-----------------------------------------------------------------------------
// Transforms a point in world space to OBB space
//-----------------------------------------------------------------------------
inline const Vector &CCollisionProperty::WorldToCollisionSpace( const Vector &in, Vector *pResult ) const { if ( !IsBoundsDefinedInEntitySpace() || ( GetCollisionAngles() == vec3_angle ) ) { VectorSubtract( in, GetCollisionOrigin(), *pResult ); } else { VectorITransform( in, CollisionToWorldTransform(), *pResult ); } return *pResult; }
//-----------------------------------------------------------------------------
// Transforms a direction in world space to OBB space
//-----------------------------------------------------------------------------
inline const Vector & CCollisionProperty::WorldDirectionToCollisionSpace( const Vector &in, Vector *pResult ) const { if ( !IsBoundsDefinedInEntitySpace() || ( GetCollisionAngles() == vec3_angle ) ) { *pResult = in; } else { VectorIRotate( in, CollisionToWorldTransform(), *pResult ); } return *pResult; }
//-----------------------------------------------------------------------------
// Computes a bounding box in world space surrounding the collision bounds
//-----------------------------------------------------------------------------
inline void CCollisionProperty::WorldSpaceAABB( Vector *pWorldMins, Vector *pWorldMaxs ) const { CollisionAABBToWorldAABB( m_vecMins, m_vecMaxs, pWorldMins, pWorldMaxs ); }
//-----------------------------------------------------------------------------
// Does a rotation make us need to recompute the surrounding box?
//-----------------------------------------------------------------------------
inline bool CCollisionProperty::DoesSequenceChangeInvalidateSurroundingBox( ) const { return ( m_nSurroundType == USE_ROTATION_EXPANDED_SEQUENCE_BOUNDS ); }
//-----------------------------------------------------------------------------
// Does a rotation make us need to recompute the surrounding box?
//-----------------------------------------------------------------------------
inline bool CCollisionProperty::DoesRotationInvalidateSurroundingBox( ) const { if ( IsSolidFlagSet(FSOLID_ROOT_PARENT_ALIGNED) ) return true;
switch ( m_nSurroundType ) { case USE_COLLISION_BOUNDS_NEVER_VPHYSICS: case USE_OBB_COLLISION_BOUNDS: case USE_BEST_COLLISION_BOUNDS: return IsBoundsDefinedInEntitySpace();
// In the case of game code, we don't really know, so we have to assume it does
case USE_HITBOXES: case USE_GAME_CODE: return true;
case USE_ROTATION_EXPANDED_BOUNDS: case USE_SPECIFIED_BOUNDS: case USE_ROTATION_EXPANDED_SEQUENCE_BOUNDS: return false;
default: Assert(0); return true; } }
#endif // COLLISIONPROPERTY_H
|