2023-10-03 17:23:56 +03:00
//========= Copyright 1996-2005, Valve Corporation, All rights reserved. ============//
2020-04-22 12:56:21 -04:00
//
// Purpose:
//
// $NoKeywords: $
//=============================================================================//
# include "cbase.h"
# include "hud_basechat.h"
# include <vgui/IScheme.h>
# include <vgui/IVGui.h>
# include "iclientmode.h"
# include "hud_macros.h"
# include "engine/IEngineSound.h"
# include "text_message.h"
# include <vgui/ILocalize.h>
# include "vguicenterprint.h"
2023-10-03 17:23:56 +03:00
# include "vgui/keycode.h"
2020-04-22 12:56:21 -04:00
# include <KeyValues.h>
# include "ienginevgui.h"
# include "c_playerresource.h"
# include "ihudlcd.h"
# include "vgui/IInput.h"
# include "vgui/ILocalize.h"
# include "multiplay_gamerules.h"
2023-10-03 17:23:56 +03:00
# include "time.h"
# include "filesystem.h"
# include "vgui_int.h"
# if defined( _X360 )
# include "xbox/xbox_win32stubs.h"
# endif
2020-04-22 12:56:21 -04:00
// memdbgon must be the last include file in a .cpp file!!!
# include "tier0/memdbgon.h"
# define CHAT_WIDTH_PERCENTAGE 0.6f
ConVar hud_saytext_time ( " hud_saytext_time " , " 12 " , 0 ) ;
ConVar cl_showtextmsg ( " cl_showtextmsg " , " 1 " , 0 , " Enable/disable text messages printing on the screen. " ) ;
2023-10-03 17:23:56 +03:00
ConVar cl_chat_active ( " cl_chat_active " , " 0 " ) ;
ConVar cl_chatfilters ( " cl_chatfilters " , " 31 " , FCVAR_CLIENTDLL | FCVAR_ARCHIVE , " Stores the chat filter settings " ) ;
2020-04-22 12:56:21 -04:00
Color g_ColorBlue ( 153 , 204 , 255 , 255 ) ;
2023-10-03 17:23:56 +03:00
Color g_ColorRed ( 255 , 63.75 , 63.75 , 255 ) ;
2020-04-22 12:56:21 -04:00
Color g_ColorGreen ( 153 , 255 , 153 , 255 ) ;
Color g_ColorDarkGreen ( 64 , 255 , 64 , 255 ) ;
2023-10-03 17:23:56 +03:00
Color g_ColorYellow ( 255 , 178.5 , 0.0 , 255 ) ;
2020-04-22 12:56:21 -04:00
Color g_ColorGrey ( 204 , 204 , 204 , 255 ) ;
2023-10-03 17:23:56 +03:00
static const char * gBugPriorityTable [ ] = {
" TODAY " ,
" ASAP " ,
" NONE " ,
NULL
} ;
static const char * gBugTokenTable [ ] = {
" re " , " regression " ,
" today " , " showstopper " ,
" asap " , " showstopper " ,
" ss " , " showstopper " ,
" show " , " showstopper " ,
// "high", "high",
" med " , " medium " ,
// "low", "low",
" none " , " feature " ,
" sugg " , " feature " ,
" feat " , " feature " ,
NULL
} ;
2020-04-22 12:56:21 -04:00
// removes all color markup characters, so Msg can deal with the string properly
// returns a pointer to str
char * RemoveColorMarkup ( char * str )
{
char * out = str ;
for ( char * in = str ; * in ! = 0 ; + + in )
{
if ( * in > 0 & & * in < COLOR_MAX )
{
continue ;
}
* out = * in ;
+ + out ;
}
* out = 0 ;
return str ;
}
// converts all '\r' characters to '\n', so that the engine can deal with the properly
// returns a pointer to str
char * ConvertCRtoNL ( char * str )
{
for ( char * ch = str ; * ch ! = 0 ; ch + + )
if ( * ch = = ' \r ' )
* ch = ' \n ' ;
return str ;
}
// converts all '\r' characters to '\n', so that the engine can deal with the properly
// returns a pointer to str
wchar_t * ConvertCRtoNL ( wchar_t * str )
{
for ( wchar_t * ch = str ; * ch ! = 0 ; ch + + )
if ( * ch = = L ' \r ' )
* ch = L ' \n ' ;
return str ;
}
void StripEndNewlineFromString ( char * str )
{
int s = strlen ( str ) - 1 ;
if ( s > = 0 )
{
if ( str [ s ] = = ' \n ' | | str [ s ] = = ' \r ' )
str [ s ] = 0 ;
}
}
void StripEndNewlineFromString ( wchar_t * str )
{
int s = wcslen ( str ) - 1 ;
if ( s > = 0 )
{
if ( str [ s ] = = L ' \n ' | | str [ s ] = = L ' \r ' )
str [ s ] = 0 ;
}
}
//-----------------------------------------------------------------------------
// Purpose: Reads a string from the current message and checks if it is translatable
//-----------------------------------------------------------------------------
2023-10-03 17:23:56 +03:00
wchar_t * ReadLocalizedString ( bf_read & msg , wchar_t * pOut , int outSize , bool bStripNewline , char * originalString , int originalSize )
2020-04-22 12:56:21 -04:00
{
char szString [ 2048 ] ;
szString [ 0 ] = 0 ;
msg . ReadString ( szString , sizeof ( szString ) ) ;
if ( originalString )
{
Q_strncpy ( originalString , szString , originalSize ) ;
}
const wchar_t * pBuf = g_pVGuiLocalize - > Find ( szString ) ;
if ( pBuf )
{
2023-10-03 17:23:56 +03:00
wcsncpy ( pOut , pBuf , outSize / sizeof ( wchar_t ) ) ;
pOut [ outSize / sizeof ( wchar_t ) - 1 ] = 0 ;
2020-04-22 12:56:21 -04:00
}
else
{
2023-10-03 17:23:56 +03:00
g_pVGuiLocalize - > ConvertANSIToUnicode ( szString , pOut , outSize ) ;
2020-04-22 12:56:21 -04:00
}
if ( bStripNewline )
StripEndNewlineFromString ( pOut ) ;
return pOut ;
}
2023-10-03 17:23:56 +03:00
//-----------------------------------------------------------------------------
// Purpose: Expands shortcuts into longer tokens
//-----------------------------------------------------------------------------
static const char * TranslateToken ( const char * pToken )
{
const char * * pKey = gBugTokenTable ;
while ( pKey [ 0 ] )
{
if ( ! V_stricmp ( pKey [ 0 ] , pToken ) )
{
return pKey [ 1 ] ;
}
pKey + = 2 ;
}
return pToken ;
}
static const char * TranslatePriorityToken ( const char * pToken )
{
const char * * pKey = gBugPriorityTable ;
while ( pKey [ 0 ] )
{
if ( ! V_stricmp ( pKey [ 0 ] , pToken ) )
{
return pKey [ 0 ] ;
}
pKey + + ;
}
return NULL ;
}
//-----------------------------------------------------------------------------
// Purpose: Converts all the strings in parentheses into a linked list of strings
// It will also null terminate the string at the first parenthesis
//-----------------------------------------------------------------------------
static CUtlLinkedList < const char * > * ParseTokens ( char * szString )
{
CUtlLinkedList < const char * > * tokens = new CUtlLinkedList < const char * > ( ) ;
// ensure that the defaults are reset
// later tokens should override these values
tokens - > AddToHead ( " NONE " ) ;
tokens - > AddToHead ( " High " ) ;
tokens - > AddToHead ( " triage " ) ;
char * pEnd = szString + V_strlen ( szString ) - 1 ;
while ( pEnd > = szString & & ( * pEnd = = ' ) ' | | * pEnd = = ' ' ) )
{
if ( * pEnd = = ' ) ' )
{
char * pToken = NULL ;
// skip any spaces
char * pTemp = pEnd - 1 ;
while ( pTemp > = szString & & * pTemp = = ' ' ) pTemp - - ;
if ( pTemp > = szString )
{
pEnd = pTemp + 1 ;
* pEnd = ' \0 ' ;
}
// skip back to the open paren (if there is one)
char * pStart = pEnd ;
while ( pStart > szString & & * pStart ! = ' ( ' ) pStart - - ;
if ( pStart > = szString )
{
* pStart = ' \0 ' ;
pToken = pStart + 1 ;
}
if ( pToken > = szString & & pToken ! = pEnd )
{
const char * pTranslatedToken = TranslateToken ( pToken ) ;
const char * pPriorityToken = TranslatePriorityToken ( pToken ) ;
tokens - > AddToTail ( pTranslatedToken ) ;
if ( pPriorityToken )
{
tokens - > AddToTail ( pPriorityToken ) ;
}
}
pEnd = pStart ;
}
else
{
// Chomp off trailing white space
* pEnd = ' \0 ' ;
}
pEnd - - ;
}
return tokens ;
}
2020-04-22 12:56:21 -04:00
//-----------------------------------------------------------------------------
// Purpose: Reads a string from the current message, converts it to unicode, and strips out color codes
//-----------------------------------------------------------------------------
2023-10-03 17:23:56 +03:00
wchar_t * ReadChatTextString ( bf_read & msg , wchar_t * pOut , int outSize , bool stripBugData )
2020-04-22 12:56:21 -04:00
{
char szString [ 2048 ] ;
szString [ 0 ] = 0 ;
msg . ReadString ( szString , sizeof ( szString ) ) ;
2023-10-03 17:23:56 +03:00
g_pVGuiLocalize - > ConvertANSIToUnicode ( szString , pOut , outSize ) ;
2020-04-22 12:56:21 -04:00
StripEndNewlineFromString ( pOut ) ;
// converts color control characters into control characters for the normal color
for ( wchar_t * test = pOut ; test & & * test ; + + test )
{
if ( * test & & ( * test < COLOR_MAX ) )
{
2023-10-03 17:23:56 +03:00
* test = COLOR_NORMAL ;
2020-04-22 12:56:21 -04:00
}
}
return pOut ;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *parent -
// *panelName -
//-----------------------------------------------------------------------------
CBaseHudChatLine : : CBaseHudChatLine ( vgui : : Panel * parent , const char * panelName ) :
vgui : : RichText ( parent , panelName )
{
m_hFont = m_hFontMarlett = 0 ;
m_flExpireTime = 0.0f ;
m_flStartTime = 0.0f ;
m_iNameLength = 0 ;
m_text = NULL ;
SetPaintBackgroundEnabled ( true ) ;
SetVerticalScrollbar ( false ) ;
}
CBaseHudChatLine : : ~ CBaseHudChatLine ( )
{
if ( m_text )
{
delete [ ] m_text ;
m_text = NULL ;
}
}
void CBaseHudChatLine : : ApplySchemeSettings ( vgui : : IScheme * pScheme )
{
BaseClass : : ApplySchemeSettings ( pScheme ) ;
m_hFont = pScheme - > GetFont ( " Default " ) ;
SetBgColor ( Color ( 0 , 0 , 0 , 100 ) ) ;
m_hFontMarlett = pScheme - > GetFont ( " Marlett " ) ;
m_clrText = pScheme - > GetColor ( " FgColor " , GetFgColor ( ) ) ;
SetFont ( m_hFont ) ;
}
void CBaseHudChatLine : : PerformFadeout ( void )
{
// Flash + Extra bright when new
float curtime = gpGlobals - > curtime ;
int lr = m_clrText [ 0 ] ;
int lg = m_clrText [ 1 ] ;
int lb = m_clrText [ 2 ] ;
if ( curtime > = m_flStartTime & & curtime < m_flStartTime + CHATLINE_FLASH_TIME )
{
float frac1 = ( curtime - m_flStartTime ) / CHATLINE_FLASH_TIME ;
float frac = frac1 ;
frac * = CHATLINE_NUM_FLASHES ;
frac * = 2 * M_PI ;
frac = cos ( frac ) ;
frac = clamp ( frac , 0.0f , 1.0f ) ;
frac * = ( 1.0f - frac1 ) ;
int r = lr , g = lg , b = lb ;
r = r + ( 255 - r ) * frac ;
g = g + ( 255 - g ) * frac ;
b = b + ( 255 - b ) * frac ;
// Draw a right facing triangle in red, faded out over time
int alpha = 63 + 192 * ( 1.0f - frac1 ) ;
alpha = clamp ( alpha , 0 , 255 ) ;
wchar_t wbuf [ 4096 ] ;
GetText ( 0 , wbuf , sizeof ( wbuf ) ) ;
SetText ( " " ) ;
InsertColorChange ( Color ( r , g , b , 255 ) ) ;
InsertString ( wbuf ) ;
}
else if ( curtime < = m_flExpireTime & & curtime > m_flExpireTime - CHATLINE_FADE_TIME )
{
float frac = ( m_flExpireTime - curtime ) / CHATLINE_FADE_TIME ;
int alpha = frac * 255 ;
alpha = clamp ( alpha , 0 , 255 ) ;
wchar_t wbuf [ 4096 ] ;
GetText ( 0 , wbuf , sizeof ( wbuf ) ) ;
SetText ( " " ) ;
InsertColorChange ( Color ( lr * frac , lg * frac , lb * frac , alpha ) ) ;
InsertString ( wbuf ) ;
}
else
{
wchar_t wbuf [ 4096 ] ;
GetText ( 0 , wbuf , sizeof ( wbuf ) ) ;
SetText ( " " ) ;
InsertColorChange ( Color ( lr , lg , lb , 255 ) ) ;
InsertString ( wbuf ) ;
}
OnThink ( ) ;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : time -
//-----------------------------------------------------------------------------
void CBaseHudChatLine : : SetExpireTime ( void )
{
m_flStartTime = gpGlobals - > curtime ;
m_flExpireTime = m_flStartTime + hud_saytext_time . GetFloat ( ) ;
m_nCount = CBaseHudChat : : m_nLineCounter + + ;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
int CBaseHudChatLine : : GetCount ( void )
{
return m_nCount ;
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : Returns true on success, false on failure.
//-----------------------------------------------------------------------------
bool CBaseHudChatLine : : IsReadyToExpire ( void )
{
// Engine disconnected, expire right away
if ( ! engine - > IsInGame ( ) & & ! engine - > IsConnected ( ) )
return true ;
if ( gpGlobals - > curtime > = m_flExpireTime )
return true ;
return false ;
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : float
//-----------------------------------------------------------------------------
float CBaseHudChatLine : : GetStartTime ( void )
{
return m_flStartTime ;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CBaseHudChatLine : : Expire ( void )
{
SetVisible ( false ) ;
// Spit out label text now
// char text[ 256 ];
// GetText( text, 256 );
// Msg( "%s\n", text );
}
//-----------------------------------------------------------------------------
// Purpose: The prompt and text entry area for chat messages
//-----------------------------------------------------------------------------
2023-10-03 17:23:56 +03:00
CBaseHudChatInputLine : : CBaseHudChatInputLine ( CBaseHudChat * parent , char const * panelName ) :
2020-04-22 12:56:21 -04:00
vgui : : Panel ( parent , panelName )
{
SetMouseInputEnabled ( false ) ;
m_pPrompt = new vgui : : Label ( this , " ChatInputPrompt " , L " Enter text: " ) ;
m_pInput = new CBaseHudChatEntry ( this , " ChatInput " , parent ) ;
m_pInput - > SetMaximumCharCount ( 127 ) ;
}
void CBaseHudChatInputLine : : ApplySchemeSettings ( vgui : : IScheme * pScheme )
{
BaseClass : : ApplySchemeSettings ( pScheme ) ;
2023-10-03 17:23:56 +03:00
2020-04-22 12:56:21 -04:00
// FIXME: Outline
2023-10-03 17:23:56 +03:00
vgui : : HFont hFont = pScheme - > GetFont ( " ChatFont " ) ;
2020-04-22 12:56:21 -04:00
m_pPrompt - > SetFont ( hFont ) ;
m_pInput - > SetFont ( hFont ) ;
m_pInput - > SetFgColor ( pScheme - > GetColor ( " Chat.TypingText " , pScheme - > GetColor ( " Panel.FgColor " , Color ( 255 , 255 , 255 , 255 ) ) ) ) ;
SetPaintBackgroundEnabled ( true ) ;
m_pPrompt - > SetPaintBackgroundEnabled ( true ) ;
m_pPrompt - > SetContentAlignment ( vgui : : Label : : a_west ) ;
m_pPrompt - > SetTextInset ( 2 , 0 ) ;
m_pInput - > SetMouseInputEnabled ( true ) ;
SetBgColor ( Color ( 0 , 0 , 0 , 0 ) ) ;
}
void CBaseHudChatInputLine : : SetPrompt ( const wchar_t * prompt )
{
Assert ( m_pPrompt ) ;
m_pPrompt - > SetText ( prompt ) ;
InvalidateLayout ( ) ;
}
void CBaseHudChatInputLine : : ClearEntry ( void )
{
Assert ( m_pInput ) ;
SetEntry ( L " " ) ;
}
void CBaseHudChatInputLine : : SetEntry ( const wchar_t * entry )
{
Assert ( m_pInput ) ;
Assert ( entry ) ;
m_pInput - > SetText ( entry ) ;
2023-10-03 17:23:56 +03:00
if ( entry & & wcslen ( entry ) > 0 )
{
m_pInput - > GotoEndOfLine ( ) ;
}
2020-04-22 12:56:21 -04:00
}
2023-10-03 17:23:56 +03:00
void CBaseHudChatInputLine : : GetMessageText ( wchar_t * buffer , int buffersizebytes )
2020-04-22 12:56:21 -04:00
{
m_pInput - > GetText ( buffer , buffersizebytes ) ;
}
void CBaseHudChatInputLine : : PerformLayout ( )
{
BaseClass : : PerformLayout ( ) ;
int wide , tall ;
GetSize ( wide , tall ) ;
int w , h ;
m_pPrompt - > GetContentSize ( w , h ) ;
m_pPrompt - > SetBounds ( 0 , 0 , w , tall ) ;
m_pInput - > SetBounds ( w + 2 , 0 , wide - w - 2 , tall ) ;
}
vgui : : Panel * CBaseHudChatInputLine : : GetInputPanel ( void )
{
return m_pInput ;
}
CHudChatFilterButton : : CHudChatFilterButton ( vgui : : Panel * pParent , const char * pName , const char * pText ) :
BaseClass ( pParent , pName , pText )
{
}
CHudChatFilterCheckButton : : CHudChatFilterCheckButton ( vgui : : Panel * pParent , const char * pName , const char * pText , int iFlag ) :
BaseClass ( pParent , pName , pText )
{
m_iFlag = iFlag ;
}
CHudChatFilterPanel : : CHudChatFilterPanel ( vgui : : Panel * pParent , const char * pName ) : BaseClass ( pParent , pName )
{
SetParent ( pParent ) ;
new CHudChatFilterCheckButton ( this , " joinleave_button " , " Sky is blue? " , CHAT_FILTER_JOINLEAVE ) ;
new CHudChatFilterCheckButton ( this , " namechange_button " , " Sky is blue? " , CHAT_FILTER_NAMECHANGE ) ;
new CHudChatFilterCheckButton ( this , " publicchat_button " , " Sky is blue? " , CHAT_FILTER_PUBLICCHAT ) ;
new CHudChatFilterCheckButton ( this , " servermsg_button " , " Sky is blue? " , CHAT_FILTER_SERVERMSG ) ;
new CHudChatFilterCheckButton ( this , " teamchange_button " , " Sky is blue? " , CHAT_FILTER_TEAMCHANGE ) ;
}
void CHudChatFilterPanel : : ApplySchemeSettings ( vgui : : IScheme * pScheme )
{
2023-10-03 17:23:56 +03:00
if ( IsConsole ( ) )
{
// not used
BaseClass : : SetVisible ( false ) ;
return ;
}
2020-04-22 12:56:21 -04:00
LoadControlSettings ( " resource/UI/ChatFilters.res " ) ;
BaseClass : : ApplySchemeSettings ( pScheme ) ;
Color cColor = pScheme - > GetColor ( " DullWhite " , GetBgColor ( ) ) ;
SetBgColor ( Color ( cColor . r ( ) , cColor . g ( ) , cColor . b ( ) , CHAT_HISTORY_ALPHA ) ) ;
SetFgColor ( pScheme - > GetColor ( " Blank " , GetFgColor ( ) ) ) ;
}
void CHudChatFilterPanel : : OnFilterButtonChecked ( vgui : : Panel * panel )
{
2023-10-03 17:23:56 +03:00
if ( IsConsole ( ) )
{
// not used
return ;
}
2020-04-22 12:56:21 -04:00
CHudChatFilterCheckButton * pButton = dynamic_cast < CHudChatFilterCheckButton * > ( panel ) ;
if ( pButton & & GetChatParent ( ) & & IsVisible ( ) )
{
if ( pButton - > IsSelected ( ) )
{
GetChatParent ( ) - > SetFilterFlag ( GetChatParent ( ) - > GetFilterFlags ( ) | pButton - > GetFilterFlag ( ) ) ;
}
else
{
GetChatParent ( ) - > SetFilterFlag ( GetChatParent ( ) - > GetFilterFlags ( ) & ~ pButton - > GetFilterFlag ( ) ) ;
}
}
}
void CHudChatFilterPanel : : SetVisible ( bool state )
{
2023-10-03 17:23:56 +03:00
if ( IsConsole ( ) )
{
// not used
return ;
}
2020-04-22 12:56:21 -04:00
if ( state = = true )
{
for ( int i = 0 ; i < GetChildCount ( ) ; i + + )
{
CHudChatFilterCheckButton * pButton = dynamic_cast < CHudChatFilterCheckButton * > ( GetChild ( i ) ) ;
if ( pButton )
{
if ( cl_chatfilters . GetInt ( ) & pButton - > GetFilterFlag ( ) )
{
pButton - > SetSelected ( true ) ;
}
else
{
pButton - > SetSelected ( false ) ;
}
}
}
}
BaseClass : : SetVisible ( state ) ;
}
void CHudChatFilterButton : : DoClick ( void )
{
2023-10-03 17:23:56 +03:00
if ( IsConsole ( ) )
{
// not used
return ;
}
2020-04-22 12:56:21 -04:00
BaseClass : : DoClick ( ) ;
CBaseHudChat * pChat = dynamic_cast < CBaseHudChat * > ( GetParent ( ) ) ;
if ( pChat )
{
pChat - > GetChatInput ( ) - > RequestFocus ( ) ;
if ( pChat - > GetChatFilterPanel ( ) )
{
if ( pChat - > GetChatFilterPanel ( ) - > IsVisible ( ) )
{
pChat - > GetChatFilterPanel ( ) - > SetVisible ( false ) ;
}
else
{
pChat - > GetChatFilterPanel ( ) - > SetVisible ( true ) ;
pChat - > GetChatFilterPanel ( ) - > MakePopup ( ) ;
pChat - > GetChatFilterPanel ( ) - > SetMouseInputEnabled ( true ) ;
}
}
}
}
CHudChatHistory : : CHudChatHistory ( vgui : : Panel * pParent , const char * panelName ) : BaseClass ( pParent , " HudChatHistory " )
{
2023-10-03 17:23:56 +03:00
SetScheme ( " ChatScheme " ) ;
2020-04-22 12:56:21 -04:00
InsertFade ( - 1 , - 1 ) ;
}
void CHudChatHistory : : ApplySchemeSettings ( vgui : : IScheme * pScheme )
{
BaseClass : : ApplySchemeSettings ( pScheme ) ;
2023-10-03 17:23:56 +03:00
SetFont ( pScheme - > GetFont ( " ChatFont " ) ) ;
2020-04-22 12:56:21 -04:00
SetAlpha ( 255 ) ;
}
2023-10-03 17:23:56 +03:00
void CHudChatHistory : : Paint ( )
{
BaseClass : : Paint ( ) ;
if ( IsAllTextAlphaZero ( ) & & HasText ( ) )
{
SetText ( " " ) ;
// Wipe
}
}
CBaseHudChat * g_pHudChat = NULL ;
CBaseHudChat * CBaseHudChat : : GetHudChat ( void )
{
Assert ( g_pHudChat ) ;
return g_pHudChat ;
}
2020-04-22 12:56:21 -04:00
int CBaseHudChat : : m_nLineCounter = 1 ;
//-----------------------------------------------------------------------------
// Purpose: Text chat input/output hud element
//-----------------------------------------------------------------------------
CBaseHudChat : : CBaseHudChat ( const char * pElementName )
: CHudElement ( pElementName ) , BaseClass ( NULL , " HudChat " )
{
2023-10-03 17:23:56 +03:00
Assert ( g_pHudChat = = NULL ) ;
g_pHudChat = this ;
2022-09-08 21:23:06 +03:00
2023-10-03 17:23:56 +03:00
vgui : : Panel * pParent = GetFullscreenClientMode ( ) - > GetViewport ( ) ;
SetParent ( pParent ) ;
2022-09-08 21:23:06 +03:00
2023-10-03 17:23:56 +03:00
vgui : : HScheme scheme = vgui : : scheme ( ) - > LoadSchemeFromFileEx ( NULL , " resource/ChatScheme.res " , " ChatScheme " ) ;
2020-04-22 12:56:21 -04:00
SetScheme ( scheme ) ;
g_pVGuiLocalize - > AddFile ( " resource/chat_%language%.txt " ) ;
2023-10-03 17:23:56 +03:00
m_nMessageMode = MM_NONE ;
cl_chat_active . SetValue ( m_nMessageMode ) ;
2020-04-22 12:56:21 -04:00
vgui : : ivgui ( ) - > AddTickSignal ( GetVPanel ( ) ) ;
// (We don't actually want input until they bring up the chat line).
MakePopup ( ) ;
SetZPos ( - 30 ) ;
SetHiddenBits ( HIDEHUD_CHAT ) ;
2023-10-03 17:23:56 +03:00
m_pFiltersButton = new CHudChatFilterButton ( this , " ChatFiltersButton " , " #chat_filterbutton " ) ;
2020-04-22 12:56:21 -04:00
if ( m_pFiltersButton )
{
m_pFiltersButton - > SetScheme ( scheme ) ;
m_pFiltersButton - > SetVisible ( true ) ;
m_pFiltersButton - > SetEnabled ( true ) ;
m_pFiltersButton - > SetMouseInputEnabled ( true ) ;
m_pFiltersButton - > SetKeyBoardInputEnabled ( false ) ;
}
m_pChatHistory = new CHudChatHistory ( this , " HudChatHistory " ) ;
CreateChatLines ( ) ;
CreateChatInputLine ( ) ;
GetChatFilterPanel ( ) ;
m_iFilterFlags = cl_chatfilters . GetInt ( ) ;
}
2023-10-03 17:23:56 +03:00
CBaseHudChat : : ~ CBaseHudChat ( )
{
g_pHudChat = NULL ;
}
2020-04-22 12:56:21 -04:00
void CBaseHudChat : : CreateChatInputLine ( void )
{
m_pChatInput = new CBaseHudChatInputLine ( this , " ChatInputLine " ) ;
m_pChatInput - > SetVisible ( false ) ;
if ( GetChatHistory ( ) )
{
GetChatHistory ( ) - > SetMaximumCharCount ( 127 * 100 ) ;
GetChatHistory ( ) - > SetVisible ( true ) ;
}
}
void CBaseHudChat : : CreateChatLines ( void )
{
m_ChatLine = new CBaseHudChatLine ( this , " ChatLine1 " ) ;
m_ChatLine - > SetVisible ( false ) ;
}
# define BACKGROUND_BORDER_WIDTH 20
CHudChatFilterPanel * CBaseHudChat : : GetChatFilterPanel ( void )
{
if ( m_pFilterPanel = = NULL )
{
m_pFilterPanel = new CHudChatFilterPanel ( this , " HudChatFilterPanel " ) ;
if ( m_pFilterPanel )
{
2023-10-03 17:23:56 +03:00
m_pFilterPanel - > SetScheme ( " ChatScheme " ) ;
2020-04-22 12:56:21 -04:00
m_pFilterPanel - > InvalidateLayout ( true , true ) ;
m_pFilterPanel - > SetMouseInputEnabled ( true ) ;
m_pFilterPanel - > SetPaintBackgroundType ( 2 ) ;
m_pFilterPanel - > SetPaintBorderEnabled ( true ) ;
m_pFilterPanel - > SetVisible ( false ) ;
}
}
return m_pFilterPanel ;
}
void CBaseHudChat : : ApplySchemeSettings ( vgui : : IScheme * pScheme )
{
2023-10-03 17:23:56 +03:00
LoadControlSettings ( " resource/UI/BaseChat.res " ) ;
2020-04-22 12:56:21 -04:00
BaseClass : : ApplySchemeSettings ( pScheme ) ;
SetPaintBackgroundType ( 2 ) ;
SetPaintBorderEnabled ( true ) ;
SetPaintBackgroundEnabled ( true ) ;
SetKeyBoardInputEnabled ( false ) ;
SetMouseInputEnabled ( false ) ;
Color cColor = pScheme - > GetColor ( " DullWhite " , GetBgColor ( ) ) ;
SetBgColor ( Color ( cColor . r ( ) , cColor . g ( ) , cColor . b ( ) , CHAT_HISTORY_ALPHA ) ) ;
GetChatHistory ( ) - > SetVerticalScrollbar ( false ) ;
2023-10-03 17:23:56 +03:00
if ( IsConsole ( ) )
{
// console has no keyboard
// panel not used as input, only as output of chat history
SetPaintBackgroundEnabled ( false ) ;
m_pChatInput - > SetVisible ( false ) ;
m_ChatLine - > SetVisible ( false ) ;
m_pFiltersButton - > SetVisible ( false ) ;
m_pFilterPanel - > SetVisible ( false ) ;
GetChatHistory ( ) - > SetBgColor ( Color ( 0 , 0 , 0 , 0 ) ) ;
}
2020-04-22 12:56:21 -04:00
}
void CBaseHudChat : : Reset ( void )
{
Clear ( ) ;
}
void CBaseHudChat : : Paint ( void )
{
}
CHudChatHistory * CBaseHudChat : : GetChatHistory ( void )
{
return m_pChatHistory ;
}
void CBaseHudChat : : Init ( void )
{
ListenForGameEvent ( " hltv_chat " ) ;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *pszName -
// iSize -
// *pbuf -
//-----------------------------------------------------------------------------
void CBaseHudChat : : MsgFunc_SayText ( bf_read & msg )
{
char szString [ 256 ] ;
int client = msg . ReadByte ( ) ;
msg . ReadString ( szString , sizeof ( szString ) ) ;
2023-10-03 17:23:56 +03:00
bool bWantsToChat = msg . ReadByte ( ) ? true : false ;
2020-04-22 12:56:21 -04:00
if ( bWantsToChat )
{
// print raw chat text
ChatPrintf ( client , CHAT_FILTER_NONE , " %s " , szString ) ;
}
else
{
// try to lookup translated string
Printf ( CHAT_FILTER_NONE , " %s " , hudtextmessage - > LookupString ( szString ) ) ;
}
CLocalPlayerFilter filter ;
C_BaseEntity : : EmitSound ( filter , SOUND_FROM_LOCAL_PLAYER , " HudChat.Message " ) ;
2023-10-03 17:23:56 +03:00
// TERROR: color console echo
//Msg( "%s", szString );
2020-04-22 12:56:21 -04:00
}
int CBaseHudChat : : GetFilterForString ( const char * pString )
{
if ( ! Q_stricmp ( pString , " #HL_Name_Change " ) )
{
return CHAT_FILTER_NAMECHANGE ;
}
return CHAT_FILTER_NONE ;
}
//-----------------------------------------------------------------------------
// Purpose: Reads in a player's Chat text from the server
//-----------------------------------------------------------------------------
void CBaseHudChat : : MsgFunc_SayText2 ( bf_read & msg )
{
// Got message during connection
if ( ! g_PR )
return ;
int client = msg . ReadByte ( ) ;
2023-10-03 17:23:56 +03:00
bool bWantsToChat = msg . ReadByte ( ) ? true : false ;
2020-04-22 12:56:21 -04:00
wchar_t szBuf [ 6 ] [ 256 ] ;
char untranslated_msg_text [ 256 ] ;
wchar_t * msg_text = ReadLocalizedString ( msg , szBuf [ 0 ] , sizeof ( szBuf [ 0 ] ) , false , untranslated_msg_text , sizeof ( untranslated_msg_text ) ) ;
// keep reading strings and using C format strings for subsituting the strings into the localised text string
ReadChatTextString ( msg , szBuf [ 1 ] , sizeof ( szBuf [ 1 ] ) ) ; // player name
2023-10-03 17:23:56 +03:00
ReadChatTextString ( msg , szBuf [ 2 ] , sizeof ( szBuf [ 2 ] ) , true ) ; // chat text
2020-04-22 12:56:21 -04:00
ReadLocalizedString ( msg , szBuf [ 3 ] , sizeof ( szBuf [ 3 ] ) , true ) ;
ReadLocalizedString ( msg , szBuf [ 4 ] , sizeof ( szBuf [ 4 ] ) , true ) ;
2022-03-01 23:00:42 +03:00
g_pVGuiLocalize - > ConstructString ( szBuf [ 5 ] , sizeof ( szBuf [ 5 ] ) , msg_text , 4 , szBuf [ 1 ] , szBuf [ 2 ] , szBuf [ 3 ] , szBuf [ 4 ] ) ;
2020-04-22 12:56:21 -04:00
char ansiString [ 512 ] ;
g_pVGuiLocalize - > ConvertUnicodeToANSI ( ConvertCRtoNL ( szBuf [ 5 ] ) , ansiString , sizeof ( ansiString ) ) ;
if ( bWantsToChat )
{
int iFilter = CHAT_FILTER_NONE ;
if ( client > 0 & & ( g_PR - > GetTeam ( client ) ! = g_PR - > GetTeam ( GetLocalPlayerIndex ( ) ) ) )
{
iFilter = CHAT_FILTER_PUBLICCHAT ;
}
// print raw chat text
ChatPrintf ( client , iFilter , " %s " , ansiString ) ;
2023-10-03 17:23:56 +03:00
// Msg( "%s\n", RemoveColorMarkup(ansiString) );
2020-04-22 12:56:21 -04:00
CLocalPlayerFilter filter ;
C_BaseEntity : : EmitSound ( filter , SOUND_FROM_LOCAL_PLAYER , " HudChat.Message " ) ;
}
else
{
// print raw chat text
ChatPrintf ( client , GetFilterForString ( untranslated_msg_text ) , " %s " , ansiString ) ;
}
}
//-----------------------------------------------------------------------------
// Message handler for text messages
// displays a string, looking them up from the titles.txt file, which can be localised
// parameters:
// byte: message direction ( HUD_PRINTCONSOLE, HUD_PRINTNOTIFY, HUD_PRINTCENTER, HUD_PRINTTALK )
// string: message
// optional parameters:
// string: message parameter 1
// string: message parameter 2
// string: message parameter 3
// string: message parameter 4
// any string that starts with the character '#' is a message name, and is used to look up the real message in titles.txt
// the next (optional) one to four strings are parameters for that string (which can also be message names if they begin with '#')
//-----------------------------------------------------------------------------
void CBaseHudChat : : MsgFunc_TextMsg ( bf_read & msg )
{
char szString [ 2048 ] ;
int msg_dest = msg . ReadByte ( ) ;
wchar_t szBuf [ 5 ] [ 256 ] ;
wchar_t outputBuf [ 256 ] ;
for ( int i = 0 ; i < 5 ; + + i )
{
msg . ReadString ( szString , sizeof ( szString ) ) ;
char * tmpStr = hudtextmessage - > LookupString ( szString , & msg_dest ) ;
const wchar_t * pBuf = g_pVGuiLocalize - > Find ( tmpStr ) ;
if ( pBuf )
{
// Copy pBuf into szBuf[i].
int nMaxChars = sizeof ( szBuf [ i ] ) / sizeof ( wchar_t ) ;
wcsncpy ( szBuf [ i ] , pBuf , nMaxChars ) ;
szBuf [ i ] [ nMaxChars - 1 ] = 0 ;
}
else
{
if ( i )
{
StripEndNewlineFromString ( tmpStr ) ; // these strings are meant for subsitution into the main strings, so cull the automatic end newlines
}
g_pVGuiLocalize - > ConvertANSIToUnicode ( tmpStr , szBuf [ i ] , sizeof ( szBuf [ i ] ) ) ;
}
}
if ( ! cl_showtextmsg . GetInt ( ) )
return ;
int len ;
switch ( msg_dest )
{
case HUD_PRINTCENTER :
2022-03-01 23:00:42 +03:00
g_pVGuiLocalize - > ConstructString ( outputBuf , sizeof ( outputBuf ) , szBuf [ 0 ] , 4 , szBuf [ 1 ] , szBuf [ 2 ] , szBuf [ 3 ] , szBuf [ 4 ] ) ;
2023-10-03 17:23:56 +03:00
GetCenterPrint ( ) - > Print ( ConvertCRtoNL ( outputBuf ) ) ;
2020-04-22 12:56:21 -04:00
break ;
case HUD_PRINTNOTIFY :
2022-03-01 23:00:42 +03:00
g_pVGuiLocalize - > ConstructString ( outputBuf , sizeof ( outputBuf ) , szBuf [ 0 ] , 4 , szBuf [ 1 ] , szBuf [ 2 ] , szBuf [ 3 ] , szBuf [ 4 ] ) ;
2020-04-22 12:56:21 -04:00
g_pVGuiLocalize - > ConvertUnicodeToANSI ( outputBuf , szString , sizeof ( szString ) ) ;
len = strlen ( szString ) ;
if ( len & & szString [ len - 1 ] ! = ' \n ' & & szString [ len - 1 ] ! = ' \r ' )
{
Q_strncat ( szString , " \n " , sizeof ( szString ) , 1 ) ;
}
Msg ( " %s " , ConvertCRtoNL ( szString ) ) ;
break ;
case HUD_PRINTTALK :
2022-03-01 23:00:42 +03:00
g_pVGuiLocalize - > ConstructString ( outputBuf , sizeof ( outputBuf ) , szBuf [ 0 ] , 4 , szBuf [ 1 ] , szBuf [ 2 ] , szBuf [ 3 ] , szBuf [ 4 ] ) ;
2020-04-22 12:56:21 -04:00
g_pVGuiLocalize - > ConvertUnicodeToANSI ( outputBuf , szString , sizeof ( szString ) ) ;
len = strlen ( szString ) ;
if ( len & & szString [ len - 1 ] ! = ' \n ' & & szString [ len - 1 ] ! = ' \r ' )
{
Q_strncat ( szString , " \n " , sizeof ( szString ) , 1 ) ;
}
Printf ( CHAT_FILTER_NONE , " %s " , ConvertCRtoNL ( szString ) ) ;
2023-10-03 17:23:56 +03:00
// TERROR: color console echo
//Msg( "%s", ConvertCRtoNL( szString ) );
2020-04-22 12:56:21 -04:00
break ;
case HUD_PRINTCONSOLE :
2022-03-01 23:00:42 +03:00
g_pVGuiLocalize - > ConstructString ( outputBuf , sizeof ( outputBuf ) , szBuf [ 0 ] , 4 , szBuf [ 1 ] , szBuf [ 2 ] , szBuf [ 3 ] , szBuf [ 4 ] ) ;
2020-04-22 12:56:21 -04:00
g_pVGuiLocalize - > ConvertUnicodeToANSI ( outputBuf , szString , sizeof ( szString ) ) ;
len = strlen ( szString ) ;
if ( len & & szString [ len - 1 ] ! = ' \n ' & & szString [ len - 1 ] ! = ' \r ' )
{
Q_strncat ( szString , " \n " , sizeof ( szString ) , 1 ) ;
}
Msg ( " %s " , ConvertCRtoNL ( szString ) ) ;
break ;
}
}
void CBaseHudChat : : MsgFunc_VoiceSubtitle ( bf_read & msg )
{
// Got message during connection
if ( ! g_PR )
return ;
if ( ! cl_showtextmsg . GetInt ( ) )
return ;
char szString [ 2048 ] ;
char szPrefix [ 64 ] ; //(Voice)
wchar_t szBuf [ 128 ] ;
int client = msg . ReadByte ( ) ;
int iMenu = msg . ReadByte ( ) ;
int iItem = msg . ReadByte ( ) ;
const char * pszSubtitle = " " ;
CGameRules * pGameRules = GameRules ( ) ;
CMultiplayRules * pMultiRules = dynamic_cast < CMultiplayRules * > ( pGameRules ) ;
Assert ( pMultiRules ) ;
if ( pMultiRules )
{
pszSubtitle = pMultiRules - > GetVoiceCommandSubtitle ( iMenu , iItem ) ;
}
SetVoiceSubtitleState ( true ) ;
const wchar_t * pBuf = g_pVGuiLocalize - > Find ( pszSubtitle ) ;
if ( pBuf )
{
// Copy pBuf into szBuf[i].
int nMaxChars = sizeof ( szBuf ) / sizeof ( wchar_t ) ;
wcsncpy ( szBuf , pBuf , nMaxChars ) ;
szBuf [ nMaxChars - 1 ] = 0 ;
}
else
{
g_pVGuiLocalize - > ConvertANSIToUnicode ( pszSubtitle , szBuf , sizeof ( szBuf ) ) ;
}
int len ;
g_pVGuiLocalize - > ConvertUnicodeToANSI ( szBuf , szString , sizeof ( szString ) ) ;
len = strlen ( szString ) ;
if ( len & & szString [ len - 1 ] ! = ' \n ' & & szString [ len - 1 ] ! = ' \r ' )
{
Q_strncat ( szString , " \n " , sizeof ( szString ) , 1 ) ;
}
const wchar_t * pVoicePrefix = g_pVGuiLocalize - > Find ( " #Voice " ) ;
g_pVGuiLocalize - > ConvertUnicodeToANSI ( pVoicePrefix , szPrefix , sizeof ( szPrefix ) ) ;
ChatPrintf ( client , CHAT_FILTER_NONE , " %c(%s) %s%c: %s " , COLOR_PLAYERNAME , szPrefix , GetDisplayedSubtitlePlayerName ( client ) , COLOR_NORMAL , ConvertCRtoNL ( szString ) ) ;
SetVoiceSubtitleState ( false ) ;
}
const char * CBaseHudChat : : GetDisplayedSubtitlePlayerName ( int clientIndex )
{
return g_PR - > GetPlayerName ( clientIndex ) ;
}
static int __cdecl SortLines ( void const * line1 , void const * line2 )
{
CBaseHudChatLine * l1 = * ( CBaseHudChatLine * * ) line1 ;
CBaseHudChatLine * l2 = * ( CBaseHudChatLine * * ) line2 ;
// Invisible at bottom
if ( l1 - > IsVisible ( ) & & ! l2 - > IsVisible ( ) )
return - 1 ;
else if ( ! l1 - > IsVisible ( ) & & l2 - > IsVisible ( ) )
return 1 ;
// Oldest start time at top
if ( l1 - > GetStartTime ( ) < l2 - > GetStartTime ( ) )
return - 1 ;
else if ( l1 - > GetStartTime ( ) > l2 - > GetStartTime ( ) )
return 1 ;
// Otherwise, compare counter
if ( l1 - > GetCount ( ) < l2 - > GetCount ( ) )
return - 1 ;
else if ( l1 - > GetCount ( ) > l2 - > GetCount ( ) )
return 1 ;
return 0 ;
}
2023-10-03 17:23:56 +03:00
2020-04-22 12:56:21 -04:00
//-----------------------------------------------------------------------------
// Purpose: Allow inheriting classes to change this spacing behavior
//-----------------------------------------------------------------------------
int CBaseHudChat : : GetChatInputOffset ( void )
{
return m_iFontHeight ;
}
//-----------------------------------------------------------------------------
2023-10-03 17:23:56 +03:00
// Purpose: Do repositioning here to avoid latency due to repositioning of vgui
2020-04-22 12:56:21 -04:00
// voice manager icon panel
//-----------------------------------------------------------------------------
void CBaseHudChat : : OnTick ( void )
{
CBaseHudChatLine * line = m_ChatLine ;
if ( line )
{
vgui : : HFont font = line - > GetFont ( ) ;
2023-10-03 17:23:56 +03:00
m_iFontHeight = vgui : : surface ( ) - > GetFontTall ( font ) + 2 ;
2020-04-22 12:56:21 -04:00
int iChatX , iChatY , iChatW , iChatH ;
GetBounds ( iChatX , iChatY , iChatW , iChatH ) ;
2023-10-03 17:23:56 +03:00
// Put input area at bottom
int iInputX , iInputY , iInputW , iInputH ;
m_pChatInput - > GetBounds ( iInputX , iInputY , iInputW , iInputH ) ;
2020-04-22 12:56:21 -04:00
m_pChatInput - > SetBounds ( iInputX , iChatH - ( m_iFontHeight * 1.75 ) , iInputW , m_iFontHeight ) ;
//Resize the History Panel so it fits more lines depending on the screen resolution.
int iChatHistoryX , iChatHistoryY , iChatHistoryW , iChatHistoryH ;
GetChatHistory ( ) - > GetBounds ( iChatHistoryX , iChatHistoryY , iChatHistoryW , iChatHistoryH ) ;
iChatHistoryH = ( iChatH - ( m_iFontHeight * 2.25 ) ) - iChatHistoryY ;
GetChatHistory ( ) - > SetBounds ( iChatHistoryX , iChatHistoryY , iChatHistoryW , iChatHistoryH ) ;
}
FadeChatHistory ( ) ;
2023-10-03 17:23:56 +03:00
if ( IsConsole ( ) )
{
// force to one time only for layout
vgui : : ivgui ( ) - > RemoveTickSignal ( GetVPanel ( ) ) ;
}
2020-04-22 12:56:21 -04:00
}
2023-10-03 17:23:56 +03:00
// Release build is crashing on long strings...sigh
# pragma optimize( "", off )
2020-04-22 12:56:21 -04:00
//-----------------------------------------------------------------------------
// Purpose:
// Input : width -
// *text -
// textlen -
// Output : int
//-----------------------------------------------------------------------------
int CBaseHudChat : : ComputeBreakChar ( int width , const char * text , int textlen )
{
CBaseHudChatLine * line = m_ChatLine ;
vgui : : HFont font = line - > GetFont ( ) ;
int currentlen = 0 ;
int lastbreak = textlen ;
for ( int i = 0 ; i < textlen ; i + + )
{
char ch = text [ i ] ;
if ( ch < = 32 )
{
lastbreak = i ;
}
wchar_t wch [ 2 ] ;
g_pVGuiLocalize - > ConvertANSIToUnicode ( & ch , wch , sizeof ( wch ) ) ;
int a , b , c ;
vgui : : surface ( ) - > GetCharABCwide ( font , wch [ 0 ] , a , b , c ) ;
currentlen + = a + b + c ;
if ( currentlen > = width )
{
// If we haven't found a whitespace char to break on before getting
// to the end, but it's still too long, break on the character just before
// this one
if ( lastbreak = = textlen )
{
lastbreak = MAX ( 0 , i - 1 ) ;
}
break ;
}
}
if ( currentlen > = width )
{
return lastbreak ;
}
return textlen ;
}
2023-10-03 17:23:56 +03:00
# pragma optimize( "", on )
2020-04-22 12:56:21 -04:00
//-----------------------------------------------------------------------------
// Purpose:
// Input : *fmt -
// ... -
//-----------------------------------------------------------------------------
2023-10-03 17:23:56 +03:00
# pragma warning( push )
# pragma warning( disable: 4748 ) // /GS can not protect parameters and local variables from local buffer overrun because optimizations are disabled in function
2020-04-22 12:56:21 -04:00
void CBaseHudChat : : Printf ( int iFilter , const char * fmt , . . . )
{
va_list marker ;
char msg [ 4096 ] ;
va_start ( marker , fmt ) ;
Q_vsnprintf ( msg , sizeof ( msg ) , fmt , marker ) ;
va_end ( marker ) ;
ChatPrintf ( 0 , iFilter , " %s " , msg ) ;
}
2023-10-03 17:23:56 +03:00
# pragma warning( pop )
# pragma optimize( "", on )
2020-04-22 12:56:21 -04:00
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CBaseHudChat : : StartMessageMode ( int iMessageModeType )
{
m_nMessageMode = iMessageModeType ;
2023-10-03 17:23:56 +03:00
cl_chat_active . SetValue ( m_nMessageMode ) ;
2020-04-22 12:56:21 -04:00
2023-10-03 17:23:56 +03:00
if ( ! IsConsole ( ) )
2020-04-22 12:56:21 -04:00
{
2023-10-03 17:23:56 +03:00
m_pChatInput - > ClearEntry ( ) ;
SetChatPrompt ( iMessageModeType ) ;
if ( GetChatHistory ( ) )
2020-04-22 12:56:21 -04:00
{
2023-10-03 17:23:56 +03:00
// TERROR: hack to get ChatFont back
GetChatHistory ( ) - > SetFont ( vgui : : scheme ( ) - > GetIScheme ( GetScheme ( ) ) - > GetFont ( " ChatFont " , false ) ) ;
GetChatHistory ( ) - > SetMouseInputEnabled ( true ) ;
GetChatHistory ( ) - > SetKeyBoardInputEnabled ( false ) ;
GetChatHistory ( ) - > SetVerticalScrollbar ( true ) ;
GetChatHistory ( ) - > ResetAllFades ( true ) ;
GetChatHistory ( ) - > SetPaintBorderEnabled ( true ) ;
GetChatHistory ( ) - > SetVisible ( true ) ;
2020-04-22 12:56:21 -04:00
}
2023-10-03 17:23:56 +03:00
vgui : : SETUP_PANEL ( this ) ;
SetKeyBoardInputEnabled ( true ) ;
SetMouseInputEnabled ( true ) ;
m_pChatInput - > SetVisible ( true ) ;
vgui : : surface ( ) - > CalculateMouseVisible ( ) ;
m_pChatInput - > RequestFocus ( ) ;
m_pChatInput - > SetPaintBorderEnabled ( true ) ;
m_pChatInput - > SetMouseInputEnabled ( true ) ;
// Place the mouse cursor near the text so people notice it.
int x , y , w , h ;
GetChatHistory ( ) - > GetBounds ( x , y , w , h ) ;
# ifndef INFESTED_DLL
vgui : : input ( ) - > SetCursorPos ( x + ( w / 2 ) , y + ( h / 2 ) ) ;
# endif
m_pFilterPanel - > SetVisible ( false ) ;
}
2020-04-22 12:56:21 -04:00
m_flHistoryFadeTime = gpGlobals - > curtime + CHAT_HISTORY_FADE_TIME ;
engine - > ClientCmd_Unrestricted ( " gameui_preventescapetoshow \n " ) ;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
2023-10-03 17:23:56 +03:00
void CBaseHudChat : : SetChatPrompt ( int iMessageModeType )
2020-04-22 12:56:21 -04:00
{
2023-10-03 17:23:56 +03:00
if ( m_nMessageMode = = MM_SAY )
2020-04-22 12:56:21 -04:00
{
2023-10-03 17:23:56 +03:00
m_pChatInput - > SetPrompt ( g_pVGuiLocalize - > FindSafe ( " #chat_say " ) ) ;
}
else
{
m_pChatInput - > SetPrompt ( g_pVGuiLocalize - > FindSafe ( " #chat_say_team " ) ) ;
2020-04-22 12:56:21 -04:00
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
2023-10-03 17:23:56 +03:00
void CBaseHudChat : : StopMessageMode ( bool bFade )
2020-04-22 12:56:21 -04:00
{
2023-10-03 17:23:56 +03:00
engine - > ClientCmd_Unrestricted ( " gameui_allowescapetoshow \n " ) ;
2020-04-22 12:56:21 -04:00
2023-10-03 17:23:56 +03:00
if ( ! IsConsole ( ) )
{
SetKeyBoardInputEnabled ( false ) ;
SetMouseInputEnabled ( false ) ;
if ( GetChatHistory ( ) )
{
GetChatHistory ( ) - > SetPaintBorderEnabled ( false ) ;
GetChatHistory ( ) - > GotoTextEnd ( ) ;
GetChatHistory ( ) - > SetMouseInputEnabled ( false ) ;
GetChatHistory ( ) - > SetVerticalScrollbar ( false ) ;
GetChatHistory ( ) - > ResetAllFades ( false , true , CHAT_HISTORY_FADE_TIME ) ;
GetChatHistory ( ) - > SelectNoText ( ) ;
}
// Clear the entry since we wont need it anymore.
m_pChatInput - > ClearEntry ( ) ;
}
m_nMessageMode = MM_NONE ; // TERROR
cl_chat_active . SetValue ( m_nMessageMode ) ;
if ( bFade )
{
m_flHistoryFadeTime = gpGlobals - > curtime + CHAT_HISTORY_FADE_TIME ;
}
else
{
m_flHistoryFadeTime = gpGlobals - > curtime ;
if ( IsConsole ( ) )
{
// console forces these off now
GetChatHistory ( ) - > ResetAllFades ( false , false , 0 ) ;
}
}
2020-04-22 12:56:21 -04:00
}
2023-10-03 17:23:56 +03:00
2020-04-22 12:56:21 -04:00
void CBaseHudChat : : FadeChatHistory ( void )
{
2023-10-03 17:23:56 +03:00
if ( IsConsole ( ) )
{
return ;
}
float frac = ( m_flHistoryFadeTime - gpGlobals - > curtime ) * CHAT_HISTORY_ONE_OVER_FADE_TIME ;
2020-04-22 12:56:21 -04:00
int alpha = frac * CHAT_HISTORY_ALPHA ;
alpha = clamp ( alpha , 0 , CHAT_HISTORY_ALPHA ) ;
if ( alpha > = 0 )
{
if ( GetChatHistory ( ) )
{
if ( IsMouseInputEnabled ( ) )
{
2023-10-03 17:23:56 +03:00
// fade in
2020-04-22 12:56:21 -04:00
SetAlpha ( 255 ) ;
GetChatHistory ( ) - > SetBgColor ( Color ( 0 , 0 , 0 , CHAT_HISTORY_ALPHA - alpha ) ) ;
2023-10-03 17:23:56 +03:00
SetBgColor ( Color ( GetBgColor ( ) . r ( ) , GetBgColor ( ) . g ( ) , GetBgColor ( ) . b ( ) , CHAT_HISTORY_ALPHA - alpha ) ) ;
2020-04-22 12:56:21 -04:00
m_pChatInput - > GetPrompt ( ) - > SetAlpha ( ( CHAT_HISTORY_ALPHA * 2 ) - alpha ) ;
m_pChatInput - > GetInputPanel ( ) - > SetAlpha ( ( CHAT_HISTORY_ALPHA * 2 ) - alpha ) ;
m_pFiltersButton - > SetAlpha ( ( CHAT_HISTORY_ALPHA * 2 ) - alpha ) ;
}
else
{
2023-10-03 17:23:56 +03:00
// fade out
2020-04-22 12:56:21 -04:00
GetChatHistory ( ) - > SetBgColor ( Color ( 0 , 0 , 0 , alpha ) ) ;
SetBgColor ( Color ( GetBgColor ( ) . r ( ) , GetBgColor ( ) . g ( ) , GetBgColor ( ) . b ( ) , alpha ) ) ;
m_pChatInput - > GetPrompt ( ) - > SetAlpha ( alpha ) ;
m_pChatInput - > GetInputPanel ( ) - > SetAlpha ( alpha ) ;
m_pFiltersButton - > SetAlpha ( alpha ) ;
}
}
}
}
void CBaseHudChat : : SetFilterFlag ( int iFilter )
{
m_iFilterFlags = iFilter ;
cl_chatfilters . SetValue ( m_iFilterFlags ) ;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
Color CBaseHudChat : : GetTextColorForClient ( TextColor colorNum , int clientIndex )
{
Color c ;
switch ( colorNum )
{
case COLOR_PLAYERNAME :
c = GetClientColor ( clientIndex ) ;
2023-10-03 17:23:56 +03:00
break ;
2020-04-22 12:56:21 -04:00
case COLOR_LOCATION :
c = g_ColorDarkGreen ;
break ;
case COLOR_ACHIEVEMENT :
{
vgui : : IScheme * pSourceScheme = vgui : : scheme ( ) - > GetIScheme ( vgui : : scheme ( ) - > GetScheme ( " SourceScheme " ) ) ;
if ( pSourceScheme )
{
c = pSourceScheme - > GetColor ( " SteamLightGreen " , GetBgColor ( ) ) ;
}
else
{
c = GetDefaultTextColor ( ) ;
}
}
break ;
default :
c = GetDefaultTextColor ( ) ;
}
return Color ( c [ 0 ] , c [ 1 ] , c [ 2 ] , 255 ) ;
}
//-----------------------------------------------------------------------------
Color CBaseHudChat : : GetDefaultTextColor ( void )
{
return g_ColorYellow ;
}
//-----------------------------------------------------------------------------
Color CBaseHudChat : : GetClientColor ( int clientIndex )
{
if ( clientIndex = = 0 ) // console msg
{
return g_ColorGreen ;
}
else if ( g_PR )
{
return g_ColorGrey ;
}
return g_ColorYellow ;
}
//-----------------------------------------------------------------------------
// Purpose: Parses a line of text for color markup and inserts it via Colorize()
//-----------------------------------------------------------------------------
void CBaseHudChatLine : : InsertAndColorizeText ( wchar_t * buf , int clientIndex )
{
if ( m_text )
{
delete [ ] m_text ;
m_text = NULL ;
}
m_textRanges . RemoveAll ( ) ;
m_text = CloneWString ( buf ) ;
CBaseHudChat * pChat = dynamic_cast < CBaseHudChat * > ( GetParent ( ) ) ;
if ( pChat = = NULL )
return ;
wchar_t * txt = m_text ;
int lineLen = wcslen ( m_text ) ;
2023-10-03 17:23:56 +03:00
if ( m_text [ 0 ] = = COLOR_PLAYERNAME | | m_text [ 0 ] = = COLOR_LOCATION | | m_text [ 0 ] = = COLOR_NORMAL | | m_text [ 0 ] = = COLOR_ACHIEVEMENT | | m_text [ 0 ] = = COLOR_MOD_CUSTOM | | m_text [ 0 ] = = COLOR_MOD_CUSTOM2 )
2020-04-22 12:56:21 -04:00
{
while ( txt & & * txt )
{
TextRange range ;
switch ( * txt )
{
case COLOR_PLAYERNAME :
case COLOR_LOCATION :
case COLOR_ACHIEVEMENT :
case COLOR_NORMAL :
2023-10-03 17:23:56 +03:00
case COLOR_MOD_CUSTOM :
case COLOR_MOD_CUSTOM2 :
2020-04-22 12:56:21 -04:00
{
// save this start
2023-10-03 17:23:56 +03:00
range . start = ( txt - m_text ) + 1 ;
2020-04-22 12:56:21 -04:00
range . color = pChat - > GetTextColorForClient ( ( TextColor ) ( * txt ) , clientIndex ) ;
range . end = lineLen ;
2023-10-03 17:23:56 +03:00
int count = m_textRanges . Count ( ) ;
if ( count )
2020-04-22 12:56:21 -04:00
{
2023-10-03 17:23:56 +03:00
m_textRanges [ count - 1 ] . end = range . start - 1 ;
2020-04-22 12:56:21 -04:00
}
2023-10-03 17:23:56 +03:00
m_textRanges . AddToTail ( range ) ;
2020-04-22 12:56:21 -04:00
}
+ + txt ;
break ;
2023-10-03 17:23:56 +03:00
default :
+ + txt ;
2020-04-22 12:56:21 -04:00
}
}
}
if ( ! m_textRanges . Count ( ) & & m_iNameLength > 0 & & m_text [ 0 ] = = COLOR_USEOLDCOLORS )
{
TextRange range ;
range . start = 0 ;
range . end = m_iNameStart ;
range . color = pChat - > GetTextColorForClient ( COLOR_NORMAL , clientIndex ) ;
m_textRanges . AddToTail ( range ) ;
range . start = m_iNameStart ;
range . end = m_iNameStart + m_iNameLength ;
range . color = pChat - > GetTextColorForClient ( COLOR_PLAYERNAME , clientIndex ) ;
m_textRanges . AddToTail ( range ) ;
range . start = range . end ;
range . end = wcslen ( m_text ) ;
range . color = pChat - > GetTextColorForClient ( COLOR_NORMAL , clientIndex ) ;
m_textRanges . AddToTail ( range ) ;
}
if ( ! m_textRanges . Count ( ) )
{
TextRange range ;
range . start = 0 ;
range . end = wcslen ( m_text ) ;
range . color = pChat - > GetTextColorForClient ( COLOR_NORMAL , clientIndex ) ;
m_textRanges . AddToTail ( range ) ;
}
for ( int i = 0 ; i < m_textRanges . Count ( ) ; + + i )
{
wchar_t * start = m_text + m_textRanges [ i ] . start ;
if ( * start > 0 & & * start < COLOR_MAX )
{
m_textRanges [ i ] . start + = 1 ;
}
}
Colorize ( ) ;
}
//-----------------------------------------------------------------------------
// Purpose: Inserts colored text into the RichText control at the given alpha
//-----------------------------------------------------------------------------
void CBaseHudChatLine : : Colorize ( int alpha )
{
2023-10-03 17:23:56 +03:00
MEM_ALLOC_CREDIT ( ) ;
2020-04-22 12:56:21 -04:00
// clear out text
SetText ( " " ) ;
CBaseHudChat * pChat = dynamic_cast < CBaseHudChat * > ( GetParent ( ) ) ;
if ( pChat & & pChat - > GetChatHistory ( ) )
{
pChat - > GetChatHistory ( ) - > InsertString ( " \n " ) ;
}
wchar_t wText [ 4096 ] ;
Color color ;
for ( int i = 0 ; i < m_textRanges . Count ( ) ; + + i )
{
wchar_t * start = m_text + m_textRanges [ i ] . start ;
int len = m_textRanges [ i ] . end - m_textRanges [ i ] . start + 1 ;
2023-10-03 17:23:56 +03:00
if ( len > 1 )
2020-04-22 12:56:21 -04:00
{
wcsncpy ( wText , start , len ) ;
wText [ len - 1 ] = 0 ;
color = m_textRanges [ i ] . color ;
2023-10-03 17:23:56 +03:00
color [ 3 ] = alpha ;
2020-04-22 12:56:21 -04:00
InsertColorChange ( color ) ;
InsertString ( wText ) ;
2023-10-03 17:23:56 +03:00
// TERROR: color console echo
ConColorMsg ( color , " %ls " , wText ) ;
2022-03-01 23:00:42 +03:00
CBaseHudChat * pChat = dynamic_cast < CBaseHudChat * > ( GetParent ( ) ) ;
2020-04-22 12:56:21 -04:00
if ( pChat & & pChat - > GetChatHistory ( ) )
{
pChat - > GetChatHistory ( ) - > InsertColorChange ( color ) ;
pChat - > GetChatHistory ( ) - > InsertString ( wText ) ;
pChat - > GetChatHistory ( ) - > InsertFade ( hud_saytext_time . GetFloat ( ) , CHAT_HISTORY_IDLE_FADE_TIME ) ;
if ( i = = m_textRanges . Count ( ) - 1 )
{
pChat - > GetChatHistory ( ) - > InsertFade ( - 1 , - 1 ) ;
}
}
}
}
2023-10-03 17:23:56 +03:00
Msg ( " \n " ) ;
2020-04-22 12:56:21 -04:00
InvalidateLayout ( true ) ;
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : CBaseHudChatLine
//-----------------------------------------------------------------------------
CBaseHudChatLine * CBaseHudChat : : FindUnusedChatLine ( void )
{
return m_ChatLine ;
}
void CBaseHudChat : : Send ( void )
{
2023-10-03 17:23:56 +03:00
if ( IsConsole ( ) )
{
// not used
return ;
}
2020-04-22 12:56:21 -04:00
2023-10-03 17:23:56 +03:00
wchar_t szTextbuf [ 1024 ] ;
2020-04-22 12:56:21 -04:00
m_pChatInput - > GetMessageText ( szTextbuf , sizeof ( szTextbuf ) ) ;
2023-10-03 17:23:56 +03:00
char ansi [ 1024 ] ;
2020-04-22 12:56:21 -04:00
g_pVGuiLocalize - > ConvertUnicodeToANSI ( szTextbuf , ansi , sizeof ( ansi ) ) ;
int len = Q_strlen ( ansi ) ;
2023-10-03 17:23:56 +03:00
// Remove this code before shipping
if ( ! V_strnicmp ( ansi , " bug! " , 4 ) | | ! V_strnicmp ( ansi , " bug: " , 4 ) )
{
char szTempStr [ 1024 ] ;
char szCommand [ 1024 ] ;
// Copy the string since we are going to hack it up in ParseTokens
V_strncpy ( szTempStr , ansi + 4 , sizeof ( szTempStr ) ) ;
// Auto submit if there is text after the keyword
// otherwise throw up the bug reporter ui
CUtlLinkedList < const char * > * tokens = ParseTokens ( szTempStr ) ;
if ( V_strlen ( szTempStr ) )
{
V_snprintf ( szCommand , sizeof ( szCommand ) , " bug -auto -title \" %s \" " , szTempStr ) ;
}
else
{
V_strncpy ( szCommand , " bug " , sizeof ( szCommand ) ) ;
}
FOR_EACH_LL ( ( * tokens ) , i )
{
V_snprintf ( szCommand , sizeof ( szCommand ) , " %s \" %s \" " , szCommand , tokens - > Element ( i ) ) ;
}
free ( tokens ) ;
//Msg("BUG: %s\n", szCommand);
engine - > ClientCmd_Unrestricted ( szCommand ) ;
}
2020-04-22 12:56:21 -04:00
// remove the \n
if ( len > 0 & &
ansi [ len - 1 ] = = ' \n ' )
{
ansi [ len - 1 ] = ' \0 ' ;
}
2023-10-03 17:23:56 +03:00
if ( len > 0 )
2020-04-22 12:56:21 -04:00
{
2023-10-03 17:23:56 +03:00
char szbuf [ 1024 ] ; // more than 128
2020-04-22 12:56:21 -04:00
Q_snprintf ( szbuf , sizeof ( szbuf ) , " %s \" %s \" " , m_nMessageMode = = MM_SAY ? " say " : " say_team " , ansi ) ;
engine - > ClientCmd_Unrestricted ( szbuf ) ;
}
m_pChatInput - > ClearEntry ( ) ;
2023-10-03 17:23:56 +03:00
m_nMessageMode = MM_NONE ; // TERROR
cl_chat_active . SetValue ( m_nMessageMode ) ;
2020-04-22 12:56:21 -04:00
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : vgui::Panel
//-----------------------------------------------------------------------------
vgui : : Panel * CBaseHudChat : : GetInputPanel ( void )
{
return m_pChatInput - > GetInputPanel ( ) ;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CBaseHudChat : : Clear ( void )
{
// Kill input prompt
StopMessageMode ( ) ;
2023-10-03 17:23:56 +03:00
m_flHistoryFadeTime = 0 ;
if ( GetChatHistory ( ) )
{
GetChatHistory ( ) - > ResetAllFades ( false , false , 0.0f ) ;
}
2020-04-22 12:56:21 -04:00
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *newmap -
//-----------------------------------------------------------------------------
void CBaseHudChat : : LevelInit ( const char * newmap )
{
Clear ( ) ;
}
void CBaseHudChat : : LevelShutdown ( void )
{
Clear ( ) ;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *fmt -
// ... -
//-----------------------------------------------------------------------------
void CBaseHudChat : : ChatPrintf ( int iPlayerIndex , int iFilter , const char * fmt , . . . )
{
va_list marker ;
char msg [ 4096 ] ;
va_start ( marker , fmt ) ;
Q_vsnprintf ( msg , sizeof ( msg ) , fmt , marker ) ;
va_end ( marker ) ;
// Strip any trailing '\n'
if ( strlen ( msg ) > 0 & & msg [ strlen ( msg ) - 1 ] = = ' \n ' )
{
msg [ strlen ( msg ) - 1 ] = 0 ;
}
// Strip leading \n characters ( or notify/color signifiers ) for empty string check
char * pmsg = msg ;
while ( * pmsg & & ( * pmsg = = ' \n ' | | ( * pmsg > 0 & & * pmsg < COLOR_MAX ) ) )
{
pmsg + + ;
}
if ( ! * pmsg )
return ;
// Now strip just newlines, since we want the color info for printing
pmsg = msg ;
while ( * pmsg & & ( * pmsg = = ' \n ' ) )
{
pmsg + + ;
}
if ( ! * pmsg )
return ;
CBaseHudChatLine * line = ( CBaseHudChatLine * ) FindUnusedChatLine ( ) ;
if ( ! line )
{
line = ( CBaseHudChatLine * ) FindUnusedChatLine ( ) ;
}
if ( ! line )
{
return ;
}
if ( iFilter ! = CHAT_FILTER_NONE )
{
if ( ! ( iFilter & GetFilterFlags ( ) ) )
return ;
}
if ( * pmsg < 32 )
{
hudlcd - > AddChatLine ( pmsg + 1 ) ;
}
else
{
hudlcd - > AddChatLine ( pmsg ) ;
}
line - > SetText ( " " ) ;
int iNameStart = 0 ;
int iNameLength = 0 ;
player_info_t sPlayerInfo ;
if ( iPlayerIndex = = 0 )
{
Q_memset ( & sPlayerInfo , 0 , sizeof ( player_info_t ) ) ;
Q_strncpy ( sPlayerInfo . name , " Console " , sizeof ( sPlayerInfo . name ) ) ;
}
else
{
engine - > GetPlayerInfo ( iPlayerIndex , & sPlayerInfo ) ;
}
int bufSize = ( strlen ( pmsg ) + 1 ) * sizeof ( wchar_t ) ;
wchar_t * wbuf = static_cast < wchar_t * > ( _alloca ( bufSize ) ) ;
if ( wbuf )
{
Color clrNameColor = GetClientColor ( iPlayerIndex ) ;
line - > SetExpireTime ( ) ;
g_pVGuiLocalize - > ConvertANSIToUnicode ( pmsg , wbuf , bufSize ) ;
// find the player's name in the unicode string, in case there is no color markup
const char * pName = sPlayerInfo . name ;
if ( pName )
{
wchar_t wideName [ MAX_PLAYER_NAME_LENGTH ] ;
g_pVGuiLocalize - > ConvertANSIToUnicode ( pName , wideName , sizeof ( wideName ) ) ;
const wchar_t * nameInString = wcsstr ( wbuf , wideName ) ;
if ( nameInString )
{
iNameStart = ( nameInString - wbuf ) ;
iNameLength = wcslen ( wideName ) ;
}
}
line - > SetVisible ( false ) ;
line - > SetNameStart ( iNameStart ) ;
line - > SetNameLength ( iNameLength ) ;
line - > SetNameColor ( clrNameColor ) ;
line - > InsertAndColorizeText ( wbuf , iPlayerIndex ) ;
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CBaseHudChat : : FireGameEvent ( IGameEvent * event )
{
const char * eventname = event - > GetName ( ) ;
if ( Q_strcmp ( " hltv_chat " , eventname ) = = 0 )
{
C_BasePlayer * player = C_BasePlayer : : GetLocalPlayer ( ) ;
if ( ! player )
return ;
ChatPrintf ( player - > entindex ( ) , CHAT_FILTER_NONE , " (SourceTV) %s " , event - > GetString ( " text " ) ) ;
}
2023-10-03 17:23:56 +03:00
}