|
|
//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose: Clones a physics object (usually with a matrix transform applied)
//
// $NoKeywords: $
//=============================================================================//
#include "cbase.h"
#include "physicsshadowclone.h"
#include "portal_util_shared.h"
#include "vphysics/object_hash.h"
#include "trains.h"
#include "props.h"
#include "model_types.h"
#include "portal/weapon_physcannon.h" //grab controllers
#include "PortalSimulation.h"
#define MAX_SHADOW_CLONE_COUNT 200
static int g_iShadowCloneCount = 0; ConVar sv_debug_physicsshadowclones("sv_debug_physicsshadowclones", "0", FCVAR_REPLICATED ); ConVar sv_use_shadow_clones( "sv_use_shadow_clones", "1", FCVAR_REPLICATED | FCVAR_CHEAT ); //should we create shadow clones?
static void DrawDebugOverlayForShadowClone( CPhysicsShadowClone *pClone );
LINK_ENTITY_TO_CLASS( physicsshadowclone, CPhysicsShadowClone );
static CUtlVector<CPhysicsShadowClone *> s_ActiveShadowClones; CUtlVector<CPhysicsShadowClone *> const &CPhysicsShadowClone::g_ShadowCloneList = s_ActiveShadowClones; static bool s_IsShadowClone[MAX_EDICTS] = { false };
static CPhysicsShadowCloneLL *s_EntityClones[MAX_EDICTS] = { NULL }; struct ShadowCloneLLEntryManager { CPhysicsShadowCloneLL m_ShadowCloneLLEntries[MAX_SHADOW_CLONE_COUNT]; CPhysicsShadowCloneLL *m_pFreeShadowCloneLLEntries[MAX_SHADOW_CLONE_COUNT]; int m_iUsedEntryIndex;
ShadowCloneLLEntryManager( void ) { m_iUsedEntryIndex = 0; for( int i = 0; i != MAX_SHADOW_CLONE_COUNT; ++i ) { m_pFreeShadowCloneLLEntries[i] = &m_ShadowCloneLLEntries[i]; } }
inline CPhysicsShadowCloneLL *Alloc( void ) { return m_pFreeShadowCloneLLEntries[m_iUsedEntryIndex++]; }
inline void Free( CPhysicsShadowCloneLL *pFree ) { m_pFreeShadowCloneLLEntries[--m_iUsedEntryIndex] = pFree; } }; static ShadowCloneLLEntryManager s_SCLLManager;
CPhysicsShadowClone::CPhysicsShadowClone( void ) { m_matrixShadowTransform.Identity(); m_matrixShadowTransform_Inverse.Identity(); m_bShadowTransformIsIdentity = true; s_ActiveShadowClones.AddToTail( this ); }
CPhysicsShadowClone::~CPhysicsShadowClone( void ) { VPhysicsDestroyObject(); VPhysicsSetObject( NULL ); m_hClonedEntity = NULL; s_ActiveShadowClones.FindAndRemove( this ); //also removed in UpdateOnRemove()
Assert( s_IsShadowClone[entindex()] == true ); s_IsShadowClone[entindex()] = false; }
void CPhysicsShadowClone::UpdateOnRemove( void ) { CBaseEntity *pSource = m_hClonedEntity; if( pSource ) { CPhysicsShadowCloneLL *pCloneListHead = s_EntityClones[pSource->entindex()]; Assert( pCloneListHead != NULL );
CPhysicsShadowCloneLL *pFind = pCloneListHead; CPhysicsShadowCloneLL *pLast = pFind; while( pFind->pClone != this ) { pLast = pFind; Assert( pFind->pNext != NULL ); pFind = pFind->pNext; }
if( pFind == pCloneListHead ) { s_EntityClones[pSource->entindex()] = pFind->pNext; } else { pLast->pNext = pFind->pNext; } s_SCLLManager.Free( pFind ); } #ifdef _DEBUG
else { //verify that it didn't weasel into a list somewhere and get left behind
for( int i = 0; i != MAX_SHADOW_CLONE_COUNT; ++i ) { CPhysicsShadowCloneLL *pCloneSearch = s_EntityClones[i]; while( pCloneSearch ) { Assert( pCloneSearch->pClone != this ); pCloneSearch = pCloneSearch->pNext; } } } #endif
VPhysicsDestroyObject(); VPhysicsSetObject( NULL ); m_hClonedEntity = NULL; s_ActiveShadowClones.FindAndRemove( this ); //also removed in Destructor
BaseClass::UpdateOnRemove(); }
void CPhysicsShadowClone::Spawn( void ) { AddFlag( FL_DONTTOUCH ); AddEffects( EF_NODRAW | EF_NOSHADOW | EF_NORECEIVESHADOW );
FullSync( false ); m_bInAssumedSyncState = false; BaseClass::Spawn();
s_IsShadowClone[entindex()] = true; }
void CPhysicsShadowClone::FullSync( bool bAllowAssumedSync ) { Assert( IsMarkedForDeletion() == false );
CBaseEntity *pClonedEntity = m_hClonedEntity.Get();
if( pClonedEntity == NULL ) { AssertMsg( VPhysicsGetObject() != NULL, "Been linkless for more than this update, something should have killed this clone." ); SetMoveType( MOVETYPE_NONE ); SetSolid( SOLID_NONE ); SetSolidFlags( 0 ); SetCollisionGroup( COLLISION_GROUP_NONE ); VPhysicsDestroyObject(); return; }
SetGroundEntity( NULL );
bool bIsSynced = bAllowAssumedSync; bool bBigChanges = true; //assume there are, and be proven wrong
if( bAllowAssumedSync ) { IPhysicsObject *pSourceObjects[1024]; int iObjectCount = pClonedEntity->VPhysicsGetObjectList( pSourceObjects, 1024 );
//scan for really big differences that would definitely require a full sync
bBigChanges = ( iObjectCount != m_CloneLinks.Count() ); if( !bBigChanges ) { for( int i = 0; i != iObjectCount; ++i ) { IPhysicsObject *pSourcePhysics = pSourceObjects[i]; IPhysicsObject *pClonedPhysics = m_CloneLinks[i].pClone;
if( (pSourcePhysics != m_CloneLinks[i].pSource) || (pSourcePhysics->IsCollisionEnabled() != pClonedPhysics->IsCollisionEnabled()) ) { bBigChanges = true; bIsSynced = false; break; }
Vector ptSourcePosition, ptClonePosition; pSourcePhysics->GetPosition( &ptSourcePosition, NULL ); if( !m_bShadowTransformIsIdentity ) ptSourcePosition = m_matrixShadowTransform * ptSourcePosition;
pClonedPhysics->GetPosition( &ptClonePosition, NULL );
if( (ptClonePosition - ptSourcePosition).LengthSqr() > 2500.0f ) { bBigChanges = true; bIsSynced = false; break; } //Vector vSourceVelocity, vCloneVelocity;
if( !pSourcePhysics->IsAsleep() ) //only allow full syncrosity if the source entity is entirely asleep
bIsSynced = false;
if( m_bInAssumedSyncState && !pClonedPhysics->IsAsleep() ) bIsSynced = false; } } else { bIsSynced = false; }
bIsSynced = false;
if( bIsSynced ) { //good enough to skip a full update
if( !m_bInAssumedSyncState ) { //do one last sync
PartialSync( true );
//if we don't do this, objects just fall out of the world (it happens, I swear)
for( int i = m_CloneLinks.Count(); --i >= 0; ) { if( (m_CloneLinks[i].pSource->GetShadowController() == NULL) && m_CloneLinks[i].pClone->IsMotionEnabled() ) { //m_CloneLinks[i].pClone->SetVelocityInstantaneous( &vec3_origin, &vec3_origin );
//m_CloneLinks[i].pClone->SetVelocity( &vec3_origin, &vec3_origin );
m_CloneLinks[i].pClone->EnableGravity( false ); m_CloneLinks[i].pClone->EnableMotion( false ); m_CloneLinks[i].pClone->Sleep(); } }
m_bInAssumedSyncState = true; } if( sv_debug_physicsshadowclones.GetBool() ) DrawDebugOverlayForShadowClone( this );
return; } } m_bInAssumedSyncState = false;
//past this point, we're committed to a broad update
if( bBigChanges ) { MoveType_t sourceMoveType = pClonedEntity->GetMoveType();
IPhysicsObject *pPhysObject = pClonedEntity->VPhysicsGetObject(); if( (sourceMoveType == MOVETYPE_CUSTOM) || (sourceMoveType == MOVETYPE_STEP) || (sourceMoveType == MOVETYPE_WALK) || (pPhysObject && ( (pPhysObject->GetGameFlags() & FVPHYSICS_PLAYER_HELD) || (pPhysObject->GetShadowController() != NULL) ) ) ) { //#ifdef _DEBUG
SetMoveType( MOVETYPE_NONE ); //to kill an assert
//#endif
//PUSH should be used sparingly, you can't stand on a MOVETYPE_PUSH object :/
SetMoveType( MOVETYPE_VPHYSICS, pClonedEntity->GetMoveCollide() ); //either an unclonable movetype, or a shadow/held object
} /*else if(sourceMoveType == MOVETYPE_STEP)
{ //SetMoveType( MOVETYPE_NONE ); //to kill an assert
SetMoveType( MOVETYPE_VPHYSICS, pClonedEntity->GetMoveCollide() ); }*/ else { //if( m_bShadowTransformIsIdentity )
SetMoveType( sourceMoveType, pClonedEntity->GetMoveCollide() ); //else
//{
// SetMoveType( MOVETYPE_NONE ); //to kill an assert
// SetMoveType( MOVETYPE_PUSH, pClonedEntity->GetMoveCollide() );
//}
}
SolidType_t sourceSolidType = pClonedEntity->GetSolid(); if( sourceSolidType == SOLID_BBOX ) SetSolid( SOLID_VPHYSICS ); else SetSolid( sourceSolidType ); //SetSolid( SOLID_VPHYSICS );
SetElasticity( pClonedEntity->GetElasticity() ); SetFriction( pClonedEntity->GetFriction() );
int iSolidFlags = pClonedEntity->GetSolidFlags() | FSOLID_CUSTOMRAYTEST; if( m_bShadowTransformIsIdentity ) iSolidFlags |= FSOLID_CUSTOMBOXTEST; //need this at least for the player or they get stuck in themselves
else iSolidFlags &= ~FSOLID_FORCE_WORLD_ALIGNED; /*if( pClonedEntity->IsPlayer() )
{ iSolidFlags |= FSOLID_CUSTOMRAYTEST | FSOLID_CUSTOMBOXTEST; }*/
SetSolidFlags( iSolidFlags );
SetEffects( pClonedEntity->GetEffects() | (EF_NODRAW | EF_NOSHADOW | EF_NORECEIVESHADOW) );
SetCollisionGroup( pClonedEntity->GetCollisionGroup() );
SetModelIndex( pClonedEntity->GetModelIndex() ); SetModelName( pClonedEntity->GetModelName() );
if( modelinfo->GetModelType( pClonedEntity->GetModel() ) == mod_studio ) SetModel( STRING( pClonedEntity->GetModelName() ) );
CCollisionProperty *pClonedCollisionProp = pClonedEntity->CollisionProp(); SetSize( pClonedCollisionProp->OBBMins(), pClonedCollisionProp->OBBMaxs() ); }
FullSyncClonedPhysicsObjects( bBigChanges ); SyncEntity( true );
if( bBigChanges ) CollisionRulesChanged();
if( sv_debug_physicsshadowclones.GetBool() ) DrawDebugOverlayForShadowClone( this ); }
void CPhysicsShadowClone::SyncEntity( bool bPullChanges ) { m_bShouldUpSync = false;
CBaseEntity *pSource, *pDest; VMatrix *pTransform; if( bPullChanges ) { pSource = m_hClonedEntity.Get(); pDest = this; pTransform = &m_matrixShadowTransform;
if( pSource == NULL ) return; } else { pSource = this; pDest = m_hClonedEntity.Get(); pTransform = &m_matrixShadowTransform_Inverse;
if( pDest == NULL ) return; }
Vector ptOrigin, vVelocity; QAngle qAngles;
ptOrigin = pSource->GetAbsOrigin(); qAngles = pSource->GetAbsAngles(); vVelocity = pSource->GetAbsVelocity();
if( !m_bShadowTransformIsIdentity ) { ptOrigin = (*pTransform) * ptOrigin; qAngles = TransformAnglesToWorldSpace( qAngles, pTransform->As3x4() ); vVelocity = pTransform->ApplyRotation( vVelocity ); } //else
//{
// pDest->SetGroundEntity( pSource->GetGroundEntity() );
//}
if( (ptOrigin != pDest->GetAbsOrigin()) || (qAngles != pDest->GetAbsAngles()) ) { pDest->Teleport( &ptOrigin, &qAngles, NULL ); } if( vVelocity != pDest->GetAbsVelocity() ) { //pDest->IncrementInterpolationFrame();
pDest->SetAbsVelocity( vec3_origin ); //the two step process helps, I don't know why, but it does
pDest->ApplyAbsVelocityImpulse( vVelocity ); } }
static void FullSyncPhysicsObject( IPhysicsObject *pSource, IPhysicsObject *pDest, const VMatrix *pTransform, bool bTeleport ) { CGrabController *pGrabController = NULL;
if( !pSource->IsAsleep() ) pDest->Wake();
float fSavedMass = 0.0f, fSavedRotationalDamping; //setting mass to 0.0f purely to kill a warning that I can't seem to kill with pragmas
if( pSource->GetGameFlags() & FVPHYSICS_PLAYER_HELD ) { //CBasePlayer *pPlayer = UTIL_PlayerByIndex( 1 );
//Assert( pPlayer );
CBaseEntity *pLookingForEntity = (CBaseEntity *)pSource->GetGameData();
CBasePlayer *pHoldingPlayer = GetPlayerHoldingEntity( pLookingForEntity ); if( pHoldingPlayer ) { pGrabController = GetGrabControllerForPlayer( pHoldingPlayer );
if ( !pGrabController ) pGrabController = GetGrabControllerForPhysCannon( pHoldingPlayer->GetActiveWeapon() ); }
AssertMsg( pGrabController, "Physics object is held, but we can't find the holding controller." ); GetSavedParamsForCarriedPhysObject( pGrabController, pSource, &fSavedMass, &fSavedRotationalDamping ); }
//Boiler plate
{ pDest->SetGameIndex( pSource->GetGameIndex() ); //what's it do?
pDest->SetCallbackFlags( pSource->GetCallbackFlags() ); //wise?
pDest->SetGameFlags( pSource->GetGameFlags() | FVPHYSICS_NO_SELF_COLLISIONS | FVPHYSICS_IS_SHADOWCLONE ); pDest->SetMaterialIndex( pSource->GetMaterialIndex() ); pDest->SetContents( pSource->GetContents() );
pDest->EnableCollisions( pSource->IsCollisionEnabled() ); pDest->EnableGravity( pSource->IsGravityEnabled() ); pDest->EnableDrag( pSource->IsDragEnabled() ); pDest->EnableMotion( pSource->IsMotionEnabled() ); }
//Damping
{ float fSpeedDamp, fRotDamp; if( pGrabController ) { pSource->GetDamping( &fSpeedDamp, NULL ); pDest->SetDamping( &fSpeedDamp, &fSavedRotationalDamping ); } else { pSource->GetDamping( &fSpeedDamp, &fRotDamp ); pDest->SetDamping( &fSpeedDamp, &fRotDamp ); } }
//stuff that we really care about
{ if( pGrabController ) pDest->SetMass( fSavedMass ); else pDest->SetMass( pSource->GetMass() );
Vector ptOrigin, vVelocity, vAngularVelocity, vInertia; QAngle qAngles;
pSource->GetPosition( &ptOrigin, &qAngles ); pSource->GetVelocity( &vVelocity, &vAngularVelocity ); vInertia = pSource->GetInertia();
if( pTransform ) { #if 0
pDest->SetPositionMatrix( pTransform->As3x4(), true ); //works like we think?
#else
ptOrigin = (*pTransform) * ptOrigin; qAngles = TransformAnglesToWorldSpace( qAngles, pTransform->As3x4() ); vVelocity = pTransform->ApplyRotation( vVelocity ); vAngularVelocity = pTransform->ApplyRotation( vAngularVelocity ); #endif
}
//avoid oversetting variables (I think that even setting them to the same value they already are disrupts the delicate physics balance)
if( vInertia != pDest->GetInertia() ) pDest->SetInertia( vInertia );
Vector ptDestOrigin, vDestVelocity, vDestAngularVelocity; QAngle qDestAngles; pDest->GetPosition( &ptDestOrigin, &qDestAngles );
if( (ptOrigin != ptDestOrigin) || (qAngles != qDestAngles) ) pDest->SetPosition( ptOrigin, qAngles, bTeleport );
//pDest->SetVelocityInstantaneous( &vec3_origin, &vec3_origin );
//pDest->Sleep();
pDest->GetVelocity( &vDestVelocity, &vDestAngularVelocity );
if( (vVelocity != vDestVelocity) || (vAngularVelocity != vDestAngularVelocity) ) pDest->SetVelocityInstantaneous( &vVelocity, &vAngularVelocity );
IPhysicsShadowController *pSourceController = pSource->GetShadowController(); if( pSourceController == NULL ) { if( pDest->GetShadowController() != NULL ) { //we don't need a shadow controller anymore
pDest->RemoveShadowController(); } } else { IPhysicsShadowController *pDestController = pDest->GetShadowController(); if( pDestController == NULL ) { //we need a shadow controller
float fMaxSpeed, fMaxAngularSpeed; pSourceController->GetMaxSpeed( &fMaxSpeed, &fMaxAngularSpeed );
pDest->SetShadow( fMaxSpeed, fMaxAngularSpeed, pSourceController->AllowsTranslation(), pSourceController->AllowsRotation() ); pDestController = pDest->GetShadowController(); pDestController->SetTeleportDistance( pSourceController->GetTeleportDistance() ); pDestController->SetPhysicallyControlled( pSourceController->IsPhysicallyControlled() ); }
//sync shadow controllers
float fTimeOffset; Vector ptTargetPosition; QAngle qTargetAngles; fTimeOffset = pSourceController->GetTargetPosition( &ptTargetPosition, &qTargetAngles );
if( pTransform ) { ptTargetPosition = (*pTransform) * ptTargetPosition; qTargetAngles = TransformAnglesToWorldSpace( qTargetAngles, pTransform->As3x4() ); }
pDestController->Update( ptTargetPosition, qTargetAngles, fTimeOffset ); }
}
//pDest->RecheckContactPoints();
}
static void PartialSyncPhysicsObject( IPhysicsObject *pSource, IPhysicsObject *pDest, const VMatrix *pTransform ) { Vector ptOrigin, vVelocity, vAngularVelocity, vInertia; QAngle qAngles;
pSource->GetPosition( &ptOrigin, &qAngles ); pSource->GetVelocity( &vVelocity, &vAngularVelocity ); vInertia = pSource->GetInertia();
if( pTransform ) { #if 0
//pDest->SetPositionMatrix( matTransform.As3x4(), true ); //works like we think?
#else
ptOrigin = (*pTransform) * ptOrigin; qAngles = TransformAnglesToWorldSpace( qAngles, pTransform->As3x4() ); vVelocity = pTransform->ApplyRotation( vVelocity ); vAngularVelocity = pTransform->ApplyRotation( vAngularVelocity ); #endif
}
//avoid oversetting variables (I think that even setting them to the same value they already are disrupts the delicate physics balance)
if( vInertia != pDest->GetInertia() ) pDest->SetInertia( vInertia );
Vector ptDestOrigin, vDestVelocity, vDestAngularVelocity; QAngle qDestAngles; pDest->GetPosition( &ptDestOrigin, &qDestAngles ); pDest->GetVelocity( &vDestVelocity, &vDestAngularVelocity );
if( (ptOrigin != ptDestOrigin) || (qAngles != qDestAngles) ) pDest->SetPosition( ptOrigin, qAngles, false );
if( (vVelocity != vDestVelocity) || (vAngularVelocity != vDestAngularVelocity) ) pDest->SetVelocity( &vVelocity, &vAngularVelocity );
pDest->EnableCollisions( pSource->IsCollisionEnabled() ); }
void CPhysicsShadowClone::FullSyncClonedPhysicsObjects( bool bTeleport ) { CBaseEntity *pClonedEntity = m_hClonedEntity.Get(); if( pClonedEntity == NULL ) { VPhysicsDestroyObject(); return; }
VMatrix *pTransform; if( m_bShadowTransformIsIdentity ) pTransform = NULL; else pTransform = &m_matrixShadowTransform;
IPhysicsObject *(pSourceObjects[1024]); int iObjectCount = pClonedEntity->VPhysicsGetObjectList( pSourceObjects, 1024 );
//easy out if nothing has changed
if( iObjectCount == m_CloneLinks.Count() ) { int i; for( i = 0; i != iObjectCount; ++i ) { if( pSourceObjects[i] == NULL ) break;
if( pSourceObjects[i] != m_CloneLinks[i].pSource ) break; }
if( i == iObjectCount ) //no changes
{ for( i = 0; i != iObjectCount; ++i ) FullSyncPhysicsObject( m_CloneLinks[i].pSource, m_CloneLinks[i].pClone, pTransform, bTeleport );
return; } }
//copy the existing list of clone links to a temp array, we're going to be starting from scratch and copying links as we need them
PhysicsObjectCloneLink_t *pExistingLinks = NULL; int iExistingLinkCount = m_CloneLinks.Count(); if( iExistingLinkCount != 0 ) { pExistingLinks = (PhysicsObjectCloneLink_t *)stackalloc( sizeof(PhysicsObjectCloneLink_t) * m_CloneLinks.Count() ); memcpy( pExistingLinks, m_CloneLinks.Base(), sizeof(PhysicsObjectCloneLink_t) * m_CloneLinks.Count() ); } m_CloneLinks.RemoveAll();
//now, go over the object list we just got from the source entity, and either copy or create links as necessary
int i; for( i = 0; i != iObjectCount; ++i ) { IPhysicsObject *pSource = pSourceObjects[i];
if( pSource == NULL ) //this really shouldn't happen, but it does >_<
continue;
PhysicsObjectCloneLink_t cloneLink;
int j; for( j = 0; j != iExistingLinkCount; ++j ) { if( pExistingLinks[j].pSource == pSource ) break; }
if( j != iExistingLinkCount ) { //copyable link found
cloneLink = pExistingLinks[j]; memset( &pExistingLinks[j], 0, sizeof( PhysicsObjectCloneLink_t ) ); //zero out this slot so we don't destroy it in cleanup
} else { //no link found to copy, create a new one
cloneLink.pSource = pSource;
//apparently some collision code gets called on creation before we've set extra game flags, so we're going to cheat a bit and temporarily set our extra flags on the source
unsigned int iOldGameFlags = pSource->GetGameFlags(); pSource->SetGameFlags( iOldGameFlags | FVPHYSICS_IS_SHADOWCLONE );
unsigned int size = physenv->GetObjectSerializeSize(pSource); byte *pBuffer = (byte *)stackalloc(size); memset( pBuffer, 0, size );
physenv->SerializeObjectToBuffer( pSource, pBuffer, size ); //this should work across physics environments because the serializer doesn't write anything about itself to the template
pSource->SetGameFlags( iOldGameFlags ); cloneLink.pClone = m_pOwnerPhysEnvironment->UnserializeObjectFromBuffer( this, pBuffer, size, false ); //unserializer has to be in the target environment
assert( cloneLink.pClone ); //there should be absolutely no case where we can't clone a valid existing physics object
stackfree(pBuffer); }
FullSyncPhysicsObject( cloneLink.pSource, cloneLink.pClone, pTransform, bTeleport );
//cloneLink.pClone->Wake();
m_CloneLinks.AddToTail( cloneLink ); }
//now go over the existing links, if any of them haven't been nullified, they need to be deleted
for( i = 0; i != iExistingLinkCount; ++i ) { if( pExistingLinks[i].pClone ) m_pOwnerPhysEnvironment->DestroyObject( pExistingLinks[i].pClone ); //also destroys shadow controller
}
VPhysicsSetObject( NULL );
IPhysicsObject *pSource = m_hClonedEntity->VPhysicsGetObject();
for( i = m_CloneLinks.Count(); --i >= 0; ) { if( m_CloneLinks[i].pSource == pSource ) { //m_CloneLinks[i].pClone->Wake();
VPhysicsSetObject( m_CloneLinks[i].pClone ); break; } }
if( (i < 0) && (m_CloneLinks.Count() != 0) ) { VPhysicsSetObject( m_CloneLinks[0].pClone ); }
stackfree( pExistingLinks );
//CollisionRulesChanged();
}
void CPhysicsShadowClone::PartialSync( bool bPullChanges ) { VMatrix *pTransform; if( bPullChanges ) { if( m_bShadowTransformIsIdentity ) pTransform = NULL; else pTransform = &m_matrixShadowTransform;
for( int i = m_CloneLinks.Count(); --i >= 0; ) PartialSyncPhysicsObject( m_CloneLinks[i].pSource, m_CloneLinks[i].pClone, pTransform ); } else { if( m_bShadowTransformIsIdentity ) pTransform = NULL; else pTransform = &m_matrixShadowTransform_Inverse;
for( int i = m_CloneLinks.Count(); --i >= 0; ) PartialSyncPhysicsObject( m_CloneLinks[i].pClone, m_CloneLinks[i].pSource, pTransform ); }
SyncEntity( bPullChanges ); }
int CPhysicsShadowClone::VPhysicsGetObjectList( IPhysicsObject **pList, int listMax ) { int iCountStop = m_CloneLinks.Count(); if( iCountStop > listMax ) iCountStop = listMax;
for( int i = 0; i != iCountStop; ++i, ++pList ) *pList = m_CloneLinks[i].pClone;
return iCountStop; }
void CPhysicsShadowClone::VPhysicsDestroyObject( void ) { VPhysicsSetObject( NULL ); for( int i = m_CloneLinks.Count(); --i >= 0; ) { Assert( m_CloneLinks[i].pClone != NULL ); m_pOwnerPhysEnvironment->DestroyObject( m_CloneLinks[i].pClone ); } m_CloneLinks.RemoveAll();
SetMoveType( MOVETYPE_NONE ); SetSolid( SOLID_NONE ); SetSolidFlags( 0 ); SetCollisionGroup( COLLISION_GROUP_NONE );
BaseClass::VPhysicsDestroyObject(); }
bool CPhysicsShadowClone::ShouldCollide( int collisionGroup, int contentsMask ) const { CBaseEntity *pClonedEntity = m_hClonedEntity.Get();
if( pClonedEntity ) return pClonedEntity->ShouldCollide( collisionGroup, contentsMask ); else return false; }
bool CPhysicsShadowClone::TestCollision( const Ray_t &ray, unsigned int fContentsMask, trace_t& trace ) { return false;
/*CBaseEntity *pSourceEntity = m_hClonedEntity.Get();
if( pSourceEntity == NULL ) return false;
enginetrace->ClipRayToEntity( ray, fContentsMask, pSourceEntity, &trace ); return trace.DidHit();*/ }
int CPhysicsShadowClone::ObjectCaps( void ) { return ((BaseClass::ObjectCaps() | FCAP_DONT_SAVE) & ~(FCAP_FORCE_TRANSITION | FCAP_ACROSS_TRANSITION | FCAP_MUST_SPAWN | FCAP_SAVE_NON_NETWORKABLE)); }
void CPhysicsShadowClone::SetCloneTransformationMatrix( const matrix3x4_t &sourceMatrix ) { m_matrixShadowTransform = sourceMatrix; m_bShadowTransformIsIdentity = m_matrixShadowTransform.IsIdentity();
if( m_matrixShadowTransform.InverseGeneral( m_matrixShadowTransform_Inverse ) == false ) { m_matrixShadowTransform.InverseTR( m_matrixShadowTransform_Inverse ); //probably not the right matrix, but we're out of options
}
FullSync(); //PartialSync( true );
}
void CPhysicsShadowClone::SetClonedEntity( EHANDLE hEntToClone ) { VPhysicsDestroyObject(); m_hClonedEntity = hEntToClone;
//FullSyncClonedPhysicsObjects();
}
EHANDLE CPhysicsShadowClone::GetClonedEntity( void ) { return m_hClonedEntity; }
//damage relays to source entity
bool CPhysicsShadowClone::PassesDamageFilter( const CTakeDamageInfo &info ) { CBaseEntity *pClonedEntity = m_hClonedEntity.Get();
if( pClonedEntity ) return pClonedEntity->PassesDamageFilter( info ); else return BaseClass::PassesDamageFilter( info ); }
bool CPhysicsShadowClone::CanBeHitByMeleeAttack( CBaseEntity *pAttacker ) { CBaseEntity *pClonedEntity = m_hClonedEntity.Get();
if( pClonedEntity ) return pClonedEntity->CanBeHitByMeleeAttack( pAttacker ); else return BaseClass::CanBeHitByMeleeAttack( pAttacker ); }
int CPhysicsShadowClone::OnTakeDamage( const CTakeDamageInfo &info ) { CBaseEntity *pClonedEntity = m_hClonedEntity.Get();
if( pClonedEntity ) return pClonedEntity->OnTakeDamage( info ); else return BaseClass::OnTakeDamage( info ); }
int CPhysicsShadowClone::TakeHealth( float flHealth, int bitsDamageType ) { CBaseEntity *pClonedEntity = m_hClonedEntity.Get();
if( pClonedEntity ) return pClonedEntity->TakeHealth( flHealth, bitsDamageType ); else return BaseClass::TakeHealth( flHealth, bitsDamageType ); }
void CPhysicsShadowClone::Event_Killed( const CTakeDamageInfo &info ) { CBaseEntity *pClonedEntity = m_hClonedEntity.Get();
if( pClonedEntity ) pClonedEntity->Event_Killed( info ); else BaseClass::Event_Killed( info ); }
CPhysicsShadowClone *CPhysicsShadowClone::CreateShadowClone( IPhysicsEnvironment *pInPhysicsEnvironment, EHANDLE hEntToClone, const char *szDebugMarker, const matrix3x4_t *pTransformationMatrix /*= NULL*/ ) { AssertMsg( szDebugMarker != NULL, "All shadow clones must have a debug marker for where it came from in debug builds." );
if( !sv_use_shadow_clones.GetBool() ) return NULL;
CBaseEntity *pClonedEntity = hEntToClone.Get(); if( pClonedEntity == NULL ) return NULL;
AssertMsg( IsShadowClone( pClonedEntity ) == false, "Shouldn't attempt to clone clones" );
if( pClonedEntity->IsMarkedForDeletion() ) return NULL;
//if( pClonedEntity->IsPlayer() )
// return NULL;
IPhysicsObject *pPhysics = pClonedEntity->VPhysicsGetObject();
if( pPhysics == NULL ) return NULL;
if( pPhysics->IsStatic() ) return NULL;
if( pClonedEntity->GetSolid() == SOLID_BSP ) return NULL;
if( pClonedEntity->GetSolidFlags() & (FSOLID_NOT_SOLID | FSOLID_TRIGGER) ) return NULL;
if( pClonedEntity->GetFlags() & (FL_WORLDBRUSH | FL_STATICPROP) ) return NULL;
/*if( FClassnameIs( pClonedEntity, "func_door" ) )
{ //only clone func_door's that are in front of the portal
return NULL; }*/
// Too many shadow clones breaks the game (too many entities)
if( g_iShadowCloneCount >= MAX_SHADOW_CLONE_COUNT ) { AssertMsg( false, "Too many shadow clones, consider upping the limit or reducing the level's physics props" ); return NULL; } ++g_iShadowCloneCount;
CPhysicsShadowClone *pClone = (CPhysicsShadowClone*)CreateEntityByName("physicsshadowclone"); s_IsShadowClone[pClone->entindex()] = true; pClone->m_pOwnerPhysEnvironment = pInPhysicsEnvironment; pClone->m_hClonedEntity = hEntToClone; DBG_CODE_NOSCOPE( pClone->m_szDebugMarker = szDebugMarker; );
CPhysicsShadowCloneLL *pCloneLLEntry = s_SCLLManager.Alloc(); pCloneLLEntry->pClone = pClone; pCloneLLEntry->pNext = s_EntityClones[pClonedEntity->entindex()]; s_EntityClones[pClonedEntity->entindex()] = pCloneLLEntry;
if( pTransformationMatrix ) { pClone->m_matrixShadowTransform = *pTransformationMatrix; pClone->m_bShadowTransformIsIdentity = pClone->m_matrixShadowTransform.IsIdentity();
if( !pClone->m_bShadowTransformIsIdentity ) { if( pClone->m_matrixShadowTransform.InverseGeneral( pClone->m_matrixShadowTransform_Inverse ) == false ) { pClone->m_matrixShadowTransform.InverseTR( pClone->m_matrixShadowTransform_Inverse ); //probably not the right matrix, but we're out of options
} } }
DispatchSpawn( pClone );
return pClone; }
void CPhysicsShadowClone::Free( void ) { VPhysicsDestroyObject();
UTIL_Remove( this );
//Too many shadow clones breaks the game (too many entities)
--g_iShadowCloneCount; }
void CPhysicsShadowClone::FullSyncAllClones( void ) { for( int i = s_ActiveShadowClones.Count(); --i >= 0; ) { s_ActiveShadowClones[i]->FullSync( true ); } }
IPhysicsObject *CPhysicsShadowClone::TranslatePhysicsToClonedEnt( const IPhysicsObject *pPhysics ) { if( m_hClonedEntity.Get() != NULL ) { for( int i = m_CloneLinks.Count(); --i >= 0; ) { if( m_CloneLinks[i].pClone == pPhysics ) return m_CloneLinks[i].pSource; } }
return NULL; }
void CPhysicsShadowClone::VPhysicsCollision( int index, gamevcollisionevent_t *pEvent ) { //the baseclass just screenshakes, makes sounds, and outputs dust, we rely on the original entity to do this when applicable
}
bool CPhysicsShadowClone::IsShadowClone( const CBaseEntity *pEntity ) { return s_IsShadowClone[pEntity->entindex()]; }
CPhysicsShadowCloneLL *CPhysicsShadowClone::GetClonesOfEntity( const CBaseEntity *pEntity ) { return s_EntityClones[pEntity->entindex()]; }
static void DrawDebugOverlayForShadowClone( CPhysicsShadowClone *pClone ) { unsigned char iColorIntensity = (pClone->IsInAssumedSyncState())?(127):(255);
int iRed = (pClone->IsUntransformedClone())?(0):(iColorIntensity); int iGreen = iColorIntensity; int iBlue = iColorIntensity;
NDebugOverlay::EntityBounds( pClone, iRed, iGreen, iBlue, (iColorIntensity>>2), 0.05f ); }
bool CTraceFilterTranslateClones::ShouldHitEntity( IHandleEntity *pEntity, int contentsMask ) { CBaseEntity *pEnt = EntityFromEntityHandle( pEntity ); if( CPhysicsShadowClone::IsShadowClone( pEnt ) ) { CBaseEntity *pClonedEntity = ((CPhysicsShadowClone *)pEnt)->GetClonedEntity(); CPortalSimulator *pSimulator = CPortalSimulator::GetSimulatorThatOwnsEntity( pClonedEntity ); if( pSimulator->m_DataAccess.Simulation.Dynamic.EntFlags[pClonedEntity->entindex()] & PSEF_IS_IN_PORTAL_HOLE ) return m_pActualFilter->ShouldHitEntity( pClonedEntity, contentsMask ); else return false; } else { return m_pActualFilter->ShouldHitEntity( pEntity, contentsMask ); } }
TraceType_t CTraceFilterTranslateClones::GetTraceType() const { return m_pActualFilter->GetTraceType(); }
|