hlsdk-portable/dlls/gman.cpp

237 lines
6.3 KiB
C++
Raw Normal View History

2016-06-04 13:24:23 +00: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.
*
* This source code contains proprietary and confidential information of
* Valve LLC and its suppliers. Access to this code is restricted to
* persons who have executed a written SDK license with Valve. Any access,
* use or distribution of this code by or to any unlicensed person is illegal.
*
****/
//=========================================================
// GMan - misunderstood servant of the people
//=========================================================
#include "extdll.h"
#include "util.h"
#include "cbase.h"
#include "monsters.h"
#include "schedule.h"
#include "weapons.h"
//=========================================================
// Monster's Anim Events Go Here
//=========================================================
class CGMan : public CBaseMonster
{
public:
void Spawn( void );
void Precache( void );
void SetYawSpeed( void );
2016-07-31 13:48:50 +00:00
int Classify ( void );
2016-06-04 13:24:23 +00:00
void HandleAnimEvent( MonsterEvent_t *pEvent );
int ISoundMask ( void );
2016-07-31 13:48:50 +00:00
int Save( CSave &save );
2016-06-04 13:24:23 +00:00
int Restore( CRestore &restore );
static TYPEDESCRIPTION m_SaveData[];
void StartTask( Task_t *pTask );
void RunTask( Task_t *pTask );
2016-07-31 13:48:50 +00:00
int TakeDamage( entvars_t *pevInflictor, entvars_t *pevAttacker, float flDamage, int bitsDamageType );
2016-06-04 13:24:23 +00:00
void TraceAttack( entvars_t *pevAttacker, float flDamage, Vector vecDir, TraceResult *ptr, int bitsDamageType);
void PlayScriptedSentence( const char *pszSentence, float duration, float volume, float attenuation, BOOL bConcurrent, CBaseEntity *pListener );
EHANDLE m_hPlayer;
EHANDLE m_hTalkTarget;
float m_flTalkTime;
};
LINK_ENTITY_TO_CLASS( monster_gman, CGMan )
2016-06-04 13:24:23 +00:00
2016-07-31 13:48:50 +00:00
TYPEDESCRIPTION CGMan::m_SaveData[] =
2016-06-04 13:24:23 +00:00
{
DEFINE_FIELD( CGMan, m_hTalkTarget, FIELD_EHANDLE ),
DEFINE_FIELD( CGMan, m_flTalkTime, FIELD_TIME ),
};
IMPLEMENT_SAVERESTORE( CGMan, CBaseMonster )
2016-06-04 13:24:23 +00:00
//=========================================================
// Classify - indicates this monster's place in the
// relationship table.
//=========================================================
2016-07-31 13:48:50 +00:00
int CGMan::Classify( void )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
return CLASS_NONE;
2016-06-04 13:24:23 +00:00
}
//=========================================================
// SetYawSpeed - allows each sequence to have a different
// turn rate associated with it.
//=========================================================
2016-07-31 13:48:50 +00:00
void CGMan::SetYawSpeed( void )
2016-06-04 13:24:23 +00:00
{
int ys;
2016-07-31 13:48:50 +00:00
switch( m_Activity )
2016-06-04 13:24:23 +00:00
{
case ACT_IDLE:
default:
ys = 90;
}
pev->yaw_speed = ys;
}
//=========================================================
// HandleAnimEvent - catches the monster-specific messages
// that occur when tagged animation frames are played.
//=========================================================
2016-07-31 13:48:50 +00:00
void CGMan::HandleAnimEvent( MonsterEvent_t *pEvent )
2016-06-04 13:24:23 +00:00
{
switch( pEvent->event )
{
case 0:
default:
CBaseMonster::HandleAnimEvent( pEvent );
break;
}
}
//=========================================================
// ISoundMask - generic monster can't hear.
//=========================================================
2016-07-31 13:48:50 +00:00
int CGMan::ISoundMask( void )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
return NULL;
2016-06-04 13:24:23 +00:00
}
//=========================================================
// Spawn
//=========================================================
2016-07-31 13:48:50 +00:00
void CGMan::Spawn()
2016-06-04 13:24:23 +00:00
{
Precache();
2016-07-31 13:48:50 +00:00
SET_MODEL( ENT( pev ), "models/gman.mdl" );
UTIL_SetSize( pev, VEC_HUMAN_HULL_MIN, VEC_HUMAN_HULL_MAX );
2016-06-04 13:24:23 +00:00
2016-07-31 13:48:50 +00:00
pev->solid = SOLID_SLIDEBOX;
2016-06-04 13:24:23 +00:00
pev->movetype = MOVETYPE_STEP;
m_bloodColor = DONT_BLEED;
2016-07-31 13:48:50 +00:00
pev->health = 100;
2016-06-04 13:24:23 +00:00
m_flFieldOfView = 0.5;// indicates the width of this monster's forward view cone ( as a dotproduct result )
m_MonsterState = MONSTERSTATE_NONE;
MonsterInit();
}
//=========================================================
// Precache - precaches all resources this monster needs
//=========================================================
2016-07-31 13:48:50 +00:00
void CGMan::Precache()
2016-06-04 13:24:23 +00:00
{
PRECACHE_MODEL( "models/gman.mdl" );
2016-07-31 13:48:50 +00:00
}
2016-06-04 13:24:23 +00:00
//=========================================================
// AI Schedules Specific to this monster
//=========================================================
2016-07-31 13:48:50 +00:00
void CGMan::StartTask( Task_t *pTask )
2016-06-04 13:24:23 +00:00
{
switch( pTask->iTask )
{
case TASK_WAIT:
2016-07-31 13:48:50 +00:00
if( m_hPlayer == NULL )
2016-06-04 13:24:23 +00:00
{
m_hPlayer = UTIL_FindEntityByClassname( NULL, "player" );
}
break;
}
CBaseMonster::StartTask( pTask );
}
2016-07-31 13:48:50 +00:00
void CGMan::RunTask( Task_t *pTask )
2016-06-04 13:24:23 +00:00
{
switch( pTask->iTask )
{
case TASK_WAIT:
// look at who I'm talking to
2016-07-31 13:48:50 +00:00
if( m_flTalkTime > gpGlobals->time && m_hTalkTarget != NULL )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
float yaw = VecToYaw( m_hTalkTarget->pev->origin - pev->origin ) - pev->angles.y;
2016-06-04 13:24:23 +00:00
2016-07-31 13:48:50 +00:00
if( yaw > 180 )
yaw -= 360;
if( yaw < -180 )
yaw += 360;
2016-06-04 13:24:23 +00:00
// turn towards vector
SetBoneController( 0, yaw );
}
// look at player, but only if playing a "safe" idle animation
2016-07-31 13:48:50 +00:00
else if( m_hPlayer != NULL && pev->sequence == 0 )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
float yaw = VecToYaw( m_hPlayer->pev->origin - pev->origin ) - pev->angles.y;
2016-06-04 13:24:23 +00:00
2016-07-31 13:48:50 +00:00
if( yaw > 180 )
yaw -= 360;
if( yaw < -180 )
yaw += 360;
2016-06-04 13:24:23 +00:00
// turn towards vector
SetBoneController( 0, yaw );
}
else
{
SetBoneController( 0, 0 );
}
CBaseMonster::RunTask( pTask );
break;
default:
SetBoneController( 0, 0 );
CBaseMonster::RunTask( pTask );
break;
}
}
//=========================================================
// Override all damage
//=========================================================
2016-07-31 13:48:50 +00:00
int CGMan::TakeDamage( entvars_t *pevInflictor, entvars_t *pevAttacker, float flDamage, int bitsDamageType )
2016-06-04 13:24:23 +00:00
{
pev->health = pev->max_health / 2; // always trigger the 50% damage aitrigger
2016-07-31 13:48:50 +00:00
if( flDamage > 0 )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
SetConditions( bits_COND_LIGHT_DAMAGE );
2016-06-04 13:24:23 +00:00
}
2016-07-31 13:48:50 +00:00
if( flDamage >= 20 )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
SetConditions( bits_COND_HEAVY_DAMAGE );
2016-06-04 13:24:23 +00:00
}
return TRUE;
}
void CGMan::TraceAttack( entvars_t *pevAttacker, float flDamage, Vector vecDir, TraceResult *ptr, int bitsDamageType)
{
UTIL_Ricochet( ptr->vecEndPos, 1.0 );
AddMultiDamage( pevAttacker, this, flDamage, bitsDamageType );
}
void CGMan::PlayScriptedSentence( const char *pszSentence, float duration, float volume, float attenuation, BOOL bConcurrent, CBaseEntity *pListener )
{
CBaseMonster::PlayScriptedSentence( pszSentence, duration, volume, attenuation, bConcurrent, pListener );
m_flTalkTime = gpGlobals->time + duration;
m_hTalkTarget = pListener;
}