Team Fortress 2 Source Code as on 22/4/2020
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.
 
 
 
 
 
 

523 lines
16 KiB

//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//=============================================================================//
#include "cbase.h"
#include "tf_hud_mediccallers.h"
#include "iclientmode.h"
#include <vgui/ILocalize.h>
#include <vgui/ISurface.h>
#include <vgui/IVGui.h>
#include "view.h"
#include "ivieweffects.h"
#include "viewrender.h"
#include "prediction.h"
#include "GameEventListener.h"
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
#define MEDICCALLER_WIDE (XRES(56))
#define MEDICCALLER_TALL (YRES(48))
#define MEDICCALLER_ARROW_WIDE (XRES(16))
#define MEDICCALLER_ARROW_TALL (YRES(24))
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
CTFMedicCallerPanel::CTFMedicCallerPanel( Panel *parent, const char *name ) : EditablePanel(parent,name)
{
m_pArrowMaterial = NULL;
m_iDrawArrow = DRAW_ARROW_UP;
m_bOnscreen = false;
m_flPanelScale = 1.0f;
m_bBurning = false;
m_bBleeding = false;
m_nCallerType = CALLER_TYPE_NORMAL;
ListenForGameEvent( "player_calledformedic" );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
CTFMedicCallerPanel::~CTFMedicCallerPanel( void )
{
if ( m_pArrowMaterial )
{
m_pArrowMaterial->DecrementReferenceCount();
}
}
//-----------------------------------------------------------------------------
// Purpose: Applies scheme settings
//-----------------------------------------------------------------------------
void CTFMedicCallerPanel::ApplySchemeSettings( vgui::IScheme *pScheme )
{
BaseClass::ApplySchemeSettings( pScheme );
LoadControlSettings( GetControlSettingFile() );
if ( m_pArrowMaterial )
{
m_pArrowMaterial->DecrementReferenceCount();
}
m_pArrowMaterial = materials->FindMaterial( "HUD/medic_arrow", TEXTURE_GROUP_VGUI );
m_pArrowMaterial->IncrementReferenceCount();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFMedicCallerPanel::PerformLayout( void )
{
BaseClass::PerformLayout();
int nWide = XRES(100), nTall = YRES(100);
bool bNormal = ( m_nCallerType == CALLER_TYPE_NORMAL );
bool bAutoCaller = ( m_nCallerType == CALLER_TYPE_AUTO );
// Adjust scale of the panel based on distance to the caller
C_TFPlayer *pLocalTFPlayer = C_TFPlayer::GetLocalTFPlayer();
if ( pLocalTFPlayer && m_hEntity )
{
Vector vecDistance = m_hEntity->GetAbsOrigin() - pLocalTFPlayer->GetAbsOrigin();
m_flPanelScale = RemapValClamped( vecDistance.LengthSqr(), 0.0f, (2000.0f * 2000.0f), 1.0f, 0.5f );
}
vgui::Panel *pPanelAuto = FindChildByName( "CallerAuto" );
if ( pPanelAuto )
{
if ( pPanelAuto->IsVisible() != bAutoCaller )
{
pPanelAuto->SetVisible( bAutoCaller );
}
if ( bAutoCaller )
{
pPanelAuto->GetSize( nWide, nTall );
pPanelAuto->SetSize ( nWide * m_flPanelScale, nTall * m_flPanelScale );
pPanelAuto->SetPos( ( GetWide() - pPanelAuto->GetWide() ) * 0.5, ( GetTall() - pPanelAuto->GetTall() ) * 0.5 );
}
}
vgui::Panel *pPanel = FindChildByName( "CallerBG" );
if ( pPanel )
{
if ( pPanel->IsVisible() != bNormal )
{
pPanel->SetVisible( bNormal );
}
if ( bNormal )
{
pPanel->GetSize( nWide, nTall );
pPanel->SetSize ( nWide * m_flPanelScale, nTall * m_flPanelScale );
pPanel->SetPos( (GetWide() - pPanel->GetWide()) * 0.5, (GetTall() - pPanel->GetTall()) * 0.5 );
}
}
vgui::Panel *pBurningPanel = FindChildByName( "CallerBurning" );
if ( pBurningPanel )
{
bool bVisible = bNormal && m_bBurning;
if ( pBurningPanel->IsVisible() != bVisible )
{
pBurningPanel->SetVisible( bVisible );
}
if ( bVisible )
{
pBurningPanel->GetSize( nWide, nTall );
pBurningPanel->SetSize ( nWide * m_flPanelScale, nTall * m_flPanelScale );
pBurningPanel->SetPos( (GetWide() - pBurningPanel->GetWide()) * 0.5, (GetTall() - pBurningPanel->GetTall()) * 0.5 );
}
}
vgui::Panel *pBleedingPanel = FindChildByName( "CallerBleeding" );
if ( pBleedingPanel )
{
bool bVisible = bNormal && m_bBleeding;
if ( pBleedingPanel->IsVisible() != bVisible )
{
pBleedingPanel->SetVisible( bVisible );
}
if ( bVisible )
{
pBleedingPanel->GetSize( nWide, nTall );
pBleedingPanel->SetSize ( nWide * m_flPanelScale, nTall * m_flPanelScale );
pBleedingPanel->SetPos( (GetWide() - pBleedingPanel->GetWide()) * 0.5, (GetTall() - pBleedingPanel->GetTall()) * 0.5 );
}
}
vgui::Panel *pPanelHealth = FindChildByName( "CallerHealth" );
if ( pPanelHealth )
{
if ( pPanelHealth->IsVisible() != bNormal )
{
pPanelHealth->SetVisible( bNormal );
}
if ( bNormal )
{
pPanelHealth->GetSize( nWide, nTall );
pPanelHealth->SetSize ( nWide * m_flPanelScale, nTall * m_flPanelScale );
pPanelHealth->SetPos( (GetWide() - pPanelHealth->GetWide()) * 0.5, (GetTall() - pPanelHealth->GetTall()) * 0.5 );
pPanelHealth->SetAlpha( 0 );
}
}
// Revive block
vgui::Panel *pPanelReviveEasy = FindChildByName( "CallerReviveEasy" );
if ( pPanelReviveEasy )
{
bool bReviveEasy = m_nCallerType == CALLER_TYPE_REVIVE_EASY;
if ( pPanelReviveEasy->IsVisible() != bReviveEasy )
{
pPanelReviveEasy->SetVisible( bReviveEasy );
}
if ( bReviveEasy )
{
pPanelReviveEasy->GetSize( nWide, nTall );
pPanelReviveEasy->SetSize ( nWide * m_flPanelScale, nTall * m_flPanelScale );
pPanelReviveEasy->SetPos( ( GetWide() - pPanelReviveEasy->GetWide() ) * 0.5, ( GetTall() - pPanelReviveEasy->GetTall() ) * 0.5 );
}
}
vgui::Panel *pPanelReviveMedium = FindChildByName( "CallerReviveMedium" );
if ( pPanelReviveMedium )
{
bool bReviveMedium = m_nCallerType == CALLER_TYPE_REVIVE_MEDIUM;
if ( pPanelReviveMedium->IsVisible() != bReviveMedium )
{
pPanelReviveMedium->SetVisible( bReviveMedium );
}
if ( bReviveMedium )
{
pPanelReviveMedium->GetSize( nWide, nTall );
pPanelReviveMedium->SetSize ( nWide * m_flPanelScale, nTall * m_flPanelScale );
pPanelReviveMedium->SetPos( ( GetWide() - pPanelReviveMedium->GetWide() ) * 0.5, ( GetTall() - pPanelReviveMedium->GetTall() ) * 0.5 );
}
}
vgui::Panel *pPanelReviveHard = FindChildByName( "CallerReviveHard" );
if ( pPanelReviveHard )
{
bool bReviveHard = m_nCallerType == CALLER_TYPE_REVIVE_HARD;
if ( pPanelReviveHard->IsVisible() != bReviveHard )
{
pPanelReviveHard->SetVisible( bReviveHard );
}
if ( bReviveHard )
{
pPanelReviveHard->GetSize( nWide, nTall );
pPanelReviveHard->SetSize ( nWide * m_flPanelScale, nTall * m_flPanelScale );
pPanelReviveHard->SetPos( ( GetWide() - pPanelReviveHard->GetWide() ) * 0.5, ( GetTall() - pPanelReviveHard->GetTall() ) * 0.5 );
}
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFMedicCallerPanel::GetCallerPosition( const Vector &vecDelta, float flRadius, float *xpos, float *ypos, float *flRotation )
{
// Player Data
Vector playerPosition = MainViewOrigin();
QAngle playerAngles = MainViewAngles();
Vector forward, right, up(0,0,1);
AngleVectors (playerAngles, &forward, NULL, NULL );
forward.z = 0;
VectorNormalize(forward);
CrossProduct( up, forward, right );
float front = DotProduct(vecDelta, forward);
float side = DotProduct(vecDelta, right);
*xpos = flRadius * -side;
*ypos = flRadius * -front;
// Get the rotation (yaw)
*flRotation = atan2(*xpos,*ypos) + M_PI;
*flRotation *= 180 / M_PI;
float yawRadians = -(*flRotation) * M_PI / 180.0f;
float ca = cos( yawRadians );
float sa = sin( yawRadians );
// Rotate it around the circle
*xpos = (int)((ScreenWidth() / 2) + (flRadius * sa));
*ypos = (int)((ScreenHeight() / 2) - (flRadius * ca));
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFMedicCallerPanel::OnTick( void )
{
if ( !m_hEntity || ( m_hEntity->IsPlayer() && !m_hEntity->IsAlive() ) || gpGlobals->curtime > m_flRemoveAt )
{
MarkForDeletion();
return;
}
// If the local player has started healing this guy, remove it too.
// Also don't draw it if we're dead.
C_TFPlayer *pLocalTFPlayer = C_TFPlayer::GetLocalTFPlayer();
if ( pLocalTFPlayer )
{
CBaseEntity *pHealTarget = pLocalTFPlayer->MedicGetHealTarget();
if ( ( pHealTarget && pHealTarget == m_hEntity ) || ( m_hEntity->IsPlayer() && !pLocalTFPlayer->IsAlive() ) )
{
MarkForDeletion();
return;
}
if ( m_hEntity->IsPlayer() )
{
C_TFPlayer *pTFPlayer = ToTFPlayer( m_hEntity );
if ( pTFPlayer )
{
// If we're pointing to an enemy spy and they are no longer disguised, remove ourselves
if ( pTFPlayer->IsPlayerClass( TF_CLASS_SPY ) &&
pTFPlayer->GetTeamNumber() != pLocalTFPlayer->GetTeamNumber() &&
!( pTFPlayer->m_Shared.InCond( TF_COND_DISGUISED ) && pTFPlayer->m_Shared.GetDisguiseTeam() == pLocalTFPlayer->GetTeamNumber() ) )
{
MarkForDeletion();
return;
}
// Updates the state of the caller panel if they are now burning or bleeding, or have stopped while caller panel is still up.
if ( m_nCallerType != CALLER_TYPE_AUTO )
{
m_bBurning = pTFPlayer->m_Shared.InCond( TF_COND_BURNING );
vgui::Panel *pBurningPanel = FindChildByName( "CallerBurning" );
if ( pBurningPanel && pBurningPanel->IsVisible() != m_bBurning )
{
pBurningPanel->SetVisible( m_bBurning );
}
vgui::Panel *pBleedingPanel = FindChildByName( "CallerBleeding" );
m_bBleeding = pTFPlayer->m_Shared.InCond( TF_COND_BLEEDING );
if ( pBleedingPanel && pBleedingPanel->IsVisible() != m_bBleeding )
{
pBleedingPanel->SetVisible( m_bBleeding );
}
}
}
}
}
if ( m_nCallerType == CALLER_TYPE_NORMAL )
{
// Tints caller panel based on health remaining.
vgui::Panel *pPanelHealth = FindChildByName( "CallerHealth" );
if ( pPanelHealth )
{
float flHealth = ( float(m_hEntity->GetHealth()) / float(m_hEntity->GetMaxHealth()) );
int iCallerHurtAlpha = 255 * ( 1 - flHealth ) + 75;
pPanelHealth->SetAlpha( clamp( iCallerHurtAlpha, 0, 255 ) );
}
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFMedicCallerPanel::PaintBackground( void )
{
// If the local player has started healing this guy, remove it too.
//Also don't draw it if we're dead.
C_TFPlayer *pLocalTFPlayer = C_TFPlayer::GetLocalTFPlayer();
if ( !pLocalTFPlayer )
return;
if ( !m_hEntity || m_hEntity->IsDormant() )
{
SetAlpha(0);
return;
}
// Reposition the callout based on our target's position
int iX, iY;
Vector vecTarget = (m_hEntity->GetAbsOrigin() + m_vecOffset);
Vector vecDelta = vecTarget - MainViewOrigin();
bool bOnscreen = GetVectorInHudSpace( vecTarget, iX, iY ); // Tested and correct - should NOT be GetVectorInScreenSpace.
int halfWidth = GetWide() / 2;
if( !bOnscreen || iX < halfWidth || iX > ScreenWidth()-halfWidth )
{
// It's off the screen. Position the callout.
VectorNormalize(vecDelta);
float xpos, ypos;
float flRotation;
float flRadius = YRES(100);
GetCallerPosition( vecDelta, flRadius, &xpos, &ypos, &flRotation );
iX = xpos;
iY = ypos;
Vector vCenter = m_hEntity->WorldSpaceCenter( );
if( MainViewRight().Dot( vCenter - MainViewOrigin() ) > 0 )
{
m_iDrawArrow = DRAW_ARROW_RIGHT;
}
else
{
m_iDrawArrow = DRAW_ARROW_LEFT;
}
// Move the icon there
SetPos( iX - halfWidth, iY - (GetTall() / 2) );
SetAlpha( 255 );
}
else
{
// On screen
// If our target isn't visible, we draw transparently
trace_t tr;
UTIL_TraceLine( vecTarget, MainViewOrigin(), MASK_OPAQUE, NULL, COLLISION_GROUP_NONE, &tr );
if ( tr.fraction >= 1.0f )
{
m_bOnscreen = true;
SetAlpha( 0 );
return;
}
m_iDrawArrow = DRAW_ARROW_UP;
SetAlpha( 92 );
SetPos( iX - halfWidth, iY - (GetTall() / 2) );
}
m_bOnscreen = false;
BaseClass::PaintBackground();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFMedicCallerPanel::Paint( void )
{
// Don't draw if our target is visible. The particle effect will be doing it for us.
if ( m_bOnscreen )
return;
BaseClass::Paint();
if ( m_iDrawArrow == DRAW_ARROW_UP )
return;
float uA,uB,yA,yB;
int x,y;
GetPos( x,y );
if ( m_iDrawArrow == DRAW_ARROW_LEFT )
{
uA = 1.0;
uB = 0.0;
yA = 0.0;
yB = 1.0;
}
else
{
uA = 0.0;
uB = 1.0;
yA = 0.0;
yB = 1.0;
x += GetWide() - MEDICCALLER_ARROW_WIDE;
}
int iyindent = (GetTall() - MEDICCALLER_ARROW_TALL) * 0.5;
y += iyindent;
CMatRenderContextPtr pRenderContext( materials );
pRenderContext->Bind( m_pArrowMaterial );
IMesh* pMesh = pRenderContext->GetDynamicMesh( true );
CMeshBuilder meshBuilder;
meshBuilder.Begin( pMesh, MATERIAL_QUADS, 1 );
meshBuilder.Position3f( x, y, 0.0f );
meshBuilder.TexCoord2f( 0, uA, yA );
meshBuilder.Color4ub( 255, 255, 255, 255 );
meshBuilder.AdvanceVertex();
meshBuilder.Position3f( x + MEDICCALLER_ARROW_WIDE, y, 0.0f );
meshBuilder.TexCoord2f( 0, uB, yA );
meshBuilder.Color4ub( 255, 255, 255, 255 );
meshBuilder.AdvanceVertex();
meshBuilder.Position3f( x + MEDICCALLER_ARROW_WIDE, y + MEDICCALLER_ARROW_TALL, 0.0f );
meshBuilder.TexCoord2f( 0, uB, yB );
meshBuilder.Color4ub( 255, 255, 255, 255 );
meshBuilder.AdvanceVertex();
meshBuilder.Position3f( x, y + MEDICCALLER_ARROW_TALL, 0.0f );
meshBuilder.TexCoord2f( 0, uA, yB );
meshBuilder.Color4ub( 255, 255, 255, 255 );
meshBuilder.AdvanceVertex();
meshBuilder.End();
pMesh->Draw();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFMedicCallerPanel::SetEntity( C_BaseEntity *pEntity, float flDuration, Vector &vecOffset )
{
m_hEntity = pEntity;
m_flRemoveAt = gpGlobals->curtime + flDuration;
m_vecOffset = vecOffset;
}
void CTFMedicCallerPanel::SetMedicCallerType( MedicCallerType nType )
{
m_nCallerType = nType;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFMedicCallerPanel::AddMedicCaller( C_BaseEntity *pEntity, float flDuration, Vector &vecOffset, MedicCallerType nType /* = CALLER_TYPE_NORMAL */ )
{
CTFMedicCallerPanel *pCaller = new CTFMedicCallerPanel( g_pClientMode->GetViewport(), "MedicCallerPanel" );
vgui::SETUP_PANEL(pCaller);
pCaller->SetBounds( 0,0, MEDICCALLER_WIDE, MEDICCALLER_TALL );
pCaller->SetEntity( pEntity, flDuration, vecOffset );
pCaller->SetMedicCallerType( nType );
pCaller->SetVisible( true );
vgui::ivgui()->AddTickSignal( pCaller->GetVPanel() );
pCaller->OnTick();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFMedicCallerPanel::FireGameEvent( IGameEvent *event )
{
if ( m_nCallerType == CALLER_TYPE_AUTO )
{
if ( Q_strcmp( event->GetName(), "player_calledformedic" ) == 0 )
{
if ( m_hEntity && m_hEntity->IsPlayer() )
{
C_TFPlayer *pTFPlayer = ToTFPlayer( m_hEntity );
if ( pTFPlayer )
{
int iCaller = engine->GetPlayerForUserID( event->GetInt( "userid" ) );
if ( pTFPlayer->GetUserID() == iCaller )
{
MarkForDeletion();
}
}
}
}
}
}