hlsdk-portable/dlls/bmodels.cpp

1044 lines
27 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.
*
****/
/*
===== bmodels.cpp ========================================================
spawn, think, and use functions for entities that use brush models
*/
#include "extdll.h"
#include "util.h"
#include "cbase.h"
#include "doors.h"
#include "movewith.h"
extern DLL_GLOBAL Vector g_vecAttackDir;
#define SF_BRUSH_ACCDCC 16// brush should accelerate and decelerate when toggled
#define SF_BRUSH_HURT 32// rotating brush that inflicts pain based on rotation speed
#define SF_ROTATING_NOT_SOLID 64 // some special rotating objects are not solid.
// covering cheesy noise1, noise2, & noise3 fields so they make more sense (for rotating fans)
#define noiseStart noise1
#define noiseStop noise2
#define noiseRunning noise3
#define SF_PENDULUM_SWING 2 // spawnflag that makes a pendulum a rope swing.
//
// BModelOrigin - calculates origin of a bmodel from absmin/size because all bmodel origins are 0 0 0
//
Vector VecBModelOrigin( entvars_t* pevBModel )
{
2019-11-11 02:59:56 +05:00
return ( pevBModel->absmin + pevBModel->absmax ) * 0.5f; //LRC - bug fix for rotating ents
// return pevBModel->absmin + ( pevBModel->size * 0.5f );
2017-12-18 02:39:44 +03:00
}
// =================== FUNC_WALL ==============================================
/*QUAKED func_wall (0 .5 .8) ?
This is just a solid wall if not inhibited
*/
class CFuncWall : public CBaseEntity
{
public:
2016-07-31 18:48:50 +05:00
void Spawn( void );
void Use( CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value );
2017-12-18 02:39:44 +03:00
virtual STATE GetState( void ) { return pev->frame?STATE_ON:STATE_OFF; };
// Bmodels don't go across transitions
2016-07-31 18:48:50 +05:00
virtual int ObjectCaps( void ) { return CBaseEntity :: ObjectCaps() & ~FCAP_ACROSS_TRANSITION; }
2017-12-18 02:39:44 +03:00
int m_iStyle;
};
LINK_ENTITY_TO_CLASS( func_wall, CFuncWall )
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
void CFuncWall::Spawn( void )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
pev->angles = g_vecZero;
pev->movetype = MOVETYPE_PUSH; // so it doesn't get pushed by anything
pev->solid = SOLID_BSP;
SET_MODEL( ENT( pev ), STRING( pev->model ) );
2017-12-18 02:39:44 +03:00
// If it can't move/go away, it's really part of the world
if (!m_pMoveWith) //LRC
2016-06-04 18:24:23 +05:00
pev->flags |= FL_WORLDBRUSH;
2017-12-18 02:39:44 +03:00
//LRC
if (m_iStyle >= 32) LIGHT_STYLE(m_iStyle, "a");
else if (m_iStyle <= -32) LIGHT_STYLE(-m_iStyle, "z");
}
2016-07-31 18:48:50 +05:00
void CFuncWall::Use( CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
if( ShouldToggle( useType, (int)( pev->frame ) ) )
2017-12-18 02:39:44 +03:00
{
pev->frame = 1 - pev->frame;
if (m_iStyle >= 32)
{
if (pev->frame)
LIGHT_STYLE(m_iStyle, "z");
else
LIGHT_STYLE(m_iStyle, "a");
}
else if (m_iStyle <= -32)
{
if (pev->frame)
LIGHT_STYLE(-m_iStyle, "a");
else
LIGHT_STYLE(-m_iStyle, "z");
}
}
}
#define SF_WALL_START_OFF 0x0001
class CFuncWallToggle : public CFuncWall
{
public:
2016-07-31 18:48:50 +05:00
void Spawn( void );
void Use( CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value );
void TurnOff( void );
void TurnOn( void );
BOOL IsOn( void );
2017-12-18 02:39:44 +03:00
virtual STATE GetState( void ) { return (pev->solid == SOLID_NOT)?STATE_OFF:STATE_ON; };
};
LINK_ENTITY_TO_CLASS( func_wall_toggle, CFuncWallToggle )
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
void CFuncWallToggle::Spawn( void )
2017-12-18 02:39:44 +03:00
{
CFuncWall::Spawn();
2016-07-31 18:48:50 +05:00
if( pev->spawnflags & SF_WALL_START_OFF )
2017-12-18 02:39:44 +03:00
TurnOff();
}
2016-07-31 18:48:50 +05:00
void CFuncWallToggle::TurnOff( void )
2017-12-18 02:39:44 +03:00
{
pev->solid = SOLID_NOT;
pev->effects |= EF_NODRAW;
UTIL_SetOrigin( this, pev->origin );
}
2016-07-31 18:48:50 +05:00
void CFuncWallToggle::TurnOn( void )
2017-12-18 02:39:44 +03:00
{
pev->solid = SOLID_BSP;
pev->effects &= ~EF_NODRAW;
UTIL_SetOrigin( this, pev->origin );
}
2016-07-31 18:48:50 +05:00
BOOL CFuncWallToggle::IsOn( void )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
if( pev->solid == SOLID_NOT )
2017-12-18 02:39:44 +03:00
return FALSE;
return TRUE;
}
2016-07-31 18:48:50 +05:00
void CFuncWallToggle::Use( CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value )
2017-12-18 02:39:44 +03:00
{
// int status = IsOn();
BOOL status = (GetState() == STATE_ON);
2016-07-31 18:48:50 +05:00
if( ShouldToggle( useType, status ) )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
if( status )
2017-12-18 02:39:44 +03:00
TurnOff();
else
TurnOn();
}
}
2016-07-31 18:48:50 +05:00
#define SF_CONVEYOR_VISUAL 0x0001
2017-12-18 02:39:44 +03:00
#define SF_CONVEYOR_NOTSOLID 0x0002
class CFuncConveyor : public CFuncWall
{
public:
2016-07-31 18:48:50 +05:00
void Spawn( void );
void Use( CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value );
void UpdateSpeed( float speed );
2017-12-18 02:39:44 +03:00
};
LINK_ENTITY_TO_CLASS( func_conveyor, CFuncConveyor )
2016-07-31 18:48:50 +05:00
void CFuncConveyor::Spawn( void )
2017-12-18 02:39:44 +03:00
{
SetMovedir( pev );
CFuncWall::Spawn();
2016-07-31 18:48:50 +05:00
if( !( pev->spawnflags & SF_CONVEYOR_VISUAL ) )
2017-12-18 02:39:44 +03:00
SetBits( pev->flags, FL_CONVEYOR );
// HACKHACK - This is to allow for some special effects
2016-07-31 18:48:50 +05:00
if( pev->spawnflags & SF_CONVEYOR_NOTSOLID )
2017-12-18 02:39:44 +03:00
{
pev->solid = SOLID_NOT;
pev->skin = 0; // Don't want the engine thinking we've got special contents on this brush
}
2019-10-13 16:49:25 +05:00
if( pev->speed == 0.0f )
pev->speed = 100.0f;
2017-12-18 02:39:44 +03:00
UpdateSpeed( pev->speed );
}
// HACKHACK -- This is ugly, but encode the speed in the rendercolor to avoid adding more data to the network stream
2016-07-31 18:48:50 +05:00
void CFuncConveyor::UpdateSpeed( float speed )
2017-12-18 02:39:44 +03:00
{
// Encode it as an integer with 4 fractional bits
2019-10-13 16:49:25 +05:00
int speedCode = (int)( fabs( speed ) * 16.0f );
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
if( speed < 0 )
2017-12-18 02:39:44 +03:00
pev->rendercolor.x = 1;
else
pev->rendercolor.x = 0;
2016-07-31 18:48:50 +05:00
pev->rendercolor.y = speedCode >> 8;
pev->rendercolor.z = speedCode & 0xFF;
2017-12-18 02:39:44 +03:00
}
2016-07-31 18:48:50 +05:00
void CFuncConveyor::Use( CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value )
2017-12-18 02:39:44 +03:00
{
pev->speed = -pev->speed;
UpdateSpeed( pev->speed );
}
// =================== FUNC_ILLUSIONARY ==============================================
/*QUAKED func_illusionary (0 .5 .8) ?
A simple entity that looks solid but lets you walk through it.
*/
class CFuncIllusionary : public CBaseToggle
{
public:
void Spawn( void );
void EXPORT SloshTouch( CBaseEntity *pOther );
void KeyValue( KeyValueData *pkvd );
2016-07-31 18:48:50 +05:00
virtual int ObjectCaps( void ) { return CBaseEntity :: ObjectCaps() & ~FCAP_ACROSS_TRANSITION; }
2017-12-18 02:39:44 +03:00
};
LINK_ENTITY_TO_CLASS( func_illusionary, CFuncIllusionary )
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
void CFuncIllusionary::KeyValue( KeyValueData *pkvd )
2017-12-18 02:39:44 +03:00
{
// LRC- surely it just parses this automatically? pev values are handled by the engine.
2016-07-31 18:48:50 +05:00
if( FStrEq( pkvd->szKeyName, "skin" ) )//skin is used for content type
2017-12-18 02:39:44 +03:00
{
2017-06-29 18:56:03 +05:00
pev->skin = atoi( pkvd->szValue );
2017-12-18 02:39:44 +03:00
pkvd->fHandled = TRUE;
}
else
CBaseToggle::KeyValue( pkvd );
}
2016-07-31 18:48:50 +05:00
void CFuncIllusionary::Spawn( void )
2017-12-18 02:39:44 +03:00
{
pev->angles = g_vecZero;
pev->movetype = MOVETYPE_NONE;
pev->solid = SOLID_NOT;// always solid_not
2016-07-31 18:48:50 +05:00
SET_MODEL( ENT( pev ), STRING( pev->model ) );
2017-12-18 02:39:44 +03:00
// I'd rather eat the network bandwidth of this than figure out how to save/restore
// these entities after they have been moved to the client, or respawn them ala Quake
// Perhaps we can do this in deathmatch only.
// MAKE_STATIC(ENT(pev));
}
// =================== FUNC_SHINE ==============================================
//LRC - shiny surfaces
class CFuncShine : public CBaseEntity
{
public:
void Spawn( void );
void Activate( void );
virtual int ObjectCaps( void ) { return CBaseEntity :: ObjectCaps() & ~FCAP_ACROSS_TRANSITION; }
void DesiredAction( void );
void EXPORT Think( void );
};
2017-12-18 23:47:12 +03:00
LINK_ENTITY_TO_CLASS( func_shine, CFuncShine )
2017-12-18 02:39:44 +03:00
extern int gmsgAddShine;
void CFuncShine :: Spawn( void )
{
pev->solid = SOLID_NOT;// always solid_not
SET_MODEL( ENT(pev), STRING(pev->model) );
pev->effects |= EF_NODRAW;
// not that we actually need to precache it here, but we do need to make sure it exists
2017-12-18 23:47:12 +03:00
PRECACHE_MODEL( STRING(pev->message) );
2017-12-18 02:39:44 +03:00
}
void CFuncShine :: Activate( void )
{
// ALERT(at_console, "Activate shine\n");
CBaseEntity::Activate();
UTIL_DesiredAction(this);
}
void CFuncShine :: DesiredAction( void )
{
if (pev->message && pev->renderamt)
{
// ALERT(at_console, "Prepare think\n");
2020-01-07 17:53:03 +05:00
pev->nextthink = gpGlobals->time + 1.5f;
2017-12-18 02:39:44 +03:00
}
}
void CFuncShine :: Think( void )
{
// ALERT(at_console, "Think shine\n");
MESSAGE_BEGIN(MSG_BROADCAST, gmsgAddShine, NULL);
WRITE_BYTE(pev->scale);
WRITE_BYTE(pev->renderamt);
WRITE_COORD(pev->absmin.x + 2); // take off 2: mins values are padded, but we just want to hug the surface
WRITE_COORD(pev->absmax.x - 2);
WRITE_COORD(pev->absmin.y + 2);
WRITE_COORD(pev->absmax.y - 2);
WRITE_COORD(pev->absmin.z + 2);
WRITE_STRING(STRING(pev->message));
MESSAGE_END();
}
// -------------------------------------------------------------------------------
//
// Monster only clip brush
//
// This brush will be solid for any entity who has the FL_MONSTERCLIP flag set
// in pev->flags
//
// otherwise it will be invisible and not solid. This can be used to keep
// specific monsters out of certain areas
//
// -------------------------------------------------------------------------------
class CFuncMonsterClip : public CFuncWall
{
public:
2016-07-31 18:48:50 +05:00
void Spawn( void );
void Use( CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value ) {} // Clear out func_wall's use function
2017-12-18 02:39:44 +03:00
};
LINK_ENTITY_TO_CLASS( func_monsterclip, CFuncMonsterClip )
2017-12-18 02:39:44 +03:00
void CFuncMonsterClip::Spawn( void )
{
CFuncWall::Spawn();
2016-07-31 18:48:50 +05:00
if( CVAR_GET_FLOAT( "showtriggers" ) == 0 )
2017-12-18 02:39:44 +03:00
pev->effects = EF_NODRAW;
pev->flags |= FL_MONSTERCLIP;
}
// =================== FUNC_ROTATING ==============================================
class CFuncRotating : public CBaseEntity
{
public:
// basic functions
void Spawn( void );
void Precache( void );
2016-07-31 18:48:50 +05:00
void EXPORT SpinUp( void );
void EXPORT SpinDown( void );
2017-12-18 02:39:44 +03:00
void KeyValue( KeyValueData* pkvd);
2016-07-31 18:48:50 +05:00
void EXPORT HurtTouch( CBaseEntity *pOther );
2017-12-18 02:39:44 +03:00
void EXPORT RotatingUse( CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value );
void EXPORT WaitForStart (); //LRC - get round 1.1.0.8's bizarre behaviour on startup
void EXPORT Rotate( void );
2016-07-31 18:48:50 +05:00
void RampPitchVol(int fUp );
2017-12-18 02:39:44 +03:00
void Blocked( CBaseEntity *pOther );
2016-07-31 18:48:50 +05:00
virtual int ObjectCaps( void ) { return CBaseEntity :: ObjectCaps() & ~FCAP_ACROSS_TRANSITION; }
virtual int Save( CSave &save );
virtual int Restore( CRestore &restore );
static TYPEDESCRIPTION m_SaveData[];
2017-12-18 02:39:44 +03:00
float m_flFanFriction;
float m_flAttenuation;
float m_flVolume;
float m_pitch;
2016-07-31 18:48:50 +05:00
int m_sounds;
2017-12-18 02:39:44 +03:00
float m_fCurSpeed; //LRC - during spin-up and spin-down, this is
// the current speed factor (between 0 and 1).
// storing this here lets us avoid the hassle of deriving it
// from pev->avelocity.
STATE m_iState; //LRC
virtual STATE GetState( void ) { return m_iState; }; //LRC
};
2016-07-31 18:48:50 +05:00
TYPEDESCRIPTION CFuncRotating::m_SaveData[] =
2017-12-18 02:39:44 +03:00
{
DEFINE_FIELD( CFuncRotating, m_flFanFriction, FIELD_FLOAT ),
DEFINE_FIELD( CFuncRotating, m_flAttenuation, FIELD_FLOAT ),
DEFINE_FIELD( CFuncRotating, m_flVolume, FIELD_FLOAT ),
DEFINE_FIELD( CFuncRotating, m_pitch, FIELD_FLOAT ),
DEFINE_FIELD( CFuncRotating, m_sounds, FIELD_INTEGER ),
DEFINE_FIELD( CFuncRotating, m_fCurSpeed, FIELD_FLOAT ),
};
IMPLEMENT_SAVERESTORE( CFuncRotating, CBaseEntity )
2017-12-18 02:39:44 +03:00
LINK_ENTITY_TO_CLASS( func_rotating, CFuncRotating )
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
void CFuncRotating::KeyValue( KeyValueData* pkvd )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
if( FStrEq( pkvd->szKeyName, "fanfriction" ) )
2017-12-18 02:39:44 +03:00
{
2019-10-13 16:49:25 +05:00
m_flFanFriction = atof( pkvd->szValue ) * 0.01f;
2017-12-18 02:39:44 +03:00
pkvd->fHandled = TRUE;
}
2016-07-31 18:48:50 +05:00
else if( FStrEq( pkvd->szKeyName, "Volume" ) )
2017-12-18 02:39:44 +03:00
{
2019-10-13 16:49:25 +05:00
m_flVolume = atof( pkvd->szValue ) * 0.1f;
2017-12-18 02:39:44 +03:00
2019-10-13 16:49:25 +05:00
if( m_flVolume > 1.0f )
m_flVolume = 1.0f;
if( m_flVolume < 0.0f )
m_flVolume = 0.0f;
2017-12-18 02:39:44 +03:00
pkvd->fHandled = TRUE;
}
2016-07-31 18:48:50 +05:00
else if( FStrEq( pkvd->szKeyName, "spawnorigin" ) )
2017-12-18 02:39:44 +03:00
{
Vector tmp;
UTIL_StringToVector( (float *)tmp, pkvd->szValue );
2016-07-31 18:48:50 +05:00
if( tmp != g_vecZero )
2017-12-18 02:39:44 +03:00
pev->origin = tmp;
}
2016-07-31 18:48:50 +05:00
else if( FStrEq( pkvd->szKeyName, "sounds" ) )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
m_sounds = atoi( pkvd->szValue );
2017-12-18 02:39:44 +03:00
pkvd->fHandled = TRUE;
}
else if (FStrEq(pkvd->szKeyName, "axes"))
{
UTIL_StringToVector( (float *)(pev->movedir), pkvd->szValue);
pkvd->fHandled = TRUE;
}
else
CBaseEntity::KeyValue( pkvd );
}
/*QUAKED func_rotating (0 .5 .8) ? START_ON REVERSE X_AXIS Y_AXIS
You need to have an origin brush as part of this entity. The
center of that brush will be
the point around which it is rotated. It will rotate around the Z
axis by default. You can
check either the X_AXIS or Y_AXIS box to change that.
"speed" determines how fast it moves; default value is 100.
"dmg" damage to inflict when blocked (2 default)
REVERSE will cause the it to rotate in the opposite direction.
*/
2016-07-31 18:48:50 +05:00
void CFuncRotating::Spawn()
2017-12-18 02:39:44 +03:00
{
m_iState = STATE_OFF;
m_fCurSpeed = 0; //LRC
// set final pitch. Must not be PITCH_NORM, since we
// plan on pitch shifting later.
m_pitch = PITCH_NORM - 1;
// maintain compatibility with previous maps
2019-10-13 16:49:25 +05:00
if( m_flVolume == 0.0f )
m_flVolume = 1.0f;
2017-12-18 02:39:44 +03:00
// if the designer didn't set a sound attenuation, default to one.
m_flAttenuation = ATTN_NORM;
2016-07-31 18:48:50 +05:00
if( FBitSet( pev->spawnflags, SF_BRUSH_ROTATE_SMALLRADIUS) )
2017-12-18 02:39:44 +03:00
{
m_flAttenuation = ATTN_IDLE;
}
2016-07-31 18:48:50 +05:00
else if( FBitSet( pev->spawnflags, SF_BRUSH_ROTATE_MEDIUMRADIUS ) )
2017-12-18 02:39:44 +03:00
{
m_flAttenuation = ATTN_STATIC;
}
2016-07-31 18:48:50 +05:00
else if( FBitSet( pev->spawnflags, SF_BRUSH_ROTATE_LARGERADIUS ) )
2017-12-18 02:39:44 +03:00
{
m_flAttenuation = ATTN_NORM;
}
// prevent divide by zero if level designer forgets friction!
2019-11-11 02:59:56 +05:00
if( m_flFanFriction <= 0.0f ) //LRC - ensure it's not negative
2017-12-18 02:39:44 +03:00
{
2019-10-13 16:49:25 +05:00
m_flFanFriction = 1.0f;
2017-12-18 02:39:44 +03:00
}
2016-07-31 18:48:50 +05:00
2017-12-18 02:39:44 +03:00
if (pev->movedir == g_vecZero)
{
2019-11-11 02:59:56 +05:00
if( FBitSet( pev->spawnflags, SF_BRUSH_ROTATE_Z_AXIS ) )
pev->movedir = Vector( 0.0f, 0.0f, 1.0f );
else if( FBitSet( pev->spawnflags, SF_BRUSH_ROTATE_X_AXIS ) )
pev->movedir = Vector( 1.0f, 0.0f, 0.0f );
else
pev->movedir = Vector( 0.0f, 1.0f, 0.0f ); // y-axis
2017-12-18 02:39:44 +03:00
}
// check for reverse rotation
2016-07-31 18:48:50 +05:00
if( FBitSet( pev->spawnflags, SF_BRUSH_ROTATE_BACKWARDS ) )
2019-10-13 16:49:25 +05:00
pev->movedir = pev->movedir * -1.0f;
2017-12-18 02:39:44 +03:00
// some rotating objects like fake volumetric lights will not be solid.
2016-07-31 18:48:50 +05:00
if( FBitSet( pev->spawnflags, SF_ROTATING_NOT_SOLID ) )
2017-12-18 02:39:44 +03:00
{
pev->solid = SOLID_NOT;
pev->skin = CONTENTS_EMPTY;
2016-07-31 18:48:50 +05:00
pev->movetype = MOVETYPE_PUSH;
2017-12-18 02:39:44 +03:00
}
else
{
2016-07-31 18:48:50 +05:00
pev->solid = SOLID_BSP;
pev->movetype = MOVETYPE_PUSH;
2017-12-18 02:39:44 +03:00
}
UTIL_SetOrigin(this, pev->origin);
2017-07-24 02:24:55 +05:00
SET_MODEL( ENT( pev ), STRING( pev->model ) );
2017-12-18 02:39:44 +03:00
2016-06-04 18:24:23 +05:00
SetUse( &CFuncRotating::RotatingUse );
2017-12-18 02:39:44 +03:00
// did level designer forget to assign speed?
2019-10-13 16:49:25 +05:00
if( pev->speed <= 0.0f )
pev->speed = 0.0f;
2017-12-18 02:39:44 +03:00
// Removed this per level designers request. -- JAY
2016-07-31 18:48:50 +05:00
// if( pev->dmg == 0 )
2017-12-18 02:39:44 +03:00
// pev->dmg = 2;
// instant-use brush?
//LRC - start immediately if unnamed, too.
2019-11-11 02:59:56 +05:00
if( FBitSet( pev->spawnflags, SF_BRUSH_ROTATE_INSTANT ) || FStringNull( pev->targetname ) )
2016-07-31 18:48:50 +05:00
{
2019-11-11 02:59:56 +05:00
SetThink( &CFuncRotating::WaitForStart );
SetNextThink( 1.5f ); // leave a magic delay for client to start up
2016-07-31 18:48:50 +05:00
}
2017-12-18 02:39:44 +03:00
// can this brush inflict pain?
2016-07-31 18:48:50 +05:00
if( FBitSet( pev->spawnflags, SF_BRUSH_HURT ) )
2017-12-18 02:39:44 +03:00
{
2016-06-04 18:24:23 +05:00
SetTouch( &CFuncRotating::HurtTouch );
2017-12-18 02:39:44 +03:00
}
2016-07-31 18:48:50 +05:00
Precache();
2016-06-04 18:24:23 +05:00
}
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
void CFuncRotating::Precache( void )
2017-12-18 02:39:44 +03:00
{
const char* szSoundFile = STRING( pev->message );
BOOL NullSound = FALSE;
2017-12-18 02:39:44 +03:00
// set up fan sounds
2019-09-24 03:00:37 +05:00
if( !FStringNull( pev->message ) && szSoundFile[0] != '\0' )
2017-12-18 02:39:44 +03:00
{
// if a path is set for a wave, use it
}
else
2017-12-18 02:39:44 +03:00
{
// otherwise use preset sound
2016-07-31 18:48:50 +05:00
switch( m_sounds )
2017-12-18 02:39:44 +03:00
{
case 1:
szSoundFile = "fans/fan1.wav";
2017-12-18 02:39:44 +03:00
break;
case 2:
szSoundFile = "fans/fan2.wav";
2017-12-18 02:39:44 +03:00
break;
case 3:
szSoundFile = "fans/fan3.wav";
2017-12-18 02:39:44 +03:00
break;
case 4:
szSoundFile = "fans/fan4.wav";
2017-12-18 02:39:44 +03:00
break;
case 5:
szSoundFile = "fans/fan5.wav";
2017-12-18 02:39:44 +03:00
break;
case 0:
default:
szSoundFile = "common/null.wav";
NullSound = TRUE;
break;
2017-12-18 02:39:44 +03:00
}
}
2016-07-31 18:48:50 +05:00
if( !NullSound )
PRECACHE_SOUND( szSoundFile );
pev->noiseRunning = MAKE_STRING( szSoundFile );
2016-07-31 18:48:50 +05:00
if( pev->avelocity != g_vecZero )
2017-12-18 02:39:44 +03:00
{
// if fan was spinning, and we went through transition or save/restore,
// make sure we restart the sound. 1.5 sec delay is magic number. KDB
2016-06-04 18:24:23 +05:00
SetThink( &CFuncRotating::SpinUp );
2019-11-11 02:59:56 +05:00
SetNextThink( 1.5f );
2017-12-18 02:39:44 +03:00
}
}
2019-11-11 02:59:56 +05:00
void CFuncRotating::WaitForStart()
2017-12-18 02:39:44 +03:00
{
2019-11-11 02:59:56 +05:00
if( gpGlobals->time > 1 ) // has the client started yet?
2017-12-18 02:39:44 +03:00
{
SUB_CallUseToggle();
}
else
{
2019-11-11 02:59:56 +05:00
SetNextThink( 0.1f );
2017-12-18 02:39:44 +03:00
}
}
//
// Touch - will hurt others based on how fast the brush is spinning
//
2016-07-31 18:48:50 +05:00
void CFuncRotating::HurtTouch( CBaseEntity *pOther )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
entvars_t *pevOther = pOther->pev;
2017-12-18 02:39:44 +03:00
// we can't hurt this thing, so we're not concerned with it
2016-07-31 18:48:50 +05:00
if( !pevOther->takedamage )
2017-12-18 02:39:44 +03:00
return;
// calculate damage based on rotation speed
pev->dmg = m_fCurSpeed / 10; //LRC
// pev->dmg = pev->avelocity.Length() / 10;
2016-07-31 18:48:50 +05:00
pOther->TakeDamage( pev, pev, pev->dmg, DMG_CRUSH );
pevOther->velocity = ( pevOther->origin - VecBModelOrigin( pev ) ).Normalize() * pev->dmg;
2017-12-18 02:39:44 +03:00
}
//
// RampPitchVol - ramp pitch and volume up to final values, based on difference
// between how fast we're going vs how fast we plan to go
//
#define FANPITCHMIN 30
#define FANPITCHMAX 100
2016-07-31 18:48:50 +05:00
void CFuncRotating::RampPitchVol( int fUp )
2017-12-18 02:39:44 +03:00
{
float fvol;
float fpitch;
int pitch;
float speedfactor = m_fCurSpeed/pev->speed;
2016-07-31 18:48:50 +05:00
2017-12-18 02:39:44 +03:00
fvol = m_flVolume * speedfactor; // slowdown volume ramps down to 0
fpitch = FANPITCHMIN + (FANPITCHMAX - FANPITCHMIN) * speedfactor;
2016-07-31 18:48:50 +05:00
pitch = (int)fpitch;
if( pitch == PITCH_NORM )
pitch = PITCH_NORM - 1;
2017-12-18 02:39:44 +03:00
// change the fan's vol and pitch
EMIT_SOUND_DYN(ENT(pev), CHAN_STATIC, (char *)STRING(pev->noiseRunning),
2016-07-31 18:48:50 +05:00
fvol, m_flAttenuation, SND_CHANGE_PITCH | SND_CHANGE_VOL, pitch );
2017-12-18 02:39:44 +03:00
}
//
// SpinUp - accelerates a non-moving func_rotating up to it's speed
//
2016-07-31 18:48:50 +05:00
void CFuncRotating::SpinUp( void )
2017-12-18 02:39:44 +03:00
{
//Vector vecAVel;//rotational velocity
2019-11-11 02:59:56 +05:00
SetNextThink( 0.1f );
2017-12-18 02:39:44 +03:00
m_fCurSpeed = m_fCurSpeed + ( pev->speed * m_flFanFriction );
UTIL_SetAvelocity(this, pev->movedir * m_fCurSpeed);
//pev->avelocity = pev->avelocity + ( pev->movedir * ( pev->speed * m_flFanFriction ) );
//vecAVel = pev->avelocity;// cache entity's rotational velocity
// if we've met or exceeded target speed, set target speed and stop thinking
if ( m_fCurSpeed >= pev->speed )
{
m_iState = STATE_ON;
m_fCurSpeed = pev->speed;
UTIL_SetAvelocity(this, pev->movedir * pev->speed);
//pev->avelocity = pev->movedir * pev->speed;// set speed in case we overshot
EMIT_SOUND_DYN(ENT(pev), CHAN_STATIC, (char *)STRING(pev->noiseRunning),
2016-07-31 18:48:50 +05:00
m_flVolume, m_flAttenuation, SND_CHANGE_PITCH | SND_CHANGE_VOL, FANPITCHMAX );
2016-06-04 18:24:23 +05:00
SetThink( &CFuncRotating::Rotate );
2017-12-18 02:39:44 +03:00
Rotate();
}
else
{
2016-07-31 18:48:50 +05:00
RampPitchVol( TRUE );
2017-12-18 02:39:44 +03:00
}
}
//
// SpinDown - decelerates a moving func_rotating to a standstill.
//
2016-07-31 18:48:50 +05:00
void CFuncRotating::SpinDown( void )
2017-12-18 02:39:44 +03:00
{
2019-11-11 02:59:56 +05:00
SetNextThink( 0.1f );
2017-12-18 02:39:44 +03:00
m_fCurSpeed = m_fCurSpeed - ( pev->speed * m_flFanFriction );
UTIL_SetAvelocity(this, pev->movedir * m_fCurSpeed);
//pev->avelocity = pev->avelocity - ( pev->movedir * ( pev->speed * m_flFanFriction ) );//spin down slower than spinup
// if we've met or exceeded target speed, set target speed and stop thinking
if (m_fCurSpeed <= 0)
{
m_iState = STATE_OFF;
m_fCurSpeed = 0;
UTIL_SetAvelocity(this, g_vecZero);
//pev->avelocity = g_vecZero;// set speed in case we overshot
2016-07-31 18:48:50 +05:00
2017-12-18 02:39:44 +03:00
// stop sound, we're done
2017-07-24 02:24:55 +05:00
EMIT_SOUND_DYN( ENT( pev ), CHAN_STATIC, STRING( pev->noiseRunning /* Stop */ ),
2017-06-29 18:56:03 +05:00
0, 0, SND_STOP, (int)m_pitch );
2017-12-18 02:39:44 +03:00
2016-06-04 18:24:23 +05:00
SetThink( &CFuncRotating::Rotate );
2017-12-18 02:39:44 +03:00
Rotate();
}
else
{
2016-07-31 18:48:50 +05:00
RampPitchVol( FALSE );
2017-12-18 02:39:44 +03:00
}
}
2016-07-31 18:48:50 +05:00
void CFuncRotating::Rotate( void )
2017-12-18 02:39:44 +03:00
{
2019-11-11 02:59:56 +05:00
SetNextThink( 10.0f );
2017-12-18 02:39:44 +03:00
}
//=========================================================
// Rotating Use - when a rotating brush is triggered
//=========================================================
2016-07-31 18:48:50 +05:00
void CFuncRotating::RotatingUse( CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value )
2017-12-18 02:39:44 +03:00
{
if (!ShouldToggle(useType)) return;
// is this a brush that should accelerate and decelerate when turned on/off (fan)?
2016-07-31 18:48:50 +05:00
if( FBitSet ( pev->spawnflags, SF_BRUSH_ACCDCC ) )
2017-12-18 02:39:44 +03:00
{
// fan is spinning, so stop it.
if ( m_fCurSpeed != 0 )
// if ( pev->avelocity != g_vecZero )
{
m_iState = STATE_TURN_OFF;
2016-06-04 18:24:23 +05:00
SetThink( &CFuncRotating::SpinDown );
2017-07-24 02:24:55 +05:00
//EMIT_SOUND_DYN( ENT( pev ), CHAN_WEAPON, STRING( pev->noiseStop ),
2016-07-31 18:48:50 +05:00
// m_flVolume, m_flAttenuation, 0, m_pitch );
2017-12-18 02:39:44 +03:00
2019-11-11 02:59:56 +05:00
SetNextThink( 0.1f );
2017-12-18 02:39:44 +03:00
}
else// fan is not moving, so start it
{
m_iState = STATE_TURN_ON;
2016-06-04 18:24:23 +05:00
SetThink( &CFuncRotating::SpinUp );
2017-07-24 02:24:55 +05:00
EMIT_SOUND_DYN( ENT( pev ), CHAN_STATIC, STRING( pev->noiseRunning ),
2019-10-13 16:49:25 +05:00
0.01f, m_flAttenuation, 0, FANPITCHMIN );
2017-12-18 02:39:44 +03:00
2019-11-11 02:59:56 +05:00
SetNextThink( 0.1f );
2017-12-18 02:39:44 +03:00
}
}
else // if ( !FBitSet ( pev->spawnflags, SF_BRUSH_ACCDCC ) )//this is a normal start/stop brush.
{
if ( m_fCurSpeed != 0 ) //LRC
// if ( pev->avelocity != g_vecZero )
{
m_iState = STATE_OFF;
// play stopping sound here
2016-06-04 18:24:23 +05:00
SetThink( &CFuncRotating::SpinDown );
2017-07-24 02:24:55 +05:00
// EMIT_SOUND_DYN( ENT( pev ), CHAN_WEAPON, STRING( pev->noiseStop ),
2016-07-31 18:48:50 +05:00
// m_flVolume, m_flAttenuation, 0, m_pitch );
2017-12-18 02:39:44 +03:00
2019-11-11 02:59:56 +05:00
SetNextThink( 0.1f );
2017-12-18 02:39:44 +03:00
// pev->avelocity = g_vecZero;
}
else
{
m_iState = STATE_ON;
EMIT_SOUND_DYN(ENT(pev), CHAN_STATIC, (char *)STRING(pev->noiseRunning),
2016-07-31 18:48:50 +05:00
m_flVolume, m_flAttenuation, 0, FANPITCHMAX );
2017-12-18 02:39:44 +03:00
//LRC
m_fCurSpeed = pev->speed;
UTIL_SetAvelocity(this, pev->movedir * pev->speed);
// pev->avelocity = pev->movedir * pev->speed;
2016-06-04 18:24:23 +05:00
SetThink( &CFuncRotating::Rotate );
2017-12-18 02:39:44 +03:00
Rotate();
}
}
}
//
// RotatingBlocked - An entity has blocked the brush
//
2016-07-31 18:48:50 +05:00
void CFuncRotating::Blocked( CBaseEntity *pOther )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
pOther->TakeDamage( pev, pev, pev->dmg, DMG_CRUSH );
2017-12-18 02:39:44 +03:00
}
//#endif
class CPendulum : public CBaseEntity
{
public:
2016-07-31 18:48:50 +05:00
void Spawn( void );
void KeyValue( KeyValueData *pkvd );
2017-12-18 02:39:44 +03:00
void EXPORT SwingThink( void );
2016-07-31 18:48:50 +05:00
void EXPORT PendulumUse( CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value );
2017-12-18 02:39:44 +03:00
void EXPORT StopThink( void );
2016-07-31 18:48:50 +05:00
void Touch( CBaseEntity *pOther );
void EXPORT RopeTouch( CBaseEntity *pOther );// this touch func makes the pendulum a rope
virtual int ObjectCaps( void ) { return CBaseEntity :: ObjectCaps() & ~FCAP_ACROSS_TRANSITION; }
virtual int Save( CSave &save );
virtual int Restore( CRestore &restore );
void Blocked( CBaseEntity *pOther );
2017-12-18 02:39:44 +03:00
virtual STATE GetState( void ) { return (pev->speed)?STATE_ON:STATE_OFF; }
2016-07-31 18:48:50 +05:00
static TYPEDESCRIPTION m_SaveData[];
float m_accel; // Acceleration
float m_distance;
float m_time;
float m_damp;
float m_maxSpeed;
float m_dampSpeed;
vec3_t m_center;
vec3_t m_start;
2017-12-18 02:39:44 +03:00
};
LINK_ENTITY_TO_CLASS( func_pendulum, CPendulum )
2017-12-18 02:39:44 +03:00
TYPEDESCRIPTION CPendulum::m_SaveData[] =
2017-12-18 02:39:44 +03:00
{
DEFINE_FIELD( CPendulum, m_accel, FIELD_FLOAT ),
DEFINE_FIELD( CPendulum, m_distance, FIELD_FLOAT ),
DEFINE_FIELD( CPendulum, m_time, FIELD_TIME ),
DEFINE_FIELD( CPendulum, m_damp, FIELD_FLOAT ),
DEFINE_FIELD( CPendulum, m_maxSpeed, FIELD_FLOAT ),
DEFINE_FIELD( CPendulum, m_dampSpeed, FIELD_FLOAT ),
DEFINE_FIELD( CPendulum, m_center, FIELD_VECTOR ),
DEFINE_FIELD( CPendulum, m_start, FIELD_VECTOR ),
};
IMPLEMENT_SAVERESTORE( CPendulum, CBaseEntity )
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
void CPendulum::KeyValue( KeyValueData *pkvd )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
if( FStrEq( pkvd->szKeyName, "distance" ) )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
m_distance = atof( pkvd->szValue );
2017-12-18 02:39:44 +03:00
pkvd->fHandled = TRUE;
}
else if (FStrEq(pkvd->szKeyName, "axes"))
{
UTIL_StringToVector( (float*)(pev->movedir), pkvd->szValue );
pkvd->fHandled = TRUE;
}
2016-07-31 18:48:50 +05:00
else if( FStrEq( pkvd->szKeyName, "damp" ) )
2017-12-18 02:39:44 +03:00
{
2019-10-13 16:49:25 +05:00
m_damp = atof( pkvd->szValue ) * 0.001f;
2017-12-18 02:39:44 +03:00
pkvd->fHandled = TRUE;
}
else
CBaseEntity::KeyValue( pkvd );
}
2016-07-31 18:48:50 +05:00
void CPendulum::Spawn( void )
2017-12-18 02:39:44 +03:00
{
// set the axis of rotation
2016-07-31 18:48:50 +05:00
CBaseToggle::AxisDir( pev );
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
if( FBitSet( pev->spawnflags, SF_DOOR_PASSABLE ) )
pev->solid = SOLID_NOT;
2017-12-18 02:39:44 +03:00
else
2016-07-31 18:48:50 +05:00
pev->solid = SOLID_BSP;
pev->movetype = MOVETYPE_PUSH;
2017-12-18 02:39:44 +03:00
UTIL_SetOrigin(this, pev->origin);
2016-07-31 18:48:50 +05:00
SET_MODEL( ENT( pev ), STRING( pev->model ) );
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
if( m_distance == 0 )
2017-12-18 02:39:44 +03:00
return;
2019-10-13 16:49:25 +05:00
if( pev->speed == 0.0f )
pev->speed = 100.0f;
2017-12-18 02:39:44 +03:00
2019-10-13 16:49:25 +05:00
m_accel = ( pev->speed * pev->speed ) / ( 2.0f * fabs( m_distance ) ); // Calculate constant acceleration from speed and distance
2017-12-18 02:39:44 +03:00
m_maxSpeed = pev->speed;
m_start = pev->angles;
2019-10-13 16:49:25 +05:00
m_center = pev->angles + ( m_distance * 0.5f ) * pev->movedir;
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
if( FBitSet( pev->spawnflags, SF_BRUSH_ROTATE_INSTANT ) )
{
2019-11-11 02:59:56 +05:00
SetThink( &CPendulum::SUB_CallUseToggle );
SetNextThink( 0.1f );
2017-12-18 02:39:44 +03:00
}
pev->speed = 0;
2016-06-04 18:24:23 +05:00
SetUse( &CPendulum::PendulumUse );
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
if( FBitSet( pev->spawnflags, SF_PENDULUM_SWING ) )
2017-12-18 02:39:44 +03:00
{
2016-06-04 18:24:23 +05:00
SetTouch( &CPendulum::RopeTouch );
2017-12-18 02:39:44 +03:00
}
}
2016-07-31 18:48:50 +05:00
void CPendulum::PendulumUse( CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value )
2017-12-18 02:39:44 +03:00
{
if (!ShouldToggle(useType)) return;
2016-07-31 18:48:50 +05:00
if( pev->speed ) // Pendulum is moving, stop it and auto-return if necessary
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
if( FBitSet( pev->spawnflags, SF_PENDULUM_AUTO_RETURN ) )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
float delta;
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
delta = CBaseToggle::AxisDelta( pev->spawnflags, pev->angles, m_start );
2017-12-18 02:39:44 +03:00
UTIL_SetAvelocity(this, m_maxSpeed * pev->movedir); //LRC
//pev->avelocity = m_maxSpeed * pev->movedir;
SetNextThink(delta / m_maxSpeed);
SetThink(&CPendulum ::StopThink);
}
else
{
2019-10-13 16:49:25 +05:00
pev->speed = 0.0f; // Dead stop
2017-12-18 02:39:44 +03:00
DontThink();
UTIL_SetAvelocity(this, g_vecZero); //LRC
//pev->avelocity = g_vecZero;
}
}
else
{
2019-11-11 02:59:56 +05:00
SetNextThink( 0.1f ); // start the pendulum moving
SetThink( &CPendulum ::SwingThink );
2017-12-18 02:39:44 +03:00
m_time = gpGlobals->time; // Save time to calculate dt
m_dampSpeed = m_maxSpeed;
}
}
void CPendulum :: StopThink( void )
{
UTIL_SetAngles(this, m_start); //LRC
//pev->angles = m_start;
2019-10-13 16:49:25 +05:00
pev->speed = 0.0f;
2017-12-18 02:39:44 +03:00
DontThink();
UTIL_SetAvelocity(this, g_vecZero); //LRC
//pev->avelocity = g_vecZero;
}
void CPendulum::Blocked( CBaseEntity *pOther )
{
m_time = gpGlobals->time;
}
void CPendulum :: SwingThink( void )
{
float delta, dt;
2016-07-31 18:48:50 +05:00
delta = CBaseToggle::AxisDelta( pev->spawnflags, pev->angles, m_center );
2017-12-18 02:39:44 +03:00
dt = gpGlobals->time - m_time; // How much time has passed?
m_time = gpGlobals->time; // Remember the last time called
2019-10-13 16:49:25 +05:00
if( delta > 0.0f && m_accel > 0.0f )
2017-12-18 02:39:44 +03:00
pev->speed -= m_accel * dt; // Integrate velocity
else
pev->speed += m_accel * dt;
2016-07-31 18:48:50 +05:00
if( pev->speed > m_maxSpeed )
2017-12-18 02:39:44 +03:00
pev->speed = m_maxSpeed;
2016-07-31 18:48:50 +05:00
else if( pev->speed < -m_maxSpeed )
2017-12-18 02:39:44 +03:00
pev->speed = -m_maxSpeed;
// scale the destdelta vector by the time spent traveling to get velocity
UTIL_SetAvelocity(this, pev->speed * pev->movedir); //LRC
//pev->avelocity = pev->speed * pev->movedir;
// ALERT(at_console, "m_damp %f, m_dampSpeed %f\n", m_damp, m_dampSpeed);
// ALERT(at_console, "SwingThink: delta %f, dt %f, speed %f, avel %f %f %f\n", delta, dt, pev->speed, pev->avelocity.x, pev->avelocity.y, pev->avelocity.z);
// Call this again
2019-11-11 02:59:56 +05:00
SetNextThink( 0.1f );
SetThink( &CPendulum::SwingThink );
2017-12-18 02:39:44 +03:00
// if (m_pMoveWith) // correct MoveWith problems associated with fast-thinking entities
// UTIL_AssignOrigin(this, m_vecMoveWithOffset + m_pMoveWith->pev->origin);
2016-07-31 18:48:50 +05:00
if( m_damp )
2017-12-18 02:39:44 +03:00
{
m_dampSpeed -= m_damp * m_dampSpeed * dt;
2019-10-13 16:49:25 +05:00
if( m_dampSpeed < 30.0f )
2017-12-18 02:39:44 +03:00
{
UTIL_SetAngles(this, m_center); //LRC
//pev->angles = m_center;
pev->speed = 0;
2017-12-18 23:47:12 +03:00
ALERT(at_console, "**CANCELLING pendulum think!\n");
2017-12-18 02:39:44 +03:00
DontThink();
UTIL_SetAvelocity(this, g_vecZero); //LRC
//pev->avelocity = g_vecZero;
}
2016-07-31 18:48:50 +05:00
else if( pev->speed > m_dampSpeed )
2017-12-18 02:39:44 +03:00
pev->speed = m_dampSpeed;
2016-07-31 18:48:50 +05:00
else if( pev->speed < -m_dampSpeed )
2017-12-18 02:39:44 +03:00
pev->speed = -m_dampSpeed;
}
}
2016-07-31 18:48:50 +05:00
void CPendulum::Touch( CBaseEntity *pOther )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
entvars_t *pevOther = pOther->pev;
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
if( pev->dmg <= 0 )
2017-12-18 02:39:44 +03:00
return;
// we can't hurt this thing, so we're not concerned with it
2016-07-31 18:48:50 +05:00
if( !pevOther->takedamage )
2017-12-18 02:39:44 +03:00
return;
// calculate damage based on rotation speed
2019-10-13 16:49:25 +05:00
float damage = pev->dmg * pev->speed * 0.01f;
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
if( damage < 0 )
2017-12-18 02:39:44 +03:00
damage = -damage;
pOther->TakeDamage( pev, pev, damage, DMG_CRUSH );
2016-07-31 18:48:50 +05:00
pevOther->velocity = ( pevOther->origin - VecBModelOrigin( pev ) ).Normalize() * damage;
2017-12-18 02:39:44 +03:00
}
2016-07-31 18:48:50 +05:00
void CPendulum::RopeTouch( CBaseEntity *pOther )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
entvars_t *pevOther = pOther->pev;
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
if( !pOther->IsPlayer() )
{
// not a player!
ALERT( at_console, "Not a client\n" );
2017-12-18 02:39:44 +03:00
return;
}
2016-07-31 18:48:50 +05:00
if( ENT( pevOther ) == pev->enemy )
{
// this player already on the rope.
2017-12-18 02:39:44 +03:00
return;
}
pev->enemy = pOther->edict();
pevOther->velocity = g_vecZero;
pevOther->movetype = MOVETYPE_NONE;
}