471 lines
11 KiB
C++
471 lines
11 KiB
C++
/***
|
|
*
|
|
* 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_redraw.cpp
|
|
//
|
|
#include <math.h>
|
|
#include "hud.h"
|
|
#include "cl_util.h"
|
|
#include "triangleapi.h"
|
|
|
|
#include <string.h>
|
|
#define MAX_LOGO_FRAMES 56
|
|
|
|
int grgLogoFrame[MAX_LOGO_FRAMES] =
|
|
{
|
|
1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 13, 13, 13, 13, 13, 12, 11, 10, 9, 8, 14, 15,
|
|
16, 17, 18, 19, 20, 20, 20, 20, 20, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29,
|
|
29, 29, 29, 29, 29, 28, 27, 26, 25, 24, 30, 31
|
|
};
|
|
|
|
|
|
extern int g_iVisibleMouse;
|
|
|
|
float HUD_GetFOV( void );
|
|
|
|
extern cvar_t *sensitivity;
|
|
|
|
// Think
|
|
void CHud::Think(void)
|
|
{
|
|
int newfov;
|
|
HUDLIST *pList = m_pHudList;
|
|
|
|
while (pList)
|
|
{
|
|
if (pList->p->m_iFlags & HUD_ACTIVE)
|
|
pList->p->Think();
|
|
pList = pList->pNext;
|
|
}
|
|
|
|
newfov = HUD_GetFOV();
|
|
if ( newfov == 0 )
|
|
{
|
|
m_iFOV = default_fov->value;
|
|
}
|
|
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 == default_fov->value )
|
|
{
|
|
// 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)default_fov->value) * CVAR_GET_FLOAT("zoom_sensitivity_ratio");
|
|
}
|
|
|
|
// think about default fov
|
|
if ( m_iFOV == 0 )
|
|
{ // only let players adjust up in fov, and only if they are not overriden by something else
|
|
m_iFOV = max( default_fov->value, 90 );
|
|
}
|
|
|
|
}
|
|
|
|
// Redraw
|
|
// step through the local data, placing the appropriate graphics & text as appropriate
|
|
// returns 1 if they've changed, 0 otherwise
|
|
int CHud :: Redraw( float flTime, int intermission )
|
|
{
|
|
m_fOldTime = m_flTime; // save time of previous redraw
|
|
m_flTime = flTime;
|
|
m_flTimeDelta = (double)m_flTime - m_fOldTime;
|
|
static int m_flShotTime = 0;
|
|
|
|
// Clock was reset, reset delta
|
|
if ( m_flTimeDelta < 0 )
|
|
m_flTimeDelta = 0;
|
|
|
|
// Bring up the scoreboard during intermission
|
|
/*if (gViewPort)
|
|
{
|
|
if ( m_iIntermission && !intermission )
|
|
{
|
|
// Have to do this here so the scoreboard goes away
|
|
m_iIntermission = intermission;
|
|
gViewPort->HideCommandMenu();
|
|
gViewPort->HideScoreBoard();
|
|
gViewPort->UpdateSpectatorPanel();
|
|
}
|
|
else if ( !m_iIntermission && intermission )
|
|
{
|
|
m_iIntermission = intermission;
|
|
gViewPort->HideCommandMenu();
|
|
gViewPort->HideVGUIMenu();
|
|
gViewPort->ShowScoreBoard();
|
|
gViewPort->UpdateSpectatorPanel();
|
|
|
|
// Take a screenshot if the client's got the cvar set
|
|
if ( CVAR_GET_FLOAT( "hud_takesshots" ) != 0 )
|
|
m_flShotTime = flTime + 1.0; // Take a screenshot in a second
|
|
}
|
|
}*/
|
|
|
|
if (m_flShotTime && m_flShotTime < flTime)
|
|
{
|
|
gEngfuncs.pfnClientCmd("snapshot\n");
|
|
m_flShotTime = 0;
|
|
}
|
|
|
|
m_iIntermission = intermission;
|
|
|
|
if ( m_pCvarDraw->value )
|
|
{
|
|
HUDLIST *pList = m_pHudList;
|
|
|
|
while (pList)
|
|
{
|
|
if ( !intermission )
|
|
{
|
|
if ( (pList->p->m_iFlags & HUD_ACTIVE) && !(m_iHideHUDDisplay & HIDEHUD_ALL) )
|
|
pList->p->Draw(flTime);
|
|
}
|
|
else
|
|
{ // it's an intermission, so only draw hud elements that are set to draw during intermissions
|
|
if ( pList->p->m_iFlags & HUD_INTERMISSION )
|
|
pList->p->Draw( flTime );
|
|
}
|
|
|
|
pList = pList->pNext;
|
|
}
|
|
}
|
|
|
|
// are we in demo mode? do we need to draw the logo in the top corner?
|
|
if (m_iLogo)
|
|
{
|
|
int x, y, i;
|
|
|
|
if (m_hsprLogo == 0)
|
|
m_hsprLogo = LoadSprite("sprites/%d_logo.spr");
|
|
|
|
SPR_Set(m_hsprLogo, 250, 250, 250 );
|
|
|
|
x = SPR_Width(m_hsprLogo, 0);
|
|
x = ScreenWidth - x;
|
|
y = SPR_Height(m_hsprLogo, 0)/2;
|
|
|
|
// Draw the logo at 20 fps
|
|
int iFrame = (int)(flTime * 20) % MAX_LOGO_FRAMES;
|
|
i = grgLogoFrame[iFrame] - 1;
|
|
|
|
SPR_DrawAdditive(i, x, y, NULL);
|
|
}
|
|
|
|
/*
|
|
if ( g_iVisibleMouse )
|
|
{
|
|
void IN_GetMousePos( int *mx, int *my );
|
|
int mx, my;
|
|
|
|
IN_GetMousePos( &mx, &my );
|
|
|
|
if (m_hsprCursor == 0)
|
|
{
|
|
char sz[256];
|
|
sprintf( sz, "sprites/cursor.spr" );
|
|
m_hsprCursor = SPR_Load( sz );
|
|
}
|
|
|
|
SPR_Set(m_hsprCursor, 250, 250, 250 );
|
|
|
|
// Draw the logo at 20 fps
|
|
SPR_DrawAdditive( 0, mx, my, NULL );
|
|
}
|
|
*/
|
|
|
|
return 1;
|
|
}
|
|
|
|
void ScaleColors( int &r, int &g, int &b, int a )
|
|
{
|
|
float x = (float)a / 255;
|
|
r = (int)(r * x);
|
|
g = (int)(g * x);
|
|
b = (int)(b * x);
|
|
}
|
|
|
|
int CHud :: DrawHudString( int xpos, int ypos, int iMaxX, char *szIt, int r, int g, int b, bool drawing )
|
|
{
|
|
int first_xpos = xpos;
|
|
// draw the string until we hit the null character or a newline character
|
|
for ( ; *szIt != 0 && *szIt != '\n'; szIt++ )
|
|
{
|
|
int next = xpos + gHUD.m_scrinfo.charWidths[ (unsigned char)*szIt ]; // variable-width fonts look cool
|
|
if ( next > iMaxX )
|
|
return xpos;
|
|
|
|
if( *szIt == '\\' && *(szIt+1) != '\n' && *(szIt+1) != 0)
|
|
{
|
|
// an escape character
|
|
|
|
switch( *(++szIt) )
|
|
{
|
|
case 'y':
|
|
UnpackRGB( r, g, b, RGB_YELLOWISH );
|
|
break;
|
|
case 'w':
|
|
r = g = b = 255;
|
|
break;
|
|
case 'R':
|
|
if( drawing ) return xpos;
|
|
return DrawHudStringReverse( iMaxX, ypos, first_xpos, szIt, r, g, b, true ); // set 'drawing' to true, to stop when '\R' is catched
|
|
case 'd':
|
|
break;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
xpos += TextMessageDrawChar( xpos, ypos, *szIt, r, g, b );
|
|
}
|
|
|
|
return xpos;
|
|
}
|
|
|
|
int CHud :: DrawHudStringLen( char *szIt )
|
|
{
|
|
int l = 0;
|
|
// draw the string until we hit the null character or a newline character
|
|
for ( ; *szIt != 0 && *szIt != '\n'; szIt++ )
|
|
{
|
|
l += gHUD.m_scrinfo.charWidths[ (unsigned char)*szIt ]; // variable-width fonts look cool
|
|
}
|
|
return l;
|
|
}
|
|
|
|
int CHud :: DrawHudNumberString( int xpos, int ypos, int iMinX, int iNumber, int r, int g, int b )
|
|
{
|
|
char szString[32];
|
|
snprintf( szString, 32, "%d", iNumber );
|
|
return DrawHudStringReverse( xpos, ypos, iMinX, szString, r, g, b );
|
|
|
|
}
|
|
|
|
// draws a string from right to left (right-aligned)
|
|
int CHud :: DrawHudStringReverse( int xpos, int ypos, int iMinX, char *szString, int r, int g, int b, bool drawing )
|
|
{
|
|
int first_xpos = xpos;
|
|
|
|
|
|
// iterate throug the string in reverse
|
|
for ( signed int i = strlen(szString); i >= 0; i-- )
|
|
{
|
|
int next = xpos - gHUD.m_scrinfo.charWidths[ (unsigned char)szString[i] ]; // variable-width fonts look cool
|
|
if ( next < iMinX )
|
|
return xpos;
|
|
xpos = next;
|
|
|
|
if( szString[i + 1] == '\\' )
|
|
{
|
|
// an escape character
|
|
|
|
switch( szString[i] )
|
|
{
|
|
case 'y':
|
|
UnpackRGB( r, g, b, RGB_YELLOWISH );
|
|
break;
|
|
case 'w':
|
|
r = g = b = 255;
|
|
break;
|
|
case 'R':
|
|
if( drawing ) return xpos;
|
|
else return DrawHudString( iMinX, ypos, first_xpos, &szString[i], r, g, b, true ); // set 'drawing' to true, to stop when '\R' is catched
|
|
case 'd':
|
|
break;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
TextMessageDrawChar( xpos, ypos, szString[i], r, g, b );
|
|
}
|
|
|
|
return xpos;
|
|
}
|
|
|
|
int CHud :: DrawHudNumber( int x, int y, int iFlags, int iNumber, int r, int g, int b)
|
|
{
|
|
int iWidth = GetSpriteRect(m_HUD_number_0).right - GetSpriteRect(m_HUD_number_0).left;
|
|
int k;
|
|
|
|
if (iNumber > 0)
|
|
{
|
|
// SPR_Draw 100's
|
|
if (iNumber >= 100)
|
|
{
|
|
k = iNumber/100;
|
|
SPR_Set(GetSprite(m_HUD_number_0 + k), r, g, b );
|
|
SPR_DrawAdditive( 0, x, y, &GetSpriteRect(m_HUD_number_0 + k));
|
|
x += iWidth;
|
|
}
|
|
else if (iFlags & (DHN_3DIGITS))
|
|
{
|
|
//SPR_DrawAdditive( 0, x, y, &rc );
|
|
x += iWidth;
|
|
}
|
|
|
|
// SPR_Draw 10's
|
|
if (iNumber >= 10)
|
|
{
|
|
k = (iNumber % 100)/10;
|
|
SPR_Set(GetSprite(m_HUD_number_0 + k), r, g, b );
|
|
SPR_DrawAdditive( 0, x, y, &GetSpriteRect(m_HUD_number_0 + k));
|
|
x += iWidth;
|
|
}
|
|
else if (iFlags & (DHN_3DIGITS | DHN_2DIGITS))
|
|
{
|
|
//SPR_DrawAdditive( 0, x, y, &rc );
|
|
x += iWidth;
|
|
}
|
|
|
|
// SPR_Draw ones
|
|
k = iNumber % 10;
|
|
SPR_Set(GetSprite(m_HUD_number_0 + k), r, g, b );
|
|
SPR_DrawAdditive(0, x, y, &GetSpriteRect(m_HUD_number_0 + k));
|
|
x += iWidth;
|
|
}
|
|
else if (iFlags & DHN_DRAWZERO)
|
|
{
|
|
SPR_Set(GetSprite(m_HUD_number_0), r, g, b );
|
|
|
|
// SPR_Draw 100's
|
|
if (iFlags & (DHN_3DIGITS))
|
|
{
|
|
//SPR_DrawAdditive( 0, x, y, &rc );
|
|
x += iWidth;
|
|
}
|
|
|
|
if (iFlags & (DHN_3DIGITS | DHN_2DIGITS))
|
|
{
|
|
//SPR_DrawAdditive( 0, x, y, &rc );
|
|
x += iWidth;
|
|
}
|
|
|
|
// SPR_Draw ones
|
|
|
|
SPR_DrawAdditive( 0, x, y, &GetSpriteRect(m_HUD_number_0));
|
|
x += iWidth;
|
|
}
|
|
|
|
return x;
|
|
}
|
|
|
|
|
|
int CHud::GetNumWidth( int iNumber, int iFlags )
|
|
{
|
|
if (iFlags & (DHN_3DIGITS))
|
|
return 3;
|
|
|
|
if (iFlags & (DHN_2DIGITS))
|
|
return 2;
|
|
|
|
if (iNumber <= 0)
|
|
{
|
|
if (iFlags & (DHN_DRAWZERO))
|
|
return 1;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
if (iNumber < 10)
|
|
return 1;
|
|
|
|
if (iNumber < 100)
|
|
return 2;
|
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
void CHud::DrawDarkRectangle(int x, int y, int wide, int tall , int r, int g, int b, int a, bool drawStroke)
|
|
{
|
|
FillRGBA( x, y, wide, tall, r, g, b, a );
|
|
|
|
gEngfuncs.pTriAPI->RenderMode( kRenderTransTexture );
|
|
gEngfuncs.pTriAPI->Begin(TRI_QUADS);
|
|
gEngfuncs.pTriAPI->Color4f(r / 255.0f, g / 255.0f, b / 255.0f, a / 255.0f);
|
|
gEngfuncs.pTriAPI->Vertex3f(x * m_flScale, (y+tall)*m_flScale, 0);
|
|
gEngfuncs.pTriAPI->Vertex3f(x * m_flScale, y*m_flScale, 0);
|
|
gEngfuncs.pTriAPI->Vertex3f((x + wide)*m_flScale, y*m_flScale, 0);
|
|
gEngfuncs.pTriAPI->Vertex3f((x + wide)*m_flScale, (y+tall)*m_flScale, 0);
|
|
gEngfuncs.pTriAPI->End();
|
|
|
|
if(drawStroke)
|
|
{
|
|
// TODO: remove this hardcoded hardcore
|
|
FillRGBA( x+1, y, wide-1, 1, 255, 140, 0, 255 );
|
|
FillRGBA( x, y, 1, tall-1, 255, 140, 0, 255 );
|
|
FillRGBA( x+wide-1, y+1, 1, tall-1, 255, 140, 0, 255 );
|
|
FillRGBA( x, y+tall-1, wide-1, 1, 255, 140, 0, 255 );
|
|
}
|
|
}
|
|
|
|
// TANKIST START
|
|
int CHud :: DrawHudNumber2( int x, int y, bool DrawZero, int iDigits, int iNumber, int r, int g, int b)
|
|
{
|
|
int iWidth = GetSpriteRect( m_HUD_number_0 ).right - GetSpriteRect( m_HUD_number_0 ).left;
|
|
x += ( iDigits - 1 ) * iWidth;
|
|
|
|
int ResX = x + iWidth;
|
|
do
|
|
{
|
|
int k = iNumber % 10;
|
|
iNumber /= 10;
|
|
SPR_Set( GetSprite( m_HUD_number_0 + k ), r, g, b );
|
|
SPR_DrawAdditive( 0, x, y, &GetSpriteRect( m_HUD_number_0 + k ) );
|
|
x -= iWidth;
|
|
iDigits--;
|
|
}
|
|
while( iNumber > 0 || ( iDigits > 0 && DrawZero ) );
|
|
|
|
return ResX;
|
|
}
|
|
|
|
int CHud :: DrawHudNumber2( int x, int y, int iNumber, int r, int g, int b)
|
|
{
|
|
int iWidth = GetSpriteRect( m_HUD_number_0 ).right - GetSpriteRect( m_HUD_number_0 ).left;
|
|
|
|
int iDigits = 0;
|
|
int temp = iNumber;
|
|
do
|
|
{
|
|
iDigits++;
|
|
temp /= 10;
|
|
}
|
|
while( temp > 0 );
|
|
|
|
x += ( iDigits - 1 ) * iWidth;
|
|
|
|
int ResX = x + iWidth;
|
|
do
|
|
{
|
|
int k = iNumber % 10;
|
|
iNumber /= 10;
|
|
SPR_Set( GetSprite( m_HUD_number_0 + k ), r, g, b );
|
|
SPR_DrawAdditive( 0, x, y, &GetSpriteRect( m_HUD_number_0 + k ) );
|
|
x -= iWidth;
|
|
}
|
|
while( iNumber > 0 );
|
|
|
|
return ResX;
|
|
}
|
|
// TANKIST END
|