2021-07-24 21:11:47 -07:00

1327 lines
40 KiB
C++
Raw Blame History

//========= Copyright <20> 1996-2005, Valve Corporation, All rights reserved. ============//
//
// Purpose: Weapon selection hud
//
// $NoKeywords: $
//=============================================================================//
#include "cbase.h"
#if defined( INCLUDE_SCALEFORM )
#include "sfhud_uniquealerts.h"
#include "hud_macros.h"
#include <vgui/ILocalize.h>
#include <vgui/ISurface.h>
#include "iclientmode.h"
#include "hud.h"
#include "hudelement.h"
#include "hud_element_helper.h"
#include "iclientmode.h"
#include "scaleformui/scaleformui.h"
#include "sfhudfreezepanel.h"
#include <engine/IEngineSound.h>
#include "vguicenterprint.h"
#include "hltvreplaysystem.h"
#include "c_team.h"
#include "clientsteamcontext.h"
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
DECLARE_HUDELEMENT( SFUniqueAlerts );
//
SFUI_BEGIN_GAME_API_DEF
SFUI_DECL_METHOD( OnAlertPanelHideAnimEnd ),
SFUI_END_GAME_API_DEF( SFUniqueAlerts, UniqueAlerts );
extern ConVar mp_dm_bonus_percent;
extern ConVar cl_draw_only_deathnotices;
extern ConVar item_debug_give_fake_random_tourney_awards;
extern ConVar mp_team_timeout_max;
//--------------------------------------------------------------------------------------------------------------
/**
* Return the number of players on the given team
*/
static int UTIL_PlayersOnTeam( int teamID, bool isAlive = false, bool allowBots = true )
{
int count = 0;
for ( int i = 1; i <= gpGlobals->maxClients; ++i )
{
CBaseEntity *entity = UTIL_PlayerByIndex( i );
if ( entity == NULL )
continue;
CBasePlayer *player = static_cast<CBasePlayer *>( entity );
if ( !allowBots && player->IsBot() )
continue;
if ( player->GetTeamNumber() != teamID )
continue;
if ( isAlive && !player->IsAlive() )
continue;
count++;
}
return count;
}
// This function was designed for the quests included in Operation: Wildfire,
// and may need to be updated for future operations.
static bool ShouldQuestShowMissionPanel( const CEconQuestDefinition *pQuestDef )
{
if ( !pQuestDef )
return false;
if ( pQuestDef->IsAnEvent() )
return false;
// No mission panel for guardian mode -- we have special case code for these quests
const char* gameMode = pQuestDef->GetGameMode();
if ( gameMode && !Q_stricmp( gameMode, "cooperative" ) )
return false;
// No mission panel for co-op strike mode. Goal is just to win.
if ( gameMode && !Q_stricmp( gameMode, "coopmission" ) )
return false;
// Show mission panel for all other quests
return true;
}
static bool ShouldQuestShowMissionPanel( CCSPlayer* pPlayer, CCSGameRules* pGameRules )
{
// Don't show quest UI if we don't have an active quest.
uint32 unQuestId = pPlayer->GetActiveQuestID();
if ( unQuestId == 0 )
return false;
// If the quest isn't valid for some reason (perhaps we are in the wrong map or mode)
// don't show the UI. We do allow showing it during transient situations (all bots,
// or during warmup)
switch ( pPlayer->GetQuestProgressReason() )
{
case QuestProgress::QUEST_OK:
case QuestProgress::QUEST_NOT_ENOUGH_PLAYERS:
case QuestProgress::QUEST_WARMUP:
break;
default:
return false;
}
// Check if the player's active quest allows showing the mission panel
if ( !ShouldQuestShowMissionPanel( GetItemSchema()->GetQuestDefinition( unQuestId ) ) )
return false;
// Don't show quest progress at game end; at this point the server updates the user's
// quest progress but the client still has the 'uncommitted' quest progress state, and
// so it double counts. Just hide in this case to avoid showing the double-counted amount.
if ( pGameRules->GetGamePhase() == GAMEPHASE_MATCH_ENDED )
return false;
// Everything is OK, show quest UI!
return true;
}
//-----------------------------------------------------------------------------
// Purpose: Constructor
//-----------------------------------------------------------------------------
SFUniqueAlerts::SFUniqueAlerts( const char *value )
: SFHudFlashInterface( value )
, m_bVisible( false )
, m_AlertState( ALERTSTATE_HIDDEN )
, m_flNextAlertTick( -1 )
, m_bShowDemolitionProgressionPanel( false )
, m_bDMBonusIsActive( false )
, m_flNextDMBonusTick( -1 )
, m_nDMCurrentWeaponPoints( 0 )
, m_nDMCurrentWeaponBonusPoints( 0 )
, m_nDMBonusWeaponSlot( -1 )
, m_bMissionVisible( false )
, m_guardianPanelInitialized( false )
, m_guardianPanelHideTick( -1 )
, m_nGuardianKills( 0 )
, m_nGuardianWeapon( -1 )
, m_nPlayerMoney( 0 )
, m_bQuestMissionActive( false )
#if defined ( ENABLE_GIFT_PANEL )
, m_bIsGiftPanelActive( false )
, m_nGlobalGiftsGivenInPeriod( 0 )
, m_nGlobalNumGiftersInPeriod( 0 )
, m_nGlobalGiftingPeriodSeconds( 0 )
, m_flLastGiftPanelUpdate( 0 )
, m_PlayerSpotlightTempList()
#endif
{
SetHiddenBits( HIDEHUD_PLAYERDEAD | HIDEHUD_MISCSTATUS );
#if defined ( ENABLE_GIFT_PANEL )
m_PlayerSpotlightTempList.RemoveAll();
#endif
}
static bool UTIL_CheckWeaponMatch(int weaponIdx, const CWeaponCSBase* pWeapon)
{
if ( !pWeapon )
return false;
const CEconItemView *pItemView = pWeapon->GetEconItemView();
if ( !pItemView )
return false;
const CEconItemDefinition *pItemDef = pItemView->GetItemDefinition();
if ( !pItemDef )
return false;
if ( weaponIdx != pItemDef->GetDefinitionIndex() )
return false;
return true;
}
void SFUniqueAlerts::ProcessMissionPanelGuardian()
{
C_CSPlayer* pPlayer = C_CSPlayer::GetLocalCSPlayer();
bool guardianForceShow = true; // Always show the mission panel in guardian mode.
bool guardianPanelChange = false;
if ( m_nGuardianKills != CSGameRules()->GetGuardianKillsRemaining() )
{
// If we got a kill, notice the update
int newKills = CSGameRules()->GetGuardianKillsRemaining();
m_nGuardianKills = newKills;
guardianPanelChange = true;
}
if ( m_nGuardianWeapon != CSGameRules()->GetGuardianSpecialWeapon() )
{
// Weapon changed, just reinitialize from nothing
m_nGuardianWeapon = CSGameRules()->GetGuardianSpecialWeapon();
guardianPanelChange = true;
m_guardianPanelInitialized = false; // force re-initialize of state.
}
bool hasSpecialWeapon = false;
bool hasSpecialWeaponEquipped = false;
if ( m_nGuardianWeapon != 0 )
{
// Check if the player has the special weapon
int numWeapons = pPlayer->WeaponCount();
for ( int iWeapon = 0; iWeapon < numWeapons; ++iWeapon )
{
const CWeaponCSBase* pWeapon = dynamic_cast< CWeaponCSBase* >( pPlayer->GetWeapon( iWeapon ) );
if ( UTIL_CheckWeaponMatch( m_nGuardianWeapon, pWeapon ) )
{
hasSpecialWeapon = true;
break;
}
}
// Check if the player has the special weapon equipped
CWeaponCSBase* pActiveWeapon = pPlayer->GetActiveCSWeapon();
hasSpecialWeaponEquipped = UTIL_CheckWeaponMatch( m_nGuardianWeapon, pActiveWeapon );
}
// Check for state changes
if ( hasSpecialWeapon != m_bGuardianHasWeapon )
{
m_bGuardianHasWeapon = hasSpecialWeapon;
guardianPanelChange = true;
}
if ( hasSpecialWeaponEquipped != m_bGuardianHasWeaponEquipped )
{
m_bGuardianHasWeaponEquipped = hasSpecialWeaponEquipped;
guardianPanelChange = true;
}
if ( guardianPanelChange )
{
if ( !m_guardianPanelInitialized )
{
m_guardianPanelInitialized = true;
// Get total number of kills required
int killsRequired = CSGameRules()->GetGuardianRequiredKills();
// Get weapon name
const CEconItemDefinition* pItemDef = NULL;
if ( m_nGuardianWeapon != 0 )
pItemDef = GetItemSchema()->GetItemDefinition( m_nGuardianWeapon );
wchar_t szQuestDescription[128];
const char* szWeaponIcon;
const wchar_t* szWeaponName;
if ( pItemDef && pItemDef->GetDefinitionIndex() != 0 )
{
szWeaponIcon = pItemDef->GetDefinitionName();
szWeaponName = g_pVGuiLocalize->Find( pItemDef->GetItemBaseName() );
}
else
{
szWeaponIcon = ""; // TODO: new "any weapon" icon
szWeaponName = g_pVGuiLocalize->Find( "#quest_weapon_any_weapon" );
}
g_pVGuiLocalize->ConstructString( szQuestDescription, sizeof( szQuestDescription ), g_pVGuiLocalize->Find( "#quest_hud_guardian_kills_with" ), 1, szWeaponName );
// function missionInitGuardian( weaponName:String, questDesc:String, killsRequired:Number, weaponCost:Number )
WITH_SFVALUEARRAY_SLOT_LOCKED( data, 4 )
{
m_pScaleformUI->ValueArray_SetElement( data, 0, szWeaponIcon );
m_pScaleformUI->ValueArray_SetElement( data, 1, szQuestDescription );
m_pScaleformUI->ValueArray_SetElement( data, 2, killsRequired );
m_pScaleformUI->ValueArray_SetElement( data, 3, 0 ); // $$$REI TODO ECON: Get weapon cost
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "missionInitGuardian", data, 4 );
}
}
if ( !m_bMissionVisible )
{
m_bMissionVisible = true;
// Need to show the panel
WITH_SLOT_LOCKED
{
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "missionShowPanel", NULL, 0 );
}
}
// Update panel with new data
int killsRemaining = CSGameRules()->GetGuardianKillsRemaining();
int money = 0; // pPlayer->GetAccount(); $$$REI TODO ECON
// function missionUpdateGuardian( killsRemaining:Number, money : Number, hasWeapon : Boolean )
WITH_SFVALUEARRAY_SLOT_LOCKED( data, 3 )
{
m_pScaleformUI->ValueArray_SetElement( data, 0, killsRemaining );
m_pScaleformUI->ValueArray_SetElement( data, 1, money );
m_pScaleformUI->ValueArray_SetElement( data, 2, m_bGuardianHasWeapon );
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "missionUpdateGuardian", data, 3 );
}
// Show panel for at least 5 seconds after any change
m_guardianPanelHideTick = Plat_FloatTime() + 5.0f;
}
else if ( guardianForceShow && !m_bMissionVisible )
{
m_bMissionVisible = true;
// Need to show the panel
WITH_SLOT_LOCKED
{
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "missionShowPanel", NULL, 0 );
}
}
if ( m_bMissionVisible && !guardianForceShow && m_guardianPanelHideTick <= Plat_FloatTime() )
{
m_bMissionVisible = false;
WITH_SLOT_LOCKED
{
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "missionHidePanel", NULL, 0 );
}
}
}
void SFUniqueAlerts::ProcessDMBonusPanel()
{
int nSecondsLeft = ( CSGameRules()->GetDMBonusStartTime() + CSGameRules()->GetDMBonusTimeLength() ) - gpGlobals->curtime;
if ( nSecondsLeft >= 0 && CSGameRules()->IsDMBonusActive() )
{
int nSlot = CSGameRules()->GetDMBonusWeaponLoadoutSlot();
if ( m_nDMBonusWeaponSlot != nSlot || m_bVisible == false )
{
m_nDMBonusWeaponSlot = nSlot;
m_nDMCurrentWeaponPoints = CSGameRules()->GetWeaponScoreForDeathmatch( nSlot );
m_nDMCurrentWeaponBonusPoints = ( ( float )( mp_dm_bonus_percent.GetInt() ) / 100.0f ) * m_nDMCurrentWeaponPoints;
CCSPlayerInventory* pPlayerInv = CSInventoryManager()->GetInventoryForPlayer( ClientSteamContext().GetLocalPlayerSteamID() );
if ( !pPlayerInv )
return;
CEconItemView* pItem = pPlayerInv->GetItemInLoadout( C_CSPlayer::GetLocalCSPlayer()->GetTeamNumber(), nSlot );
if ( !pItem )
return;
ShowDMBonusWeapon( pItem, nSlot, nSecondsLeft );
}
else
{
WITH_SFVALUEARRAY_SLOT_LOCKED( data, 1 )
{
m_pScaleformUI->ValueArray_SetElement( data, 0, nSecondsLeft );
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "setDMBonusTimer", data, 1 );
}
}
}
else
{
m_bDMBonusIsActive = false;
WITH_SLOT_LOCKED
{
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "hideDMBonusWeapon", NULL, 0 );
}
C_RecipientFilter filter;
filter.AddRecipient( C_CSPlayer::GetLocalPlayer() );
C_BaseEntity::EmitSound( filter, SOUND_FROM_WORLD, "UI.DeathMatchBonusAlertEnd" );
}
}
void SFUniqueAlerts::ProcessAlertBar()
{
// get the round restart time first
float flEndTime = CSGameRules()->GetRoundRestartTime() - 0.5f;
bool bIsRestarting = CSGameRules()->IsGameRestarting();
if ( g_HltvReplaySystem.GetHltvReplayDelay() > 0 )
{
//ShowHltvReplayAlertPanel( int( CL_GetHltvReplayTicksLeft() * gpGlobals->interval_per_tick ) );
HideAlertText(); // we don't need the DELAYED REPLAY strip anymore
}
else if ( bIsRestarting )
{
int nLeft = ( int )flEndTime - ( int )gpGlobals->curtime;
if ( nLeft >= 0 )
{
wchar_t szNotice[128] = L"";
wchar_t wzSecs[16];
V_swprintf_safe( wzSecs, L"%d", nLeft );
if ( CSGameRules()->IsWarmupPeriod() )
{
if ( nLeft == 0 )
g_pVGuiLocalize->ConstructString( szNotice, sizeof( szNotice ), g_pVGuiLocalize->Find( "#SFUI_Notice_Alert_Match_Starting" ), 1, wzSecs );
else
g_pVGuiLocalize->ConstructString( szNotice, sizeof( szNotice ), g_pVGuiLocalize->Find( "#SFUI_Notice_Alert_Match_Starting_In" ), 1, wzSecs );
}
else
{
if ( nLeft == 0 )
g_pVGuiLocalize->ConstructString( szNotice, sizeof( szNotice ), g_pVGuiLocalize->Find( "#SFUI_Notice_Alert_Match_Restarting" ), 1, wzSecs );
else
g_pVGuiLocalize->ConstructString( szNotice, sizeof( szNotice ), g_pVGuiLocalize->Find( "#SFUI_Notice_Alert_Match_Restarting_In" ), 1, wzSecs );
}
ShowAlertText( szNotice );
}
}
else if ( CSGameRules()->IsWarmupPeriod() )
{
ShowWarmupAlertPanel();
}
else if ( CSGameRules()->IsFreezePeriod() )// we're paused
{
if ( CSGameRules()->IsTimeOutActive() )
{
int nTimeLeftInSec;
int nTimeOutsRemaining;
int iTeamIndex;
if ( CSGameRules()->IsTerroristTimeOutActive() )
{
nTimeOutsRemaining = CSGameRules( )->GetTerroristTimeOuts( );
nTimeLeftInSec = ( int )CSGameRules()->GetTerroristTimeOutRemaining();
iTeamIndex = TEAM_TERRORIST;
}
else if ( CSGameRules()->IsCTTimeOutActive() )
{
nTimeOutsRemaining = CSGameRules( )->GetCTTimeOuts( );
nTimeLeftInSec = ( int )CSGameRules()->GetCTTimeOutRemaining();
iTeamIndex = TEAM_CT;
}
else
{
return;
}
const char *szTeam = "#SFUI_Notice_Alert_Timeout";
if ( nTimeLeftInSec > 0 )
{
int nMinLeft = nTimeLeftInSec / 60;
int nSecLeft = nTimeLeftInSec - ( nMinLeft * 60 );
wchar_t szNotice[64] = L"";
wchar_t wzTime[32] = L"";
if ( mp_team_timeout_max.GetInt() > 1 )
{
_snwprintf( wzTime, ARRAYSIZE( wzTime ), L"%d:%02d (%d remaining)", nMinLeft, nSecLeft, nTimeOutsRemaining );
}
else
{
_snwprintf( wzTime, ARRAYSIZE( wzTime ), L"%d:%02d", nMinLeft, nSecLeft );
}
// get the team
C_Team *pTeam = GetGlobalTeam( iTeamIndex );
// se if we have a custom clan name
wchar_t wszSafeName[ MAX_TEAM_NAME_LENGTH ];
wszSafeName[ 0 ] = L'\0';
if ( ( pTeam == NULL ) || StringIsEmpty( pTeam->Get_ClanName( ) ) )
{
// if not, just use the default T or CT labels
switch ( iTeamIndex )
{
case TEAM_TERRORIST:
V_snwprintf( wszSafeName, ARRAYSIZE( wszSafeName ), g_pVGuiLocalize->Find( "#terrorists" ) );
break;
case TEAM_CT:
V_snwprintf( wszSafeName, ARRAYSIZE( wszSafeName ), g_pVGuiLocalize->Find( "#counter-terrorists" ) );
break;
}
}
else
{
wchar_t wszName[ MAX_TEAM_NAME_LENGTH ];
// we have a custom team name, convert to wide
g_pVGuiLocalize->ConvertANSIToUnicode( pTeam->Get_ClanName( ), wszName, sizeof( wszName ) );
// now make the team name string safe
g_pScaleformUI->MakeStringSafe( wszName, wszSafeName, sizeof( wszName ) );
}
g_pVGuiLocalize->ConstructString( szNotice, sizeof( szNotice ), g_pVGuiLocalize->Find( szTeam ), 2, wszSafeName, wzTime );
ShowAlertText( szNotice );
}
}
else if ( CSGameRules()->IsMatchWaitingForResume() )
{
ShowAlertText( g_pVGuiLocalize->Find( "#SFUI_Notice_Alert_Freeze_Pause" ) );
}
else
{
HideAlertText();
}
}
else
{
HideAlertText(); // Is this a good thing to do here? seems like we might be overriding someone else....
}
}
#if defined ( ENABLE_GIFT_PANEL )
void SFUniqueAlerts::ProcessGiftPanel()
{
C_CSPlayer *pLocalPlayer = C_CSPlayer::GetLocalCSPlayer();
AssertMsg( pLocalPlayer, "Should have checked GetLocalCSPlayer was valid in ProcessInput()" );
if ( ( CSGameRules()->IsWarmupPeriod() || ( CSGameRules()->IsFreezePeriod() && CSGameRules()->GetTotalRoundsPlayed() < 1 ) ) &&
!( engine->GetDemoPlaybackParameters() && engine->GetDemoPlaybackParameters()->m_bAnonymousPlayerIdentity )
)
{
// don't update every frame
if ( ( m_flLastGiftPanelUpdate + 1.0f ) <= gpGlobals->curtime )
{
m_flLastGiftPanelUpdate = gpGlobals->curtime;
m_nGlobalGiftsGivenInPeriod = CSGameRules()->m_numGlobalGiftsGiven;
m_nGlobalNumGiftersInPeriod = CSGameRules()->m_numGlobalGifters;
m_nGlobalGiftingPeriodSeconds = CSGameRules()->m_numGlobalGiftsPeriodSeconds;
if ( m_nGlobalGiftsGivenInPeriod > 0 )
{
// get the data from gamerules about:
// how many gifts were given
// how many players gifted them
// the steamIDs of the five "spotlighted" players
AccountID_t uiLocalAccountID = steamapicontext->SteamUser()->GetSteamID().GetAccountID();
bool bIsLocalPlayer = false;
m_PlayerSpotlightTempList.RemoveAll();
for ( int i = 0; i < MAX_PLAYER_GIFT_DROP_DISPLAY; i++ )
{
AccountID_t uiAccountID = CSGameRules()->m_arrFeaturedGiftersAccounts[i];
if ( uiAccountID == 0 )
break;
int nGiftNum = CSGameRules()->m_arrFeaturedGiftersGifts[i];
if ( pLocalPlayer && uiLocalAccountID == uiAccountID )
bIsLocalPlayer = true;
if ( m_PlayerSpotlightTempList.Count() < MAX_PLAYER_GIFT_DROP_DISPLAY )
{
GiftTempList_t drop;
//drop.nDefindex = nDefindex;
drop.uiAccountID = uiAccountID;
drop.nNumGifts = nGiftNum;
drop.bIsLocalPlayer = bIsLocalPlayer;
m_PlayerSpotlightTempList.AddToTail( drop );
}
}
WITH_SLOT_LOCKED
{
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "ClearAllPlayerReceivedGift", NULL, 0 );
}
// send the list of player ids - we'll show these names on screen
if ( m_nGlobalGiftsGivenInPeriod && m_PlayerSpotlightTempList.Count() > 0 )
{
for ( int i = 0; i < m_PlayerSpotlightTempList.Count(); i++ )
{
CSteamID steamID( m_PlayerSpotlightTempList[i].uiAccountID, steamapicontext->SteamUser()->GetSteamID().GetEUniverse(), k_EAccountTypeIndividual );
// only send the first 5
if ( i < MAX_PLAYER_GIFT_DROP_DISPLAY )
{
WITH_SFVALUEARRAY_SLOT_LOCKED( args, 4 )
{
char xuidText[255];
xuidText[0] = '\0';
V_snprintf( xuidText, ARRAYSIZE( xuidText ), "%llu", steamID.ConvertToUint64() );
m_pScaleformUI->ValueArray_SetElement( args, 0, xuidText );
m_pScaleformUI->ValueArray_SetElement( args, 1, m_PlayerSpotlightTempList[i].nDefindex );
m_pScaleformUI->ValueArray_SetElement( args, 2, m_PlayerSpotlightTempList[i].nNumGifts );
m_pScaleformUI->ValueArray_SetElement( args, 3, m_PlayerSpotlightTempList[i].bIsLocalPlayer );
m_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "AddPlayerReceivedGift", args, 4 );
}
}
}
WITH_SFVALUEARRAY_SLOT_LOCKED( data, 3 )
{
m_pScaleformUI->ValueArray_SetElement( data, 0, m_nGlobalGiftsGivenInPeriod );
m_pScaleformUI->ValueArray_SetElement( data, 1, m_nGlobalNumGiftersInPeriod );
m_pScaleformUI->ValueArray_SetElement( data, 2, m_nGlobalGiftingPeriodSeconds );
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "UpdatePanelAndItemNumbers", data, 3 );
}
}
m_bIsGiftPanelActive = true;
}
}
}
else if ( m_bIsGiftPanelActive == true )
{
m_bIsGiftPanelActive = false;
m_nGlobalGiftsGivenInPeriod = 0;
// remove our temp list
m_PlayerSpotlightTempList.RemoveAll();
WITH_SLOT_LOCKED
{
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "HidePanelAndItemNumbers", NULL, 0 );
}
}
}
#endif // ENABLE_GIFT_PANEL
#if MISSIONPANEL_ENABLE_QUEST_PROGRESSION
void SFUniqueAlerts::ProcessMissionPanelQuest()
{
// TODO: Only tick every quarter second
C_CSPlayer* pPlayer = C_CSPlayer::GetLocalCSPlayer();
if ( !pPlayer )
return;
uint32 unQuestId = pPlayer->GetActiveQuestID();
if ( !unQuestId )
return;
const CEconQuestDefinition* pQuestDef = GetItemSchema()->GetQuestDefinition( unQuestId );
if ( !pQuestDef )
return;
// Get current progress
uint32 numPointsRemaining, numPointsUncommitted;
if ( !InventoryManager()->BGetPlayerQuestIdPointsRemaining( unQuestId, numPointsRemaining, numPointsUncommitted ) )
return;
// Get the required # of points to complete the quest.
// TODO: This interface is awkward, I think it was originally to support quests with
// multiple objectives, but this array is always size 1 (or maybe sometimes 0, which means '1 point required').
int requiredPoints = 1;
const CCopyableUtlVector< uint32 >& points = pQuestDef->GetQuestPoints();
if ( !points.IsEmpty() )
requiredPoints = points[0];
int currentPoints = ( requiredPoints - numPointsRemaining ) + numPointsUncommitted;
if ( currentPoints < 0 )
currentPoints = 0;
if ( currentPoints > requiredPoints )
currentPoints = requiredPoints;
// Initialize panel on first access
if(!m_bMissionVisible)
{
m_bMissionVisible = true;
// Get the current quest information
WITH_SFVALUEARRAY_SLOT_LOCKED( args, 2 )
{
m_pScaleformUI->ValueArray_SetElement( args, 0, currentPoints );
m_pScaleformUI->ValueArray_SetElement( args, 1, requiredPoints );
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "missionSetGoal", args, 2 );
}
// Get mission description
const char* szDescToken = pQuestDef->GetHudDesscriptionLocToken();
const wchar_t *szDescUnformatted = NULL;
wchar_t szDesc[1024] = L"";
wchar_t szTrimmedDesc[1024] = L"";
if ( szDescToken != NULL && szDescToken[0] != '\0' )
szDescUnformatted = g_pVGuiLocalize->Find( szDescToken );
if ( szDescUnformatted != NULL )
{
// TODO: Factor out this code (it, or something like it, is used in a few places)
// Get a fake item for this quest
static CSchemaItemDefHandle hItemDefQuest( "quest" );
// create a new KV that has localized strings in it
KeyValues *pKVLocalizedQuestStrings = new KeyValues( "LocalizedQuestStrings" );
KeyValues::AutoDelete autodelete( pKVLocalizedQuestStrings );
g_pVGuiLocalize->ConstructString( szDesc, sizeof( szDesc ), szDescUnformatted, pKVLocalizedQuestStrings );
// Remove double spaces caused by missing optional tags
UTIL_TrimEmptyWhitespaceFromHTML( szTrimmedDesc, sizeof( szTrimmedDesc ), szDesc );
}
WITH_SFVALUEARRAY_SLOT_LOCKED( args, 1 )
{
m_pScaleformUI->ValueArray_SetElement( args, 0, szTrimmedDesc );
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "missionSetDescription", args, 1 );
}
WITH_SFVALUEARRAY_SLOT_LOCKED( args, 1 )
{
// Currently not using Extra Info area. Could put bonus requirements/points there
m_pScaleformUI->ValueArray_SetElement( args, 0, "" );
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "missionSetExtraInfo", args, 1 );
}
const char* szIcon = pQuestDef->GetIcon();
if ( !szIcon )
szIcon = "";
WITH_SFVALUEARRAY_SLOT_LOCKED( args, 2 )
{
m_pScaleformUI->ValueArray_SetElement( args, 0, szIcon );
m_pScaleformUI->ValueArray_SetElement( args, 1, true ); // always show solid icon for quests for now.
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "missionUpdateIcon", args, 2 );
}
WITH_SLOT_LOCKED
{
// Show the panel!
m_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "missionShowPanel", NULL, 0 );
}
}
// Update current quest points
WITH_SFVALUEARRAY_SLOT_LOCKED( args, 1 )
{
m_pScaleformUI->ValueArray_SetElement( args, 0, currentPoints );
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "missionUpdateGoal", args, 1 );
}
}
#endif // MISSIONPANEL_ENABLE_QUEST_PROGRESSION
void SFUniqueAlerts::ProcessInput()
{
if ( !m_pScaleformUI )
return;
C_CSPlayer *pPlayer = C_CSPlayer::GetLocalCSPlayer();
if ( !pPlayer || !CSGameRules() )
return;
if ( m_bShowDemolitionProgressionPanel )
{
m_bShowDemolitionProgressionPanel = false;
UpdateGGWeaponIconList();
}
if ( CSGameRules()->IsPlayingCoopGuardian() )
{
// Check if we need to update the guardian panel
// TODO: Maybe only do this once every .25 seconds or so? Doesn't seem to be a
// performance issue at the moment, so leaving it as is for now.
ProcessMissionPanelGuardian();
}
#if MISSIONPANEL_ENABLE_QUEST_PROGRESSION
else if( ShouldQuestShowMissionPanel(pPlayer, CSGameRules()) )
{
ProcessMissionPanelQuest();
}
#endif
else if( m_bMissionVisible )
{
// Somehow we ended up with the mission panel visible even though no mission is active. Hide it.
m_bMissionVisible = false;
WITH_SLOT_LOCKED
{
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "missionHidePanel", NULL, 0 );
}
}
if ( m_bDMBonusIsActive && m_flNextDMBonusTick <= Plat_FloatTime() )
{
m_flNextDMBonusTick = Plat_FloatTime() + 0.25f;
ProcessDMBonusPanel();
}
if ( m_bVisible && m_flNextAlertTick <= gpGlobals->curtime )
{
m_flNextAlertTick = gpGlobals->curtime + 1;
ProcessAlertBar();
}
#if defined ( ENABLE_GIFT_PANEL )
ProcessGiftPanel();
#endif
}
void SFUniqueAlerts::OnTimeJump()
{
m_flNextAlertTick = -1; // refresh notices on next Think
}
void SFUniqueAlerts::ShowDMBonusWeapon( CEconItemView* pItem, int nPos, int nSecondsLeft )
{
C_CSPlayer *pLocalPlayer = C_CSPlayer::GetLocalCSPlayer();
if ( !pLocalPlayer )
return;
WITH_SFVALUEARRAY_SLOT_LOCKED( data, 5 )
{
m_pScaleformUI->ValueArray_SetElement( data, 0, pItem->GetStaticData()->GetDefinitionName() );
m_pScaleformUI->ValueArray_SetElement( data, 1, g_pVGuiLocalize->Find( pItem->GetStaticData()->GetItemBaseName() ) );
m_pScaleformUI->ValueArray_SetElement( data, 2, nSecondsLeft );
m_pScaleformUI->ValueArray_SetElement( data, 3, m_nDMCurrentWeaponPoints );
m_pScaleformUI->ValueArray_SetElement( data, 4, m_nDMCurrentWeaponBonusPoints );
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "showDMBonusWeapon", data, 5 );
}
CBaseCombatWeapon *pOwnedWeapon = pLocalPlayer->Weapon_GetPosition( ( loadout_positions_t )nPos );
// If we already have the bonus weapon, switch to it.
if ( pOwnedWeapon || pLocalPlayer->CanPlayerBuy( false ) )
GetCenterPrint()->Print( "#SFUI_Notice_DM_BonusSwitchTo" );
else
GetCenterPrint()->Print( "#SFUI_Notice_DM_BonusRespawn" );
C_RecipientFilter filter;
filter.AddRecipient( pLocalPlayer );
C_BaseEntity::EmitSound( filter, SOUND_FROM_WORLD, "UI.DeathMatchBonusAlertStart" );
}
void SFUniqueAlerts::ShowQuestProgress(uint32 numPointsEarned, uint32 numPointsTotal, uint32 numPointsMax, const char* weaponName, const char* questDesc)
{
#if MISSIONPANEL_ENABLE_QUEST_PROGRESSION
C_CSPlayer *pLocalPlayer = C_CSPlayer::GetLocalCSPlayer();
if (!pLocalPlayer)
return;
//function showQuestProgress( itemName:String, questDesc:String, nProgress:Number, nMaxProgress:Number, nPoints:Number)
WITH_SFVALUEARRAY_SLOT_LOCKED(data, 5)
{
m_pScaleformUI->ValueArray_SetElement(data, 0, weaponName);
m_pScaleformUI->ValueArray_SetElement(data, 1, questDesc);
m_pScaleformUI->ValueArray_SetElement(data, 2, (int)numPointsTotal);
m_pScaleformUI->ValueArray_SetElement(data, 3, (int)numPointsMax);
m_pScaleformUI->ValueArray_SetElement(data, 4, (int)numPointsEarned);
g_pScaleformUI->Value_InvokeWithoutReturn(m_FlashAPI, "showQuestProgress", data, 5);
}
// TODO: Better sound effects?
C_RecipientFilter filter;
filter.AddRecipient(pLocalPlayer);
C_BaseEntity::EmitSound(filter, SOUND_FROM_WORLD, "UI.DeathMatchBonusAlertStart");
#endif
}
void SFUniqueAlerts::LevelInit( void )
{
if ( !FlashAPIIsValid() )
{
SFUI_REQUEST_ELEMENT( SF_SS_SLOT( GET_ACTIVE_SPLITSCREEN_SLOT() ), g_pScaleformUI, SFUniqueAlerts, this, UniqueAlerts );
}
else
{
WITH_SLOT_LOCKED
{
m_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "onLevelReset", NULL, 0 );
}
}
// New level, reset everything
m_AlertState = ALERTSTATE_HIDDEN;
m_flNextAlertTick = -1;
m_bShowDemolitionProgressionPanel = false;
m_bDMBonusIsActive = false;
m_flNextDMBonusTick = -1;
m_nDMCurrentWeaponPoints = 0;
m_nDMCurrentWeaponBonusPoints = 0;
m_nDMBonusWeaponSlot = -1;
m_bMissionVisible = false;
m_guardianPanelInitialized = false;
m_guardianPanelHideTick = -1;
m_nGuardianKills = 0;
m_nGuardianWeapon = -1;
m_nPlayerMoney = 0;
m_bGuardianHasWeapon = false;
m_bGuardianHasWeaponEquipped = false;
m_bQuestMissionActive = false;
#if defined ( ENABLE_GIFT_PANEL )
m_bIsGiftPanelActive = false;
m_nGlobalGiftsGivenInPeriod = 0;
m_nGlobalNumGiftersInPeriod = 0;
m_nGlobalGiftingPeriodSeconds = 0;
m_flLastGiftPanelUpdate = 0;
m_PlayerSpotlightTempList.Purge();
#endif
}
void SFUniqueAlerts::LevelShutdown( void )
{
if ( FlashAPIIsValid() )
{
RemoveFlashElement();
}
}
bool SFUniqueAlerts::ShouldDraw( void )
{
bool bTrainingMode = CSGameRules() && CSGameRules()->IsPlayingTraining();
return
cl_drawhud.GetBool()
&& cl_draw_only_deathnotices.GetBool() == false
&& !IsTakingAFreezecamScreenshot()
&& !bTrainingMode
&& CHudElement::ShouldDraw();
}
void SFUniqueAlerts::SetActive( bool bActive )
{
if ( FlashAPIIsValid() )
{
if ( bActive != m_bVisible )
{
if(bActive)
{
Show();
}
else
{
Hide();
}
}
}
CHudElement::SetActive( bActive );
}
void SFUniqueAlerts::FlashReady( void )
{
if ( m_FlashAPI && m_pScaleformUI )
{
ListenForGameEvent( "player_spawn" );
ListenForGameEvent( "round_start" );
ListenForGameEvent( "round_announce_final" );
ListenForGameEvent( "round_announce_match_point" );
ListenForGameEvent( "round_announce_last_round_half" );
ListenForGameEvent( "round_announce_match_start" );
ListenForGameEvent( "round_announce_warmup" );
ListenForGameEvent( "dm_bonus_weapon_start" );
ListenForGameEvent( "gg_killed_enemy" );
}
}
bool SFUniqueAlerts::PreUnloadFlash( void )
{
return true;
}
void SFUniqueAlerts::FireGameEvent( IGameEvent *event )
{
if ( !FlashAPIIsValid() )
return;
const char *type = event->GetName();
C_CSPlayer *pLocalPlayer = C_CSPlayer::GetLocalCSPlayer();
if ( !pLocalPlayer )
return;
int EventUserID = event->GetInt( "userid", -1 );
int LocalPlayerID = ( pLocalPlayer != NULL ) ? pLocalPlayer->GetUserID() : -2;
if ( Q_strcmp( "round_announce_match_start", type ) == 0 )
{
ShowAlertText( g_pVGuiLocalize->Find( "#SFUI_Notice_Alert_Match_Start" ), true );
C_RecipientFilter filter;
filter.AddRecipient( pLocalPlayer );
C_BaseEntity::EmitSound( filter, SOUND_FROM_WORLD, "Music.Match_Start_Stinger" );
}
else if ( Q_strcmp( "round_start", type ) == 0 )
{
if ( CSGameRules() && CSGameRules()->IsPlayingGunGameTRBomb()/* && !pLocalPlayer->IsControllingBot()*/ )
{
m_bShowDemolitionProgressionPanel = true;
}
if ( pLocalPlayer->IsHLTV() )
HideAlertText();
m_flNextAlertTick = -1;
if ( !CSGameRules()->IsFreezePeriod() && !CSGameRules()->IsWarmupPeriod() )
{
WITH_SLOT_LOCKED
{
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "HidePanelAndItemNumbers", NULL, 0 );
}
}
}
else if ( Q_strcmp( "round_announce_final", type ) == 0 ||
Q_strcmp( "round_announce_last_round_half", type ) == 0 ||
Q_strcmp( "round_announce_match_point", type ) == 0 ||
Q_strcmp( "player_spawn", type ) == 0 )
{
if ( Q_strcmp( "round_announce_final", type ) == 0 )
{
ShowAlertText( g_pVGuiLocalize->Find( "#SFUI_Notice_Alert_Final_Round" ), true );
C_RecipientFilter filter;
filter.AddRecipient( pLocalPlayer );
C_BaseEntity::EmitSound( filter, SOUND_FROM_WORLD, "Music.Final_Round_Stinger" );
}
else if ( Q_strcmp( "round_announce_match_point", type ) == 0 )
{
ShowAlertText( g_pVGuiLocalize->Find( "#SFUI_Notice_Alert_Match_Point" ), true );
C_RecipientFilter filter;
filter.AddRecipient( pLocalPlayer );
C_BaseEntity::EmitSound( filter, SOUND_FROM_WORLD, "Music.Match_Point_Stinger" );
}
else if ( Q_strcmp( "round_announce_last_round_half", type ) == 0 )
{
ShowAlertText( g_pVGuiLocalize->Find( "#SFUI_Notice_Alert_Last_Round_Half" ), true );
}
else if ( Q_strcmp( "player_spawn", type ) == 0 && EventUserID == LocalPlayerID )
{
HideAlertText();
}
}
else if ( Q_strcmp( "round_announce_warmup", type ) == 0 )
{
HideAlertText();
}
else if ( Q_strcmp( "dm_bonus_weapon_start", type ) == 0 )
{
// don't do the initial show if we are set not to draw right now
// we'll fill the data out during the ProcessInput
if ( ShouldDraw() )
{
int nTime = event->GetInt( "time", 0 );
// int nWepID = event->GetInt( "wepID", -1 );
int nPos = event->GetInt( "Pos", 0 );
CCSPlayerInventory* pPlayerInv = CSInventoryManager()->GetInventoryForPlayer( ClientSteamContext().GetLocalPlayerSteamID() );
if ( !pPlayerInv )
return;
m_nDMBonusWeaponSlot = nPos;
//CSGameRules()->GetDMBonusWeaponLoadoutSlot()
CEconItemView* pItem = pPlayerInv->GetItemInLoadout( C_CSPlayer::GetLocalCSPlayer()->GetTeamNumber(), nPos );
if ( !pItem )
return;
m_bDMBonusIsActive = true;
m_flNextDMBonusTick = Plat_FloatTime() + 0.25f;
if ( CSGameRules() )
{
m_nDMCurrentWeaponPoints = CSGameRules()->GetWeaponScoreForDeathmatch( nPos );
m_nDMCurrentWeaponBonusPoints = ( ( float )( mp_dm_bonus_percent.GetInt() ) / 100.0f ) * m_nDMCurrentWeaponPoints;
}
ShowDMBonusWeapon( pItem, nPos, nTime );
}
}
else if ( Q_strcmp( "gg_killed_enemy", type ) == 0 )
{
if ( CSGameRules()->IsPlayingGunGame() && pLocalPlayer && pLocalPlayer->GetUserID() == event->GetInt( "attackerid" ) )
{
if ( event->GetInt( "bonus" ) )
{
WITH_SLOT_LOCKED
{
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "registerBonusKill", NULL, 0 );
}
}
}
}
}
void SFUniqueAlerts::Show( void )
{
if ( !m_pScaleformUI || !m_FlashAPI )
return;
m_bVisible = true;
#if defined ( ENABLE_GIFT_PANEL )
m_flLastGiftPanelUpdate = gpGlobals->curtime;
#endif
ScaleformDisplayInfo dinfo;
m_pScaleformUI->Value_GetDisplayInfo( m_FlashAPI, &dinfo );
dinfo.SetVisibility( true );
m_pScaleformUI->Value_SetDisplayInfo( m_FlashAPI, &dinfo );
}
void SFUniqueAlerts::Hide( void )
{
m_bVisible = false;
if ( !m_pScaleformUI || !m_FlashAPI)
return;
HideAlertText();
//Msg( "------------------------ SFUniqueAlerts::Hide\n" );
ScaleformDisplayInfo dinfo;
m_pScaleformUI->Value_GetDisplayInfo( m_FlashAPI, &dinfo );
dinfo.SetVisibility( false );
m_pScaleformUI->Value_SetDisplayInfo( m_FlashAPI, &dinfo );
}
void SFUniqueAlerts::Reset( void )
{
}
void SFUniqueAlerts::UpdateGGWeaponIconList( void )
{
C_CSPlayer *pPlayer = (C_CSPlayer::GetLocalCSPlayer() && C_CSPlayer::GetLocalCSPlayer()->GetObserverTarget()) ? dynamic_cast<C_CSPlayer*>(C_CSPlayer::GetLocalCSPlayer()->GetObserverTarget()) : C_CSPlayer::GetLocalCSPlayer();
if ( !pPlayer )
return;
int nMaxIndex = CSGameRules()->GetNumProgressiveGunGameWeapons( pPlayer->GetTeamNumber() );
int nextWeaponID = -1;
for ( int i = 0; i < nMaxIndex; i++ )
{
nextWeaponID = CSGameRules()->GetProgressiveGunGameWeapon( i, pPlayer->GetTeamNumber() );
const char *pchClassName = "unknown";
const char *pchPrintName = "unknown";
const CEconItemDefinition *pDef = GetItemSchema()->GetItemDefinition( nextWeaponID );
if ( pDef && pDef->GetDefinitionIndex() != 0 )
{
pchClassName = pDef->GetDefinitionName();
pchPrintName = pDef->GetItemBaseName();
}
else
{
const CCSWeaponInfo* pWeaponInfo = GetWeaponInfo( (CSWeaponID)nextWeaponID );
if ( !pWeaponInfo )
continue;
pchClassName = pWeaponInfo->szClassName;
pchPrintName = pWeaponInfo->szPrintName;
}
bool bSelected = pPlayer->GetPlayerGunGameWeaponIndex() == i;
WITH_SFVALUEARRAY_SLOT_LOCKED( data, 4 )
{
m_pScaleformUI->ValueArray_SetElement( data, 0, i );
m_pScaleformUI->ValueArray_SetElement( data, 1, pchClassName );
m_pScaleformUI->ValueArray_SetElement( data, 2, pchPrintName );
m_pScaleformUI->ValueArray_SetElement( data, 3, bSelected );
m_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "setWeaponsIconsAndShow", data, 4 );
}
}
}
void SFUniqueAlerts::ShowWarmupAlertPanel( void )
{
C_CSPlayer *pPlayer = C_CSPlayer::GetLocalCSPlayer();
if ( !pPlayer || !CSGameRules() || !m_pScaleformUI )
return;
if ( !m_bVisible || !m_bActive )
{
//DevMsg( "SFUniqueAlerts:ShowWarmupAlertPanel : RETURNING : m_bVisible == %s, m_bActive == %s\n", m_bVisible ? "true" : "false", m_bActive ? "true" : "false" );
return;
}
float flEndTime = CSGameRules()->GetWarmupPeriodEndTime() - 0.5f;
bool bIsRestarting = CSGameRules()->IsGameRestarting();
int nTimeLeftInSec = (int)flEndTime - (int)gpGlobals->curtime;
if ( nTimeLeftInSec > 0 )
{
int nMinLeft = nTimeLeftInSec / 60;
int nSecLeft = nTimeLeftInSec - ( nMinLeft * 60 );
wchar_t szNotice[64] = L"";
wchar_t wzTime[8] = L"";
if ( !CSGameRules()->IsWarmupPeriodPaused() )
{
V_swprintf_safe( wzTime, L"%d:%02d", nMinLeft, nSecLeft );
}
if ( nTimeLeftInSec <= 5 )
{
g_pVGuiLocalize->ConstructString( szNotice, sizeof( szNotice ), g_pVGuiLocalize->Find( "#SFUI_Notice_Alert_Warmup_Period_Ending" ), 1, wzTime );
ShowAlertText( szNotice );
if ( !bIsRestarting )
pPlayer->EmitSound("Alert.WarmupTimeoutBeep");
}
else
{
if ( CSGameRules()->IsQueuedMatchmaking() )
{
// client-side UTIL_HumansInGame
int nTotalPlayers = 0;
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
{
CCSPlayer *pPlayer = ( CCSPlayer* )UTIL_PlayerByIndex( i );
if ( pPlayer && !pPlayer->IsSpectator() && !pPlayer->IsBot() )
nTotalPlayers++;
}
int numHumansNeeded = 10;
if ( CSGameRules()->IsPlayingCoopGuardian() || CSGameRules()->IsPlayingCoopMission() )
numHumansNeeded = 2;
if ( nTotalPlayers < numHumansNeeded )
{
g_pVGuiLocalize->ConstructString( szNotice, sizeof( szNotice ), g_pVGuiLocalize->Find( "#SFUI_Notice_Alert_Waiting_For_Players" ), 1, wzTime );
}
else
{
g_pVGuiLocalize->ConstructString( szNotice, sizeof( szNotice ), g_pVGuiLocalize->Find( "#SFUI_Notice_Alert_Warmup_Period" ), 1, wzTime );
}
}
else
{
g_pVGuiLocalize->ConstructString( szNotice, sizeof( szNotice ), g_pVGuiLocalize->Find( "#SFUI_Notice_Alert_Warmup_Period" ), 1, wzTime );
}
ShowAlertText( szNotice );
}
}
}
void SFUniqueAlerts::ShowHltvReplayAlertPanel( int nHltvReplayLeft )
{
wchar_t szNotice[ 128 ] = L"";
wchar_t wzSecs[ 16 ] = L"";
if ( IsDebug() )
{
Msg( "ShowHltvReplayAlertPanel\n" );
V_swprintf_safe( wzSecs, L"%d @ frame %d", nHltvReplayLeft, gpGlobals->tickcount );
}
else if ( nHltvReplayLeft > 0 ) // 0 may mean we either haven't started streaming yet (time hasn't warped back yet), or have almost finished streaming (to the warped point or beyond), or the tick rate is invalid so our timeout computations are invalid, or that we're playing past the point we recorded as the end of playback... either way, there's no need to show that to the user
{
V_swprintf_safe( wzSecs, L"%d", nHltvReplayLeft );
}
g_pVGuiLocalize->ConstructString( szNotice, sizeof( szNotice ), g_pVGuiLocalize->Find( "#SFUI_Notice_Alert_Replaying" ), 1, wzSecs );
ShowAlertText( szNotice );
}
void SFUniqueAlerts::ShowAlertText( const wchar_t *szMsg, bool oneShot )
{
if ( !m_FlashAPI || !m_pScaleformUI )
return;
WITH_SFVALUEARRAY_SLOT_LOCKED( data, 1 )
{
m_pScaleformUI->ValueArray_SetElement( data, 0, szMsg );
if ( oneShot )
{
m_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "setAlertTextAndShow", data, 1 );
m_AlertState = ALERTSTATE_HIDING; // animation will automatically hide at some point
}
else if ( m_AlertState == ALERTSTATE_SHOWING )
{
m_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "updateAlertText", data, 1 );
}
else // either HIDING or HIDDEN, need to switch to 'new message' animation
{
m_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "setAlertTextAndShowPause", data, 1 );
m_AlertState = ALERTSTATE_SHOWING;
}
}
}
void SFUniqueAlerts::HideAlertText()
{
if(m_AlertState != ALERTSTATE_HIDDEN)
{
WITH_SFVALUEARRAY_SLOT_LOCKED( data, 1 )
{
g_pScaleformUI->Value_InvokeWithoutReturn( m_FlashAPI, "hideFlash", NULL, 0 );
}
m_AlertState = ALERTSTATE_HIDING;
}
}
// Scaleform callbacks
void SFUniqueAlerts::OnAlertPanelHideAnimEnd( SCALEFORM_CALLBACK_ARGS_DECL )
{
m_AlertState = ALERTSTATE_HIDDEN;
}
#endif // INCLUDE_SCALEFORM