hlsdk-portable/cl_dll/hud.cpp

617 lines
13 KiB
C++
Raw Normal View History

2016-06-04 13:24:23 +00:00
/***
*
* 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
//
#include "hud.h"
#include "cl_util.h"
#include <string.h>
#include <stdio.h>
#include "parsemsg.h"
#include "hud_servers.h"
#include "demo.h"
#include "demo_api.h"
cvar_t *hud_textmode;
float g_hud_text_color[3];
2016-07-03 13:39:55 +00:00
extern client_sprite_t *GetSpriteList( client_sprite_t *pList, const char *psz, int iRes, int iCount );
2016-06-04 13:24:23 +00:00
extern cvar_t *sensitivity;
cvar_t *cl_lw = NULL;
cvar_t *cl_viewbob = NULL;
2020-04-08 07:17:35 +00:00
cvar_t *adjust_fov = NULL;
2016-06-04 13:24:23 +00:00
2016-07-03 13:39:55 +00:00
void ShutdownInput( void );
2016-06-04 13:24:23 +00:00
2016-07-03 13:39:55 +00:00
//DECLARE_MESSAGE( m_Logo, Logo )
int __MsgFunc_Logo( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
2016-07-03 13:39:55 +00:00
return gHUD.MsgFunc_Logo( pszName, iSize, pbuf );
2016-06-04 13:24:23 +00:00
}
2016-07-03 13:39:55 +00:00
//DECLARE_MESSAGE( m_Logo, Logo )
int __MsgFunc_ResetHUD( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
2016-07-03 13:39:55 +00:00
return gHUD.MsgFunc_ResetHUD( pszName, iSize, pbuf );
2016-06-04 13:24:23 +00:00
}
2016-07-03 13:39:55 +00:00
int __MsgFunc_InitHUD( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
gHUD.MsgFunc_InitHUD( pszName, iSize, pbuf );
return 1;
}
2016-07-03 13:39:55 +00:00
int __MsgFunc_ViewMode( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
gHUD.MsgFunc_ViewMode( pszName, iSize, pbuf );
return 1;
}
2016-07-03 13:39:55 +00:00
int __MsgFunc_SetFOV( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
return gHUD.MsgFunc_SetFOV( pszName, iSize, pbuf );
}
2016-07-03 13:39:55 +00:00
int __MsgFunc_Concuss( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
return gHUD.MsgFunc_Concuss( pszName, iSize, pbuf );
}
2016-07-03 13:39:55 +00:00
int __MsgFunc_GameMode( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
return gHUD.MsgFunc_GameMode( pszName, iSize, pbuf );
}
int __MsgFunc_PlayMP3( const char *pszName, int iSize, void *pbuf )
2016-08-27 19:39:01 +00:00
{
const char *pszSound;
char cmd[64], path[64];
int loop;
BEGIN_READ( pbuf, iSize );
pszSound = READ_STRING();
loop = READ_BYTE();
sprintf( path, "sound/%s", pszSound );
2018-10-25 03:06:16 +00:00
if( !IsXashFWGS() && gEngfuncs.pfnGetCvarPointer( "gl_overbright" ) )
{
sprintf( cmd, "mp3 play %s\n", path );
gEngfuncs.pfnClientCmd( cmd );
}
else
gEngfuncs.pfnPrimeMusicStream( path, loop );
return 1;
2016-08-27 19:39:01 +00:00
}
2016-06-04 13:24:23 +00:00
// TFFree Command Menu
2016-07-03 13:39:55 +00:00
void __CmdFunc_OpenCommandMenu( void )
2016-06-04 13:24:23 +00:00
{
}
// TFC "special" command
2016-07-03 13:39:55 +00:00
void __CmdFunc_InputPlayerSpecial( void )
2016-06-04 13:24:23 +00:00
{
}
2016-07-03 13:39:55 +00:00
void __CmdFunc_CloseCommandMenu( void )
2016-06-04 13:24:23 +00:00
{
}
void __CmdFunc_ForceCloseCommandMenu( void )
{
}
void __CmdFunc_ToggleServerBrowser( void )
{
}
// TFFree Command Menu Message Handlers
2016-07-03 13:39:55 +00:00
int __MsgFunc_ValClass( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
return 0;
}
2016-07-03 13:39:55 +00:00
int __MsgFunc_TeamNames( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
return 0;
}
2016-07-03 13:39:55 +00:00
int __MsgFunc_Feign( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
return 0;
}
2016-07-03 13:39:55 +00:00
int __MsgFunc_Detpack( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
return 0;
}
2016-07-03 13:39:55 +00:00
int __MsgFunc_VGUIMenu( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
return 0;
}
2016-07-03 13:39:55 +00:00
int __MsgFunc_BuildSt( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
return 0;
}
2016-07-03 13:39:55 +00:00
int __MsgFunc_RandomPC( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
return 0;
}
2016-07-03 13:39:55 +00:00
int __MsgFunc_ServerName( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
return 0;
}
2016-07-03 13:39:55 +00:00
int __MsgFunc_Spectator( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
return 0;
}
2016-07-03 13:39:55 +00:00
int __MsgFunc_AllowSpec( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
return 0;
}
// This is called every time the DLL is loaded
2016-07-03 13:39:55 +00:00
void CHud::Init( void )
2016-06-04 13:24:23 +00:00
{
HOOK_MESSAGE( Logo );
HOOK_MESSAGE( ResetHUD );
HOOK_MESSAGE( GameMode );
HOOK_MESSAGE( InitHUD );
HOOK_MESSAGE( ViewMode );
HOOK_MESSAGE( SetFOV );
HOOK_MESSAGE( Concuss );
HOOK_MESSAGE( PlayMP3 );
2016-06-04 13:24:23 +00:00
// TFFree CommandMenu
HOOK_COMMAND( "+commandmenu", OpenCommandMenu );
HOOK_COMMAND( "-commandmenu", CloseCommandMenu );
HOOK_COMMAND( "ForceCloseCommandMenu", ForceCloseCommandMenu );
HOOK_COMMAND( "special", InputPlayerSpecial );
HOOK_COMMAND( "togglebrowser", ToggleServerBrowser );
HOOK_MESSAGE( ValClass );
HOOK_MESSAGE( TeamNames );
HOOK_MESSAGE( Feign );
HOOK_MESSAGE( Detpack );
HOOK_MESSAGE( BuildSt );
HOOK_MESSAGE( RandomPC );
HOOK_MESSAGE( ServerName );
HOOK_MESSAGE( Spectator );
HOOK_MESSAGE( AllowSpec );
// 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;
2019-11-03 19:05:51 +00:00
CVAR_CREATE( "zoom_sensitivity_ratio", "1.2", FCVAR_ARCHIVE );
CVAR_CREATE( "cl_autowepswitch", "1", FCVAR_ARCHIVE | FCVAR_USERINFO );
2019-11-03 19:05:51 +00:00
default_fov = CVAR_CREATE( "default_fov", "90", FCVAR_ARCHIVE );
2016-06-04 13:24:23 +00:00
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 );
2020-04-08 07:17:35 +00:00
adjust_fov = gEngfuncs.pfnGetCvarPointer( "r_adjust_fov" ); // Xash3D widescreen fix
2016-06-04 13:24:23 +00:00
m_pSpriteList = NULL;
// Clear any old HUD list
2016-07-03 13:39:55 +00:00
if( m_pHudList )
2016-06-04 13:24:23 +00:00
{
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_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_MOTD.Init();
m_Scoreboard.Init();
m_Menu.Init();
2016-07-03 13:39:55 +00:00
MsgFunc_ResetHUD( 0, 0, NULL );
2016-06-04 13:24:23 +00:00
}
// CHud destructor
// cleans up memory allocated for m_rg* arrays
2016-07-03 13:39:55 +00:00
CHud::~CHud()
2016-06-04 13:24:23 +00:00
{
2016-07-03 13:39:55 +00:00
delete[] m_rghSprites;
delete[] m_rgrcRects;
delete[] m_rgszSpriteNames;
2016-06-04 13:24:23 +00:00
2016-07-03 13:39:55 +00:00
if( m_pHudList )
2016-06-04 13:24:23 +00:00
{
HUDLIST *pList;
2016-07-03 13:39:55 +00:00
while( m_pHudList )
2016-06-04 13:24:23 +00:00
{
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
2016-07-03 13:39:55 +00:00
int CHud::GetSpriteIndex( const char *SpriteName )
2016-06-04 13:24:23 +00:00
{
// look through the loaded sprite name list for SpriteName
2016-07-03 13:39:55 +00:00
for( int i = 0; i < m_iSpriteCount; i++ )
2016-06-04 13:24:23 +00:00
{
2016-07-03 13:39:55 +00:00
if( strncmp( SpriteName, m_rgszSpriteNames + ( i * MAX_SPRITE_NAME_LENGTH), MAX_SPRITE_NAME_LENGTH ) == 0 )
2016-06-04 13:24:23 +00:00
return i;
}
return -1; // invalid sprite
}
2016-07-03 13:39:55 +00:00
void CHud::VidInit( void )
2016-06-04 13:24:23 +00:00
{
2016-04-17 20:30:05 +00:00
int j;
2016-06-04 13:24:23 +00:00
m_scrinfo.iSize = sizeof(m_scrinfo);
2016-07-03 13:39:55 +00:00
GetScreenInfo( &m_scrinfo );
2016-06-04 13:24:23 +00:00
// ----------
// Load Sprites
// ---------
2016-07-03 13:39:55 +00:00
//m_hsprFont = LoadSprite("sprites/%d_font.spr");
2016-06-04 13:24:23 +00:00
m_hsprLogo = 0;
m_hsprCursor = 0;
2016-07-03 13:39:55 +00:00
if( ScreenWidth < 640 )
2016-06-04 13:24:23 +00:00
m_iRes = 320;
else
m_iRes = 640;
// Only load this once
2016-07-03 13:39:55 +00:00
if( !m_pSpriteList )
2016-06-04 13:24:23 +00:00
{
// we need to load the hud.txt, and all sprites within
2016-07-03 13:39:55 +00:00
m_pSpriteList = SPR_GetList( "sprites/hud.txt", &m_iSpriteCountAllRes );
2016-06-04 13:24:23 +00:00
2016-07-03 13:39:55 +00:00
if( m_pSpriteList )
2016-06-04 13:24:23 +00:00
{
// count the number of sprites of the appropriate res
m_iSpriteCount = 0;
client_sprite_t *p = m_pSpriteList;
2016-07-03 13:39:55 +00:00
for( j = 0; j < m_iSpriteCountAllRes; j++ )
2016-06-04 13:24:23 +00:00
{
2016-07-03 13:39:55 +00:00
if( p->iRes == m_iRes )
2016-06-04 13:24:23 +00:00
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;
2016-07-03 13:39:55 +00:00
for( j = 0; j < m_iSpriteCountAllRes; j++ )
2016-06-04 13:24:23 +00:00
{
2016-07-03 13:39:55 +00:00
if( p->iRes == m_iRes )
2016-06-04 13:24:23 +00:00
{
char sz[256];
2016-07-03 13:39:55 +00:00
sprintf( sz, "sprites/%s.spr", p->szSprite );
m_rghSprites[index] = SPR_Load( sz );
2016-06-04 13:24:23 +00:00
m_rgrcRects[index] = p->rc;
strncpy( &m_rgszSpriteNames[index * MAX_SPRITE_NAME_LENGTH], p->szName, MAX_SPRITE_NAME_LENGTH );
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;
2016-07-03 13:39:55 +00:00
for( j = 0; j < m_iSpriteCountAllRes; j++ )
2016-06-04 13:24:23 +00:00
{
2016-07-03 13:39:55 +00:00
if( p->iRes == m_iRes )
2016-06-04 13:24:23 +00:00
m_iSpriteCount++;
p++;
}
2016-07-03 13:39:55 +00:00
delete[] m_rghSprites;
delete[] m_rgrcRects;
delete[] m_rgszSpriteNames;
2016-06-04 13:24:23 +00:00
// 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;
2016-07-03 13:39:55 +00:00
for( j = 0; j < m_iSpriteCountAllRes; j++ )
2016-06-04 13:24:23 +00:00
{
2016-07-03 13:39:55 +00:00
if( p->iRes == m_iRes )
2016-06-04 13:24:23 +00:00
{
char sz[256];
sprintf( sz, "sprites/%s.spr", p->szSprite );
2016-07-03 13:39:55 +00:00
m_rghSprites[index] = SPR_Load( sz );
2016-06-04 13:24:23 +00:00
m_rgrcRects[index] = p->rc;
strncpy( &m_rgszSpriteNames[index * MAX_SPRITE_NAME_LENGTH], p->szName, MAX_SPRITE_NAME_LENGTH );
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;
}
2016-06-04 13:24:23 +00:00
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_Scoreboard.VidInit();
m_MOTD.VidInit();
}
2016-07-03 13:39:55 +00:00
int CHud::MsgFunc_Logo( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
BEGIN_READ( pbuf, iSize );
// update Train data
m_iLogo = READ_BYTE();
return 1;
}
float g_lastFOV = 0.0;
/*
============
COM_FileBase
============
*/
// Extracts the base name of a file (no path, no extension, assumes '/' as path separator)
2016-07-03 13:39:55 +00:00
void COM_FileBase ( const char *in, char *out )
2016-06-04 13:24:23 +00:00
{
int len, start, end;
len = strlen( in );
2016-07-03 13:39:55 +00:00
2016-06-04 13:24:23 +00:00
// scan backward for '.'
end = len - 1;
2016-07-03 13:39:55 +00:00
while( end && in[end] != '.' && in[end] != '/' && in[end] != '\\' )
2016-06-04 13:24:23 +00:00
end--;
2016-07-03 13:39:55 +00:00
if( in[end] != '.' ) // no '.', copy to end
end = len - 1;
2016-06-04 13:24:23 +00:00
else
end--; // Found ',', copy to left of '.'
// Scan backward for '/'
2016-07-03 13:39:55 +00:00
start = len - 1;
while( start >= 0 && in[start] != '/' && in[start] != '\\' )
2016-06-04 13:24:23 +00:00
start--;
2016-07-03 13:39:55 +00:00
if( in[start] != '/' && in[start] != '\\' )
2016-06-04 13:24:23 +00:00
start = 0;
else
start++;
// Length of new sting
len = end - start + 1;
// Copy partial string
strncpy( out, &in[start], len );
2016-07-03 13:39:55 +00:00
2016-06-04 13:24:23 +00:00
// Terminate it
out[len] = 0;
}
/*
=================
HUD_IsGame
=================
*/
int HUD_IsGame( const char *game )
{
const char *gamedir;
2016-07-03 13:39:55 +00:00
char gd[1024];
2016-06-04 13:24:23 +00:00
gamedir = gEngfuncs.pfnGetGameDirectory();
2016-07-03 13:39:55 +00:00
if( gamedir && gamedir[0] )
2016-06-04 13:24:23 +00:00
{
COM_FileBase( gamedir, gd );
2016-07-03 13:39:55 +00:00
if( !stricmp( gd, game ) )
2016-06-04 13:24:23 +00:00
return 1;
}
return 0;
}
/*
=====================
HUD_GetFOV
Returns last FOV
=====================
*/
float HUD_GetFOV( void )
{
2016-07-03 13:39:55 +00:00
if( gEngfuncs.pDemoAPI->IsRecording() )
2016-06-04 13:24:23 +00:00
{
// Write it
int i = 0;
2016-07-03 13:39:55 +00:00
unsigned char buf[100];
2016-06-04 13:24:23 +00:00
// Active
2016-07-03 13:39:55 +00:00
*(float *)&buf[i] = g_lastFOV;
i += sizeof(float);
2016-06-04 13:24:23 +00:00
Demo_WriteBuffer( TYPE_ZOOM, i, buf );
}
2016-07-03 13:39:55 +00:00
if( gEngfuncs.pDemoAPI->IsPlayingback() )
2016-06-04 13:24:23 +00:00
{
g_lastFOV = g_demozoom;
}
return g_lastFOV;
}
2016-07-03 13:39:55 +00:00
int CHud::MsgFunc_SetFOV( const char *pszName, int iSize, void *pbuf )
2016-06-04 13:24:23 +00:00
{
BEGIN_READ( pbuf, iSize );
int newfov = READ_BYTE();
int def_fov = CVAR_GET_FLOAT( "default_fov" );
g_lastFOV = newfov;
2016-07-03 13:39:55 +00:00
if( newfov == 0 )
2016-06-04 13:24:23 +00:00
{
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
2016-07-03 13:39:55 +00:00
if( m_iFOV == def_fov )
2016-06-04 13:24:23 +00:00
{
// 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;
}
2016-07-03 13:39:55 +00:00
void CHud::AddHudElem( CHudBase *phudelem )
2016-06-04 13:24:23 +00:00
{
HUDLIST *pdl, *ptemp;
2016-07-03 13:39:55 +00:00
//phudelem->Think();
2016-06-04 13:24:23 +00:00
2016-07-03 13:39:55 +00:00
if( !phudelem )
2016-06-04 13:24:23 +00:00
return;
2016-07-03 13:39:55 +00:00
pdl = (HUDLIST *)malloc( sizeof(HUDLIST) );
if( !pdl )
2016-06-04 13:24:23 +00:00
return;
2016-07-03 13:39:55 +00:00
memset( pdl, 0, sizeof(HUDLIST) );
2016-06-04 13:24:23 +00:00
pdl->p = phudelem;
2016-07-03 13:39:55 +00:00
if( !m_pHudList )
2016-06-04 13:24:23 +00:00
{
m_pHudList = pdl;
return;
}
ptemp = m_pHudList;
2016-07-03 13:39:55 +00:00
while( ptemp->pNext )
2016-06-04 13:24:23 +00:00
ptemp = ptemp->pNext;
ptemp->pNext = pdl;
}
float CHud::GetSensitivity( void )
{
return m_flMouseSensitivity;
}
2016-08-27 19:39:01 +00:00