2334 lines
64 KiB
C++
2334 lines
64 KiB
C++
//========= Copyright Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose: Player for Portal.
|
|
//
|
|
//=============================================================================//
|
|
|
|
#include "cbase.h"
|
|
#include "portal_player.h"
|
|
#include "globalstate.h"
|
|
#include "trains.h"
|
|
#include "game.h"
|
|
#include "portal_player_shared.h"
|
|
#include "predicted_viewmodel.h"
|
|
#include "in_buttons.h"
|
|
#include "portal_gamerules.h"
|
|
#include "weapon_portalgun.h"
|
|
#include "portal/weapon_physcannon.h"
|
|
#include "KeyValues.h"
|
|
#include "team.h"
|
|
#include "eventqueue.h"
|
|
#include "weapon_portalbase.h"
|
|
#include "engine/IEngineSound.h"
|
|
#include "ai_basenpc.h"
|
|
#include "SoundEmitterSystem/isoundemittersystembase.h"
|
|
#include "prop_portal_shared.h"
|
|
#include "player_pickup.h" // for player pickup code
|
|
#include "vphysics/player_controller.h"
|
|
#include "datacache/imdlcache.h"
|
|
#include "bone_setup.h"
|
|
#include "portal_gamestats.h"
|
|
#include "physicsshadowclone.h"
|
|
#include "physics_prop_ragdoll.h"
|
|
#include "soundenvelope.h"
|
|
#include "ai_speech.h" // For expressors, vcd playing
|
|
#include "sceneentity.h" // has the VCD precache function
|
|
|
|
// Max mass the player can lift with +use
|
|
#define PORTAL_PLAYER_MAX_LIFT_MASS 85
|
|
#define PORTAL_PLAYER_MAX_LIFT_SIZE 128
|
|
|
|
extern CBaseEntity *g_pLastSpawn;
|
|
|
|
extern void respawn(CBaseEntity *pEdict, bool fCopyCorpse);
|
|
|
|
|
|
// -------------------------------------------------------------------------------- //
|
|
// Player animation event. Sent to the client when a player fires, jumps, reloads, etc..
|
|
// -------------------------------------------------------------------------------- //
|
|
|
|
class CTEPlayerAnimEvent : public CBaseTempEntity
|
|
{
|
|
public:
|
|
DECLARE_CLASS( CTEPlayerAnimEvent, CBaseTempEntity );
|
|
DECLARE_SERVERCLASS();
|
|
|
|
CTEPlayerAnimEvent( const char *name ) : CBaseTempEntity( name )
|
|
{
|
|
}
|
|
|
|
CNetworkHandle( CBasePlayer, m_hPlayer );
|
|
CNetworkVar( int, m_iEvent );
|
|
CNetworkVar( int, m_nData );
|
|
};
|
|
|
|
IMPLEMENT_SERVERCLASS_ST_NOBASE( CTEPlayerAnimEvent, DT_TEPlayerAnimEvent )
|
|
SendPropEHandle( SENDINFO( m_hPlayer ) ),
|
|
SendPropInt( SENDINFO( m_iEvent ), Q_log2( PLAYERANIMEVENT_COUNT ) + 1, SPROP_UNSIGNED ),
|
|
SendPropInt( SENDINFO( m_nData ), 32 ),
|
|
END_SEND_TABLE()
|
|
|
|
static CTEPlayerAnimEvent g_TEPlayerAnimEvent( "PlayerAnimEvent" );
|
|
|
|
void TE_PlayerAnimEvent( CBasePlayer *pPlayer, PlayerAnimEvent_t event, int nData )
|
|
{
|
|
CPVSFilter filter( (const Vector&)pPlayer->EyePosition() );
|
|
|
|
g_TEPlayerAnimEvent.m_hPlayer = pPlayer;
|
|
g_TEPlayerAnimEvent.m_iEvent = event;
|
|
g_TEPlayerAnimEvent.m_nData = nData;
|
|
g_TEPlayerAnimEvent.Create( filter, 0 );
|
|
}
|
|
|
|
|
|
|
|
//=================================================================================
|
|
//
|
|
// Ragdoll Entity
|
|
//
|
|
class CPortalRagdoll : public CBaseAnimatingOverlay, public CDefaultPlayerPickupVPhysics
|
|
{
|
|
public:
|
|
|
|
DECLARE_CLASS( CPortalRagdoll, CBaseAnimatingOverlay );
|
|
DECLARE_SERVERCLASS();
|
|
DECLARE_DATADESC();
|
|
|
|
CPortalRagdoll()
|
|
{
|
|
m_hPlayer.Set( NULL );
|
|
m_vecRagdollOrigin.Init();
|
|
m_vecRagdollVelocity.Init();
|
|
}
|
|
|
|
// Transmit ragdolls to everyone.
|
|
virtual int UpdateTransmitState()
|
|
{
|
|
return SetTransmitState( FL_EDICT_ALWAYS );
|
|
}
|
|
|
|
// In case the client has the player entity, we transmit the player index.
|
|
// In case the client doesn't have it, we transmit the player's model index, origin, and angles
|
|
// so they can create a ragdoll in the right place.
|
|
CNetworkHandle( CBaseEntity, m_hPlayer ); // networked entity handle
|
|
CNetworkVector( m_vecRagdollVelocity );
|
|
CNetworkVector( m_vecRagdollOrigin );
|
|
};
|
|
|
|
LINK_ENTITY_TO_CLASS( portal_ragdoll, CPortalRagdoll );
|
|
|
|
IMPLEMENT_SERVERCLASS_ST_NOBASE( CPortalRagdoll, DT_PortalRagdoll )
|
|
SendPropVector( SENDINFO(m_vecRagdollOrigin), -1, SPROP_COORD ),
|
|
SendPropEHandle( SENDINFO( m_hPlayer ) ),
|
|
SendPropModelIndex( SENDINFO( m_nModelIndex ) ),
|
|
SendPropInt ( SENDINFO(m_nForceBone), 8, 0 ),
|
|
SendPropVector ( SENDINFO(m_vecForce), -1, SPROP_NOSCALE ),
|
|
SendPropVector( SENDINFO( m_vecRagdollVelocity ) ),
|
|
END_SEND_TABLE()
|
|
|
|
|
|
BEGIN_DATADESC( CPortalRagdoll )
|
|
|
|
DEFINE_FIELD( m_vecRagdollOrigin, FIELD_POSITION_VECTOR ),
|
|
DEFINE_FIELD( m_hPlayer, FIELD_EHANDLE ),
|
|
DEFINE_FIELD( m_vecRagdollVelocity, FIELD_VECTOR ),
|
|
|
|
END_DATADESC()
|
|
|
|
|
|
|
|
|
|
|
|
LINK_ENTITY_TO_CLASS( player, CPortal_Player );
|
|
|
|
IMPLEMENT_SERVERCLASS_ST(CPortal_Player, DT_Portal_Player)
|
|
SendPropExclude( "DT_BaseAnimating", "m_flPlaybackRate" ),
|
|
SendPropExclude( "DT_BaseAnimating", "m_nSequence" ),
|
|
SendPropExclude( "DT_BaseAnimating", "m_nNewSequenceParity" ),
|
|
SendPropExclude( "DT_BaseAnimating", "m_nResetEventsParity" ),
|
|
SendPropExclude( "DT_BaseEntity", "m_angRotation" ),
|
|
SendPropExclude( "DT_BaseAnimatingOverlay", "overlay_vars" ),
|
|
SendPropExclude( "DT_BaseFlex", "m_viewtarget" ),
|
|
SendPropExclude( "DT_BaseFlex", "m_flexWeight" ),
|
|
SendPropExclude( "DT_BaseFlex", "m_blinktoggle" ),
|
|
|
|
// portal_playeranimstate and clientside animation takes care of these on the client
|
|
SendPropExclude( "DT_ServerAnimationData" , "m_flCycle" ),
|
|
SendPropExclude( "DT_AnimTimeMustBeFirst" , "m_flAnimTime" ),
|
|
|
|
|
|
SendPropAngle( SENDINFO_VECTORELEM(m_angEyeAngles, 0), 11, SPROP_CHANGES_OFTEN ),
|
|
SendPropAngle( SENDINFO_VECTORELEM(m_angEyeAngles, 1), 11, SPROP_CHANGES_OFTEN ),
|
|
SendPropEHandle( SENDINFO( m_hRagdoll ) ),
|
|
SendPropInt( SENDINFO( m_iSpawnInterpCounter), 4 ),
|
|
SendPropInt( SENDINFO( m_iPlayerSoundType), 3 ),
|
|
SendPropBool( SENDINFO( m_bHeldObjectOnOppositeSideOfPortal) ),
|
|
SendPropEHandle( SENDINFO( m_pHeldObjectPortal ) ),
|
|
SendPropBool( SENDINFO( m_bPitchReorientation ) ),
|
|
SendPropEHandle( SENDINFO( m_hPortalEnvironment ) ),
|
|
SendPropEHandle( SENDINFO( m_hSurroundingLiquidPortal ) ),
|
|
SendPropBool( SENDINFO( m_bSuppressingCrosshair ) ),
|
|
SendPropExclude( "DT_BaseAnimating", "m_flPoseParameter" ),
|
|
|
|
END_SEND_TABLE()
|
|
|
|
BEGIN_DATADESC( CPortal_Player )
|
|
|
|
DEFINE_SOUNDPATCH( m_pWooshSound ),
|
|
|
|
DEFINE_FIELD( m_bHeldObjectOnOppositeSideOfPortal, FIELD_BOOLEAN ),
|
|
DEFINE_FIELD( m_pHeldObjectPortal, FIELD_EHANDLE ),
|
|
DEFINE_FIELD( m_bIntersectingPortalPlane, FIELD_BOOLEAN ),
|
|
DEFINE_FIELD( m_bStuckOnPortalCollisionObject, FIELD_BOOLEAN ),
|
|
DEFINE_FIELD( m_fTimeLastHurt, FIELD_TIME ),
|
|
DEFINE_FIELD( m_StatsThisLevel.iNumPortalsPlaced, FIELD_INTEGER ),
|
|
DEFINE_FIELD( m_StatsThisLevel.iNumStepsTaken, FIELD_INTEGER ),
|
|
DEFINE_FIELD( m_StatsThisLevel.fNumSecondsTaken, FIELD_FLOAT ),
|
|
DEFINE_FIELD( m_fTimeLastNumSecondsUpdate, FIELD_TIME ),
|
|
DEFINE_FIELD( m_iNumCamerasDetatched, FIELD_INTEGER ),
|
|
DEFINE_FIELD( m_bPitchReorientation, FIELD_BOOLEAN ),
|
|
DEFINE_FIELD( m_bIsRegenerating, FIELD_BOOLEAN ),
|
|
DEFINE_FIELD( m_fNeuroToxinDamageTime, FIELD_TIME ),
|
|
DEFINE_FIELD( m_hPortalEnvironment, FIELD_EHANDLE ),
|
|
DEFINE_FIELD( m_flExpressionLoopTime, FIELD_TIME ),
|
|
DEFINE_FIELD( m_iszExpressionScene, FIELD_STRING ),
|
|
DEFINE_FIELD( m_hExpressionSceneEnt, FIELD_EHANDLE ),
|
|
DEFINE_FIELD( m_vecTotalBulletForce, FIELD_VECTOR ),
|
|
DEFINE_FIELD( m_bSilentDropAndPickup, FIELD_BOOLEAN ),
|
|
DEFINE_FIELD( m_hRagdoll, FIELD_EHANDLE ),
|
|
DEFINE_FIELD( m_angEyeAngles, FIELD_VECTOR ),
|
|
DEFINE_FIELD( m_iPlayerSoundType, FIELD_INTEGER ),
|
|
DEFINE_FIELD( m_qPrePortalledViewAngles, FIELD_VECTOR ),
|
|
DEFINE_FIELD( m_bFixEyeAnglesFromPortalling, FIELD_BOOLEAN ),
|
|
DEFINE_FIELD( m_matLastPortalled, FIELD_VMATRIX_WORLDSPACE ),
|
|
DEFINE_FIELD( m_vWorldSpaceCenterHolder, FIELD_POSITION_VECTOR ),
|
|
DEFINE_FIELD( m_hSurroundingLiquidPortal, FIELD_EHANDLE ),
|
|
DEFINE_FIELD( m_bSuppressingCrosshair, FIELD_BOOLEAN ),
|
|
//DEFINE_FIELD ( m_PlayerAnimState, CPortalPlayerAnimState ),
|
|
//DEFINE_FIELD ( m_StatsThisLevel, PortalPlayerStatistics_t ),
|
|
|
|
DEFINE_EMBEDDEDBYREF( m_pExpresser ),
|
|
|
|
END_DATADESC()
|
|
|
|
ConVar sv_regeneration_wait_time ("sv_regeneration_wait_time", "1.0", FCVAR_REPLICATED );
|
|
|
|
const char *g_pszChellModel = "models/player/chell.mdl";
|
|
const char *g_pszPlayerModel = g_pszChellModel;
|
|
|
|
|
|
#define MAX_COMBINE_MODELS 4
|
|
#define MODEL_CHANGE_INTERVAL 5.0f
|
|
#define TEAM_CHANGE_INTERVAL 5.0f
|
|
|
|
#define PORTALPLAYER_PHYSDAMAGE_SCALE 4.0f
|
|
|
|
extern ConVar sv_turbophysics;
|
|
|
|
//----------------------------------------------------
|
|
// Player Physics Shadow
|
|
//----------------------------------------------------
|
|
#define VPHYS_MAX_DISTANCE 2.0
|
|
#define VPHYS_MAX_VEL 10
|
|
#define VPHYS_MAX_DISTSQR (VPHYS_MAX_DISTANCE*VPHYS_MAX_DISTANCE)
|
|
#define VPHYS_MAX_VELSQR (VPHYS_MAX_VEL*VPHYS_MAX_VEL)
|
|
|
|
|
|
extern float IntervalDistance( float x, float x0, float x1 );
|
|
|
|
//disable 'this' : used in base member initializer list
|
|
#pragma warning( disable : 4355 )
|
|
|
|
CPortal_Player::CPortal_Player()
|
|
{
|
|
|
|
m_PlayerAnimState = CreatePortalPlayerAnimState( this );
|
|
CreateExpresser();
|
|
|
|
UseClientSideAnimation();
|
|
|
|
m_angEyeAngles.Init();
|
|
|
|
m_iLastWeaponFireUsercmd = 0;
|
|
|
|
m_iSpawnInterpCounter = 0;
|
|
|
|
m_bHeldObjectOnOppositeSideOfPortal = false;
|
|
m_pHeldObjectPortal = 0;
|
|
|
|
m_bIntersectingPortalPlane = false;
|
|
|
|
m_bPitchReorientation = false;
|
|
|
|
m_bSilentDropAndPickup = false;
|
|
|
|
m_iszExpressionScene = NULL_STRING;
|
|
m_hExpressionSceneEnt = NULL;
|
|
m_flExpressionLoopTime = 0.0f;
|
|
m_bSuppressingCrosshair = false;
|
|
}
|
|
|
|
CPortal_Player::~CPortal_Player( void )
|
|
{
|
|
ClearSceneEvents( NULL, true );
|
|
|
|
if ( m_PlayerAnimState )
|
|
m_PlayerAnimState->Release();
|
|
|
|
CPortalRagdoll *pRagdoll = dynamic_cast<CPortalRagdoll*>( m_hRagdoll.Get() );
|
|
if( pRagdoll )
|
|
{
|
|
UTIL_Remove( pRagdoll );
|
|
}
|
|
}
|
|
|
|
void CPortal_Player::UpdateOnRemove( void )
|
|
{
|
|
BaseClass::UpdateOnRemove();
|
|
}
|
|
|
|
void CPortal_Player::Precache( void )
|
|
{
|
|
BaseClass::Precache();
|
|
|
|
PrecacheScriptSound( "PortalPlayer.EnterPortal" );
|
|
PrecacheScriptSound( "PortalPlayer.ExitPortal" );
|
|
|
|
PrecacheScriptSound( "PortalPlayer.Woosh" );
|
|
PrecacheScriptSound( "PortalPlayer.FallRecover" );
|
|
|
|
PrecacheModel ( "sprites/glow01.vmt" );
|
|
|
|
//Precache Citizen models
|
|
PrecacheModel( g_pszPlayerModel );
|
|
PrecacheModel( g_pszChellModel );
|
|
|
|
PrecacheScriptSound( "NPC_Citizen.die" );
|
|
}
|
|
|
|
void CPortal_Player::CreateSounds()
|
|
{
|
|
if ( !m_pWooshSound )
|
|
{
|
|
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
|
|
|
|
CPASAttenuationFilter filter( this );
|
|
|
|
m_pWooshSound = controller.SoundCreate( filter, entindex(), "PortalPlayer.Woosh" );
|
|
controller.Play( m_pWooshSound, 0, 100 );
|
|
}
|
|
}
|
|
|
|
void CPortal_Player::StopLoopingSounds()
|
|
{
|
|
if ( m_pWooshSound )
|
|
{
|
|
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
|
|
|
|
controller.SoundDestroy( m_pWooshSound );
|
|
m_pWooshSound = NULL;
|
|
}
|
|
|
|
BaseClass::StopLoopingSounds();
|
|
}
|
|
|
|
void CPortal_Player::GiveAllItems( void )
|
|
{
|
|
EquipSuit();
|
|
|
|
CBasePlayer::GiveAmmo( 255, "Pistol");
|
|
CBasePlayer::GiveAmmo( 32, "357" );
|
|
|
|
CBasePlayer::GiveAmmo( 255, "AR2" );
|
|
CBasePlayer::GiveAmmo( 3, "AR2AltFire" );
|
|
CBasePlayer::GiveAmmo( 255, "SMG1");
|
|
CBasePlayer::GiveAmmo( 3, "smg1_grenade");
|
|
|
|
CBasePlayer::GiveAmmo( 255, "Buckshot");
|
|
CBasePlayer::GiveAmmo( 16, "XBowBolt" );
|
|
|
|
CBasePlayer::GiveAmmo( 3, "rpg_round");
|
|
CBasePlayer::GiveAmmo( 6, "grenade" );
|
|
|
|
GiveNamedItem( "weapon_crowbar" );
|
|
GiveNamedItem( "weapon_physcannon" );
|
|
|
|
GiveNamedItem( "weapon_pistol" );
|
|
GiveNamedItem( "weapon_357" );
|
|
|
|
GiveNamedItem( "weapon_smg1" );
|
|
GiveNamedItem( "weapon_ar2" );
|
|
|
|
GiveNamedItem( "weapon_shotgun" );
|
|
GiveNamedItem( "weapon_crossbow" );
|
|
|
|
GiveNamedItem( "weapon_rpg" );
|
|
GiveNamedItem( "weapon_frag" );
|
|
|
|
GiveNamedItem( "weapon_bugbait" );
|
|
|
|
//GiveNamedItem( "weapon_physcannon" );
|
|
CWeaponPortalgun *pPortalGun = static_cast<CWeaponPortalgun*>( GiveNamedItem( "weapon_portalgun" ) );
|
|
|
|
if ( !pPortalGun )
|
|
{
|
|
pPortalGun = static_cast<CWeaponPortalgun*>( Weapon_OwnsThisType( "weapon_portalgun" ) );
|
|
}
|
|
|
|
if ( pPortalGun )
|
|
{
|
|
pPortalGun->SetCanFirePortal1();
|
|
pPortalGun->SetCanFirePortal2();
|
|
}
|
|
}
|
|
|
|
void CPortal_Player::GiveDefaultItems( void )
|
|
{
|
|
castable_string_t st( "suit_no_sprint" );
|
|
GlobalEntity_SetState( st, GLOBAL_OFF );
|
|
inputdata_t in;
|
|
InputDisableFlashlight( in );
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Sets specific defaults.
|
|
//-----------------------------------------------------------------------------
|
|
void CPortal_Player::Spawn(void)
|
|
{
|
|
SetPlayerModel();
|
|
|
|
BaseClass::Spawn();
|
|
|
|
CreateSounds();
|
|
|
|
pl.deadflag = false;
|
|
RemoveSolidFlags( FSOLID_NOT_SOLID );
|
|
|
|
RemoveEffects( EF_NODRAW );
|
|
StopObserverMode();
|
|
|
|
GiveDefaultItems();
|
|
|
|
m_nRenderFX = kRenderNormal;
|
|
|
|
m_Local.m_iHideHUD = 0;
|
|
|
|
AddFlag(FL_ONGROUND); // set the player on the ground at the start of the round.
|
|
|
|
m_impactEnergyScale = PORTALPLAYER_PHYSDAMAGE_SCALE;
|
|
|
|
RemoveFlag( FL_FROZEN );
|
|
|
|
m_iSpawnInterpCounter = (m_iSpawnInterpCounter + 1) % 8;
|
|
|
|
m_Local.m_bDucked = false;
|
|
|
|
SetPlayerUnderwater(false);
|
|
|
|
#ifdef PORTAL_MP
|
|
PickTeam();
|
|
#endif
|
|
}
|
|
|
|
void CPortal_Player::Activate( void )
|
|
{
|
|
BaseClass::Activate();
|
|
m_fTimeLastNumSecondsUpdate = gpGlobals->curtime;
|
|
}
|
|
|
|
void CPortal_Player::NotifySystemEvent(CBaseEntity *pNotify, notify_system_event_t eventType, const notify_system_event_params_t ¶ms )
|
|
{
|
|
// On teleport, we send event for tracking fling achievements
|
|
if ( eventType == NOTIFY_EVENT_TELEPORT )
|
|
{
|
|
CProp_Portal *pEnteredPortal = dynamic_cast<CProp_Portal*>( pNotify );
|
|
IGameEvent *event = gameeventmanager->CreateEvent( "portal_player_portaled" );
|
|
if ( event )
|
|
{
|
|
event->SetInt( "userid", GetUserID() );
|
|
event->SetBool( "portal2", pEnteredPortal->m_bIsPortal2 );
|
|
gameeventmanager->FireEvent( event );
|
|
}
|
|
}
|
|
|
|
BaseClass::NotifySystemEvent( pNotify, eventType, params );
|
|
}
|
|
|
|
void CPortal_Player::OnRestore( void )
|
|
{
|
|
BaseClass::OnRestore();
|
|
if ( m_pExpresser )
|
|
{
|
|
m_pExpresser->SetOuter ( this );
|
|
}
|
|
}
|
|
|
|
//bool CPortal_Player::StartObserverMode( int mode )
|
|
//{
|
|
// //Do nothing.
|
|
//
|
|
// return false;
|
|
//}
|
|
|
|
bool CPortal_Player::ValidatePlayerModel( const char *pModel )
|
|
{
|
|
if ( !Q_stricmp( g_pszPlayerModel, pModel ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if ( !Q_stricmp( g_pszChellModel, pModel ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void CPortal_Player::SetPlayerModel( void )
|
|
{
|
|
const char *szModelName = NULL;
|
|
const char *pszCurrentModelName = modelinfo->GetModelName( GetModel());
|
|
|
|
szModelName = engine->GetClientConVarValue( engine->IndexOfEdict( edict() ), "cl_playermodel" );
|
|
|
|
if ( ValidatePlayerModel( szModelName ) == false )
|
|
{
|
|
char szReturnString[512];
|
|
|
|
if ( ValidatePlayerModel( pszCurrentModelName ) == false )
|
|
{
|
|
pszCurrentModelName = g_pszPlayerModel;
|
|
}
|
|
|
|
Q_snprintf( szReturnString, sizeof (szReturnString ), "cl_playermodel %s\n", pszCurrentModelName );
|
|
engine->ClientCommand ( edict(), szReturnString );
|
|
|
|
szModelName = pszCurrentModelName;
|
|
}
|
|
|
|
int modelIndex = modelinfo->GetModelIndex( szModelName );
|
|
|
|
if ( modelIndex == -1 )
|
|
{
|
|
szModelName = g_pszPlayerModel;
|
|
|
|
char szReturnString[512];
|
|
|
|
Q_snprintf( szReturnString, sizeof (szReturnString ), "cl_playermodel %s\n", szModelName );
|
|
engine->ClientCommand ( edict(), szReturnString );
|
|
}
|
|
|
|
SetModel( szModelName );
|
|
m_iPlayerSoundType = (int)PLAYER_SOUNDS_CITIZEN;
|
|
}
|
|
|
|
|
|
bool CPortal_Player::Weapon_Switch( CBaseCombatWeapon *pWeapon, int viewmodelindex )
|
|
{
|
|
bool bRet = BaseClass::Weapon_Switch( pWeapon, viewmodelindex );
|
|
|
|
return bRet;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
void CPortal_Player::UpdateExpression( void )
|
|
{
|
|
if ( !m_pExpresser )
|
|
return;
|
|
|
|
int iConcept = CONCEPT_CHELL_IDLE;
|
|
if ( GetHealth() <= 0 )
|
|
{
|
|
iConcept = CONCEPT_CHELL_DEAD;
|
|
}
|
|
|
|
GetExpresser()->SetOuter( this );
|
|
|
|
ClearExpression();
|
|
AI_Response *response = SpeakFindResponse( g_pszChellConcepts[iConcept] );
|
|
if ( !response )
|
|
{
|
|
m_flExpressionLoopTime = gpGlobals->curtime + RandomFloat(30,40);
|
|
return;
|
|
}
|
|
|
|
char szScene[256] = { 0 };
|
|
response->GetResponse( szScene, sizeof(szScene) );
|
|
|
|
// Ignore updates that choose the same scene
|
|
if ( m_iszExpressionScene != NULL_STRING && stricmp( STRING(m_iszExpressionScene), szScene ) == 0 )
|
|
return;
|
|
|
|
if ( m_hExpressionSceneEnt )
|
|
{
|
|
ClearExpression();
|
|
}
|
|
|
|
m_iszExpressionScene = AllocPooledString( szScene );
|
|
float flDuration = InstancedScriptedScene( this, szScene, &m_hExpressionSceneEnt, 0.0, true, NULL );
|
|
m_flExpressionLoopTime = gpGlobals->curtime + flDuration;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
void CPortal_Player::ClearExpression( void )
|
|
{
|
|
if ( m_hExpressionSceneEnt != NULL )
|
|
{
|
|
StopScriptedScene( this, m_hExpressionSceneEnt );
|
|
}
|
|
m_flExpressionLoopTime = gpGlobals->curtime;
|
|
}
|
|
|
|
|
|
void CPortal_Player::PreThink( void )
|
|
{
|
|
QAngle vOldAngles = GetLocalAngles();
|
|
QAngle vTempAngles = GetLocalAngles();
|
|
|
|
vTempAngles = EyeAngles();
|
|
|
|
if ( vTempAngles[PITCH] > 180.0f )
|
|
{
|
|
vTempAngles[PITCH] -= 360.0f;
|
|
}
|
|
|
|
SetLocalAngles( vTempAngles );
|
|
|
|
BaseClass::PreThink();
|
|
|
|
if( (m_afButtonPressed & IN_JUMP) )
|
|
{
|
|
Jump();
|
|
}
|
|
|
|
//Reset bullet force accumulator, only lasts one frame
|
|
m_vecTotalBulletForce = vec3_origin;
|
|
|
|
SetLocalAngles( vOldAngles );
|
|
}
|
|
|
|
void CPortal_Player::PostThink( void )
|
|
{
|
|
BaseClass::PostThink();
|
|
|
|
// Store the eye angles pitch so the client can compute its animation state correctly.
|
|
m_angEyeAngles = EyeAngles();
|
|
|
|
QAngle angles = GetLocalAngles();
|
|
angles[PITCH] = 0;
|
|
SetLocalAngles( angles );
|
|
|
|
// Regenerate heath after 3 seconds
|
|
if ( IsAlive() && GetHealth() < GetMaxHealth() )
|
|
{
|
|
// Color to overlay on the screen while the player is taking damage
|
|
color32 hurtScreenOverlay = {64,0,0,64};
|
|
|
|
if ( gpGlobals->curtime > m_fTimeLastHurt + sv_regeneration_wait_time.GetFloat() )
|
|
{
|
|
TakeHealth( 1, DMG_GENERIC );
|
|
m_bIsRegenerating = true;
|
|
|
|
if ( GetHealth() >= GetMaxHealth() )
|
|
{
|
|
m_bIsRegenerating = false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_bIsRegenerating = false;
|
|
UTIL_ScreenFade( this, hurtScreenOverlay, 1.0f, 0.1f, FFADE_IN|FFADE_PURGE );
|
|
}
|
|
}
|
|
|
|
UpdatePortalPlaneSounds();
|
|
UpdateWooshSounds();
|
|
|
|
m_PlayerAnimState->Update( m_angEyeAngles[YAW], m_angEyeAngles[PITCH] );
|
|
|
|
if ( IsAlive() && m_flExpressionLoopTime >= 0 && gpGlobals->curtime > m_flExpressionLoopTime )
|
|
{
|
|
// Random expressions need to be cleared, because they don't loop. So if we
|
|
// pick the same one again, we want to restart it.
|
|
ClearExpression();
|
|
m_iszExpressionScene = NULL_STRING;
|
|
UpdateExpression();
|
|
}
|
|
|
|
UpdateSecondsTaken();
|
|
|
|
// Try to fix the player if they're stuck
|
|
if ( m_bStuckOnPortalCollisionObject )
|
|
{
|
|
Vector vForward = ((CProp_Portal*)m_hPortalEnvironment.Get())->m_vPrevForward;
|
|
Vector vNewPos = GetAbsOrigin() + vForward * gpGlobals->frametime * -1000.0f;
|
|
Teleport( &vNewPos, NULL, &vForward );
|
|
m_bStuckOnPortalCollisionObject = false;
|
|
}
|
|
}
|
|
|
|
void CPortal_Player::PlayerDeathThink(void)
|
|
{
|
|
float flForward;
|
|
|
|
SetNextThink( gpGlobals->curtime + 0.1f );
|
|
|
|
if (GetFlags() & FL_ONGROUND)
|
|
{
|
|
flForward = GetAbsVelocity().Length() - 20;
|
|
if (flForward <= 0)
|
|
{
|
|
SetAbsVelocity( vec3_origin );
|
|
}
|
|
else
|
|
{
|
|
Vector vecNewVelocity = GetAbsVelocity();
|
|
VectorNormalize( vecNewVelocity );
|
|
vecNewVelocity *= flForward;
|
|
SetAbsVelocity( vecNewVelocity );
|
|
}
|
|
}
|
|
|
|
if ( HasWeapons() )
|
|
{
|
|
// we drop the guns here because weapons that have an area effect and can kill their user
|
|
// will sometimes crash coming back from CBasePlayer::Killed() if they kill their owner because the
|
|
// player class sometimes is freed. It's safer to manipulate the weapons once we know
|
|
// we aren't calling into any of their code anymore through the player pointer.
|
|
PackDeadPlayerItems();
|
|
}
|
|
|
|
if (GetModelIndex() && (!IsSequenceFinished()) && (m_lifeState == LIFE_DYING))
|
|
{
|
|
StudioFrameAdvance( );
|
|
|
|
m_iRespawnFrames++;
|
|
if ( m_iRespawnFrames < 60 ) // animations should be no longer than this
|
|
return;
|
|
}
|
|
|
|
if (m_lifeState == LIFE_DYING)
|
|
m_lifeState = LIFE_DEAD;
|
|
|
|
StopAnimation();
|
|
|
|
IncrementInterpolationFrame();
|
|
m_flPlaybackRate = 0.0;
|
|
|
|
int fAnyButtonDown = (m_nButtons & ~IN_SCORE);
|
|
|
|
// Strip out the duck key from this check if it's toggled
|
|
if ( (fAnyButtonDown & IN_DUCK) && GetToggledDuckState())
|
|
{
|
|
fAnyButtonDown &= ~IN_DUCK;
|
|
}
|
|
|
|
// wait for all buttons released
|
|
if ( m_lifeState == LIFE_DEAD )
|
|
{
|
|
if ( fAnyButtonDown || gpGlobals->curtime < m_flDeathTime + DEATH_ANIMATION_TIME )
|
|
return;
|
|
|
|
if ( g_pGameRules->FPlayerCanRespawn( this ) )
|
|
{
|
|
m_lifeState = LIFE_RESPAWNABLE;
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
// if the player has been dead for one second longer than allowed by forcerespawn,
|
|
// forcerespawn isn't on. Send the player off to an intermission camera until they
|
|
// choose to respawn.
|
|
if ( g_pGameRules->IsMultiplayer() && ( gpGlobals->curtime > (m_flDeathTime + DEATH_ANIMATION_TIME) ) && !IsObserver() )
|
|
{
|
|
// go to dead camera.
|
|
StartObserverMode( m_iObserverLastMode );
|
|
}
|
|
|
|
// wait for any button down, or mp_forcerespawn is set and the respawn time is up
|
|
if (!fAnyButtonDown
|
|
&& !( g_pGameRules->IsMultiplayer() && forcerespawn.GetInt() > 0 && (gpGlobals->curtime > (m_flDeathTime + 5))) )
|
|
return;
|
|
|
|
m_nButtons = 0;
|
|
m_iRespawnFrames = 0;
|
|
|
|
//Msg( "Respawn\n");
|
|
|
|
respawn( this, !IsObserver() );// don't copy a corpse if we're in deathcam.
|
|
SetNextThink( TICK_NEVER_THINK );
|
|
}
|
|
|
|
void CPortal_Player::UpdatePortalPlaneSounds( void )
|
|
{
|
|
CProp_Portal *pPortal = m_hPortalEnvironment;
|
|
if ( pPortal && pPortal->m_bActivated )
|
|
{
|
|
Vector vVelocity;
|
|
GetVelocity( &vVelocity, NULL );
|
|
|
|
if ( !vVelocity.IsZero() )
|
|
{
|
|
Vector vMin, vMax;
|
|
CollisionProp()->WorldSpaceAABB( &vMin, &vMax );
|
|
|
|
Vector vEarCenter = ( vMax + vMin ) / 2.0f;
|
|
Vector vDiagonal = vMax - vMin;
|
|
|
|
if ( !m_bIntersectingPortalPlane )
|
|
{
|
|
vDiagonal *= 0.25f;
|
|
|
|
if ( UTIL_IsBoxIntersectingPortal( vEarCenter, vDiagonal, pPortal ) )
|
|
{
|
|
m_bIntersectingPortalPlane = true;
|
|
|
|
CPASAttenuationFilter filter( this );
|
|
CSoundParameters params;
|
|
if ( GetParametersForSound( "PortalPlayer.EnterPortal", params, NULL ) )
|
|
{
|
|
EmitSound_t ep( params );
|
|
ep.m_nPitch = 80.0f + vVelocity.Length() * 0.03f;
|
|
ep.m_flVolume = MIN( 0.3f + vVelocity.Length() * 0.00075f, 1.0f );
|
|
|
|
EmitSound( filter, entindex(), ep );
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
vDiagonal *= 0.30f;
|
|
|
|
if ( !UTIL_IsBoxIntersectingPortal( vEarCenter, vDiagonal, pPortal ) )
|
|
{
|
|
m_bIntersectingPortalPlane = false;
|
|
|
|
CPASAttenuationFilter filter( this );
|
|
CSoundParameters params;
|
|
if ( GetParametersForSound( "PortalPlayer.ExitPortal", params, NULL ) )
|
|
{
|
|
EmitSound_t ep( params );
|
|
ep.m_nPitch = 80.0f + vVelocity.Length() * 0.03f;
|
|
ep.m_flVolume = MIN( 0.3f + vVelocity.Length() * 0.00075f, 1.0f );
|
|
|
|
EmitSound( filter, entindex(), ep );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if ( m_bIntersectingPortalPlane )
|
|
{
|
|
m_bIntersectingPortalPlane = false;
|
|
|
|
CPASAttenuationFilter filter( this );
|
|
CSoundParameters params;
|
|
if ( GetParametersForSound( "PortalPlayer.ExitPortal", params, NULL ) )
|
|
{
|
|
EmitSound_t ep( params );
|
|
Vector vVelocity;
|
|
GetVelocity( &vVelocity, NULL );
|
|
ep.m_nPitch = 80.0f + vVelocity.Length() * 0.03f;
|
|
ep.m_flVolume = MIN( 0.3f + vVelocity.Length() * 0.00075f, 1.0f );
|
|
|
|
EmitSound( filter, entindex(), ep );
|
|
}
|
|
}
|
|
}
|
|
|
|
void CPortal_Player::UpdateWooshSounds( void )
|
|
{
|
|
if ( m_pWooshSound )
|
|
{
|
|
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
|
|
|
|
float fWooshVolume = GetAbsVelocity().Length() - MIN_FLING_SPEED;
|
|
|
|
if ( fWooshVolume < 0.0f )
|
|
{
|
|
controller.SoundChangeVolume( m_pWooshSound, 0.0f, 0.1f );
|
|
return;
|
|
}
|
|
|
|
fWooshVolume /= 2000.0f;
|
|
if ( fWooshVolume > 1.0f )
|
|
fWooshVolume = 1.0f;
|
|
|
|
controller.SoundChangeVolume( m_pWooshSound, fWooshVolume, 0.1f );
|
|
// controller.SoundChangePitch( m_pWooshSound, fWooshVolume + 0.5f, 0.1f );
|
|
}
|
|
}
|
|
|
|
void CPortal_Player::FireBullets ( const FireBulletsInfo_t &info )
|
|
{
|
|
NoteWeaponFired();
|
|
|
|
BaseClass::FireBullets( info );
|
|
}
|
|
|
|
void CPortal_Player::NoteWeaponFired( void )
|
|
{
|
|
Assert( m_pCurrentCommand );
|
|
if( m_pCurrentCommand )
|
|
{
|
|
m_iLastWeaponFireUsercmd = m_pCurrentCommand->command_number;
|
|
}
|
|
}
|
|
|
|
extern ConVar sv_maxunlag;
|
|
|
|
bool CPortal_Player::WantsLagCompensationOnEntity( const CBasePlayer *pPlayer, const CUserCmd *pCmd, const CBitVec<MAX_EDICTS> *pEntityTransmitBits ) const
|
|
{
|
|
// No need to lag compensate at all if we're not attacking in this command and
|
|
// we haven't attacked recently.
|
|
if ( !( pCmd->buttons & IN_ATTACK ) && (pCmd->command_number - m_iLastWeaponFireUsercmd > 5) )
|
|
return false;
|
|
|
|
// If this entity hasn't been transmitted to us and acked, then don't bother lag compensating it.
|
|
if ( pEntityTransmitBits && !pEntityTransmitBits->Get( pPlayer->entindex() ) )
|
|
return false;
|
|
|
|
const Vector &vMyOrigin = GetAbsOrigin();
|
|
const Vector &vHisOrigin = pPlayer->GetAbsOrigin();
|
|
|
|
// get max distance player could have moved within max lag compensation time,
|
|
// multiply by 1.5 to to avoid "dead zones" (sqrt(2) would be the exact value)
|
|
float maxDistance = 1.5 * pPlayer->MaxSpeed() * sv_maxunlag.GetFloat();
|
|
|
|
// If the player is within this distance, lag compensate them in case they're running past us.
|
|
if ( vHisOrigin.DistTo( vMyOrigin ) < maxDistance )
|
|
return true;
|
|
|
|
// If their origin is not within a 45 degree cone in front of us, no need to lag compensate.
|
|
Vector vForward;
|
|
AngleVectors( pCmd->viewangles, &vForward );
|
|
|
|
Vector vDiff = vHisOrigin - vMyOrigin;
|
|
VectorNormalize( vDiff );
|
|
|
|
float flCosAngle = 0.707107f; // 45 degree angle
|
|
if ( vForward.Dot( vDiff ) < flCosAngle )
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
void CPortal_Player::DoAnimationEvent( PlayerAnimEvent_t event, int nData )
|
|
{
|
|
m_PlayerAnimState->DoAnimationEvent( event, nData );
|
|
TE_PlayerAnimEvent( this, event, nData ); // Send to any clients who can see this guy.
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Override setup bones so that is uses the render angles from
|
|
// the Portal animation state to setup the hitboxes.
|
|
//-----------------------------------------------------------------------------
|
|
void CPortal_Player::SetupBones( matrix3x4_t *pBoneToWorld, int boneMask )
|
|
{
|
|
VPROF_BUDGET( "CBaseAnimating::SetupBones", VPROF_BUDGETGROUP_SERVER_ANIM );
|
|
|
|
// Set the mdl cache semaphore.
|
|
MDLCACHE_CRITICAL_SECTION();
|
|
|
|
// Get the studio header.
|
|
Assert( GetModelPtr() );
|
|
CStudioHdr *pStudioHdr = GetModelPtr( );
|
|
|
|
Vector pos[MAXSTUDIOBONES];
|
|
Quaternion q[MAXSTUDIOBONES];
|
|
|
|
// Adjust hit boxes based on IK driven offset.
|
|
Vector adjOrigin = GetAbsOrigin() + Vector( 0, 0, m_flEstIkOffset );
|
|
|
|
// FIXME: pass this into Studio_BuildMatrices to skip transforms
|
|
CBoneBitList boneComputed;
|
|
if ( m_pIk )
|
|
{
|
|
m_iIKCounter++;
|
|
m_pIk->Init( pStudioHdr, GetAbsAngles(), adjOrigin, gpGlobals->curtime, m_iIKCounter, boneMask );
|
|
GetSkeleton( pStudioHdr, pos, q, boneMask );
|
|
|
|
m_pIk->UpdateTargets( pos, q, pBoneToWorld, boneComputed );
|
|
CalculateIKLocks( gpGlobals->curtime );
|
|
m_pIk->SolveDependencies( pos, q, pBoneToWorld, boneComputed );
|
|
}
|
|
else
|
|
{
|
|
GetSkeleton( pStudioHdr, pos, q, boneMask );
|
|
}
|
|
|
|
CBaseAnimating *pParent = dynamic_cast< CBaseAnimating* >( GetMoveParent() );
|
|
if ( pParent )
|
|
{
|
|
// We're doing bone merging, so do special stuff here.
|
|
CBoneCache *pParentCache = pParent->GetBoneCache();
|
|
if ( pParentCache )
|
|
{
|
|
BuildMatricesWithBoneMerge(
|
|
pStudioHdr,
|
|
m_PlayerAnimState->GetRenderAngles(),
|
|
adjOrigin,
|
|
pos,
|
|
q,
|
|
pBoneToWorld,
|
|
pParent,
|
|
pParentCache );
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
Studio_BuildMatrices(
|
|
pStudioHdr,
|
|
m_PlayerAnimState->GetRenderAngles(),
|
|
adjOrigin,
|
|
pos,
|
|
q,
|
|
-1,
|
|
GetModelScale(), // Scaling
|
|
pBoneToWorld,
|
|
boneMask );
|
|
}
|
|
|
|
|
|
// Set the activity based on an event or current state
|
|
void CPortal_Player::SetAnimation( PLAYER_ANIM playerAnim )
|
|
{
|
|
return;
|
|
}
|
|
|
|
CAI_Expresser *CPortal_Player::CreateExpresser()
|
|
{
|
|
Assert( !m_pExpresser );
|
|
|
|
if ( m_pExpresser )
|
|
{
|
|
delete m_pExpresser;
|
|
}
|
|
|
|
m_pExpresser = new CAI_Expresser(this);
|
|
if ( !m_pExpresser)
|
|
{
|
|
return NULL;
|
|
}
|
|
m_pExpresser->Connect(this);
|
|
|
|
return m_pExpresser;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
CAI_Expresser *CPortal_Player::GetExpresser()
|
|
{
|
|
if ( m_pExpresser )
|
|
{
|
|
m_pExpresser->Connect(this);
|
|
}
|
|
return m_pExpresser;
|
|
}
|
|
|
|
|
|
extern int gEvilImpulse101;
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Player reacts to bumping a weapon.
|
|
// Input : pWeapon - the weapon that the player bumped into.
|
|
// Output : Returns true if player picked up the weapon
|
|
//-----------------------------------------------------------------------------
|
|
bool CPortal_Player::BumpWeapon( CBaseCombatWeapon *pWeapon )
|
|
{
|
|
CBaseCombatCharacter *pOwner = pWeapon->GetOwner();
|
|
|
|
// Can I have this weapon type?
|
|
if ( !IsAllowedToPickupWeapons() )
|
|
return false;
|
|
|
|
if ( pOwner || !Weapon_CanUse( pWeapon ) || !g_pGameRules->CanHavePlayerItem( this, pWeapon ) )
|
|
{
|
|
if ( gEvilImpulse101 )
|
|
{
|
|
UTIL_Remove( pWeapon );
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// Don't let the player fetch weapons through walls (use MASK_SOLID so that you can't pickup through windows)
|
|
if( !pWeapon->FVisible( this, MASK_SOLID ) && !(GetFlags() & FL_NOTARGET) )
|
|
{
|
|
return false;
|
|
}
|
|
|
|
CWeaponPortalgun *pPickupPortalgun = dynamic_cast<CWeaponPortalgun*>( pWeapon );
|
|
|
|
bool bOwnsWeaponAlready = !!Weapon_OwnsThisType( pWeapon->GetClassname(), pWeapon->GetSubType());
|
|
|
|
if ( bOwnsWeaponAlready == true )
|
|
{
|
|
// If we picked up a second portal gun set the bool to alow secondary fire
|
|
if ( pPickupPortalgun )
|
|
{
|
|
CWeaponPortalgun *pPortalGun = static_cast<CWeaponPortalgun*>( Weapon_OwnsThisType( pWeapon->GetClassname() ) );
|
|
|
|
if ( pPickupPortalgun->CanFirePortal1() )
|
|
pPortalGun->SetCanFirePortal1();
|
|
|
|
if ( pPickupPortalgun->CanFirePortal2() )
|
|
pPortalGun->SetCanFirePortal2();
|
|
|
|
UTIL_Remove( pWeapon );
|
|
return true;
|
|
}
|
|
|
|
//If we have room for the ammo, then "take" the weapon too.
|
|
if ( Weapon_EquipAmmoOnly( pWeapon ) )
|
|
{
|
|
pWeapon->CheckRespawn();
|
|
|
|
UTIL_Remove( pWeapon );
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
pWeapon->CheckRespawn();
|
|
Weapon_Equip( pWeapon );
|
|
|
|
// If we're holding and object before picking up portalgun, drop it
|
|
if ( pPickupPortalgun )
|
|
{
|
|
ForceDropOfCarriedPhysObjects( GetPlayerHeldEntity( this ) );
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void CPortal_Player::ShutdownUseEntity( void )
|
|
{
|
|
ShutdownPickupController( m_hUseEntity );
|
|
}
|
|
|
|
const Vector& CPortal_Player::WorldSpaceCenter( ) const
|
|
{
|
|
m_vWorldSpaceCenterHolder = GetAbsOrigin();
|
|
m_vWorldSpaceCenterHolder.z += ( (IsDucked()) ? (VEC_DUCK_HULL_MAX_SCALED( this ).z) : (VEC_HULL_MAX_SCALED( this ).z) ) * 0.5f;
|
|
return m_vWorldSpaceCenterHolder;
|
|
}
|
|
|
|
void CPortal_Player::Teleport( const Vector *newPosition, const QAngle *newAngles, const Vector *newVelocity )
|
|
{
|
|
Vector oldOrigin = GetLocalOrigin();
|
|
QAngle oldAngles = GetLocalAngles();
|
|
BaseClass::Teleport( newPosition, newAngles, newVelocity );
|
|
m_angEyeAngles = pl.v_angle;
|
|
|
|
m_PlayerAnimState->Teleport( newPosition, newAngles, this );
|
|
}
|
|
|
|
void CPortal_Player::VPhysicsShadowUpdate( IPhysicsObject *pPhysics )
|
|
{
|
|
if( m_hPortalEnvironment.Get() == NULL )
|
|
return BaseClass::VPhysicsShadowUpdate( pPhysics );
|
|
|
|
|
|
//below is mostly a cut/paste of existing CBasePlayer::VPhysicsShadowUpdate code with some minor tweaks to avoid getting stuck in stuff when in a portal environment
|
|
if ( sv_turbophysics.GetBool() )
|
|
return;
|
|
|
|
Vector newPosition;
|
|
|
|
bool physicsUpdated = m_pPhysicsController->GetShadowPosition( &newPosition, NULL ) > 0 ? true : false;
|
|
|
|
// UNDONE: If the player is penetrating, but the player's game collisions are not stuck, teleport the physics shadow to the game position
|
|
if ( pPhysics->GetGameFlags() & FVPHYSICS_PENETRATING )
|
|
{
|
|
CUtlVector<CBaseEntity *> list;
|
|
PhysGetListOfPenetratingEntities( this, list );
|
|
for ( int i = list.Count()-1; i >= 0; --i )
|
|
{
|
|
// filter out anything that isn't simulated by vphysics
|
|
// UNDONE: Filter out motion disabled objects?
|
|
if ( list[i]->GetMoveType() == MOVETYPE_VPHYSICS )
|
|
{
|
|
// I'm currently stuck inside a moving object, so allow vphysics to
|
|
// apply velocity to the player in order to separate these objects
|
|
m_touchedPhysObject = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( m_pPhysicsController->IsInContact() || (m_afPhysicsFlags & PFLAG_VPHYSICS_MOTIONCONTROLLER) )
|
|
{
|
|
m_touchedPhysObject = true;
|
|
}
|
|
|
|
if ( IsFollowingPhysics() )
|
|
{
|
|
m_touchedPhysObject = true;
|
|
}
|
|
|
|
if ( GetMoveType() == MOVETYPE_NOCLIP )
|
|
{
|
|
m_oldOrigin = GetAbsOrigin();
|
|
return;
|
|
}
|
|
|
|
if ( phys_timescale.GetFloat() == 0.0f )
|
|
{
|
|
physicsUpdated = false;
|
|
}
|
|
|
|
if ( !physicsUpdated )
|
|
return;
|
|
|
|
IPhysicsObject *pPhysGround = GetGroundVPhysics();
|
|
|
|
Vector newVelocity;
|
|
pPhysics->GetPosition( &newPosition, 0 );
|
|
m_pPhysicsController->GetShadowVelocity( &newVelocity );
|
|
|
|
|
|
|
|
Vector tmp = GetAbsOrigin() - newPosition;
|
|
if ( !m_touchedPhysObject && !(GetFlags() & FL_ONGROUND) )
|
|
{
|
|
tmp.z *= 0.5f; // don't care about z delta as much
|
|
}
|
|
|
|
float dist = tmp.LengthSqr();
|
|
float deltaV = (newVelocity - GetAbsVelocity()).LengthSqr();
|
|
|
|
float maxDistErrorSqr = VPHYS_MAX_DISTSQR;
|
|
float maxVelErrorSqr = VPHYS_MAX_VELSQR;
|
|
if ( IsRideablePhysics(pPhysGround) )
|
|
{
|
|
maxDistErrorSqr *= 0.25;
|
|
maxVelErrorSqr *= 0.25;
|
|
}
|
|
|
|
if ( dist >= maxDistErrorSqr || deltaV >= maxVelErrorSqr || (pPhysGround && !m_touchedPhysObject) )
|
|
{
|
|
if ( m_touchedPhysObject || pPhysGround )
|
|
{
|
|
// BUGBUG: Rewrite this code using fixed timestep
|
|
if ( deltaV >= maxVelErrorSqr )
|
|
{
|
|
Vector dir = GetAbsVelocity();
|
|
float len = VectorNormalize(dir);
|
|
float dot = DotProduct( newVelocity, dir );
|
|
if ( dot > len )
|
|
{
|
|
dot = len;
|
|
}
|
|
else if ( dot < -len )
|
|
{
|
|
dot = -len;
|
|
}
|
|
|
|
VectorMA( newVelocity, -dot, dir, newVelocity );
|
|
|
|
if ( m_afPhysicsFlags & PFLAG_VPHYSICS_MOTIONCONTROLLER )
|
|
{
|
|
float val = Lerp( 0.1f, len, dot );
|
|
VectorMA( newVelocity, val - len, dir, newVelocity );
|
|
}
|
|
|
|
if ( !IsRideablePhysics(pPhysGround) )
|
|
{
|
|
if ( !(m_afPhysicsFlags & PFLAG_VPHYSICS_MOTIONCONTROLLER ) && IsSimulatingOnAlternateTicks() )
|
|
{
|
|
newVelocity *= 0.5f;
|
|
}
|
|
ApplyAbsVelocityImpulse( newVelocity );
|
|
}
|
|
}
|
|
|
|
trace_t trace;
|
|
UTIL_TraceEntity( this, newPosition, newPosition, MASK_PLAYERSOLID, this, COLLISION_GROUP_PLAYER_MOVEMENT, &trace );
|
|
if ( !trace.allsolid && !trace.startsolid )
|
|
{
|
|
SetAbsOrigin( newPosition );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
trace_t trace;
|
|
|
|
Ray_t ray;
|
|
ray.Init( GetAbsOrigin(), GetAbsOrigin(), WorldAlignMins(), WorldAlignMaxs() );
|
|
|
|
CTraceFilterSimple OriginalTraceFilter( this, COLLISION_GROUP_PLAYER_MOVEMENT );
|
|
CTraceFilterTranslateClones traceFilter( &OriginalTraceFilter );
|
|
UTIL_Portal_TraceRay_With( m_hPortalEnvironment, ray, MASK_PLAYERSOLID, &traceFilter, &trace );
|
|
|
|
// current position is not ok, fixup
|
|
if ( trace.allsolid || trace.startsolid )
|
|
{
|
|
//try again with new position
|
|
ray.Init( newPosition, newPosition, WorldAlignMins(), WorldAlignMaxs() );
|
|
UTIL_Portal_TraceRay_With( m_hPortalEnvironment, ray, MASK_PLAYERSOLID, &traceFilter, &trace );
|
|
|
|
if( trace.startsolid == false )
|
|
{
|
|
SetAbsOrigin( newPosition );
|
|
}
|
|
else
|
|
{
|
|
if( !FindClosestPassableSpace( this, newPosition - GetAbsOrigin(), MASK_PLAYERSOLID ) )
|
|
{
|
|
// Try moving the player closer to the center of the portal
|
|
CProp_Portal *pPortal = m_hPortalEnvironment.Get();
|
|
newPosition += ( pPortal->GetAbsOrigin() - WorldSpaceCenter() ) * 0.1f;
|
|
SetAbsOrigin( newPosition );
|
|
|
|
DevMsg( "Hurting the player for FindClosestPassableSpaceFailure!" );
|
|
|
|
// Deal 1 damage per frame... this will kill a player very fast, but allow for the above correction to fix some cases
|
|
CTakeDamageInfo info( this, this, vec3_origin, vec3_origin, 1, DMG_CRUSH );
|
|
OnTakeDamage( info );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( m_touchedPhysObject )
|
|
{
|
|
// check my position (physics object could have simulated into my position
|
|
// physics is not very far away, check my position
|
|
trace_t trace;
|
|
UTIL_TraceEntity( this, GetAbsOrigin(), GetAbsOrigin(),
|
|
MASK_PLAYERSOLID, this, COLLISION_GROUP_PLAYER_MOVEMENT, &trace );
|
|
|
|
// is current position ok?
|
|
if ( trace.allsolid || trace.startsolid )
|
|
{
|
|
// stuck????!?!?
|
|
//Msg("Stuck on %s\n", trace.m_pEnt->GetClassname());
|
|
SetAbsOrigin( newPosition );
|
|
UTIL_TraceEntity( this, GetAbsOrigin(), GetAbsOrigin(),
|
|
MASK_PLAYERSOLID, this, COLLISION_GROUP_PLAYER_MOVEMENT, &trace );
|
|
if ( trace.allsolid || trace.startsolid )
|
|
{
|
|
//Msg("Double Stuck\n");
|
|
SetAbsOrigin( m_oldOrigin );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
m_oldOrigin = GetAbsOrigin();
|
|
}
|
|
|
|
bool CPortal_Player::UseFoundEntity( CBaseEntity *pUseEntity )
|
|
{
|
|
bool usedSomething = false;
|
|
|
|
//!!!UNDONE: traceline here to prevent +USEing buttons through walls
|
|
int caps = pUseEntity->ObjectCaps();
|
|
variant_t emptyVariant;
|
|
|
|
if ( m_afButtonPressed & IN_USE )
|
|
{
|
|
// Robin: Don't play sounds for NPCs, because NPCs will allow respond with speech.
|
|
if ( !pUseEntity->MyNPCPointer() )
|
|
{
|
|
EmitSound( "HL2Player.Use" );
|
|
}
|
|
}
|
|
|
|
if ( ( (m_nButtons & IN_USE) && (caps & FCAP_CONTINUOUS_USE) ) ||
|
|
( (m_afButtonPressed & IN_USE) && (caps & (FCAP_IMPULSE_USE|FCAP_ONOFF_USE)) ) )
|
|
{
|
|
if ( caps & FCAP_CONTINUOUS_USE )
|
|
m_afPhysicsFlags |= PFLAG_USING;
|
|
|
|
pUseEntity->AcceptInput( "Use", this, this, emptyVariant, USE_TOGGLE );
|
|
|
|
usedSomething = true;
|
|
}
|
|
// UNDONE: Send different USE codes for ON/OFF. Cache last ONOFF_USE object to send 'off' if you turn away
|
|
else if ( (m_afButtonReleased & IN_USE) && (pUseEntity->ObjectCaps() & FCAP_ONOFF_USE) ) // BUGBUG This is an "off" use
|
|
{
|
|
pUseEntity->AcceptInput( "Use", this, this, emptyVariant, USE_TOGGLE );
|
|
|
|
usedSomething = true;
|
|
}
|
|
|
|
#if HL2_SINGLE_PRIMARY_WEAPON_MODE
|
|
|
|
//Check for weapon pick-up
|
|
if ( m_afButtonPressed & IN_USE )
|
|
{
|
|
CBaseCombatWeapon *pWeapon = dynamic_cast<CBaseCombatWeapon *>(pUseEntity);
|
|
|
|
if ( ( pWeapon != NULL ) && ( Weapon_CanSwitchTo( pWeapon ) ) )
|
|
{
|
|
//Try to take ammo or swap the weapon
|
|
if ( Weapon_OwnsThisType( pWeapon->GetClassname(), pWeapon->GetSubType() ) )
|
|
{
|
|
Weapon_EquipAmmoOnly( pWeapon );
|
|
}
|
|
else
|
|
{
|
|
Weapon_DropSlot( pWeapon->GetSlot() );
|
|
Weapon_Equip( pWeapon );
|
|
}
|
|
|
|
usedSomething = true;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
return usedSomething;
|
|
}
|
|
|
|
//bool CPortal_Player::StartReplayMode( float fDelay, float fDuration, int iEntity )
|
|
//{
|
|
// if ( !BaseClass::StartReplayMode( fDelay, fDuration, 1 ) )
|
|
// return false;
|
|
//
|
|
// CSingleUserRecipientFilter filter( this );
|
|
// filter.MakeReliable();
|
|
//
|
|
// UserMessageBegin( filter, "KillCam" );
|
|
//
|
|
// EHANDLE hPlayer = this;
|
|
//
|
|
// if ( m_hObserverTarget.Get() )
|
|
// {
|
|
// WRITE_EHANDLE( m_hObserverTarget ); // first target
|
|
// WRITE_EHANDLE( hPlayer ); //second target
|
|
// }
|
|
// else
|
|
// {
|
|
// WRITE_EHANDLE( hPlayer ); // first target
|
|
// WRITE_EHANDLE( 0 ); //second target
|
|
// }
|
|
// MessageEnd();
|
|
//
|
|
// return true;
|
|
//}
|
|
//
|
|
//void CPortal_Player::StopReplayMode()
|
|
//{
|
|
// BaseClass::StopReplayMode();
|
|
//
|
|
// CSingleUserRecipientFilter filter( this );
|
|
// filter.MakeReliable();
|
|
//
|
|
// UserMessageBegin( filter, "KillCam" );
|
|
// WRITE_EHANDLE( 0 );
|
|
// WRITE_EHANDLE( 0 );
|
|
// MessageEnd();
|
|
//}
|
|
|
|
void CPortal_Player::PlayerUse( void )
|
|
{
|
|
// Was use pressed or released?
|
|
if ( ! ((m_nButtons | m_afButtonPressed | m_afButtonReleased) & IN_USE) )
|
|
return;
|
|
|
|
if ( m_afButtonPressed & IN_USE )
|
|
{
|
|
// Currently using a latched entity?
|
|
if ( ClearUseEntity() )
|
|
{
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
if ( m_afPhysicsFlags & PFLAG_DIROVERRIDE )
|
|
{
|
|
m_afPhysicsFlags &= ~PFLAG_DIROVERRIDE;
|
|
m_iTrain = TRAIN_NEW|TRAIN_OFF;
|
|
return;
|
|
}
|
|
else
|
|
{ // Start controlling the train!
|
|
CBaseEntity *pTrain = GetGroundEntity();
|
|
if ( pTrain && !(m_nButtons & IN_JUMP) && (GetFlags() & FL_ONGROUND) && (pTrain->ObjectCaps() & FCAP_DIRECTIONAL_USE) && pTrain->OnControls(this) )
|
|
{
|
|
m_afPhysicsFlags |= PFLAG_DIROVERRIDE;
|
|
m_iTrain = TrainSpeed(pTrain->m_flSpeed, ((CFuncTrackTrain*)pTrain)->GetMaxSpeed());
|
|
m_iTrain |= TRAIN_NEW;
|
|
EmitSound( "HL2Player.TrainUse" );
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Tracker 3926: We can't +USE something if we're climbing a ladder
|
|
if ( GetMoveType() == MOVETYPE_LADDER )
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
|
|
CBaseEntity *pUseEntity = FindUseEntity();
|
|
|
|
bool usedSomething = false;
|
|
|
|
// Found an object
|
|
if ( pUseEntity )
|
|
{
|
|
SetHeldObjectOnOppositeSideOfPortal( false );
|
|
|
|
// TODO: Removed because we no longer have ghost animatings. May need to rework this code.
|
|
//// If we found a ghost animating then it needs to be held across a portal
|
|
//CGhostAnimating *pGhostAnimating = dynamic_cast<CGhostAnimating*>( pUseEntity );
|
|
//if ( pGhostAnimating )
|
|
//{
|
|
// CProp_Portal *pPortal = NULL;
|
|
|
|
// CPortalSimulator *pPortalSimulator = CPortalSimulator::GetSimulatorThatOwnsEntity( pGhostAnimating->GetSourceEntity() );
|
|
|
|
// //HACKHACK: This assumes all portal simulators are a member of a prop_portal
|
|
// pPortal = (CProp_Portal *)(((char *)pPortalSimulator) - ((int)&(((CProp_Portal *)0)->m_PortalSimulator)));
|
|
// Assert( (&(pPortal->m_PortalSimulator)) == pPortalSimulator ); //doublechecking the hack
|
|
|
|
// if ( pPortal )
|
|
// {
|
|
// SetHeldObjectPortal( pPortal->m_hLinkedPortal );
|
|
// SetHeldObjectOnOppositeSideOfPortal( true );
|
|
// }
|
|
//}
|
|
usedSomething = UseFoundEntity( pUseEntity );
|
|
}
|
|
else
|
|
{
|
|
Vector forward;
|
|
EyeVectors( &forward, NULL, NULL );
|
|
Vector start = EyePosition();
|
|
|
|
Ray_t rayPortalTest;
|
|
rayPortalTest.Init( start, start + forward * PLAYER_USE_RADIUS );
|
|
|
|
float fMustBeCloserThan = 2.0f;
|
|
|
|
CProp_Portal *pPortal = UTIL_Portal_FirstAlongRay( rayPortalTest, fMustBeCloserThan );
|
|
|
|
if ( pPortal )
|
|
{
|
|
SetHeldObjectPortal( pPortal );
|
|
pUseEntity = FindUseEntityThroughPortal();
|
|
}
|
|
|
|
if ( pUseEntity )
|
|
{
|
|
SetHeldObjectOnOppositeSideOfPortal( true );
|
|
usedSomething = UseFoundEntity( pUseEntity );
|
|
}
|
|
else if ( m_afButtonPressed & IN_USE )
|
|
{
|
|
// Signal that we want to play the deny sound, unless the user is +USEing on a ladder!
|
|
// The sound is emitted in ItemPostFrame, since that occurs after GameMovement::ProcessMove which
|
|
// lets the ladder code unset this flag.
|
|
m_bPlayUseDenySound = true;
|
|
}
|
|
}
|
|
|
|
// Debounce the use key
|
|
if ( usedSomething && pUseEntity )
|
|
{
|
|
m_Local.m_nOldButtons |= IN_USE;
|
|
m_afButtonPressed &= ~IN_USE;
|
|
}
|
|
}
|
|
|
|
void CPortal_Player::PlayerRunCommand(CUserCmd *ucmd, IMoveHelper *moveHelper)
|
|
{
|
|
if( m_bFixEyeAnglesFromPortalling )
|
|
{
|
|
//the idea here is to handle the notion that the player has portalled, but they sent us an angle update before receiving that message.
|
|
//If we don't handle this here, we end up sending back their old angles which makes them hiccup their angles for a frame
|
|
float fOldAngleDiff = fabs( AngleDistance( ucmd->viewangles.x, m_qPrePortalledViewAngles.x ) );
|
|
fOldAngleDiff += fabs( AngleDistance( ucmd->viewangles.y, m_qPrePortalledViewAngles.y ) );
|
|
fOldAngleDiff += fabs( AngleDistance( ucmd->viewangles.z, m_qPrePortalledViewAngles.z ) );
|
|
|
|
float fCurrentAngleDiff = fabs( AngleDistance( ucmd->viewangles.x, pl.v_angle.x ) );
|
|
fCurrentAngleDiff += fabs( AngleDistance( ucmd->viewangles.y, pl.v_angle.y ) );
|
|
fCurrentAngleDiff += fabs( AngleDistance( ucmd->viewangles.z, pl.v_angle.z ) );
|
|
|
|
if( fCurrentAngleDiff > fOldAngleDiff )
|
|
ucmd->viewangles = TransformAnglesToWorldSpace( ucmd->viewangles, m_matLastPortalled.As3x4() );
|
|
|
|
m_bFixEyeAnglesFromPortalling = false;
|
|
}
|
|
|
|
BaseClass::PlayerRunCommand( ucmd, moveHelper );
|
|
}
|
|
|
|
|
|
bool CPortal_Player::ClientCommand( const CCommand &args )
|
|
{
|
|
if ( FStrEq( args[0], "spectate" ) )
|
|
{
|
|
// do nothing.
|
|
return true;
|
|
}
|
|
|
|
return BaseClass::ClientCommand( args );
|
|
}
|
|
|
|
void CPortal_Player::CheatImpulseCommands( int iImpulse )
|
|
{
|
|
switch ( iImpulse )
|
|
{
|
|
case 101:
|
|
{
|
|
if( sv_cheats->GetBool() )
|
|
{
|
|
GiveAllItems();
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
BaseClass::CheatImpulseCommands( iImpulse );
|
|
}
|
|
}
|
|
|
|
void CPortal_Player::CreateViewModel( int index /*=0*/ )
|
|
{
|
|
BaseClass::CreateViewModel( index );
|
|
return;
|
|
Assert( index >= 0 && index < MAX_VIEWMODELS );
|
|
|
|
if ( GetViewModel( index ) )
|
|
return;
|
|
|
|
CPredictedViewModel *vm = ( CPredictedViewModel * )CreateEntityByName( "predicted_viewmodel" );
|
|
if ( vm )
|
|
{
|
|
vm->SetAbsOrigin( GetAbsOrigin() );
|
|
vm->SetOwner( this );
|
|
vm->SetIndex( index );
|
|
DispatchSpawn( vm );
|
|
vm->FollowEntity( this, false );
|
|
m_hViewModel.Set( index, vm );
|
|
}
|
|
}
|
|
|
|
bool CPortal_Player::BecomeRagdollOnClient( const Vector &force )
|
|
{
|
|
return true;//BaseClass::BecomeRagdollOnClient( force );
|
|
}
|
|
|
|
void CPortal_Player::CreateRagdollEntity( const CTakeDamageInfo &info )
|
|
{
|
|
if ( m_hRagdoll )
|
|
{
|
|
UTIL_RemoveImmediate( m_hRagdoll );
|
|
m_hRagdoll = NULL;
|
|
}
|
|
|
|
#if PORTAL_HIDE_PLAYER_RAGDOLL
|
|
AddSolidFlags( FSOLID_NOT_SOLID );
|
|
AddEffects( EF_NODRAW | EF_NOSHADOW );
|
|
AddEFlags( EFL_NO_DISSOLVE );
|
|
#endif // PORTAL_HIDE_PLAYER_RAGDOLL
|
|
CBaseEntity *pRagdoll = CreateServerRagdoll( this, m_nForceBone, info, COLLISION_GROUP_INTERACTIVE_DEBRIS, true );
|
|
pRagdoll->m_takedamage = DAMAGE_NO;
|
|
m_hRagdoll = pRagdoll;
|
|
|
|
/*
|
|
// If we already have a ragdoll destroy it.
|
|
CPortalRagdoll *pRagdoll = dynamic_cast<CPortalRagdoll*>( m_hRagdoll.Get() );
|
|
if( pRagdoll )
|
|
{
|
|
UTIL_Remove( pRagdoll );
|
|
pRagdoll = NULL;
|
|
}
|
|
Assert( pRagdoll == NULL );
|
|
|
|
// Create a ragdoll.
|
|
pRagdoll = dynamic_cast<CPortalRagdoll*>( CreateEntityByName( "portal_ragdoll" ) );
|
|
if ( pRagdoll )
|
|
{
|
|
|
|
|
|
pRagdoll->m_hPlayer = this;
|
|
pRagdoll->m_vecRagdollOrigin = GetAbsOrigin();
|
|
pRagdoll->m_vecRagdollVelocity = GetAbsVelocity();
|
|
pRagdoll->m_nModelIndex = m_nModelIndex;
|
|
pRagdoll->m_nForceBone = m_nForceBone;
|
|
pRagdoll->CopyAnimationDataFrom( this );
|
|
pRagdoll->SetOwnerEntity( this );
|
|
pRagdoll->m_flAnimTime = gpGlobals->curtime;
|
|
pRagdoll->m_flPlaybackRate = 0.0;
|
|
pRagdoll->SetCycle( 0 );
|
|
pRagdoll->ResetSequence( 0 );
|
|
|
|
float fSequenceDuration = SequenceDuration( GetSequence() );
|
|
float fPreviousCycle = clamp(GetCycle()-( 0.1 * ( 1 / fSequenceDuration ) ),0.f,1.f);
|
|
float fCurCycle = GetCycle();
|
|
matrix3x4_t pBoneToWorld[MAXSTUDIOBONES], pBoneToWorldNext[MAXSTUDIOBONES];
|
|
SetupBones( pBoneToWorldNext, BONE_USED_BY_ANYTHING );
|
|
SetCycle( fPreviousCycle );
|
|
SetupBones( pBoneToWorld, BONE_USED_BY_ANYTHING );
|
|
SetCycle( fCurCycle );
|
|
|
|
pRagdoll->InitRagdoll( info.GetDamageForce(), m_nForceBone, info.GetDamagePosition(), pBoneToWorld, pBoneToWorldNext, 0.1f, COLLISION_GROUP_INTERACTIVE_DEBRIS, true );
|
|
pRagdoll->SetMoveType( MOVETYPE_VPHYSICS );
|
|
pRagdoll->SetSolid( SOLID_VPHYSICS );
|
|
if ( IsDissolving() )
|
|
{
|
|
pRagdoll->TransferDissolveFrom( this );
|
|
}
|
|
|
|
Vector mins, maxs;
|
|
mins = CollisionProp()->OBBMins();
|
|
maxs = CollisionProp()->OBBMaxs();
|
|
pRagdoll->CollisionProp()->SetCollisionBounds( mins, maxs );
|
|
pRagdoll->SetCollisionGroup( COLLISION_GROUP_INTERACTIVE_DEBRIS );
|
|
}
|
|
|
|
// Turn off the player.
|
|
AddSolidFlags( FSOLID_NOT_SOLID );
|
|
AddEffects( EF_NODRAW | EF_NOSHADOW );
|
|
SetMoveType( MOVETYPE_NONE );
|
|
|
|
// Save ragdoll handle.
|
|
m_hRagdoll = pRagdoll;
|
|
*/
|
|
}
|
|
|
|
void CPortal_Player::Jump( void )
|
|
{
|
|
g_PortalGameStats.Event_PlayerJump( GetAbsOrigin(), GetAbsVelocity() );
|
|
BaseClass::Jump();
|
|
}
|
|
|
|
void CPortal_Player::Event_Killed( const CTakeDamageInfo &info )
|
|
{
|
|
//update damage info with our accumulated physics force
|
|
CTakeDamageInfo subinfo = info;
|
|
subinfo.SetDamageForce( m_vecTotalBulletForce );
|
|
|
|
// show killer in death cam mode
|
|
// chopped down version of SetObserverTarget without the team check
|
|
//if( info.GetAttacker() )
|
|
//{
|
|
// // set new target
|
|
// m_hObserverTarget.Set( info.GetAttacker() );
|
|
//}
|
|
//else
|
|
// m_hObserverTarget.Set( NULL );
|
|
|
|
UpdateExpression();
|
|
|
|
// Note: since we're dead, it won't draw us on the client, but we don't set EF_NODRAW
|
|
// because we still want to transmit to the clients in our PVS.
|
|
CreateRagdollEntity( info );
|
|
|
|
BaseClass::Event_Killed( subinfo );
|
|
|
|
#if PORTAL_HIDE_PLAYER_RAGDOLL
|
|
// Fizzle all portals so they don't see the player disappear
|
|
int iPortalCount = CProp_Portal_Shared::AllPortals.Count();
|
|
CProp_Portal **pPortals = CProp_Portal_Shared::AllPortals.Base();
|
|
for( int i = 0; i != iPortalCount; ++i )
|
|
{
|
|
CProp_Portal *pTempPortal = pPortals[i];
|
|
|
|
if( pTempPortal && pTempPortal->m_bActivated )
|
|
{
|
|
pTempPortal->Fizzle();
|
|
}
|
|
}
|
|
#endif // PORTAL_HIDE_PLAYER_RAGDOLL
|
|
|
|
if ( (info.GetDamageType() & DMG_DISSOLVE) && !(m_hRagdoll.Get()->GetEFlags() & EFL_NO_DISSOLVE) )
|
|
{
|
|
if ( m_hRagdoll )
|
|
{
|
|
m_hRagdoll->GetBaseAnimating()->Dissolve( NULL, gpGlobals->curtime, false, ENTITY_DISSOLVE_NORMAL );
|
|
}
|
|
}
|
|
|
|
m_lifeState = LIFE_DYING;
|
|
StopZooming();
|
|
|
|
if ( GetObserverTarget() )
|
|
{
|
|
//StartReplayMode( 3, 3, GetObserverTarget()->entindex() );
|
|
//StartObserverMode( OBS_MODE_DEATHCAM );
|
|
}
|
|
}
|
|
|
|
int CPortal_Player::OnTakeDamage( const CTakeDamageInfo &inputInfo )
|
|
{
|
|
CTakeDamageInfo inputInfoCopy( inputInfo );
|
|
|
|
// If you shoot yourself, make it hurt but push you less
|
|
if ( inputInfoCopy.GetAttacker() == this && inputInfoCopy.GetDamageType() == DMG_BULLET )
|
|
{
|
|
inputInfoCopy.ScaleDamage( 5.0f );
|
|
inputInfoCopy.ScaleDamageForce( 0.05f );
|
|
}
|
|
|
|
CBaseEntity *pAttacker = inputInfoCopy.GetAttacker();
|
|
CBaseEntity *pInflictor = inputInfoCopy.GetInflictor();
|
|
|
|
bool bIsTurret = false;
|
|
|
|
if ( pAttacker && FClassnameIs( pAttacker, "npc_portal_turret_floor" ) )
|
|
bIsTurret = true;
|
|
|
|
// Refuse damage from prop_glados_core.
|
|
if ( (pAttacker && FClassnameIs( pAttacker, "prop_glados_core" )) ||
|
|
(pInflictor && FClassnameIs( pInflictor, "prop_glados_core" )) )
|
|
{
|
|
inputInfoCopy.SetDamage(0.0f);
|
|
}
|
|
|
|
if ( bIsTurret && ( inputInfoCopy.GetDamageType() & DMG_BULLET ) )
|
|
{
|
|
Vector vLateralForce = inputInfoCopy.GetDamageForce();
|
|
vLateralForce.z = 0.0f;
|
|
|
|
// Push if the player is moving against the force direction
|
|
if ( GetAbsVelocity().Dot( vLateralForce ) < 0.0f )
|
|
ApplyAbsVelocityImpulse( vLateralForce );
|
|
}
|
|
else if ( ( inputInfoCopy.GetDamageType() & DMG_CRUSH ) )
|
|
{
|
|
if ( bIsTurret )
|
|
{
|
|
inputInfoCopy.SetDamage( inputInfoCopy.GetDamage() * 0.5f );
|
|
}
|
|
|
|
if ( inputInfoCopy.GetDamage() >= 10.0f )
|
|
{
|
|
EmitSound( "PortalPlayer.BonkYelp" );
|
|
}
|
|
}
|
|
else if ( ( inputInfoCopy.GetDamageType() & DMG_SHOCK ) || ( inputInfoCopy.GetDamageType() & DMG_BURN ) )
|
|
{
|
|
EmitSound( "PortalPortal.PainYelp" );
|
|
}
|
|
|
|
int ret = BaseClass::OnTakeDamage( inputInfoCopy );
|
|
|
|
// Copy the multidamage damage origin over what the base class wrote, because
|
|
// that gets translated correctly though portals.
|
|
m_DmgOrigin = inputInfo.GetDamagePosition();
|
|
|
|
if ( GetHealth() < 100 )
|
|
{
|
|
m_fTimeLastHurt = gpGlobals->curtime;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
int CPortal_Player::OnTakeDamage_Alive( const CTakeDamageInfo &info )
|
|
{
|
|
// set damage type sustained
|
|
m_bitsDamageType |= info.GetDamageType();
|
|
|
|
if ( !CBaseCombatCharacter::OnTakeDamage_Alive( info ) )
|
|
return 0;
|
|
|
|
CBaseEntity * attacker = info.GetAttacker();
|
|
|
|
if ( !attacker )
|
|
return 0;
|
|
|
|
Vector vecDir = vec3_origin;
|
|
if ( info.GetInflictor() )
|
|
{
|
|
vecDir = info.GetInflictor()->WorldSpaceCenter() - Vector ( 0, 0, 10 ) - WorldSpaceCenter();
|
|
VectorNormalize( vecDir );
|
|
}
|
|
|
|
if ( info.GetInflictor() && (GetMoveType() == MOVETYPE_WALK) &&
|
|
( !attacker->IsSolidFlagSet(FSOLID_TRIGGER)) )
|
|
{
|
|
Vector force = vecDir;// * -DamageForce( WorldAlignSize(), info.GetBaseDamage() );
|
|
if ( force.z > 250.0f )
|
|
{
|
|
force.z = 250.0f;
|
|
}
|
|
ApplyAbsVelocityImpulse( force );
|
|
}
|
|
|
|
// fire global game event
|
|
|
|
IGameEvent * event = gameeventmanager->CreateEvent( "player_hurt" );
|
|
if ( event )
|
|
{
|
|
event->SetInt("userid", GetUserID() );
|
|
event->SetInt("health", MAX(0, m_iHealth) );
|
|
event->SetInt("priority", 5 ); // HLTV event priority, not transmitted
|
|
|
|
if ( attacker->IsPlayer() )
|
|
{
|
|
CBasePlayer *player = ToBasePlayer( attacker );
|
|
event->SetInt("attacker", player->GetUserID() ); // hurt by other player
|
|
}
|
|
else
|
|
{
|
|
event->SetInt("attacker", 0 ); // hurt by "world"
|
|
}
|
|
|
|
gameeventmanager->FireEvent( event );
|
|
}
|
|
|
|
// Insert a combat sound so that nearby NPCs hear battle
|
|
if ( attacker->IsNPC() )
|
|
{
|
|
CSoundEnt::InsertSound( SOUND_COMBAT, GetAbsOrigin(), 512, 0.5, this );//<<TODO>>//magic number
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
|
|
void CPortal_Player::ForceDuckThisFrame( void )
|
|
{
|
|
if( m_Local.m_bDucked != true )
|
|
{
|
|
//m_Local.m_bDucking = false;
|
|
m_Local.m_bDucked = true;
|
|
ForceButtons( IN_DUCK );
|
|
AddFlag( FL_DUCKING );
|
|
SetVCollisionState( GetAbsOrigin(), GetAbsVelocity(), VPHYS_CROUCH );
|
|
}
|
|
}
|
|
|
|
void CPortal_Player::UnDuck( void )
|
|
{
|
|
if( m_Local.m_bDucked != false )
|
|
{
|
|
m_Local.m_bDucked = false;
|
|
UnforceButtons( IN_DUCK );
|
|
RemoveFlag( FL_DUCKING );
|
|
SetVCollisionState( GetAbsOrigin(), GetAbsVelocity(), VPHYS_WALK );
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Overload for portal-- Our player can lift his own mass.
|
|
// Input : *pObject - The object to lift
|
|
// bLimitMassAndSize - check for mass/size limits
|
|
//-----------------------------------------------------------------------------
|
|
void CPortal_Player::PickupObject(CBaseEntity *pObject, bool bLimitMassAndSize )
|
|
{
|
|
// can't pick up what you're standing on
|
|
if ( GetGroundEntity() == pObject )
|
|
return;
|
|
|
|
if ( bLimitMassAndSize == true )
|
|
{
|
|
if ( CBasePlayer::CanPickupObject( pObject, PORTAL_PLAYER_MAX_LIFT_MASS, PORTAL_PLAYER_MAX_LIFT_SIZE ) == false )
|
|
return;
|
|
}
|
|
|
|
// Can't be picked up if NPCs are on me
|
|
if ( pObject->HasNPCsOnIt() )
|
|
return;
|
|
|
|
PlayerPickupObject( this, pObject );
|
|
}
|
|
|
|
void CPortal_Player::ForceDropOfCarriedPhysObjects( CBaseEntity *pOnlyIfHoldingThis )
|
|
{
|
|
m_bHeldObjectOnOppositeSideOfPortal = false;
|
|
BaseClass::ForceDropOfCarriedPhysObjects( pOnlyIfHoldingThis );
|
|
}
|
|
|
|
void CPortal_Player::IncrementPortalsPlaced( void )
|
|
{
|
|
m_StatsThisLevel.iNumPortalsPlaced++;
|
|
|
|
if ( m_iBonusChallenge == PORTAL_CHALLENGE_PORTALS )
|
|
SetBonusProgress( static_cast<int>( m_StatsThisLevel.iNumPortalsPlaced ) );
|
|
}
|
|
|
|
void CPortal_Player::IncrementStepsTaken( void )
|
|
{
|
|
m_StatsThisLevel.iNumStepsTaken++;
|
|
|
|
if ( m_iBonusChallenge == PORTAL_CHALLENGE_STEPS )
|
|
SetBonusProgress( static_cast<int>( m_StatsThisLevel.iNumStepsTaken ) );
|
|
}
|
|
|
|
void CPortal_Player::UpdateSecondsTaken( void )
|
|
{
|
|
float fSecondsSinceLastUpdate = ( gpGlobals->curtime - m_fTimeLastNumSecondsUpdate );
|
|
m_StatsThisLevel.fNumSecondsTaken += fSecondsSinceLastUpdate;
|
|
m_fTimeLastNumSecondsUpdate = gpGlobals->curtime;
|
|
|
|
if ( m_iBonusChallenge == PORTAL_CHALLENGE_TIME )
|
|
SetBonusProgress( static_cast<int>( m_StatsThisLevel.fNumSecondsTaken ) );
|
|
|
|
if ( m_fNeuroToxinDamageTime > 0.0f )
|
|
{
|
|
float fTimeRemaining = m_fNeuroToxinDamageTime - gpGlobals->curtime;
|
|
|
|
if ( fTimeRemaining < 0.0f )
|
|
{
|
|
CTakeDamageInfo info;
|
|
info.SetDamage( gpGlobals->frametime * 50.0f );
|
|
info.SetDamageType( DMG_NERVEGAS );
|
|
TakeDamage( info );
|
|
fTimeRemaining = 0.0f;
|
|
}
|
|
|
|
PauseBonusProgress( false );
|
|
SetBonusProgress( static_cast<int>( fTimeRemaining ) );
|
|
}
|
|
}
|
|
|
|
void CPortal_Player::ResetThisLevelStats( void )
|
|
{
|
|
m_StatsThisLevel.iNumPortalsPlaced = 0;
|
|
m_StatsThisLevel.iNumStepsTaken = 0;
|
|
m_StatsThisLevel.fNumSecondsTaken = 0.0f;
|
|
|
|
if ( m_iBonusChallenge != PORTAL_CHALLENGE_NONE )
|
|
SetBonusProgress( 0 );
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Update the area bits variable which is networked down to the client to determine
|
|
// which area portals should be closed based on visibility.
|
|
// Input : *pvs - pvs to be used to determine visibility of the portals
|
|
//-----------------------------------------------------------------------------
|
|
void CPortal_Player::UpdatePortalViewAreaBits( unsigned char *pvs, int pvssize )
|
|
{
|
|
Assert ( pvs );
|
|
|
|
int iPortalCount = CProp_Portal_Shared::AllPortals.Count();
|
|
if( iPortalCount == 0 )
|
|
return;
|
|
|
|
CProp_Portal **pPortals = CProp_Portal_Shared::AllPortals.Base();
|
|
int *portalArea = (int *)stackalloc( sizeof( int ) * iPortalCount );
|
|
bool *bUsePortalForVis = (bool *)stackalloc( sizeof( bool ) * iPortalCount );
|
|
|
|
unsigned char *portalTempBits = (unsigned char *)stackalloc( sizeof( unsigned char ) * 32 * iPortalCount );
|
|
COMPILE_TIME_ASSERT( (sizeof( unsigned char ) * 32) >= sizeof( ((CPlayerLocalData*)0)->m_chAreaBits ) );
|
|
|
|
// setup area bits for these portals
|
|
for ( int i = 0; i < iPortalCount; ++i )
|
|
{
|
|
CProp_Portal* pLocalPortal = pPortals[ i ];
|
|
// Make sure this portal is active before adding it's location to the pvs
|
|
if ( pLocalPortal && pLocalPortal->m_bActivated )
|
|
{
|
|
CProp_Portal* pRemotePortal = pLocalPortal->m_hLinkedPortal.Get();
|
|
|
|
// Make sure this portal's linked portal is in the PVS before we add what it can see
|
|
if ( pRemotePortal && pRemotePortal->m_bActivated && pRemotePortal->NetworkProp() &&
|
|
pRemotePortal->NetworkProp()->IsInPVS( edict(), pvs, pvssize ) )
|
|
{
|
|
portalArea[ i ] = engine->GetArea( pPortals[ i ]->GetAbsOrigin() );
|
|
|
|
if ( portalArea [ i ] >= 0 )
|
|
{
|
|
bUsePortalForVis[ i ] = true;
|
|
}
|
|
|
|
engine->GetAreaBits( portalArea[ i ], &portalTempBits[ i * 32 ], sizeof( unsigned char ) * 32 );
|
|
}
|
|
}
|
|
}
|
|
|
|
// Use the union of player-view area bits and the portal-view area bits of each portal
|
|
for ( int i = 0; i < m_Local.m_chAreaBits.Count(); i++ )
|
|
{
|
|
for ( int j = 0; j < iPortalCount; ++j )
|
|
{
|
|
// If this portal is active, in PVS and it's location is valid
|
|
if ( bUsePortalForVis[ j ] )
|
|
{
|
|
m_Local.m_chAreaBits.Set( i, m_Local.m_chAreaBits[ i ] | portalTempBits[ (j * 32) + i ] );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
// AddPortalCornersToEnginePVS
|
|
// Subroutine to wrap the adding of portal corners to the PVS which is called once for the setup of each portal.
|
|
// input - pPortal: the portal we are viewing 'out of' which needs it's corners added to the PVS
|
|
//////////////////////////////////////////////////////////////////////////
|
|
void AddPortalCornersToEnginePVS( CProp_Portal* pPortal )
|
|
{
|
|
Assert ( pPortal );
|
|
|
|
if ( !pPortal )
|
|
return;
|
|
|
|
Vector vForward, vRight, vUp;
|
|
pPortal->GetVectors( &vForward, &vRight, &vUp );
|
|
|
|
// Center of the remote portal
|
|
Vector ptOrigin = pPortal->GetAbsOrigin();
|
|
|
|
// Distance offsets to the different edges of the portal... Used in the placement checks
|
|
Vector vToTopEdge = vUp * ( PORTAL_HALF_HEIGHT - PORTAL_BUMP_FORGIVENESS );
|
|
Vector vToBottomEdge = -vToTopEdge;
|
|
Vector vToRightEdge = vRight * ( PORTAL_HALF_WIDTH - PORTAL_BUMP_FORGIVENESS );
|
|
Vector vToLeftEdge = -vToRightEdge;
|
|
|
|
// Distance to place PVS points away from portal, to avoid being in solid
|
|
Vector vForwardBump = vForward * 1.0f;
|
|
|
|
// Add center and edges to the engine PVS
|
|
engine->AddOriginToPVS( ptOrigin + vForwardBump);
|
|
engine->AddOriginToPVS( ptOrigin + vToTopEdge + vToLeftEdge + vForwardBump );
|
|
engine->AddOriginToPVS( ptOrigin + vToTopEdge + vToRightEdge + vForwardBump );
|
|
engine->AddOriginToPVS( ptOrigin + vToBottomEdge + vToLeftEdge + vForwardBump );
|
|
engine->AddOriginToPVS( ptOrigin + vToBottomEdge + vToRightEdge + vForwardBump );
|
|
}
|
|
|
|
void PortalSetupVisibility( CBaseEntity *pPlayer, int area, unsigned char *pvs, int pvssize )
|
|
{
|
|
int iPortalCount = CProp_Portal_Shared::AllPortals.Count();
|
|
if( iPortalCount == 0 )
|
|
return;
|
|
|
|
CProp_Portal **pPortals = CProp_Portal_Shared::AllPortals.Base();
|
|
for( int i = 0; i != iPortalCount; ++i )
|
|
{
|
|
CProp_Portal *pPortal = pPortals[i];
|
|
|
|
if ( pPortal && pPortal->m_bActivated )
|
|
{
|
|
if ( pPortal->NetworkProp()->IsInPVS( pPlayer->edict(), pvs, pvssize ) )
|
|
{
|
|
if ( engine->CheckAreasConnected( area, pPortal->NetworkProp()->AreaNum() ) )
|
|
{
|
|
CProp_Portal *pLinkedPortal = static_cast<CProp_Portal*>( pPortal->m_hLinkedPortal.Get() );
|
|
if ( pLinkedPortal )
|
|
{
|
|
AddPortalCornersToEnginePVS ( pLinkedPortal );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CPortal_Player::SetupVisibility( CBaseEntity *pViewEntity, unsigned char *pvs, int pvssize )
|
|
{
|
|
BaseClass::SetupVisibility( pViewEntity, pvs, pvssize );
|
|
|
|
int area = pViewEntity ? pViewEntity->NetworkProp()->AreaNum() : NetworkProp()->AreaNum();
|
|
|
|
// At this point the EyePosition has been added as a view origin, but if we are currently stuck
|
|
// in a portal, our EyePosition may return a point in solid. Find the reflected eye position
|
|
// and use that as a vis origin instead.
|
|
if ( m_hPortalEnvironment )
|
|
{
|
|
CProp_Portal *pPortal = NULL, *pRemotePortal = NULL;
|
|
pPortal = m_hPortalEnvironment;
|
|
pRemotePortal = pPortal->m_hLinkedPortal;
|
|
|
|
if ( pPortal && pRemotePortal && pPortal->m_bActivated && pRemotePortal->m_bActivated )
|
|
{
|
|
Vector ptPortalCenter = pPortal->GetAbsOrigin();
|
|
Vector vPortalForward;
|
|
pPortal->GetVectors( &vPortalForward, NULL, NULL );
|
|
|
|
Vector eyeOrigin = EyePosition();
|
|
Vector vEyeToPortalCenter = ptPortalCenter - eyeOrigin;
|
|
|
|
float fPortalDist = vPortalForward.Dot( vEyeToPortalCenter );
|
|
if( fPortalDist > 0.0f ) //eye point is behind portal
|
|
{
|
|
// Move eye origin to it's transformed position on the other side of the portal
|
|
UTIL_Portal_PointTransform( pPortal->MatrixThisToLinked(), eyeOrigin, eyeOrigin );
|
|
|
|
// Use this as our view origin (as this is where the client will be displaying from)
|
|
engine->AddOriginToPVS( eyeOrigin );
|
|
if ( !pViewEntity || pViewEntity->IsPlayer() )
|
|
{
|
|
area = engine->GetArea( eyeOrigin );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
PortalSetupVisibility( this, area, pvs, pvssize );
|
|
}
|
|
|
|
|
|
#ifdef PORTAL_MP
|
|
|
|
CBaseEntity* CPortal_Player::EntSelectSpawnPoint( void )
|
|
{
|
|
CBaseEntity *pSpot = NULL;
|
|
CBaseEntity *pLastSpawnPoint = g_pLastSpawn;
|
|
edict_t *player = edict();
|
|
const char *pSpawnpointName = "info_player_start";
|
|
|
|
/*if ( HL2MPRules()->IsTeamplay() == true )
|
|
{
|
|
if ( GetTeamNumber() == TEAM_COMBINE )
|
|
{
|
|
pSpawnpointName = "info_player_combine";
|
|
pLastSpawnPoint = g_pLastCombineSpawn;
|
|
}
|
|
else if ( GetTeamNumber() == TEAM_REBELS )
|
|
{
|
|
pSpawnpointName = "info_player_rebel";
|
|
pLastSpawnPoint = g_pLastRebelSpawn;
|
|
}
|
|
|
|
if ( gEntList.FindEntityByClassname( NULL, pSpawnpointName ) == NULL )
|
|
{
|
|
pSpawnpointName = "info_player_deathmatch";
|
|
pLastSpawnPoint = g_pLastSpawn;
|
|
}
|
|
}*/
|
|
|
|
pSpot = pLastSpawnPoint;
|
|
// Randomize the start spot
|
|
for ( int i = random->RandomInt(1,5); i > 0; i-- )
|
|
pSpot = gEntList.FindEntityByClassname( pSpot, pSpawnpointName );
|
|
if ( !pSpot ) // skip over the null point
|
|
pSpot = gEntList.FindEntityByClassname( pSpot, pSpawnpointName );
|
|
|
|
CBaseEntity *pFirstSpot = pSpot;
|
|
|
|
do
|
|
{
|
|
if ( pSpot )
|
|
{
|
|
// check if pSpot is valid
|
|
if ( g_pGameRules->IsSpawnPointValid( pSpot, this ) )
|
|
{
|
|
if ( pSpot->GetLocalOrigin() == vec3_origin )
|
|
{
|
|
pSpot = gEntList.FindEntityByClassname( pSpot, pSpawnpointName );
|
|
continue;
|
|
}
|
|
|
|
// if so, go to pSpot
|
|
goto ReturnSpot;
|
|
}
|
|
}
|
|
// increment pSpot
|
|
pSpot = gEntList.FindEntityByClassname( pSpot, pSpawnpointName );
|
|
} while ( pSpot != pFirstSpot ); // loop if we're not back to the start
|
|
|
|
// we haven't found a place to spawn yet, so kill any guy at the first spawn point and spawn there
|
|
if ( pSpot )
|
|
{
|
|
CBaseEntity *ent = NULL;
|
|
for ( CEntitySphereQuery sphere( pSpot->GetAbsOrigin(), 128 ); (ent = sphere.GetCurrentEntity()) != NULL; sphere.NextEntity() )
|
|
{
|
|
// if ent is a client, kill em (unless they are ourselves)
|
|
if ( ent->IsPlayer() && !(ent->edict() == player) )
|
|
ent->TakeDamage( CTakeDamageInfo( GetContainingEntity(INDEXENT(0)), GetContainingEntity(INDEXENT(0)), 300, DMG_GENERIC ) );
|
|
}
|
|
goto ReturnSpot;
|
|
}
|
|
|
|
if ( !pSpot )
|
|
{
|
|
pSpot = gEntList.FindEntityByClassname( pSpot, "info_player_start" );
|
|
|
|
if ( pSpot )
|
|
goto ReturnSpot;
|
|
}
|
|
|
|
ReturnSpot:
|
|
|
|
/*if ( HL2MPRules()->IsTeamplay() == true )
|
|
{
|
|
if ( GetTeamNumber() == TEAM_COMBINE )
|
|
{
|
|
g_pLastCombineSpawn = pSpot;
|
|
}
|
|
else if ( GetTeamNumber() == TEAM_REBELS )
|
|
{
|
|
g_pLastRebelSpawn = pSpot;
|
|
}
|
|
}*/
|
|
|
|
g_pLastSpawn = pSpot;
|
|
|
|
//m_flSlamProtectTime = gpGlobals->curtime + 0.5;
|
|
|
|
return pSpot;
|
|
}
|
|
|
|
void CPortal_Player::PickTeam( void )
|
|
{
|
|
//picks lowest or random
|
|
CTeam *pCombine = g_Teams[TEAM_COMBINE];
|
|
CTeam *pRebels = g_Teams[TEAM_REBELS];
|
|
if ( pCombine->GetNumPlayers() > pRebels->GetNumPlayers() )
|
|
{
|
|
ChangeTeam( TEAM_REBELS );
|
|
}
|
|
else if ( pCombine->GetNumPlayers() < pRebels->GetNumPlayers() )
|
|
{
|
|
ChangeTeam( TEAM_COMBINE );
|
|
}
|
|
else
|
|
{
|
|
ChangeTeam( random->RandomInt( TEAM_COMBINE, TEAM_REBELS ) );
|
|
}
|
|
}
|
|
|
|
#endif
|
|
|
|
CON_COMMAND( startadmiregloves, "Starts the admire gloves animation." )
|
|
{
|
|
CPortal_Player *pPlayer = (CPortal_Player *)UTIL_GetCommandClient();
|
|
if( pPlayer == NULL )
|
|
pPlayer = GetPortalPlayer( 1 ); //last ditch effort
|
|
|
|
if( pPlayer )
|
|
pPlayer->StartAdmireGlovesAnimation();
|
|
}
|
|
|
|
CON_COMMAND( displayportalplayerstats, "Displays current level stats for portals placed, steps taken, and seconds taken." )
|
|
{
|
|
CPortal_Player *pPlayer = (CPortal_Player *)UTIL_GetCommandClient();
|
|
if( pPlayer == NULL )
|
|
pPlayer = GetPortalPlayer( 1 ); //last ditch effort
|
|
|
|
if( pPlayer )
|
|
{
|
|
int iMinutes = static_cast<int>( pPlayer->NumSecondsTaken() / 60.0f );
|
|
int iSeconds = static_cast<int>( pPlayer->NumSecondsTaken() ) % 60;
|
|
|
|
CFmtStr msg;
|
|
NDebugOverlay::ScreenText( 0.5f, 0.5f, msg.sprintf( "Portals Placed: %d\nSteps Taken: %d\nTime: %d:%d", pPlayer->NumPortalsPlaced(), pPlayer->NumStepsTaken(), iMinutes, iSeconds ), 255, 255, 255, 150, 5.0f );
|
|
}
|
|
}
|
|
|
|
CON_COMMAND( startneurotoxins, "Starts the nerve gas timer." )
|
|
{
|
|
CPortal_Player *pPlayer = (CPortal_Player *)UTIL_GetCommandClient();
|
|
if( pPlayer == NULL )
|
|
pPlayer = GetPortalPlayer( 1 ); //last ditch effort
|
|
|
|
float fCoundownTime = 180.0f;
|
|
|
|
if ( args.ArgC() > 1 )
|
|
fCoundownTime = atof( args[ 1 ] );
|
|
|
|
if( pPlayer )
|
|
pPlayer->SetNeuroToxinDamageTime( fCoundownTime );
|
|
}
|