css_enhanced_waf/engine/world.cpp
2024-08-23 20:55:44 +02:00

566 lines
16 KiB
C++

//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $Workfile: $
// $Date: $
// $NoKeywords: $
//===========================================================================//
#include "quakedef.h"
#include "world.h"
#include "eiface.h"
#include "server.h"
#include "cmodel_engine.h"
#include "gl_model_private.h"
#include "sv_main.h"
#include "vengineserver_impl.h"
#include "collisionutils.h"
#include "vphysics_interface.h"
#include "ispatialpartitioninternal.h"
#include "staticpropmgr.h"
#include "shadowmgr.h"
#include "string_t.h"
#include "enginetrace.h"
#include "sys_dll.h"
#include "cdll_engine_int.h"
#include "icliententitylist.h"
#include "client_class.h"
#include "icliententity.h"
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
//============================================================================
/*
===============
SV_ClearWorld
===============
*/
void SV_ClearWorld (void)
{
MDLCACHE_COARSE_LOCK_(g_pMDLCache);
// Clean up static props from the previous level
#if !defined( SWDS )
g_pShadowMgr->LevelShutdown();
#endif // SWDS
StaticPropMgr()->LevelShutdown();
for ( int i = 0; i < 3; i++ )
{
if ( host_state.worldmodel->mins[i] < MIN_COORD_INTEGER || host_state.worldmodel->maxs[i] > MAX_COORD_INTEGER )
{
Host_EndGame(true, "Map coordinate extents are too large!!\nCheck for errors!\n" );
}
}
SpatialPartition()->Init( host_state.worldmodel->mins, host_state.worldmodel->maxs );
// Load all static props into the spatial partition
StaticPropMgr()->LevelInit();
#if !defined( SWDS )
g_pShadowMgr->LevelInit( host_state.worldbrush->numsurfaces );
#endif
}
//-----------------------------------------------------------------------------
// Trigger world-space bounds
//-----------------------------------------------------------------------------
static void CM_TriggerWorldSpaceBounds( ICollideable *pCollideable, Vector *pMins, Vector *pMaxs )
{
if ( pCollideable->GetSolidFlags() & FSOLID_USE_TRIGGER_BOUNDS )
{
pCollideable->WorldSpaceTriggerBounds( pMins, pMaxs );
}
else
{
CM_WorldSpaceBounds( pCollideable, pMins, pMaxs );
}
}
static void CM_GetCollideableTriggerTestBox( ICollideable *pCollide, Vector *pMins, Vector *pMaxs, bool bUseAccurateBbox )
{
if ( bUseAccurateBbox && pCollide->GetSolid() == SOLID_BBOX )
{
*pMins = pCollide->OBBMins();
*pMaxs = pCollide->OBBMaxs();
}
else
{
const Vector &vecStart = pCollide->GetCollisionOrigin();
pCollide->WorldSpaceSurroundingBounds( pMins, pMaxs );
*pMins -= vecStart;
*pMaxs -= vecStart;
}
}
//-----------------------------------------------------------------------------
// Little enumeration class used to try touching all triggers
//-----------------------------------------------------------------------------
class CTouchLinks : public IPartitionEnumerator
{
public:
CTouchLinks( edict_t* pEnt, const Vector* pPrevAbsOrigin, bool accurateBboxTriggerChecks ) : m_TouchedEntities( 8, 8 )
{
m_pEnt = pEnt;
m_pCollide = pEnt->GetCollideable();
Assert( m_pCollide );
Vector vecMins, vecMaxs;
CM_GetCollideableTriggerTestBox( m_pCollide, &vecMins, &vecMaxs, accurateBboxTriggerChecks );
const Vector &vecStart = m_pCollide->GetCollisionOrigin();
if (pPrevAbsOrigin)
{
m_Ray.Init( *pPrevAbsOrigin, vecStart, vecMins, vecMaxs );
}
else
{
m_Ray.Init( vecStart, vecStart, vecMins, vecMaxs );
}
}
IterationRetval_t EnumElement( IHandleEntity *pHandleEntity )
{
// Static props should never be in the trigger list
Assert( !StaticPropMgr()->IsStaticProp( pHandleEntity ) );
IServerUnknown *pUnk = static_cast<IServerUnknown*>( pHandleEntity );
Assert( pUnk );
// Convert the IHandleEntity to an edict_t*...
// Context is the thing we're testing everything against
edict_t* pTouch = pUnk->GetNetworkable()->GetEdict();
// Can't bump against itself
if ( pTouch == m_pEnt )
return ITERATION_CONTINUE;
IServerEntity *pTriggerEntity = pTouch->GetIServerEntity();
if ( !pTriggerEntity )
return ITERATION_CONTINUE;
// Hmmm.. everything in this list should be a trigger....
ICollideable *pTriggerCollideable = pTriggerEntity->GetCollideable();
if ( !m_pCollide->ShouldTouchTrigger(pTriggerCollideable->GetSolidFlags()) )
return ITERATION_CONTINUE;
Assert(pTriggerCollideable->GetSolidFlags() & FSOLID_TRIGGER );
if ( pTriggerCollideable->GetSolidFlags() & FSOLID_USE_TRIGGER_BOUNDS )
{
Vector vecTriggerMins, vecTriggerMaxs;
pTriggerCollideable->WorldSpaceTriggerBounds( &vecTriggerMins, &vecTriggerMaxs );
if ( !IsBoxIntersectingRay( vecTriggerMins, vecTriggerMaxs, m_Ray ) )
{
return ITERATION_CONTINUE;
}
}
else
{
trace_t tr;
g_pEngineTraceServer->ClipRayToCollideable( m_Ray, MASK_SOLID, pTriggerCollideable, &tr );
if ( !(tr.contents & MASK_SOLID) )
return ITERATION_CONTINUE;
}
m_TouchedEntities.AddToTail( pTouch );
return ITERATION_CONTINUE;
}
void HandleTouchedEntities( )
{
for ( int i = 0; i < m_TouchedEntities.Count(); ++i )
{
serverGameEnts->MarkEntitiesAsTouching( m_TouchedEntities[i], m_pEnt );
}
}
Ray_t m_Ray;
private:
edict_t *m_pEnt;
ICollideable *m_pCollide;
CUtlVector< edict_t* > m_TouchedEntities;
};
// enumerator class that's used to update touch links for a trigger when
// it moves or changes solid type
class CTriggerMoved : public IPartitionEnumerator
{
public:
CTriggerMoved( bool accurateBboxTriggerChecks ) : m_TouchedEntities( 8, 8 )
{
m_bAccurateBBoxCheck = accurateBboxTriggerChecks;
}
void TriggerMoved( edict_t *pTriggerEntity )
{
m_pTriggerEntity = pTriggerEntity;
m_pTrigger = pTriggerEntity->GetCollideable();
m_triggerSolidFlags = m_pTrigger->GetSolidFlags();
Vector vecAbsMins, vecAbsMaxs;
CM_TriggerWorldSpaceBounds( m_pTrigger, &vecAbsMins, &vecAbsMaxs );
SpatialPartition()->EnumerateElementsInBox( PARTITION_ENGINE_SOLID_EDICTS,
vecAbsMins, vecAbsMaxs, false, this );
}
IterationRetval_t EnumElement( IHandleEntity *pHandleEntity )
{
// skip static props, the game DLL doesn't care about them
if ( StaticPropMgr()->IsStaticProp( pHandleEntity ) )
return ITERATION_CONTINUE;
IServerUnknown *pUnk = static_cast< IServerUnknown* >( pHandleEntity );
Assert( pUnk );
// Convert the user ID to and edict_t*...
edict_t* pTouch = pUnk->GetNetworkable()->GetEdict();
Assert( pTouch );
ICollideable *pTouchCollide = pUnk->GetCollideable();
// Can't ever touch itself because it's in the other list
if ( pTouchCollide == m_pTrigger )
return ITERATION_CONTINUE;
if ( !pTouchCollide->ShouldTouchTrigger(m_triggerSolidFlags) )
return ITERATION_CONTINUE;
IServerEntity *serverEntity = pTouch->GetIServerEntity();
if ( !serverEntity )
return ITERATION_CONTINUE;
// FIXME: Should we be using the surrounding bounds here?
Vector vecMins, vecMaxs;
CM_GetCollideableTriggerTestBox( pTouchCollide, &vecMins, &vecMaxs, m_bAccurateBBoxCheck );
const Vector &vecStart = pTouchCollide->GetCollisionOrigin();
Ray_t ray;
ray.Init( vecStart, vecStart, vecMins, vecMaxs );
if ( m_pTrigger->GetSolidFlags() & FSOLID_USE_TRIGGER_BOUNDS )
{
Vector vecTriggerMins, vecTriggerMaxs;
m_pTrigger->WorldSpaceTriggerBounds( &vecTriggerMins, &vecTriggerMaxs );
if ( !IsBoxIntersectingRay( vecTriggerMins, vecTriggerMaxs, ray ) )
{
return ITERATION_CONTINUE;
}
}
else
{
trace_t tr;
g_pEngineTraceServer->ClipRayToCollideable( ray, MASK_SOLID, m_pTrigger, &tr );
if ( !(tr.contents & MASK_SOLID) )
return ITERATION_CONTINUE;
}
m_TouchedEntities.AddToTail( pTouch );
return ITERATION_CONTINUE;
}
void HandleTouchedEntities( )
{
for ( int i = 0; i < m_TouchedEntities.Count(); ++i )
{
serverGameEnts->MarkEntitiesAsTouching( m_TouchedEntities[i], m_pTriggerEntity );
}
}
private:
edict_t* m_pTriggerEntity;
ICollideable* m_pTrigger;
int m_triggerSolidFlags;
Vector m_vecDelta;
CUtlVector< edict_t* > m_TouchedEntities;
bool m_bAccurateBBoxCheck;
};
//-----------------------------------------------------------------------------
// Touches triggers. Or, if it is a trigger, causes other things to touch it
// returns true if untouch needs to be checked
//-----------------------------------------------------------------------------
void SV_TriggerMoved( edict_t *pTriggerEnt, bool accurateBboxTriggerChecks )
{
CTriggerMoved triggerEnum( accurateBboxTriggerChecks );
triggerEnum.TriggerMoved( pTriggerEnt );
triggerEnum.HandleTouchedEntities( );
}
void SV_SolidMoved( edict_t *pSolidEnt, ICollideable *pSolidCollide, const Vector* pPrevAbsOrigin, bool accurateBboxTriggerChecks )
{
if (!pPrevAbsOrigin)
{
CTouchLinks touchEnumerator(pSolidEnt, NULL, accurateBboxTriggerChecks);
Vector vecWorldMins, vecWorldMaxs;
pSolidCollide->WorldSpaceSurroundingBounds( &vecWorldMins, &vecWorldMaxs );
SpatialPartition()->EnumerateElementsInBox( PARTITION_ENGINE_TRIGGER_EDICTS,
vecWorldMins, vecWorldMaxs, false, &touchEnumerator );
touchEnumerator.HandleTouchedEntities( );
}
else
{
CTouchLinks touchEnumerator(pSolidEnt, pPrevAbsOrigin, accurateBboxTriggerChecks);
// A version that checks against an extruded ray indicating the motion
SpatialPartition()->EnumerateElementsAlongRay( PARTITION_ENGINE_TRIGGER_EDICTS,
touchEnumerator.m_Ray, false, &touchEnumerator );
touchEnumerator.HandleTouchedEntities( );
}
}
CON_COMMAND(cl_numtriggers, "Number of triggers in map")
{
class CTestTriggerMask : public IPartitionEnumerator
{
public:
IterationRetval_t EnumElement( IHandleEntity *pHandleEntity )
{
IClientUnknown *pUnk = static_cast< IClientUnknown* >( pHandleEntity );
m_nCount++;
return ITERATION_CONTINUE;
}
int m_nCount = 0;
};
static CTestTriggerMask s_Enumerator;
s_Enumerator.m_nCount = 0;
SpatialPartition()->EnumerateElementsInBox( PARTITION_CLIENT_TRIGGER_ENTITIES,
Vector(vec_t( SHRT_MAX ) * -0.5), Vector(vec_t( SHRT_MAX ) * 0.5), false, &s_Enumerator );
ConMsg("Triggers in map: %i\n", s_Enumerator.m_nCount);
}
#ifndef DEDICATED
//-----------------------------------------------------------------------------
// Little enumeration class used to try touching all triggers
//-----------------------------------------------------------------------------
class CTouchLinks_ClientSide : public IPartitionEnumerator
{
public:
CTouchLinks_ClientSide( IClientEntity *pEnt, const Vector* pPrevAbsOrigin, bool accurateBboxTriggerChecks ) : m_TouchedEntities( 8, 8 )
{
m_pEnt = pEnt;
m_pCollide = pEnt->GetCollideable();
Assert( m_pCollide );
Vector vecMins, vecMaxs;
CM_GetCollideableTriggerTestBox( m_pCollide, &vecMins, &vecMaxs, accurateBboxTriggerChecks );
const Vector &vecStart = m_pCollide->GetCollisionOrigin();
if (pPrevAbsOrigin)
{
m_Ray.Init( *pPrevAbsOrigin, vecStart, vecMins, vecMaxs );
}
else
{
m_Ray.Init( vecStart, vecStart, vecMins, vecMaxs );
}
}
IterationRetval_t EnumElement( IHandleEntity *pHandleEntity )
{
// Static props should never be in the trigger list
Assert( !StaticPropMgr()->IsStaticProp( pHandleEntity ) );
IClientUnknown *pUnk = static_cast<IClientUnknown*>( pHandleEntity );
Assert( pUnk );
// Convert the IHandleEntity to an edict_t*...
// Context is the thing we're testing everything against
IClientEntity *pTriggerEntity = pUnk->GetIClientEntity();
// Can't bump against itself
if ( pTriggerEntity == m_pEnt )
return ITERATION_CONTINUE;
// Hmmm.. everything in this list should be a trigger....
ICollideable *pTriggerCollideable = pTriggerEntity->GetCollideable();
int nTriggerSolidFlags = pTriggerCollideable->GetSolidFlags();
if ( !m_pCollide->ShouldTouchTrigger(pTriggerCollideable->GetSolidFlags()) )
return ITERATION_CONTINUE;
if ( pTriggerCollideable->GetSolidFlags() & FSOLID_USE_TRIGGER_BOUNDS )
{
Vector vecTriggerMins, vecTriggerMaxs;
pTriggerCollideable->WorldSpaceTriggerBounds( &vecTriggerMins, &vecTriggerMaxs );
if ( !IsBoxIntersectingRay( vecTriggerMins, vecTriggerMaxs, m_Ray ) )
{
return ITERATION_CONTINUE;
}
}
else
{
trace_t tr;
g_pEngineTraceClient->ClipRayToCollideable( m_Ray, MASK_SOLID, pTriggerCollideable, &tr );
if ( !(tr.contents & MASK_SOLID) )
return ITERATION_CONTINUE;
}
m_TouchedEntities.AddToTail( pTriggerEntity );
return ITERATION_CONTINUE;
}
void HandleTouchedEntities()
{
for ( int i = 0; i < m_TouchedEntities.Count(); ++i )
{
g_ClientDLL->MarkEntitiesAsTouching( m_TouchedEntities[i], m_pEnt );
}
}
Ray_t m_Ray;
private:
IClientEntity *m_pEnt;
ICollideable *m_pCollide;
CUtlVector< IClientEntity* > m_TouchedEntities;
};
// enumerator class that's used to update touch links for a trigger when
// it moves or changes solid type
class CTriggerMoved_ClientSide : public IPartitionEnumerator
{
public:
CTriggerMoved_ClientSide( bool accurateBboxTriggerChecks ) : m_TouchedEntities( 8, 8 )
{
m_bAccurateBBoxCheck = accurateBboxTriggerChecks;
}
void TriggerMoved( IClientEntity *pTriggerEntity )
{
m_pTriggerEntity = pTriggerEntity;
m_pTrigger = pTriggerEntity->GetCollideable();
m_triggerSolidFlags = m_pTrigger->GetSolidFlags();
Vector vecAbsMins, vecAbsMaxs;
CM_TriggerWorldSpaceBounds( m_pTrigger, &vecAbsMins, &vecAbsMaxs );
SpatialPartition()->EnumerateElementsInBox( PARTITION_CLIENT_SOLID_EDICTS,
vecAbsMins, vecAbsMaxs, false, this );
}
IterationRetval_t EnumElement( IHandleEntity *pHandleEntity )
{
// skip static props, the game DLL doesn't care about them
if ( StaticPropMgr()->IsStaticProp( pHandleEntity ) )
return ITERATION_CONTINUE;
IClientUnknown *pUnk = static_cast< IClientUnknown* >( pHandleEntity );
Assert( pUnk );
// Convert the user ID to and edict_t*...
IClientEntity* pTouch = pUnk->GetIClientEntity();
Assert( pTouch );
ICollideable *pTouchCollide = pUnk->GetCollideable();
// Can't ever touch itself because it's in the other list
if ( pTouchCollide == m_pTrigger )
return ITERATION_CONTINUE;
if ( !pTouchCollide->ShouldTouchTrigger(m_triggerSolidFlags) )
return ITERATION_CONTINUE;
// FIXME: Should we be using the surrounding bounds here?
Vector vecMins, vecMaxs;
CM_GetCollideableTriggerTestBox( pTouchCollide, &vecMins, &vecMaxs, m_bAccurateBBoxCheck );
const Vector &vecStart = pTouchCollide->GetCollisionOrigin();
Ray_t ray;
ray.Init( vecStart, vecStart, vecMins, vecMaxs );
if ( m_pTrigger->GetSolidFlags() & FSOLID_USE_TRIGGER_BOUNDS )
{
Vector vecTriggerMins, vecTriggerMaxs;
m_pTrigger->WorldSpaceTriggerBounds( &vecTriggerMins, &vecTriggerMaxs );
if ( !IsBoxIntersectingRay( vecTriggerMins, vecTriggerMaxs, ray ) )
{
return ITERATION_CONTINUE;
}
}
else
{
trace_t tr;
g_pEngineTraceClient->ClipRayToCollideable( ray, MASK_SOLID, m_pTrigger, &tr );
if ( !(tr.contents & MASK_SOLID) )
return ITERATION_CONTINUE;
}
m_TouchedEntities.AddToTail( pTouch );
return ITERATION_CONTINUE;
}
void HandleTouchedEntities( )
{
for ( int i = 0; i < m_TouchedEntities.Count(); ++i )
{
g_ClientDLL->MarkEntitiesAsTouching( m_TouchedEntities[i], m_pTriggerEntity );
}
}
private:
IClientEntity* m_pTriggerEntity;
ICollideable* m_pTrigger;
int m_triggerSolidFlags;
Vector m_vecDelta;
CUtlVector< IClientEntity* > m_TouchedEntities;
bool m_bAccurateBBoxCheck;
};
#include "debugoverlay.h"
// XYZ_TODO: Remove these on listen servers
void CL_TriggerMoved( IClientEntity *pTriggerEnt, bool accurateBboxTriggerChecks )
{
CTriggerMoved_ClientSide triggerEnum( accurateBboxTriggerChecks );
triggerEnum.TriggerMoved( pTriggerEnt );
triggerEnum.HandleTouchedEntities();
}
void CL_SolidMoved( IClientEntity *pTriggerEnt, ICollideable *pSolidCollide, const Vector* pPrevAbsOrigin, bool accurateBboxTriggerChecks )
{
if (!pPrevAbsOrigin)
{
CTouchLinks_ClientSide touchEnumerator(pTriggerEnt, NULL, accurateBboxTriggerChecks);
Vector vecWorldMins, vecWorldMaxs;
pSolidCollide->WorldSpaceSurroundingBounds( &vecWorldMins, &vecWorldMaxs );
SpatialPartition()->EnumerateElementsInBox( PARTITION_CLIENT_TRIGGER_ENTITIES,
vecWorldMins, vecWorldMaxs, false, &touchEnumerator );
touchEnumerator.HandleTouchedEntities();
}
else
{
CTouchLinks_ClientSide touchEnumerator(pTriggerEnt, pPrevAbsOrigin, accurateBboxTriggerChecks);
// A version that checks against an extruded ray indicating the motion
SpatialPartition()->EnumerateElementsAlongRay( PARTITION_CLIENT_TRIGGER_ENTITIES,
touchEnumerator.m_Ray, false, &touchEnumerator );
touchEnumerator.HandleTouchedEntities();
}
}
#endif