mirror of
https://github.com/alliedmodders/hl2sdk.git
synced 2025-01-10 02:58:48 +08:00
2900 lines
88 KiB
C++
2900 lines
88 KiB
C++
//====== Copyright <20> 1996-2005, Valve Corporation, All rights reserved. =====//
|
||
//
|
||
// Purpose: Client-side CBasePlayer.
|
||
//
|
||
// - Manages the player's flashlight effect.
|
||
//
|
||
//===========================================================================//
|
||
#include "cbase.h"
|
||
#include "c_baseplayer.h"
|
||
#include "flashlighteffect.h"
|
||
#include "weapon_selection.h"
|
||
#include "history_resource.h"
|
||
#include "iinput.h"
|
||
#include "input.h"
|
||
#include "ammodef.h"
|
||
#include "view.h"
|
||
#include "iviewrender.h"
|
||
#include "iclientmode.h"
|
||
#include "in_buttons.h"
|
||
#include "engine/IEngineSound.h"
|
||
#include "c_soundscape.h"
|
||
#include "usercmd.h"
|
||
#include "c_playerresource.h"
|
||
#include "iclientvehicle.h"
|
||
#include "view_shared.h"
|
||
#include "movevars_shared.h"
|
||
#include "prediction.h"
|
||
#include "tier0/vprof.h"
|
||
#include "filesystem.h"
|
||
#include "bitbuf.h"
|
||
#include "KeyValues.h"
|
||
#include "particles_simple.h"
|
||
#include "fx_water.h"
|
||
#include "hltvcamera.h"
|
||
#if defined( REPLAY_ENABLED )
|
||
#include "replaycamera.h"
|
||
#endif
|
||
#include "toolframework/itoolframework.h"
|
||
#include "toolframework_client.h"
|
||
#include "view_scene.h"
|
||
#include "c_vguiscreen.h"
|
||
#include "datacache/imdlcache.h"
|
||
#include "vgui/isurface.h"
|
||
#include "voice_status.h"
|
||
#include "fx.h"
|
||
#include "cellcoord.h"
|
||
|
||
#include "debugoverlay_shared.h"
|
||
|
||
#ifdef DEMOPOLISH_ENABLED
|
||
#include "demo_polish/demo_polish.h"
|
||
#endif
|
||
|
||
// memdbgon must be the last include file in a .cpp file!!!
|
||
#include "tier0/memdbgon.h"
|
||
|
||
// Don't alias here
|
||
#if defined( CBasePlayer )
|
||
#undef CBasePlayer
|
||
#endif
|
||
|
||
int g_nKillCamMode = OBS_MODE_NONE;
|
||
int g_nKillCamTarget1 = 0;
|
||
int g_nKillCamTarget2 = 0;
|
||
|
||
extern ConVar mp_forcecamera; // in gamevars_shared.h
|
||
extern ConVar r_mapextents;
|
||
|
||
#define FLASHLIGHT_DISTANCE 1000
|
||
#define MAX_VGUI_INPUT_MODE_SPEED 30
|
||
#define MAX_VGUI_INPUT_MODE_SPEED_SQ (MAX_VGUI_INPUT_MODE_SPEED*MAX_VGUI_INPUT_MODE_SPEED)
|
||
|
||
static Vector WALL_MIN(-WALL_OFFSET,-WALL_OFFSET,-WALL_OFFSET);
|
||
static Vector WALL_MAX(WALL_OFFSET,WALL_OFFSET,WALL_OFFSET);
|
||
|
||
bool CommentaryModeShouldSwallowInput( C_BasePlayer *pPlayer );
|
||
|
||
extern ConVar default_fov;
|
||
extern ConVar sensitivity;
|
||
|
||
static C_BasePlayer *s_pLocalPlayer[ MAX_SPLITSCREEN_PLAYERS ];
|
||
|
||
static ConVar cl_customsounds ( "cl_customsounds", "0", 0, "Enable customized player sound playback" );
|
||
static ConVar spec_track ( "spec_track", "0", 0, "Tracks an entity in spec mode" );
|
||
static ConVar cl_smooth ( "cl_smooth", "1", 0, "Smooth view/eye origin after prediction errors" );
|
||
static ConVar cl_smoothtime (
|
||
"cl_smoothtime",
|
||
"0.1",
|
||
0,
|
||
"Smooth client's view after prediction error over this many seconds",
|
||
true, 0.01, // min/max is 0.01/2.0
|
||
true, 2.0
|
||
);
|
||
|
||
ConVar spec_freeze_time( "spec_freeze_time", "4.0", FCVAR_CHEAT | FCVAR_REPLICATED, "Time spend frozen in observer freeze cam." );
|
||
ConVar spec_freeze_traveltime( "spec_freeze_traveltime", "0.4", FCVAR_CHEAT | FCVAR_REPLICATED, "Time taken to zoom in to frame a target in observer freeze cam.", true, 0.01, false, 0 );
|
||
ConVar spec_freeze_distance_min( "spec_freeze_distance_min", "96", FCVAR_CHEAT, "Minimum random distance from the target to stop when framing them in observer freeze cam." );
|
||
ConVar spec_freeze_distance_max( "spec_freeze_distance_max", "200", FCVAR_CHEAT, "Maximum random distance from the target to stop when framing them in observer freeze cam." );
|
||
|
||
bool IsDemoPolishRecording();
|
||
|
||
// -------------------------------------------------------------------------------- //
|
||
// RecvTable for CPlayerState.
|
||
// -------------------------------------------------------------------------------- //
|
||
|
||
BEGIN_RECV_TABLE_NOBASE(CPlayerState, DT_PlayerState)
|
||
RecvPropInt (RECVINFO(deadflag)),
|
||
END_RECV_TABLE()
|
||
|
||
|
||
BEGIN_RECV_TABLE_NOBASE( CPlayerLocalData, DT_Local )
|
||
RecvPropArray3( RECVINFO_ARRAY(m_chAreaBits), RecvPropInt(RECVINFO(m_chAreaBits[0]))),
|
||
RecvPropArray3( RECVINFO_ARRAY(m_chAreaPortalBits), RecvPropInt(RECVINFO(m_chAreaPortalBits[0]))),
|
||
RecvPropInt(RECVINFO(m_iHideHUD)),
|
||
|
||
// View
|
||
|
||
RecvPropFloat(RECVINFO(m_flFOVRate)),
|
||
|
||
RecvPropInt (RECVINFO(m_bDucked)),
|
||
RecvPropInt (RECVINFO(m_bDucking)),
|
||
RecvPropInt (RECVINFO(m_bInDuckJump)),
|
||
RecvPropInt (RECVINFO(m_nDuckTimeMsecs)),
|
||
RecvPropInt (RECVINFO(m_nDuckJumpTimeMsecs)),
|
||
RecvPropInt (RECVINFO(m_nJumpTimeMsecs)),
|
||
RecvPropFloat (RECVINFO(m_flFallVelocity)),
|
||
|
||
#if PREDICTION_ERROR_CHECK_LEVEL > 1
|
||
RecvPropFloat (RECVINFO_NAME( m_vecPunchAngle.m_Value[0], m_vecPunchAngle[0])),
|
||
RecvPropFloat (RECVINFO_NAME( m_vecPunchAngle.m_Value[1], m_vecPunchAngle[1])),
|
||
RecvPropFloat (RECVINFO_NAME( m_vecPunchAngle.m_Value[2], m_vecPunchAngle[2] )),
|
||
RecvPropFloat (RECVINFO_NAME( m_vecPunchAngleVel.m_Value[0], m_vecPunchAngleVel[0] )),
|
||
RecvPropFloat (RECVINFO_NAME( m_vecPunchAngleVel.m_Value[1], m_vecPunchAngleVel[1] )),
|
||
RecvPropFloat (RECVINFO_NAME( m_vecPunchAngleVel.m_Value[2], m_vecPunchAngleVel[2] )),
|
||
#else
|
||
RecvPropVector (RECVINFO(m_vecPunchAngle)),
|
||
RecvPropVector (RECVINFO(m_vecPunchAngleVel)),
|
||
#endif
|
||
|
||
RecvPropInt (RECVINFO(m_bDrawViewmodel)),
|
||
RecvPropInt (RECVINFO(m_bWearingSuit)),
|
||
RecvPropBool (RECVINFO(m_bPoisoned)),
|
||
RecvPropFloat (RECVINFO(m_flStepSize)),
|
||
RecvPropInt (RECVINFO(m_bAllowAutoMovement)),
|
||
|
||
// 3d skybox data
|
||
RecvPropInt(RECVINFO(m_skybox3d.scale)),
|
||
RecvPropVector(RECVINFO(m_skybox3d.origin)),
|
||
RecvPropInt(RECVINFO(m_skybox3d.area)),
|
||
|
||
// 3d skybox fog data
|
||
RecvPropInt( RECVINFO( m_skybox3d.fog.enable ) ),
|
||
RecvPropInt( RECVINFO( m_skybox3d.fog.blend ) ),
|
||
RecvPropVector( RECVINFO( m_skybox3d.fog.dirPrimary ) ),
|
||
RecvPropInt( RECVINFO( m_skybox3d.fog.colorPrimary ), 0, RecvProxy_Int32ToColor32 ),
|
||
RecvPropInt( RECVINFO( m_skybox3d.fog.colorSecondary ), 0, RecvProxy_Int32ToColor32 ),
|
||
RecvPropFloat( RECVINFO( m_skybox3d.fog.start ) ),
|
||
RecvPropFloat( RECVINFO( m_skybox3d.fog.end ) ),
|
||
RecvPropFloat( RECVINFO( m_skybox3d.fog.maxdensity ) ),
|
||
RecvPropFloat( RECVINFO( m_skybox3d.fog.HDRColorScale ) ),
|
||
|
||
// audio data
|
||
RecvPropVector( RECVINFO( m_audio.localSound[0] ) ),
|
||
RecvPropVector( RECVINFO( m_audio.localSound[1] ) ),
|
||
RecvPropVector( RECVINFO( m_audio.localSound[2] ) ),
|
||
RecvPropVector( RECVINFO( m_audio.localSound[3] ) ),
|
||
RecvPropVector( RECVINFO( m_audio.localSound[4] ) ),
|
||
RecvPropVector( RECVINFO( m_audio.localSound[5] ) ),
|
||
RecvPropVector( RECVINFO( m_audio.localSound[6] ) ),
|
||
RecvPropVector( RECVINFO( m_audio.localSound[7] ) ),
|
||
RecvPropInt( RECVINFO( m_audio.soundscapeIndex ) ),
|
||
RecvPropInt( RECVINFO( m_audio.localBits ) ),
|
||
RecvPropInt( RECVINFO( m_audio.entIndex ) ),
|
||
|
||
END_RECV_TABLE()
|
||
|
||
// -------------------------------------------------------------------------------- //
|
||
// This data only gets sent to clients that ARE this player entity.
|
||
// -------------------------------------------------------------------------------- //
|
||
|
||
BEGIN_RECV_TABLE_NOBASE( C_BasePlayer, DT_LocalPlayerExclusive )
|
||
|
||
RecvPropDataTable ( RECVINFO_DT(m_Local),0, &REFERENCE_RECV_TABLE(DT_Local) ),
|
||
|
||
RecvPropFloat ( RECVINFO(m_vecViewOffset[0]) ),
|
||
RecvPropFloat ( RECVINFO(m_vecViewOffset[1]) ),
|
||
RecvPropFloat ( RECVINFO(m_vecViewOffset[2]) ),
|
||
RecvPropFloat ( RECVINFO(m_flFriction) ),
|
||
|
||
RecvPropArray3 ( RECVINFO_ARRAY(m_iAmmo), RecvPropInt( RECVINFO(m_iAmmo[0])) ),
|
||
|
||
RecvPropInt ( RECVINFO(m_fOnTarget) ),
|
||
|
||
RecvPropInt ( RECVINFO( m_nTickBase ) ),
|
||
RecvPropInt ( RECVINFO( m_nNextThinkTick ) ),
|
||
|
||
RecvPropEHandle ( RECVINFO( m_hLastWeapon ) ),
|
||
|
||
RecvPropFloat ( RECVINFO(m_vecVelocity[0]), 0, C_BasePlayer::RecvProxy_LocalVelocityX ),
|
||
RecvPropFloat ( RECVINFO(m_vecVelocity[1]), 0, C_BasePlayer::RecvProxy_LocalVelocityY ),
|
||
RecvPropFloat ( RECVINFO(m_vecVelocity[2]), 0, C_BasePlayer::RecvProxy_LocalVelocityZ ),
|
||
|
||
RecvPropVector ( RECVINFO( m_vecBaseVelocity ) ),
|
||
|
||
RecvPropEHandle ( RECVINFO( m_hConstraintEntity)),
|
||
RecvPropVector ( RECVINFO( m_vecConstraintCenter) ),
|
||
RecvPropFloat ( RECVINFO( m_flConstraintRadius )),
|
||
RecvPropFloat ( RECVINFO( m_flConstraintWidth )),
|
||
RecvPropFloat ( RECVINFO( m_flConstraintSpeedFactor )),
|
||
RecvPropBool ( RECVINFO( m_bConstraintPastRadius )),
|
||
|
||
RecvPropFloat ( RECVINFO( m_flDeathTime )),
|
||
|
||
RecvPropInt ( RECVINFO( m_nWaterLevel ) ),
|
||
RecvPropFloat ( RECVINFO( m_flLaggedMovementValue )),
|
||
|
||
RecvPropEHandle ( RECVINFO( m_hTonemapController ) ),
|
||
|
||
END_RECV_TABLE()
|
||
|
||
|
||
// -------------------------------------------------------------------------------- //
|
||
// DT_BasePlayer datatable.
|
||
// -------------------------------------------------------------------------------- //
|
||
IMPLEMENT_CLIENTCLASS_DT(C_BasePlayer, DT_BasePlayer, CBasePlayer)
|
||
// We have both the local and nonlocal data in here, but the server proxies
|
||
// only send one.
|
||
RecvPropDataTable( "localdata", 0, 0, &REFERENCE_RECV_TABLE(DT_LocalPlayerExclusive) ),
|
||
|
||
RecvPropDataTable(RECVINFO_DT(pl), 0, &REFERENCE_RECV_TABLE(DT_PlayerState), DataTableRecvProxy_StaticDataTable),
|
||
|
||
RecvPropInt (RECVINFO(m_iFOV)),
|
||
RecvPropInt (RECVINFO(m_iFOVStart)),
|
||
RecvPropFloat (RECVINFO(m_flFOVTime)),
|
||
RecvPropInt (RECVINFO(m_iDefaultFOV)),
|
||
RecvPropEHandle (RECVINFO(m_hZoomOwner)),
|
||
|
||
RecvPropEHandle( RECVINFO(m_hVehicle) ),
|
||
RecvPropEHandle( RECVINFO(m_hUseEntity) ),
|
||
|
||
RecvPropEHandle ( RECVINFO( m_hViewEntity ) ), // L4D: send view entity to everyone for first-person spectating
|
||
RecvPropEHandle ( RECVINFO( m_hGroundEntity ) ),
|
||
|
||
RecvPropInt (RECVINFO(m_iHealth)),
|
||
RecvPropInt (RECVINFO(m_lifeState)),
|
||
|
||
RecvPropInt (RECVINFO(m_iBonusProgress)),
|
||
RecvPropInt (RECVINFO(m_iBonusChallenge)),
|
||
|
||
RecvPropFloat (RECVINFO(m_flMaxspeed)),
|
||
RecvPropInt (RECVINFO(m_fFlags)),
|
||
|
||
|
||
RecvPropInt (RECVINFO(m_iObserverMode), 0, C_BasePlayer::RecvProxy_ObserverMode ),
|
||
RecvPropEHandle (RECVINFO(m_hObserverTarget), C_BasePlayer::RecvProxy_ObserverTarget ),
|
||
RecvPropArray ( RecvPropEHandle( RECVINFO( m_hViewModel[0] ) ), m_hViewModel ),
|
||
|
||
|
||
RecvPropString( RECVINFO(m_szLastPlaceName) ),
|
||
RecvPropVector( RECVINFO(m_vecLadderNormal) ),
|
||
RecvPropInt (RECVINFO(m_ladderSurfaceProps) ),
|
||
|
||
RecvPropInt( RECVINFO( m_ubEFNoInterpParity ) ),
|
||
|
||
RecvPropEHandle( RECVINFO( m_hPostProcessCtrl ) ), // Send to everybody - for spectating
|
||
RecvPropEHandle( RECVINFO( m_hColorCorrectionCtrl ) ), // Send to everybody - for spectating
|
||
|
||
// fog data
|
||
RecvPropEHandle( RECVINFO( m_PlayerFog.m_hCtrl ) ),
|
||
|
||
END_RECV_TABLE()
|
||
|
||
BEGIN_PREDICTION_DATA_NO_BASE( CPlayerState )
|
||
|
||
DEFINE_PRED_FIELD( deadflag, FIELD_BOOLEAN, FTYPEDESC_INSENDTABLE ),
|
||
// DEFINE_FIELD( netname, string_t ),
|
||
// DEFINE_FIELD( fixangle, FIELD_INTEGER ),
|
||
// DEFINE_FIELD( anglechange, FIELD_FLOAT ),
|
||
// DEFINE_FIELD( v_angle, FIELD_VECTOR ),
|
||
|
||
END_PREDICTION_DATA()
|
||
|
||
BEGIN_PREDICTION_DATA_NO_BASE( CPlayerLocalData )
|
||
|
||
// DEFINE_PRED_TYPEDESCRIPTION( m_skybox3d, sky3dparams_t ),
|
||
// DEFINE_PRED_TYPEDESCRIPTION( m_fog, fogparams_t ),
|
||
// DEFINE_PRED_TYPEDESCRIPTION( m_audio, audioparams_t ),
|
||
DEFINE_FIELD( m_nStepside, FIELD_INTEGER ),
|
||
|
||
DEFINE_PRED_FIELD( m_iHideHUD, FIELD_INTEGER, FTYPEDESC_INSENDTABLE ),
|
||
#if PREDICTION_ERROR_CHECK_LEVEL > 1
|
||
DEFINE_PRED_FIELD( m_vecPunchAngle, FIELD_VECTOR, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD( m_vecPunchAngleVel, FIELD_VECTOR, FTYPEDESC_INSENDTABLE ),
|
||
#else
|
||
DEFINE_PRED_FIELD_TOL( m_vecPunchAngle, FIELD_VECTOR, FTYPEDESC_INSENDTABLE, 0.125f ),
|
||
DEFINE_PRED_FIELD_TOL( m_vecPunchAngleVel, FIELD_VECTOR, FTYPEDESC_INSENDTABLE, 0.125f ),
|
||
#endif
|
||
DEFINE_PRED_FIELD( m_bDrawViewmodel, FIELD_BOOLEAN, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD( m_bWearingSuit, FIELD_BOOLEAN, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD( m_bPoisoned, FIELD_BOOLEAN, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD( m_bAllowAutoMovement, FIELD_BOOLEAN, FTYPEDESC_INSENDTABLE ),
|
||
|
||
DEFINE_PRED_FIELD( m_bDucked, FIELD_BOOLEAN, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD( m_bDucking, FIELD_BOOLEAN, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD( m_bInDuckJump, FIELD_BOOLEAN, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD( m_nDuckTimeMsecs, FIELD_INTEGER, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD( m_nDuckJumpTimeMsecs, FIELD_INTEGER, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD( m_nJumpTimeMsecs, FIELD_INTEGER, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD_TOL( m_flFallVelocity, FIELD_FLOAT, FTYPEDESC_INSENDTABLE, 0.5f ),
|
||
// DEFINE_PRED_FIELD( m_nOldButtons, FIELD_INTEGER, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_FIELD( m_nOldButtons, FIELD_INTEGER ),
|
||
DEFINE_PRED_FIELD( m_flStepSize, FIELD_FLOAT, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_FIELD( m_flFOVRate, FIELD_FLOAT ),
|
||
|
||
END_PREDICTION_DATA()
|
||
|
||
BEGIN_PREDICTION_DATA( C_BasePlayer )
|
||
|
||
DEFINE_PRED_TYPEDESCRIPTION( m_Local, CPlayerLocalData ),
|
||
DEFINE_PRED_TYPEDESCRIPTION( pl, CPlayerState ),
|
||
|
||
DEFINE_PRED_FIELD( m_iFOV, FIELD_INTEGER, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD( m_hZoomOwner, FIELD_EHANDLE, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD( m_flFOVTime, FIELD_FLOAT, 0 ),
|
||
DEFINE_PRED_FIELD( m_iFOVStart, FIELD_INTEGER, 0 ),
|
||
|
||
DEFINE_PRED_FIELD( m_hVehicle, FIELD_EHANDLE, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD_TOL( m_flMaxspeed, FIELD_FLOAT, FTYPEDESC_INSENDTABLE, 0.5f ),
|
||
DEFINE_PRED_FIELD( m_iHealth, FIELD_INTEGER, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD( m_iBonusProgress, FIELD_INTEGER, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD( m_iBonusChallenge, FIELD_INTEGER, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD( m_fOnTarget, FIELD_BOOLEAN, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD( m_nNextThinkTick, FIELD_INTEGER, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD( m_lifeState, FIELD_CHARACTER, FTYPEDESC_INSENDTABLE ),
|
||
DEFINE_PRED_FIELD( m_nWaterLevel, FIELD_CHARACTER, FTYPEDESC_INSENDTABLE ),
|
||
|
||
DEFINE_PRED_FIELD_TOL( m_vecBaseVelocity, FIELD_VECTOR, FTYPEDESC_INSENDTABLE, 0.05 ),
|
||
|
||
DEFINE_FIELD( m_nButtons, FIELD_INTEGER ),
|
||
DEFINE_FIELD( m_flWaterJumpTime, FIELD_FLOAT ),
|
||
DEFINE_FIELD( m_nImpulse, FIELD_INTEGER ),
|
||
DEFINE_FIELD( m_flStepSoundTime, FIELD_FLOAT ),
|
||
DEFINE_FIELD( m_flSwimSoundTime, FIELD_FLOAT ),
|
||
DEFINE_FIELD( m_vecLadderNormal, FIELD_VECTOR ),
|
||
DEFINE_FIELD( m_ladderSurfaceProps, FIELD_INTEGER ),
|
||
DEFINE_FIELD( m_flPhysics, FIELD_INTEGER ),
|
||
DEFINE_AUTO_ARRAY( m_szAnimExtension, FIELD_CHARACTER ),
|
||
DEFINE_FIELD( m_afButtonLast, FIELD_INTEGER ),
|
||
DEFINE_FIELD( m_afButtonPressed, FIELD_INTEGER ),
|
||
DEFINE_FIELD( m_afButtonReleased, FIELD_INTEGER ),
|
||
// DEFINE_FIELD( m_vecOldViewAngles, FIELD_VECTOR ),
|
||
|
||
// DEFINE_ARRAY( m_iOldAmmo, FIELD_INTEGER, MAX_AMMO_TYPES ),
|
||
|
||
//DEFINE_FIELD( m_hOldVehicle, FIELD_EHANDLE ),
|
||
// DEFINE_FIELD( m_pModelLight, dlight_t* ),
|
||
// DEFINE_FIELD( m_pEnvironmentLight, dlight_t* ),
|
||
// DEFINE_FIELD( m_pBrightLight, dlight_t* ),
|
||
DEFINE_PRED_FIELD( m_hLastWeapon, FIELD_EHANDLE, FTYPEDESC_INSENDTABLE ),
|
||
|
||
DEFINE_PRED_FIELD( m_nTickBase, FIELD_INTEGER, FTYPEDESC_INSENDTABLE ),
|
||
|
||
DEFINE_PRED_FIELD( m_hGroundEntity, FIELD_EHANDLE, FTYPEDESC_INSENDTABLE ),
|
||
|
||
DEFINE_PRED_ARRAY( m_hViewModel, FIELD_EHANDLE, MAX_VIEWMODELS, FTYPEDESC_INSENDTABLE ),
|
||
|
||
DEFINE_FIELD( m_surfaceFriction, FIELD_FLOAT ),
|
||
|
||
END_PREDICTION_DATA()
|
||
|
||
|
||
LINK_ENTITY_TO_CLASS( player, C_BasePlayer );
|
||
|
||
|
||
// -------------------------------------------------------------------------------- //
|
||
// Functions.
|
||
// -------------------------------------------------------------------------------- //
|
||
C_BasePlayer::C_BasePlayer() : m_iv_vecViewOffset( "C_BasePlayer::m_iv_vecViewOffset" )
|
||
{
|
||
AddVar( &m_vecViewOffset, &m_iv_vecViewOffset, LATCH_SIMULATION_VAR );
|
||
|
||
#ifdef _DEBUG
|
||
m_vecLadderNormal.Init();
|
||
m_ladderSurfaceProps = 0;
|
||
m_vecOldViewAngles.Init();
|
||
#endif
|
||
m_hViewEntity = NULL;
|
||
|
||
for ( int i = 0; i < MAX_SPLITSCREEN_PLAYERS; i++ )
|
||
{
|
||
m_bFlashlightEnabled[ i ] = false;
|
||
}
|
||
|
||
m_pCurrentVguiScreen = NULL;
|
||
m_pCurrentCommand = NULL;
|
||
|
||
m_flPredictionErrorTime = -100;
|
||
m_StuckLast = 0;
|
||
m_bWasFrozen = false;
|
||
|
||
m_bResampleWaterSurface = true;
|
||
|
||
ResetObserverMode();
|
||
|
||
m_vecPredictionError.Init();
|
||
m_flPredictionErrorTime = 0;
|
||
|
||
m_surfaceProps = 0;
|
||
m_pSurfaceData = NULL;
|
||
m_surfaceFriction = 1.0f;
|
||
m_chTextureType = 0;
|
||
m_nSplitScreenSlot = -1;
|
||
m_bIsLocalPlayer = false;
|
||
m_afButtonForced = 0;
|
||
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose:
|
||
//-----------------------------------------------------------------------------
|
||
C_BasePlayer::~C_BasePlayer()
|
||
{
|
||
DeactivateVguiScreen( m_pCurrentVguiScreen.Get() );
|
||
for ( int i = 0; i < MAX_SPLITSCREEN_PLAYERS; ++i )
|
||
{
|
||
if ( this == s_pLocalPlayer[ i ] )
|
||
{
|
||
s_pLocalPlayer[ i ] = NULL;
|
||
}
|
||
else if ( s_pLocalPlayer[ i ] )
|
||
{
|
||
s_pLocalPlayer[ i ]->RemoveSplitScreenPlayer( this );
|
||
}
|
||
|
||
if ( m_bFlashlightEnabled[ i ] )
|
||
{
|
||
FlashlightEffectManager( i ).TurnOffFlashlight( true );
|
||
m_bFlashlightEnabled[ i ] = false;
|
||
}
|
||
}
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose:
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::Spawn( void )
|
||
{
|
||
// Clear all flags except for FL_FULLEDICT
|
||
ClearFlags();
|
||
AddFlag( FL_CLIENT );
|
||
|
||
int effects = GetEffects() & EF_NOSHADOW;
|
||
SetEffects( effects );
|
||
|
||
m_iFOV = 0; // init field of view.
|
||
|
||
SetModel( "models/player.mdl" );
|
||
|
||
Precache();
|
||
|
||
SetThink(NULL);
|
||
|
||
SharedSpawn();
|
||
|
||
m_bWasFreezeFraming = false;
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose:
|
||
//-----------------------------------------------------------------------------
|
||
bool C_BasePlayer::AudioStateIsUnderwater( Vector vecMainViewOrigin )
|
||
{
|
||
if ( IsObserver() )
|
||
{
|
||
// Just check the view position
|
||
int cont = enginetrace->GetPointContents_WorldOnly( vecMainViewOrigin, MASK_WATER );
|
||
return (cont & MASK_WATER) ? true : false;
|
||
}
|
||
|
||
return ( GetWaterLevel() >= WL_Eyes );
|
||
}
|
||
|
||
#if defined( REPLAY_ENABLED )
|
||
bool C_BasePlayer::IsReplay() const
|
||
{
|
||
return ( IsLocalPlayer( const_cast< C_BasePlayer * >( this ) ) && engine->IsReplay() );
|
||
}
|
||
#endif
|
||
|
||
CBaseEntity *C_BasePlayer::GetObserverTarget() const // returns players targer or NULL
|
||
{
|
||
if ( IsHLTV() )
|
||
{
|
||
return HLTVCamera()->GetPrimaryTarget();
|
||
}
|
||
#if defined( REPLAY_ENABLED )
|
||
if ( IsReplay() )
|
||
{
|
||
return ReplayCamera()->GetPrimaryTarget();
|
||
}
|
||
#endif
|
||
|
||
if ( GetObserverMode() == OBS_MODE_ROAMING )
|
||
{
|
||
return NULL; // no target in roaming mode
|
||
}
|
||
else
|
||
{
|
||
return m_hObserverTarget;
|
||
}
|
||
}
|
||
|
||
// Helper method to fix up visiblity across split screen for view models when observer target or mode changes
|
||
static void UpdateViewmodelVisibility( C_BasePlayer *player )
|
||
{
|
||
// Update view model visibility
|
||
for ( int i = 0; i < MAX_VIEWMODELS; i++ )
|
||
{
|
||
CBaseViewModel *vm = player->GetViewModel( i );
|
||
if ( !vm )
|
||
continue;
|
||
vm->UpdateVisibility();
|
||
}
|
||
}
|
||
|
||
// Called from Recv Proxy, mainly to reset tone map scale
|
||
void C_BasePlayer::SetObserverTarget( EHANDLE hObserverTarget )
|
||
{
|
||
// If the observer target is changing to an entity that the client doesn't know about yet,
|
||
// it can resolve to NULL. If the client didn't have an observer target before, then
|
||
// comparing EHANDLEs directly will see them as equal, since it uses Get(), and compares
|
||
// NULL to NULL. To combat this, we need to check against GetEntryIndex() and
|
||
// GetSerialNumber().
|
||
if ( hObserverTarget.GetEntryIndex() != m_hObserverTarget.GetEntryIndex() ||
|
||
hObserverTarget.GetSerialNumber() != m_hObserverTarget.GetSerialNumber())
|
||
{
|
||
// Init based on the new handle's entry index and serial number, so that it's Get()
|
||
// has a chance to become non-NULL even if it currently resolves to NULL.
|
||
m_hObserverTarget.Init( hObserverTarget.GetEntryIndex(), hObserverTarget.GetSerialNumber() );
|
||
|
||
IGameEvent *event = gameeventmanager->CreateEvent( "spec_target_updated" );
|
||
if ( event )
|
||
{
|
||
gameeventmanager->FireEventClientSide( event );
|
||
}
|
||
|
||
if ( IsLocalPlayer( this ) )
|
||
{
|
||
ACTIVE_SPLITSCREEN_PLAYER_GUARD_ENT( this );
|
||
ResetToneMapping( -1.0f ); // This forces the tonemapping scalar to the average of min and max
|
||
}
|
||
UpdateViewmodelVisibility( this );
|
||
}
|
||
}
|
||
|
||
int C_BasePlayer::GetObserverMode() const
|
||
{
|
||
if ( IsHLTV() )
|
||
{
|
||
return HLTVCamera()->GetMode();
|
||
}
|
||
#if defined( REPLAY_ENABLED )
|
||
if ( IsReplay() )
|
||
{
|
||
return ReplayCamera()->GetMode();
|
||
}
|
||
#endif
|
||
|
||
return m_iObserverMode;
|
||
}
|
||
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Used by prediction, sets the view angles for the player
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::SetLocalViewAngles( const QAngle &viewAngles )
|
||
{
|
||
pl.v_angle = viewAngles;
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose:
|
||
// Input : ang -
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::SetViewAngles( const QAngle& ang )
|
||
{
|
||
SetLocalAngles( ang );
|
||
SetNetworkAngles( ang );
|
||
}
|
||
|
||
|
||
surfacedata_t* C_BasePlayer::GetGroundSurface()
|
||
{
|
||
//
|
||
// Find the name of the material that lies beneath the player.
|
||
//
|
||
Vector start, end;
|
||
VectorCopy( GetAbsOrigin(), start );
|
||
VectorCopy( start, end );
|
||
|
||
// Straight down
|
||
end.z -= 64;
|
||
|
||
// Fill in default values, just in case.
|
||
|
||
Ray_t ray;
|
||
ray.Init( start, end, GetPlayerMins(), GetPlayerMaxs() );
|
||
|
||
trace_t trace;
|
||
UTIL_TraceRay( ray, MASK_PLAYERSOLID_BRUSHONLY, this, COLLISION_GROUP_PLAYER_MOVEMENT, &trace );
|
||
|
||
if ( trace.fraction == 1.0f )
|
||
return NULL; // no ground
|
||
|
||
return physprops->GetSurfaceData( trace.surface.surfaceProps );
|
||
}
|
||
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// returns the player name
|
||
//-----------------------------------------------------------------------------
|
||
const char * C_BasePlayer::GetPlayerName()
|
||
{
|
||
return g_PR ? g_PR->GetPlayerName( entindex() ) : "";
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Is the player dead?
|
||
//-----------------------------------------------------------------------------
|
||
bool C_BasePlayer::IsPlayerDead()
|
||
{
|
||
return pl.deadflag == true;
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose:
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::SetVehicleRole( int nRole )
|
||
{
|
||
if ( !IsInAVehicle() )
|
||
return;
|
||
|
||
// HL2 has only a player in a vehicle.
|
||
if ( nRole > VEHICLE_ROLE_DRIVER )
|
||
return;
|
||
|
||
char szCmd[64];
|
||
Q_snprintf( szCmd, sizeof( szCmd ), "vehicleRole %i\n", nRole );
|
||
engine->ServerCmd( szCmd );
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: Store original ammo data to see what has changed
|
||
// Input : bnewentity -
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::OnPreDataChanged( DataUpdateType_t updateType )
|
||
{
|
||
for (int i = 0; i < MAX_AMMO_TYPES; ++i)
|
||
{
|
||
m_iOldAmmo[i] = GetAmmoCount(i);
|
||
}
|
||
|
||
m_bWasFreezeFraming = (GetObserverMode() == OBS_MODE_FREEZECAM);
|
||
m_hOldFogController = m_PlayerFog.m_hCtrl;
|
||
|
||
BaseClass::OnPreDataChanged( updateType );
|
||
}
|
||
|
||
void C_BasePlayer::PreDataUpdate( DataUpdateType_t updateType )
|
||
{
|
||
BaseClass::PreDataUpdate( updateType );
|
||
|
||
m_ubOldEFNoInterpParity = m_ubEFNoInterpParity;
|
||
}
|
||
|
||
void C_BasePlayer::CheckForLocalPlayer( int nSplitScreenSlot )
|
||
{
|
||
// Make sure s_pLocalPlayer is correct
|
||
int iLocalPlayerIndex = ( nSplitScreenSlot != -1 ) ? engine->GetLocalPlayer() : 0;
|
||
|
||
if ( g_nKillCamMode )
|
||
iLocalPlayerIndex = g_nKillCamTarget1;
|
||
|
||
if ( iLocalPlayerIndex == index )
|
||
{
|
||
Assert( s_pLocalPlayer[ nSplitScreenSlot ] == NULL );
|
||
s_pLocalPlayer[ nSplitScreenSlot ] = this;
|
||
m_bIsLocalPlayer = true;
|
||
|
||
// Tell host player about the parasitic splitscreen user
|
||
if ( nSplitScreenSlot != 0 )
|
||
{
|
||
Assert( s_pLocalPlayer[ 0 ] );
|
||
m_nSplitScreenSlot = nSplitScreenSlot;
|
||
m_hSplitOwner = s_pLocalPlayer[ 0 ];
|
||
if ( s_pLocalPlayer[ 0 ] )
|
||
{
|
||
s_pLocalPlayer[ 0 ]->AddSplitScreenPlayer( this );
|
||
}
|
||
}
|
||
else
|
||
{
|
||
// We're the host, not the parasite...
|
||
m_nSplitScreenSlot = 0;
|
||
m_hSplitOwner = NULL;
|
||
}
|
||
|
||
if ( nSplitScreenSlot == 0 )
|
||
{
|
||
// Reset our sound mixed in case we were in a freeze cam when we
|
||
// changed level, which would cause the snd_soundmixer to be left modified.
|
||
ConVar *pVar = (ConVar *)cvar->FindVar( "snd_soundmixer" );
|
||
pVar->Revert();
|
||
}
|
||
}
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose:
|
||
// Input : updateType -
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::PostDataUpdate( DataUpdateType_t updateType )
|
||
{
|
||
// This has to occur here as opposed to OnDataChanged so that EHandles to the player created
|
||
// on this same frame are not stomped because prediction thinks there
|
||
// isn't a local player yet!!!
|
||
|
||
int nSlot = -1;
|
||
|
||
FOR_EACH_VALID_SPLITSCREEN_PLAYER( i )
|
||
{
|
||
int nIndex = engine->GetSplitScreenPlayer( i );
|
||
if ( nIndex == index )
|
||
{
|
||
nSlot = i;
|
||
break;
|
||
}
|
||
}
|
||
ACTIVE_SPLITSCREEN_PLAYER_GUARD( nSlot );
|
||
|
||
bool bRecheck = false;
|
||
if ( nSlot != -1 && !s_pLocalPlayer[ nSlot ] )
|
||
{
|
||
bRecheck = true;
|
||
}
|
||
if ( updateType == DATA_UPDATE_CREATED || bRecheck )
|
||
{
|
||
CheckForLocalPlayer( nSlot );
|
||
}
|
||
|
||
bool bForceEFNoInterp = ( m_ubOldEFNoInterpParity != m_ubEFNoInterpParity );
|
||
|
||
if ( IsLocalPlayer( this ) )
|
||
{
|
||
SetSimulatedEveryTick( true );
|
||
}
|
||
else
|
||
{
|
||
SetSimulatedEveryTick( false );
|
||
|
||
// estimate velocity for non local players
|
||
float flTimeDelta = m_flSimulationTime - m_flOldSimulationTime;
|
||
|
||
if (IsParentChanging())
|
||
{
|
||
bForceEFNoInterp = true;
|
||
}
|
||
|
||
if ( flTimeDelta > 0 && !( IsEffectActive(EF_NOINTERP) || bForceEFNoInterp ) )
|
||
{
|
||
Vector newVelo = (GetNetworkOrigin() - GetOldOrigin() ) / flTimeDelta;
|
||
// This code used to call SetAbsVelocity, which is a no-no since we are in networking and if
|
||
// in hieararchy, the parent velocity might not be set up yet.
|
||
// On top of that GetNetworkOrigin and GetOldOrigin are local coordinates
|
||
// So we'll just set the local vel and avoid an Assert here
|
||
SetLocalVelocity( newVelo );
|
||
}
|
||
}
|
||
|
||
BaseClass::PostDataUpdate( updateType );
|
||
|
||
// Only care about this for local player
|
||
if ( IsLocalPlayer( this ) )
|
||
{
|
||
QAngle angles;
|
||
engine->GetViewAngles( angles );
|
||
if ( updateType == DATA_UPDATE_CREATED )
|
||
{
|
||
SetLocalViewAngles( angles );
|
||
m_flOldPlayerZ = GetLocalOrigin().z;
|
||
}
|
||
SetLocalAngles( angles );
|
||
|
||
if ( !m_bWasFreezeFraming && GetObserverMode() == OBS_MODE_FREEZECAM )
|
||
{
|
||
m_vecFreezeFrameStart = MainViewOrigin( GetSplitScreenPlayerSlot() );
|
||
m_flFreezeFrameStartTime = gpGlobals->curtime;
|
||
m_flFreezeFrameDistance = RandomFloat( spec_freeze_distance_min.GetFloat(), spec_freeze_distance_max.GetFloat() );
|
||
m_flFreezeZOffset = RandomFloat( -30, 20 );
|
||
m_bSentFreezeFrame = false;
|
||
|
||
IGameEvent *pEvent = gameeventmanager->CreateEvent( "show_freezepanel" );
|
||
if ( pEvent )
|
||
{
|
||
pEvent->SetInt( "killer", GetObserverTarget() ? GetObserverTarget()->entindex() : 0 );
|
||
gameeventmanager->FireEventClientSide( pEvent );
|
||
}
|
||
|
||
// Force the sound mixer to the freezecam mixer
|
||
ConVar *pVar = (ConVar *)cvar->FindVar( "snd_soundmixer" );
|
||
pVar->SetValue( "FreezeCam_Only" );
|
||
}
|
||
else if ( m_bWasFreezeFraming && GetObserverMode() != OBS_MODE_FREEZECAM )
|
||
{
|
||
IGameEvent *pEvent = gameeventmanager->CreateEvent( "hide_freezepanel" );
|
||
if ( pEvent )
|
||
{
|
||
gameeventmanager->FireEventClientSide( pEvent );
|
||
}
|
||
|
||
view->FreezeFrame(0);
|
||
|
||
ConVar *pVar = (ConVar *)cvar->FindVar( "snd_soundmixer" );
|
||
pVar->Revert();
|
||
}
|
||
}
|
||
|
||
// If we are updated while paused, allow the player origin to be snapped by the
|
||
// server if we receive a packet from the server
|
||
if ( engine->IsPaused() || bForceEFNoInterp )
|
||
{
|
||
ResetLatched();
|
||
}
|
||
#ifdef DEMOPOLISH_ENABLED
|
||
if ( engine->IsRecordingDemo() &&
|
||
IsDemoPolishRecording() )
|
||
{
|
||
m_bBonePolishSetup = true;
|
||
matrix3x4a_t dummyBones[MAXSTUDIOBONES];
|
||
C_BaseEntity::SetAbsQueriesValid( true );
|
||
ForceSetupBonesAtTime( dummyBones, gpGlobals->curtime );
|
||
C_BaseEntity::SetAbsQueriesValid( false );
|
||
m_bBonePolishSetup = false;
|
||
}
|
||
#endif
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose:
|
||
//-----------------------------------------------------------------------------
|
||
bool C_BasePlayer::CanSetSoundMixer( void )
|
||
{
|
||
// Can't set sound mixers when we're in freezecam mode, since it has a code-enforced mixer
|
||
return (GetObserverMode() != OBS_MODE_FREEZECAM);
|
||
}
|
||
|
||
void C_BasePlayer::ReceiveMessage( int classID, bf_read &msg )
|
||
{
|
||
if ( classID != GetClientClass()->m_ClassID )
|
||
{
|
||
// message is for subclass
|
||
BaseClass::ReceiveMessage( classID, msg );
|
||
return;
|
||
}
|
||
|
||
int messageType = msg.ReadByte();
|
||
|
||
switch( messageType )
|
||
{
|
||
case PLAY_PLAYER_JINGLE:
|
||
PlayPlayerJingle();
|
||
break;
|
||
}
|
||
}
|
||
|
||
void C_BasePlayer::OnRestore()
|
||
{
|
||
BaseClass::OnRestore();
|
||
|
||
if ( IsLocalPlayer( this ) )
|
||
{
|
||
ACTIVE_SPLITSCREEN_PLAYER_GUARD_ENT( this );
|
||
|
||
// debounce the attack key, for if it was used for restore
|
||
input->ClearInputButton( IN_ATTACK | IN_ATTACK2 );
|
||
// GetButtonBits() has to be called for the above to take effect
|
||
input->GetButtonBits( false );
|
||
}
|
||
|
||
// For ammo history icons to current value so they don't flash on level transtions
|
||
int ammoTypes = GetAmmoDef()->NumAmmoTypes();
|
||
// ammodef is 1 based, use <=
|
||
for ( int i = 0; i <= ammoTypes; i++ )
|
||
{
|
||
m_iOldAmmo[i] = GetAmmoCount(i);
|
||
}
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: Process incoming data
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::OnDataChanged( DataUpdateType_t updateType )
|
||
{
|
||
#if !defined( NO_ENTITY_PREDICTION )
|
||
if ( IsLocalPlayer( this ) )
|
||
{
|
||
SetPredictionEligible( true );
|
||
}
|
||
#endif
|
||
|
||
BaseClass::OnDataChanged( updateType );
|
||
|
||
// Only care about this for local player
|
||
if ( IsLocalPlayer( this ) )
|
||
{
|
||
int nSlot = GetSplitScreenPlayerSlot();
|
||
// Reset engine areabits pointer, but only for main local player (not piggybacked split screen users)
|
||
if ( nSlot == 0 )
|
||
{
|
||
render->SetAreaState( m_Local.m_chAreaBits, m_Local.m_chAreaPortalBits );
|
||
}
|
||
|
||
// Check for Ammo pickups.
|
||
int ammoTypes = GetAmmoDef()->NumAmmoTypes();
|
||
for ( int i = 0; i <= ammoTypes; i++ )
|
||
{
|
||
if ( GetAmmoCount(i) > m_iOldAmmo[i] )
|
||
{
|
||
// Only add this to the correct Hud
|
||
ACTIVE_SPLITSCREEN_PLAYER_GUARD( nSlot );
|
||
|
||
// Don't add to ammo pickup if the ammo doesn't do it
|
||
const FileWeaponInfo_t *pWeaponData = gWR.GetWeaponFromAmmo(i);
|
||
|
||
if ( !pWeaponData || !( pWeaponData->iFlags & ITEM_FLAG_NOAMMOPICKUPS ) )
|
||
{
|
||
// We got more ammo for this ammo index. Add it to the ammo history
|
||
CHudHistoryResource *pHudHR = GET_HUDELEMENT( CHudHistoryResource );
|
||
if( pHudHR )
|
||
{
|
||
pHudHR->AddToHistory( HISTSLOT_AMMO, i, abs(GetAmmoCount(i) - m_iOldAmmo[i]) );
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
// Only add this to the correct Hud
|
||
{
|
||
ACTIVE_SPLITSCREEN_PLAYER_GUARD( nSlot );
|
||
Soundscape_Update( m_Local.m_audio );
|
||
}
|
||
|
||
if ( m_hOldFogController != m_PlayerFog.m_hCtrl )
|
||
{
|
||
FogControllerChanged( updateType == DATA_UPDATE_CREATED );
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Did we just enter a vehicle this frame?
|
||
//-----------------------------------------------------------------------------
|
||
bool C_BasePlayer::JustEnteredVehicle()
|
||
{
|
||
if ( !IsInAVehicle() )
|
||
return false;
|
||
|
||
return ( m_hOldVehicle == m_hVehicle );
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Are we in VGUI input mode?.
|
||
//-----------------------------------------------------------------------------
|
||
bool C_BasePlayer::IsInVGuiInputMode() const
|
||
{
|
||
return (m_pCurrentVguiScreen.Get() != NULL);
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Are we inputing to a view model vgui screen
|
||
//-----------------------------------------------------------------------------
|
||
bool C_BasePlayer::IsInViewModelVGuiInputMode() const
|
||
{
|
||
C_BaseEntity *pScreenEnt = m_pCurrentVguiScreen.Get();
|
||
|
||
if ( !pScreenEnt )
|
||
return false;
|
||
|
||
Assert( dynamic_cast<C_VGuiScreen*>(pScreenEnt) );
|
||
C_VGuiScreen *pVguiScreen = static_cast<C_VGuiScreen*>(pScreenEnt);
|
||
|
||
return ( pVguiScreen->IsAttachedToViewModel() && pVguiScreen->AcceptsInput() );
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Check to see if we're in vgui input mode...
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::DetermineVguiInputMode( CUserCmd *pCmd )
|
||
{
|
||
// If we're dead, close down and abort!
|
||
if ( !IsAlive() )
|
||
{
|
||
DeactivateVguiScreen( m_pCurrentVguiScreen.Get() );
|
||
m_pCurrentVguiScreen.Set( NULL );
|
||
return;
|
||
}
|
||
|
||
// If we're in vgui mode *and* we're holding down mouse buttons,
|
||
// stay in vgui mode even if we're outside the screen bounds
|
||
if (m_pCurrentVguiScreen.Get() && (pCmd->buttons & (IN_ATTACK | IN_ATTACK2)) )
|
||
{
|
||
SetVGuiScreenButtonState( m_pCurrentVguiScreen.Get(), pCmd->buttons );
|
||
|
||
// Kill all attack inputs if we're in vgui screen mode
|
||
pCmd->buttons &= ~(IN_ATTACK | IN_ATTACK2);
|
||
return;
|
||
}
|
||
|
||
// We're not in vgui input mode if we're moving, or have hit a key
|
||
// that will make us move...
|
||
|
||
// Not in vgui mode if we're moving too quickly
|
||
// ROBIN: Disabled movement preventing VGUI screen usage
|
||
//if (GetVelocity().LengthSqr() > MAX_VGUI_INPUT_MODE_SPEED_SQ)
|
||
if ( 0 )
|
||
{
|
||
DeactivateVguiScreen( m_pCurrentVguiScreen.Get() );
|
||
m_pCurrentVguiScreen.Set( NULL );
|
||
return;
|
||
}
|
||
|
||
// Don't enter vgui mode if we've got combat buttons held down
|
||
bool bAttacking = false;
|
||
if ( ((pCmd->buttons & IN_ATTACK) || (pCmd->buttons & IN_ATTACK2)) && !m_pCurrentVguiScreen.Get() )
|
||
{
|
||
bAttacking = true;
|
||
}
|
||
|
||
// Not in vgui mode if we're pushing any movement key at all
|
||
// Not in vgui mode if we're in a vehicle...
|
||
// ROBIN: Disabled movement preventing VGUI screen usage
|
||
//if ((pCmd->forwardmove > MAX_VGUI_INPUT_MODE_SPEED) ||
|
||
// (pCmd->sidemove > MAX_VGUI_INPUT_MODE_SPEED) ||
|
||
// (pCmd->upmove > MAX_VGUI_INPUT_MODE_SPEED) ||
|
||
// (pCmd->buttons & IN_JUMP) ||
|
||
// (bAttacking) )
|
||
if ( bAttacking || IsInAVehicle() )
|
||
{
|
||
DeactivateVguiScreen( m_pCurrentVguiScreen.Get() );
|
||
m_pCurrentVguiScreen.Set( NULL );
|
||
return;
|
||
}
|
||
|
||
// Don't interact with world screens when we're in a menu
|
||
if ( vgui::surface()->IsCursorVisible() )
|
||
{
|
||
DeactivateVguiScreen( m_pCurrentVguiScreen.Get() );
|
||
m_pCurrentVguiScreen.Set( NULL );
|
||
return;
|
||
}
|
||
|
||
// Not in vgui mode if there are no nearby screens
|
||
C_BaseEntity *pOldScreen = m_pCurrentVguiScreen.Get();
|
||
|
||
m_pCurrentVguiScreen = FindNearbyVguiScreen( EyePosition(), pCmd->viewangles, GetTeamNumber() );
|
||
|
||
if (pOldScreen != m_pCurrentVguiScreen)
|
||
{
|
||
DeactivateVguiScreen( pOldScreen );
|
||
ActivateVguiScreen( m_pCurrentVguiScreen.Get() );
|
||
}
|
||
|
||
if (m_pCurrentVguiScreen.Get())
|
||
{
|
||
SetVGuiScreenButtonState( m_pCurrentVguiScreen.Get(), pCmd->buttons );
|
||
|
||
// Kill all attack inputs if we're in vgui screen mode
|
||
pCmd->buttons &= ~(IN_ATTACK | IN_ATTACK2);
|
||
}
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: Input handling
|
||
//-----------------------------------------------------------------------------
|
||
bool C_BasePlayer::CreateMove( float flInputSampleTime, CUserCmd *pCmd )
|
||
{
|
||
// Allow the vehicle to clamp the view angles
|
||
if ( IsInAVehicle() )
|
||
{
|
||
IClientVehicle *pVehicle = m_hVehicle.Get()->GetClientVehicle();
|
||
if ( pVehicle )
|
||
{
|
||
pVehicle->UpdateViewAngles( this, pCmd );
|
||
engine->SetViewAngles( pCmd->viewangles );
|
||
}
|
||
}
|
||
else
|
||
{
|
||
#ifndef _X360
|
||
if ( joy_autosprint.GetBool() )
|
||
#endif
|
||
{
|
||
if ( input->KeyState( &in_joyspeed ) != 0.0f )
|
||
{
|
||
pCmd->buttons |= IN_SPEED;
|
||
}
|
||
}
|
||
|
||
CBaseCombatWeapon *pWeapon = GetActiveWeapon();
|
||
if ( pWeapon )
|
||
{
|
||
pWeapon->CreateMove( flInputSampleTime, pCmd, m_vecOldViewAngles );
|
||
}
|
||
}
|
||
|
||
// If the frozen flag is set, prevent view movement (server prevents the rest of the movement)
|
||
if ( GetFlags() & FL_FROZEN )
|
||
{
|
||
// Don't stomp the first time we get frozen
|
||
if ( m_bWasFrozen )
|
||
{
|
||
// Stomp the new viewangles with old ones
|
||
pCmd->viewangles = m_vecOldViewAngles;
|
||
engine->SetViewAngles( pCmd->viewangles );
|
||
}
|
||
else
|
||
{
|
||
m_bWasFrozen = true;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
m_bWasFrozen = false;
|
||
}
|
||
|
||
m_vecOldViewAngles = pCmd->viewangles;
|
||
|
||
// Check to see if we're in vgui input mode...
|
||
DetermineVguiInputMode( pCmd );
|
||
|
||
return true;
|
||
}
|
||
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: Player has changed to a new team
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::TeamChange( int iNewTeam )
|
||
{
|
||
// Base class does nothing
|
||
}
|
||
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: Creates, destroys, and updates the flashlight effect as needed.
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::UpdateFlashlight()
|
||
{
|
||
ASSERT_LOCAL_PLAYER_RESOLVABLE();
|
||
int iSsPlayer = GET_ACTIVE_SPLITSCREEN_SLOT();
|
||
|
||
// TERROR: if we're in-eye spectating, use that player's flashlight
|
||
C_BasePlayer *pFlashlightPlayer = this;
|
||
if ( !IsAlive() )
|
||
{
|
||
if ( GetObserverMode() == OBS_MODE_IN_EYE )
|
||
{
|
||
pFlashlightPlayer = ToBasePlayer( GetObserverTarget() );
|
||
}
|
||
}
|
||
|
||
if ( pFlashlightPlayer )
|
||
{
|
||
FlashlightEffectManager().SetEntityIndex( pFlashlightPlayer->index );
|
||
}
|
||
|
||
// The dim light is the flashlight.
|
||
if ( pFlashlightPlayer && pFlashlightPlayer->IsAlive() && pFlashlightPlayer->IsEffectActive( EF_DIMLIGHT ) && !pFlashlightPlayer->GetViewEntity() )
|
||
{
|
||
// Make sure we're using the proper flashlight texture
|
||
const char *pszTextureName = pFlashlightPlayer->GetFlashlightTextureName();
|
||
if ( !m_bFlashlightEnabled[ iSsPlayer ] )
|
||
{
|
||
// Turned on the headlight; create it.
|
||
if ( pszTextureName )
|
||
{
|
||
FlashlightEffectManager().TurnOnFlashlight( pFlashlightPlayer->index, pszTextureName, pFlashlightPlayer->GetFlashlightFOV(),
|
||
pFlashlightPlayer->GetFlashlightFarZ(), pFlashlightPlayer->GetFlashlightLinearAtten() );
|
||
}
|
||
else
|
||
{
|
||
FlashlightEffectManager().TurnOnFlashlight( pFlashlightPlayer->index );
|
||
}
|
||
m_bFlashlightEnabled[ iSsPlayer ] = true;
|
||
}
|
||
}
|
||
else if ( m_bFlashlightEnabled[ iSsPlayer ] )
|
||
{
|
||
// Turned off the flashlight; delete it.
|
||
FlashlightEffectManager().TurnOffFlashlight();
|
||
m_bFlashlightEnabled[ iSsPlayer ] = false;
|
||
}
|
||
|
||
if ( pFlashlightPlayer && m_bFlashlightEnabled[ iSsPlayer ] )
|
||
{
|
||
Vector vecForward, vecRight, vecUp;
|
||
Vector vecPos;
|
||
//Check to see if we have an externally specified flashlight origin, if not, use eye vectors/render origin
|
||
if ( pFlashlightPlayer->m_vecFlashlightOrigin != vec3_origin && pFlashlightPlayer->m_vecFlashlightOrigin.IsValid() )
|
||
{
|
||
vecPos = pFlashlightPlayer->m_vecFlashlightOrigin;
|
||
vecForward = pFlashlightPlayer->m_vecFlashlightForward;
|
||
vecRight = pFlashlightPlayer->m_vecFlashlightRight;
|
||
vecUp = pFlashlightPlayer->m_vecFlashlightUp;
|
||
}
|
||
else
|
||
{
|
||
EyeVectors( &vecForward, &vecRight, &vecUp );
|
||
vecPos = GetRenderOrigin() + m_vecViewOffset;
|
||
}
|
||
|
||
// Update the light with the new position and direction.
|
||
FlashlightEffectManager().UpdateFlashlight( vecPos, vecForward, vecRight, vecUp, pFlashlightPlayer->GetFlashlightFOV(),
|
||
pFlashlightPlayer->CastsFlashlightShadows(), pFlashlightPlayer->GetFlashlightFarZ(), pFlashlightPlayer->GetFlashlightLinearAtten(),
|
||
pFlashlightPlayer->GetFlashlightTextureName() );
|
||
}
|
||
}
|
||
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: Creates player flashlight if it's active
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::Flashlight( void )
|
||
{
|
||
UpdateFlashlight();
|
||
}
|
||
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: Turns off flashlight if it's active (TERROR)
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::TurnOffFlashlight( void )
|
||
{
|
||
ASSERT_LOCAL_PLAYER_RESOLVABLE();
|
||
int nSlot = GET_ACTIVE_SPLITSCREEN_SLOT();
|
||
|
||
if ( m_bFlashlightEnabled[ nSlot ] )
|
||
{
|
||
FlashlightEffectManager().TurnOffFlashlight();
|
||
m_bFlashlightEnabled[ nSlot ] = false;
|
||
}
|
||
}
|
||
|
||
|
||
extern float UTIL_WaterLevel( const Vector &position, float minz, float maxz );
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose:
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::CreateWaterEffects( void )
|
||
{
|
||
// Must be completely submerged to bother
|
||
if ( GetWaterLevel() < 3 )
|
||
{
|
||
m_bResampleWaterSurface = true;
|
||
return;
|
||
}
|
||
|
||
// Do special setup if this is our first time back underwater
|
||
if ( m_bResampleWaterSurface )
|
||
{
|
||
// Reset our particle timer
|
||
m_tWaterParticleTimer.Init( 32 );
|
||
|
||
// Find the surface of the water to clip against
|
||
m_flWaterSurfaceZ = UTIL_WaterLevel( WorldSpaceCenter(), WorldSpaceCenter().z, WorldSpaceCenter().z + 256 );
|
||
m_bResampleWaterSurface = false;
|
||
}
|
||
|
||
// Make sure the emitter is setup
|
||
if ( m_pWaterEmitter == NULL )
|
||
{
|
||
if ( ( m_pWaterEmitter = WaterDebrisEffect::Create( "splish" ) ) == NULL )
|
||
return;
|
||
}
|
||
|
||
Vector vecVelocity;
|
||
GetVectors( &vecVelocity, NULL, NULL );
|
||
|
||
Vector offset = WorldSpaceCenter();
|
||
|
||
m_pWaterEmitter->SetSortOrigin( offset );
|
||
|
||
SimpleParticle *pParticle;
|
||
|
||
float curTime = gpGlobals->frametime;
|
||
|
||
// Add as many particles as we need
|
||
while ( m_tWaterParticleTimer.NextEvent( curTime ) )
|
||
{
|
||
offset = WorldSpaceCenter() + ( vecVelocity * 128.0f ) + RandomVector( -128, 128 );
|
||
|
||
// Make sure we don't start out of the water!
|
||
if ( offset.z > m_flWaterSurfaceZ )
|
||
{
|
||
offset.z = ( m_flWaterSurfaceZ - 8.0f );
|
||
}
|
||
|
||
pParticle = (SimpleParticle *) m_pWaterEmitter->AddParticle( sizeof(SimpleParticle), g_Mat_Fleck_Cement[random->RandomInt(0,1)], offset );
|
||
|
||
if (pParticle == NULL)
|
||
continue;
|
||
|
||
pParticle->m_flLifetime = 0.0f;
|
||
pParticle->m_flDieTime = random->RandomFloat( 2.0f, 4.0f );
|
||
|
||
pParticle->m_vecVelocity = RandomVector( -2.0f, 2.0f );
|
||
|
||
//FIXME: We should tint these based on the water's fog value!
|
||
float color = random->RandomInt( 32, 128 );
|
||
pParticle->m_uchColor[0] = color;
|
||
pParticle->m_uchColor[1] = color;
|
||
pParticle->m_uchColor[2] = color;
|
||
|
||
pParticle->m_uchStartSize = 1;
|
||
pParticle->m_uchEndSize = 1;
|
||
|
||
pParticle->m_uchStartAlpha = 255;
|
||
pParticle->m_uchEndAlpha = 0;
|
||
|
||
pParticle->m_flRoll = random->RandomInt( 0, 360 );
|
||
pParticle->m_flRollDelta = random->RandomFloat( -0.5f, 0.5f );
|
||
}
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Called when not in tactical mode. Allows view to be overriden for things like driving a tank.
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::OverrideView( CViewSetup *pSetup )
|
||
{
|
||
}
|
||
|
||
bool C_BasePlayer::ShouldInterpolate()
|
||
{
|
||
// always interpolate myself
|
||
if ( IsLocalPlayer( this ) )
|
||
return true;
|
||
|
||
// always interpolate entity if followed by HLTV/Replay
|
||
#if defined( REPLAY_ENABLED )
|
||
if ( HLTVCamera()->GetCameraMan() == this ||
|
||
ReplayCamera()->GetCameraMan() == this )
|
||
return true;
|
||
#else
|
||
if ( HLTVCamera()->GetCameraMan() == this )
|
||
return true;
|
||
#endif
|
||
|
||
return BaseClass::ShouldInterpolate();
|
||
}
|
||
|
||
|
||
bool C_BasePlayer::ShouldDraw()
|
||
{
|
||
return ( this != GetSplitScreenViewPlayer() || C_BasePlayer::ShouldDrawLocalPlayer() || (GetObserverMode() == OBS_MODE_DEATHCAM ) ) &&
|
||
BaseClass::ShouldDraw();
|
||
}
|
||
|
||
int C_BasePlayer::DrawModel( int flags, const RenderableInstance_t &instance )
|
||
{
|
||
// if local player is spectating this player in first person mode, don't draw it
|
||
C_BasePlayer * player = C_BasePlayer::GetLocalPlayer();
|
||
|
||
if ( player && player->IsObserver() )
|
||
{
|
||
if ( player->GetObserverMode() == OBS_MODE_IN_EYE &&
|
||
player->GetObserverTarget() == this &&
|
||
!input->CAM_IsThirdPerson() )
|
||
return 0;
|
||
}
|
||
|
||
return BaseClass::DrawModel( flags, instance );
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Computes the render mode for this player
|
||
//-----------------------------------------------------------------------------
|
||
PlayerRenderMode_t C_BasePlayer::GetPlayerRenderMode( int nSlot )
|
||
{
|
||
// check if local player chases owner of this weapon in first person
|
||
C_BasePlayer *pLocalPlayer = GetSplitScreenViewPlayer( nSlot );
|
||
if ( !pLocalPlayer )
|
||
return PLAYER_RENDER_THIRDPERSON;
|
||
|
||
if ( pLocalPlayer->IsObserver() )
|
||
{
|
||
if ( pLocalPlayer->GetObserverTarget() != this )
|
||
return PLAYER_RENDER_THIRDPERSON;
|
||
if ( pLocalPlayer->GetObserverMode() != OBS_MODE_IN_EYE )
|
||
return PLAYER_RENDER_THIRDPERSON;
|
||
}
|
||
else
|
||
{
|
||
if ( pLocalPlayer != this )
|
||
return PLAYER_RENDER_THIRDPERSON;
|
||
}
|
||
|
||
if ( input->CAM_IsThirdPerson( nSlot ) )
|
||
return PLAYER_RENDER_THIRDPERSON;
|
||
|
||
// if ( IsInThirdPersonView() )
|
||
// return PLAYER_RENDER_THIRDPERSON;
|
||
|
||
return PLAYER_RENDER_FIRSTPERSON;
|
||
}
|
||
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose:
|
||
//-----------------------------------------------------------------------------
|
||
Vector C_BasePlayer::GetChaseCamViewOffset( CBaseEntity *target )
|
||
{
|
||
C_BasePlayer *player = ToBasePlayer( target );
|
||
|
||
if ( player && player->IsAlive() )
|
||
{
|
||
if( player->GetFlags() & FL_DUCKING )
|
||
return VEC_DUCK_VIEW;
|
||
|
||
return VEC_VIEW;
|
||
}
|
||
|
||
// assume it's the players ragdoll
|
||
return VEC_DEAD_VIEWHEIGHT;
|
||
}
|
||
|
||
void C_BasePlayer::CalcChaseCamView(Vector& eyeOrigin, QAngle& eyeAngles, float& fov)
|
||
{
|
||
C_BaseEntity *target = GetObserverTarget();
|
||
|
||
if ( !target )
|
||
{
|
||
// just copy a save in-map position
|
||
VectorCopy( EyePosition(), eyeOrigin );
|
||
VectorCopy( EyeAngles(), eyeAngles );
|
||
return;
|
||
};
|
||
|
||
// If our target isn't visible, we're at a camera point of some kind.
|
||
// Instead of letting the player rotate around an invisible point, treat
|
||
// the point as a fixed camera.
|
||
if ( !target->GetBaseAnimating() && !target->GetModel() )
|
||
{
|
||
CalcRoamingView( eyeOrigin, eyeAngles, fov );
|
||
return;
|
||
}
|
||
|
||
// QAngle tmpangles;
|
||
|
||
Vector forward, viewpoint;
|
||
|
||
// GetObserverCamOrigin() returns ragdoll pos if player is ragdolled
|
||
Vector origin = target->GetObserverCamOrigin();
|
||
|
||
VectorAdd( origin, GetChaseCamViewOffset( target ), origin );
|
||
|
||
QAngle viewangles;
|
||
|
||
if ( GetObserverMode() == OBS_MODE_IN_EYE )
|
||
{
|
||
viewangles = eyeAngles;
|
||
}
|
||
else if ( IsLocalPlayer( this ) )
|
||
{
|
||
engine->GetViewAngles( viewangles );
|
||
}
|
||
else
|
||
{
|
||
viewangles = EyeAngles();
|
||
}
|
||
|
||
m_flObserverChaseDistance += gpGlobals->frametime*48.0f;
|
||
|
||
float flMaxDistance = CHASE_CAM_DISTANCE;
|
||
if ( target && target->IsBaseTrain() )
|
||
{
|
||
// if this is a train, we want to be back a little further so we can see more of it
|
||
flMaxDistance *= 2.5f;
|
||
}
|
||
m_flObserverChaseDistance = clamp( m_flObserverChaseDistance, 16, flMaxDistance );
|
||
|
||
AngleVectors( viewangles, &forward );
|
||
|
||
VectorNormalize( forward );
|
||
|
||
VectorMA(origin, -m_flObserverChaseDistance, forward, viewpoint );
|
||
|
||
trace_t trace;
|
||
CTraceFilterNoNPCsOrPlayer filter( target, COLLISION_GROUP_NONE );
|
||
C_BaseEntity::PushEnableAbsRecomputations( false ); // HACK don't recompute positions while doing RayTrace
|
||
UTIL_TraceHull( origin, viewpoint, WALL_MIN, WALL_MAX, MASK_SOLID, &filter, &trace );
|
||
C_BaseEntity::PopEnableAbsRecomputations();
|
||
|
||
if (trace.fraction < 1.0)
|
||
{
|
||
viewpoint = trace.endpos;
|
||
m_flObserverChaseDistance = VectorLength(origin - eyeOrigin);
|
||
}
|
||
|
||
VectorCopy( viewangles, eyeAngles );
|
||
VectorCopy( viewpoint, eyeOrigin );
|
||
|
||
fov = GetFOV();
|
||
}
|
||
|
||
void C_BasePlayer::CalcRoamingView(Vector& eyeOrigin, QAngle& eyeAngles, float& fov)
|
||
{
|
||
C_BaseEntity *target = GetObserverTarget();
|
||
|
||
if ( !target )
|
||
{
|
||
target = this;
|
||
}
|
||
|
||
m_flObserverChaseDistance = 0.0;
|
||
|
||
eyeOrigin = target->EyePosition();
|
||
eyeAngles = target->EyeAngles();
|
||
|
||
if ( spec_track.GetInt() > 0 )
|
||
{
|
||
C_BaseEntity *target = ClientEntityList().GetBaseEntity( spec_track.GetInt() );
|
||
|
||
if ( target )
|
||
{
|
||
Vector v = target->GetAbsOrigin(); v.z += 54;
|
||
QAngle a; VectorAngles( v - eyeOrigin, a );
|
||
|
||
NormalizeAngles( a );
|
||
eyeAngles = a;
|
||
engine->SetViewAngles( a );
|
||
}
|
||
}
|
||
|
||
// Apply a smoothing offset to smooth out prediction errors.
|
||
Vector vSmoothOffset;
|
||
GetPredictionErrorSmoothingVector( vSmoothOffset );
|
||
eyeOrigin += vSmoothOffset;
|
||
|
||
fov = GetFOV();
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: Calculate the view for the player while he's in freeze frame observer mode
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::CalcFreezeCamView( Vector& eyeOrigin, QAngle& eyeAngles, float& fov )
|
||
{
|
||
C_BaseEntity *pTarget = GetObserverTarget();
|
||
if ( !pTarget )
|
||
{
|
||
CalcDeathCamView( eyeOrigin, eyeAngles, fov );
|
||
return;
|
||
}
|
||
|
||
// Zoom towards our target
|
||
float flCurTime = (gpGlobals->curtime - m_flFreezeFrameStartTime);
|
||
float flBlendPerc = clamp( flCurTime / spec_freeze_traveltime.GetFloat(), 0, 1 );
|
||
flBlendPerc = SimpleSpline( flBlendPerc );
|
||
|
||
Vector vecCamDesired = pTarget->GetObserverCamOrigin(); // Returns ragdoll origin if they're ragdolled
|
||
VectorAdd( vecCamDesired, GetChaseCamViewOffset( pTarget ), vecCamDesired );
|
||
Vector vecCamTarget = vecCamDesired;
|
||
if ( pTarget->IsAlive() )
|
||
{
|
||
// Look at their chest, not their head
|
||
Vector maxs = GameRules()->GetViewVectors()->m_vHullMax;
|
||
vecCamTarget.z -= (maxs.z * 0.5);
|
||
}
|
||
else
|
||
{
|
||
vecCamTarget.z += VEC_DEAD_VIEWHEIGHT.z; // look over ragdoll, not through
|
||
}
|
||
|
||
// Figure out a view position in front of the target
|
||
Vector vecEyeOnPlane = eyeOrigin;
|
||
vecEyeOnPlane.z = vecCamTarget.z;
|
||
Vector vecTargetPos = vecCamTarget;
|
||
Vector vecToTarget = vecTargetPos - vecEyeOnPlane;
|
||
VectorNormalize( vecToTarget );
|
||
|
||
// Stop a few units away from the target, and shift up to be at the same height
|
||
vecTargetPos = vecCamTarget - (vecToTarget * m_flFreezeFrameDistance);
|
||
float flEyePosZ = pTarget->EyePosition().z;
|
||
vecTargetPos.z = flEyePosZ + m_flFreezeZOffset;
|
||
|
||
// Now trace out from the target, so that we're put in front of any walls
|
||
trace_t trace;
|
||
C_BaseEntity::PushEnableAbsRecomputations( false ); // HACK don't recompute positions while doing RayTrace
|
||
UTIL_TraceHull( vecCamTarget, vecTargetPos, WALL_MIN, WALL_MAX, MASK_SOLID, pTarget, COLLISION_GROUP_NONE, &trace );
|
||
C_BaseEntity::PopEnableAbsRecomputations();
|
||
if (trace.fraction < 1.0)
|
||
{
|
||
// The camera's going to be really close to the target. So we don't end up
|
||
// looking at someone's chest, aim close freezecams at the target's eyes.
|
||
vecTargetPos = trace.endpos;
|
||
vecCamTarget = vecCamDesired;
|
||
|
||
// To stop all close in views looking up at character's chins, move the view up.
|
||
vecTargetPos.z += fabs(vecCamTarget.z - vecTargetPos.z) * 0.85;
|
||
C_BaseEntity::PushEnableAbsRecomputations( false ); // HACK don't recompute positions while doing RayTrace
|
||
UTIL_TraceHull( vecCamTarget, vecTargetPos, WALL_MIN, WALL_MAX, MASK_SOLID, pTarget, COLLISION_GROUP_NONE, &trace );
|
||
C_BaseEntity::PopEnableAbsRecomputations();
|
||
vecTargetPos = trace.endpos;
|
||
}
|
||
|
||
// Look directly at the target
|
||
vecToTarget = vecCamTarget - vecTargetPos;
|
||
VectorNormalize( vecToTarget );
|
||
VectorAngles( vecToTarget, eyeAngles );
|
||
|
||
VectorLerp( m_vecFreezeFrameStart, vecTargetPos, flBlendPerc, eyeOrigin );
|
||
|
||
if ( flCurTime >= spec_freeze_traveltime.GetFloat() && !m_bSentFreezeFrame )
|
||
{
|
||
IGameEvent *pEvent = gameeventmanager->CreateEvent( "freezecam_started" );
|
||
if ( pEvent )
|
||
{
|
||
gameeventmanager->FireEventClientSide( pEvent );
|
||
}
|
||
|
||
m_bSentFreezeFrame = true;
|
||
view->FreezeFrame( spec_freeze_time.GetFloat() );
|
||
}
|
||
}
|
||
|
||
void C_BasePlayer::CalcInEyeCamView(Vector& eyeOrigin, QAngle& eyeAngles, float& fov)
|
||
{
|
||
C_BaseEntity *target = GetObserverTarget();
|
||
|
||
if ( !target )
|
||
{
|
||
// just copy a save in-map position
|
||
VectorCopy( EyePosition(), eyeOrigin );
|
||
VectorCopy( EyeAngles(), eyeAngles );
|
||
return;
|
||
};
|
||
|
||
if ( !target->IsAlive() )
|
||
{
|
||
// if dead, show from 3rd person
|
||
CalcChaseCamView( eyeOrigin, eyeAngles, fov );
|
||
return;
|
||
}
|
||
|
||
fov = GetFOV(); // TODO use tragets FOV
|
||
|
||
m_flObserverChaseDistance = 0.0;
|
||
|
||
eyeAngles = target->EyeAngles();
|
||
eyeOrigin = target->GetAbsOrigin();
|
||
|
||
// Apply punch angle
|
||
VectorAdd( eyeAngles, GetPunchAngle(), eyeAngles );
|
||
|
||
#if defined( REPLAY_ENABLED )
|
||
if( g_bEngineIsHLTV || engine->IsReplay() )
|
||
#else
|
||
if( g_bEngineIsHLTV )
|
||
#endif
|
||
{
|
||
if ( target->GetFlags() & FL_DUCKING )
|
||
{
|
||
eyeOrigin += VEC_DUCK_VIEW;
|
||
}
|
||
else
|
||
{
|
||
eyeOrigin += VEC_VIEW;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
Vector offset = m_vecViewOffset;
|
||
eyeOrigin += offset; // hack hack
|
||
}
|
||
|
||
engine->SetViewAngles( eyeAngles );
|
||
}
|
||
|
||
void C_BasePlayer::CalcDeathCamView(Vector& eyeOrigin, QAngle& eyeAngles, float& fov)
|
||
{
|
||
CBaseEntity * pKiller = NULL;
|
||
|
||
if ( mp_forcecamera.GetInt() == OBS_ALLOW_ALL )
|
||
{
|
||
// if mp_forcecamera is off let user see killer or look around
|
||
pKiller = GetObserverTarget();
|
||
eyeAngles = EyeAngles();
|
||
}
|
||
|
||
float interpolation = ( gpGlobals->curtime - m_flDeathTime ) / DEATH_ANIMATION_TIME;
|
||
interpolation = clamp( interpolation, 0.0f, 1.0f );
|
||
|
||
m_flObserverChaseDistance += gpGlobals->frametime*48.0f;
|
||
m_flObserverChaseDistance = clamp( m_flObserverChaseDistance, 16, CHASE_CAM_DISTANCE );
|
||
|
||
QAngle aForward = eyeAngles;
|
||
Vector origin = EyePosition();
|
||
|
||
IRagdoll *pRagdoll = GetRepresentativeRagdoll();
|
||
if ( pRagdoll )
|
||
{
|
||
origin = pRagdoll->GetRagdollOrigin();
|
||
origin.z += VEC_DEAD_VIEWHEIGHT.z; // look over ragdoll, not through
|
||
}
|
||
|
||
if ( pKiller && pKiller->IsPlayer() && (pKiller != this) )
|
||
{
|
||
Vector vKiller = pKiller->EyePosition() - origin;
|
||
QAngle aKiller; VectorAngles( vKiller, aKiller );
|
||
InterpolateAngles( aForward, aKiller, eyeAngles, interpolation );
|
||
};
|
||
|
||
Vector vForward; AngleVectors( eyeAngles, &vForward );
|
||
|
||
VectorNormalize( vForward );
|
||
|
||
VectorMA( origin, -m_flObserverChaseDistance, vForward, eyeOrigin );
|
||
|
||
trace_t trace; // clip against world
|
||
C_BaseEntity::PushEnableAbsRecomputations( false ); // HACK don't recompute positions while doing RayTrace
|
||
UTIL_TraceHull( origin, eyeOrigin, WALL_MIN, WALL_MAX, MASK_SOLID, this, COLLISION_GROUP_NONE, &trace );
|
||
C_BaseEntity::PopEnableAbsRecomputations();
|
||
|
||
if (trace.fraction < 1.0)
|
||
{
|
||
eyeOrigin = trace.endpos;
|
||
m_flObserverChaseDistance = VectorLength(origin - eyeOrigin);
|
||
}
|
||
|
||
fov = GetFOV();
|
||
}
|
||
|
||
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: Return the weapon to have open the weapon selection on, based upon our currently active weapon
|
||
// Base class just uses the weapon that's currently active.
|
||
//-----------------------------------------------------------------------------
|
||
C_BaseCombatWeapon *C_BasePlayer::GetActiveWeaponForSelection( void )
|
||
{
|
||
return GetActiveWeapon();
|
||
}
|
||
|
||
C_BaseAnimating* C_BasePlayer::GetRenderedWeaponModel()
|
||
{
|
||
// Attach to either their weapon model or their view model.
|
||
if ( ShouldDrawLocalPlayer() || !IsLocalPlayer( this ) )
|
||
{
|
||
return GetActiveWeapon();
|
||
}
|
||
else
|
||
{
|
||
return GetViewModel();
|
||
}
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: Gets a pointer to the local player, if it exists yet.
|
||
// static method
|
||
//-----------------------------------------------------------------------------
|
||
C_BasePlayer *C_BasePlayer::GetLocalPlayer( int nSlot /*= -1*/ )
|
||
{
|
||
if ( nSlot == -1 )
|
||
{
|
||
// ASSERT_LOCAL_PLAYER_RESOLVABLE();
|
||
return s_pLocalPlayer[ GET_ACTIVE_SPLITSCREEN_SLOT() ];
|
||
}
|
||
return s_pLocalPlayer[ nSlot ];
|
||
}
|
||
|
||
void C_BasePlayer::SetRemoteSplitScreenPlayerViewsAreLocalPlayer( bool bSet )
|
||
{
|
||
for( int i = 0; i != MAX_SPLITSCREEN_PLAYERS; ++i )
|
||
{
|
||
if( !IsLocalSplitScreenPlayer( i ) )
|
||
{
|
||
s_pLocalPlayer[i] = bSet ? GetSplitScreenViewPlayer( i ) : NULL;
|
||
}
|
||
}
|
||
}
|
||
|
||
bool C_BasePlayer::HasAnyLocalPlayer()
|
||
{
|
||
FOR_EACH_VALID_SPLITSCREEN_PLAYER( i )
|
||
{
|
||
if ( s_pLocalPlayer[ i ] )
|
||
return true;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
int C_BasePlayer::GetSplitScreenSlotForPlayer( C_BaseEntity *pl )
|
||
{
|
||
C_BasePlayer *pPlayer = ToBasePlayer( pl );
|
||
|
||
if ( !pPlayer )
|
||
{
|
||
Assert( 0 );
|
||
return -1;
|
||
}
|
||
return pPlayer->GetSplitScreenPlayerSlot();
|
||
}
|
||
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose:
|
||
// Input : bThirdperson -
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::ThirdPersonSwitch( bool bThirdperson )
|
||
{
|
||
// We've switch from first to third, or vice versa.
|
||
UpdateVisibility();
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: single place to decide whether the local player should draw
|
||
//-----------------------------------------------------------------------------
|
||
bool C_BasePlayer::ShouldDrawLocalPlayer()
|
||
{
|
||
int nSlot = GetSplitScreenPlayerSlot();
|
||
|
||
|
||
|
||
ACTIVE_SPLITSCREEN_PLAYER_GUARD( nSlot );
|
||
return input->CAM_IsThirdPerson() || ( ToolsEnabled() && ToolFramework_IsThirdPersonCamera() );
|
||
}
|
||
|
||
//----------------------------------------------------------------------------
|
||
// Hooks into the fast path render system
|
||
//----------------------------------------------------------------------------
|
||
IClientModelRenderable *C_BasePlayer::GetClientModelRenderable()
|
||
{
|
||
// Honor base class eligibility
|
||
if ( !BaseClass::GetClientModelRenderable() )
|
||
return NULL;
|
||
|
||
// No fast path for firstperson local players
|
||
if ( IsLocalPlayer( this ) )
|
||
{
|
||
bool bThirdPerson = input->CAM_IsThirdPerson() || ( ToolsEnabled() && ToolFramework_IsThirdPersonCamera() );
|
||
if ( !bThirdPerson )
|
||
{
|
||
return NULL;
|
||
}
|
||
}
|
||
|
||
// if local player is spectating this player in first person mode, don't use fast path, so we can skip drawing it
|
||
C_BasePlayer *localPlayer = C_BasePlayer::GetLocalPlayer();
|
||
if ( localPlayer && localPlayer->IsObserver() )
|
||
{
|
||
if ( localPlayer->GetObserverMode() == OBS_MODE_IN_EYE &&
|
||
localPlayer->GetObserverTarget() == this &&
|
||
!input->CAM_IsThirdPerson() )
|
||
return NULL;
|
||
}
|
||
|
||
// don't use fastpath for teammates (causes extra work for glows)
|
||
if ( localPlayer && localPlayer->GetTeamNumber() == GetTeamNumber() )
|
||
{
|
||
return NULL;
|
||
}
|
||
|
||
return this;
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose:
|
||
// Output : Returns true on success, false on failure.
|
||
//-----------------------------------------------------------------------------
|
||
bool C_BasePlayer::IsLocalPlayer( const C_BaseEntity *pEntity )
|
||
{
|
||
if ( !pEntity ||
|
||
!pEntity->IsPlayer() )
|
||
return false;
|
||
|
||
return static_cast< const C_BasePlayer * >( pEntity )->m_bIsLocalPlayer;
|
||
}
|
||
|
||
int C_BasePlayer::GetUserID( void ) const
|
||
{
|
||
player_info_t pi;
|
||
|
||
if ( !engine->GetPlayerInfo( entindex(), &pi ) )
|
||
return -1;
|
||
|
||
return pi.userID;
|
||
}
|
||
|
||
|
||
// For weapon prediction
|
||
void C_BasePlayer::SetAnimation( PLAYER_ANIM playerAnim )
|
||
{
|
||
// FIXME
|
||
}
|
||
|
||
void C_BasePlayer::UpdateClientData( void )
|
||
{
|
||
// Update all the items
|
||
for ( int i = 0; i < WeaponCount(); i++ )
|
||
{
|
||
if ( GetWeapon(i) ) // each item updates it's successors
|
||
GetWeapon(i)->UpdateClientData( this );
|
||
}
|
||
}
|
||
|
||
// Prediction stuff
|
||
void C_BasePlayer::PreThink( void )
|
||
{
|
||
#if !defined( NO_ENTITY_PREDICTION )
|
||
ItemPreFrame();
|
||
|
||
UpdateClientData();
|
||
|
||
UpdateUnderwaterState();
|
||
|
||
// Update the player's fog data if necessary.
|
||
UpdateFogController();
|
||
|
||
if (m_lifeState >= LIFE_DYING)
|
||
return;
|
||
|
||
//
|
||
// If we're not on the ground, we're falling. Update our falling velocity.
|
||
//
|
||
if ( !( GetFlags() & FL_ONGROUND ) )
|
||
{
|
||
m_Local.m_flFallVelocity = -GetAbsVelocity().z;
|
||
}
|
||
#endif
|
||
}
|
||
|
||
void C_BasePlayer::PostThink( void )
|
||
{
|
||
#if !defined( NO_ENTITY_PREDICTION )
|
||
MDLCACHE_CRITICAL_SECTION();
|
||
|
||
if ( IsAlive())
|
||
{
|
||
UpdateCollisionBounds();
|
||
|
||
if ( !CommentaryModeShouldSwallowInput( this ) )
|
||
{
|
||
// do weapon stuff
|
||
ItemPostFrame();
|
||
}
|
||
|
||
if ( GetFlags() & FL_ONGROUND )
|
||
{
|
||
m_Local.m_flFallVelocity = 0;
|
||
}
|
||
|
||
// Don't allow bogus sequence on player
|
||
if ( GetSequence() == -1 )
|
||
{
|
||
SetSequence( 0 );
|
||
}
|
||
|
||
StudioFrameAdvance();
|
||
}
|
||
|
||
// Even if dead simulate entities
|
||
SimulatePlayerSimulatedEntities();
|
||
#endif
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: send various tool messages - viewoffset, and base class messages (flex and bones)
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::GetToolRecordingState( KeyValues *msg )
|
||
{
|
||
if ( !ToolsEnabled() )
|
||
return;
|
||
|
||
VPROF_BUDGET( "C_BasePlayer::GetToolRecordingState", VPROF_BUDGETGROUP_TOOLS );
|
||
|
||
BaseClass::GetToolRecordingState( msg );
|
||
|
||
msg->SetBool( "baseplayer", true );
|
||
msg->SetBool( "localplayer", IsLocalPlayer( this ) );
|
||
msg->SetString( "playername", GetPlayerName() );
|
||
|
||
static CameraRecordingState_t state;
|
||
state.m_flFOV = GetFOV();
|
||
|
||
float flZNear = view->GetZNear();
|
||
float flZFar = view->GetZFar();
|
||
CalcView( state.m_vecEyePosition, state.m_vecEyeAngles, flZNear, flZFar, state.m_flFOV );
|
||
state.m_bThirdPerson = !engine->IsPaused() && ::input->CAM_IsThirdPerson();
|
||
|
||
// this is a straight copy from ClientModeShared::OverrideView,
|
||
// When that method is removed in favor of rolling it into CalcView,
|
||
// then this code can (should!) be removed
|
||
if ( state.m_bThirdPerson )
|
||
{
|
||
Vector cam_ofs;
|
||
::input->CAM_GetCameraOffset( cam_ofs );
|
||
|
||
QAngle camAngles;
|
||
camAngles[ PITCH ] = cam_ofs[ PITCH ];
|
||
camAngles[ YAW ] = cam_ofs[ YAW ];
|
||
camAngles[ ROLL ] = 0;
|
||
|
||
Vector camForward, camRight, camUp;
|
||
AngleVectors( camAngles, &camForward, &camRight, &camUp );
|
||
|
||
VectorMA( state.m_vecEyePosition, -cam_ofs[ ROLL ], camForward, state.m_vecEyePosition );
|
||
|
||
// Override angles from third person camera
|
||
VectorCopy( camAngles, state.m_vecEyeAngles );
|
||
}
|
||
|
||
msg->SetPtr( "camera", &state );
|
||
}
|
||
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: Simulate the player for this frame
|
||
//-----------------------------------------------------------------------------
|
||
bool C_BasePlayer::Simulate()
|
||
{
|
||
//Frame updates
|
||
if ( C_BasePlayer::IsLocalPlayer( this ) )
|
||
{
|
||
ACTIVE_SPLITSCREEN_PLAYER_GUARD_ENT( this );
|
||
|
||
//Update the flashlight
|
||
Flashlight();
|
||
|
||
// Update the player's fog data if necessary.
|
||
UpdateFogController();
|
||
}
|
||
else
|
||
{
|
||
// update step sounds for all other players
|
||
Vector vel;
|
||
EstimateAbsVelocity( vel );
|
||
UpdateStepSound( GetGroundSurface(), GetAbsOrigin(), vel );
|
||
}
|
||
|
||
BaseClass::Simulate();
|
||
|
||
// Server says don't interpolate this frame, so set previous info to new info.
|
||
if ( IsEffectActive( EF_NOINTERP ) || Teleported() )
|
||
{
|
||
ResetLatched();
|
||
}
|
||
return true;
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose:
|
||
// Output : CBaseViewModel
|
||
//-----------------------------------------------------------------------------
|
||
C_BaseViewModel *C_BasePlayer::GetViewModel( int index /*= 0*/ )
|
||
{
|
||
Assert( index >= 0 && index < MAX_VIEWMODELS );
|
||
|
||
C_BaseViewModel *vm = m_hViewModel[ index ];
|
||
|
||
if ( GetObserverMode() == OBS_MODE_IN_EYE )
|
||
{
|
||
C_BasePlayer *target = ToBasePlayer( GetObserverTarget() );
|
||
|
||
// get the targets viewmodel unless the target is an observer itself
|
||
if ( target && target != this && !target->IsObserver() )
|
||
{
|
||
vm = target->GetViewModel( index );
|
||
}
|
||
}
|
||
|
||
return vm;
|
||
}
|
||
|
||
C_BaseCombatWeapon *C_BasePlayer::GetActiveWeapon( void ) const
|
||
{
|
||
const C_BasePlayer *fromPlayer = this;
|
||
|
||
// if localplayer is in InEye spectator mode, return weapon on chased player
|
||
if ( ( C_BasePlayer::IsLocalPlayer( const_cast< C_BasePlayer * >( fromPlayer) ) ) && ( GetObserverMode() == OBS_MODE_IN_EYE) )
|
||
{
|
||
C_BaseEntity *target = GetObserverTarget();
|
||
|
||
if ( target && target->IsPlayer() )
|
||
{
|
||
fromPlayer = ToBasePlayer( target );
|
||
}
|
||
}
|
||
|
||
return fromPlayer->C_BaseCombatCharacter::GetActiveWeapon();
|
||
}
|
||
|
||
//=========================================================
|
||
// Autoaim
|
||
// set crosshair position to point to enemey
|
||
//=========================================================
|
||
Vector C_BasePlayer::GetAutoaimVector( float flScale )
|
||
{
|
||
// Never autoaim a predicted weapon (for now)
|
||
Vector forward;
|
||
AngleVectors( GetAbsAngles() + m_Local.m_vecPunchAngle, &forward );
|
||
return forward;
|
||
}
|
||
|
||
void C_BasePlayer::PlayPlayerJingle()
|
||
{
|
||
if ( !cl_customsounds.GetBool() )
|
||
return;
|
||
|
||
// Find player sound for shooter
|
||
player_info_t info;
|
||
engine->GetPlayerInfo( entindex(), &info );
|
||
|
||
// Doesn't have a jingle sound
|
||
if ( !info.customFiles[1] )
|
||
return;
|
||
|
||
char soundhex[ 16 ];
|
||
Q_binarytohex( (byte *)&info.customFiles[1], sizeof( info.customFiles[1] ), soundhex, sizeof( soundhex ) );
|
||
|
||
// See if logo has been downloaded.
|
||
char fullsoundname[ 512 ];
|
||
Q_snprintf( fullsoundname, sizeof( fullsoundname ), "sound/temp/%s.wav", soundhex );
|
||
|
||
if ( !filesystem->FileExists( fullsoundname ) )
|
||
{
|
||
char custname[ 512 ];
|
||
Q_snprintf( custname, sizeof( custname ), "downloads/%s.dat", soundhex );
|
||
// it may have been downloaded but not copied under materials folder
|
||
if ( !filesystem->FileExists( custname ) )
|
||
return; // not downloaded yet
|
||
|
||
// copy from download folder to materials/temp folder
|
||
// this is done since material system can access only materials/*.vtf files
|
||
|
||
if ( !engine->CopyFile( custname, fullsoundname) )
|
||
return;
|
||
}
|
||
|
||
Q_snprintf( fullsoundname, sizeof( fullsoundname ), "temp/%s.wav", soundhex );
|
||
|
||
CLocalPlayerFilter filter;
|
||
|
||
EmitSound_t ep;
|
||
ep.m_nChannel = CHAN_VOICE;
|
||
ep.m_pSoundName = fullsoundname;
|
||
ep.m_flVolume = VOL_NORM;
|
||
ep.m_SoundLevel = SNDLVL_NORM;
|
||
|
||
C_BaseEntity::EmitSound( filter, GetSoundSourceIndex(), ep );
|
||
}
|
||
|
||
// Stuff for prediction
|
||
void C_BasePlayer::SetSuitUpdate(char *name, int fgroup, int iNoRepeat)
|
||
{
|
||
// FIXME: Do something here?
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose:
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::ResetAutoaim( void )
|
||
{
|
||
#if 0
|
||
if (m_vecAutoAim.x != 0 || m_vecAutoAim.y != 0)
|
||
{
|
||
m_vecAutoAim = QAngle( 0, 0, 0 );
|
||
engine->CrosshairAngle( edict(), 0, 0 );
|
||
}
|
||
#endif
|
||
m_fOnTarget = false;
|
||
}
|
||
|
||
bool C_BasePlayer::ShouldPredict( void )
|
||
{
|
||
#if !defined( NO_ENTITY_PREDICTION )
|
||
// Do this before calling into baseclass so prediction data block gets allocated
|
||
if ( IsLocalPlayer( this ) )
|
||
{
|
||
return true;
|
||
}
|
||
#endif
|
||
return false;
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: Return the player who will predict this entity
|
||
//-----------------------------------------------------------------------------
|
||
C_BasePlayer *C_BasePlayer::GetPredictionOwner( void )
|
||
{
|
||
return this;
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: Special processing for player simulation
|
||
// NOTE: Don't chain to BaseClass!!!!
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::PhysicsSimulate( void )
|
||
{
|
||
#if !defined( NO_ENTITY_PREDICTION )
|
||
VPROF( "C_BasePlayer::PhysicsSimulate" );
|
||
// If we've got a moveparent, we must simulate that first.
|
||
CBaseEntity *pMoveParent = GetMoveParent();
|
||
if (pMoveParent)
|
||
{
|
||
pMoveParent->PhysicsSimulate();
|
||
}
|
||
|
||
// Make sure not to simulate this guy twice per frame
|
||
if (m_nSimulationTick == gpGlobals->tickcount)
|
||
return;
|
||
|
||
m_nSimulationTick = gpGlobals->tickcount;
|
||
|
||
if ( !IsLocalPlayer( this ) )
|
||
return;
|
||
|
||
C_CommandContext *ctx = GetCommandContext();
|
||
Assert( ctx );
|
||
Assert( ctx->needsprocessing );
|
||
if ( !ctx->needsprocessing )
|
||
return;
|
||
|
||
ctx->needsprocessing = false;
|
||
|
||
// Handle FL_FROZEN.
|
||
if(GetFlags() & FL_FROZEN)
|
||
{
|
||
ctx->cmd.forwardmove = 0;
|
||
ctx->cmd.sidemove = 0;
|
||
ctx->cmd.upmove = 0;
|
||
ctx->cmd.buttons = 0;
|
||
ctx->cmd.impulse = 0;
|
||
//VectorCopy ( pl.v_angle, ctx->cmd.viewangles );
|
||
}
|
||
|
||
// Run the next command
|
||
MoveHelper()->SetHost( this );
|
||
prediction->RunCommand(
|
||
this,
|
||
&ctx->cmd,
|
||
MoveHelper() );
|
||
MoveHelper()->SetHost( NULL );
|
||
#endif
|
||
}
|
||
|
||
const QAngle& C_BasePlayer::GetPunchAngle()
|
||
{
|
||
return m_Local.m_vecPunchAngle.Get();
|
||
}
|
||
|
||
|
||
void C_BasePlayer::SetPunchAngle( const QAngle &angle )
|
||
{
|
||
m_Local.m_vecPunchAngle = angle;
|
||
}
|
||
|
||
|
||
float C_BasePlayer::GetWaterJumpTime() const
|
||
{
|
||
return m_flWaterJumpTime;
|
||
}
|
||
|
||
void C_BasePlayer::SetWaterJumpTime( float flWaterJumpTime )
|
||
{
|
||
m_flWaterJumpTime = flWaterJumpTime;
|
||
}
|
||
|
||
float C_BasePlayer::GetSwimSoundTime() const
|
||
{
|
||
return m_flSwimSoundTime;
|
||
}
|
||
|
||
void C_BasePlayer::SetSwimSoundTime( float flSwimSoundTime )
|
||
{
|
||
m_flSwimSoundTime = flSwimSoundTime;
|
||
}
|
||
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: Return true if this object can be +used by the player
|
||
//-----------------------------------------------------------------------------
|
||
bool C_BasePlayer::IsUseableEntity( CBaseEntity *pEntity, unsigned int requiredCaps )
|
||
{
|
||
return false;
|
||
}
|
||
|
||
C_BaseEntity* C_BasePlayer::GetUseEntity( void ) const
|
||
{
|
||
return m_hUseEntity;
|
||
}
|
||
|
||
C_BaseEntity* C_BasePlayer::GetPotentialUseEntity( void ) const
|
||
{
|
||
return GetUseEntity();
|
||
}
|
||
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose:
|
||
// Output : float
|
||
//-----------------------------------------------------------------------------
|
||
float C_BasePlayer::GetFOV( void ) const
|
||
{
|
||
if ( GetObserverMode() == OBS_MODE_IN_EYE )
|
||
{
|
||
C_BasePlayer *pTargetPlayer = ToBasePlayer( GetObserverTarget() );
|
||
|
||
// get fov from observer target. Not if target is observer itself
|
||
if ( pTargetPlayer && !pTargetPlayer->IsObserver() )
|
||
{
|
||
return pTargetPlayer->GetFOV();
|
||
}
|
||
}
|
||
|
||
// Allow our vehicle to override our FOV if it's currently at the default FOV.
|
||
float flDefaultFOV;
|
||
IClientVehicle *pVehicle = const_cast< C_BasePlayer * >(this)->GetVehicle();
|
||
if ( pVehicle )
|
||
{
|
||
const_cast< C_BasePlayer * >(this)->CacheVehicleView();
|
||
flDefaultFOV = ( m_flVehicleViewFOV == 0 ) ? GetDefaultFOV() : m_flVehicleViewFOV;
|
||
}
|
||
else
|
||
{
|
||
flDefaultFOV = GetDefaultFOV();
|
||
}
|
||
|
||
float fFOV = ( m_iFOV == 0 ) ? flDefaultFOV : m_iFOV;
|
||
|
||
// Don't do lerping during prediction. It's only necessary when actually rendering,
|
||
// and it'll cause problems due to prediction timing messiness.
|
||
if ( !prediction->InPrediction() )
|
||
{
|
||
// See if we need to lerp the values for local player
|
||
if ( IsLocalPlayer( this ) && ( fFOV != m_iFOVStart ) && (m_Local.m_flFOVRate > 0.0f ) )
|
||
{
|
||
float deltaTime = (float)( gpGlobals->curtime - m_flFOVTime ) / m_Local.m_flFOVRate;
|
||
|
||
#if !defined( NO_ENTITY_PREDICTION )
|
||
if ( GetPredictable() )
|
||
{
|
||
// m_flFOVTime was set to a predicted time in the future, because the FOV change was predicted.
|
||
deltaTime = (float)( GetFinalPredictedTime() - m_flFOVTime );
|
||
deltaTime += ( gpGlobals->interpolation_amount * TICK_INTERVAL );
|
||
deltaTime /= m_Local.m_flFOVRate;
|
||
}
|
||
#endif
|
||
|
||
if ( deltaTime >= 1.0f )
|
||
{
|
||
//If we're past the zoom time, just take the new value and stop lerping
|
||
const_cast<C_BasePlayer *>(this)->m_iFOVStart = fFOV;
|
||
}
|
||
else
|
||
{
|
||
fFOV = SimpleSplineRemapValClamped( deltaTime, 0.0f, 1.0f, (float) m_iFOVStart, fFOV );
|
||
}
|
||
}
|
||
}
|
||
|
||
return fFOV;
|
||
}
|
||
|
||
void C_BasePlayer::RecvProxy_LocalVelocityX( const CRecvProxyData *pData, void *pStruct, void *pOut )
|
||
{
|
||
C_BasePlayer *pPlayer = (C_BasePlayer *) pStruct;
|
||
|
||
Assert( pPlayer );
|
||
|
||
float flNewVel_x = pData->m_Value.m_Float;
|
||
|
||
Vector vecVelocity = pPlayer->GetLocalVelocity();
|
||
|
||
if( vecVelocity.x != flNewVel_x ) // Should this use an epsilon check?
|
||
{
|
||
vecVelocity.x = flNewVel_x;
|
||
pPlayer->SetLocalVelocity( vecVelocity );
|
||
}
|
||
}
|
||
|
||
void C_BasePlayer::RecvProxy_LocalVelocityY( const CRecvProxyData *pData, void *pStruct, void *pOut )
|
||
{
|
||
C_BasePlayer *pPlayer = (C_BasePlayer *) pStruct;
|
||
|
||
Assert( pPlayer );
|
||
|
||
float flNewVel_y = pData->m_Value.m_Float;
|
||
|
||
Vector vecVelocity = pPlayer->GetLocalVelocity();
|
||
|
||
if( vecVelocity.y != flNewVel_y )
|
||
{
|
||
vecVelocity.y = flNewVel_y;
|
||
pPlayer->SetLocalVelocity( vecVelocity );
|
||
}
|
||
}
|
||
|
||
void C_BasePlayer::RecvProxy_LocalVelocityZ( const CRecvProxyData *pData, void *pStruct, void *pOut )
|
||
{
|
||
C_BasePlayer *pPlayer = (C_BasePlayer *) pStruct;
|
||
|
||
Assert( pPlayer );
|
||
|
||
float flNewVel_z = pData->m_Value.m_Float;
|
||
|
||
Vector vecVelocity = pPlayer->GetLocalVelocity();
|
||
|
||
if( vecVelocity.z != flNewVel_z )
|
||
{
|
||
vecVelocity.z = flNewVel_z;
|
||
pPlayer->SetLocalVelocity( vecVelocity );
|
||
}
|
||
}
|
||
|
||
void C_BasePlayer::RecvProxy_ObserverMode( const CRecvProxyData *pData, void *pStruct, void *pOut )
|
||
{
|
||
RecvProxy_Int32ToInt32( pData, pStruct, pOut );
|
||
|
||
C_BasePlayer *pPlayer = (C_BasePlayer *) pStruct;
|
||
Assert( pPlayer );
|
||
if ( C_BasePlayer::IsLocalPlayer( pPlayer ) )
|
||
{
|
||
UpdateViewmodelVisibility( pPlayer );
|
||
}
|
||
|
||
}
|
||
|
||
void C_BasePlayer::RecvProxy_ObserverTarget( const CRecvProxyData *pData, void *pStruct, void *pOut )
|
||
{
|
||
C_BasePlayer *pPlayer = (C_BasePlayer *) pStruct;
|
||
|
||
Assert( pPlayer );
|
||
|
||
EHANDLE hTarget;
|
||
|
||
RecvProxy_IntToEHandle( pData, pStruct, &hTarget );
|
||
|
||
pPlayer->SetObserverTarget( hTarget );
|
||
}
|
||
|
||
void C_BasePlayer::RecvProxy_LocalOriginXY( const CRecvProxyData *pData, void *pStruct, void *pOut )
|
||
{
|
||
((float*)pOut)[0] = pData->m_Value.m_Vector[0];
|
||
((float*)pOut)[1] = pData->m_Value.m_Vector[1];
|
||
}
|
||
|
||
void C_BasePlayer::RecvProxy_LocalOriginZ( const CRecvProxyData *pData, void *pStruct, void *pOut )
|
||
{
|
||
*((float*)pOut) = pData->m_Value.m_Float;
|
||
}
|
||
|
||
void C_BasePlayer::RecvProxy_NonLocalOriginXY( const CRecvProxyData *pData, void *pStruct, void *pOut )
|
||
{
|
||
((float*)pOut)[0] = pData->m_Value.m_Vector[0];
|
||
((float*)pOut)[1] = pData->m_Value.m_Vector[1];
|
||
}
|
||
|
||
void C_BasePlayer::RecvProxy_NonLocalOriginZ( const CRecvProxyData *pData, void *pStruct, void *pOut )
|
||
{
|
||
*((float*)pOut) = pData->m_Value.m_Float;
|
||
}
|
||
|
||
void C_BasePlayer::RecvProxy_NonLocalCellOriginXY( const CRecvProxyData *pData, void *pStruct, void *pOut )
|
||
{
|
||
C_BasePlayer *player = (C_BasePlayer *) pStruct;
|
||
|
||
player->m_vecCellOrigin.x = pData->m_Value.m_Vector[0];
|
||
player->m_vecCellOrigin.y = pData->m_Value.m_Vector[1];
|
||
|
||
register int const cellwidth = player->m_cellwidth; // Load it into a register
|
||
((float*)pOut)[0] = CoordFromCell( cellwidth, player->m_cellX, pData->m_Value.m_Vector[0] );
|
||
((float*)pOut)[1] = CoordFromCell( cellwidth, player->m_cellY, pData->m_Value.m_Vector[1] );
|
||
}
|
||
|
||
void C_BasePlayer::RecvProxy_NonLocalCellOriginZ( const CRecvProxyData *pData, void *pStruct, void *pOut )
|
||
{
|
||
C_BasePlayer *player = (C_BasePlayer *) pStruct;
|
||
|
||
player->m_vecCellOrigin.z = pData->m_Value.m_Float;
|
||
|
||
register int const cellwidth = player->m_cellwidth; // Load it into a register
|
||
*((float*)pOut) = CoordFromCell( cellwidth, player->m_cellZ, pData->m_Value.m_Float );
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: Remove this player from a vehicle
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::LeaveVehicle( void )
|
||
{
|
||
if ( NULL == m_hVehicle.Get() )
|
||
return;
|
||
|
||
// Let server do this for now
|
||
#if 0
|
||
IClientVehicle *pVehicle = GetVehicle();
|
||
Assert( pVehicle );
|
||
|
||
int nRole = pVehicle->GetPassengerRole( this );
|
||
Assert( nRole != VEHICLE_ROLE_NONE );
|
||
|
||
SetParent( NULL );
|
||
|
||
// Find the first non-blocked exit point:
|
||
Vector vNewPos = GetAbsOrigin();
|
||
QAngle qAngles = GetAbsAngles();
|
||
pVehicle->GetPassengerExitPoint( nRole, &vNewPos, &qAngles );
|
||
OnVehicleEnd( vNewPos );
|
||
SetAbsOrigin( vNewPos );
|
||
SetAbsAngles( qAngles );
|
||
|
||
m_Local.m_iHideHUD &= ~HIDEHUD_WEAPONSELECTION;
|
||
RemoveEffects( EF_NODRAW );
|
||
|
||
SetMoveType( MOVETYPE_WALK );
|
||
SetCollisionGroup( COLLISION_GROUP_PLAYER );
|
||
|
||
qAngles[ROLL] = 0;
|
||
SnapEyeAngles( qAngles );
|
||
|
||
m_hVehicle = NULL;
|
||
pVehicle->SetPassenger(nRole, NULL);
|
||
|
||
Weapon_Switch( m_hLastWeapon );
|
||
#endif
|
||
}
|
||
|
||
|
||
float C_BasePlayer::GetMinFOV() const
|
||
{
|
||
if ( gpGlobals->maxClients == 1 )
|
||
{
|
||
// Let them do whatever they want, more or less, in single player
|
||
return 5;
|
||
}
|
||
else
|
||
{
|
||
return 75;
|
||
}
|
||
}
|
||
|
||
float C_BasePlayer::GetFinalPredictedTime() const
|
||
{
|
||
return ( m_nFinalPredictedTick * TICK_INTERVAL );
|
||
}
|
||
|
||
void C_BasePlayer::NotePredictionError( const Vector &vDelta )
|
||
{
|
||
// don't worry about prediction errors when dead
|
||
if ( !IsAlive() )
|
||
return;
|
||
|
||
#if !defined( NO_ENTITY_PREDICTION )
|
||
Vector vOldDelta;
|
||
|
||
GetPredictionErrorSmoothingVector( vOldDelta );
|
||
|
||
// sum all errors within smoothing time
|
||
m_vecPredictionError = vDelta + vOldDelta;
|
||
|
||
// remember when last error happened
|
||
m_flPredictionErrorTime = gpGlobals->curtime;
|
||
|
||
ResetLatched();
|
||
#endif
|
||
}
|
||
|
||
|
||
// offset curtime and setup bones at that time using fake interpolation
|
||
// fake interpolation means we don't have reliable interpolation history (the local player doesn't animate locally)
|
||
// so we just modify cycle and origin directly and use that as a fake guess
|
||
void C_BasePlayer::ForceSetupBonesAtTimeFakeInterpolation( matrix3x4a_t *pBonesOut, float curtimeOffset )
|
||
{
|
||
// we don't have any interpolation data, so fake it
|
||
float cycle = m_flCycle;
|
||
Vector origin = GetLocalOrigin();
|
||
|
||
// blow the cached prev bones
|
||
InvalidateBoneCache();
|
||
// reset root position to flTime
|
||
Interpolate( gpGlobals->curtime + curtimeOffset );
|
||
|
||
// force cycle back by boneDt
|
||
m_flCycle = fmod( 10 + cycle + GetPlaybackRate() * curtimeOffset, 1.0f );
|
||
SetLocalOrigin( origin + curtimeOffset * GetLocalVelocity() );
|
||
// Setup bone state to extrapolate physics velocity
|
||
SetupBones( pBonesOut, MAXSTUDIOBONES, BONE_USED_BY_ANYTHING, gpGlobals->curtime + curtimeOffset );
|
||
|
||
m_flCycle = cycle;
|
||
SetLocalOrigin( origin );
|
||
}
|
||
|
||
void C_BasePlayer::GetRagdollInitBoneArrays( matrix3x4a_t *pDeltaBones0, matrix3x4a_t *pDeltaBones1, matrix3x4a_t *pCurrentBones, float boneDt )
|
||
{
|
||
if ( !C_BasePlayer::IsLocalPlayer( this ) )
|
||
{
|
||
BaseClass::GetRagdollInitBoneArrays(pDeltaBones0, pDeltaBones1, pCurrentBones, boneDt);
|
||
return;
|
||
}
|
||
ForceSetupBonesAtTimeFakeInterpolation( pDeltaBones0, -boneDt );
|
||
ForceSetupBonesAtTimeFakeInterpolation( pDeltaBones1, 0 );
|
||
float ragdollCreateTime = PhysGetSyncCreateTime();
|
||
if ( ragdollCreateTime != gpGlobals->curtime )
|
||
{
|
||
ForceSetupBonesAtTimeFakeInterpolation( pCurrentBones, ragdollCreateTime - gpGlobals->curtime );
|
||
}
|
||
else
|
||
{
|
||
SetupBones( pCurrentBones, MAXSTUDIOBONES, BONE_USED_BY_ANYTHING, gpGlobals->curtime );
|
||
}
|
||
}
|
||
|
||
|
||
void C_BasePlayer::GetPredictionErrorSmoothingVector( Vector &vOffset )
|
||
{
|
||
#if !defined( NO_ENTITY_PREDICTION )
|
||
if ( engine->IsPlayingDemo() || !cl_smooth.GetInt() || !cl_predict->GetInt() || engine->IsPaused() )
|
||
{
|
||
vOffset.Init();
|
||
return;
|
||
}
|
||
|
||
float errorAmount = ( gpGlobals->curtime - m_flPredictionErrorTime ) / cl_smoothtime.GetFloat();
|
||
|
||
if ( errorAmount >= 1.0f )
|
||
{
|
||
vOffset.Init();
|
||
return;
|
||
}
|
||
|
||
errorAmount = 1.0f - errorAmount;
|
||
|
||
vOffset = m_vecPredictionError * errorAmount;
|
||
#else
|
||
vOffset.Init();
|
||
#endif
|
||
}
|
||
|
||
|
||
IRagdoll* C_BasePlayer::GetRepresentativeRagdoll() const
|
||
{
|
||
return m_pRagdoll;
|
||
}
|
||
|
||
IMaterial *C_BasePlayer::GetHeadLabelMaterial( void )
|
||
{
|
||
if ( GetClientVoiceMgr() == NULL )
|
||
return NULL;
|
||
|
||
return GetClientVoiceMgr()->GetHeadLabelMaterial();
|
||
}
|
||
|
||
bool IsInFreezeCam( void )
|
||
{
|
||
C_BasePlayer *pPlayer = C_BasePlayer::GetLocalPlayer();
|
||
if ( pPlayer && pPlayer->GetObserverMode() == OBS_MODE_FREEZECAM )
|
||
return true;
|
||
|
||
return false;
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: Set the fog controller data per player.
|
||
// Input : &inputdata -
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::FogControllerChanged( bool bSnap )
|
||
{
|
||
if ( m_PlayerFog.m_hCtrl )
|
||
{
|
||
fogparams_t *pFogParams = &(m_PlayerFog.m_hCtrl->m_fog);
|
||
|
||
/*
|
||
Msg("Updating Fog Target: (%d,%d,%d) %.0f,%.0f -> (%d,%d,%d) %.0f,%.0f (%.2f seconds)\n",
|
||
m_CurrentFog.colorPrimary.GetR(), m_CurrentFog.colorPrimary.GetB(), m_CurrentFog.colorPrimary.GetG(),
|
||
m_CurrentFog.start.Get(), m_CurrentFog.end.Get(),
|
||
pFogParams->colorPrimary.GetR(), pFogParams->colorPrimary.GetB(), pFogParams->colorPrimary.GetG(),
|
||
pFogParams->start.Get(), pFogParams->end.Get(), pFogParams->duration.Get() );*/
|
||
|
||
|
||
// Setup the fog color transition.
|
||
m_PlayerFog.m_OldColor = m_CurrentFog.colorPrimary;
|
||
m_PlayerFog.m_flOldStart = m_CurrentFog.start;
|
||
m_PlayerFog.m_flOldEnd = m_CurrentFog.end;
|
||
m_PlayerFog.m_flOldMaxDensity = m_CurrentFog.maxdensity;
|
||
m_PlayerFog.m_flOldHDRColorScale = m_CurrentFog.HDRColorScale;
|
||
m_PlayerFog.m_flOldFarZ = m_CurrentFog.farz;
|
||
|
||
m_PlayerFog.m_NewColor = pFogParams->colorPrimary;
|
||
m_PlayerFog.m_flNewStart = pFogParams->start;
|
||
m_PlayerFog.m_flNewEnd = pFogParams->end;
|
||
m_PlayerFog.m_flNewMaxDensity = pFogParams->maxdensity;
|
||
m_PlayerFog.m_flNewHDRColorScale = pFogParams->HDRColorScale;
|
||
m_PlayerFog.m_flNewFarZ = pFogParams->farz;
|
||
|
||
m_PlayerFog.m_flTransitionTime = bSnap ? -1 : gpGlobals->curtime;
|
||
|
||
m_CurrentFog = *pFogParams;
|
||
|
||
// Update the fog player's local fog data with the fog controller's data if need be.
|
||
UpdateFogController();
|
||
}
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
// Purpose: Check to see that the controllers data is up to date.
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::UpdateFogController( void )
|
||
{
|
||
if ( m_PlayerFog.m_hCtrl )
|
||
{
|
||
// Don't bother copying while we're transitioning, since it'll be stomped in UpdateFogBlend();
|
||
if ( m_PlayerFog.m_flTransitionTime == -1 && (m_hOldFogController == m_PlayerFog.m_hCtrl) )
|
||
{
|
||
fogparams_t *pFogParams = &(m_PlayerFog.m_hCtrl->m_fog);
|
||
if ( m_CurrentFog != *pFogParams )
|
||
{
|
||
/*
|
||
Msg("FORCING UPDATE: (%d,%d,%d) %.0f,%.0f -> (%d,%d,%d) %.0f,%.0f (%.2f seconds)\n",
|
||
m_CurrentFog.colorPrimary.GetR(), m_CurrentFog.colorPrimary.GetB(), m_CurrentFog.colorPrimary.GetG(),
|
||
m_CurrentFog.start.Get(), m_CurrentFog.end.Get(),
|
||
pFogParams->colorPrimary.GetR(), pFogParams->colorPrimary.GetB(), pFogParams->colorPrimary.GetG(),
|
||
pFogParams->start.Get(), pFogParams->end.Get(), pFogParams->duration.Get() );*/
|
||
|
||
|
||
m_CurrentFog = *pFogParams;
|
||
}
|
||
}
|
||
}
|
||
else
|
||
{
|
||
if ( m_CurrentFog.farz != -1 || m_CurrentFog.enable != false )
|
||
{
|
||
// No fog controller in this level. Use default fog parameters.
|
||
m_CurrentFog.farz = -1;
|
||
m_CurrentFog.enable = false;
|
||
}
|
||
}
|
||
|
||
// Update the fog blending state - of necessary.
|
||
UpdateFogBlend();
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
//
|
||
//-----------------------------------------------------------------------------
|
||
void C_BasePlayer::UpdateFogBlend( void )
|
||
{
|
||
// Transition.
|
||
if ( m_PlayerFog.m_flTransitionTime != -1 )
|
||
{
|
||
float flTimeDelta = gpGlobals->curtime - m_PlayerFog.m_flTransitionTime;
|
||
if ( flTimeDelta < m_CurrentFog.duration )
|
||
{
|
||
float flScale = flTimeDelta / m_CurrentFog.duration;
|
||
m_CurrentFog.colorPrimary.SetR( ( m_PlayerFog.m_NewColor.r * flScale ) + ( m_PlayerFog.m_OldColor.r * ( 1.0f - flScale ) ) );
|
||
m_CurrentFog.colorPrimary.SetG( ( m_PlayerFog.m_NewColor.g * flScale ) + ( m_PlayerFog.m_OldColor.g * ( 1.0f - flScale ) ) );
|
||
m_CurrentFog.colorPrimary.SetB( ( m_PlayerFog.m_NewColor.b * flScale ) + ( m_PlayerFog.m_OldColor.b * ( 1.0f - flScale ) ) );
|
||
m_CurrentFog.start.Set( ( m_PlayerFog.m_flNewStart * flScale ) + ( ( m_PlayerFog.m_flOldStart * ( 1.0f - flScale ) ) ) );
|
||
m_CurrentFog.end.Set( ( m_PlayerFog.m_flNewEnd * flScale ) + ( ( m_PlayerFog.m_flOldEnd * ( 1.0f - flScale ) ) ) );
|
||
m_CurrentFog.maxdensity.Set( ( m_PlayerFog.m_flNewMaxDensity * flScale ) + ( ( m_PlayerFog.m_flOldMaxDensity * ( 1.0f - flScale ) ) ) );
|
||
m_CurrentFog.HDRColorScale.Set( ( m_PlayerFog.m_flNewHDRColorScale * flScale ) + ( ( m_PlayerFog.m_flOldHDRColorScale * ( 1.0f - flScale ) ) ) );
|
||
|
||
// Lerp to a sane FarZ (default value comes from CViewRender::GetZFar())
|
||
float newFarZ = m_PlayerFog.m_flNewFarZ;
|
||
if ( newFarZ <= 0 )
|
||
newFarZ = r_mapextents.GetFloat() * 1.73205080757f;
|
||
|
||
float oldFarZ = m_PlayerFog.m_flOldFarZ;
|
||
if ( oldFarZ <= 0 )
|
||
oldFarZ = r_mapextents.GetFloat() * 1.73205080757f;
|
||
|
||
m_CurrentFog.farz.Set( ( newFarZ * flScale ) + ( ( oldFarZ * ( 1.0f - flScale ) ) ) );
|
||
}
|
||
else
|
||
{
|
||
// Slam the final fog values.
|
||
m_CurrentFog.colorPrimary.SetR( m_PlayerFog.m_NewColor.r );
|
||
m_CurrentFog.colorPrimary.SetG( m_PlayerFog.m_NewColor.g );
|
||
m_CurrentFog.colorPrimary.SetB( m_PlayerFog.m_NewColor.b );
|
||
m_CurrentFog.start.Set( m_PlayerFog.m_flNewStart );
|
||
m_CurrentFog.end.Set( m_PlayerFog.m_flNewEnd );
|
||
m_CurrentFog.maxdensity.Set( m_PlayerFog.m_flNewMaxDensity );
|
||
m_CurrentFog.HDRColorScale.Set( m_PlayerFog.m_flNewHDRColorScale );
|
||
m_CurrentFog.farz.Set( m_PlayerFog.m_flNewFarZ );
|
||
m_PlayerFog.m_flTransitionTime = -1;
|
||
|
||
/*
|
||
Msg("Finished transition to (%d,%d,%d) %.0f,%.0f\n",
|
||
m_CurrentFog.colorPrimary.GetR(), m_CurrentFog.colorPrimary.GetB(), m_CurrentFog.colorPrimary.GetG(),
|
||
m_CurrentFog.start.Get(), m_CurrentFog.end.Get() );*/
|
||
|
||
}
|
||
}
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
//
|
||
//-----------------------------------------------------------------------------
|
||
C_PostProcessController* C_BasePlayer::GetActivePostProcessController() const
|
||
{
|
||
return m_hPostProcessCtrl.Get();
|
||
}
|
||
|
||
//-----------------------------------------------------------------------------
|
||
//
|
||
//-----------------------------------------------------------------------------
|
||
C_ColorCorrection* C_BasePlayer::GetActiveColorCorrection() const
|
||
{
|
||
return m_hColorCorrectionCtrl.Get();
|
||
}
|
||
|
||
bool C_BasePlayer::PreRender( int nSplitScreenPlayerSlot )
|
||
{
|
||
if ( !IsVisible() ||
|
||
!GetClientMode()->ShouldDrawLocalPlayer( this ) )
|
||
{
|
||
return true;
|
||
}
|
||
|
||
// Add in lighting effects
|
||
return CreateLightEffects();
|
||
}
|
||
|
||
bool C_BasePlayer::IsSplitScreenPartner( C_BasePlayer *pPlayer )
|
||
{
|
||
if ( !pPlayer || pPlayer == this )
|
||
return false;
|
||
|
||
for ( int i = 0; i < MAX_SPLITSCREEN_PLAYERS; ++i )
|
||
{
|
||
if ( s_pLocalPlayer[i] == pPlayer )
|
||
return true;
|
||
}
|
||
|
||
return false;
|
||
}
|
||
|
||
int C_BasePlayer::GetSplitScreenPlayerSlot()
|
||
{
|
||
return m_nSplitScreenSlot;
|
||
}
|
||
|
||
bool C_BasePlayer::IsSplitScreenPlayer() const
|
||
{
|
||
return m_nSplitScreenSlot >= 1;
|
||
}
|
||
|
||
bool C_BasePlayer::ShouldRegenerateOriginFromCellBits() const
|
||
{
|
||
// Don't use cell bits for local players.
|
||
// Assumes full update for local players!
|
||
int nSlot = GET_ACTIVE_SPLITSCREEN_SLOT();
|
||
int nIndex = engine->GetSplitScreenPlayer( nSlot );
|
||
if ( nIndex == entindex() )
|
||
return false;
|
||
|
||
return BaseClass::ShouldRegenerateOriginFromCellBits();
|
||
}
|
||
|
||
|
||
void CC_DumpClientSoundscapeData( const CCommand& args )
|
||
{
|
||
C_BasePlayer *pPlayer = C_BasePlayer::GetLocalPlayer();
|
||
if ( !pPlayer )
|
||
return;
|
||
|
||
Msg("Client Soundscape data dump:\n");
|
||
Msg(" Position: %.2f %.2f %.2f\n", pPlayer->GetAbsOrigin().x, pPlayer->GetAbsOrigin().y, pPlayer->GetAbsOrigin().z );
|
||
Msg(" soundscape index: %d\n", pPlayer->m_Local.m_audio.soundscapeIndex );
|
||
Msg(" entity index: %d\n", pPlayer->m_Local.m_audio.entIndex );
|
||
bool bFoundOne = false;
|
||
for ( int i = 0; i < NUM_AUDIO_LOCAL_SOUNDS; i++ )
|
||
{
|
||
if ( pPlayer->m_Local.m_audio.localBits & (1<<i) )
|
||
{
|
||
if ( !bFoundOne )
|
||
{
|
||
Msg(" Sound Positions:\n");
|
||
bFoundOne = true;
|
||
}
|
||
|
||
Vector vecPos = pPlayer->m_Local.m_audio.localSound[i];
|
||
Msg(" %d: %.2f %.2f %.2f\n", i, vecPos.x,vecPos.y, vecPos.z );
|
||
}
|
||
}
|
||
|
||
Msg("End dump.\n");
|
||
}
|
||
static ConCommand soundscape_dumpclient("soundscape_dumpclient", CC_DumpClientSoundscapeData, "Dumps the client's soundscape data.\n", FCVAR_CHEAT);
|