source-engine/game/client/tf/vgui/quest_notification_panel.cpp

534 lines
16 KiB
C++
Raw Normal View History

2020-04-22 12:56:21 -04:00
//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
//=============================================================================//
#include "cbase.h"
#include "quest_notification_panel.h"
#include "vgui/ISurface.h"
#include "ienginevgui.h"
#include "hudelement.h"
#include "iclientmode.h"
#include "basemodel_panel.h"
#include "tf_item_inventory.h"
#include "quest_log_panel.h"
#include "econ_controls.h"
#include "c_tf_player.h"
#include <vgui_controls/AnimationController.h>
#include "engine/IEngineSound.h"
#include "econ_item_system.h"
#include "tf_hud_item_progress_tracker.h"
#include "tf_spectatorgui.h"
#include "econ_quests.h"
// memdbgon must be the last include file in a .cpp file!!!
#include <tier0/memdbgon.h>
ConVar tf_quest_notification_line_delay( "tf_quest_notification_line_delay", "1.2", FCVAR_ARCHIVE );
extern ISoundEmitterSystemBase *soundemitterbase;
CQuestNotificationPanel *g_pQuestNotificationPanel = NULL;
DECLARE_HUDELEMENT( CQuestNotificationPanel );
CQuestNotification::CQuestNotification( CEconItem *pItem )
: m_hItem( pItem )
{}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
float CQuestNotification::Present( CQuestNotificationPanel* pNotificationPanel )
{
m_timerDialog.Start( tf_quest_notification_line_delay.GetFloat() );
return 0.f;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
CQuestNotification_Speaking::CQuestNotification_Speaking( CEconItem *pItem )
: CQuestNotification( pItem )
{
m_pszSoundToSpeak = NULL;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
float CQuestNotification_Speaking::Present( CQuestNotificationPanel* pNotificationPanel )
{
CQuestNotification::Present( pNotificationPanel );
if ( m_hItem )
{
C_BasePlayer *pPlayer = C_BasePlayer::GetLocalPlayer();
if ( !pPlayer )
return 0.f;
CTFPlayer* pTFPlayer = ToTFPlayer( pPlayer );
if ( !pTFPlayer )
return 0.f;
const GameItemDefinition_t *pItemDef = m_hItem->GetItemDefinition();
// Get our quest theme
const CQuestThemeDefinition *pTheme = pItemDef->GetQuestDef()->GetQuestTheme();
if ( pTheme )
{
// Get the sound we need to speak
m_pszSoundToSpeak = GetSoundEntry( pTheme, pTFPlayer->GetPlayerClass()->GetClassIndex() );
float flPresentTime = 0.f;
if ( m_pszSoundToSpeak )
{
flPresentTime = enginesound->GetSoundDuration( m_pszSoundToSpeak ) + m_timerDialog.GetCountdownDuration() + 1.f;
m_timerShow.Start( enginesound->GetSoundDuration( m_pszSoundToSpeak ) + m_timerDialog.GetCountdownDuration() + 1.f );
}
return flPresentTime;
}
}
return 0.f;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CQuestNotification_Speaking::Update( CQuestNotificationPanel* pNotificationPanel )
{
if ( m_timerDialog.IsElapsed() && m_timerDialog.HasStarted() && m_hItem )
{
m_timerDialog.Invalidate();
// Play it!
if ( m_pszSoundToSpeak )
{
vgui::surface()->PlaySound( m_pszSoundToSpeak );
}
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
bool CQuestNotification_Speaking::IsDone() const
{
return m_timerShow.IsElapsed() && m_timerShow.HasStarted();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
const char *CQuestNotification_NewQuest::GetSoundEntry( const CQuestThemeDefinition* pTheme, int nClassIndex )
{
return pTheme->GetGiveSoundForClass( nClassIndex );
}
bool CQuestNotification_NewQuest::ShouldPresent() const
{
C_BasePlayer *pPlayer = C_BasePlayer::GetLocalPlayer();
if ( !pPlayer )
return false;
CTFPlayer* pTFPlayer = ToTFPlayer( pPlayer );
if ( !pTFPlayer )
return false;
IViewPortPanel* pSpecGuiPanel = gViewPortInterface->FindPanelByName( PANEL_SPECGUI );
if ( !pTFPlayer->IsAlive() )
{
if ( !pSpecGuiPanel || !pSpecGuiPanel->IsVisible() )
return false;
}
else
{
// Local player is in a spawn room
if ( pTFPlayer->m_Shared.GetRespawnTouchCount() <= 0 )
return false;
}
return true;
}
CQuestNotification_CompletedQuest::CQuestNotification_CompletedQuest( CEconItem *pItem )
: CQuestNotification_Speaking( pItem )
{
const char *pszSoundName = UTIL_GetRandomSoundFromEntry( "Quest.StatusTickComplete" );
m_PresentTimer.Start( enginesound->GetSoundDuration( pszSoundName ) - 2.f );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
const char *CQuestNotification_CompletedQuest::GetSoundEntry( const CQuestThemeDefinition* pTheme, int nClassIndex )
{
return pTheme->GetCompleteSoundForClass( nClassIndex );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
bool CQuestNotification_CompletedQuest::ShouldPresent() const
{
return m_PresentTimer.IsElapsed() && m_PresentTimer.HasStarted();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
const char *CQuestNotification_FullyCompletedQuest::GetSoundEntry( const CQuestThemeDefinition* pTheme, int nClassIndex )
{
return pTheme->GetFullyCompleteSoundForClass( nClassIndex );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
CQuestNotificationPanel::CQuestNotificationPanel( const char *pszElementName )
: CHudElement( pszElementName )
, EditablePanel( NULL, "QuestNotificationPanel" )
, m_flTimeSinceLastShown( 0.f )
, m_bIsPresenting( false )
, m_mapNotifiedItemIDs( DefLessFunc( itemid_t ) )
, m_bInitialized( false )
, m_pMainContainer( NULL )
{
Panel *pParent = g_pClientMode->GetViewport();
SetParent( pParent );
g_pQuestNotificationPanel = this;
ListenForGameEvent( "player_death" );
ListenForGameEvent( "inventory_updated" );
ListenForGameEvent( "player_initial_spawn" );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
CQuestNotificationPanel::~CQuestNotificationPanel()
{}
void CQuestNotificationPanel::ApplySchemeSettings( IScheme *pScheme )
{
BaseClass::ApplySchemeSettings( pScheme );
// Default, load pauling
LoadControlSettings( "Resource/UI/econ/QuestNotificationPanel_Pauling_standard.res" );
m_pMainContainer = FindControl< EditablePanel >( "MainContainer", true );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CQuestNotificationPanel::PerformLayout()
{
BaseClass::PerformLayout();
CExLabel* pNewQuestLabel = FindControl< CExLabel >( "NewQuestText", true );
if ( pNewQuestLabel )
{
const wchar_t *pszText = NULL;
const char *pszTextKey = "#QuestNotification_Accept";
if ( pszTextKey )
{
pszText = g_pVGuiLocalize->Find( pszTextKey );
}
if ( pszText )
{
wchar_t wzFinal[512] = L"";
UTIL_ReplaceKeyBindings( pszText, 0, wzFinal, sizeof( wzFinal ) );
pNewQuestLabel->SetText( wzFinal );
}
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CQuestNotificationPanel::FireGameEvent( IGameEvent * event )
{
const char *pszName = event->GetName();
if ( FStrEq( pszName, "inventory_updated" ) || FStrEq( pszName, "player_death" ) )
{
CheckForNotificationOpportunities();
}
else if ( FStrEq( pszName, "player_initial_spawn" ) )
{
CTFPlayer *pNewPlayer = ToTFPlayer( UTIL_PlayerByIndex( event->GetInt( "index" ) ) );
if ( pNewPlayer == C_BasePlayer::GetLocalPlayer() )
{
// Reset every round
m_mapNotifiedItemIDs.Purge();
m_vecNotifications.PurgeAndDeleteElements();
m_timerNotificationCooldown.Start( 0 );
m_bInitialized = false;
}
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CQuestNotificationPanel::Reset()
{
CheckForNotificationOpportunities();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CQuestNotificationPanel::CheckForNotificationOpportunities()
{
// Suppress making new notifications while in competitive play
if ( TFGameRules() && TFGameRules()->IsCompetitiveMode() )
return;
FOR_EACH_VEC_BACK( m_vecNotifications, i )
{
// Clean up old entires for items that are now gone
if ( m_vecNotifications[i]->GetItemHandle() == NULL )
{
delete m_vecNotifications[i];
m_vecNotifications.Remove( i );
}
}
CPlayerInventory *pInv = InventoryManager()->GetLocalInventory();
Assert( pInv );
if ( pInv )
{
for ( int i = 0 ; i < pInv->GetItemCount(); ++i )
{
CEconItemView *pItem = pInv->GetItem( i );
// Check if this is a quest at all
if ( pItem->GetItemDefinition()->GetQuestDef() == NULL )
continue;
CQuestNotification* pNotification = NULL;
if ( IsUnacknowledged( pItem->GetInventoryPosition() ) )
{
pNotification = new CQuestNotification_NewQuest( pItem->GetSOCData() );
}
else if ( IsQuestItemFullyCompleted( pItem ) ) // Fully completed
{
pNotification = new CQuestNotification_FullyCompletedQuest( pItem->GetSOCData() );
}
else if ( IsQuestItemReadyToTurnIn( pItem ) ) // Ready to turn in
{
pNotification = new CQuestNotification_CompletedQuest( pItem->GetSOCData() );
}
else
{
// Clean up any pending notifications for normal quests
FOR_EACH_VEC_BACK( m_vecNotifications, j )
{
if ( m_vecNotifications[j]->GetItemHandle() == pItem->GetSOCData() )
{
delete m_vecNotifications[j];
m_vecNotifications.Remove( j );
}
}
}
if ( pNotification && !AddNotificationForItem( pItem, pNotification ) )
{
delete pNotification;
pNotification = NULL;
}
}
m_bInitialized = pInv->GetOwner().IsValid();
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
bool CQuestNotificationPanel::AddNotificationForItem( const CEconItemView *pItem, CQuestNotification* pNotification )
{
bool bTypeAlreadyInQueue = false;
// Check if there's already a notification of this type
FOR_EACH_VEC_BACK( m_vecNotifications, i )
{
// There's already a quest of this type in queue, no need to add another
if ( m_vecNotifications[i]->GetType() == pNotification->GetType() )
{
bTypeAlreadyInQueue = true;
break;
}
}
// Find the notified bits
auto idx = m_mapNotifiedItemIDs.Find( pItem->GetItemID() );
if ( idx == m_mapNotifiedItemIDs.InvalidIndex() )
{
// Create if missing
idx = m_mapNotifiedItemIDs.Insert( pItem->GetItemID() );
m_mapNotifiedItemIDs[ idx ].SetSize( CQuestNotification::NUM_NOTIFICATION_TYPES );
FOR_EACH_VEC( m_mapNotifiedItemIDs[ idx ], i )
{
m_mapNotifiedItemIDs[ idx ][ i ] = 0.f;
}
}
// Check if we've already done a notification for this type recently
if ( Plat_FloatTime() < m_mapNotifiedItemIDs[ idx ][ pNotification->GetType() ] || m_mapNotifiedItemIDs[ idx ][ pNotification->GetType() ] == NEVER_REPEAT )
{
return false;
}
bool bNotificationUsed = false;
// Don't play completed notifications unless they happen mid-play
if ( !bTypeAlreadyInQueue && ( m_bInitialized || pNotification->GetType() == CQuestNotification::NOTIFICATION_TYPE_NEW_QUEST ) )
{
// Add notification
m_vecNotifications.AddToTail( pNotification );
bNotificationUsed = true;
}
// Mark that we've created a notification of this type for this item
m_mapNotifiedItemIDs[ idx ][ pNotification->GetType() ] = pNotification->GetReplayTime() == NEVER_REPEAT ? NEVER_REPEAT : Plat_FloatTime() + pNotification->GetReplayTime();
return bNotificationUsed;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
bool CQuestNotificationPanel::ShouldDraw()
{
C_BasePlayer *pPlayer = C_BasePlayer::GetLocalPlayer();
if ( !pPlayer )
return false;
CTFPlayer* pTFPlayer = ToTFPlayer( pPlayer );
if ( !pTFPlayer )
return false;
// Not selected a class, so they haven't joined in
if ( pTFPlayer->IsPlayerClass( 0 ) )
return false;
if ( !CHudElement::ShouldDraw() )
return false;
if ( TFGameRules() && TFGameRules()->IsCompetitiveMode() )
return false;
return true;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CQuestNotificationPanel::OnThink()
{
if ( !ShouldDraw() )
return;
bool bHasStarted = m_animTimer.HasStarted();
float flShowProgress = bHasStarted ? 1.f : 0.f;
const float flTransitionTime = 0.5f;
Update();
if ( bHasStarted )
{
// Transitions
if ( m_animTimer.GetElapsedTime() < flTransitionTime )
{
flShowProgress = Bias( m_animTimer.GetElapsedTime() / flTransitionTime, 0.75f );
}
else if ( ( m_animTimer.GetRemainingTime() + 1.f ) < flTransitionTime )
{
flShowProgress = Bias( Max( 0.0f, m_animTimer.GetRemainingTime() + 1.f ) / flTransitionTime, 0.25f );
}
}
// Move the main container around
if ( m_pMainContainer )
{
int nY = g_pSpectatorGUI && g_pSpectatorGUI->IsVisible() ? g_pSpectatorGUI->GetTopBarHeight() : 0;
float flXPos = RemapValClamped( flShowProgress, 0.f, 1.f, 0.f, m_pMainContainer->GetWide() + XRES( 4 ) );
m_pMainContainer->SetPos( GetWide() - (int)flXPos, nY );
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
bool CQuestNotificationPanel::ShouldPresent()
{
if ( !m_timerNotificationCooldown.IsElapsed() )
return false;
// We need notifications!
if ( m_vecNotifications.IsEmpty() )
return false;
// It's been a few seconds since we were last shown
if ( ( Plat_FloatTime() - m_flTimeSinceLastShown ) < 1.5f )
return false;
return true;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CQuestNotificationPanel::Update()
{
bool bAllowedToShow = ShouldPresent();
if ( bAllowedToShow && !m_bIsPresenting )
{
if ( m_vecNotifications.Head()->ShouldPresent() )
{
float flPresentTime = m_vecNotifications.Head()->Present( this );
m_animTimer.Start( flPresentTime );
m_timerHoldUp.Start( 3.f );
// Notification sound
vgui::surface()->PlaySound( "ui/quest_alert.wav" );
m_bIsPresenting = true;
}
}
else if ( !bAllowedToShow && m_bIsPresenting && m_timerHoldUp.IsElapsed() )
{
m_flTimeSinceLastShown = Plat_FloatTime();
// Play the slide-out animation
g_pClientMode->GetViewportAnimationController()->StartAnimationSequence( this, "QuestNotification_Hide" );
m_bIsPresenting = false;
}
else if ( m_bIsPresenting ) // We are presenting a notification
{
if ( m_vecNotifications.Count() )
{
m_vecNotifications.Head()->Update( this );
// Check if the notification is done
if ( m_vecNotifications.Head()->IsDone() )
{
// Start our cooldown
m_timerNotificationCooldown.Start( 1.f );
// We're done with this notification
delete m_vecNotifications.Head();
m_vecNotifications.Remove( 0 );
}
}
}
}