Modified source engine (2017) developed by valve and leaked in 2020. Not for commercial purporses
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.

1704 lines
50 KiB

5 years ago
//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//=============================================================================//
#include "cbase.h"
#include "tf_classmenu.h"
#include "IGameUIFuncs.h" // for key bindings
#include "tf_hud_notification_panel.h"
#include "character_info_panel.h"
#include "playerspawncache.h"
#include "iclientmode.h"
#include "econ_gcmessages.h"
#include "gc_clientsystem.h"
#include "vgui/IInput.h"
#include <vgui_controls/PanelListPanel.h>
#include <vgui_controls/ScrollBarSlider.h>
#include "tf_gamerules.h"
#include "engine/IEngineSound.h"
#include "inputsystem/iinputsystem.h"
#if defined( REPLAY_ENABLED )
#include "replay/iclientreplaycontext.h"
#include "replay/ireplaysystem.h"
#include "replay/ienginereplay.h"
#include "replay/shared_defs.h"
#endif
extern IGameUIFuncs *gameuifuncs; // for key binding details
using namespace vgui;
ConVar _cl_classmenuopen( "_cl_classmenuopen", "0", 0, "internal cvar used to tell server when class menu is open" );
#if defined( REPLAY_ENABLED )
ConVar replay_replaywelcomedlgcount( "replay_replaywelcomedlgcount", "0", FCVAR_CLIENTDLL | FCVAR_DONTRECORD | FCVAR_ARCHIVE | FCVAR_HIDDEN, "The number of times the replay help dialog has displayed." );
#endif
ConVar tf_mvm_classupgradehelpcount( "tf_mvm_classupgradehelpcount", "0", FCVAR_CLIENTDLL | FCVAR_DONTRECORD | FCVAR_ARCHIVE | FCVAR_HIDDEN, "The number of times the player upgrade help dialog has displayed." );
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
CTFClassTipsItemPanel::CTFClassTipsItemPanel( Panel *parent, const char *name, int iListItemID ) : BaseClass( parent, name )
{
m_pTipIcon = new vgui::ImagePanel( this, "TipIcon" );
m_pTipLabel = new CExLabel( this, "TipLabel", "" );
// m_pTipIcon = dynamic_cast<vgui::ImagePanel*>( FindChildByName( "TipIcon" ) );
// m_pTipLabel = dynamic_cast<CExLabel*>( FindChildByName( "TipLabel" ) );
}
CTFClassTipsItemPanel::~CTFClassTipsItemPanel()
{
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFClassTipsItemPanel::SetClassTip( const wchar_t *pwszText, const char *pszIcon )
{
// Set tf_english string and .res icon path
if ( m_pTipLabel && m_pTipIcon )
{
if ( pszIcon )
{
m_pTipIcon->SetImage( pszIcon );
}
else
{
m_pTipIcon->SetVisible( false );
// int iX, iY = 0;
// m_pTipLabel->GetPos( iX, iY );
// int nIconWidth = m_pTipIcon->GetWide();
// m_pTipLabel->SetPos( ( iX - nIconWidth ), iY );
}
m_pTipLabel->SetText( pwszText );
}
InvalidateLayout( true, true );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFClassTipsItemPanel::ApplySchemeSettings( IScheme* pScheme )
{
BaseClass::ApplySchemeSettings( pScheme );
LoadControlSettings( "Resource/UI/ClassTipsItem.res" );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
class CTFClassTipsListPanel : public PanelListPanel
{
private:
DECLARE_CLASS_SIMPLE( CTFClassTipsListPanel, PanelListPanel );
public:
CTFClassTipsListPanel( Panel *parent, const char *panelName )
: PanelListPanel( parent, panelName )
{
m_pScrollBar = GetScrollbar();
m_pUpArrow = new CExImageButton( this, "UpArrow", "" );
if ( m_pUpArrow )
{
m_pUpArrow->AddActionSignalTarget( m_pScrollBar );
m_pUpArrow->SetCommand(new KeyValues("ScrollButtonPressed", "index", 0));
m_pUpArrow->GetImage()->SetShouldScaleImage( true );
m_pUpArrow->SetFgColor( Color( 255, 255, 255, 255 ) );
m_pUpArrow->SetAlpha( 255 );
m_pUpArrow->SetPaintBackgroundEnabled( false );
m_pUpArrow->SetVisible( false );
m_pUpArrow->PassMouseTicksTo( m_pScrollBar );
m_pUpArrow->SetImageDefault( "chalkboard_scroll_up" );
}
m_pDownArrow = new CExImageButton( this, "DownArrow", "" );
if ( m_pDownArrow )
{
m_pDownArrow->AddActionSignalTarget( m_pScrollBar );
m_pDownArrow->SetCommand(new KeyValues("ScrollButtonPressed", "index", 1));
m_pDownArrow->GetImage()->SetShouldScaleImage( true );
m_pDownArrow->SetFgColor( Color( 255, 255, 255, 255 ) );
m_pDownArrow->SetAlpha( 255 );
m_pDownArrow->SetPaintBackgroundEnabled( false );
m_pDownArrow->SetVisible( false );
m_pDownArrow->PassMouseTicksTo( m_pScrollBar );
m_pDownArrow->SetImageDefault( "chalkboard_scroll_down" );
}
if ( m_pScrollBar )
{
m_pScrollBar->SetOverriddenButtons( m_pUpArrow, m_pDownArrow );
m_pScrollBar->SetVisible( false );
}
m_pLine = new vgui::ImagePanel( this, "Line" );
m_pBox = new vgui::ImagePanel( this, "Box" );
if ( m_pLine )
{
m_pLine->SetImage( "chalkboard_scroll_line" );
m_pLine->SetShouldScaleImage( true );
}
if ( m_pBox )
{
m_pBox->SetImage( "chalkboard_scroll_box" );
m_pBox->SetShouldScaleImage( true );
}
SetScrollBarImagesVisible( false );
vgui::ivgui()->AddTickSignal( GetVPanel() );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
~CTFClassTipsListPanel()
{
ivgui()->RemoveTickSignal( GetVPanel() );
}
private:
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
int GetListHeight( void )
{
int nHeight = 0;
for ( int i = FirstItem(); i < GetItemCount(); i++ )
{
vgui::Panel *pClassTipsItemPanel = GetItemPanel( i );
if ( pClassTipsItemPanel )
{
nHeight += pClassTipsItemPanel->GetTall();
}
}
return nHeight;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void SetScrollBarImagesVisible( bool visible )
{
if ( m_pDownArrow && m_pDownArrow->IsVisible() != visible )
{
m_pDownArrow->SetVisible( visible );
m_pDownArrow->SetEnabled( visible );
}
if ( m_pUpArrow && m_pUpArrow->IsVisible() != visible )
{
m_pUpArrow->SetVisible( visible );
m_pUpArrow->SetEnabled( visible );
}
if ( m_pLine && m_pLine->IsVisible() != visible )
{
m_pLine->SetVisible( visible );
}
if ( m_pBox && m_pBox->IsVisible() != visible )
{
m_pBox->SetVisible( visible );
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void OnTick()
{
if ( !IsVisible() )
return;
if ( m_pDownArrow && m_pUpArrow && m_pLine && m_pBox )
{
if ( m_pScrollBar && m_pScrollBar->IsVisible() && GetListHeight() > GetTall() )
{
SetScrollBarImagesVisible ( true );
// set the alpha on the up arrow
int nMin, nMax;
m_pScrollBar->GetRange( nMin, nMax );
int nScrollPos = m_pScrollBar->GetValue();
int nRangeWindow = m_pScrollBar->GetRangeWindow();
int nBottom = nMax - nRangeWindow;
if ( nBottom < 0 )
{
nBottom = 0;
}
int nAlpha = ( nScrollPos - nMin <= 0 ) ? 90 : 255;
m_pUpArrow->SetAlpha( nAlpha );
// set the alpha on the down arrow
nAlpha = ( nScrollPos >= nBottom ) ? 90 : 255;
m_pDownArrow->SetAlpha( nAlpha );
ScrollBarSlider *pSlider = m_pScrollBar->GetSlider();
if ( pSlider && pSlider->GetRangeWindow() > 0 )
{
int x, y, w, t, min, max;
m_pLine->GetBounds( x, y, w, t );
pSlider->GetNobPos( min, max );
m_pBox->SetBounds( x, y + min, w, ( max - min ) );
}
}
else
{
// turn off our images
SetScrollBarImagesVisible ( false );
}
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void ApplySchemeSettings( IScheme *pScheme )
{
BaseClass::ApplySchemeSettings( pScheme );
if ( m_pScrollBar )
{
m_pScrollBar->SetZPos( 500 );
m_pUpArrow->SetZPos( 501 );
m_pDownArrow->SetZPos( 501 );
// turn off painting the vertical scrollbar
m_pScrollBar->SetPaintBackgroundEnabled( false );
m_pScrollBar->SetPaintBorderEnabled( false );
m_pScrollBar->SetPaintEnabled( false );
m_pScrollBar->SetScrollbarButtonsVisible( false );
m_pScrollBar->GetButton(0)->SetMouseInputEnabled( false );
m_pScrollBar->GetButton(1)->SetMouseInputEnabled( false );
if ( m_pScrollBar->IsVisible() )
{
int nMin, nMax;
m_pScrollBar->GetRange( nMin, nMax );
m_pScrollBar->SetValue( nMin );
int nScrollbarWide = m_pScrollBar->GetWide();
int wide, tall;
GetSize( wide, tall );
if ( m_pUpArrow )
{
m_pUpArrow->SetBounds( wide - nScrollbarWide, 0, nScrollbarWide, nScrollbarWide );
m_pUpArrow->GetImage()->SetSize( nScrollbarWide, nScrollbarWide );
}
if ( m_pLine )
{
m_pLine->SetBounds( wide - nScrollbarWide, nScrollbarWide, nScrollbarWide, tall - ( 2 * nScrollbarWide ) );
}
if ( m_pBox )
{
m_pBox->SetBounds( wide - nScrollbarWide, nScrollbarWide, nScrollbarWide, nScrollbarWide );
}
if ( m_pDownArrow )
{
m_pDownArrow->SetBounds( wide - nScrollbarWide, tall - nScrollbarWide, nScrollbarWide, nScrollbarWide );
m_pDownArrow->GetImage()->SetSize( nScrollbarWide, nScrollbarWide );
}
SetScrollBarImagesVisible( false );
}
}
}
vgui::ScrollBar *m_pScrollBar;
CExImageButton *m_pUpArrow;
CExImageButton *m_pDownArrow;
vgui::ImagePanel *m_pLine;
vgui::ImagePanel *m_pBox;
};
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
class CTFClassTipsPanel : public EditablePanel
{
private:
DECLARE_CLASS_SIMPLE( CTFClassTipsPanel, EditablePanel );
public:
CTFClassTipsPanel( Panel *parent, const char *panelName )
: EditablePanel( parent, panelName )
{
m_pClassTipsListPanel = new CTFClassTipsListPanel( this, "ClassTipsListPanel" );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
~CTFClassTipsPanel()
{
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void SetClass( int iClass )
{
const char *pPathID = IsX360() ? "MOD" : "GAME";
m_fmtResFilename.sprintf( "classes/%s.res", g_aRawPlayerClassNames[ iClass ] );
if ( !g_pFullFileSystem->FileExists( m_fmtResFilename.Access(), pPathID ) &&
g_pFullFileSystem->FileExists( "classes/default.res", pPathID ) )
{
m_fmtResFilename.sprintf( "classes/default.res" );
}
if ( m_pClassTipsListPanel )
{
m_pClassTipsListPanel->DeleteAllItems();
int nScrollToItem = 0;
// Get tip count
const wchar_t *wzTipCount = g_pVGuiLocalize->Find( CFmtStr( "ClassTips_%d_Count", iClass ) );
int nTipCount = wzTipCount ? _wtoi( wzTipCount ) : 0;
for ( int iTip = 1; iTip < nTipCount+1; ++iTip )
{
const wchar_t *pwszText = g_pVGuiLocalize->Find( CFmtStr( "#ClassTips_%d_%d", iClass, iTip ) );
const wchar_t *pwszTextMvM = g_pVGuiLocalize->Find( CFmtStr( "#ClassTips_%d_%d_MvM", iClass, iTip ) );
wchar_t *pwszIcon = g_pVGuiLocalize->Find( CFmtStr( "ClassTips_%d_%d_Icon", iClass, iTip ) );
char szIcon[MAX_PATH];
szIcon[0] = 0;
if ( pwszIcon )
{
g_pVGuiLocalize->ConvertUnicodeToANSI( pwszIcon, szIcon, sizeof( szIcon ) );
}
// Don't load MvM tips outside the mode
if ( pwszTextMvM )
{
if ( !TFGameRules()->IsMannVsMachineMode() )
continue;
// If we're MvM mode, remember first MvM tip
if ( !nScrollToItem )
nScrollToItem = iTip;
}
// Create a TipsItemPanel for each tip
if ( pwszText || pwszTextMvM )
{
CTFClassTipsItemPanel *pClassTipsItemPanel = new CTFClassTipsItemPanel( this, "ClassTipsItemPanel", iTip );
if ( pwszText )
{
pClassTipsItemPanel->SetClassTip( pwszText, szIcon );
}
else if ( pwszTextMvM )
{
pClassTipsItemPanel->SetClassTip( pwszTextMvM, szIcon );
}
m_pClassTipsListPanel->AddItem( NULL, pClassTipsItemPanel );
}
}
if ( m_pClassTipsListPanel->GetItemCount() > 0 )
{
m_pClassTipsListPanel->SetFirstColumnWidth( 0 );
m_pClassTipsListPanel->ScrollToItem( nScrollToItem );
}
}
InvalidateLayout( true, true );
}
private:
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
virtual void ApplySchemeSettings( IScheme *pScheme )
{
BaseClass::ApplySchemeSettings( pScheme );
LoadControlSettings( "Resource/UI/ClassTipsList.res" );
}
CTFClassTipsListPanel *m_pClassTipsListPanel;
CFmtStr m_fmtResFilename;
};
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
CTFClassMenu::CTFClassMenu( IViewPort *pViewPort )
: CClassMenu( pViewPort )
{
MakePopup();
m_mouseoverButtons.RemoveAll();
m_iClassMenuKey = BUTTON_CODE_INVALID;
m_iCurrentClassIndex = TF_CLASS_HEAVYWEAPONS;
#ifdef _X360
m_pFooter = new CTFFooter( this, "Footer" );
#endif
m_pTFPlayerModelPanel = NULL;
m_pSelectAClassLabel = NULL;
m_pClassTipsPanel = new CTFClassTipsPanel( this, "ClassTipsPanel" );
Q_memset( m_pClassButtons, 0, sizeof( m_pClassButtons ) );
#ifndef _X360
char tempName[MAX_PATH];
for ( int i = 0 ; i < CLASS_COUNT_IMAGES ; ++i )
{
Q_snprintf( tempName, sizeof( tempName ), "countImage%d", i );
m_ClassCountImages[i] = new CTFImagePanel( this, tempName );
}
m_pCountLabel = NULL;
m_pLocalPlayerImage = new CTFImagePanel( this, "localPlayerImage" );
m_pLocalPlayerBG = new CTFImagePanel( this, "localPlayerBG" );
m_iLocalPlayerClass = TEAM_UNASSIGNED;
m_pClassButtons[TF_CLASS_SCOUT] = new CExImageButton( this, "scout", "", this );
m_pClassButtons[TF_CLASS_SOLDIER] = new CExImageButton( this, "soldier", "", this );
m_pClassButtons[TF_CLASS_PYRO] = new CExImageButton( this, "pyro", "", this );
m_pClassButtons[TF_CLASS_DEMOMAN] = new CExImageButton( this, "demoman", "", this );
m_pClassButtons[TF_CLASS_MEDIC] = new CExImageButton( this, "medic", "", this );
m_pClassButtons[TF_CLASS_HEAVYWEAPONS] = new CExImageButton( this, "heavyweapons", "", this );
m_pClassButtons[TF_CLASS_SNIPER] = new CExImageButton( this, "sniper", "", this );
m_pClassButtons[TF_CLASS_ENGINEER] = new CExImageButton( this, "engineer", "", this );
m_pClassButtons[TF_CLASS_SPY] = new CExImageButton( this, "spy", "", this );
m_pClassButtons[TF_CLASS_RANDOM] = new CExImageButton( this, "random", "", this );
#endif
m_pEditLoadoutButton = NULL;
m_nBaseMusicGuid = -1;
ListenForGameEvent( "localplayer_changeteam" );
ListenForGameEvent( "show_match_summary" );
Q_memset( m_pMvmUpgradeImages, 0, sizeof( m_pMvmUpgradeImages ) );
m_pMvmUpgradeImages[TF_CLASS_SCOUT] = new vgui::ImagePanel( this, "MvMUpgradeImageScout" );
m_pMvmUpgradeImages[TF_CLASS_SOLDIER] = new vgui::ImagePanel( this, "MvMUpgradeImageSolider" );
m_pMvmUpgradeImages[TF_CLASS_PYRO] = new vgui::ImagePanel( this, "MvMUpgradeImagePyro" );
m_pMvmUpgradeImages[TF_CLASS_DEMOMAN] = new vgui::ImagePanel( this, "MvMUpgradeImageDemoman" );
m_pMvmUpgradeImages[TF_CLASS_MEDIC] = new vgui::ImagePanel( this, "MvMUpgradeImageMedic" );
m_pMvmUpgradeImages[TF_CLASS_HEAVYWEAPONS] = new vgui::ImagePanel( this, "MvMUpgradeImageHeavy" );
m_pMvmUpgradeImages[TF_CLASS_SNIPER] = new vgui::ImagePanel( this, "MvMUpgradeImageSniper" );
m_pMvmUpgradeImages[TF_CLASS_ENGINEER] = new vgui::ImagePanel( this, "MvMUpgradeImageEngineer" );
m_pMvmUpgradeImages[TF_CLASS_SPY] = new vgui::ImagePanel( this, "MvMUpgradeImageSpy" );
vgui::ivgui()->AddTickSignal( GetVPanel() );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFClassMenu::ApplySchemeSettings( IScheme *pScheme )
{
BaseClass::ApplySchemeSettings( pScheme );
for ( int i = 0; i < TF_CLASS_MENU_BUTTONS; i++ )
{
m_pClassHintIcons[i] = nullptr;
}
// Load the .res file
if ( ::input->IsSteamControllerActive() )
{
LoadControlSettings( "Resource/UI/ClassSelection_SC.res" );
m_pCancelHintIcon = dynamic_cast< CSCHintIcon* >( FindChildByName( "CancelHintIcon" ) );
m_pEditLoadoutHintIcon = dynamic_cast< CSCHintIcon* >( FindChildByName( "EditLoadoutHintIcon" ) );
m_pClassHintIcons[TF_CLASS_SCOUT] = dynamic_cast< CSCHintIcon* >( FindChildByName( "ScoutHintIcon" ) );
m_pClassHintIcons[TF_CLASS_SOLDIER] = dynamic_cast< CSCHintIcon* >( FindChildByName( "SoldierHintIcon" ) );
m_pClassHintIcons[TF_CLASS_PYRO] = dynamic_cast< CSCHintIcon* >( FindChildByName( "PyroHintIcon" ) );
m_pClassHintIcons[TF_CLASS_DEMOMAN] = dynamic_cast< CSCHintIcon* >( FindChildByName( "DemomanHintIcon" ) );
m_pClassHintIcons[TF_CLASS_HEAVYWEAPONS] = dynamic_cast< CSCHintIcon* >( FindChildByName( "HeavyHintIcon" ) );
m_pClassHintIcons[TF_CLASS_MEDIC] = dynamic_cast< CSCHintIcon* >( FindChildByName( "MedicHintIcon" ) );
m_pClassHintIcons[TF_CLASS_SPY] = dynamic_cast< CSCHintIcon* >( FindChildByName( "SpyHintIcon" ) );
m_pClassHintIcons[TF_CLASS_ENGINEER] = dynamic_cast< CSCHintIcon* >( FindChildByName( "EngineerHintIcon" ) );
m_pClassHintIcons[TF_CLASS_SNIPER] = dynamic_cast< CSCHintIcon* >( FindChildByName( "SniperHintIcon" ) );
m_pClassHintIcons[TF_CLASS_RANDOM] = dynamic_cast< CSCHintIcon* >( FindChildByName( "RandomHintIcon" ) );
for ( int i = 0; i < TF_CLASS_MENU_BUTTONS; i++ )
{
if ( m_pClassHintIcons[i] )
{
m_pClassHintIcons[i]->SetVisible( false );
}
}
SetMouseInputEnabled( false );
}
else
{
LoadControlSettings( "Resource/UI/ClassSelection.res" );
m_pCancelHintIcon = m_pEditLoadoutHintIcon = nullptr;
SetMouseInputEnabled( true );
}
m_pTFPlayerModelPanel = dynamic_cast<CTFPlayerModelPanel*>( FindChildByName("TFPlayerModel") );
m_pSelectAClassLabel = dynamic_cast<CExLabel*>( FindChildByName( "ClassMenuSelect" ) );
m_pEditLoadoutButton = dynamic_cast<CExButton*>( FindChildByName( "EditLoadoutButton" ) );
const char *pTeamExtension = GetTeamNumber() == TF_TEAM_BLUE ? "blu" : "red";
for ( int i = 0; i < ARRAYSIZE( m_pClassButtons ); ++i )
{
if ( !m_pClassButtons[i] )
continue;
if ( !IsValidTFPlayerClass( i ) && i != TF_CLASS_RANDOM )
continue;
m_pClassButtons[i]->SetImageSelected( CFmtStr( "class_sel_sm_%s_%s", g_aRawPlayerClassNamesShort[i], pTeamExtension ).Access() );
if( i != TF_CLASS_RANDOM )
{
m_pClassButtons[i]->SetArmedSound("misc/null.wav");
}
}
}
void CTFClassMenu::PerformLayout()
{
BaseClass::PerformLayout();
#ifndef _X360
m_pCountLabel = dynamic_cast< CExLabel * >( FindChildByName( "CountLabel" ) );
if ( m_pCountLabel )
{
m_pCountLabel->SizeToContents();
}
#endif
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
int CTFClassMenu::GetCurrentPlayerClass()
{
int iClass = TF_CLASS_HEAVYWEAPONS;
C_TFPlayer *pLocalPlayer = C_TFPlayer::GetLocalTFPlayer();
if ( pLocalPlayer && pLocalPlayer->m_Shared.GetDesiredPlayerClassIndex() != TF_CLASS_UNDEFINED )
{
iClass = pLocalPlayer->m_Shared.GetDesiredPlayerClassIndex();
}
return iClass;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
CExImageButton *CTFClassMenu::GetCurrentClassButton()
{
const int iClass = GetCurrentPlayerClass();
m_iCurrentClassIndex = iRemapIndexToClass[ iClass ];
return m_pClassButtons[iClass];
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFClassMenu::ShowPanel( bool bShow )
{
if ( bShow )
{
// Hide the other class menu
if ( gViewPortInterface )
{
gViewPortInterface->ShowPanel( GetTeamNumber() == TF_TEAM_BLUE ? PANEL_CLASS_RED : PANEL_CLASS_BLUE, false );
}
// can't change class if you're on the losing team during the "bonus time" after a team has won the round
if ( ( TFGameRules()->State_Get() == GR_STATE_TEAM_WIN &&
C_TFPlayer::GetLocalTFPlayer() &&
C_TFPlayer::GetLocalTFPlayer()->GetTeamNumber() != TFGameRules()->GetWinningTeam()
&& C_TFPlayer::GetLocalTFPlayer()->GetTeamNumber() != TEAM_SPECTATOR
&& C_TFPlayer::GetLocalTFPlayer()->GetTeamNumber() != TEAM_UNASSIGNED
&& GetSpectatorMode() == OBS_MODE_NONE ) ||
TFGameRules()->State_Get() == GR_STATE_GAME_OVER ||
( TFGameRules()->IsInTraining() && C_TFPlayer::GetLocalTFPlayer() &&
( C_TFPlayer::GetLocalTFPlayer()->GetPlayerClass() == NULL || C_TFPlayer::GetLocalTFPlayer()->GetPlayerClass()->GetClassIndex() != TF_CLASS_UNDEFINED ) ) )
{
SetVisible( false );
CHudNotificationPanel *pNotifyPanel = GET_HUDELEMENT( CHudNotificationPanel );
if ( pNotifyPanel )
{
if ( C_TFPlayer::GetLocalTFPlayer() )
{
pNotifyPanel->SetupNotifyCustom( "#TF_CantChangeClassNow", "ico_notify_flag_moving", C_TFPlayer::GetLocalTFPlayer()->GetTeamNumber() );
}
}
return;
}
engine->CheckPoint( "ClassMenu" );
// Force us to reload our scheme, in case Steam Controller stuff has changed.
InvalidateLayout( true, true );
Activate();
m_iClassMenuKey = gameuifuncs->GetButtonCodeForBind( "changeclass" );
m_iScoreBoardKey = gameuifuncs->GetButtonCodeForBind( "showscores" );
SelectClass( GetCurrentPlayerClass() );
}
else
{
SetVisible( false );
if ( m_pTFPlayerModelPanel )
{
m_pTFPlayerModelPanel->ClearCarriedItems();
}
}
}
const char *g_pszLegacyClassSelectVCDWeapons[TF_LAST_NORMAL_CLASS] =
{
"", // TF_CLASS_UNDEFINED = 0,
"", // TF_CLASS_SCOUT, // weapons handled individually
"", // TF_CLASS_SNIPER, // weapons handled individually
"", // TF_CLASS_SOLDIER, // weapons handled individually
"tf_weapon_grenadelauncher", // TF_CLASS_DEMOMAN,
"tf_weapon_medigun", // TF_CLASS_MEDIC,
"tf_weapon_minigun", // TF_CLASS_HEAVYWEAPONS,
"tf_weapon_flamethrower", // TF_CLASS_PYRO,
"", // TF_CLASS_SPY, // weapons handled individually
"tf_weapon_wrench", // TF_CLASS_ENGINEER,
};
int g_iLegacyClassSelectWeaponSlots[TF_LAST_NORMAL_CLASS] =
{
LOADOUT_POSITION_PRIMARY, // TF_CLASS_UNDEFINED = 0,
LOADOUT_POSITION_PRIMARY, // TF_CLASS_SCOUT, // TF_FIRST_NORMAL_CLASS
LOADOUT_POSITION_PRIMARY, // TF_CLASS_SNIPER,
LOADOUT_POSITION_PRIMARY, // TF_CLASS_SOLDIER,
LOADOUT_POSITION_PRIMARY, // TF_CLASS_DEMOMAN,
LOADOUT_POSITION_SECONDARY, // TF_CLASS_MEDIC,
LOADOUT_POSITION_PRIMARY, // TF_CLASS_HEAVYWEAPONS,
LOADOUT_POSITION_PRIMARY, // TF_CLASS_PYRO,
LOADOUT_POSITION_MELEE, // TF_CLASS_SPY,
LOADOUT_POSITION_MELEE, // TF_CLASS_ENGINEER,
};
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFClassMenu::UpdateButtonSelectionStates( int iClass )
{
// Set the correct button as selected, all other buttons as not selected
for ( int i = 0; i < ARRAYSIZE( m_pClassButtons ); ++i )
{
if ( !m_pClassButtons[ i ] )
continue;
m_pClassButtons[ i ]->SetSelected( i == iClass );
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFClassMenu::SelectClass( int iClass )
{
if ( !engine->IsInGame() )
return;
if ( !m_pTFPlayerModelPanel )
return;
if ( !m_pClassTipsPanel )
return;
if ( !m_pEditLoadoutButton )
return;
// Update select hint icon for Steam Controller
for ( int i = 0; i < TF_CLASS_MENU_BUTTONS; i++ )
{
if ( m_pClassHintIcons[i] )
{
m_pClassHintIcons[i]->SetVisible( i == iClass );
}
}
// Were we random? If so, we'll force our class to refresh later to prevent the
// model panel thinking the class hasn't changed.
const bool bClassWasRandom = m_iCurrentClassIndex == TF_CLASS_RANDOM;
// Cache current player class
m_iCurrentClassIndex = iClass;
UpdateButtonSelectionStates( iClass );
bool bRandomClass = iClass == TF_CLASS_RANDOM;
if ( !IsValidTFPlayerClass( iClass ) && !bRandomClass )
{
m_pTFPlayerModelPanel->SetVisible( false );
m_pTFPlayerModelPanel->ClearCarriedItems();
return;
}
m_pTFPlayerModelPanel->SetVisible( true );
m_pTFPlayerModelPanel->ClearCarriedItems();
if ( bRandomClass )
{
m_pEditLoadoutButton->SetVisible( false );
if ( m_pEditLoadoutHintIcon )
{
m_pEditLoadoutHintIcon->SetVisible( false );
}
MDLHandle_t hModel = mdlcache->FindMDL( "models/class_menu/random_class_icon.mdl" );
m_pTFPlayerModelPanel->SetMDL( hModel );
m_pTFPlayerModelPanel->SetSequence( ACT_IDLE );
m_pTFPlayerModelPanel->InvalidateLayout( true, true ); // Updates position
m_pTFPlayerModelPanel->SetSkin( GetTeamNumber() == TF_TEAM_RED ? 0 : 1 );
mdlcache->Release( hModel ); // counterbalance addref from within FindMDL
}
else
{
bool bIsRobot = false;
// Check for Robot
static CSchemaAttributeDefHandle pAttrDef_PlayerRobot( "appear as mvm robot" );
for ( int i = 0; i < CLASS_LOADOUT_POSITION_COUNT; i++ )
{
CEconItemView *pItemData = TFInventoryManager()->GetItemInLoadoutForClass( iClass, i );
if ( !pItemData )
continue;
if ( FindAttribute( pItemData, pAttrDef_PlayerRobot ) )
{
bIsRobot = true;
break;
}
}
/*if ( pLocalPlayer )
{
int iRobot = 0;
CALL_ATTRIB_HOOK_INT_ON_OTHER( pLocalPlayer, iRobot, appear_as_mvm_robot );
bIsRobot = iRobot ? true : false;
}*/
m_pTFPlayerModelPanel->SetToPlayerClass( iClass, bIsRobot, bClassWasRandom );
m_pEditLoadoutButton->SetVisible( true );
if ( m_pEditLoadoutHintIcon )
{
m_pEditLoadoutHintIcon->SetVisible( true );
}
C_BasePlayer *pLocalPlayer = C_BasePlayer::GetLocalPlayer();
if ( pLocalPlayer )
{
int iTeam = GetTeamNumber();
m_pTFPlayerModelPanel->SetTeam( iTeam );
}
LoadItems();
}
m_pClassTipsPanel->SetClass( iClass );
enginesound->StopSoundByGuid( m_nBaseMusicGuid );
CBroadcastRecipientFilter filter;
char nClassMusicStr[64];
if ( TFGameRules() && TFGameRules()->IsMannVsMachineMode() )
{
sprintf( nClassMusicStr, "music.mvm_class_menu_0%i", iClass );
}
else
{
sprintf( nClassMusicStr, "music.class_menu_0%i", iClass );
}
CBaseEntity::EmitSound( filter, SOUND_FROM_UI_PANEL, nClassMusicStr );
m_nBaseMusicGuid = enginesound->GetGuidForLastSoundEmitted();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFClassMenu::LoadItems()
{
const int iClass = m_pTFPlayerModelPanel->GetPlayerClass();
m_pTFPlayerModelPanel->ClearCarriedItems();
static CSchemaAttributeDefHandle pAttrDef_DisableFancyLoadoutAnim( "disable fancy class select anim" );
bool bCanUseFancyClassSelectAnimation = true;
static CSchemaAttributeDefHandle pAttrDef_ClassSelectOverrideVCD( "class select override vcd" );
CAttribute_String attrClassSelectOverrideVCD;
const char *pszVCD = "class_select";
for ( int i = 0; i < CLASS_LOADOUT_POSITION_COUNT; i++ )
{
CEconItemView *pItemData = TFInventoryManager()->GetItemInLoadoutForClass( iClass, i );
if ( pItemData && pItemData->IsValid() )
{
m_pTFPlayerModelPanel->AddCarriedItem( pItemData );
// Certain items have different shapes and would interfere with our class select animations.
bCanUseFancyClassSelectAnimation = bCanUseFancyClassSelectAnimation
&& !pItemData->FindAttribute( pAttrDef_DisableFancyLoadoutAnim );
// Some items want to override the class select VCD
if ( pItemData->FindAttribute( pAttrDef_ClassSelectOverrideVCD, &attrClassSelectOverrideVCD ) )
{
const char *pszClassSelectOverrideVCD = attrClassSelectOverrideVCD.value().c_str();
if ( pszClassSelectOverrideVCD && *pszClassSelectOverrideVCD )
{
pszVCD = pszClassSelectOverrideVCD;
}
}
}
}
m_pTFPlayerModelPanel->PlayVCD( bCanUseFancyClassSelectAnimation ? pszVCD : NULL, g_pszLegacyClassSelectVCDWeapons[iClass] );
m_pTFPlayerModelPanel->HoldItemInSlot( g_iLegacyClassSelectWeaponSlots[iClass] );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFClassMenu::OnKeyCodePressed( KeyCode code )
{
m_KeyRepeat.KeyDown( code );
if ( code > KEY_0 && code <= KEY_9 )
{
const int iButton = code - KEY_0;
const int iClass = iRemapIndexToClass[ iButton ];
SelectClass( iClass );
Go();
}
else if ( code > KEY_PAD_0 && code <= KEY_PAD_9 )
{
const int iButton = code - KEY_PAD_0;
const int iClass = iRemapIndexToClass[ iButton ];
SelectClass( iClass );
Go();
}
else if( code == KEY_ENTER || code == KEY_SPACE || code == KEY_XBUTTON_A || code == KEY_XBUTTON_RTRIGGER || code == STEAMCONTROLLER_A )
{
Go();
}
else if ( ( m_iClassMenuKey != BUTTON_CODE_INVALID && m_iClassMenuKey == code ) ||
code == KEY_XBUTTON_BACK ||
code == KEY_XBUTTON_B ||
code == STEAMCONTROLLER_B ||
code == KEY_0 ||
code == KEY_PAD_0 )
{
C_TFPlayer *pLocalPlayer = C_TFPlayer::GetLocalTFPlayer();
if ( pLocalPlayer && ( pLocalPlayer->GetPlayerClass()->GetClassIndex() != TF_CLASS_UNDEFINED ) )
{
ShowPanel( false );
}
}
else if( code == KEY_XBUTTON_RIGHT || code == KEY_XSTICK1_RIGHT || code == STEAMCONTROLLER_DPAD_RIGHT )
{
int loopCheck = 0;
int nCurrentClass = GetRemappedMenuIndexForClass( m_iCurrentClassIndex );
do
{
loopCheck++;
nCurrentClass++;
nCurrentClass = ( nCurrentClass % TF_CLASS_MENU_BUTTONS );
} while( ( m_pClassButtons[ iRemapIndexToClass[nCurrentClass] ] == NULL ) && ( loopCheck < TF_CLASS_MENU_BUTTONS ) );
SelectClass( iRemapIndexToClass[ nCurrentClass ] );
}
else if ( code == STEAMCONTROLLER_Y )
{
OnCommand( "openloadout" );
}
else if( code == KEY_XBUTTON_LEFT || code == KEY_XSTICK1_LEFT || code == STEAMCONTROLLER_DPAD_LEFT )
{
int loopCheck = 0;
int nCurrentClass = GetRemappedMenuIndexForClass( m_iCurrentClassIndex );
do
{
loopCheck++;
nCurrentClass--;
if( nCurrentClass <= 0 )
{
nCurrentClass = GetRemappedMenuIndexForClass( TF_CLASS_RANDOM );
}
} while( ( m_pClassButtons[ iRemapIndexToClass[nCurrentClass] ] == NULL ) && ( loopCheck < TF_CLASS_MENU_BUTTONS ) );
SelectClass( iRemapIndexToClass[ nCurrentClass ] );
}
else if( code == KEY_XBUTTON_UP || code == KEY_XSTICK1_UP || code == STEAMCONTROLLER_DPAD_UP )
{
// Scroll class info text up
if ( g_lastPanel )
{
CExRichText *pRichText = dynamic_cast< CExRichText * >( g_lastPanel->FindChildByName( "classInfo" ) );
if ( pRichText )
{
PostMessage( pRichText, new KeyValues("MoveScrollBarDirect", "delta", 1) );
}
}
}
else if( code == KEY_XBUTTON_DOWN || code == KEY_XSTICK1_DOWN || code == STEAMCONTROLLER_DPAD_DOWN )
{
// Scroll class info text up
if ( g_lastPanel )
{
CExRichText *pRichText = dynamic_cast< CExRichText * >( g_lastPanel->FindChildByName( "classInfo" ) );
if ( pRichText )
{
PostMessage( pRichText, new KeyValues("MoveScrollBarDirect", "delta", -1) );
}
}
}
else
{
BaseClass::OnKeyCodePressed( code );
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFClassMenu::OnKeyCodeReleased( vgui::KeyCode code )
{
m_KeyRepeat.KeyUp( code );
BaseClass::OnKeyCodeReleased( code );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFClassMenu::OnThink()
{
vgui::KeyCode code = m_KeyRepeat.KeyRepeated();
if ( code )
{
OnKeyCodePressed( code );
}
// Get mouse cursor position
int aCursorPos[2];
vgui::input()->GetCursorPos( aCursorPos[0], aCursorPos[1] );
// Go through all buttons - if the mouse is within one, select that class
for ( int i = 0; i < ARRAYSIZE( m_pClassButtons ); ++i )
{
if ( !m_pClassButtons[ i ] )
continue;
if ( m_iCurrentClassIndex != i && m_pClassButtons[ i ]->IsWithin( aCursorPos[0], aCursorPos[1] ) )
{
SelectClass( i );
}
}
//Always hide the health... this needs to be done every frame because a message from the server keeps resetting this.
C_BasePlayer *pLocalPlayer = C_BasePlayer::GetLocalPlayer();
if ( pLocalPlayer )
{
pLocalPlayer->m_Local.m_iHideHUD |= HIDEHUD_HEALTH;
}
BaseClass::OnThink();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFClassMenu::SetCancelButtonVisible( bool bVisible )
{
SetVisibleButton( "CancelButton", bVisible );
if ( m_pCancelHintIcon )
{
m_pCancelHintIcon->SetVisible( bVisible );
}
if ( m_pSelectAClassLabel )
{
m_pSelectAClassLabel->SetVisible( !bVisible );
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFClassMenu::Update()
{
C_TFPlayer *pLocalPlayer = C_TFPlayer::GetLocalTFPlayer();
// Force them to pick a class if they haven't picked one yet.
if ( ( pLocalPlayer && pLocalPlayer->m_Shared.GetDesiredPlayerClassIndex() != TF_CLASS_UNDEFINED ) )
{
#ifdef _X360
if ( m_pFooter )
{
m_pFooter->ShowButtonLabel( "cancel", true );
}
#else
SetCancelButtonVisible( true );
if ( TFGameRules() && TFGameRules()->IsInHighlanderMode() )
{
SetVisibleButton( "ResetButton", true );
}
else
{
SetVisibleButton( "ResetButton", false );
}
#endif
}
else
{
#ifdef _X360
if ( m_pFooter )
{
m_pFooter->ShowButtonLabel( "cancel", false );
}
#else
SetCancelButtonVisible( false );
SetVisibleButton( "ResetButton", false );
#endif
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
Panel *CTFClassMenu::CreateControlByName( const char *controlName )
{
if ( !Q_stricmp( "CIconPanel", controlName ) )
{
return new CIconPanel( this, "icon_panel" );
}
else
{
return BaseClass::CreateControlByName( controlName );
}
}
//-----------------------------------------------------------------------------
// Catch the mouseover event and set the active class
//-----------------------------------------------------------------------------
void CTFClassMenu::OnShowPage( vgui::Panel *panel, const char *pagename )
{
for ( int i = 0; i < TF_CLASS_MENU_BUTTONS; i++ )
{
if (m_pClassButtons[i] == panel )
{
// SelectClass( i );
break;
}
}
}
//-----------------------------------------------------------------------------
// Draw nothing
//-----------------------------------------------------------------------------
void CTFClassMenu::PaintBackground( void )
{
}
//-----------------------------------------------------------------------------
// Do things that should be done often, eg number of players in the
// selected class
//-----------------------------------------------------------------------------
void CTFClassMenu::OnTick( void )
{
//When a player changes teams, their class and team values don't get here
//necessarily before the command to update the class menu. This leads to the cancel button
//being visible and people cancelling before they have a class. check for class == TF_CLASS_UNDEFINED and if so
//hide the cancel button
if ( !IsVisible() )
return;
#ifndef _X360
C_TFPlayer *pLocalPlayer = C_TFPlayer::GetLocalTFPlayer();
// Force them to pick a class if they haven't picked one yet.
if ( pLocalPlayer && pLocalPlayer->m_Shared.GetDesiredPlayerClassIndex() == TF_CLASS_UNDEFINED )
{
SetCancelButtonVisible( false );
SetVisibleButton( "ResetButton", false );
}
UpdateClassCounts();
#endif
BaseClass::OnTick();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFClassMenu::OnClose()
{
C_BasePlayer *pLocalPlayer = C_BasePlayer::GetLocalPlayer();
if ( pLocalPlayer )
{
// Clear the HIDEHUD_HEALTH bit we hackily added. Turns out prediction
// was restoring these bits every frame. Unfortunately, prediction
// is off for karts which means the spell hud item would disappear if you
// brought up this menu and returned.
pLocalPlayer->m_Local.m_iHideHUD &= ~HIDEHUD_HEALTH;
}
ShowPanel( false );
BaseClass::OnClose();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFClassMenu::SetVisible( bool state )
{
BaseClass::SetVisible( state );
m_KeyRepeat.Reset();
if ( state )
{
engine->ServerCmd( "menuopen" ); // to the server
engine->ClientCmd( "_cl_classmenuopen 1" ); // for other panels
CBroadcastRecipientFilter filter;
if ( TFGameRules() && TFGameRules()->IsMannVsMachineMode() )
{
CBaseEntity::EmitSound( filter, SOUND_FROM_UI_PANEL, "music.mvm_class_menu" );
}
else
{
CBaseEntity::EmitSound( filter, SOUND_FROM_UI_PANEL, "music.class_menu" );
}
CheckMvMUpgrades();
}
else
{
engine->ServerCmd( "menuclosed" );
engine->ClientCmd( "_cl_classmenuopen 0" );
if ( TFGameRules() && TFGameRules()->IsMannVsMachineMode() )
{
CBaseEntity::StopSound( SOUND_FROM_UI_PANEL, "music.mvm_class_menu" );
}
else
{
CBaseEntity::StopSound( SOUND_FROM_UI_PANEL, "music.class_menu" );
}
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFClassMenu::Go()
{
const int iClass = m_iCurrentClassIndex;
if ( iClass == TF_CLASS_UNDEFINED )
return;
// Check class limits
if ( TFGameRules() && !TFGameRules()->CanPlayerChooseClass( C_TFPlayer::GetLocalTFPlayer(), iClass ) )
return;
#if defined( REPLAY_ENABLED )
// Display replay recording message if appropriate
int &nDisplayedConnectedRecording = CPlayerSpawnCache::Instance().m_Data.m_nDisplayedConnectedRecording;
if ( g_pReplay->IsReplayEnabled() &&
!g_pEngineClientReplay->IsPlayingReplayDemo() && // FIXME: We shouldn't need this here but for some reason the engine thinks a replay is recording during demo playback, even though replay_recording has a FCVAR_DONTRECORD flag
!nDisplayedConnectedRecording &&
replay_replaywelcomedlgcount.GetInt() <= MAX_TIMES_TO_SHOW_REPLAY_WELCOME_DLG )
{
wchar_t wText[256];
wchar wKeyBind[80];
char szText[256];
const char *pSaveReplayKey = engine->Key_LookupBinding( "save_replay" );
if ( !pSaveReplayKey )
{
pSaveReplayKey = "< not bound >";
}
g_pVGuiLocalize->ConvertANSIToUnicode( pSaveReplayKey, wKeyBind, sizeof( wKeyBind ) );
g_pVGuiLocalize->ConstructString_safe( wText, g_pVGuiLocalize->Find( "#Replay_ConnectRecording" ), 1, wKeyBind );
g_pVGuiLocalize->ConvertUnicodeToANSI( wText, szText, sizeof( szText ) );
extern ConVar replay_msgduration_connectrecording;
g_pClientMode->DisplayReplayMessage( szText, replay_msgduration_connectrecording.GetFloat(), false, NULL, true );
// Don't execute this clause next time the player spawns, unless the cache has been cleared
++nDisplayedConnectedRecording;
// Increment (archives)
replay_replaywelcomedlgcount.SetValue( replay_replaywelcomedlgcount.GetInt() + 1 );
}
#endif
// This will complete any pending replay, commit if necessary, and clear - this way when the player respawns
// we will start with a fresh replay for the new life.
g_pClientReplayContext->OnPlayerClassChanged();
// Change class
BaseClass::OnCommand( CFmtStr( "joinclass %s", g_aRawPlayerClassNames[ iClass ] ).Access() );
CBroadcastRecipientFilter filter;
if ( TFGameRules() && TFGameRules()->IsMannVsMachineMode() )
{
CBaseEntity::EmitSound( filter, SOUND_FROM_UI_PANEL, "music.mvm_class_select" );
}
else
{
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFClassMenu::OnCommand( const char *command )
{
if ( !V_strnicmp( command, "select", 6 ) )
{
const char *pClass = command + 6;
const int iClass = atoi( pClass );
// Avoid restarting the animation if the user selected on the same class
if ( iClass != m_iCurrentClassIndex )
{
SelectClass( iClass );
}
else
{
// Ensure selection states, in case the user clicks a button and drags away
UpdateButtonSelectionStates( iClass );
}
Go();
}
else if ( !V_strnicmp( command, "resetclass", 10 ) )
{
if ( TFGameRules() && !TFGameRules()->IsInHighlanderMode() )
return;
engine->ClientCmd( const_cast<char *>( command ) );
}
else if ( !V_strnicmp( command, "openloadout", 11 ) )
{
// Let this panel know when you've closed, so we can reload items
EconUI()->AddPanelCloseListener( this );
// Make the back button close, rather than go back to the econ root panel
EconUI()->SetClosePanel( -m_iCurrentClassIndex );
// Set team number, so the model's color will match
EconUI()->SetDefaultTeam( GetTeamNumber() );
// Go directly to the loadout for the selected class
EconUI()->OpenEconUI( -m_iCurrentClassIndex );
}
else
{
BaseClass::OnCommand( command );
}
}
//-----------------------------------------------------------------------------
// Purpose: Console command to select a class
//-----------------------------------------------------------------------------
void CTFClassMenu::Join_Class( const CCommand &args )
{
if ( args.ArgC() > 1 )
{
char cmd[256];
Q_snprintf( cmd, sizeof( cmd ), "joinclass %s", args.Arg( 1 ) );
OnCommand( cmd );
ShowPanel( false );
}
}
static const char *g_sDialogVariables[] = {
"",
"numScout",
"numSoldier",
"numPyro",
"numDemoman",
"numHeavy",
"numEngineer",
"numMedic",
"numSniper",
"numSpy",
"",
};
static const char *g_sClassImagesBlue[] = {
"",
"class_sel_sm_scout_blu",
"class_sel_sm_soldier_blu",
"class_sel_sm_pyro_blu",
"class_sel_sm_demo_blu",
"class_sel_sm_heavy_blu",
"class_sel_sm_engineer_blu",
"class_sel_sm_medic_blu",
"class_sel_sm_sniper_blu",
"class_sel_sm_spy_blu",
"class_sel_sm_scout_blu",
};
static const char *g_sClassImagesRed[] = {
"",
"class_sel_sm_scout_red",
"class_sel_sm_soldier_red",
"class_sel_sm_pyro_red",
"class_sel_sm_demo_red",
"class_sel_sm_heavy_red",
"class_sel_sm_engineer_red",
"class_sel_sm_medic_red",
"class_sel_sm_sniper_red",
"class_sel_sm_spy_red",
"class_sel_sm_scout_red",
};
int g_ClassDefinesRemap[] = {
0,
TF_CLASS_SCOUT,
TF_CLASS_SOLDIER,
TF_CLASS_PYRO,
TF_CLASS_DEMOMAN,
TF_CLASS_HEAVYWEAPONS,
TF_CLASS_ENGINEER,
TF_CLASS_MEDIC,
TF_CLASS_SNIPER,
TF_CLASS_SPY,
TF_CLASS_CIVILIAN,
};
void CTFClassMenu::UpdateNumClassLabels( int iTeam )
{
#ifndef _X360
int nTotalCount = 0;
// count how many of each class there are
if ( !g_TF_PR )
return;
if ( iTeam < FIRST_GAME_TEAM || iTeam >= TF_TEAM_COUNT ) // invalid team number
return;
C_TFPlayer *pLocalPlayer = C_TFPlayer::GetLocalTFPlayer();
bool bSpectator = pLocalPlayer && pLocalPlayer->GetTeamNumber() == TEAM_SPECTATOR;
int iLocalPlayerClass = TF_CLASS_UNDEFINED;
if ( pLocalPlayer )
{
iLocalPlayerClass = pLocalPlayer->m_Shared.GetDesiredPlayerClassIndex();
}
if ( iLocalPlayerClass == TF_CLASS_UNDEFINED )
{
m_iLocalPlayerClass = iLocalPlayerClass;
if ( m_pLocalPlayerImage && m_pLocalPlayerImage->IsVisible() )
{
m_pLocalPlayerImage->SetVisible( false );
}
if ( m_pLocalPlayerBG && m_pLocalPlayerBG->IsVisible() )
{
m_pLocalPlayerBG->SetVisible( false );
}
}
for( int i = TF_FIRST_NORMAL_CLASS ; i <= TF_LAST_NORMAL_CLASS ; i++ )
{
if ( bSpectator == true )
{
SetDialogVariable( g_sDialogVariables[i], "" );
continue;
}
int classCount = g_TF_PR->GetCountForPlayerClass( iTeam, g_ClassDefinesRemap[i], false );
int iClassLimit = TFGameRules()->GetClassLimit( g_ClassDefinesRemap[i] );
if ( iClassLimit != NO_CLASS_LIMIT )
{
if ( classCount >= iClassLimit )
{
if ( classCount > 0 )
{
wchar_t wTemp[32];
wchar_t wzCount[10];
_snwprintf( wzCount, ARRAYSIZE( wzCount ), L"%d", classCount );
g_pVGuiLocalize->ConstructString_safe( wTemp, g_pVGuiLocalize->Find("TF_ClassLimitHit"), 1, wzCount );
SetDialogVariable( g_sDialogVariables[i], wTemp );
}
else
{
SetDialogVariable( g_sDialogVariables[i], g_pVGuiLocalize->Find("TF_ClassLimitHit_None") );
}
}
else
{
wchar_t wTemp[32];
wchar_t wzCount[10];
_snwprintf( wzCount, ARRAYSIZE( wzCount ), L"%d", classCount );
wchar_t wzMax[10];
_snwprintf( wzMax, ARRAYSIZE( wzMax ), L"%d", iClassLimit );
g_pVGuiLocalize->ConstructString_safe( wTemp, g_pVGuiLocalize->Find("TF_ClassLimitUnder"), 2, wzCount, wzMax );
SetDialogVariable( g_sDialogVariables[i], wTemp );
}
}
else if ( classCount > 0 )
{
SetDialogVariable( g_sDialogVariables[i], classCount );
}
else
{
SetDialogVariable( g_sDialogVariables[i], "" );
}
if ( g_ClassDefinesRemap[i] == iLocalPlayerClass )
{
// take 1 off the count for the images since the local player has their own image already
if ( classCount > 0 )
{
classCount--;
}
if ( m_pLocalPlayerImage )
{
if ( !m_pLocalPlayerImage->IsVisible() )
{
m_pLocalPlayerImage->SetVisible( true );
}
if ( m_iLocalPlayerClass != iLocalPlayerClass )
{
m_iLocalPlayerClass = iLocalPlayerClass;
m_pLocalPlayerImage->SetImage( iTeam == TF_TEAM_BLUE ? g_sClassImagesBlue[i] : g_sClassImagesRed[i] );
}
}
if ( m_pLocalPlayerBG && !m_pLocalPlayerBG->IsVisible() )
{
m_pLocalPlayerBG->SetVisible( true );
}
}
if ( nTotalCount < CLASS_COUNT_IMAGES )
{
for ( int j = 0 ; j < classCount ; ++j )
{
CTFImagePanel *pImage = m_ClassCountImages[nTotalCount];
if ( pImage )
{
pImage->SetVisible( true );
pImage->SetImage( iTeam == TF_TEAM_BLUE ? g_sClassImagesBlue[i] : g_sClassImagesRed[i] );
}
nTotalCount++;
if ( nTotalCount >= CLASS_COUNT_IMAGES )
{
break;
}
}
}
}
if ( nTotalCount == 0 )
{
// no classes for our team yet
if ( m_pCountLabel && m_pCountLabel->IsVisible() )
{
m_pCountLabel->SetVisible( false );
}
}
else
{
if ( m_pCountLabel && !m_pCountLabel->IsVisible() )
{
m_pCountLabel->SetVisible( true );
}
}
// turn off any unused images
while ( nTotalCount < CLASS_COUNT_IMAGES )
{
CTFImagePanel *pImage = m_ClassCountImages[nTotalCount];
if ( pImage )
{
pImage->SetVisible( false );
}
nTotalCount++;
}
#endif
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFClassMenu::FireGameEvent( IGameEvent *event )
{
const char *pszEventName = event->GetName();
// when we are changing levels
if ( FStrEq( pszEventName, "localplayer_changeteam" ) )
{
if ( IsVisible() )
{
C_BasePlayer *pLocalPlayer = C_BasePlayer::GetLocalPlayer();
if ( pLocalPlayer )
{
int iTeam = pLocalPlayer->GetTeamNumber();
if ( iTeam != GetTeamNumber() )
{
ShowPanel( false );
if ( iTeam == TF_TEAM_BLUE )
{
gViewPortInterface->ShowPanel( PANEL_CLASS_BLUE, true );
}
else
{
gViewPortInterface->ShowPanel( PANEL_CLASS_RED, true );
}
}
}
}
}
else if ( FStrEq( pszEventName, "show_match_summary" ) )
{
if ( IsVisible() )
{
ShowPanel( false );
}
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CTFClassMenu::OnEconUIClosed()
{
// Reload items on model panel, in case anything's changed
LoadItems();
}
int g_nNumUpgradeIconsForLastHint = 0;
//-----------------------------------------------------------------------------
void CTFClassMenu::CheckMvMUpgrades()
{
// Set MvM Icons invisible
for ( int icons = 0; icons < ARRAYSIZE( m_pMvmUpgradeImages ); ++icons )
{
if ( m_pMvmUpgradeImages[icons] == NULL )
continue;
m_pMvmUpgradeImages[icons]->SetVisible( false );
}
if ( !TFGameRules() || !TFGameRules()->IsMannVsMachineMode() )
return;
CMannVsMachineStats *pStats = MannVsMachineStats_GetInstance();
if ( !pStats )
return;
CUtlVector< CUpgradeInfo > *upgrades = pStats->GetLocalPlayerUpgrades();
int nShowUpgradingHint = -1;
int nNumUpgradeIconsForHint = 0;
for ( int i = 0; i < upgrades->Count(); ++i )
{
vgui::Panel *pUpgradeImage = m_pMvmUpgradeImages[upgrades->Element(i).m_iPlayerClass];
if ( !pUpgradeImage )
continue;
if ( !pUpgradeImage->IsVisible() )
{
pUpgradeImage->SetVisible( true );
nNumUpgradeIconsForHint++;
// Only show the hint if we've shown it 3 or less times ever
if ( nShowUpgradingHint == -1 && tf_mvm_classupgradehelpcount.GetInt() < 3 )
{
int nY;
pUpgradeImage->GetPos( nShowUpgradingHint, nY );
nShowUpgradingHint += pUpgradeImage->GetWide() / 2;
}
}
}
// Only show the hint if there are more upgrade icon than the last time we openned the menu
if ( nShowUpgradingHint != -1 && g_nNumUpgradeIconsForLastHint < nNumUpgradeIconsForHint )
{
CExplanationPopup *pPopup = dynamic_cast< CExplanationPopup* >( FindChildByName("StartExplanation") );
if ( pPopup )
{
pPopup->SetCalloutInParentsX( nShowUpgradingHint );
pPopup->Popup();
g_nNumUpgradeIconsForLastHint = nNumUpgradeIconsForHint;
tf_mvm_classupgradehelpcount.SetValue( tf_mvm_classupgradehelpcount.GetInt() + 1 );
}
}
}