hlsdk-portable/dlls/prop.cpp

1329 lines
35 KiB
C++
Raw Normal View History

2016-02-24 21:26:16 +00:00
/***
*
* Copyright (c) 1996-2001, Valve LLC. All rights reserved.
*
* This product contains software technology licensed from Id
* Software, Inc. ("Id Technology"). Id Technology (c) 1996 Id Software, Inc.
* All Rights Reserved.
*
* Use, distribution, and modification of this source code and/or resulting
* object code is restricted to non-commercial enhancements to products from
* Valve LLC. All other use, distribution, or modification is prohibited
* without written permission from Valve LLC.
*
****/
/*
===== generic grenade.cpp ========================================================
*/
#include "extdll.h"
#include "util.h"
#include "cbase.h"
#include "monsters.h"
#include "weapons.h"
#include "nodes.h"
#include "soundent.h"
#include "decals.h"
#include "player.h"
#include "explode.h"
#include "gamerules.h"
#define SF_PROP_RESPAWN 8 // enable autorespawn
2016-08-17 12:20:18 +00:00
#define SF_PROP_BREAKABLE 16 // enable break/explode
2016-02-24 21:26:16 +00:00
#define SF_PROP_FIXED 32 // don't move untill touch
2016-08-17 12:20:18 +00:00
typedef enum { expRandom, expDirected} Explosions;
typedef enum { matGlass = 0, matWood, matMetal, matFlesh, matCinderBlock, matCeilingTile, matComputer, matUnbreakableGlass, matRocks, matNone, matLastMaterial } Materials;
2016-02-24 21:26:16 +00:00
// round() has problems, so just implement it here
static inline int myround(float f)
{
return (int)(f + 0.5);
}
2016-02-24 21:26:16 +00:00
//extern "C" void AngleVectors (const vec3_t angles, vec3_t forward, vec3_t right, vec3_t up);
Vector UTIL_AngleVectorsF(const Vector &angles)
{
2016-03-02 14:22:22 +00:00
float rgflVecOut[3];
float rgflVecIn[3];
2016-08-17 12:20:18 +00:00
angles.CopyToArray(rgflVecIn);
g_engfuncs.pfnAngleVectors(rgflVecIn, rgflVecOut, NULL, NULL);
return Vector(rgflVecOut);
2016-02-24 21:26:16 +00:00
}
2016-08-17 12:20:18 +00:00
Vector UTIL_AngleVectorsR(const Vector &angles)
2016-02-24 21:26:16 +00:00
{
2016-03-02 14:22:22 +00:00
float rgflVecOut[3];
float rgflVecIn[3];
2016-08-17 12:20:18 +00:00
angles.CopyToArray(rgflVecIn);
g_engfuncs.pfnAngleVectors(rgflVecIn, NULL, rgflVecOut, NULL);
return Vector(rgflVecOut);
2016-02-24 21:26:16 +00:00
}
2016-08-17 12:20:18 +00:00
Vector UTIL_AngleVectorsU(const Vector &angles)
2016-02-24 21:26:16 +00:00
{
2016-03-02 14:22:22 +00:00
float rgflVecOut[3];
float rgflVecIn[3];
2016-08-17 12:20:18 +00:00
angles.CopyToArray(rgflVecIn);
g_engfuncs.pfnAngleVectors(rgflVecIn, NULL, rgflVecOut, NULL);
return Vector(rgflVecOut);
2016-02-24 21:26:16 +00:00
}
//===================grenade
enum PropShape
{
2016-03-02 14:22:22 +00:00
SHAPE_CYL_H = 0,
SHAPE_CYL_V,
SHAPE_BOX,
SHAPE_GENERIC,
SHAPE_SPHERE,
SHAPE_NOROTATE
2016-02-24 21:26:16 +00:00
};
class CProp : public CBaseEntity
{
public:
2016-08-17 12:20:18 +00:00
void Spawn(void);
2016-03-02 14:22:22 +00:00
void Precache();
void EXPORT BounceTouch(CBaseEntity *pOther);
//void EXPORT SlideTouch(CBaseEntity *pOther);
virtual void Use(CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value);
virtual void Force(CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value);
int TakeDamage(entvars_t* pevInflictor, entvars_t* pevAttacker, float flDamage, int bitsDamageType);
virtual int ObjectCaps(void) { return (CBaseEntity::ObjectCaps() & ~FCAP_ACROSS_TRANSITION) | FCAP_IMPULSE_USE | FCAP_CONTINUOUS_USE; }
virtual void BounceSound(void);
virtual int BloodColor(void) { return DONT_BLEED; }
virtual void Killed(entvars_t *pevAttacker, int iGib);
2016-03-23 19:08:13 +00:00
virtual float TouchGravGun( CBaseEntity *attacker, int stage )
2016-03-21 16:07:17 +06:00
{
2016-03-23 19:08:13 +00:00
float speed = 2500;
2016-03-25 20:24:38 +00:00
if( pev->deadflag )
return 0;
2016-05-22 12:47:14 +00:00
pev->movetype = MOVETYPE_BOUNCE;
2016-03-23 20:13:22 +06:00
if(stage)
{
pev->nextthink = gpGlobals->time + m_flRespawnTime;
2016-03-25 20:24:38 +00:00
SetThink( &CProp::RespawnThink );
2016-03-23 20:13:22 +06:00
}
2016-03-23 19:08:13 +00:00
if( stage == 2 )
{
UTIL_MakeVectors( attacker->pev->v_angle + attacker->pev->punchangle);
Vector atarget = UTIL_VecToAngles(gpGlobals->v_forward);
pev->angles.x = UTIL_AngleMod(pev->angles.x);
pev->angles.y = UTIL_AngleMod(pev->angles.y);
pev->angles.z = UTIL_AngleMod(pev->angles.z);
atarget.x = UTIL_AngleMod(atarget.x);
atarget.y = UTIL_AngleMod(atarget.y);
atarget.z = UTIL_AngleMod(atarget.z);
pev->avelocity.x = UTIL_AngleDiff(atarget.x, pev->angles.x) * 10;
pev->avelocity.y = UTIL_AngleDiff(atarget.y, pev->angles.y) * 10;
pev->avelocity.z = UTIL_AngleDiff(atarget.z, pev->angles.z) * 10;
}
2016-03-28 18:38:09 +06:00
if( stage == 3 )
{
pev->avelocity.y = pev->avelocity.y*1.5 + RANDOM_FLOAT(100, -100);
pev->avelocity.x = pev->avelocity.x*1.5 + RANDOM_FLOAT(100, -100);
}
2016-03-27 03:56:54 +06:00
if( !m_attacker || m_attacker == this )
2016-03-23 20:13:22 +06:00
{
m_owner2 = attacker;
m_attacker = attacker;
2016-03-23 19:08:13 +00:00
return speed;
2016-03-23 20:13:22 +06:00
}
2016-03-27 03:56:54 +06:00
if( !m_owner2 && m_attacker && ( pev->velocity.Length() < 400) )
2016-03-23 20:13:22 +06:00
m_attacker = attacker;
2016-03-23 19:08:13 +00:00
return speed;
2016-03-23 20:13:22 +06:00
if( ( stage == 2 ) && ( m_attacker == attacker ) )
{
m_owner2 = attacker;
2016-03-23 19:08:13 +00:00
return speed;
2016-03-23 20:13:22 +06:00
}
2016-03-23 19:08:13 +00:00
return 0;
2016-03-21 16:07:17 +06:00
}
2016-03-02 14:22:22 +00:00
void CheckRotate();
2016-03-02 15:01:27 +00:00
void EXPORT RespawnThink();
void EXPORT AngleThink();
void EXPORT DeployThink();
void EXPORT DieThink();
2016-03-02 14:22:22 +00:00
void DamageSound( void );
void PropRespawn();
2016-08-17 12:20:18 +00:00
void KeyValue( KeyValueData* pkvd);
2016-03-02 14:22:22 +00:00
static const char *pSoundsWood[];
static const char *pSoundsFlesh[];
static const char *pSoundsGlass[];
static const char *pSoundsMetal[];
static const char *pSoundsConcrete[];
static const char *pSpawnObjects[];
2016-08-17 12:20:18 +00:00
inline BOOL Explodable( void ) { return ExplosionMagnitude() > 0; }
inline int ExplosionMagnitude( void ) { return pev->impulse; }
inline void ExplosionSetMagnitude( int magnitude ) { pev->impulse = magnitude; }
2016-03-02 14:22:22 +00:00
static void MaterialSoundPrecache( Materials precacheMaterial );
static void MaterialSoundRandom( edict_t *pEdict, Materials soundMaterial, float volume );
static const char **MaterialSoundList( Materials precacheMaterial, int &soundCount );
2016-08-17 12:20:18 +00:00
void EXPORT Die( void );
2016-03-02 14:22:22 +00:00
BOOL m_bBarrel;
float m_flFloorFriction;
float m_flCollideFriction;
// hull sizes
Vector minsH, maxsH;
Vector minsV, maxsV;
// spawn backup;
Vector spawnOrigin;
Vector spawnAngles;
2016-03-23 16:11:21 +06:00
EHANDLE m_owner2;
EHANDLE m_attacker;
2016-03-02 14:22:22 +00:00
float m_flNextAttack;
float m_flRespawnTime;
PropShape m_shape;
PropShape m_oldshape;
2016-03-23 16:11:21 +06:00
EHANDLE m_pHolstered;
2016-03-02 14:22:22 +00:00
float m_flSpawnHealth;
2016-08-17 12:20:18 +00:00
int m_idShard;
float m_angle;
int m_iszGibModel;
Materials m_Material;
Explosions m_Explosion;
2016-03-02 14:22:22 +00:00
int m_iaCustomAnglesX[10];
int m_iaCustomAnglesZ[10];
2016-02-24 21:26:16 +00:00
};
2016-08-17 12:20:18 +00:00
LINK_ENTITY_TO_CLASS(prop, CProp);
2016-02-24 21:26:16 +00:00
const char *CProp::pSoundsWood[] =
{
2016-03-02 14:22:22 +00:00
"debris/wood1.wav",
"debris/wood2.wav",
"debris/wood3.wav",
2016-02-24 21:26:16 +00:00
};
const char *CProp::pSoundsFlesh[] =
{
2016-03-02 14:22:22 +00:00
"debris/flesh1.wav",
"debris/flesh2.wav",
"debris/flesh3.wav",
"debris/flesh5.wav",
"debris/flesh6.wav",
"debris/flesh7.wav",
2016-02-24 21:26:16 +00:00
};
const char *CProp::pSoundsMetal[] =
{
2016-03-02 14:22:22 +00:00
"debris/metal1.wav",
"debris/metal2.wav",
"debris/metal3.wav",
2016-02-24 21:26:16 +00:00
};
const char *CProp::pSoundsConcrete[] =
{
2016-03-02 14:22:22 +00:00
"debris/concrete1.wav",
"debris/concrete2.wav",
"debris/concrete3.wav",
2016-02-24 21:26:16 +00:00
};
2016-08-17 12:20:18 +00:00
2016-02-24 21:26:16 +00:00
const char *CProp::pSoundsGlass[] =
{
2016-03-02 14:22:22 +00:00
"debris/glass1.wav",
"debris/glass2.wav",
"debris/glass3.wav",
2016-02-24 21:26:16 +00:00
};
const char **CProp::MaterialSoundList( Materials precacheMaterial, int &soundCount )
{
2016-08-17 12:20:18 +00:00
const char **pSoundList = NULL;
2016-03-02 14:22:22 +00:00
2016-08-17 12:20:18 +00:00
switch ( precacheMaterial )
2016-03-02 14:22:22 +00:00
{
case matWood:
pSoundList = pSoundsWood;
2016-08-17 12:20:18 +00:00
soundCount = ARRAYSIZE(pSoundsWood);
2016-03-02 14:22:22 +00:00
break;
case matFlesh:
pSoundList = pSoundsFlesh;
2016-08-17 12:20:18 +00:00
soundCount = ARRAYSIZE(pSoundsFlesh);
2016-03-02 14:22:22 +00:00
break;
case matComputer:
case matUnbreakableGlass:
case matGlass:
pSoundList = pSoundsGlass;
2016-08-17 12:20:18 +00:00
soundCount = ARRAYSIZE(pSoundsGlass);
2016-03-02 14:22:22 +00:00
break;
2016-08-17 12:20:18 +00:00
2016-03-02 14:22:22 +00:00
case matMetal:
pSoundList = pSoundsMetal;
2016-08-17 12:20:18 +00:00
soundCount = ARRAYSIZE(pSoundsMetal);
2016-03-02 14:22:22 +00:00
break;
2016-08-17 12:20:18 +00:00
2016-03-02 14:22:22 +00:00
case matCinderBlock:
case matRocks:
2016-08-18 17:49:35 +00:00
case matCeilingTile:
2016-03-02 14:22:22 +00:00
pSoundList = pSoundsConcrete;
2016-08-17 12:20:18 +00:00
soundCount = ARRAYSIZE(pSoundsConcrete);
2016-03-02 14:22:22 +00:00
break;
2016-08-17 12:20:18 +00:00
2016-03-02 14:22:22 +00:00
case matNone:
default:
soundCount = 0;
break;
}
return pSoundList;
2016-02-24 21:26:16 +00:00
}
void CProp::MaterialSoundPrecache( Materials precacheMaterial )
{
2016-08-17 12:20:18 +00:00
const char **pSoundList;
int i, soundCount = 0;
2016-02-24 21:26:16 +00:00
2016-03-02 14:22:22 +00:00
pSoundList = MaterialSoundList( precacheMaterial, soundCount );
2016-02-24 21:26:16 +00:00
2016-08-17 12:20:18 +00:00
for ( i = 0; i < soundCount; i++ )
2016-03-02 14:22:22 +00:00
{
PRECACHE_SOUND( (char *)pSoundList[i] );
}
2016-02-24 21:26:16 +00:00
}
void CProp::MaterialSoundRandom( edict_t *pEdict, Materials soundMaterial, float volume )
{
2016-08-17 12:20:18 +00:00
const char **pSoundList;
int soundCount = 0;
2016-02-24 21:26:16 +00:00
2016-03-02 14:22:22 +00:00
pSoundList = MaterialSoundList( soundMaterial, soundCount );
2016-02-24 21:26:16 +00:00
2016-08-17 12:20:18 +00:00
if ( soundCount )
EMIT_SOUND( pEdict, CHAN_BODY, pSoundList[ RANDOM_LONG(0,soundCount-1) ], volume, 1.0 );
2016-02-24 21:26:16 +00:00
}
void CProp::Precache( void )
{
2016-03-23 19:37:51 +06:00
const char *pGibName = NULL;
2016-03-02 14:22:22 +00:00
if( !pev->model )
pev->model = MAKE_STRING( "models/xash/barrel_brown.mdl" );
2016-08-18 17:49:35 +00:00
if( m_Material == matGlass && !(pev->spawnflags & SF_PROP_BREAKABLE) )
m_Material = matWood;
2016-08-17 12:20:18 +00:00
switch (m_Material)
2016-03-02 14:22:22 +00:00
{
case matWood:
pGibName = "models/woodgibs.mdl";
2016-08-17 12:20:18 +00:00
PRECACHE_SOUND("debris/bustcrate1.wav");
PRECACHE_SOUND("debris/bustcrate2.wav");
2016-03-02 14:22:22 +00:00
break;
case matFlesh:
pGibName = "models/fleshgibs.mdl";
2016-08-17 12:20:18 +00:00
PRECACHE_SOUND("debris/bustflesh1.wav");
PRECACHE_SOUND("debris/bustflesh2.wav");
2016-03-02 14:22:22 +00:00
break;
case matComputer:
2016-08-17 12:20:18 +00:00
PRECACHE_SOUND("buttons/spark5.wav");
PRECACHE_SOUND("buttons/spark6.wav");
2016-03-02 14:22:22 +00:00
pGibName = "models/computergibs.mdl";
2016-08-17 12:20:18 +00:00
PRECACHE_SOUND("debris/bustmetal1.wav");
PRECACHE_SOUND("debris/bustmetal2.wav");
2016-03-02 14:22:22 +00:00
break;
2016-08-17 12:20:18 +00:00
2016-03-02 14:22:22 +00:00
case matUnbreakableGlass:
case matGlass:
pGibName = "models/glassgibs.mdl";
2016-08-17 12:20:18 +00:00
PRECACHE_SOUND("debris/bustglass1.wav");
PRECACHE_SOUND("debris/bustglass2.wav");
2016-03-02 14:22:22 +00:00
break;
case matMetal:
pGibName = "models/metalplategibs.mdl";
2016-08-17 12:20:18 +00:00
PRECACHE_SOUND("debris/bustmetal1.wav");
PRECACHE_SOUND("debris/bustmetal2.wav");
2016-03-02 14:22:22 +00:00
break;
case matCinderBlock:
pGibName = "models/cindergibs.mdl";
2016-08-17 12:20:18 +00:00
PRECACHE_SOUND("debris/bustconcrete1.wav");
PRECACHE_SOUND("debris/bustconcrete2.wav");
2016-03-02 14:22:22 +00:00
break;
case matRocks:
pGibName = "models/rockgibs.mdl";
2016-08-17 12:20:18 +00:00
PRECACHE_SOUND("debris/bustconcrete1.wav");
PRECACHE_SOUND("debris/bustconcrete2.wav");
2016-03-02 14:22:22 +00:00
break;
case matCeilingTile:
pGibName = "models/ceilinggibs.mdl";
2016-08-17 12:20:18 +00:00
PRECACHE_SOUND ("debris/bustceiling.wav");
2016-03-02 14:22:22 +00:00
break;
}
MaterialSoundPrecache( m_Material );
if ( m_iszGibModel )
pGibName = STRING(m_iszGibModel);
2016-03-23 19:37:51 +06:00
if( pGibName )
m_idShard = PRECACHE_MODEL( (char *)pGibName );
2016-03-02 14:22:22 +00:00
PRECACHE_MODEL( (char *)STRING(pev->model) );
2016-02-24 21:26:16 +00:00
}
void CProp::DamageSound( void )
{
2016-03-02 14:22:22 +00:00
int pitch;
float fvol;
char *rgpsz[6];
2016-03-23 19:37:51 +06:00
int i = 0;
2016-03-02 14:22:22 +00:00
int material = m_Material;
2016-02-24 21:26:16 +00:00
2016-08-17 12:20:18 +00:00
// if (RANDOM_LONG(0,1))
// return;
2016-02-24 21:26:16 +00:00
2016-08-17 12:20:18 +00:00
if (RANDOM_LONG(0,2))
2016-03-02 14:22:22 +00:00
pitch = PITCH_NORM;
else
2016-08-17 12:20:18 +00:00
pitch = 95 + RANDOM_LONG(0,34);
2016-03-02 14:22:22 +00:00
2016-08-17 12:20:18 +00:00
fvol = RANDOM_FLOAT(0.75, 1.0);
2016-03-02 14:22:22 +00:00
2016-08-17 12:20:18 +00:00
if (material == matComputer && RANDOM_LONG(0,1))
2016-03-02 14:22:22 +00:00
material = matMetal;
2016-08-17 12:20:18 +00:00
switch (material)
2016-03-02 14:22:22 +00:00
{
case matComputer:
case matGlass:
case matUnbreakableGlass:
rgpsz[0] = "debris/glass1.wav";
rgpsz[1] = "debris/glass2.wav";
rgpsz[2] = "debris/glass3.wav";
i = 3;
break;
2016-08-17 12:20:18 +00:00
2016-03-02 14:22:22 +00:00
case matWood:
rgpsz[0] = "debris/wood1.wav";
rgpsz[1] = "debris/wood2.wav";
rgpsz[2] = "debris/wood3.wav";
i = 3;
break;
2016-08-17 12:20:18 +00:00
2016-03-02 14:22:22 +00:00
case matMetal:
rgpsz[0] = "debris/metal1.wav";
rgpsz[1] = "debris/metal3.wav";
rgpsz[2] = "debris/metal2.wav";
i = 2;
break;
2016-08-17 12:20:18 +00:00
2016-03-02 14:22:22 +00:00
case matFlesh:
rgpsz[0] = "debris/flesh1.wav";
rgpsz[1] = "debris/flesh2.wav";
rgpsz[2] = "debris/flesh3.wav";
rgpsz[3] = "debris/flesh5.wav";
rgpsz[4] = "debris/flesh6.wav";
rgpsz[5] = "debris/flesh7.wav";
i = 6;
break;
2016-08-17 12:20:18 +00:00
2016-03-02 14:22:22 +00:00
case matRocks:
case matCinderBlock:
rgpsz[0] = "debris/concrete1.wav";
rgpsz[1] = "debris/concrete2.wav";
rgpsz[2] = "debris/concrete3.wav";
i = 3;
break;
2016-08-17 12:20:18 +00:00
2016-03-02 14:22:22 +00:00
case matCeilingTile:
// UNDONE: no ceiling tile shard sound yet
2016-08-18 17:49:35 +00:00
rgpsz[0] = "debris/concrete1.wav";
rgpsz[1] = "debris/concrete2.wav";
rgpsz[2] = "debris/concrete3.wav";
i = 3;
2016-03-02 14:22:22 +00:00
break;
}
2016-08-17 12:20:18 +00:00
if (i)
EMIT_SOUND_DYN(ENT(pev), CHAN_VOICE, rgpsz[RANDOM_LONG(0,i-1)], fvol, ATTN_NORM, 0, pitch);
2016-02-24 21:26:16 +00:00
}
void CProp::Die( void )
{
2016-03-02 14:22:22 +00:00
Vector vecSpot;// shard origin
Vector vecVelocity;// shard velocity
char cFlag = 0;
int pitch;
float fvol;
2016-08-17 12:20:18 +00:00
pitch = 95 + RANDOM_LONG(0,29);
2016-03-02 14:22:22 +00:00
2016-08-17 12:20:18 +00:00
if (pitch > 97 && pitch < 103)
2016-03-02 14:22:22 +00:00
pitch = 100;
// The more negative pev->health, the louder
// the sound should be.
2016-08-17 12:20:18 +00:00
fvol = RANDOM_FLOAT(0.85, 1.0) + (fabs(pev->health) / 100.0);
2016-03-02 14:22:22 +00:00
2016-08-17 12:20:18 +00:00
if (fvol > 1.0)
2016-03-02 14:22:22 +00:00
fvol = 1.0;
2016-08-17 12:20:18 +00:00
switch (m_Material)
2016-03-02 14:22:22 +00:00
{
case matGlass:
2016-08-17 12:20:18 +00:00
switch ( RANDOM_LONG(0,1) )
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
case 0: EMIT_SOUND_DYN(ENT(pev), CHAN_VOICE, "debris/bustglass1.wav", fvol, ATTN_NORM, 0, pitch);
2016-03-02 14:22:22 +00:00
break;
2016-08-17 12:20:18 +00:00
case 1: EMIT_SOUND_DYN(ENT(pev), CHAN_VOICE, "debris/bustglass2.wav", fvol, ATTN_NORM, 0, pitch);
2016-03-02 14:22:22 +00:00
break;
}
cFlag = BREAK_GLASS;
break;
2016-08-17 12:20:18 +00:00
2016-03-02 14:22:22 +00:00
case matWood:
2016-08-17 12:20:18 +00:00
switch ( RANDOM_LONG(0,1) )
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
case 0: EMIT_SOUND_DYN(ENT(pev), CHAN_VOICE, "debris/bustcrate1.wav", fvol, ATTN_NORM, 0, pitch);
2016-03-02 14:22:22 +00:00
break;
2016-08-17 12:20:18 +00:00
case 1: EMIT_SOUND_DYN(ENT(pev), CHAN_VOICE, "debris/bustcrate2.wav", fvol, ATTN_NORM, 0, pitch);
2016-03-02 14:22:22 +00:00
break;
}
cFlag = BREAK_WOOD;
break;
2016-08-17 12:20:18 +00:00
2016-03-02 14:22:22 +00:00
case matComputer:
case matMetal:
2016-08-17 12:20:18 +00:00
switch ( RANDOM_LONG(0,1) )
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
case 0: EMIT_SOUND_DYN(ENT(pev), CHAN_VOICE, "debris/bustmetal1.wav", fvol, ATTN_NORM, 0, pitch);
2016-03-02 14:22:22 +00:00
break;
2016-08-17 12:20:18 +00:00
case 1: EMIT_SOUND_DYN(ENT(pev), CHAN_VOICE, "debris/bustmetal2.wav", fvol, ATTN_NORM, 0, pitch);
2016-03-02 14:22:22 +00:00
break;
}
cFlag = BREAK_METAL;
break;
2016-08-17 12:20:18 +00:00
2016-03-02 14:22:22 +00:00
case matFlesh:
2016-08-17 12:20:18 +00:00
switch ( RANDOM_LONG(0,1) )
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
case 0: EMIT_SOUND_DYN(ENT(pev), CHAN_VOICE, "debris/bustflesh1.wav", fvol, ATTN_NORM, 0, pitch);
2016-03-02 14:22:22 +00:00
break;
2016-08-17 12:20:18 +00:00
case 1: EMIT_SOUND_DYN(ENT(pev), CHAN_VOICE, "debris/bustflesh2.wav", fvol, ATTN_NORM, 0, pitch);
2016-03-02 14:22:22 +00:00
break;
}
cFlag = BREAK_FLESH;
break;
2016-08-17 12:20:18 +00:00
2016-03-02 14:22:22 +00:00
case matRocks:
case matCinderBlock:
2016-08-17 12:20:18 +00:00
switch ( RANDOM_LONG(0,1) )
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
case 0: EMIT_SOUND_DYN(ENT(pev), CHAN_VOICE, "debris/bustconcrete1.wav", fvol, ATTN_NORM, 0, pitch);
2016-03-02 14:22:22 +00:00
break;
2016-08-17 12:20:18 +00:00
case 1: EMIT_SOUND_DYN(ENT(pev), CHAN_VOICE, "debris/bustconcrete2.wav", fvol, ATTN_NORM, 0, pitch);
2016-03-02 14:22:22 +00:00
break;
}
cFlag = BREAK_CONCRETE;
break;
2016-08-17 12:20:18 +00:00
2016-03-02 14:22:22 +00:00
case matCeilingTile:
2016-08-17 12:20:18 +00:00
EMIT_SOUND_DYN(ENT(pev), CHAN_VOICE, "debris/bustceiling.wav", fvol, ATTN_NORM, 0, pitch);
2016-03-02 14:22:22 +00:00
break;
}
2016-08-17 12:20:18 +00:00
if (m_Explosion == expDirected)
2016-03-02 14:22:22 +00:00
vecVelocity = g_vecAttackDir * 200;
else
{
vecVelocity.x = 0;
vecVelocity.y = 0;
vecVelocity.z = 0;
}
vecSpot = pev->origin + (pev->mins + pev->maxs) * 0.5;
MESSAGE_BEGIN( MSG_PVS, SVC_TEMPENTITY, vecSpot );
WRITE_BYTE( TE_BREAKMODEL);
// position
WRITE_COORD( vecSpot.x );
WRITE_COORD( vecSpot.y );
WRITE_COORD( vecSpot.z );
// size
WRITE_COORD( pev->size.x);
WRITE_COORD( pev->size.y);
WRITE_COORD( pev->size.z);
// velocity
WRITE_COORD( vecVelocity.x );
WRITE_COORD( vecVelocity.y );
WRITE_COORD( vecVelocity.z );
// randomization
2016-06-04 15:23:46 +03:00
WRITE_BYTE( 8 );
2016-03-02 14:22:22 +00:00
// Model
WRITE_SHORT( m_idShard ); //model id#
// # of shards
2016-06-04 15:23:46 +03:00
WRITE_BYTE( 50 ); // let client decide
2016-03-02 14:22:22 +00:00
// duration
2016-06-04 15:23:46 +03:00
WRITE_BYTE( 250 );// 2.5 seconds
2016-03-02 14:22:22 +00:00
// flags
WRITE_BYTE( cFlag );
MESSAGE_END();
float size = pev->size.x;
2016-08-17 12:20:18 +00:00
if ( size < pev->size.y )
2016-03-02 14:22:22 +00:00
size = pev->size.y;
2016-08-17 12:20:18 +00:00
if ( size < pev->size.z )
2016-03-02 14:22:22 +00:00
size = pev->size.z;
// !!! HACK This should work!
// Build a box above the entity that looks like an 8 pixel high sheet
Vector mins = pev->absmin;
Vector maxs = pev->absmax;
mins.z = pev->absmax.z;
maxs.z += 8;
// BUGBUG -- can only find 256 entities on a breakable -- should be enough
CBaseEntity *pList[256];
int count = UTIL_EntitiesInBox( pList, 256, mins, maxs, FL_ONGROUND );
2016-08-17 12:20:18 +00:00
if ( count )
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
for ( int i = 0; i < count; i++ )
2016-03-02 14:22:22 +00:00
{
ClearBits( pList[i]->pev->flags, FL_ONGROUND );
pList[i]->pev->groundentity = NULL;
}
}
// Don't fire something that could fire myself
pev->targetname = 0;
pev->solid = SOLID_NOT;
// Fire targets on break
SUB_UseTargets( NULL, USE_TOGGLE, 0 );
2016-03-23 19:37:51 +06:00
if ( Explodable() && (m_attacker) )
2016-03-02 14:22:22 +00:00
{
2016-03-23 16:11:21 +06:00
ExplosionCreate( pev->origin, pev->angles, m_attacker->edict(), ExplosionMagnitude(), FALSE );
RadiusDamage ( pev->origin, pev, m_attacker->pev, ExplosionMagnitude(), ExplosionMagnitude() * 2.5 , CLASS_NONE, DMG_BLAST );
2016-03-02 14:22:22 +00:00
}
2016-08-17 12:20:18 +00:00
UTIL_SetSize(pev, Vector( 0, 0, 0), Vector(0, 0, 0) );
2016-02-24 21:26:16 +00:00
}
2016-08-17 12:20:18 +00:00
void CProp::Killed(entvars_t *pevAttacker, int iGib)
2016-02-24 21:26:16 +00:00
{
2016-03-02 14:22:22 +00:00
pev->takedamage = DAMAGE_NO;
pev->deadflag = DEAD_DEAD;
pev->solid = SOLID_NOT;
pev->effects |= EF_NODRAW;
pev->nextthink = gpGlobals->time + m_flRespawnTime;
SetThink( &CProp::RespawnThink );
2016-03-03 20:31:01 +00:00
SetTouch( NULL );
SetUse( NULL );
2016-06-04 15:23:46 +03:00
//m_owner2 = NULL;
//m_attacker = NULL;
2016-02-24 21:26:16 +00:00
}
2016-08-17 12:20:18 +00:00
void CProp::Use(CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value)
2016-02-24 21:26:16 +00:00
{
2016-08-17 12:20:18 +00:00
if( pev->health <= 0)
2016-03-02 15:01:27 +00:00
return;
2016-05-22 12:47:14 +00:00
pev->movetype = MOVETYPE_BOUNCE;
2016-03-23 16:11:21 +06:00
if (m_owner2 != pActivator)
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
if (pev->velocity.Length() < 100 && pActivator->IsPlayer())
2016-03-02 14:22:22 +00:00
{
2016-03-23 16:11:21 +06:00
m_owner2 = m_attacker = pActivator;
2016-03-02 14:22:22 +00:00
}
else
return;
}
if( pActivator->IsPlayer() )
{
2016-03-23 16:11:21 +06:00
m_pHolstered = pActivator;
CBasePlayer *player = (CBasePlayer*)pActivator;
if( player )
2016-03-02 14:22:22 +00:00
{
2016-03-23 16:11:21 +06:00
if ( player->m_pActiveItem )
2016-03-02 14:22:22 +00:00
{
2016-03-23 16:11:21 +06:00
CBasePlayerWeapon *weapon = (CBasePlayerWeapon *) player->m_pActiveItem->GetWeaponPtr();
2016-03-02 14:22:22 +00:00
//m_Holstered->m_pActiveItem->Holster(); // strange bug here. ValveWHY?
// HACK: prevent attack
if( weapon )
{
weapon->m_flNextPrimaryAttack += 0.1;
weapon->m_flNextSecondaryAttack += 0.1;
}
2016-03-23 16:11:21 +06:00
player->m_iHideHUD |= HIDEHUD_WEAPONS;
player->pev->weaponmodel = 0;
player->pev->viewmodel = 0;
2016-03-02 14:22:22 +00:00
}
SetThink( &CProp::DeployThink );
pev->nextthink = gpGlobals->time + 0.2;
}
}
2016-03-23 16:11:21 +06:00
Vector target = pActivator->pev->origin + UTIL_GetAimVector(m_owner2->edict(), 1000) * 50;
2016-03-02 14:22:22 +00:00
target.z = target.z + 32;
pev->velocity = (target - VecBModelOrigin(pev)) * 10;
2016-03-23 16:11:21 +06:00
Vector atarget = UTIL_VecToAngles(UTIL_GetAimVector(m_owner2->edict(), 1000));
2016-03-02 14:22:22 +00:00
pev->angles.x = UTIL_AngleMod(pev->angles.x);
pev->angles.y = UTIL_AngleMod(pev->angles.y);
pev->angles.z = UTIL_AngleMod(pev->angles.z);
atarget.x = UTIL_AngleMod(atarget.x);
atarget.y = UTIL_AngleMod(atarget.y);
atarget.z = UTIL_AngleMod(atarget.z);
pev->avelocity.x = UTIL_AngleDiff(atarget.x, pev->angles.x) * 10;
pev->avelocity.y = UTIL_AngleDiff(atarget.y, pev->angles.y) * 10;
pev->avelocity.z = UTIL_AngleDiff(atarget.z, pev->angles.z) * 10;
//pev->angles.z += (0 - pev->angles.z) * 0.06;
if ((pActivator->pev->button & (IN_ATTACK)))
{
2016-03-23 16:11:21 +06:00
pev->velocity = UTIL_GetAimVector(m_owner2->edict(), 1000) * 1000;
2016-03-02 14:22:22 +00:00
pev->avelocity.y = pev->avelocity.y*1.5 + RANDOM_FLOAT(100, -100);
pev->avelocity.x = pev->avelocity.x*1.5 + RANDOM_FLOAT(100, -100);
//pev->avelocity.z = pev->avelocity.z*0.5 + RANDOM_FLOAT ( 100, -100 );
}
2016-08-17 12:20:18 +00:00
if ((pActivator->pev->button & (IN_ATTACK2)))
2016-03-02 14:22:22 +00:00
{
//m_Horizontal = false;
//pev->angles.z = 0;
}
2016-08-17 12:20:18 +00:00
// m_Horizontal = (fabs(UTIL_AngleDiff(pev->angles.z, 90)) < 20) || ( sin(pev->angles.x/180*M_PI) > 0.1);
2016-03-02 14:22:22 +00:00
// CheckRotate();
2016-08-17 12:20:18 +00:00
//ALERT( at_console, "Prop use!\n");
2016-02-24 21:26:16 +00:00
}
2016-08-17 12:20:18 +00:00
void CProp::Force(CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value)
2016-02-24 21:26:16 +00:00
{
2016-03-02 15:01:27 +00:00
if( pev->health <= 0 )
return;
2016-03-23 16:11:21 +06:00
if (m_owner2 != pActivator)
2016-03-02 14:22:22 +00:00
{
if (pev->velocity.Length() < 100 && pActivator->IsPlayer())
2016-03-23 16:11:21 +06:00
m_attacker = pActivator;
2016-03-02 14:22:22 +00:00
else
return;
}
2016-08-17 12:20:18 +00:00
if ((pActivator->pev->button & (IN_ATTACK)))
2016-03-02 14:22:22 +00:00
{
2016-03-23 16:11:21 +06:00
pev->velocity = UTIL_GetAimVector(m_owner2->edict(), 3000) * 1000;
2016-03-02 14:22:22 +00:00
pev->avelocity.y = pev->avelocity.y*1.5 + RANDOM_FLOAT(100, -100);
pev->avelocity.x = pev->avelocity.x*1.5 + RANDOM_FLOAT(100, -100);
//pev->avelocity.z = pev->avelocity.z*0.5 + RANDOM_FLOAT ( 100, -100 );
}
2016-08-17 12:20:18 +00:00
if ((pActivator->pev->button & (IN_ATTACK2)))
2016-03-02 14:22:22 +00:00
{
//m_Horizontal = false;
//pev->angles.z = 0;
}
pev->nextthink = gpGlobals->time + m_flRespawnTime;
2016-08-17 12:20:18 +00:00
SetThink( &CProp::RespawnThink);
2016-02-24 21:26:16 +00:00
}
void CProp::CheckRotate()
{
2016-03-02 14:22:22 +00:00
if( m_shape != SHAPE_CYL_H && m_shape != SHAPE_CYL_V )
{
UTIL_SetSize(pev, minsH, maxsH);
return;
}
2016-08-17 12:20:18 +00:00
if( (fabs(UTIL_AngleDiff(pev->angles.z, 90)) < 20) ||
(fabs(sin(pev->angles.x / 180 * M_PI)) > 0.3) )
2016-03-02 14:22:22 +00:00
m_shape = SHAPE_CYL_H;
else
m_shape = SHAPE_CYL_V;
2016-08-17 12:20:18 +00:00
if (m_oldshape != m_shape)
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
if (m_shape == SHAPE_CYL_H)
2016-03-02 14:22:22 +00:00
{
pev->angles.y += 90;
2016-08-17 12:20:18 +00:00
ALERT(at_console, "setH: %f %f %f\n", pev->angles.x, pev->angles.y, pev->angles.z);
2016-03-02 14:22:22 +00:00
2016-08-17 12:20:18 +00:00
UTIL_SetSize(pev, minsH, maxsH);
2016-03-02 14:22:22 +00:00
}
2016-08-17 12:20:18 +00:00
else if (m_shape == SHAPE_CYL_V)
2016-03-02 14:22:22 +00:00
{
Vector mins = pev->absmin;
Vector maxs = pev->absmax;
mins.z = pev->absmax.z;
maxs.z += 10;
2016-08-17 12:20:18 +00:00
ALERT(at_console, "setV: %f %f %f\n", pev->angles.x, pev->angles.y, pev->angles.z);
2016-03-02 14:22:22 +00:00
// BUGBUG -- can only find 256 entities on a prop -- should be enough
CBaseEntity *pList[256];
int count = UTIL_EntitiesInBox( pList, 256, mins, maxs, FL_ONGROUND );
2016-08-17 12:20:18 +00:00
if ( count )
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
for ( int i = 0; i < count; i++ )
2016-03-02 14:22:22 +00:00
{
pList[i]->pev->origin.z += 10;
}
}
pev->origin.z += 10;
//pev->angles.y -= 90;
2016-08-17 12:20:18 +00:00
UTIL_SetSize(pev, minsV, maxsV);
2016-03-02 14:22:22 +00:00
}
2016-08-17 12:20:18 +00:00
//DROP_TO_FLOOR(edict());
2016-03-02 14:22:22 +00:00
//pev->origin.z += 0.5;
m_oldshape = m_shape;
}
2016-02-24 21:26:16 +00:00
}
void CProp::DeployThink( void )
{
2016-03-02 14:22:22 +00:00
if( m_pHolstered )
{
2016-03-23 16:11:21 +06:00
CBasePlayer *player = (CBasePlayer *)(CBaseEntity*)m_pHolstered;
if( player->m_pActiveItem )
2016-03-02 14:22:22 +00:00
{
2016-03-23 16:11:21 +06:00
player->m_pActiveItem->Deploy();
CBasePlayerWeapon *weapon = (CBasePlayerWeapon *) player->m_pActiveItem->GetWeaponPtr();
2016-03-02 14:22:22 +00:00
if( weapon )
{
weapon->m_flNextPrimaryAttack = 0;
weapon->m_flNextSecondaryAttack = 0;
}
}
2016-03-23 16:11:21 +06:00
player ->m_iHideHUD &= ~HIDEHUD_WEAPONS;
2016-03-02 14:22:22 +00:00
m_pHolstered = NULL;
}
if( m_pfnThink == &CProp::DeployThink )
{
pev->nextthink = gpGlobals->time + m_flRespawnTime;
SetThink( &CProp::RespawnThink );
}
2016-02-24 21:26:16 +00:00
}
2016-08-17 12:20:18 +00:00
void CProp::BounceTouch(CBaseEntity *pOther)
2016-02-24 21:26:16 +00:00
{
2016-03-02 15:01:27 +00:00
if( pev->health <= 0 )
2016-03-27 03:56:54 +06:00
{
DieThink();
2016-03-02 15:01:27 +00:00
return;
2016-03-27 03:56:54 +06:00
}
2016-05-22 12:47:14 +00:00
pev->movetype = MOVETYPE_BOUNCE;
2016-03-02 14:22:22 +00:00
//ALERT( at_console, "BounceTouch: %f %f %f\n", pev->angles.x, pev->angles.y, pev->angles.z );
// only do damage if we're moving fairly fast
DeployThink();
2016-08-17 12:20:18 +00:00
if ( m_flNextAttack < gpGlobals->time && pev->velocity.Length() > 300)
2016-03-02 14:22:22 +00:00
{
2016-03-26 18:44:18 +00:00
if( !m_attacker )
m_attacker = this;
2016-03-23 16:11:21 +06:00
entvars_t *pevOwner = m_attacker->pev;
2016-03-02 14:22:22 +00:00
if (pevOwner)
{
float dmg = 50 + pev->velocity.Length() / 40;
2016-03-23 16:11:21 +06:00
if (pOther == m_owner2)
2016-03-02 14:22:22 +00:00
{
dmg = 5;
2016-08-17 12:20:18 +00:00
if (pOther->pev->button & (IN_USE))
2016-03-02 14:22:22 +00:00
{
dmg = 1;
}
}
TraceResult tr = UTIL_GetGlobalTrace();
ClearMultiDamage();
2016-08-17 12:20:18 +00:00
pOther->TraceAttack(pevOwner, dmg, gpGlobals->v_forward, &tr, DMG_CLUB);
ApplyMultiDamage(pev, pevOwner);
2016-03-02 14:22:22 +00:00
}
m_flNextAttack = gpGlobals->time + 1.0; // debounce
}
2016-03-23 19:08:13 +00:00
if( (pOther != m_owner2) && (pev->spawnflags & SF_PROP_BREAKABLE) && (pev->velocity.Length() > 1800) )
2016-03-02 14:22:22 +00:00
{
2016-03-02 15:01:27 +00:00
pev->nextthink = gpGlobals->time + 0.1;
SetThink( &CProp::DieThink );
2016-03-02 14:22:22 +00:00
}
pev->velocity = pev->velocity + pOther->pev->velocity;
2016-08-17 12:20:18 +00:00
float dp = cos(M_PI / 180 * UTIL_AngleDiff(UTIL_VecToAngles(pev->velocity).y, pev->angles.y));
if (pev->flags & FL_ONGROUND || fabs(pev->velocity.z) < 40)
2016-03-02 14:22:22 +00:00
{
CheckRotate();
2016-08-17 12:20:18 +00:00
if (m_shape == SHAPE_CYL_H)
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
pev->velocity.x *= fabs(dp) * 0.8 + 0.2;
pev->velocity.y *= fabs(dp) * 0.8 + 0.2;
2016-03-02 14:22:22 +00:00
pev->velocity.z -= 20;
pev->avelocity.x = -dp*pev->velocity.Length()* 1.5;
pev->avelocity.y = 0;
pev->avelocity.z = 0;
2016-08-17 12:20:18 +00:00
pev->angles.z += UTIL_AngleDiff(90, pev->angles.z) * 0.7;
2016-03-02 14:22:22 +00:00
//AngleThink();
}
2016-08-17 12:20:18 +00:00
else if (m_shape == SHAPE_CYL_V)
2016-03-02 14:22:22 +00:00
{
// pev->angles.z *= 0.3;
//pev->angles.x *= 0.3;
//AngleThink();
//CheckRotate();
pev->velocity.z *= m_flFloorFriction;
pev->velocity.x *= m_flFloorFriction;
pev->velocity.y *= m_flFloorFriction;
pev->velocity.z -= 10;
2016-08-17 12:20:18 +00:00
pev->avelocity.y = pev->avelocity.y*0.4 + RANDOM_FLOAT(30, -30);
2016-03-02 14:22:22 +00:00
}
else if( m_shape == SHAPE_SPHERE )
{
pev->velocity.z -= 20;
2016-08-17 12:20:18 +00:00
pev->avelocity.x = -cos(M_PI / 180 * UTIL_AngleDiff(UTIL_VecToAngles(pev->velocity).y, pev->angles.y))*pev->velocity.Length()* 1.5;
pev->avelocity.y = -sin(M_PI / 180 * UTIL_AngleDiff(UTIL_VecToAngles(pev->velocity).y, pev->angles.y))*pev->velocity.Length()* 1.5;;
2016-03-02 14:22:22 +00:00
pev->avelocity.z = 0;
}
else if( m_shape == SHAPE_BOX || m_shape == SHAPE_GENERIC )
{
pev->velocity.z *= m_flFloorFriction;
pev->velocity.x *= m_flFloorFriction;
pev->velocity.y *= m_flFloorFriction;
pev->velocity.z -= 10;
}
2016-08-17 12:20:18 +00:00
2016-03-02 14:22:22 +00:00
}
else
{
{
pev->velocity.z *= 0.3;
pev->velocity.y *= m_flCollideFriction;
pev->velocity.x *= m_flCollideFriction;
if( m_shape != SHAPE_SPHERE )
{
2016-08-17 12:20:18 +00:00
pev->avelocity.y = pev->avelocity.y*0.4 + RANDOM_FLOAT(100, -100);
pev->avelocity.x = pev->avelocity.x*0.5 + RANDOM_FLOAT(100, -100);
2016-03-02 14:22:22 +00:00
}
}
2016-08-17 12:20:18 +00:00
//pev->avelocity.z = pev->avelocity.z*0.5 + RANDOM_FLOAT ( 1, -1 );
2016-03-02 14:22:22 +00:00
BounceSound();
}
pev->framerate = pev->velocity.Length() / 200.0;
2016-08-17 12:20:18 +00:00
if (pev->framerate > 1.0)
2016-03-02 14:22:22 +00:00
pev->framerate = 1;
2016-08-17 12:20:18 +00:00
else if (pev->framerate < 0.2)
2016-03-02 14:22:22 +00:00
{
CheckRotate();
AngleThink();
2016-08-17 12:20:18 +00:00
if (pev->angles.z == 0 || pev->angles.z == 90)
2016-03-02 14:22:22 +00:00
pev->framerate = 0;
else
pev->framerate = 0.2;
}
2016-02-24 21:26:16 +00:00
}
2016-08-17 12:20:18 +00:00
void CProp::BounceSound(void)
2016-02-24 21:26:16 +00:00
{
2016-08-18 17:49:35 +00:00
MaterialSoundRandom( edict(), m_Material, 0.2 );
2016-02-24 21:26:16 +00:00
}
2016-08-17 12:20:18 +00:00
void CProp::Spawn(void)
2016-02-24 21:26:16 +00:00
{
2016-08-17 12:20:18 +00:00
2016-03-02 14:22:22 +00:00
Precache();
if( minsH == g_vecZero )
{
// default barrel parameters
2016-08-17 12:20:18 +00:00
minsV = Vector(-10, -10, -17);
maxsV = Vector(10, 10, 18);
minsH = Vector(-10, -10, -10);
maxsH = Vector(10, 10, 13);
2016-03-02 14:22:22 +00:00
}
m_flCollideFriction = 0.7;
m_flFloorFriction = 0.5;
2016-05-22 12:47:14 +00:00
if( !(pev->spawnflags & SF_PROP_FIXED ) )
{
UTIL_SetSize( pev, minsH, maxsH );
if( m_shape < 3 )
CheckRotate();
UTIL_SetOrigin( pev, pev->origin );
DROP_TO_FLOOR( edict() );
}
2016-03-02 14:22:22 +00:00
spawnOrigin = pev->origin;
spawnAngles = pev->angles;
m_flSpawnHealth = pev->health;
if( m_flSpawnHealth <= 0 )
m_flSpawnHealth = 30;
if( !m_flRespawnTime )
2016-06-04 15:23:46 +03:00
m_flRespawnTime = 80;
2016-03-02 14:22:22 +00:00
pev->dmg = 100;
PropRespawn();
2016-02-24 21:26:16 +00:00
}
void CProp::PropRespawn()
{
2016-08-17 12:20:18 +00:00
UTIL_SetSize(pev, Vector( 0, 0, 0), Vector(0, 0, 0) );
2016-03-03 20:31:01 +00:00
pev->effects &= ~EF_NODRAW;
2016-05-22 12:47:14 +00:00
if( pev->spawnflags & SF_PROP_FIXED )
pev->movetype = MOVETYPE_NONE;
else
pev->movetype = MOVETYPE_BOUNCE;
2016-03-02 14:22:22 +00:00
pev->solid = SOLID_SLIDEBOX;
pev->takedamage = DAMAGE_YES;
2016-03-03 20:31:01 +00:00
pev->health = m_flSpawnHealth;
2016-03-02 14:22:22 +00:00
pev->velocity = pev->avelocity = g_vecZero;
2016-03-03 20:31:01 +00:00
pev->angles = spawnAngles;
pev->deadflag = DEAD_NO;
2016-08-17 12:20:18 +00:00
SET_MODEL( ENT(pev), STRING(pev->model) );
m_oldshape = (PropShape)-1;
2016-03-02 14:22:22 +00:00
CheckRotate();
2016-03-03 20:31:01 +00:00
SetTouch( &CProp::BounceTouch );
SetUse( &CProp::Use );
2016-03-02 14:22:22 +00:00
pev->framerate = 1.0f;
UTIL_SetOrigin( pev, spawnOrigin );
2016-06-04 15:23:46 +03:00
m_owner2 = NULL;
m_attacker = NULL;
2016-02-24 21:26:16 +00:00
}
void CProp::RespawnThink()
{
2016-08-17 12:20:18 +00:00
if( !(pev->spawnflags & SF_PROP_RESPAWN))
2016-03-02 15:01:27 +00:00
{
if( pev->health <= 0 )
{
pev->nextthink = gpGlobals->time + 0.1;
SetThink( &CBaseEntity::SUB_Remove );
}
2016-03-02 14:22:22 +00:00
return;
2016-03-02 15:01:27 +00:00
}
2016-03-02 14:22:22 +00:00
PropRespawn();
2016-02-24 21:26:16 +00:00
}
2016-08-17 12:20:18 +00:00
2016-02-24 21:26:16 +00:00
void CProp::AngleThink()
{
2016-03-02 14:22:22 +00:00
pev->nextthink = gpGlobals->time + m_flRespawnTime;
SetThink( &CProp::RespawnThink);
2016-08-17 12:20:18 +00:00
if (!(pev->flags & FL_ONGROUND || fabs(pev->velocity.z) < 40))
2016-03-02 14:22:22 +00:00
{
m_owner2 = m_attacker = 0;
return;
}
2016-08-17 12:20:18 +00:00
if (m_shape == SHAPE_CYL_H)
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
pev->angles.z += UTIL_AngleDiff(90, pev->angles.z) * 0.7;
if (fabs(UTIL_AngleDiff(90, pev->angles.z)) > 0.1)
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
SetThink( &CProp::AngleThink);
2016-03-02 14:22:22 +00:00
pev->nextthink = gpGlobals->time + 0.1;
}
//ALERT( at_console, "AngleThink: %f %f %f\n", pev->angles.x, pev->angles.y, pev->angles.z );
pev->avelocity.y = pev->avelocity.z = 0;
}
2016-08-17 12:20:18 +00:00
else if (m_shape == SHAPE_CYL_V)
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
if (fabs(UTIL_AngleDiff(90, pev->angles.z)) > 0.1)
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
SetThink( &CProp::AngleThink);
2016-03-02 14:22:22 +00:00
pev->nextthink = gpGlobals->time + 0.1;
}
2016-08-17 12:20:18 +00:00
pev->angles.z += UTIL_AngleDiff(0, pev->angles.z) * 0.7;
2016-03-02 14:22:22 +00:00
//pev->angles.x += UTIL_AngleDiff( 0, pev->angles.x ) * 0.3;
pev->avelocity.x = pev->avelocity.y = pev->avelocity.z = 0;
}
2016-08-17 12:20:18 +00:00
else if (m_shape == SHAPE_BOX)
2016-03-02 14:22:22 +00:00
{
Vector iangles;
iangles.x = myround( pev->angles.x / 90 ) * 90;
iangles.y = myround( pev->angles.y / 90 ) * 90;
iangles.z = myround( pev->angles.z / 90 ) * 90;
2016-03-02 14:22:22 +00:00
if (fabs(UTIL_AngleDiff(iangles.x, pev->angles.x)) > 0.1 ||
//fabs(UTIL_AngleDiff(iangles.y, pev->angles.y)) > 0.1 ||
fabs(UTIL_AngleDiff(iangles.z, pev->angles.z)) > 0.1)
{
2016-08-17 12:20:18 +00:00
SetThink( &CProp::AngleThink);
2016-03-02 14:22:22 +00:00
pev->nextthink = gpGlobals->time + 0.1;
}
2016-08-17 12:20:18 +00:00
pev->angles.x += UTIL_AngleDiff(iangles.x, pev->angles.x) * 0.6;
//pev->angles.y += UTIL_AngleDiff(iangles.y, pev->angles.y) * 0.6;
pev->angles.z += UTIL_AngleDiff(iangles.z, pev->angles.z) * 0.6;
2016-03-02 14:22:22 +00:00
pev->avelocity.x = pev->avelocity.y = pev->avelocity.z = 0;
}
2016-08-17 12:20:18 +00:00
else if (m_shape == SHAPE_NOROTATE)
2016-03-02 14:22:22 +00:00
{
pev->avelocity.x = pev->avelocity.y = pev->avelocity.z = 0;
Vector iangles = spawnAngles;
2016-08-17 12:20:18 +00:00
if (fabs(UTIL_AngleDiff(iangles.x, pev->angles.x)) > 0.1 ||
fabs(UTIL_AngleDiff(iangles.y, pev->angles.y)) > 0.1 ||
fabs(UTIL_AngleDiff(iangles.z, pev->angles.z)) > 0.1)
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
SetThink( &CProp::AngleThink);
2016-03-02 14:22:22 +00:00
pev->nextthink = gpGlobals->time + 0.1;
}
2016-08-17 12:20:18 +00:00
pev->angles.x += UTIL_AngleDiff(iangles.x, pev->angles.x) * 0.6;
pev->angles.y += UTIL_AngleDiff(iangles.y, pev->angles.y) * 0.6;
pev->angles.z += UTIL_AngleDiff(iangles.z, pev->angles.z) * 0.6;
2016-03-02 14:22:22 +00:00
}
2016-08-17 12:20:18 +00:00
else if (m_shape == SHAPE_GENERIC)
2016-03-02 14:22:22 +00:00
{
float ianglex = 0, ianglez = 0, imaxanglediff=360.0f;
int i;
2016-03-02 14:22:22 +00:00
// if first number is zero, it is angle
// all other zeroes is array end
for( i = 0; (i < 10) && ( (i == 0 ) || m_iaCustomAnglesX[i] ); i++)
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
float anglediff = fabs(UTIL_AngleDiff(pev->angles.x, m_iaCustomAnglesX[i]));
2016-03-02 14:22:22 +00:00
if( imaxanglediff > anglediff )
{
ianglex = m_iaCustomAnglesX[i];
imaxanglediff = anglediff;
}
}
imaxanglediff=360.0f;
for( i = 0; (i < 10) && ( (i == 0 ) || m_iaCustomAnglesZ[i] ); i++)
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
float anglediff = fabs(UTIL_AngleDiff(pev->angles.z, m_iaCustomAnglesZ[i]));
2016-03-02 14:22:22 +00:00
if( imaxanglediff > anglediff )
{
ianglez = m_iaCustomAnglesZ[i];
imaxanglediff = anglediff;
}
}
2016-08-17 12:20:18 +00:00
if (fabs(UTIL_AngleDiff(ianglex, pev->angles.x)) > 0.1 ||
fabs(UTIL_AngleDiff(ianglez, pev->angles.z)) > 0.1 )
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
SetThink( &CProp::AngleThink);
2016-03-02 14:22:22 +00:00
pev->nextthink = gpGlobals->time + 0.1;
}
2016-08-17 12:20:18 +00:00
pev->angles.x += UTIL_AngleDiff(ianglex, pev->angles.x) * 0.6;
pev->angles.z += UTIL_AngleDiff(ianglez, pev->angles.z) * 0.6;
2016-03-02 14:22:22 +00:00
pev->avelocity.x = pev->avelocity.y = pev->avelocity.z = 0;
}
2016-08-17 12:20:18 +00:00
pev->angles.x = UTIL_AngleMod(pev->angles.x);
pev->angles.y = UTIL_AngleMod(pev->angles.y);
pev->angles.z = UTIL_AngleMod(pev->angles.z);
2016-02-24 21:26:16 +00:00
}
2016-03-02 15:01:27 +00:00
void CProp::DieThink()
{
2016-08-17 12:20:18 +00:00
Killed( VARS(m_attacker), GIB_NORMAL );
2016-03-02 15:01:27 +00:00
Die();
}
2016-02-24 21:26:16 +00:00
2016-08-17 12:20:18 +00:00
int CProp::TakeDamage(entvars_t* pevInflictor, entvars_t* pevAttacker, float flDamage, int bitsDamageType)
2016-02-24 21:26:16 +00:00
{
2016-03-02 14:22:22 +00:00
Vector r = (pevInflictor->origin - pev->origin);
2016-03-06 20:13:35 +00:00
if ( (!m_attacker
2016-07-17 20:11:00 +00:00
|| (pev->velocity.Length() < 700)) && ((CBaseEntity*)GET_PRIVATE(ENT(pevAttacker)))
2016-03-06 20:13:35 +00:00
&& ((CBaseEntity*)GET_PRIVATE(ENT(pevAttacker)))->IsPlayer())
2016-03-23 16:11:21 +06:00
m_attacker.Set(ENT(pevAttacker));
2016-03-02 14:22:22 +00:00
DeployThink();
pev->velocity = r * flDamage / -7;
2016-08-17 12:20:18 +00:00
pev->avelocity.x = pev->avelocity.x*0.5 + RANDOM_FLOAT(100, -100);
ALERT(at_console, "Takedmg: %s %s %f %f\n", STRING(pevInflictor->classname), STRING(pevAttacker->classname), flDamage, pev->health );
2016-03-02 14:22:22 +00:00
// now some func_breakable code
2016-08-17 12:20:18 +00:00
if ( !(pev->spawnflags & SF_PROP_BREAKABLE ) )
2016-03-02 14:22:22 +00:00
return 0;
2016-03-02 15:01:27 +00:00
if ( pev->health <= 0 )
2016-03-21 16:07:17 +06:00
return 0;
2016-03-02 14:22:22 +00:00
// Breakables take double damage from the crowbar
2016-08-17 12:20:18 +00:00
if ( bitsDamageType & DMG_CLUB )
2016-03-02 14:22:22 +00:00
flDamage *= 2;
// Boxes / glass / etc. don't take much poison damage, just the impact of the dart - consider that 10%
2016-08-17 12:20:18 +00:00
if ( bitsDamageType & DMG_POISON )
2016-03-02 14:22:22 +00:00
flDamage *= 0.1;
g_vecAttackDir = r.Normalize();
// do the damage
pev->health -= flDamage;
2016-08-17 12:20:18 +00:00
if ( pev->health <= 0 )
2016-03-02 14:22:22 +00:00
{
2016-03-02 15:01:27 +00:00
// delayed explode
SetThink( &CProp::DieThink );
pev->nextthink = gpGlobals->time + 0.2;
2016-03-02 14:22:22 +00:00
return 0;
}
// Make a shard noise each time func breakable is hit.
// Don't play shard noise if cbreakable actually died.
2016-08-17 12:20:18 +00:00
2016-03-02 14:22:22 +00:00
DamageSound();
return 1;
2016-02-24 21:26:16 +00:00
}
void CProp::KeyValue( KeyValueData* pkvd )
{
2016-08-17 12:20:18 +00:00
ALERT( at_console, "%s %s\n", pkvd->szKeyName, pkvd->szValue);
2016-03-02 14:22:22 +00:00
// UNDONE_WC: explicitly ignoring these fields, but they shouldn't be in the map file!
2016-08-17 12:20:18 +00:00
if (FStrEq(pkvd->szKeyName, "explosion"))
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
if (!stricmp(pkvd->szValue, "directed"))
2016-03-02 14:22:22 +00:00
m_Explosion = expDirected;
2016-08-17 12:20:18 +00:00
else if (!stricmp(pkvd->szValue, "random"))
2016-03-02 14:22:22 +00:00
m_Explosion = expRandom;
else
m_Explosion = expRandom;
pkvd->fHandled = TRUE;
}
2016-08-17 12:20:18 +00:00
else if (FStrEq(pkvd->szKeyName, "material"))
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
int i = atoi( pkvd->szValue);
2016-03-02 14:22:22 +00:00
// 0:glass, 1:metal, 2:flesh, 3:wood
2016-08-17 12:20:18 +00:00
if ((i < 0) || (i >= matLastMaterial))
2016-03-02 14:22:22 +00:00
m_Material = matWood;
else
m_Material = (Materials)i;
pkvd->fHandled = TRUE;
}
2016-08-17 12:20:18 +00:00
else if (FStrEq(pkvd->szKeyName, "shape"))
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
int i = atoi( pkvd->szValue);
2016-03-02 14:22:22 +00:00
2016-08-17 12:20:18 +00:00
if ((i < 0) || (i >= SHAPE_NOROTATE))
2016-03-02 14:22:22 +00:00
m_shape = SHAPE_NOROTATE;
else
m_shape = (PropShape)i;
pkvd->fHandled = TRUE;
}
2016-08-17 12:20:18 +00:00
else if (FStrEq(pkvd->szKeyName, "gibmodel") )
2016-03-02 14:22:22 +00:00
{
2016-08-17 12:20:18 +00:00
m_iszGibModel = ALLOC_STRING(pkvd->szValue);
2016-03-02 14:22:22 +00:00
pkvd->fHandled = TRUE;
}
2016-08-17 12:20:18 +00:00
else if (FStrEq(pkvd->szKeyName, "explodemagnitude") )
2016-03-02 14:22:22 +00:00
{
ExplosionSetMagnitude( atoi( pkvd->szValue ) );
pkvd->fHandled = TRUE;
}
2016-08-17 12:20:18 +00:00
else if (FStrEq(pkvd->szKeyName, "respawntime") )
2016-03-02 14:22:22 +00:00
{
m_flRespawnTime = atof( pkvd->szValue );
pkvd->fHandled = TRUE;
}
2016-08-17 12:20:18 +00:00
else if (FStrEq(pkvd->szKeyName, "customanglesx"))
2016-03-02 14:22:22 +00:00
{
UTIL_StringToIntArray( m_iaCustomAnglesX, ARRAYSIZE( m_iaCustomAnglesX ), pkvd->szValue );
pkvd->fHandled = TRUE;
}
2016-08-17 12:20:18 +00:00
else if (FStrEq(pkvd->szKeyName, "customanglesz"))
2016-03-02 14:22:22 +00:00
{
UTIL_StringToIntArray( m_iaCustomAnglesZ, ARRAYSIZE( m_iaCustomAnglesZ ), pkvd->szValue );
pkvd->fHandled = TRUE;
}
2016-08-17 12:20:18 +00:00
else if (FStrEq(pkvd->szKeyName, "hmin"))
2016-03-02 14:22:22 +00:00
{
UTIL_StringToVector( minsH, pkvd->szValue );
pkvd->fHandled = TRUE;
}
2016-08-17 12:20:18 +00:00
else if (FStrEq(pkvd->szKeyName, "hmax"))
2016-03-02 14:22:22 +00:00
{
UTIL_StringToVector( maxsH, pkvd->szValue );
pkvd->fHandled = TRUE;
}
2016-08-17 12:20:18 +00:00
else if (FStrEq(pkvd->szKeyName, "vmin"))
2016-03-02 14:22:22 +00:00
{
UTIL_StringToVector( minsV, pkvd->szValue );
pkvd->fHandled = TRUE;
}
2016-08-17 12:20:18 +00:00
else if (FStrEq(pkvd->szKeyName, "vmax"))
2016-03-02 14:22:22 +00:00
{
UTIL_StringToVector( maxsV, pkvd->szValue );
pkvd->fHandled = TRUE;
}
else
CBaseEntity::KeyValue( pkvd );
2016-02-24 21:26:16 +00:00
}
2016-05-22 23:59:11 +00:00
#include <stdio.h>
void DumpProps()
{
edict_t *pEdict = g_engfuncs.pfnPEntityOfEntIndex( 1 );
FILE *f = fopen( "props.ent", "wb" );
if( !f )
return;
for ( int j = 1; j < gpGlobals->maxEntities; j++, pEdict++ )
{
if ( pEdict->free ) // Not in use
continue;
if ( strcmp( STRING( pEdict->v.classname ), "prop" ) )
continue;
CProp *prop = (CProp *)CBaseEntity::Instance(pEdict);
fprintf( f, "{\n");
fprintf( f, "\"classname\" \"prop\"\n");
if( pEdict->v.model )
fprintf( f, "\"model\" \"%s\"\n", STRING( pEdict->v.model ));
fprintf( f, "\"spawnflags\" \"%d\"\n", pEdict->v.spawnflags );
fprintf( f, "\"angles\" \"%d %d %d\"\n", (int)pEdict->v.angles.x, (int)pEdict->v.angles.y, (int)pEdict->v.angles.z );
fprintf( f, "\"origin\" \"%d %d %d\"\n", (int)pEdict->v.origin.x, (int)pEdict->v.origin.y, (int)pEdict->v.origin.z );
if( prop->m_shape != 1 )
fprintf( f, "\"shape\" \"%d\"\n", (int)(((int)prop->m_shape) > 1? (int)(prop->m_shape): (int)(!(int)prop->m_shape)) );
fprintf( f, "\"hmin\" \"%d %d %d\"\n", (int)prop->minsH.x, (int)prop->minsH.y, (int)prop->minsH.z );
fprintf( f, "\"hmax\" \"%d %d %d\"\n", (int)prop->maxsH.x, (int)prop->maxsH.y, (int)prop->maxsH.z );
if( prop->m_shape > 1 )
{
fprintf( f, "\"vmin\" \"%d %d %d\"\n", (int)prop->minsV.x, (int)prop->minsV.y, (int)prop->minsV.z );
fprintf( f, "\"vmax\" \"%d %d %d\"\n", (int)prop->maxsV.x, (int)prop->maxsV.y, (int)prop->maxsV.z );
}
if( prop->m_Material )
fprintf( f, "\"material\" \"%d\"\n", prop->m_Material );
if( pEdict->v.health != 30 )
fprintf( f, "\"health\" \"%d\"\n", (int)pEdict->v.health );
if( prop->m_shape == SHAPE_GENERIC)
{
int i;
if( !( !prop->m_iaCustomAnglesX[0] && !prop->m_iaCustomAnglesX[1] ) )
{
fprintf( f, "\"customanglesx\" \"%d", prop->m_iaCustomAnglesX[0] );
for( i = 1; (i < 10) && ( prop->m_iaCustomAnglesX[i] ); i++ )
fprintf( f, " %d", prop->m_iaCustomAnglesX[i] );
fprintf( f, "\"\n" );
}
if( !( !prop->m_iaCustomAnglesZ[0] && !prop->m_iaCustomAnglesZ[1] ) )
{
fprintf( f, "\"customanglesz\" \"%d", prop->m_iaCustomAnglesZ[0] );
for( i = 1; (i < 10) && ( prop->m_iaCustomAnglesZ[i] ); i++ )
fprintf( f, " %d", prop->m_iaCustomAnglesZ[i] );
fprintf( f, "\"\n" );
}
}
if( prop->m_Explosion == expDirected )
fprintf( f, "\"explosion\" \"directed\"\n" );
if( prop->m_iszGibModel )
fprintf( f, "\"gibmodel\" \"%s\"\n", STRING( prop->m_iszGibModel ) );
if( pEdict->v.impulse )
fprintf( f, "\"impulse\" \"%d\"\n", pEdict->v.impulse );
fprintf( f, "}\n" );
}
}