Counter Strike : Global Offensive Source Code
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

139 lines
4.4 KiB

//========= Copyright © 1996-2005, Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $Workfile: $
// $Date: $
// $NoKeywords: $
//=============================================================================//
#include "cbase.h"
#include "animation.h"
#include "baseviewmodel.h"
#include "player.h"
#include <keyvalues.h>
#include "studio.h"
#include "vguiscreen.h"
#include "saverestore_utlvector.h"
#include "hltvdirector.h"
#include "replaydirector.h"
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
void SendProxy_AnimTime( const void *pStruct, const void *pVarData, DVariant *pOut, int iElement, int objectID );
void SendProxy_SequenceChanged( const void *pStruct, const void *pVarData, DVariant *pOut, int iElement, int objectID );
//-----------------------------------------------------------------------------
// Purpose: Save Data for Base Weapon object
//-----------------------------------------------------------------------------//
BEGIN_DATADESC( CBaseViewModel )
DEFINE_FIELD( m_hOwner, FIELD_EHANDLE ),
// Client only
// DEFINE_FIELD( m_LagAnglesHistory, CInterpolatedVar < QAngle > ),
// DEFINE_FIELD( m_vLagAngles, FIELD_VECTOR ),
DEFINE_FIELD( m_nViewModelIndex, FIELD_INTEGER ),
DEFINE_FIELD( m_flTimeWeaponIdle, FIELD_FLOAT ),
DEFINE_FIELD( m_nAnimationParity, FIELD_INTEGER ),
// Client only
// DEFINE_FIELD( m_nOldAnimationParity, FIELD_INTEGER ),
DEFINE_FIELD( m_vecLastFacing, FIELD_VECTOR ),
DEFINE_FIELD( m_hWeapon, FIELD_EHANDLE ),
DEFINE_UTLVECTOR( m_hScreens, FIELD_EHANDLE ),
// Read from weapons file
// DEFINE_FIELD( m_sVMName, FIELD_STRING ),
// DEFINE_FIELD( m_sAnimationPrefix, FIELD_STRING ),
// ---------------------------------------------------------------------
// Don't save these, init to 0 and regenerate
// DEFINE_FIELD( m_Activity, FIELD_INTEGER ),
END_DATADESC()
int CBaseViewModel::UpdateTransmitState()
{
if ( IsEffectActive( EF_NODRAW ) )
{
return SetTransmitState( FL_EDICT_DONTSEND );
}
return SetTransmitState( FL_EDICT_FULLCHECK );
}
int CBaseViewModel::ShouldTransmit( const CCheckTransmitInfo *pInfo )
{
// Check if recipient owns this weapon viewmodel
CBasePlayer *pOwner = ToBasePlayer( m_hOwner );
if ( pOwner &&
( pOwner->edict() == pInfo->m_pClientEnt ||
// If we're using the other guys network connection in split screen, then also force transmit
pOwner->IsSplitScreenUserOnEdict( pInfo->m_pClientEnt ) ) )
{
return FL_EDICT_ALWAYS;
}
// check if recipient (or one of his splitscreen parasites) is spectating the owner of this viewmodel
CBasePlayer *pPlayer = ToBasePlayer( CBaseEntity::Instance( pInfo->m_pClientEnt ) );
if ( pPlayer)
{
// Bug 28591: In splitscreen, when the second slot is the one in spectator mode, it wouldn't
// get the viewmodel for the spectatee.
//
// The new logic is to loop through the splitscreen parasites (as well as the host player)
// and see if any of them are observing the viewmodel owner, and if so, FL_EDICT_ALWAYS the vm for them, too.
// This container was the source of most of the allocation cost in CS:GO so using a CUtlVectorFixed to avoid
// allocations is important.
CUtlVectorFixedGrowable< CBasePlayer *, MAX_SPLITSCREEN_CLIENTS > checkList;
checkList.AddToTail( pPlayer );
CUtlVector< CHandle< CBasePlayer > > &vecParasites = pPlayer->GetSplitScreenPlayers();
for ( int i = 0; i < vecParasites.Count(); ++i )
{
checkList.AddToTail( vecParasites[ i ] );
}
for ( int i = 0; i < checkList.Count(); ++i )
{
CBasePlayer *pPlayer = checkList[ i ];
if ( !pPlayer )
continue;
if ( pPlayer->IsHLTV() || pPlayer->IsReplay() )
{
// if this is the HLTV or Replay client, transmit all viewmodels in our PVS
return FL_EDICT_PVSCHECK;
}
if ( (pPlayer->GetObserverMode() == OBS_MODE_IN_EYE) && (pPlayer->GetObserverTarget() == pOwner) )
{
return FL_EDICT_ALWAYS;
}
}
}
// Don't send to anyone else except the local player or his spectator
return FL_EDICT_DONTSEND;
}
void CBaseViewModel::SetTransmit( CCheckTransmitInfo *pInfo, bool bAlways )
{
// Are we already marked for transmission?
if ( pInfo->m_pTransmitEdict->Get( entindex() ) )
return;
BaseClass::SetTransmit( pInfo, bAlways );
// Force our screens to be sent too.
for ( int i=0; i < m_hScreens.Count(); i++ )
{
CVGuiScreen *pScreen = m_hScreens[i].Get();
if ( pScreen )
pScreen->SetTransmit( pInfo, bAlways );
}
}