553 lines
12 KiB
C++
Raw Normal View History

2017-12-18 02:39:44 +03:00
/***
*
* Copyright (c) 1996-2002, 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.
*
****/
#include "extdll.h"
#include "util.h"
#include "cbase.h"
#include "animation.h"
#include "effects.h"
#define XEN_PLANT_GLOW_SPRITE "sprites/flare3.spr"
#define XEN_PLANT_HIDE_TIME 5
class CActAnimating : public CBaseAnimating
{
public:
2016-07-31 18:48:50 +05:00
void SetActivity( Activity act );
2017-12-18 02:39:44 +03:00
inline Activity GetActivity( void ) { return m_Activity; }
2016-07-31 18:48:50 +05:00
virtual int ObjectCaps( void ) { return CBaseAnimating::ObjectCaps() & ~FCAP_ACROSS_TRANSITION; }
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
virtual int Save( CSave &save );
virtual int Restore( CRestore &restore );
static TYPEDESCRIPTION m_SaveData[];
2017-12-18 02:39:44 +03:00
private:
2016-07-31 18:48:50 +05:00
Activity m_Activity;
2017-12-18 02:39:44 +03:00
};
TYPEDESCRIPTION CActAnimating::m_SaveData[] =
2017-12-18 02:39:44 +03:00
{
DEFINE_FIELD( CActAnimating, m_Activity, FIELD_INTEGER ),
};
IMPLEMENT_SAVERESTORE( CActAnimating, CBaseAnimating )
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
void CActAnimating::SetActivity( Activity act )
{
2017-12-18 02:39:44 +03:00
int sequence = LookupActivity( act );
2016-07-31 18:48:50 +05:00
if( sequence != ACTIVITY_NOT_AVAILABLE )
2017-12-18 02:39:44 +03:00
{
pev->sequence = sequence;
m_Activity = act;
pev->frame = 0;
2016-07-31 18:48:50 +05:00
ResetSequenceInfo();
2017-12-18 02:39:44 +03:00
}
}
class CXenPLight : public CActAnimating
{
public:
2016-07-31 18:48:50 +05:00
void Spawn( void );
void Precache( void );
void Touch( CBaseEntity *pOther );
void Think( void );
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
void LightOn( void );
void LightOff( void );
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
virtual int Save( CSave &save );
virtual int Restore( CRestore &restore );
static TYPEDESCRIPTION m_SaveData[];
2017-12-18 02:39:44 +03:00
private:
2016-07-31 18:48:50 +05:00
CSprite *m_pGlow;
2017-12-18 02:39:44 +03:00
};
LINK_ENTITY_TO_CLASS( xen_plantlight, CXenPLight )
2017-12-18 02:39:44 +03:00
TYPEDESCRIPTION CXenPLight::m_SaveData[] =
2017-12-18 02:39:44 +03:00
{
DEFINE_FIELD( CXenPLight, m_pGlow, FIELD_CLASSPTR ),
};
IMPLEMENT_SAVERESTORE( CXenPLight, CActAnimating )
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
void CXenPLight::Spawn( void )
2017-12-18 02:39:44 +03:00
{
Precache();
2016-07-31 18:48:50 +05:00
SET_MODEL( ENT( pev ), "models/light.mdl" );
pev->movetype = MOVETYPE_NONE;
pev->solid = SOLID_TRIGGER;
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
UTIL_SetSize( pev, Vector( -80, -80, 0 ), Vector( 80, 80, 32 ) );
2017-12-18 02:39:44 +03:00
SetActivity( ACT_IDLE );
SetNextThink( 0.1 );
2016-07-31 18:48:50 +05:00
pev->frame = RANDOM_FLOAT( 0, 255 );
2017-12-18 02:39:44 +03:00
m_pGlow = CSprite::SpriteCreate( XEN_PLANT_GLOW_SPRITE, pev->origin + Vector(0,0,(pev->mins.z+pev->maxs.z)*0.5), FALSE );
2017-06-29 18:56:03 +05:00
m_pGlow->SetTransparency( kRenderGlow, (int)pev->rendercolor.x, (int)pev->rendercolor.y, (int)pev->rendercolor.z, (int)pev->renderamt, (int)pev->renderfx );
2017-12-18 02:39:44 +03:00
m_pGlow->SetAttachment( edict(), 1 );
}
2016-07-31 18:48:50 +05:00
void CXenPLight::Precache( void )
2017-12-18 02:39:44 +03:00
{
PRECACHE_MODEL( "models/light.mdl" );
PRECACHE_MODEL( XEN_PLANT_GLOW_SPRITE );
}
2016-07-31 18:48:50 +05:00
void CXenPLight::Think( void )
2017-12-18 02:39:44 +03:00
{
StudioFrameAdvance();
SetNextThink( 0.1 );
switch( GetActivity() )
{
case ACT_CROUCH:
2016-07-31 18:48:50 +05:00
if( m_fSequenceFinished )
2017-12-18 02:39:44 +03:00
{
SetActivity( ACT_CROUCHIDLE );
LightOff();
}
break;
case ACT_CROUCHIDLE:
2016-07-31 18:48:50 +05:00
if( gpGlobals->time > pev->dmgtime )
2017-12-18 02:39:44 +03:00
{
SetActivity( ACT_STAND );
LightOn();
}
break;
case ACT_STAND:
2016-07-31 18:48:50 +05:00
if( m_fSequenceFinished )
2017-12-18 02:39:44 +03:00
SetActivity( ACT_IDLE );
break;
case ACT_IDLE:
default:
break;
}
}
2016-07-31 18:48:50 +05:00
void CXenPLight::Touch( CBaseEntity *pOther )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
if( pOther->IsPlayer() )
2017-12-18 02:39:44 +03:00
{
pev->dmgtime = gpGlobals->time + XEN_PLANT_HIDE_TIME;
2016-07-31 18:48:50 +05:00
if( GetActivity() == ACT_IDLE || GetActivity() == ACT_STAND )
2017-12-18 02:39:44 +03:00
{
SetActivity( ACT_CROUCH );
}
}
}
2016-07-31 18:48:50 +05:00
void CXenPLight::LightOn( void )
2017-12-18 02:39:44 +03:00
{
SUB_UseTargets( this, USE_ON, 0 );
2016-07-31 18:48:50 +05:00
if( m_pGlow )
2017-12-18 02:39:44 +03:00
m_pGlow->pev->effects &= ~EF_NODRAW;
}
2016-07-31 18:48:50 +05:00
void CXenPLight::LightOff( void )
2017-12-18 02:39:44 +03:00
{
SUB_UseTargets( this, USE_OFF, 0 );
2016-07-31 18:48:50 +05:00
if( m_pGlow )
2017-12-18 02:39:44 +03:00
m_pGlow->pev->effects |= EF_NODRAW;
}
class CXenHair : public CActAnimating
{
public:
2016-07-31 18:48:50 +05:00
void Spawn( void );
void Precache( void );
void Think( void );
2017-12-18 02:39:44 +03:00
};
LINK_ENTITY_TO_CLASS( xen_hair, CXenHair )
2017-12-18 02:39:44 +03:00
#define SF_HAIR_SYNC 0x0001
void CXenHair::Spawn( void )
{
Precache();
SET_MODEL( edict(), "models/hair.mdl" );
2016-07-31 18:48:50 +05:00
UTIL_SetSize( pev, Vector( -4, -4, 0 ), Vector( 4, 4, 32 ) );
2017-12-18 02:39:44 +03:00
pev->sequence = 0;
2016-07-31 18:48:50 +05:00
if( !( pev->spawnflags & SF_HAIR_SYNC ) )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
pev->frame = RANDOM_FLOAT( 0, 255 );
2017-12-18 02:39:44 +03:00
pev->framerate = RANDOM_FLOAT( 0.7, 1.4 );
}
2016-07-31 18:48:50 +05:00
ResetSequenceInfo();
2017-12-18 02:39:44 +03:00
pev->solid = SOLID_NOT;
pev->movetype = MOVETYPE_NONE;
SetNextThink( RANDOM_FLOAT( 0.1, 0.4 ) ); // Load balance these a bit
}
void CXenHair::Think( void )
{
StudioFrameAdvance();
SetNextThink( 0.5 );
}
void CXenHair::Precache( void )
{
PRECACHE_MODEL( "models/hair.mdl" );
}
class CXenTreeTrigger : public CBaseEntity
{
public:
2016-07-31 18:48:50 +05:00
void Touch( CBaseEntity *pOther );
2017-12-18 02:39:44 +03:00
static CXenTreeTrigger *TriggerCreate( edict_t *pOwner, const Vector &position );
};
LINK_ENTITY_TO_CLASS( xen_ttrigger, CXenTreeTrigger )
2016-06-04 18:24:23 +05:00
2016-07-31 18:48:50 +05:00
CXenTreeTrigger *CXenTreeTrigger::TriggerCreate( edict_t *pOwner, const Vector &position )
2017-12-18 02:39:44 +03:00
{
CXenTreeTrigger *pTrigger = GetClassPtr( (CXenTreeTrigger *)NULL );
pTrigger->pev->origin = position;
2016-07-31 18:48:50 +05:00
pTrigger->pev->classname = MAKE_STRING( "xen_ttrigger" );
2017-12-18 02:39:44 +03:00
pTrigger->pev->solid = SOLID_TRIGGER;
pTrigger->pev->movetype = MOVETYPE_NONE;
pTrigger->pev->owner = pOwner;
return pTrigger;
}
void CXenTreeTrigger::Touch( CBaseEntity *pOther )
{
2016-07-31 18:48:50 +05:00
if( pev->owner )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
CBaseEntity *pEntity = CBaseEntity::Instance( pev->owner );
2017-12-18 02:39:44 +03:00
pEntity->Touch( pOther );
}
}
#define TREE_AE_ATTACK 1
class CXenTree : public CActAnimating
{
public:
2016-07-31 18:48:50 +05:00
void Spawn( void );
void Precache( void );
void Touch( CBaseEntity *pOther );
void Think( void );
int TakeDamage( entvars_t *pevInflictor, entvars_t *pevAttacker, float flDamage, int bitsDamageType ) { Attack(); return 0; }
void HandleAnimEvent( MonsterEvent_t *pEvent );
void Attack( void );
int Classify( void ) { return CLASS_BARNACLE; }
virtual int Save( CSave &save );
virtual int Restore( CRestore &restore );
static TYPEDESCRIPTION m_SaveData[];
2017-12-18 02:39:44 +03:00
static const char *pAttackHitSounds[];
static const char *pAttackMissSounds[];
private:
CXenTreeTrigger *m_pTrigger;
};
LINK_ENTITY_TO_CLASS( xen_tree, CXenTree )
2017-12-18 02:39:44 +03:00
TYPEDESCRIPTION CXenTree::m_SaveData[] =
2017-12-18 02:39:44 +03:00
{
DEFINE_FIELD( CXenTree, m_pTrigger, FIELD_CLASSPTR ),
};
IMPLEMENT_SAVERESTORE( CXenTree, CActAnimating )
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
void CXenTree::Spawn( void )
2017-12-18 02:39:44 +03:00
{
Precache();
2016-07-31 18:48:50 +05:00
SET_MODEL( ENT( pev ), "models/tree.mdl" );
pev->movetype = MOVETYPE_NONE;
pev->solid = SOLID_BBOX;
2017-12-18 02:39:44 +03:00
pev->takedamage = DAMAGE_YES;
2016-07-31 18:48:50 +05:00
UTIL_SetSize( pev, Vector( -30, -30, 0 ), Vector( 30, 30, 188 ) );
2017-12-18 02:39:44 +03:00
SetActivity( ACT_IDLE );
SetNextThink( 0.1 );
2016-07-31 18:48:50 +05:00
pev->frame = RANDOM_FLOAT( 0, 255 );
2017-12-18 02:39:44 +03:00
pev->framerate = RANDOM_FLOAT( 0.7, 1.4 );
Vector triggerPosition;
UTIL_MakeVectorsPrivate( pev->angles, triggerPosition, NULL, NULL );
2016-07-31 18:48:50 +05:00
triggerPosition = pev->origin + ( triggerPosition * 64 );
2017-12-18 02:39:44 +03:00
// Create the trigger
m_pTrigger = CXenTreeTrigger::TriggerCreate( edict(), triggerPosition );
UTIL_SetSize( m_pTrigger->pev, Vector( -24, -24, 0 ), Vector( 24, 24, 128 ) );
}
const char *CXenTree::pAttackHitSounds[] =
2017-12-18 02:39:44 +03:00
{
"zombie/claw_strike1.wav",
"zombie/claw_strike2.wav",
"zombie/claw_strike3.wav",
};
const char *CXenTree::pAttackMissSounds[] =
2017-12-18 02:39:44 +03:00
{
"zombie/claw_miss1.wav",
"zombie/claw_miss2.wav",
};
2016-07-31 18:48:50 +05:00
void CXenTree::Precache( void )
2017-12-18 02:39:44 +03:00
{
PRECACHE_MODEL( "models/tree.mdl" );
PRECACHE_MODEL( XEN_PLANT_GLOW_SPRITE );
PRECACHE_SOUND_ARRAY( pAttackHitSounds );
PRECACHE_SOUND_ARRAY( pAttackMissSounds );
}
2016-07-31 18:48:50 +05:00
void CXenTree::Touch( CBaseEntity *pOther )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
if( !pOther->IsPlayer() && FClassnameIs( pOther->pev, "monster_bigmomma" ) )
2017-12-18 02:39:44 +03:00
return;
Attack();
}
2016-07-31 18:48:50 +05:00
void CXenTree::Attack( void )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
if( GetActivity() == ACT_IDLE )
2017-12-18 02:39:44 +03:00
{
SetActivity( ACT_MELEE_ATTACK1 );
pev->framerate = RANDOM_FLOAT( 1.0, 1.4 );
EMIT_SOUND_ARRAY_DYN( CHAN_WEAPON, pAttackMissSounds );
}
}
2016-07-31 18:48:50 +05:00
void CXenTree::HandleAnimEvent( MonsterEvent_t *pEvent )
2017-12-18 02:39:44 +03:00
{
switch( pEvent->event )
{
case TREE_AE_ATTACK:
{
CBaseEntity *pList[8];
BOOL sound = FALSE;
2016-07-31 18:48:50 +05:00
int count = UTIL_EntitiesInBox( pList, 8, m_pTrigger->pev->absmin, m_pTrigger->pev->absmax, FL_MONSTER | FL_CLIENT );
2017-12-18 02:39:44 +03:00
Vector forward;
UTIL_MakeVectorsPrivate( pev->angles, forward, NULL, NULL );
2016-07-31 18:48:50 +05:00
for( int i = 0; i < count; i++ )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
if( pList[i] != this )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
if( pList[i]->pev->owner != edict() )
2017-12-18 02:39:44 +03:00
{
sound = TRUE;
pList[i]->TakeDamage( pev, pev, 25, DMG_CRUSH | DMG_SLASH );
pList[i]->pev->punchangle.x = 15;
pList[i]->pev->velocity = pList[i]->pev->velocity + forward * 100;
}
}
}
2016-07-31 18:48:50 +05:00
if( sound )
2017-12-18 02:39:44 +03:00
{
EMIT_SOUND_ARRAY_DYN( CHAN_WEAPON, pAttackHitSounds );
}
}
return;
}
CActAnimating::HandleAnimEvent( pEvent );
}
2016-07-31 18:48:50 +05:00
void CXenTree::Think( void )
2017-12-18 02:39:44 +03:00
{
float flInterval = StudioFrameAdvance();
SetNextThink( 0.1 );
DispatchAnimEvents( flInterval );
switch( GetActivity() )
{
case ACT_MELEE_ATTACK1:
2016-07-31 18:48:50 +05:00
if( m_fSequenceFinished )
2017-12-18 02:39:44 +03:00
{
SetActivity( ACT_IDLE );
pev->framerate = RANDOM_FLOAT( 0.6, 1.4 );
}
break;
default:
case ACT_IDLE:
break;
}
}
// UNDONE: These need to smoke somehow when they take damage
// Touch behavior?
// Cause damage in smoke area
//
// Spores
//
class CXenSpore : public CActAnimating
{
public:
2016-07-31 18:48:50 +05:00
void Spawn( void );
void Precache( void );
void Touch( CBaseEntity *pOther );
void Think( void );
int TakeDamage( entvars_t *pevInflictor, entvars_t *pevAttacker, float flDamage, int bitsDamageType ) { Attack(); return 0; }
//void HandleAnimEvent( MonsterEvent_t *pEvent );
void Attack( void ) {}
2017-12-18 02:39:44 +03:00
static const char *pModelNames[];
};
class CXenSporeSmall : public CXenSpore
{
2016-07-31 18:48:50 +05:00
void Spawn( void );
2017-12-18 02:39:44 +03:00
};
class CXenSporeMed : public CXenSpore
{
2016-07-31 18:48:50 +05:00
void Spawn( void );
2017-12-18 02:39:44 +03:00
};
class CXenSporeLarge : public CXenSpore
{
2016-07-31 18:48:50 +05:00
void Spawn( void );
2017-12-18 02:39:44 +03:00
static const Vector m_hullSizes[];
};
// Fake collision box for big spores
class CXenHull : public CPointEntity
{
public:
static CXenHull *CreateHull( CBaseEntity *source, const Vector &mins, const Vector &maxs, const Vector &offset );
2016-07-31 18:48:50 +05:00
int Classify( void ) { return CLASS_BARNACLE; }
2017-12-18 02:39:44 +03:00
};
2016-07-31 18:48:50 +05:00
CXenHull *CXenHull::CreateHull( CBaseEntity *source, const Vector &mins, const Vector &maxs, const Vector &offset )
2017-12-18 02:39:44 +03:00
{
CXenHull *pHull = GetClassPtr( (CXenHull *)NULL );
UTIL_SetOrigin( pHull, source->pev->origin + offset );
2017-07-24 02:24:55 +05:00
SET_MODEL( pHull->edict(), STRING( source->pev->model ) );
2017-12-18 02:39:44 +03:00
pHull->pev->solid = SOLID_BBOX;
2017-07-24 02:24:55 +05:00
pHull->pev->classname = MAKE_STRING( "xen_hull" );
2017-12-18 02:39:44 +03:00
pHull->pev->movetype = MOVETYPE_NONE;
pHull->pev->owner = source->edict();
UTIL_SetSize( pHull->pev, mins, maxs );
pHull->pev->renderamt = 0;
pHull->pev->rendermode = kRenderTransTexture;
// pHull->pev->effects = EF_NODRAW;
return pHull;
}
LINK_ENTITY_TO_CLASS( xen_spore_small, CXenSporeSmall )
LINK_ENTITY_TO_CLASS( xen_spore_medium, CXenSporeMed )
LINK_ENTITY_TO_CLASS( xen_spore_large, CXenSporeLarge )
LINK_ENTITY_TO_CLASS( xen_hull, CXenHull )
2017-12-18 02:39:44 +03:00
void CXenSporeSmall::Spawn( void )
{
pev->skin = 0;
CXenSpore::Spawn();
2016-07-31 18:48:50 +05:00
UTIL_SetSize( pev, Vector( -16, -16, 0 ), Vector( 16, 16, 64) );
2017-12-18 02:39:44 +03:00
}
2017-12-18 02:39:44 +03:00
void CXenSporeMed::Spawn( void )
{
pev->skin = 1;
CXenSpore::Spawn();
2016-07-31 18:48:50 +05:00
UTIL_SetSize( pev, Vector( -40, -40, 0 ), Vector( 40, 40, 120 ) );
2017-12-18 02:39:44 +03:00
}
// I just eyeballed these -- fill in hulls for the legs
const Vector CXenSporeLarge::m_hullSizes[] =
2017-12-18 02:39:44 +03:00
{
Vector( 90, -25, 0 ),
Vector( 25, 75, 0 ),
Vector( -15, -100, 0 ),
Vector( -90, -35, 0 ),
Vector( -90, 60, 0 ),
};
void CXenSporeLarge::Spawn( void )
{
pev->skin = 2;
CXenSpore::Spawn();
2016-07-31 18:48:50 +05:00
UTIL_SetSize( pev, Vector( -48, -48, 110 ), Vector( 48, 48, 240 ) );
2017-12-18 02:39:44 +03:00
Vector forward, right;
UTIL_MakeVectorsPrivate( pev->angles, forward, right, NULL );
// Rotate the leg hulls into position
2017-06-29 18:56:03 +05:00
for( int i = 0; i < (int)ARRAYSIZE( m_hullSizes ); i++ )
2016-07-31 18:48:50 +05:00
CXenHull::CreateHull( this, Vector( -12, -12, 0 ), Vector( 12, 12, 120 ), ( m_hullSizes[i].x * forward ) + ( m_hullSizes[i].y * right ) );
2017-12-18 02:39:44 +03:00
}
void CXenSpore :: Spawn( void )
{
Precache();
2016-07-31 18:48:50 +05:00
SET_MODEL( ENT( pev ), pModelNames[pev->skin] );
pev->movetype = MOVETYPE_NONE;
pev->solid = SOLID_BBOX;
2017-12-18 02:39:44 +03:00
pev->takedamage = DAMAGE_YES;
//SetActivity( ACT_IDLE );
2017-12-18 02:39:44 +03:00
pev->sequence = 0;
2016-07-31 18:48:50 +05:00
pev->frame = RANDOM_FLOAT( 0, 255 );
2017-12-18 02:39:44 +03:00
pev->framerate = RANDOM_FLOAT( 0.7, 1.4 );
2016-07-31 18:48:50 +05:00
ResetSequenceInfo();
2017-12-18 02:39:44 +03:00
SetNextThink( RANDOM_FLOAT( 0.1, 0.4 ) ); // Load balance these a bit
}
const char *CXenSpore::pModelNames[] =
2017-12-18 02:39:44 +03:00
{
"models/fungus(small).mdl",
"models/fungus.mdl",
"models/fungus(large).mdl",
};
2016-07-31 18:48:50 +05:00
void CXenSpore::Precache( void )
2017-12-18 02:39:44 +03:00
{
2017-07-24 02:24:55 +05:00
PRECACHE_MODEL( pModelNames[pev->skin] );
2017-12-18 02:39:44 +03:00
}
2016-07-31 18:48:50 +05:00
void CXenSpore::Touch( CBaseEntity *pOther )
2017-12-18 02:39:44 +03:00
{
}
2016-07-31 18:48:50 +05:00
void CXenSpore::Think( void )
2017-12-18 02:39:44 +03:00
{
float flInterval = StudioFrameAdvance();
SetNextThink( 0.1 );
#if 0
DispatchAnimEvents( flInterval );
switch( GetActivity() )
{
default:
case ACT_IDLE:
break;
}
#endif
}