source-engine/game/client/particle_collision.cpp
2023-10-03 17:23:56 +03:00

377 lines
12 KiB
C++

//========= Copyright © 1996-2005, Valve Corporation, All rights reserved. ============//
//
// Purpose: Local fast collision system for particles
//
// $NoKeywords: $
//=============================================================================//
#include "cbase.h"
#include "particle_collision.h"
#include "engine/IVDebugOverlay.h"
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
#ifdef _XBOX
#define __DEBUG_PARTICLE_COLLISION_RETEST 0
#else
#define __DEBUG_PARTICLE_COLLISION_RETEST 1
#endif // _XBOX
#define __DEBUG_PARTICLE_COLLISION_OVERLAY 0
#define __DEBUG_PARTICLE_COLLISION_OVERLAY_LIFETIME 0.1f
#define NUM_DISCREET_STEPS 8.0f
#define NUM_SIMULATION_SECONDS 2.0f
#define COLLISION_EPSILON 0.01f
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
CBaseSimpleCollision::CBaseSimpleCollision( void )
{
ClearActivePlanes();
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : &origin -
// radius -
//-----------------------------------------------------------------------------
void CBaseSimpleCollision::Setup( const Vector &origin, float speed, float gravity )
{
TestForPlane( origin, Vector( 1, 0, 0 ), speed, gravity );
TestForPlane( origin, Vector( -1, 0, 0 ), speed, gravity );
TestForPlane( origin, Vector( 0, 1, 0 ), speed, gravity );
TestForPlane( origin, Vector( 0, -1, 0 ), speed, gravity );
TestForPlane( origin, Vector( 0, 0, 1 ), speed, gravity );
TestForPlane( origin, Vector( 0, 0, -1 ), speed, gravity );
}
//-----------------------------------------------------------------------------
// Purpose: Trace line for super-simplified traces
// Input : &start - start position
// &end - end position
// *pTrace - trace structure to fill
// coarse - tests again with a real trace unless coarse is set
//-----------------------------------------------------------------------------
void CBaseSimpleCollision::TraceLine( const Vector &start, const Vector &end, trace_t *pTrace, bool coarse )
{
//Iterate over all active planes
for ( int i = 0; i < m_nActivePlanes; i++ )
{
//Must be a valid plane
if ( m_collisionPlanes[i].m_Dist == -1.0f )
continue;
//Get our information about the relation to this plane
float dot1 = m_collisionPlanes[i].DistTo(start);
float dot2 = m_collisionPlanes[i].DistTo(end);
//Don't consider particles on the backside of planes
if ( dot1 < -COLLISION_EPSILON )
continue;
//Must be crossing the plane's boundary
if ( ( dot1 > COLLISION_EPSILON ) == ( dot2 > COLLISION_EPSILON ) )
continue;
//Find the intersection point
float t = dot1 / (dot1 - dot2);
Vector vIntersection = start + (end - start) * t;
//Fake the collision info
pTrace->endpos = vIntersection;
pTrace->fraction = t - COLLISION_EPSILON;
pTrace->plane.normal = m_collisionPlanes[i].m_Normal;
pTrace->plane.dist = m_collisionPlanes[i].m_Dist;
//If we need an exact trace, test again on a successful hit
if ( ( coarse == false ) && ( pTrace->fraction < 1.0f ) )
{
UTIL_TraceLine( start, end, MASK_SOLID_BRUSHONLY, NULL, COLLISION_GROUP_NONE, pTrace );
}
#if __DEBUG_PARTICLE_COLLISION_OVERLAY
debugoverlay->AddBoxOverlay( vIntersection, Vector(-1,-1,-1), Vector(1,1,1), QAngle(0,0,0), 0, 255, 0, 16, __DEBUG_PARTICLE_COLLISION_OVERLAY_LIFETIME );
#endif //__DEBUG_PARTICLE_COLLISION_OVERLAY
//Done
return;
}
//Fell through, so clear all the fields
pTrace->plane.normal[0] = 0.0f;
pTrace->plane.normal[1] = 0.0f;
pTrace->plane.normal[2] = 0.0f;
pTrace->plane.dist = 0.0f;
pTrace->fraction = 1.0f;
pTrace->allsolid = false;
pTrace->startsolid = false;
pTrace->m_pEnt = NULL;
}
//-----------------------------------------------------------------------------
// Purpose: Tests the planes against all others for validity
// Input : *plane - plane to test
//-----------------------------------------------------------------------------
void CBaseSimpleCollision::ConsiderPlane( cplane_t *plane )
{
//Test against all other active planes
for ( int i = 0; i < m_nActivePlanes; i++ )
{
if ( m_collisionPlanes[i].m_Dist != -1.0f )
{
//Test for coplanar
if ( ( m_collisionPlanes[i].m_Normal == plane->normal ) && ( m_collisionPlanes[i].m_Dist == plane->dist ) )
return;
}
}
//Don't overrun
if ( m_nActivePlanes >= MAX_COLLISION_PLANES )
return;
//Take it
m_collisionPlanes[m_nActivePlanes].m_Dist = plane->dist;
m_collisionPlanes[m_nActivePlanes].m_Normal = plane->normal;
m_nActivePlanes++;
}
//-----------------------------------------------------------------------------
// Purpose: Runs a simulation of the average particle's movement, looking for collisions along the way
// Input : &start - start of the simulation
// &dir - direction of travel
// speed - speed of the particle
// gravity - gravity being used
//-----------------------------------------------------------------------------
void CBaseSimpleCollision::TestForPlane( const Vector &start, const Vector &dir, float speed, float gravity )
{
trace_t tr;
Vector testStart, testEnd;
testStart = start;
//Setup our step increments
float dStepTime = (NUM_SIMULATION_SECONDS/NUM_DISCREET_STEPS);
Vector vStepIncr = dir * ( speed * dStepTime );
float flGravIncr = gravity*dStepTime;
//Simulate collsions in discreet steps
for ( int i = 1; i <= NUM_DISCREET_STEPS; i++ )
{
testEnd = testStart + vStepIncr;
testEnd[2] -= flGravIncr * (0.5f*(dStepTime*i)*(dStepTime*i) );
//Trace the line
UTIL_TraceLine( testStart, testEnd, MASK_SOLID_BRUSHONLY, NULL, COLLISION_GROUP_NONE, &tr );
//See if we found one
if ( tr.fraction != 1.0f )
{
#if __DEBUG_PARTICLE_COLLISION_OVERLAY
debugoverlay->AddLineOverlay( testStart, tr.endpos, 255, 0, 0, true, __DEBUG_PARTICLE_COLLISION_OVERLAY_LIFETIME );
QAngle angles;
VectorAngles( tr.plane.normal,angles );
angles[PITCH] += 90;
debugoverlay->AddBoxOverlay( tr.endpos, Vector(-64,-64,0), Vector(64,64,0), angles, 255, 0, 0, 16, __DEBUG_PARTICLE_COLLISION_OVERLAY_LIFETIME );
#endif //__DEBUG_PARTICLE_COLLISION_OVERLAY
//Test the plane against a set of criteria
ConsiderPlane( &tr.plane );
return;
}
//We missed
#if __DEBUG_PARTICLE_COLLISION_OVERLAY
debugoverlay->AddLineOverlay( testStart, tr.endpos, 0, 128.0f+(128.0f*((float)i/(float)NUM_DISCREET_STEPS)), 0, true, __DEBUG_PARTICLE_COLLISION_OVERLAY_LIFETIME );
#endif //__DEBUG_PARTICLE_COLLISION_OVERLAY
//Save that position for the next round
testStart = testEnd;
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CBaseSimpleCollision::ClearActivePlanes( void )
{
for ( int i = 0; i < MAX_COLLISION_PLANES; i++ )
{
m_collisionPlanes[i].m_Dist = -1.0f;
m_collisionPlanes[i].m_Normal.Init();
}
m_nActivePlanes = 0;
}
//
// CParticleCollision
//
//-----------------------------------------------------------------------------
// Constructor
//-----------------------------------------------------------------------------
CParticleCollision::CParticleCollision( void )
{
m_flGravity = 800.0f;
m_flCollisionDampen = 0.5f;
m_flAngularCollisionDampen = 0.25f;
ClearActivePlanes();
}
//-----------------------------------------------------------------------------
// Purpose: Test for surrounding collision surfaces for quick collision testing for the particle system
// Input : &origin - starting position
// *dir - direction of movement (if NULL, will do a point emission test in four directions)
// angularSpread - looseness of the spread
// minSpeed - minimum speed
// maxSpeed - maximum speed
// gravity - particle gravity for the sytem
// dampen - dampening amount on collisions
//-----------------------------------------------------------------------------
void CParticleCollision::Setup( const Vector &origin, const Vector *dir, float angularSpread, float minSpeed, float maxSpeed, float gravity, float dampen )
{
//Take the information for this simulation
m_flGravity = gravity;
m_flCollisionDampen = dampen;
m_nActivePlanes = 0;
//We take a rough estimation of the spray
float speedAvg = (minSpeed+maxSpeed)*0.5f;
//Point or directed?
if ( dir == NULL )
{
//Test all around
TestForPlane( origin, Vector( 1, 0, 0 ), speedAvg, gravity );
TestForPlane( origin, Vector( -1, 0, 0 ), speedAvg, gravity );
TestForPlane( origin, Vector( 0, 1, 0 ), speedAvg, gravity );
TestForPlane( origin, Vector( 0, -1, 0 ), speedAvg, gravity );
TestForPlane( origin, Vector( 0, 0, 1 ), speedAvg, gravity );
TestForPlane( origin, Vector( 0, 0, -1 ), speedAvg, gravity );
}
else
{
Vector vSkewDir, vRight;
QAngle vAngles;
//FIXME: Quicker conversion?
//FIXME: We need to factor in the angular spread instead
VectorAngles( *dir, vAngles );
AngleVectors( vAngles, NULL, &vRight, NULL );
//Test straight
TestForPlane( origin, *dir, speedAvg, gravity );
vSkewDir = vRight;
//Test right
TestForPlane( origin, vSkewDir, speedAvg, gravity );
vSkewDir *= -1.0f;
//Test left
TestForPlane( origin, vSkewDir, speedAvg, gravity );
#if __DEBUG_PARTICLE_COLLISION_OVERLAY
DevMsg( 1, "CParticleCollision: Found %d active plane(s)\n", m_nActivePlanes );
#endif //__DEBUG_PARTICLE_COLLISION_OVERLAY
}
}
//-----------------------------------------------------------------------------
// Purpose: Simulate movement, with collision
// Input : &origin - position of the particle
// &velocity - velocity of the particle
// &rollDelta - roll delta of the particle
// timeDelta - time step
//-----------------------------------------------------------------------------
bool CParticleCollision::MoveParticle( Vector &origin, Vector &velocity, float *rollDelta, float timeDelta, trace_t *pTrace )
{
//Don't bother with non-moving particles
if ( velocity == vec3_origin )
return false;
//Factor in gravity
velocity[2] -= m_flGravity * timeDelta;
//Move
Vector testPosition = ( origin + ( velocity * timeDelta ) );
//Only collide if we have active planes
if ( m_nActivePlanes > 0 )
{
//Collide
TraceLine( origin, testPosition, pTrace );
//See if we hit something
if ( pTrace->fraction != 1.0f )
{
#if __DEBUG_PARTICLE_COLLISION_RETEST
//Retest the collision with a true trace line to avoid errant collisions
UTIL_TraceLine( origin, testPosition, MASK_SOLID_BRUSHONLY, NULL, COLLISION_GROUP_NONE, pTrace );
#endif //__DEBUG_RETEST_COLLISION
//Did we hit anything?
if ( pTrace->fraction != 1.0f )
{
//See if we've settled
if ( ( pTrace->plane.normal[2] >= 0.5f ) && ( fabs( velocity[2] ) <= 48.0f ) )
{
//Leave the particle at the collision point
origin += velocity * ( (pTrace->fraction-COLLISION_EPSILON) * timeDelta );
//Stop the particle
velocity = vec3_origin;
if ( rollDelta != NULL )
{
*rollDelta = 0.0f;
}
return false;
}
else
{
//Move the particle to the collision point
origin += velocity * ( (pTrace->fraction-COLLISION_EPSILON) * timeDelta );
//Find the reflection vector
float proj = velocity.Dot( pTrace->plane.normal );
velocity += pTrace->plane.normal * (-proj*2.0f);
//Apply dampening
velocity *= random->RandomFloat( (m_flCollisionDampen-0.1f), (m_flCollisionDampen+0.1f) );
//Dampen the roll of the particles
if ( rollDelta != NULL )
{
(*rollDelta) *= -0.25f;
}
return true;
}
}
else
{
#if __DEBUG_PARTICLE_COLLISION_OVERLAY
//Display a false hit
debugoverlay->AddBoxOverlay( pTrace->endpos, Vector(-1,-1,-1), Vector(1,1,1), QAngle(0,0,0), 255, 0, 0, 16, __DEBUG_PARTICLE_COLLISION_OVERLAY_LIFETIME );
#endif //__DEBUG_PARTICLE_COLLISION_OVERLAY
}
}
}
//Simple move, no collision
origin = testPosition;
return false;
}