hlsdk-portable/dlls/genericmonster.cpp

228 lines
6.3 KiB
C++
Raw Normal View History

2016-06-04 18:24:23 +05: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.
*
****/
//=========================================================
// Generic Monster - purely for scripted sequence work.
//=========================================================
#include "extdll.h"
#include "util.h"
#include "cbase.h"
#include "monsters.h"
#include "schedule.h"
// For holograms, make them not solid so the player can walk through them
#define SF_GENERICMONSTER_NOTSOLID 4
#define SF_HEAD_CONTROLLER 8
2016-06-04 18:24:23 +05:00
//=========================================================
// Monster's Anim Events Go Here
//=========================================================
2018-01-13 15:04:16 +05:00
class CGenericMonster : public CBaseMonster
2016-06-04 18:24:23 +05:00
{
public:
void Spawn( void );
void Precache( void );
void SetYawSpeed( void );
2016-07-31 18:48:50 +05:00
int Classify( void );
2016-06-04 18:24:23 +05:00
void HandleAnimEvent( MonsterEvent_t *pEvent );
2016-07-31 18:48:50 +05:00
int ISoundMask( void );
2018-01-13 15:04:16 +05:00
void PlayScriptedSentence( const char *pszSentence, float duration, float volume, float attenuation, BOOL bConcurrent, CBaseEntity *pListener );
void IdleHeadTurn( Vector &vecFriend );
void EXPORT MonsterThink();
int Save( CSave &save );
int Restore( CRestore &restore );
static TYPEDESCRIPTION m_SaveData[];
private:
float m_talkTime;
EHANDLE m_hTalkTarget;
float m_flIdealYaw;
float m_flCurrentYaw;
2016-06-04 18:24:23 +05:00
};
LINK_ENTITY_TO_CLASS( monster_generic, CGenericMonster )
2016-06-04 18:24:23 +05:00
2018-01-13 15:04:16 +05:00
TYPEDESCRIPTION CGenericMonster::m_SaveData[] =
{
DEFINE_FIELD( CGenericMonster, m_talkTime, FIELD_FLOAT ),
DEFINE_FIELD( CGenericMonster, m_hTalkTarget, FIELD_EHANDLE ),
DEFINE_FIELD( CGenericMonster, m_flIdealYaw, FIELD_FLOAT ),
DEFINE_FIELD( CGenericMonster, m_flCurrentYaw, FIELD_FLOAT ),
};
IMPLEMENT_SAVERESTORE( CGenericMonster, CBaseMonster )
2016-06-04 18:24:23 +05:00
//=========================================================
// Classify - indicates this monster's place in the
// relationship table.
//=========================================================
2016-07-31 18:48:50 +05:00
int CGenericMonster::Classify( void )
2016-06-04 18:24:23 +05:00
{
2016-07-31 18:48:50 +05:00
return CLASS_PLAYER_ALLY;
2016-06-04 18:24:23 +05:00
}
//=========================================================
// SetYawSpeed - allows each sequence to have a different
// turn rate associated with it.
//=========================================================
2016-07-31 18:48:50 +05:00
void CGenericMonster::SetYawSpeed( void )
2016-06-04 18:24:23 +05:00
{
int ys;
2016-07-31 18:48:50 +05:00
switch( m_Activity )
2016-06-04 18:24:23 +05: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 18:48:50 +05:00
void CGenericMonster::HandleAnimEvent( MonsterEvent_t *pEvent )
2016-06-04 18:24:23 +05:00
{
switch( pEvent->event )
{
case 0:
default:
CBaseMonster::HandleAnimEvent( pEvent );
break;
}
}
//=========================================================
// ISoundMask - generic monster can't hear.
//=========================================================
2016-07-31 18:48:50 +05:00
int CGenericMonster::ISoundMask( void )
2016-06-04 18:24:23 +05:00
{
2017-06-29 18:56:03 +05:00
return 0;
2016-06-04 18:24:23 +05:00
}
//=========================================================
// Spawn
//=========================================================
2016-07-31 18:48:50 +05:00
void CGenericMonster::Spawn()
2016-06-04 18:24:23 +05:00
{
Precache();
2016-07-31 18:48:50 +05:00
SET_MODEL( ENT( pev ), STRING( pev->model ) );
2016-06-04 18:24:23 +05:00
/*
2016-07-31 18:48:50 +05:00
if( FStrEq( STRING( pev->model ), "models/player.mdl" ) )
UTIL_SetSize( pev, VEC_HUMAN_HULL_MIN, VEC_HUMAN_HULL_MAX );
2016-06-04 18:24:23 +05:00
else
2016-07-31 18:48:50 +05:00
UTIL_SetSize( pev, VEC_HULL_MIN, VEC_HULL_MAX);
2016-06-04 18:24:23 +05:00
*/
2016-07-31 18:48:50 +05:00
if( FStrEq( STRING( pev->model ), "models/player.mdl" ) || FStrEq( STRING( pev->model ), "models/holo.mdl" ) )
UTIL_SetSize( pev, VEC_HULL_MIN, VEC_HULL_MAX );
2016-06-04 18:24:23 +05:00
else
2016-07-31 18:48:50 +05:00
UTIL_SetSize( pev, VEC_HUMAN_HULL_MIN, VEC_HUMAN_HULL_MAX );
2016-06-04 18:24:23 +05:00
2016-07-31 18:48:50 +05:00
pev->solid = SOLID_SLIDEBOX;
pev->movetype = MOVETYPE_STEP;
m_bloodColor = BLOOD_COLOR_RED;
pev->health = 8;
m_flFieldOfView = 0.5;// indicates the width of this monster's forward view cone ( as a dotproduct result )
m_MonsterState = MONSTERSTATE_NONE;
2016-06-04 18:24:23 +05:00
MonsterInit();
if( pev->spawnflags & SF_HEAD_CONTROLLER )
{
m_afCapability = bits_CAP_TURN_HEAD;
}
2018-01-13 15:04:16 +05:00
m_flIdealYaw = m_flCurrentYaw = 0;
2016-07-31 18:48:50 +05:00
if( pev->spawnflags & SF_GENERICMONSTER_NOTSOLID )
2016-06-04 18:24:23 +05:00
{
pev->solid = SOLID_NOT;
pev->takedamage = DAMAGE_NO;
}
}
//=========================================================
// Precache - precaches all resources this monster needs
//=========================================================
2016-07-31 18:48:50 +05:00
void CGenericMonster::Precache()
2016-06-04 18:24:23 +05:00
{
2017-07-24 02:24:55 +05:00
PRECACHE_MODEL( STRING( pev->model ) );
2016-07-31 18:48:50 +05:00
}
2016-06-04 18:24:23 +05:00
2018-01-13 15:04:16 +05:00
void CGenericMonster::PlayScriptedSentence( const char *pszSentence, float duration, float volume, float attenuation, BOOL bConcurrent, CBaseEntity *pListener )
{
m_talkTime = gpGlobals->time + duration;
PlaySentence( pszSentence, duration, volume, attenuation );
m_hTalkTarget = pListener;
}
void CGenericMonster::IdleHeadTurn( Vector &vecFriend )
{
// turn head in desired direction only if ent has a turnable head
if( m_afCapability & bits_CAP_TURN_HEAD )
{
float yaw = VecToYaw( vecFriend - pev->origin ) - pev->angles.y;
if( yaw > 180 )
yaw -= 360;
if( yaw < -180 )
yaw += 360;
m_flIdealYaw = yaw;
}
}
void CGenericMonster::MonsterThink()
{
if( m_afCapability & bits_CAP_TURN_HEAD )
{
if( m_hTalkTarget != 0 )
{
if( gpGlobals->time > m_talkTime )
{
m_flIdealYaw = 0;
m_hTalkTarget = 0;
}
else
{
IdleHeadTurn( m_hTalkTarget->pev->origin );
}
}
if( m_flCurrentYaw != m_flIdealYaw )
{
if( m_flCurrentYaw <= m_flIdealYaw )
{
m_flCurrentYaw += Q_min( m_flIdealYaw - m_flCurrentYaw, 20.0f );
}
else
{
m_flCurrentYaw -= Q_min( m_flCurrentYaw - m_flIdealYaw, 20.0f );
}
SetBoneController( 0, m_flCurrentYaw );
}
}
CBaseMonster::MonsterThink();
}
2016-06-04 18:24:23 +05:00
//=========================================================
// AI Schedules Specific to this monster
//=========================================================