//========= Copyright Valve Corporation, All rights reserved. ============// // // Purpose: // //=============================================================================// #include "cbase.h" #include "weapon_selection.h" #include "iclientmode.h" #include "history_resource.h" #include "input.h" #include "../hud_crosshair.h" #include "VGuiMatSurface/IMatSystemSurface.h" #include <KeyValues.h> #include <vgui/IScheme.h> #include <vgui/ISurface.h> #include <vgui/ISystem.h> #include <vgui_controls/AnimationController.h> #include <vgui_controls/Panel.h> #include "vgui/ILocalize.h" // memdbgon must be the last include file in a .cpp file!!! #include "tier0/memdbgon.h" ConVar hud_showemptyweaponslots( "hud_showemptyweaponslots", "1", FCVAR_ARCHIVE, "Shows slots for missing weapons when recieving weapons out of order" ); #define SELECTION_TIMEOUT_THRESHOLD 0.5f // Seconds #define SELECTION_FADEOUT_TIME 0.75f #define PLUS_DISPLAY_TIMEOUT 0.5f // Seconds #define PLUS_FADEOUT_TIME 0.75f #define FASTSWITCH_DISPLAY_TIMEOUT 1.5f #define FASTSWITCH_FADEOUT_TIME 1.5f #define CAROUSEL_SMALL_DISPLAY_ALPHA 200.0f #define FASTSWITCH_SMALL_DISPLAY_ALPHA 160.0f #define MAX_CAROUSEL_SLOTS 5 //----------------------------------------------------------------------------- // Purpose: hl2 weapon selection hud element //----------------------------------------------------------------------------- class CHudWeaponSelection : public CBaseHudWeaponSelection, public vgui::Panel { DECLARE_CLASS_SIMPLE( CHudWeaponSelection, vgui::Panel ); public: CHudWeaponSelection(const char *pElementName ); virtual bool ShouldDraw(); virtual void OnWeaponPickup( C_BaseCombatWeapon *pWeapon ); virtual void CycleToNextWeapon( void ); virtual void CycleToPrevWeapon( void ); virtual C_BaseCombatWeapon *GetWeaponInSlot( int iSlot, int iSlotPos ); virtual void SelectWeaponSlot( int iSlot ); virtual C_BaseCombatWeapon *GetSelectedWeapon( void ) { return m_hSelectedWeapon; } virtual void OpenSelection( void ); virtual void HideSelection( void ); virtual void LevelInit(); protected: virtual void OnThink(); virtual void Paint(); virtual void ApplySchemeSettings(vgui::IScheme *pScheme); virtual bool IsWeaponSelectable() { if (IsInSelectionMode()) return true; return false; } virtual void SetWeaponSelected() { CBaseHudWeaponSelection::SetWeaponSelected(); switch( hud_fastswitch.GetInt() ) { case HUDTYPE_FASTSWITCH: case HUDTYPE_CAROUSEL: ActivateFastswitchWeaponDisplay( GetSelectedWeapon() ); break; case HUDTYPE_PLUS: ActivateWeaponHighlight( GetSelectedWeapon() ); break; default: // do nothing break; } } private: C_BaseCombatWeapon *FindNextWeaponInWeaponSelection(int iCurrentSlot, int iCurrentPosition); C_BaseCombatWeapon *FindPrevWeaponInWeaponSelection(int iCurrentSlot, int iCurrentPosition); void DrawLargeWeaponBox( C_BaseCombatWeapon *pWeapon, bool bSelected, int x, int y, int wide, int tall, Color color, float alpha, int number ); void ActivateFastswitchWeaponDisplay( C_BaseCombatWeapon *pWeapon ); void ActivateWeaponHighlight( C_BaseCombatWeapon *pWeapon ); float GetWeaponBoxAlpha( bool bSelected ); int GetLastPosInSlot( int iSlot ) const; void FastWeaponSwitch( int iWeaponSlot ); void PlusTypeFastWeaponSwitch( int iWeaponSlot ); virtual void SetSelectedWeapon( C_BaseCombatWeapon *pWeapon ) { m_hSelectedWeapon = pWeapon; } virtual void SetSelectedSlot( int slot ) { m_iSelectedSlot = slot; } void SetSelectedSlideDir( int dir ) { m_iSelectedSlideDir = dir; } void DrawBox(int x, int y, int wide, int tall, Color color, float normalizedAlpha, int number); CPanelAnimationVar( vgui::HFont, m_hNumberFont, "NumberFont", "HudSelectionNumbers" ); CPanelAnimationVar( vgui::HFont, m_hTextFont, "TextFont", "HudSelectionText" ); CPanelAnimationVar( float, m_flBlur, "Blur", "0" ); CPanelAnimationVarAliasType( float, m_flSmallBoxSize, "SmallBoxSize", "32", "proportional_float" ); CPanelAnimationVarAliasType( float, m_flLargeBoxWide, "LargeBoxWide", "108", "proportional_float" ); CPanelAnimationVarAliasType( float, m_flLargeBoxTall, "LargeBoxTall", "72", "proportional_float" ); CPanelAnimationVarAliasType( float, m_flMediumBoxWide, "MediumBoxWide", "75", "proportional_float" ); CPanelAnimationVarAliasType( float, m_flMediumBoxTall, "MediumBoxTall", "50", "proportional_float" ); CPanelAnimationVarAliasType( float, m_flBoxGap, "BoxGap", "12", "proportional_float" ); CPanelAnimationVarAliasType( float, m_flSelectionNumberXPos, "SelectionNumberXPos", "4", "proportional_float" ); CPanelAnimationVarAliasType( float, m_flSelectionNumberYPos, "SelectionNumberYPos", "4", "proportional_float" ); CPanelAnimationVarAliasType( float, m_flTextYPos, "TextYPos", "54", "proportional_float" ); CPanelAnimationVar( float, m_flAlphaOverride, "Alpha", "0" ); CPanelAnimationVar( float, m_flSelectionAlphaOverride, "SelectionAlpha", "0" ); CPanelAnimationVar( Color, m_TextColor, "TextColor", "SelectionTextFg" ); CPanelAnimationVar( Color, m_NumberColor, "NumberColor", "SelectionNumberFg" ); CPanelAnimationVar( Color, m_EmptyBoxColor, "EmptyBoxColor", "SelectionEmptyBoxBg" ); CPanelAnimationVar( Color, m_BoxColor, "BoxColor", "SelectionBoxBg" ); CPanelAnimationVar( Color, m_SelectedBoxColor, "SelectedBoxColor", "SelectionSelectedBoxBg" ); CPanelAnimationVar( Color, m_SelectedFgColor, "SelectedFgColor", "FgColor" ); CPanelAnimationVar( Color, m_BrightBoxColor, "SelectedFgColor", "BgColor" ); CPanelAnimationVar( float, m_flWeaponPickupGrowTime, "SelectionGrowTime", "0.1" ); CPanelAnimationVar( float, m_flTextScan, "TextScan", "1.0" ); bool m_bFadingOut; // fastswitch weapon display struct WeaponBox_t { int m_iSlot; int m_iSlotPos; }; CUtlVector<WeaponBox_t> m_WeaponBoxes; int m_iSelectedWeaponBox; int m_iSelectedSlideDir; int m_iSelectedBoxPosition; int m_iSelectedSlot; C_BaseCombatWeapon *m_pLastWeapon; CPanelAnimationVar( float, m_flHorizWeaponSelectOffsetPoint, "WeaponBoxOffset", "0" ); }; DECLARE_HUDELEMENT( CHudWeaponSelection ); using namespace vgui; //----------------------------------------------------------------------------- // Purpose: Constructor //----------------------------------------------------------------------------- CHudWeaponSelection::CHudWeaponSelection( const char *pElementName ) : CBaseHudWeaponSelection(pElementName), BaseClass(NULL, "HudWeaponSelection") { vgui::Panel *pParent = g_pClientMode->GetViewport(); SetParent( pParent ); m_bFadingOut = false; } //----------------------------------------------------------------------------- // Purpose: sets up display for showing weapon pickup //----------------------------------------------------------------------------- void CHudWeaponSelection::OnWeaponPickup( C_BaseCombatWeapon *pWeapon ) { // add to pickup history CHudHistoryResource *pHudHR = GET_HUDELEMENT( CHudHistoryResource ); if ( pHudHR ) { pHudHR->AddToHistory( pWeapon ); } } //----------------------------------------------------------------------------- // Purpose: updates animation status //----------------------------------------------------------------------------- void CHudWeaponSelection::OnThink( void ) { float flSelectionTimeout = SELECTION_TIMEOUT_THRESHOLD; float flSelectionFadeoutTime = SELECTION_FADEOUT_TIME; if ( hud_fastswitch.GetBool() ) { flSelectionTimeout = FASTSWITCH_DISPLAY_TIMEOUT; flSelectionFadeoutTime = FASTSWITCH_FADEOUT_TIME; } // Time out after awhile of inactivity if ( ( gpGlobals->curtime - m_flSelectionTime ) > flSelectionTimeout ) { if (!m_bFadingOut) { // start fading out g_pClientMode->GetViewportAnimationController()->StartAnimationSequence( "FadeOutWeaponSelectionMenu" ); m_bFadingOut = true; } else if ( gpGlobals->curtime - m_flSelectionTime > flSelectionTimeout + flSelectionFadeoutTime ) { // finished fade, close HideSelection(); } } else if (m_bFadingOut) { // stop us fading out, show the animation again g_pClientMode->GetViewportAnimationController()->StartAnimationSequence( "OpenWeaponSelectionMenu" ); m_bFadingOut = false; } } //----------------------------------------------------------------------------- // Purpose: returns true if the panel should draw //----------------------------------------------------------------------------- bool CHudWeaponSelection::ShouldDraw() { C_BasePlayer *pPlayer = C_BasePlayer::GetLocalPlayer(); if ( !pPlayer ) { if ( IsInSelectionMode() ) { HideSelection(); } return false; } bool bret = CBaseHudWeaponSelection::ShouldDraw(); if ( !bret ) return false; // draw weapon selection a little longer if in fastswitch so we can see what we've selected if ( hud_fastswitch.GetBool() && ( gpGlobals->curtime - m_flSelectionTime ) < (FASTSWITCH_DISPLAY_TIMEOUT + FASTSWITCH_FADEOUT_TIME) ) return true; return ( m_bSelectionVisible ) ? true : false; } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CHudWeaponSelection::LevelInit() { CHudElement::LevelInit(); m_iSelectedWeaponBox = -1; m_iSelectedSlideDir = 0; m_pLastWeapon = NULL; } //----------------------------------------------------------------------------- // Purpose: starts animating the center of the draw point to the newly selected weapon //----------------------------------------------------------------------------- void CHudWeaponSelection::ActivateFastswitchWeaponDisplay( C_BaseCombatWeapon *pSelectedWeapon ) { C_BasePlayer *pPlayer = C_BasePlayer::GetLocalPlayer(); if ( !pPlayer ) return; // make sure all our configuration data is read MakeReadyForUse(); m_WeaponBoxes.RemoveAll(); m_iSelectedWeaponBox = 0; // find out where our selected weapon is in the full list int cWeapons = 0; int iLastSelectedWeaponBox = -1; for ( int i = 0; i < MAX_WEAPON_SLOTS; i++ ) { for (int slotpos = 0; slotpos < MAX_WEAPON_POSITIONS; slotpos++) { C_BaseCombatWeapon *pWeapon = GetWeaponInSlot( i, slotpos ); if ( !pWeapon ) continue; WeaponBox_t box = { i, slotpos }; m_WeaponBoxes.AddToTail( box ); if ( pWeapon == pSelectedWeapon ) { m_iSelectedWeaponBox = cWeapons; } if ( pWeapon == m_pLastWeapon ) { iLastSelectedWeaponBox = cWeapons; } cWeapons++; } } if ( iLastSelectedWeaponBox == -1 ) { // unexpected failure, no last weapon to scroll from, default to snap behavior m_pLastWeapon = NULL; } // calculate where we would have to start drawing for this weapon to slide into center float flStart, flStop, flTime; if ( !m_pLastWeapon || m_iSelectedSlideDir == 0 || m_flHorizWeaponSelectOffsetPoint != 0 ) { // no previous weapon or weapon selected directly or selection during slide, snap to exact position m_pLastWeapon = pSelectedWeapon; flStart = flStop = flTime = 0; } else { // offset display for a scroll // causing selected weapon to slide into position // scroll direction based on user's "previous" or "next" selection int numIcons = 0; int start = iLastSelectedWeaponBox; for (int i=0; i<cWeapons; i++) { // count icons in direction of slide to destination if ( start == m_iSelectedWeaponBox ) break; if ( m_iSelectedSlideDir < 0 ) { start--; } else { start++; } // handle wraparound in either direction start = ( start + cWeapons ) % cWeapons; numIcons++; } flStart = numIcons * (m_flLargeBoxWide + m_flBoxGap); if ( m_iSelectedSlideDir < 0 ) flStart *= -1; flStop = 0; // shorten duration for scrolling when desired weapon is farther away // otherwise a large skip in the same duration causes the scroll to fly too fast flTime = numIcons * 0.20f; if ( numIcons > 1 ) flTime *= 0.5f; } m_flHorizWeaponSelectOffsetPoint = flStart; g_pClientMode->GetViewportAnimationController()->RunAnimationCommand( this, "WeaponBoxOffset", flStop, 0, flTime, AnimationController::INTERPOLATOR_LINEAR ); // start the highlight after the scroll completes m_flBlur = 7.f; g_pClientMode->GetViewportAnimationController()->RunAnimationCommand( this, "Blur", 0, flTime, 0.75f, AnimationController::INTERPOLATOR_DEACCEL ); } //----------------------------------------------------------------------------- // Purpose: starts animating the highlight for the selected weapon //----------------------------------------------------------------------------- void CHudWeaponSelection::ActivateWeaponHighlight( C_BaseCombatWeapon *pSelectedWeapon ) { C_BasePlayer *pPlayer = C_BasePlayer::GetLocalPlayer(); if ( !pPlayer ) return; // make sure all our configuration data is read MakeReadyForUse(); C_BaseCombatWeapon *pWeapon = GetWeaponInSlot( m_iSelectedSlot, m_iSelectedBoxPosition ); if ( !pWeapon ) return; // start the highlight after the scroll completes g_pClientMode->GetViewportAnimationController()->StartAnimationSequence( "WeaponHighlight" ); } //----------------------------------------------------------------------------- // Purpose: returns an (per frame animating) alpha value for different weapon boxes //----------------------------------------------------------------------------- float CHudWeaponSelection::GetWeaponBoxAlpha( bool bSelected ) { float alpha; if ( bSelected ) { alpha = m_flSelectionAlphaOverride; } else { alpha = m_flSelectionAlphaOverride * (m_flAlphaOverride / 255.0f); } return alpha; } //------------------------------------------------------------------------- // Purpose: draws the selection area //------------------------------------------------------------------------- void CHudWeaponSelection::Paint() { int width; int xpos; int ypos; if (!ShouldDraw()) return; C_BasePlayer *pPlayer = C_BasePlayer::GetLocalPlayer(); if ( !pPlayer ) return; // find and display our current selection C_BaseCombatWeapon *pSelectedWeapon = NULL; switch ( hud_fastswitch.GetInt() ) { case HUDTYPE_FASTSWITCH: case HUDTYPE_CAROUSEL: pSelectedWeapon = pPlayer->GetActiveWeapon(); break; default: pSelectedWeapon = GetSelectedWeapon(); break; } if ( !pSelectedWeapon ) return; bool bPushedViewport = false; if( hud_fastswitch.GetInt() == HUDTYPE_FASTSWITCH || hud_fastswitch.GetInt() == HUDTYPE_PLUS ) { CMatRenderContextPtr pRenderContext( materials ); if( pRenderContext->GetRenderTarget() ) { surface()->PushFullscreenViewport(); bPushedViewport = true; } } // interpolate the selected box size between the small box size and the large box size // interpolation has been removed since there is no weapon pickup animation anymore, so it's all at the largest size float percentageDone = 1.0f; //min(1.0f, (gpGlobals->curtime - m_flPickupStartTime) / m_flWeaponPickupGrowTime); int largeBoxWide = m_flSmallBoxSize + ((m_flLargeBoxWide - m_flSmallBoxSize) * percentageDone); int largeBoxTall = m_flSmallBoxSize + ((m_flLargeBoxTall - m_flSmallBoxSize) * percentageDone); Color selectedColor; for (int i = 0; i < 4; i++) { selectedColor[i] = m_BoxColor[i] + ((m_SelectedBoxColor[i] - m_BoxColor[i]) * percentageDone); } switch ( hud_fastswitch.GetInt() ) { case HUDTYPE_CAROUSEL: { // carousel style - flat line of items ypos = 0; if ( m_iSelectedWeaponBox == -1 || m_WeaponBoxes.Count() <= 1 ) { // nothing to do return; } else if ( m_WeaponBoxes.Count() < MAX_CAROUSEL_SLOTS ) { // draw the selected weapon as a 1 of n style width = (m_WeaponBoxes.Count()-1) * (m_flLargeBoxWide+m_flBoxGap) + m_flLargeBoxWide; xpos = (GetWide() - width)/2; for ( int i=0; i<m_WeaponBoxes.Count(); i++ ) { C_BaseCombatWeapon *pWeapon = GetWeaponInSlot( m_WeaponBoxes[i].m_iSlot, m_WeaponBoxes[i].m_iSlotPos ); if ( !pWeapon ) break; float alpha = GetWeaponBoxAlpha( i == m_iSelectedWeaponBox ); if ( i == m_iSelectedWeaponBox ) { // draw selected in highlighted style DrawLargeWeaponBox( pWeapon, true, xpos, ypos, m_flLargeBoxWide, m_flLargeBoxTall, selectedColor, alpha, -1 ); } else { DrawLargeWeaponBox( pWeapon, false, xpos, ypos, m_flLargeBoxWide, m_flLargeBoxTall / 1.5f, m_BoxColor, alpha, -1 ); } xpos += (m_flLargeBoxWide + m_flBoxGap); } } else { // draw the selected weapon in the center, as a continuous scrolling carosuel // draw at center the current selected and all items to its right xpos = GetWide()/2 + m_flHorizWeaponSelectOffsetPoint - largeBoxWide/2; int i = m_iSelectedWeaponBox; while ( 1 ) { C_BaseCombatWeapon *pWeapon = GetWeaponInSlot( m_WeaponBoxes[i].m_iSlot, m_WeaponBoxes[i].m_iSlotPos ); if ( !pWeapon ) break; float alpha; if ( i == m_iSelectedWeaponBox && !m_flHorizWeaponSelectOffsetPoint ) { // draw selected in highlighted style alpha = GetWeaponBoxAlpha( true ); DrawLargeWeaponBox( pWeapon, true, xpos, ypos, largeBoxWide, largeBoxTall, selectedColor, alpha, -1 ); } else { alpha = GetWeaponBoxAlpha( false ); DrawLargeWeaponBox( pWeapon, false, xpos, ypos, largeBoxWide, largeBoxTall / 1.5f, m_BoxColor, alpha, -1 ); } // advance until past edge xpos += (largeBoxWide + m_flBoxGap); if ( xpos >= GetWide() ) break; ++i; if ( i >= m_WeaponBoxes.Count() ) { // wraparound i = 0; } } // draw all items left of center xpos = GetWide()/2 + m_flHorizWeaponSelectOffsetPoint - (3*largeBoxWide/2 + m_flBoxGap); i = m_iSelectedWeaponBox - 1; while ( 1 ) { if ( i < 0 ) { // wraparound i = m_WeaponBoxes.Count() - 1; } C_BaseCombatWeapon *pWeapon = GetWeaponInSlot( m_WeaponBoxes[i].m_iSlot, m_WeaponBoxes[i].m_iSlotPos ); if ( !pWeapon ) break; float alpha; if ( i == m_iSelectedWeaponBox && !m_flHorizWeaponSelectOffsetPoint ) { // draw selected in highlighted style alpha = GetWeaponBoxAlpha( true ); DrawLargeWeaponBox( pWeapon, true, xpos, ypos, largeBoxWide, largeBoxTall, selectedColor, alpha, -1 ); } else { alpha = GetWeaponBoxAlpha( false ); DrawLargeWeaponBox( pWeapon, false, xpos, ypos, largeBoxWide, largeBoxTall / 1.5f, m_BoxColor, alpha, -1 ); } // retreat until past edge xpos -= (largeBoxWide + m_flBoxGap); if ( xpos + largeBoxWide <= 0 ) break; --i; } } } break; case HUDTYPE_PLUS: { float fCenterX, fCenterY; bool bBehindCamera = false; CHudCrosshair::GetDrawPosition( &fCenterX, &fCenterY, &bBehindCamera ); // if the crosshair is behind the camera, don't draw it if( bBehindCamera ) return; // bucket style int screenCenterX = (int) fCenterX; int screenCenterY = (int) fCenterY - 15; // Height isn't quite screen height, so adjust for center alignment // Modifiers for the four directions. Used to change the x and y offsets // of each box based on which bucket we're drawing. Bucket directions are // 0 = UP, 1 = RIGHT, 2 = DOWN, 3 = LEFT int xModifiers[] = { 0, 1, 0, -1, -1, 1 }; int yModifiers[] = { -1, 0, 1, 0, 1, 1 }; // Draw the four buckets for ( int i = 0; i < MAX_WEAPON_SLOTS; ++i ) { // Set the top left corner so the first box would be centered in the screen. int xPos = screenCenterX -( m_flMediumBoxWide / 2 ); int yPos = screenCenterY -( m_flMediumBoxTall / 2 ); // Find out how many positions to draw - an empty position should still // be drawn if there is an active weapon in any slots past it. int lastSlotPos = -1; for ( int slotPos = 0; slotPos < MAX_WEAPON_POSITIONS; ++slotPos ) { C_BaseCombatWeapon *pWeapon = GetWeaponInSlot( i, slotPos ); if ( pWeapon ) { lastSlotPos = slotPos; } } // Draw the weapons in this bucket for ( int slotPos = 0; slotPos <= lastSlotPos; ++slotPos ) { // Offset the box position xPos += ( m_flMediumBoxWide + 5 ) * xModifiers[ i ]; yPos += ( m_flMediumBoxTall + 5 ) * yModifiers[ i ]; int boxWide = m_flMediumBoxWide; int boxTall = m_flMediumBoxTall; int x = xPos; int y = yPos; C_BaseCombatWeapon *pWeapon = GetWeaponInSlot( i, slotPos ); bool selectedWeapon = false; if ( i == m_iSelectedSlot && slotPos == m_iSelectedBoxPosition ) { // This is a bit of a misnomer... we really are asking "Is this the selected slot"? selectedWeapon = true; } // Draw the box with the appropriate icon DrawLargeWeaponBox( pWeapon, selectedWeapon, x, y, boxWide, boxTall, selectedWeapon ? selectedColor : m_BoxColor, GetWeaponBoxAlpha( selectedWeapon ), -1 ); } } } break; case HUDTYPE_BUCKETS: { // bucket style width = (MAX_WEAPON_SLOTS - 1) * (m_flSmallBoxSize + m_flBoxGap) + largeBoxWide; xpos = (GetWide() - width) / 2; ypos = 0; int iActiveSlot = (pSelectedWeapon ? pSelectedWeapon->GetSlot() : -1); // draw the bucket set // iterate over all the weapon slots for ( int i = 0; i < MAX_WEAPON_SLOTS; i++ ) { if ( i == iActiveSlot ) { bool bDrawBucketNumber = true; int iLastPos = GetLastPosInSlot( i ); for (int slotpos = 0; slotpos <= iLastPos; slotpos++) { C_BaseCombatWeapon *pWeapon = GetWeaponInSlot( i, slotpos ); if ( !pWeapon ) { if ( !hud_showemptyweaponslots.GetBool() ) continue; DrawBox( xpos, ypos, largeBoxWide, largeBoxTall, m_EmptyBoxColor, m_flAlphaOverride, bDrawBucketNumber ? i + 1 : -1 ); } else { bool bSelected = (pWeapon == pSelectedWeapon); DrawLargeWeaponBox( pWeapon, bSelected, xpos, ypos, largeBoxWide, largeBoxTall, bSelected ? selectedColor : m_BoxColor, GetWeaponBoxAlpha( bSelected ), bDrawBucketNumber ? i + 1 : -1 ); } // move down to the next bucket ypos += (largeBoxTall + m_flBoxGap); bDrawBucketNumber = false; } xpos += largeBoxWide; } else { // check to see if there is a weapons in this bucket if ( GetFirstPos( i ) ) { // draw has weapon in slot DrawBox(xpos, ypos, m_flSmallBoxSize, m_flSmallBoxSize, m_BoxColor, m_flAlphaOverride, i + 1); } else { // draw empty slot DrawBox(xpos, ypos, m_flSmallBoxSize, m_flSmallBoxSize, m_EmptyBoxColor, m_flAlphaOverride, -1); } xpos += m_flSmallBoxSize; } // reset position ypos = 0; xpos += m_flBoxGap; } } break; default: { // do nothing } break; } if( bPushedViewport ) { surface()->PopFullscreenViewport(); } } //----------------------------------------------------------------------------- // Purpose: draws a single weapon selection box //----------------------------------------------------------------------------- void CHudWeaponSelection::DrawLargeWeaponBox( C_BaseCombatWeapon *pWeapon, bool bSelected, int xpos, int ypos, int boxWide, int boxTall, Color selectedColor, float alpha, int number ) { Color col = bSelected ? m_SelectedFgColor : GetFgColor(); switch ( hud_fastswitch.GetInt() ) { case HUDTYPE_BUCKETS: { // draw box for selected weapon DrawBox( xpos, ypos, boxWide, boxTall, selectedColor, alpha, number ); // draw icon col[3] *= (alpha / 255.0f); if ( pWeapon->GetSpriteActive() ) { // find the center of the box to draw in int iconWidth = pWeapon->GetSpriteActive()->Width(); int iconHeight = pWeapon->GetSpriteActive()->Height(); int x_offs = (boxWide - iconWidth) / 2; int y_offs; if ( bSelected && hud_fastswitch.GetInt() != 0 ) { // place the icon aligned with the non-selected version y_offs = (boxTall / 1.5f - iconHeight) / 2; } else { y_offs = (boxTall - iconHeight) / 2; } if (!pWeapon->CanBeSelected()) { // unselectable weapon, display as such col = Color(255, 0, 0, col[3]); } else if (bSelected) { // currently selected weapon, display brighter col[3] = alpha; // draw an active version over the top pWeapon->GetSpriteActive()->DrawSelf( xpos + x_offs, ypos + y_offs, col ); } // draw the inactive version pWeapon->GetSpriteInactive()->DrawSelf( xpos + x_offs, ypos + y_offs, col ); } } break; case HUDTYPE_PLUS: case HUDTYPE_CAROUSEL: { if ( !pWeapon ) { // draw red box for an empty bubble if( bSelected ) { selectedColor.SetColor( 255, 0, 0, 40 ); } DrawBox( xpos, ypos, boxWide, boxTall, selectedColor, alpha, number ); return; } else { // draw box for selected weapon DrawBox( xpos, ypos, boxWide, boxTall, selectedColor, alpha, number ); } int iconWidth; int iconHeight; int x_offs; int y_offs; // draw icon col[3] *= (alpha / 255.0f); if ( pWeapon->GetSpriteInactive() ) { iconWidth = pWeapon->GetSpriteInactive()->Width(); iconHeight = pWeapon->GetSpriteInactive()->Height(); x_offs = (boxWide - iconWidth) / 2; if ( bSelected && HUDTYPE_CAROUSEL == hud_fastswitch.GetInt() ) { // place the icon aligned with the non-selected version y_offs = (boxTall/1.5f - iconHeight) / 2; } else { y_offs = (boxTall - iconHeight) / 2; } if ( !pWeapon->CanBeSelected() ) { // unselectable weapon, display as such col = Color(255, 0, 0, col[3]); } // draw the inactive version pWeapon->GetSpriteInactive()->DrawSelf( xpos + x_offs, ypos + y_offs, iconWidth, iconHeight, col ); } if ( bSelected && pWeapon->GetSpriteActive() ) { // find the center of the box to draw in iconWidth = pWeapon->GetSpriteActive()->Width(); iconHeight = pWeapon->GetSpriteActive()->Height(); x_offs = (boxWide - iconWidth) / 2; if ( HUDTYPE_CAROUSEL == hud_fastswitch.GetInt() ) { // place the icon aligned with the non-selected version y_offs = (boxTall/1.5f - iconHeight) / 2; } else { y_offs = (boxTall - iconHeight) / 2; } col[3] = 255; for (float fl = m_flBlur; fl > 0.0f; fl -= 1.0f) { if (fl >= 1.0f) { pWeapon->GetSpriteActive()->DrawSelf( xpos + x_offs, ypos + y_offs, col ); } else { // draw a percentage of the last one col[3] *= fl; pWeapon->GetSpriteActive()->DrawSelf( xpos + x_offs, ypos + y_offs, col ); } } } } break; default: { // do nothing } break; } if ( HUDTYPE_PLUS == hud_fastswitch.GetInt() ) { // No text in plus bucket method return; } // draw text col = m_TextColor; const FileWeaponInfo_t &weaponInfo = pWeapon->GetWpnData(); if ( bSelected ) { wchar_t text[128]; wchar_t *tempString = g_pVGuiLocalize->Find(weaponInfo.szPrintName); // setup our localized string if ( tempString ) { #ifdef WIN32 _snwprintf(text, sizeof(text)/sizeof(wchar_t) - 1, L"%s", tempString); #else _snwprintf(text, sizeof(text)/sizeof(wchar_t) - 1, L"%S", tempString); #endif text[sizeof(text)/sizeof(wchar_t) - 1] = 0; } else { // string wasn't found by g_pVGuiLocalize->Find() g_pVGuiLocalize->ConvertANSIToUnicode(weaponInfo.szPrintName, text, sizeof(text)); } surface()->DrawSetTextColor( col ); surface()->DrawSetTextFont( m_hTextFont ); // count the position int slen = 0, charCount = 0, maxslen = 0; int firstslen = 0; { for (wchar_t *pch = text; *pch != 0; pch++) { if (*pch == '\n') { // newline character, drop to the next line if (slen > maxslen) { maxslen = slen; } if (!firstslen) { firstslen = slen; } slen = 0; } else if (*pch == '\r') { // do nothing } else { slen += surface()->GetCharacterWidth( m_hTextFont, *pch ); charCount++; } } } if (slen > maxslen) { maxslen = slen; } if (!firstslen) { firstslen = maxslen; } int tx = xpos + ((m_flLargeBoxWide - firstslen) / 2); int ty = ypos + (int)m_flTextYPos; surface()->DrawSetTextPos( tx, ty ); // adjust the charCount by the scan amount charCount *= m_flTextScan; for (wchar_t *pch = text; charCount > 0; pch++) { if (*pch == '\n') { // newline character, move to the next line surface()->DrawSetTextPos( xpos + ((boxWide - slen) / 2), ty + (surface()->GetFontTall(m_hTextFont) * 1.1f)); } else if (*pch == '\r') { // do nothing } else { surface()->DrawUnicodeChar(*pch); charCount--; } } } } //----------------------------------------------------------------------------- // Purpose: draws a selection box //----------------------------------------------------------------------------- void CHudWeaponSelection::DrawBox(int x, int y, int wide, int tall, Color color, float normalizedAlpha, int number) { BaseClass::DrawBox( x, y, wide, tall, color, normalizedAlpha / 255.0f ); // draw the number if (number >= 0) { Color numberColor = m_NumberColor; numberColor[3] *= normalizedAlpha / 255.0f; surface()->DrawSetTextColor(numberColor); surface()->DrawSetTextFont(m_hNumberFont); wchar_t wch = '0' + number; surface()->DrawSetTextPos(x + m_flSelectionNumberXPos, y + m_flSelectionNumberYPos); surface()->DrawUnicodeChar(wch); } } //----------------------------------------------------------------------------- // Purpose: hud scheme settings //----------------------------------------------------------------------------- void CHudWeaponSelection::ApplySchemeSettings(vgui::IScheme *pScheme) { BaseClass::ApplySchemeSettings(pScheme); SetPaintBackgroundEnabled(false); // set our size int screenWide, screenTall; int x, y; GetPos(x, y); GetHudSize(screenWide, screenTall); if ( hud_fastswitch.GetInt() == HUDTYPE_CAROUSEL ) { // need bounds to be exact width for proper clipping during scroll int width = MAX_CAROUSEL_SLOTS*m_flLargeBoxWide + (MAX_CAROUSEL_SLOTS-1)*m_flBoxGap; SetBounds( (screenWide-width)/2, y, width, screenTall - y); } else { SetBounds( x, y, screenWide - x, screenTall - y ); } } //----------------------------------------------------------------------------- // Purpose: Opens weapon selection control //----------------------------------------------------------------------------- void CHudWeaponSelection::OpenSelection( void ) { Assert(!IsInSelectionMode()); CBaseHudWeaponSelection::OpenSelection(); g_pClientMode->GetViewportAnimationController()->StartAnimationSequence("OpenWeaponSelectionMenu"); m_iSelectedBoxPosition = 0; m_iSelectedSlot = -1; } //----------------------------------------------------------------------------- // Purpose: Closes weapon selection control immediately //----------------------------------------------------------------------------- void CHudWeaponSelection::HideSelection( void ) { CBaseHudWeaponSelection::HideSelection(); g_pClientMode->GetViewportAnimationController()->StartAnimationSequence("CloseWeaponSelectionMenu"); m_bFadingOut = false; } //----------------------------------------------------------------------------- // Purpose: Returns the next available weapon item in the weapon selection //----------------------------------------------------------------------------- C_BaseCombatWeapon *CHudWeaponSelection::FindNextWeaponInWeaponSelection(int iCurrentSlot, int iCurrentPosition) { C_BasePlayer *pPlayer = C_BasePlayer::GetLocalPlayer(); if ( !pPlayer ) return NULL; C_BaseCombatWeapon *pNextWeapon = NULL; // search all the weapons looking for the closest next int iLowestNextSlot = MAX_WEAPON_SLOTS; int iLowestNextPosition = MAX_WEAPON_POSITIONS; for ( int i = 0; i < MAX_WEAPONS; i++ ) { C_BaseCombatWeapon *pWeapon = pPlayer->GetWeapon(i); if ( !pWeapon ) continue; if ( CanBeSelectedInHUD( pWeapon ) ) { int weaponSlot = pWeapon->GetSlot(), weaponPosition = pWeapon->GetPosition(); // see if this weapon is further ahead in the selection list if ( weaponSlot > iCurrentSlot || (weaponSlot == iCurrentSlot && weaponPosition > iCurrentPosition) ) { // see if this weapon is closer than the current lowest if ( weaponSlot < iLowestNextSlot || (weaponSlot == iLowestNextSlot && weaponPosition < iLowestNextPosition) ) { iLowestNextSlot = weaponSlot; iLowestNextPosition = weaponPosition; pNextWeapon = pWeapon; } } } } return pNextWeapon; } //----------------------------------------------------------------------------- // Purpose: Returns the prior available weapon item in the weapon selection //----------------------------------------------------------------------------- C_BaseCombatWeapon *CHudWeaponSelection::FindPrevWeaponInWeaponSelection(int iCurrentSlot, int iCurrentPosition) { C_BasePlayer *pPlayer = C_BasePlayer::GetLocalPlayer(); if ( !pPlayer ) return NULL; C_BaseCombatWeapon *pPrevWeapon = NULL; // search all the weapons looking for the closest next int iLowestPrevSlot = -1; int iLowestPrevPosition = -1; for ( int i = 0; i < MAX_WEAPONS; i++ ) { C_BaseCombatWeapon *pWeapon = pPlayer->GetWeapon(i); if ( !pWeapon ) continue; if ( CanBeSelectedInHUD( pWeapon ) ) { int weaponSlot = pWeapon->GetSlot(), weaponPosition = pWeapon->GetPosition(); // see if this weapon is further ahead in the selection list if ( weaponSlot < iCurrentSlot || (weaponSlot == iCurrentSlot && weaponPosition < iCurrentPosition) ) { // see if this weapon is closer than the current lowest if ( weaponSlot > iLowestPrevSlot || (weaponSlot == iLowestPrevSlot && weaponPosition > iLowestPrevPosition) ) { iLowestPrevSlot = weaponSlot; iLowestPrevPosition = weaponPosition; pPrevWeapon = pWeapon; } } } } return pPrevWeapon; } //----------------------------------------------------------------------------- // Purpose: Moves the selection to the next item in the menu //----------------------------------------------------------------------------- void CHudWeaponSelection::CycleToNextWeapon( void ) { // Get the local player. C_BasePlayer *pPlayer = C_BasePlayer::GetLocalPlayer(); if ( !pPlayer ) return; m_pLastWeapon = pPlayer->GetActiveWeapon(); C_BaseCombatWeapon *pNextWeapon = NULL; if ( IsInSelectionMode() ) { // find the next selection spot C_BaseCombatWeapon *pWeapon = GetSelectedWeapon(); if ( !pWeapon ) return; pNextWeapon = FindNextWeaponInWeaponSelection( pWeapon->GetSlot(), pWeapon->GetPosition() ); } else { // open selection at the current place pNextWeapon = pPlayer->GetActiveWeapon(); if ( pNextWeapon ) { pNextWeapon = FindNextWeaponInWeaponSelection( pNextWeapon->GetSlot(), pNextWeapon->GetPosition() ); } } if ( !pNextWeapon ) { // wrap around back to start pNextWeapon = FindNextWeaponInWeaponSelection(-1, -1); } if ( pNextWeapon ) { SetSelectedWeapon( pNextWeapon ); SetSelectedSlideDir( 1 ); if ( !IsInSelectionMode() ) { OpenSelection(); } // Play the "cycle to next weapon" sound pPlayer->EmitSound( "Player.WeaponSelectionMoveSlot" ); } } //----------------------------------------------------------------------------- // Purpose: Moves the selection to the previous item in the menu //----------------------------------------------------------------------------- void CHudWeaponSelection::CycleToPrevWeapon( void ) { // Get the local player. C_BasePlayer *pPlayer = C_BasePlayer::GetLocalPlayer(); if ( !pPlayer ) return; m_pLastWeapon = pPlayer->GetActiveWeapon(); C_BaseCombatWeapon *pNextWeapon = NULL; if ( IsInSelectionMode() ) { // find the next selection spot C_BaseCombatWeapon *pWeapon = GetSelectedWeapon(); if ( !pWeapon ) return; pNextWeapon = FindPrevWeaponInWeaponSelection( pWeapon->GetSlot(), pWeapon->GetPosition() ); } else { // open selection at the current place pNextWeapon = pPlayer->GetActiveWeapon(); if ( pNextWeapon ) { pNextWeapon = FindPrevWeaponInWeaponSelection( pNextWeapon->GetSlot(), pNextWeapon->GetPosition() ); } } if ( !pNextWeapon ) { // wrap around back to end of weapon list pNextWeapon = FindPrevWeaponInWeaponSelection(MAX_WEAPON_SLOTS, MAX_WEAPON_POSITIONS); } if ( pNextWeapon ) { SetSelectedWeapon( pNextWeapon ); SetSelectedSlideDir( -1 ); if ( !IsInSelectionMode() ) { OpenSelection(); } // Play the "cycle to next weapon" sound pPlayer->EmitSound( "Player.WeaponSelectionMoveSlot" ); } } //----------------------------------------------------------------------------- // Purpose: returns the # of the last weapon in the specified slot //----------------------------------------------------------------------------- int CHudWeaponSelection::GetLastPosInSlot( int iSlot ) const { C_BasePlayer *player = C_BasePlayer::GetLocalPlayer(); int iMaxSlotPos; if ( !player ) return -1; iMaxSlotPos = -1; for ( int i = 0; i < MAX_WEAPONS; i++ ) { C_BaseCombatWeapon *pWeapon = player->GetWeapon(i); if ( pWeapon == NULL ) continue; if ( pWeapon->GetSlot() == iSlot && pWeapon->GetPosition() > iMaxSlotPos ) iMaxSlotPos = pWeapon->GetPosition(); } return iMaxSlotPos; } //----------------------------------------------------------------------------- // Purpose: returns the weapon in the specified slot //----------------------------------------------------------------------------- C_BaseCombatWeapon *CHudWeaponSelection::GetWeaponInSlot( int iSlot, int iSlotPos ) { C_BasePlayer *player = C_BasePlayer::GetLocalPlayer(); if ( !player ) return NULL; for ( int i = 0; i < MAX_WEAPONS; i++ ) { C_BaseCombatWeapon *pWeapon = player->GetWeapon(i); if ( pWeapon == NULL ) continue; if ( pWeapon->GetSlot() == iSlot && pWeapon->GetPosition() == iSlotPos ) return pWeapon; } return NULL; } //----------------------------------------------------------------------------- // Purpose: Opens the next weapon in the slot //----------------------------------------------------------------------------- void CHudWeaponSelection::FastWeaponSwitch( int iWeaponSlot ) { // get the slot the player's weapon is in C_BasePlayer *pPlayer = C_BasePlayer::GetLocalPlayer(); if ( !pPlayer ) return; m_pLastWeapon = NULL; // see where we should start selection int iPosition = -1; C_BaseCombatWeapon *pActiveWeapon = pPlayer->GetActiveWeapon(); if ( pActiveWeapon && pActiveWeapon->GetSlot() == iWeaponSlot ) { // start after this weapon iPosition = pActiveWeapon->GetPosition(); } C_BaseCombatWeapon *pNextWeapon = NULL; // search for the weapon after the current one pNextWeapon = FindNextWeaponInWeaponSelection(iWeaponSlot, iPosition); // make sure it's in the same bucket if ( !pNextWeapon || pNextWeapon->GetSlot() != iWeaponSlot ) { // just look for any weapon in this slot pNextWeapon = FindNextWeaponInWeaponSelection(iWeaponSlot, -1); } // see if we found a weapon that's different from the current and in the selected slot if ( pNextWeapon && pNextWeapon != pActiveWeapon && pNextWeapon->GetSlot() == iWeaponSlot ) { // select the new weapon ::input->MakeWeaponSelection( pNextWeapon ); } else if ( pNextWeapon != pActiveWeapon ) { // error sound pPlayer->EmitSound( "Player.DenyWeaponSelection" ); } if ( HUDTYPE_CAROUSEL != hud_fastswitch.GetInt() ) { // kill any fastswitch display m_flSelectionTime = 0.0f; } } //----------------------------------------------------------------------------- // Purpose: Opens the next weapon in the slot //----------------------------------------------------------------------------- void CHudWeaponSelection::PlusTypeFastWeaponSwitch( int iWeaponSlot ) { // get the slot the player's weapon is in C_BasePlayer *pPlayer = C_BasePlayer::GetLocalPlayer(); if ( !pPlayer ) return; m_pLastWeapon = NULL; int newSlot = m_iSelectedSlot; // Changing slot number does not necessarily mean we need to change the slot - the player could be // scrolling through the same slot but in the opposite direction. Slot pairs are 0,2 and 1,3 - so // compare the 0 bits to see if we're within a pair. Otherwise, reset the box to the zero position. if ( -1 == m_iSelectedSlot || ( ( m_iSelectedSlot ^ iWeaponSlot ) & 1 ) ) { // Changing vertical/horizontal direction. Reset the selected box position to zero. m_iSelectedBoxPosition = 0; m_iSelectedSlot = iWeaponSlot; } else { // Still in the same horizontal/vertical direction. Determine which way we're moving in the slot. int increment = 1; if ( m_iSelectedSlot != iWeaponSlot ) { // Decrementing within the slot. If we're at the zero position in this slot, // jump to the zero position of the opposite slot. This also counts as our increment. increment = -1; if ( 0 == m_iSelectedBoxPosition ) { newSlot = ( m_iSelectedSlot + 2 ) % 4; increment = 0; } } // Find out of the box position is at the end of the slot int lastSlotPos = -1; for ( int slotPos = 0; slotPos < MAX_WEAPON_POSITIONS; ++slotPos ) { C_BaseCombatWeapon *pWeapon = GetWeaponInSlot( newSlot, slotPos ); if ( pWeapon ) { lastSlotPos = slotPos; } } // Increment/Decrement the selected box position if ( m_iSelectedBoxPosition + increment <= lastSlotPos ) { m_iSelectedBoxPosition += increment; m_iSelectedSlot = newSlot; } else { // error sound pPlayer->EmitSound( "Player.DenyWeaponSelection" ); return; } } // Select the weapon in this position bool bWeaponSelected = false; C_BaseCombatWeapon *pActiveWeapon = pPlayer->GetActiveWeapon(); C_BaseCombatWeapon *pWeapon = GetWeaponInSlot( m_iSelectedSlot, m_iSelectedBoxPosition ); if ( pWeapon ) { if ( pWeapon != pActiveWeapon ) { // Select the new weapon ::input->MakeWeaponSelection( pWeapon ); SetSelectedWeapon( pWeapon ); bWeaponSelected = true; } } if ( !bWeaponSelected ) { // Still need to set this to make hud display appear SetSelectedWeapon( pPlayer->GetActiveWeapon() ); } } //----------------------------------------------------------------------------- // Purpose: Moves selection to the specified slot //----------------------------------------------------------------------------- void CHudWeaponSelection::SelectWeaponSlot( int iSlot ) { // iSlot is one higher than it should be, since it's the number key, not the 0-based index into the weapons --iSlot; // Get the local player. C_BasePlayer *pPlayer = C_BasePlayer::GetLocalPlayer(); if ( !pPlayer ) return; // Don't try and read past our possible number of slots if ( iSlot >= MAX_WEAPON_SLOTS ) return; // Make sure the player's allowed to switch weapons if ( pPlayer->IsAllowedToSwitchWeapons() == false ) return; switch( hud_fastswitch.GetInt() ) { case HUDTYPE_FASTSWITCH: case HUDTYPE_CAROUSEL: { FastWeaponSwitch( iSlot ); return; } case HUDTYPE_PLUS: { if ( !IsInSelectionMode() ) { // open the weapon selection OpenSelection(); } PlusTypeFastWeaponSwitch( iSlot ); ActivateWeaponHighlight( GetSelectedWeapon() ); } break; case HUDTYPE_BUCKETS: { int slotPos = 0; C_BaseCombatWeapon *pActiveWeapon = GetSelectedWeapon(); // start later in the list if ( IsInSelectionMode() && pActiveWeapon && pActiveWeapon->GetSlot() == iSlot ) { slotPos = pActiveWeapon->GetPosition() + 1; } // find the weapon in this slot pActiveWeapon = GetNextActivePos( iSlot, slotPos ); if ( !pActiveWeapon ) { pActiveWeapon = GetNextActivePos( iSlot, 0 ); } if ( pActiveWeapon != NULL ) { if ( !IsInSelectionMode() ) { // open the weapon selection OpenSelection(); } // Mark the change SetSelectedWeapon( pActiveWeapon ); SetSelectedSlideDir( 0 ); } } default: { // do nothing } break; } pPlayer->EmitSound( "Player.WeaponSelectionMoveSlot" ); }