/*** * * Copyright (c) 1996-2002, Valve LLC. All rights reserved. * * This product contains software technology licensed from Id * Software, Inc. ("Id Technology"). Id Technology (c) 1996 Id Software, Inc. * All Rights Reserved. * * Use, distribution, and modification of this source code and/or resulting * object code is restricted to non-commercial enhancements to products from * Valve LLC. All other use, distribution, or modification is prohibited * without written permission from Valve LLC. * ****/ // // hud.cpp // // implementation of CHud class // //LRC - define to help track what calls are made on changelevel, save/restore, etc //#define ENGINE_DEBUG #include "hud.h" #include "cl_util.h" #include #include #include "parsemsg.h" #if USE_VGUI #include "vgui_int.h" #include "vgui_TeamFortressViewport.h" #endif #include "demo.h" #include "demo_api.h" hud_player_info_t g_PlayerInfoList[MAX_PLAYERS+1]; // player info from the engine extra_player_info_t g_PlayerExtraInfo[MAX_PLAYERS+1]; // additional player info sent directly to the client dll team_info_t g_TeamInfo[MAX_TEAMS + 1]; int g_IsSpectator[MAX_PLAYERS+1]; int g_iPlayerClass; int g_iTeamNumber; int g_iUser1 = 0; int g_iUser2 = 0; int g_iUser3 = 0; #if USE_VGUI #include "vgui_ScorePanel.h" class CHLVoiceStatusHelper : public IVoiceStatusHelper { public: virtual void GetPlayerTextColor(int entindex, int color[3]) { color[0] = color[1] = color[2] = 255; if( entindex >= 0 && entindex < sizeof(g_PlayerExtraInfo)/sizeof(g_PlayerExtraInfo[0]) ) { int iTeam = g_PlayerExtraInfo[entindex].teamnumber; if ( iTeam < 0 ) { iTeam = 0; } iTeam = iTeam % iNumberOfTeamColors; color[0] = iTeamColors[iTeam][0]; color[1] = iTeamColors[iTeam][1]; color[2] = iTeamColors[iTeam][2]; } } virtual void UpdateCursorState() { gViewPort->UpdateCursorState(); } virtual int GetAckIconHeight() { return ScreenHeight - gHUD.m_iFontHeight*3 - 6; } virtual bool CanShowSpeakerLabels() { if( gViewPort && gViewPort->m_pScoreBoard ) return !gViewPort->m_pScoreBoard->isVisible(); else return false; } }; static CHLVoiceStatusHelper g_VoiceStatusHelper; #endif cvar_t *hud_textmode; float g_hud_text_color[3]; extern client_sprite_t *GetSpriteList( client_sprite_t *pList, const char *psz, int iRes, int iCount ); extern cvar_t *sensitivity; cvar_t *cl_lw = NULL; cvar_t *cl_viewbob = NULL; void ShutdownInput( void ); //DECLARE_MESSAGE( m_Logo, Logo ) int __MsgFunc_Logo( const char *pszName, int iSize, void *pbuf ) { return gHUD.MsgFunc_Logo( pszName, iSize, pbuf ); } //DECLARE_MESSAGE( m_Logo, Logo ) //LRC int __MsgFunc_HUDColor(const char *pszName, int iSize, void *pbuf) { return gHUD.MsgFunc_HUDColor(pszName, iSize, pbuf ); } //LRC int __MsgFunc_SetFog(const char *pszName, int iSize, void *pbuf) { gHUD.MsgFunc_SetFog( pszName, iSize, pbuf ); return 1; } //LRC int __MsgFunc_KeyedDLight(const char *pszName, int iSize, void *pbuf) { gHUD.MsgFunc_KeyedDLight( pszName, iSize, pbuf ); return 1; } //LRC int __MsgFunc_AddShine(const char *pszName, int iSize, void *pbuf) { gHUD.MsgFunc_AddShine( pszName, iSize, pbuf ); return 1; } //LRC int __MsgFunc_SetSky(const char *pszName, int iSize, void *pbuf) { gHUD.MsgFunc_SetSky( pszName, iSize, pbuf ); return 1; } int __MsgFunc_ResetHUD( const char *pszName, int iSize, void *pbuf ) { #ifdef ENGINE_DEBUG CONPRINT("## ResetHUD\n"); #endif return gHUD.MsgFunc_ResetHUD( pszName, iSize, pbuf ); } int __MsgFunc_InitHUD( const char *pszName, int iSize, void *pbuf ) { #ifdef ENGINE_DEBUG CONPRINT("## InitHUD\n"); #endif gHUD.MsgFunc_InitHUD( pszName, iSize, pbuf ); return 1; } int __MsgFunc_ViewMode( const char *pszName, int iSize, void *pbuf ) { gHUD.MsgFunc_ViewMode( pszName, iSize, pbuf ); return 1; } int __MsgFunc_SetFOV( const char *pszName, int iSize, void *pbuf ) { return gHUD.MsgFunc_SetFOV( pszName, iSize, pbuf ); } int __MsgFunc_Concuss( const char *pszName, int iSize, void *pbuf ) { return gHUD.MsgFunc_Concuss( pszName, iSize, pbuf ); } int __MsgFunc_GameMode( const char *pszName, int iSize, void *pbuf ) { return gHUD.MsgFunc_GameMode( pszName, iSize, pbuf ); } // TFFree Command Menu void __CmdFunc_OpenCommandMenu( void ) { #if USE_VGUI if ( gViewPort ) { gViewPort->ShowCommandMenu( gViewPort->m_StandardMenu ); } #endif } // TFC "special" command void __CmdFunc_InputPlayerSpecial( void ) { #if USE_VGUI if ( gViewPort ) { gViewPort->InputPlayerSpecial(); } #endif } void __CmdFunc_CloseCommandMenu( void ) { #if USE_VGUI if ( gViewPort ) { gViewPort->InputSignalHideCommandMenu(); } #endif } void __CmdFunc_ForceCloseCommandMenu( void ) { #if USE_VGUI if ( gViewPort ) { gViewPort->HideCommandMenu(); } #endif } // TFFree Command Menu Message Handlers int __MsgFunc_ValClass( const char *pszName, int iSize, void *pbuf ) { #if USE_VGUI if (gViewPort) return gViewPort->MsgFunc_ValClass( pszName, iSize, pbuf ); #endif return 0; } int __MsgFunc_TeamNames( const char *pszName, int iSize, void *pbuf ) { #if USE_VGUI if (gViewPort) return gViewPort->MsgFunc_TeamNames( pszName, iSize, pbuf ); #endif return 0; } int __MsgFunc_Feign( const char *pszName, int iSize, void *pbuf ) { #if USE_VGUI if (gViewPort) return gViewPort->MsgFunc_Feign( pszName, iSize, pbuf ); #endif return 0; } int __MsgFunc_Detpack( const char *pszName, int iSize, void *pbuf ) { #if USE_VGUI if (gViewPort) return gViewPort->MsgFunc_Detpack( pszName, iSize, pbuf ); #endif return 0; } int __MsgFunc_VGUIMenu( const char *pszName, int iSize, void *pbuf ) { #if USE_VGUI if (gViewPort) return gViewPort->MsgFunc_VGUIMenu( pszName, iSize, pbuf ); #endif return 0; } #if USE_VGUI && !USE_NOVGUI_MOTD int __MsgFunc_MOTD(const char *pszName, int iSize, void *pbuf) { if (gViewPort) return gViewPort->MsgFunc_MOTD( pszName, iSize, pbuf ); return 0; } #endif int __MsgFunc_BuildSt( const char *pszName, int iSize, void *pbuf ) { #if USE_VGUI if (gViewPort) return gViewPort->MsgFunc_BuildSt( pszName, iSize, pbuf ); #endif return 0; } int __MsgFunc_RandomPC( const char *pszName, int iSize, void *pbuf ) { #if USE_VGUI if (gViewPort) return gViewPort->MsgFunc_RandomPC( pszName, iSize, pbuf ); #endif return 0; } int __MsgFunc_ServerName( const char *pszName, int iSize, void *pbuf ) { #if USE_VGUI if (gViewPort) return gViewPort->MsgFunc_ServerName( pszName, iSize, pbuf ); #endif return 0; } #if USE_VGUI && !USE_NOVGUI_SCOREBOARD int __MsgFunc_ScoreInfo(const char *pszName, int iSize, void *pbuf) { if (gViewPort) return gViewPort->MsgFunc_ScoreInfo( pszName, iSize, pbuf ); return 0; } int __MsgFunc_TeamScore(const char *pszName, int iSize, void *pbuf) { if (gViewPort) return gViewPort->MsgFunc_TeamScore( pszName, iSize, pbuf ); return 0; } int __MsgFunc_TeamInfo(const char *pszName, int iSize, void *pbuf) { if (gViewPort) return gViewPort->MsgFunc_TeamInfo( pszName, iSize, pbuf ); return 0; } #endif int __MsgFunc_Spectator( const char *pszName, int iSize, void *pbuf ) { #if USE_VGUI if (gViewPort) return gViewPort->MsgFunc_Spectator( pszName, iSize, pbuf ); #endif return 0; } #if USE_VGUI int __MsgFunc_SpecFade(const char *pszName, int iSize, void *pbuf) { if (gViewPort) return gViewPort->MsgFunc_SpecFade( pszName, iSize, pbuf ); return 0; } int __MsgFunc_ResetFade(const char *pszName, int iSize, void *pbuf) { if (gViewPort) return gViewPort->MsgFunc_ResetFade( pszName, iSize, pbuf ); return 0; } #endif int __MsgFunc_AllowSpec( const char *pszName, int iSize, void *pbuf ) { #if USE_VGUI if (gViewPort) return gViewPort->MsgFunc_AllowSpec( pszName, iSize, pbuf ); #endif return 0; } // This is called every time the DLL is loaded void CHud::Init( void ) { #ifdef ENGINE_DEBUG CONPRINT("## CHud::Init\n"); #endif HOOK_MESSAGE( Logo ); HOOK_MESSAGE( ResetHUD ); HOOK_MESSAGE( GameMode ); HOOK_MESSAGE( InitHUD ); HOOK_MESSAGE( ViewMode ); HOOK_MESSAGE( SetFOV ); HOOK_MESSAGE( Concuss ); HOOK_MESSAGE( HUDColor ); //LRC HOOK_MESSAGE( SetFog ); //LRC HOOK_MESSAGE( KeyedDLight ); //LRC HOOK_MESSAGE( AddShine ); //LRC HOOK_MESSAGE( SetSky ); //LRC // TFFree CommandMenu HOOK_COMMAND( "+commandmenu", OpenCommandMenu ); HOOK_COMMAND( "-commandmenu", CloseCommandMenu ); HOOK_COMMAND( "ForceCloseCommandMenu", ForceCloseCommandMenu ); HOOK_COMMAND( "special", InputPlayerSpecial ); HOOK_MESSAGE( ValClass ); HOOK_MESSAGE( TeamNames ); HOOK_MESSAGE( Feign ); HOOK_MESSAGE( Detpack ); //HOOK_MESSAGE( MOTD ); HOOK_MESSAGE( BuildSt ); HOOK_MESSAGE( RandomPC ); HOOK_MESSAGE( ServerName ); //HOOK_MESSAGE( ScoreInfo ); //HOOK_MESSAGE( TeamScore ); //HOOK_MESSAGE( TeamInfo ); #if USE_VGUI && !USE_NOVGUI_MOTD HOOK_MESSAGE( MOTD ); #endif #if USE_VGUI && !USE_NOVGUI_SCOREBOARD HOOK_MESSAGE( ScoreInfo ); HOOK_MESSAGE( TeamScore ); HOOK_MESSAGE( TeamInfo ); #endif HOOK_MESSAGE( Spectator ); HOOK_MESSAGE( AllowSpec ); #if USE_VGUI HOOK_MESSAGE( SpecFade ); HOOK_MESSAGE( ResetFade ); #endif // VGUI Menus HOOK_MESSAGE( VGUIMenu ); CVAR_CREATE( "hud_classautokill", "1", FCVAR_ARCHIVE | FCVAR_USERINFO ); // controls whether or not to suicide immediately on TF class switch CVAR_CREATE( "hud_takesshots", "0", FCVAR_ARCHIVE ); // controls whether or not to automatically take screenshots at the end of a round hud_textmode = CVAR_CREATE ( "hud_textmode", "0", FCVAR_ARCHIVE ); m_iLogo = 0; m_iFOV = 0; m_iHUDColor = 0x00FFFFFF; //255,255,255 -- LRC CVAR_CREATE( "zoom_sensitivity_ratio", "1.2", FCVAR_ARCHIVE ); CVAR_CREATE( "cl_autowepswitch", "1", FCVAR_ARCHIVE | FCVAR_USERINFO ); default_fov = CVAR_CREATE( "default_fov", "90", FCVAR_ARCHIVE ); m_pCvarStealMouse = CVAR_CREATE( "hud_capturemouse", "1", FCVAR_ARCHIVE ); m_pCvarDraw = CVAR_CREATE( "hud_draw", "1", FCVAR_ARCHIVE ); cl_lw = gEngfuncs.pfnGetCvarPointer( "cl_lw" ); cl_viewbob = CVAR_CREATE( "cl_viewbob", "0", FCVAR_ARCHIVE ); m_pSpriteList = NULL; m_pShinySurface = NULL; //LRC // Clear any old HUD list if( m_pHudList ) { HUDLIST *pList; while ( m_pHudList ) { pList = m_pHudList; m_pHudList = m_pHudList->pNext; free( pList ); } m_pHudList = NULL; } // In case we get messages before the first update -- time will be valid m_flTime = 1.0; m_iNoConsolePrint = 0; m_Ammo.Init(); m_Health.Init(); m_SayText.Init(); m_Spectator.Init(); m_Geiger.Init(); m_Train.Init(); m_Battery.Init(); m_Flash.Init(); m_Message.Init(); m_StatusBar.Init(); m_DeathNotice.Init(); m_AmmoSecondary.Init(); m_TextMessage.Init(); m_StatusIcons.Init(); m_NoiseEffect.Init(); m_BlackBar.Init(); m_Particle.Init(); // (LRC) -- 30/08/02 November235: Particles to Order #if USE_VGUI GetClientVoiceMgr()->Init(&g_VoiceStatusHelper, (vgui::Panel**)&gViewPort); #endif #if !USE_VGUI || USE_NOVGUI_MOTD m_MOTD.Init(); #endif #if !USE_VGUI || USE_NOVGUI_SCOREBOARD m_Scoreboard.Init(); #endif m_Particle.Init(); // (LRC) -- 30/08/02 November235: Particles to Order m_Menu.Init(); MsgFunc_ResetHUD( 0, 0, NULL ); } // CHud destructor // cleans up memory allocated for m_rg* arrays CHud::~CHud() { #ifdef ENGINE_DEBUG CONPRINT("## CHud::destructor\n"); #endif delete[] m_rghSprites; delete[] m_rgrcRects; delete[] m_rgszSpriteNames; if( m_pHudList ) { HUDLIST *pList; while( m_pHudList ) { pList = m_pHudList; m_pHudList = m_pHudList->pNext; free( pList ); } m_pHudList = NULL; } } // GetSpriteIndex() // searches through the sprite list loaded from hud.txt for a name matching SpriteName // returns an index into the gHUD.m_rghSprites[] array // returns 0 if sprite not found int CHud::GetSpriteIndex( const char *SpriteName ) { // look through the loaded sprite name list for SpriteName for( int i = 0; i < m_iSpriteCount; i++ ) { if( strncmp( SpriteName, m_rgszSpriteNames + ( i * MAX_SPRITE_NAME_LENGTH), MAX_SPRITE_NAME_LENGTH ) == 0 ) return i; } return -1; // invalid sprite } void CHud::VidInit( void ) { #ifdef ENGINE_DEBUG CONPRINT("## CHud::VidInit\n"); #endif int j; m_scrinfo.iSize = sizeof(m_scrinfo); GetScreenInfo( &m_scrinfo ); // ---------- // Load Sprites // --------- //m_hsprFont = LoadSprite("sprites/%d_font.spr"); m_hsprLogo = 0; m_hsprCursor = 0; if( ScreenWidth < 640 ) m_iRes = 320; else m_iRes = 640; // Only load this once if( !m_pSpriteList ) { // we need to load the hud.txt, and all sprites within m_pSpriteList = SPR_GetList( "sprites/hud.txt", &m_iSpriteCountAllRes ); if( m_pSpriteList ) { // count the number of sprites of the appropriate res m_iSpriteCount = 0; client_sprite_t *p = m_pSpriteList; for( j = 0; j < m_iSpriteCountAllRes; j++ ) { if( p->iRes == m_iRes ) m_iSpriteCount++; p++; } // allocated memory for sprite handle arrays m_rghSprites = new HSPRITE[m_iSpriteCount]; m_rgrcRects = new wrect_t[m_iSpriteCount]; m_rgszSpriteNames = new char[m_iSpriteCount * MAX_SPRITE_NAME_LENGTH]; p = m_pSpriteList; int index = 0; for( j = 0; j < m_iSpriteCountAllRes; j++ ) { if( p->iRes == m_iRes ) { char sz[256]; sprintf( sz, "sprites/%s.spr", p->szSprite ); m_rghSprites[index] = SPR_Load( sz ); m_rgrcRects[index] = p->rc; strncpy( &m_rgszSpriteNames[index * MAX_SPRITE_NAME_LENGTH], p->szName, MAX_SPRITE_NAME_LENGTH - 1 ); m_rgszSpriteNames[index * MAX_SPRITE_NAME_LENGTH + ( MAX_SPRITE_NAME_LENGTH - 1 )] = '\0'; index++; } p++; } } } else { // we have already have loaded the sprite reference from hud.txt, but // we need to make sure all the sprites have been loaded (we've gone through a transition, or loaded a save game) client_sprite_t *p = m_pSpriteList; // count the number of sprites of the appropriate res m_iSpriteCount = 0; for( j = 0; j < m_iSpriteCountAllRes; j++ ) { if( p->iRes == m_iRes ) m_iSpriteCount++; p++; } delete[] m_rghSprites; delete[] m_rgrcRects; delete[] m_rgszSpriteNames; // allocated memory for sprite handle arrays m_rghSprites = new HSPRITE[m_iSpriteCount]; m_rgrcRects = new wrect_t[m_iSpriteCount]; m_rgszSpriteNames = new char[m_iSpriteCount * MAX_SPRITE_NAME_LENGTH]; p = m_pSpriteList; int index = 0; for( j = 0; j < m_iSpriteCountAllRes; j++ ) { if( p->iRes == m_iRes ) { char sz[256]; sprintf( sz, "sprites/%s.spr", p->szSprite ); m_rghSprites[index] = SPR_Load( sz ); m_rgrcRects[index] = p->rc; strncpy( &m_rgszSpriteNames[index * MAX_SPRITE_NAME_LENGTH], p->szName, MAX_SPRITE_NAME_LENGTH - 1 ); m_rgszSpriteNames[index * MAX_SPRITE_NAME_LENGTH + ( MAX_SPRITE_NAME_LENGTH - 1 )] = '\0'; index++; } p++; } } // assumption: number_1, number_2, etc, are all listed and loaded sequentially m_HUD_number_0 = GetSpriteIndex( "number_0" ); if( m_HUD_number_0 == -1 ) { const char *msg = "There is something wrong with your game data! Please, reinstall\n"; if( HUD_MessageBox( msg ) ) { gEngfuncs.pfnClientCmd( "quit\n" ); } return; } m_iFontHeight = m_rgrcRects[m_HUD_number_0].bottom - m_rgrcRects[m_HUD_number_0].top; m_Ammo.VidInit(); m_Health.VidInit(); m_Spectator.VidInit(); m_Geiger.VidInit(); m_Train.VidInit(); m_Battery.VidInit(); m_Flash.VidInit(); m_Message.VidInit(); m_StatusBar.VidInit(); m_DeathNotice.VidInit(); m_SayText.VidInit(); m_Menu.VidInit(); m_AmmoSecondary.VidInit(); m_TextMessage.VidInit(); m_StatusIcons.VidInit(); m_NoiseEffect.VidInit(); m_BlackBar.VidInit(); m_Particle.VidInit(); // (LRC) -- 30/08/02 November235: Particles to Order #if USE_VGUI GetClientVoiceMgr()->VidInit(); #endif #if !USE_VGUI || USE_NOVGUI_MOTD m_MOTD.VidInit(); #endif #if !USE_VGUI || USE_NOVGUI_SCOREBOARD m_Scoreboard.VidInit(); #endif } int CHud::MsgFunc_Logo( const char *pszName, int iSize, void *pbuf ) { BEGIN_READ( pbuf, iSize ); // update Train data m_iLogo = READ_BYTE(); return 1; } //LRC int CHud::MsgFunc_HUDColor(const char *pszName, int iSize, void *pbuf) { BEGIN_READ( pbuf, iSize ); m_iHUDColor = READ_LONG(); return 1; } float g_lastFOV = 0.0; /* ============ COM_FileBase ============ */ // Extracts the base name of a file (no path, no extension, assumes '/' as path separator) void COM_FileBase ( const char *in, char *out ) { int len, start, end; len = strlen( in ); // scan backward for '.' end = len - 1; while( end && in[end] != '.' && in[end] != '/' && in[end] != '\\' ) end--; if( in[end] != '.' ) // no '.', copy to end end = len - 1; else end--; // Found ',', copy to left of '.' // Scan backward for '/' start = len - 1; while( start >= 0 && in[start] != '/' && in[start] != '\\' ) start--; if( in[start] != '/' && in[start] != '\\' ) start = 0; else start++; // Length of new sting len = end - start + 1; // Copy partial string strncpy( out, &in[start], len ); // Terminate it out[len] = 0; } /* ================= HUD_IsGame ================= */ int HUD_IsGame( const char *game ) { const char *gamedir; char gd[1024]; gamedir = gEngfuncs.pfnGetGameDirectory(); if( gamedir && gamedir[0] ) { COM_FileBase( gamedir, gd ); if( !stricmp( gd, game ) ) return 1; } return 0; } /* ===================== HUD_GetFOV Returns last FOV ===================== */ float HUD_GetFOV( void ) { if( gEngfuncs.pDemoAPI->IsRecording() ) { // Write it int i = 0; unsigned char buf[100]; // Active *(float *)&buf[i] = g_lastFOV; i += sizeof(float); Demo_WriteBuffer( TYPE_ZOOM, i, buf ); } if( gEngfuncs.pDemoAPI->IsPlayingback() ) { g_lastFOV = g_demozoom; } return g_lastFOV; } int CHud::MsgFunc_SetFOV( const char *pszName, int iSize, void *pbuf ) { BEGIN_READ( pbuf, iSize ); int newfov = READ_BYTE(); int def_fov = CVAR_GET_FLOAT( "default_fov" ); g_lastFOV = newfov; if( newfov == 0 ) { m_iFOV = def_fov; } else { m_iFOV = newfov; } // the clients fov is actually set in the client data update section of the hud // Set a new sensitivity if( m_iFOV == def_fov ) { // reset to saved sensitivity m_flMouseSensitivity = 0; } else { // set a new sensitivity that is proportional to the change from the FOV default m_flMouseSensitivity = sensitivity->value * ((float)newfov / (float)def_fov) * CVAR_GET_FLOAT("zoom_sensitivity_ratio"); } return 1; } void CHud::AddHudElem( CHudBase *phudelem ) { HUDLIST *pdl, *ptemp; //phudelem->Think(); if( !phudelem ) return; pdl = (HUDLIST *)malloc( sizeof(HUDLIST) ); if( !pdl ) return; memset( pdl, 0, sizeof(HUDLIST) ); pdl->p = phudelem; if( !m_pHudList ) { m_pHudList = pdl; return; } ptemp = m_pHudList; while( ptemp->pNext ) ptemp = ptemp->pNext; ptemp->pNext = pdl; } float CHud::GetSensitivity( void ) { return m_flMouseSensitivity; } void CHud::GetAllPlayersInfo() { for( int i = 1; i < MAX_PLAYERS; i++ ) { GetPlayerInfo( i, &g_PlayerInfoList[i] ); if( g_PlayerInfoList[i].thisplayer ) { #if USE_VGUI if(gViewPort) gViewPort->m_pScoreBoard->m_iPlayerNum = i; #endif #if !USE_VGUI || USE_NOVGUI_SCOREBOARD m_Scoreboard.m_iPlayerNum = i; // !!!HACK: this should be initialized elsewhere... maybe gotten from the engine #endif } } }