306 lines
7.1 KiB
C++
306 lines
7.1 KiB
C++
//========= Copyright Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose:
|
|
//
|
|
//=============================================================================//
|
|
|
|
#include "cbase.h"
|
|
#include "fx.h"
|
|
#include "c_gib.h"
|
|
#include "c_te_effect_dispatch.h"
|
|
#include "iefx.h"
|
|
#include "decals.h"
|
|
|
|
#define HUMAN_GIBS 1
|
|
#define ALIEN_GIBS 2
|
|
|
|
#define MAX_GIBS 4
|
|
|
|
#define HUMAN_GIB_COUNT 6
|
|
#define ALIEN_GIB_COUNT 4
|
|
|
|
const char *pHumanGibsModel = "models/gibs/hgibs.mdl";
|
|
const char *pAlienGibsModel = "models/gibs/agibs.mdl";
|
|
|
|
void GetBloodColorHL1( int bloodtype, unsigned char &r, unsigned char &g, unsigned char &b )
|
|
{
|
|
if (bloodtype == BLOOD_COLOR_RED)
|
|
{
|
|
r = 64;
|
|
g = 0;
|
|
b = 0;
|
|
}
|
|
else if ( bloodtype == BLOOD_COLOR_GREEN )
|
|
{
|
|
r = 195;
|
|
g = 195;
|
|
b = 0;
|
|
}
|
|
else if ( bloodtype == BLOOD_COLOR_YELLOW )
|
|
{
|
|
r = 0;
|
|
g = 195;
|
|
b = 195;
|
|
}
|
|
}
|
|
|
|
class C_HL1Gib : public C_Gib
|
|
{
|
|
typedef C_BaseAnimating BaseClass;
|
|
public:
|
|
|
|
static C_HL1Gib *CreateClientsideGib( const char *pszModelName, Vector vecOrigin, Vector vecForceDir, AngularImpulse vecAngularImp )
|
|
{
|
|
C_HL1Gib *pGib = new C_HL1Gib;
|
|
|
|
if ( pGib == NULL )
|
|
return NULL;
|
|
|
|
if ( pGib->InitializeGib( pszModelName, vecOrigin, vecForceDir, vecAngularImp ) == false )
|
|
return NULL;
|
|
|
|
return pGib;
|
|
}
|
|
|
|
// Decal the surface
|
|
virtual void HitSurface( C_BaseEntity *pOther )
|
|
{
|
|
int index = -1;
|
|
if ( m_iType == HUMAN_GIBS )
|
|
{
|
|
if ( !UTIL_IsLowViolence() ) // no blood decals if we're low violence.
|
|
{
|
|
index = decalsystem->GetDecalIndexForName( "Blood" );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
index = decalsystem->GetDecalIndexForName( "YellowBlood" );
|
|
}
|
|
|
|
if ( index >= 0 )
|
|
{
|
|
effects->DecalShoot( index, pOther->entindex(), pOther->GetModel(), pOther->GetAbsOrigin(), pOther->GetAbsAngles(), GetAbsOrigin(), 0, 0 );
|
|
}
|
|
|
|
|
|
if ( GetFlags() & FL_ONGROUND )
|
|
{
|
|
QAngle vAngles = GetAbsAngles();
|
|
QAngle vAngularVelocity = GetLocalAngularVelocity();
|
|
|
|
SetAbsVelocity( GetAbsVelocity() * 0.9 );
|
|
|
|
vAngles.x = 0;
|
|
vAngles.z = 0;
|
|
vAngularVelocity.x = 0;
|
|
vAngularVelocity.z = 0;
|
|
|
|
SetAbsAngles( vAngles );
|
|
SetLocalAngularVelocity( vAngularVelocity );
|
|
}
|
|
}
|
|
|
|
virtual void ClientThink( void );
|
|
|
|
int m_iType;
|
|
};
|
|
|
|
void C_HL1Gib::ClientThink( void )
|
|
{
|
|
SetRenderMode( kRenderTransAlpha );
|
|
m_nRenderFX = kRenderFxFadeSlow;
|
|
|
|
if ( m_clrRender->a == 5 )
|
|
{
|
|
Release();
|
|
return;
|
|
}
|
|
|
|
SetNextClientThink( gpGlobals->curtime + 1.0f );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : &origin -
|
|
//-----------------------------------------------------------------------------
|
|
void FX_HL1Gib( const Vector &origin, const Vector &direction, float scale, int iType, int iHealth, int iColor )
|
|
{
|
|
Vector offset;
|
|
int i;
|
|
|
|
offset = RandomVector( -16, 16 ) + origin;
|
|
|
|
if ( iType == HUMAN_GIBS )
|
|
{
|
|
Vector vVelocity;
|
|
AngularImpulse aImpulse;
|
|
|
|
// mix in some noise
|
|
vVelocity.x = random->RandomFloat( -100,100 );
|
|
vVelocity.y = random->RandomFloat ( -100,100 );
|
|
vVelocity.z = random->RandomFloat ( 200,300 );
|
|
|
|
aImpulse.x = random->RandomFloat ( 100, 200 );
|
|
aImpulse.y = random->RandomFloat ( 100, 300 );
|
|
|
|
if ( iHealth > -50)
|
|
{
|
|
vVelocity = vVelocity * 0.7;
|
|
}
|
|
else if ( iHealth > -200)
|
|
{
|
|
vVelocity = vVelocity * 2;
|
|
}
|
|
else
|
|
{
|
|
vVelocity = vVelocity * 4;
|
|
}
|
|
|
|
|
|
C_HL1Gib *pGib = C_HL1Gib::CreateClientsideGib( pHumanGibsModel, offset, vVelocity * 2, aImpulse );
|
|
|
|
//Spawn a head.
|
|
if ( pGib )
|
|
{
|
|
pGib->m_nBody = 0;
|
|
pGib->m_iType = iType;
|
|
}
|
|
}
|
|
|
|
// Spawn all the unique gibs
|
|
for ( i = 0; i < MAX_GIBS; i++ )
|
|
{
|
|
const char *pModelName = NULL;
|
|
int iNumBody = 0;
|
|
|
|
offset = RandomVector( -16, 16 ) + origin;
|
|
|
|
//TODO
|
|
Vector vVelocity = direction;
|
|
AngularImpulse aAImpulse;
|
|
|
|
// mix in some noise
|
|
vVelocity.x += random->RandomFloat( -0.25, 0.25 );
|
|
vVelocity.y += random->RandomFloat ( -0.25, 0.25 );
|
|
vVelocity.z += random->RandomFloat ( -0.25, 0.25 );
|
|
|
|
vVelocity = vVelocity * random->RandomFloat ( 300, 400 );
|
|
|
|
if ( iHealth > -50)
|
|
{
|
|
vVelocity = vVelocity * 0.7;
|
|
}
|
|
else if ( iHealth > -200)
|
|
{
|
|
vVelocity = vVelocity * 2;
|
|
}
|
|
else
|
|
{
|
|
vVelocity = vVelocity * 4;
|
|
}
|
|
|
|
aAImpulse.x = random->RandomFloat ( 100, 200 );
|
|
aAImpulse.y = random->RandomFloat ( 100, 300 );
|
|
|
|
if ( iType == HUMAN_GIBS )
|
|
{
|
|
pModelName = pHumanGibsModel;
|
|
iNumBody = HUMAN_GIB_COUNT;
|
|
}
|
|
else
|
|
{
|
|
pModelName = pAlienGibsModel;
|
|
iNumBody = ALIEN_GIB_COUNT;
|
|
}
|
|
|
|
|
|
C_HL1Gib *pGib = C_HL1Gib::CreateClientsideGib( pModelName, offset, vVelocity * 2, aAImpulse );
|
|
|
|
if ( pGib )
|
|
{
|
|
if ( iType == HUMAN_GIBS )
|
|
pGib->m_nBody = random->RandomInt( 1, iNumBody-1 );
|
|
else
|
|
pGib->m_nBody = random->RandomInt( 0, iNumBody-1 );
|
|
|
|
pGib->m_iType = iType;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Throw some blood (unless we're low violence, then we're done)
|
|
//
|
|
if ( iColor == BLOOD_COLOR_RED && UTIL_IsLowViolence() )
|
|
return;
|
|
|
|
CSmartPtr<CSimpleEmitter> pSimple = CSimpleEmitter::Create( "FX_HL1Gib" );
|
|
pSimple->SetSortOrigin( origin );
|
|
|
|
Vector vDir;
|
|
|
|
vDir.Random( -1.0f, 1.0f );
|
|
|
|
for ( i = 0; i < 4; i++ )
|
|
{
|
|
SimpleParticle *sParticle = (SimpleParticle *) pSimple->AddParticle( sizeof( SimpleParticle ), g_Mat_BloodPuff[0], origin );
|
|
|
|
if ( sParticle == NULL )
|
|
return;
|
|
|
|
sParticle->m_flLifetime = 0.0f;
|
|
sParticle->m_flDieTime = 1;
|
|
|
|
float speed = random->RandomFloat( 32.0f, 128.0f );
|
|
|
|
sParticle->m_vecVelocity = vDir * -speed;
|
|
sParticle->m_vecVelocity[2] -= 16.0f;
|
|
|
|
GetBloodColorHL1( iColor, sParticle->m_uchColor[0], sParticle->m_uchColor[1], sParticle->m_uchColor[2] );
|
|
|
|
sParticle->m_uchStartAlpha = 255;
|
|
sParticle->m_uchEndAlpha = 0;
|
|
sParticle->m_uchStartSize = random->RandomInt( 16, 32 );
|
|
sParticle->m_uchEndSize = sParticle->m_uchStartSize*random->RandomInt( 1, 4 );
|
|
sParticle->m_flRoll = random->RandomInt( 0, 360 );
|
|
sParticle->m_flRollDelta = random->RandomFloat( -4.0f, 4.0f );
|
|
}
|
|
|
|
for ( i = 0; i < 4; i++ )
|
|
{
|
|
SimpleParticle *sParticle = (SimpleParticle *) pSimple->AddParticle( sizeof( SimpleParticle ), g_Mat_BloodPuff[1], origin );
|
|
|
|
if ( sParticle == NULL )
|
|
{
|
|
return;
|
|
}
|
|
|
|
sParticle->m_flLifetime = 0.0f;
|
|
sParticle->m_flDieTime = 1;
|
|
|
|
float speed = random->RandomFloat( 16.0f, 128.0f );
|
|
|
|
sParticle->m_vecVelocity = vDir * -speed;
|
|
sParticle->m_vecVelocity[2] -= 16.0f;
|
|
|
|
GetBloodColorHL1( iColor, sParticle->m_uchColor[0], sParticle->m_uchColor[1], sParticle->m_uchColor[2] );
|
|
|
|
sParticle->m_uchStartAlpha = random->RandomInt( 64, 128 );
|
|
sParticle->m_uchEndAlpha = 0;
|
|
sParticle->m_uchStartSize = random->RandomInt( 16, 32 );
|
|
sParticle->m_uchEndSize = sParticle->m_uchStartSize*random->RandomInt( 1, 4 );
|
|
sParticle->m_flRoll = random->RandomInt( 0, 360 );
|
|
sParticle->m_flRollDelta = random->RandomFloat( -2.0f, 2.0f );
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : &data -
|
|
//-----------------------------------------------------------------------------
|
|
void HL1GibCallback( const CEffectData &data )
|
|
{
|
|
FX_HL1Gib( data.m_vOrigin, data.m_vNormal, data.m_flScale, data.m_nMaterial, -data.m_nHitBox, data.m_nColor );
|
|
}
|
|
|
|
DECLARE_CLIENT_EFFECT( "HL1Gib", HL1GibCallback );
|