summaryrefslogtreecommitdiff
path: root/engine/tmessage.cpp
diff options
context:
space:
mode:
authorFluorescentCIAAfricanAmerican <[email protected]>2020-04-22 12:56:21 -0400
committerFluorescentCIAAfricanAmerican <[email protected]>2020-04-22 12:56:21 -0400
commit3bf9df6b2785fa6d951086978a3e66f49427166a (patch)
tree2c0f1f0c63c4832882bc93814ebd2c2b1c6224e5 /engine/tmessage.cpp
downloadarchived-source-engine-2018-hl2-src-3bf9df6b2785fa6d951086978a3e66f49427166a.tar.xz
archived-source-engine-2018-hl2-src-3bf9df6b2785fa6d951086978a3e66f49427166a.zip
Diffstat (limited to 'engine/tmessage.cpp')
-rw-r--r--engine/tmessage.cpp636
1 files changed, 636 insertions, 0 deletions
diff --git a/engine/tmessage.cpp b/engine/tmessage.cpp
new file mode 100644
index 0000000..a31ce05
--- /dev/null
+++ b/engine/tmessage.cpp
@@ -0,0 +1,636 @@
+//========= Copyright Valve Corporation, All rights reserved. ============//
+//
+// Purpose:
+//
+// $NoKeywords: $
+//
+//=============================================================================//
+#include "quakedef.h"
+#include "cdll_int.h"
+#include "draw.h"
+#include "tmessage.h"
+#include "common.h"
+#include "characterset.h"
+#include "mem_fgets.h"
+#include "tier0/icommandline.h"
+
+// memdbgon must be the last include file in a .cpp file!!!
+#include "tier0/memdbgon.h"
+
+#define MSGFILE_NAME 0
+#define MSGFILE_TEXT 1
+#define MAX_MESSAGES 600 // I don't know if this table will balloon like every other feature in Half-Life
+ // But, for now, I've set this to a reasonable value
+// Defined in other files.
+static characterset_t g_WhiteSpace;
+
+client_textmessage_t gMessageParms;
+client_textmessage_t *gMessageTable = NULL;
+int gMessageTableCount = 0;
+
+char gNetworkTextMessageBuffer[MAX_NETMESSAGE][512];
+const char *gNetworkMessageNames[MAX_NETMESSAGE] = { NETWORK_MESSAGE1, NETWORK_MESSAGE2, NETWORK_MESSAGE3, NETWORK_MESSAGE4, NETWORK_MESSAGE5, NETWORK_MESSAGE6 };
+
+client_textmessage_t gNetworkTextMessage[MAX_NETMESSAGE] =
+{
+ { 0, // effect
+ 255,255,255,255,
+ 255,255,255,255,
+ -1.0f, // x
+ -1.0f, // y
+ 0.0f, // fadein
+ 0.0f, // fadeout
+ 0.0f, // holdtime
+ 0.0f, // fxTime,
+ NULL,//pVGuiSchemeFontName (NULL == default)
+ NETWORK_MESSAGE1, // pName message name.
+ gNetworkTextMessageBuffer[0] } // pMessage
+};
+
+char gDemoMessageBuffer[512];
+client_textmessage_t tm_demomessage =
+{
+ 0, // effect
+ 255,255,255,255,
+ 255,255,255,255,
+ -1.0f, // x
+ -1.0f, // y
+ 0.0f, // fadein
+ 0.0f, // fadeout
+ 0.0f, // holdtime
+ 0.0f, // fxTime,
+ NULL,// pVGuiSchemeFontName (NULL == default)
+ DEMO_MESSAGE, // pName message name.
+ gDemoMessageBuffer // pMessage
+};
+
+static client_textmessage_t orig_demo_message = tm_demomessage;
+
+static void TextMessageParse( byte *pMemFile, int fileSize );
+
+// The string "pText" is assumed to have all whitespace from both ends cut out
+int IsComment( char *pText )
+{
+ if ( pText )
+ {
+ int length = strlen( pText );
+
+ if ( length >= 2 && pText[0] == '/' && pText[1] == '/' )
+ return 1;
+
+ // No text?
+ if ( length > 0 )
+ return 0;
+ }
+ // No text is a comment too
+ return 1;
+}
+
+
+// The string "pText" is assumed to have all whitespace from both ends cut out
+int IsStartOfText( char *pText )
+{
+ if ( pText )
+ {
+ if ( pText[0] == '{' )
+ return 1;
+ }
+ return 0;
+}
+
+
+// The string "pText" is assumed to have all whitespace from both ends cut out
+int IsEndOfText( char *pText )
+{
+ if ( pText )
+ {
+ if ( pText[0] == '}' )
+ return 1;
+ }
+ return 0;
+}
+
+
+#if 0
+int IsWhiteSpace( char space )
+{
+ if ( space == ' ' || space == '\t' || space == '\r' || space == '\n' )
+ return 1;
+ return 0;
+}
+#else
+
+#define IsWhiteSpace(space) IN_CHARACTERSET( g_WhiteSpace, space )
+#endif
+
+
+const char *SkipSpace( const char *pText )
+{
+ if ( pText )
+ {
+ int pos = 0;
+ while ( pText[pos] && IsWhiteSpace( pText[pos] ) )
+ pos++;
+ return pText + pos;
+ }
+
+ return NULL;
+}
+
+
+const char *SkipText( const char *pText )
+{
+ if ( pText )
+ {
+ int pos = 0;
+ while ( pText[pos] && !IsWhiteSpace( pText[pos] ) )
+ pos++;
+ return pText + pos;
+ }
+
+ return NULL;
+}
+
+
+int ParseFloats( const char *pText, float *pFloat, int count )
+{
+ const char *pTemp = pText;
+ int index = 0;
+
+ while ( pTemp && count > 0 )
+ {
+ // Skip current token / float
+ pTemp = SkipText( pTemp );
+ // Skip any whitespace in between
+ pTemp = SkipSpace( pTemp );
+ if ( pTemp )
+ {
+ // Parse a float
+ pFloat[index] = (float)atof( pTemp );
+ count--;
+ index++;
+ }
+ }
+
+ if ( count == 0 )
+ return 1;
+
+ return 0;
+}
+
+int ParseString( char const *pText, char *buf, size_t bufsize )
+{
+ const char *pTemp = pText;
+
+ // Skip current token / float
+ pTemp = SkipText( pTemp );
+ // Skip any whitespace in between
+ pTemp = SkipSpace( pTemp );
+
+ if ( pTemp )
+ {
+ char const *pStart = pTemp;
+ pTemp = SkipText( pTemp );
+
+ int len = min( pTemp - pStart + 1, (int)bufsize - 1 );
+ Q_strncpy( buf, pStart, len );
+ buf[ len ] = 0;
+ return 1;
+ }
+
+ return 0;
+}
+
+
+// Trims all whitespace from the front and end of a string
+void TrimSpace( const char *source, char *dest )
+{
+ int start, end, length;
+
+ start = 0;
+ end = strlen( source );
+
+ while ( source[start] && IsWhiteSpace( source[start] ) )
+ start++;
+
+ end--;
+ while ( end > 0 && IsWhiteSpace( source[end] ) )
+ end--;
+
+ end++;
+
+ length = end - start;
+ if ( length > 0 )
+ memcpy( dest, source + start, length );
+ else
+ length = 0;
+
+ // Terminate the dest string
+ dest[ length ] = 0;
+}
+
+
+int IsToken( const char *pText, const char *pTokenName )
+{
+ if ( !pText || !pTokenName )
+ return 0;
+
+ if ( !Q_strnicmp( pText+1, pTokenName, strlen(pTokenName) ) )
+ return 1;
+
+ return 0;
+}
+
+static char g_pchSkipName[ 64 ];
+
+int ParseDirective( const char *pText )
+{
+ if ( pText && pText[0] == '$' )
+ {
+ float tempFloat[8];
+
+ if ( IsToken( pText, "position" ) )
+ {
+ if ( ParseFloats( pText, tempFloat, 2 ) )
+ {
+ gMessageParms.x = tempFloat[0];
+ gMessageParms.y = tempFloat[1];
+ }
+ }
+ else if ( IsToken( pText, "effect" ) )
+ {
+ if ( ParseFloats( pText, tempFloat, 1 ) )
+ {
+ gMessageParms.effect = (int)tempFloat[0];
+ }
+ }
+ else if ( IsToken( pText, "fxtime" ) )
+ {
+ if ( ParseFloats( pText, tempFloat, 1 ) )
+ {
+ gMessageParms.fxtime = tempFloat[0];
+ }
+ }
+ else if ( IsToken( pText, "color2" ) )
+ {
+ if ( ParseFloats( pText, tempFloat, 3 ) )
+ {
+ gMessageParms.r2 = (int)tempFloat[0];
+ gMessageParms.g2 = (int)tempFloat[1];
+ gMessageParms.b2 = (int)tempFloat[2];
+ }
+ }
+ else if ( IsToken( pText, "color" ) )
+ {
+ if ( ParseFloats( pText, tempFloat, 3 ) )
+ {
+ gMessageParms.r1 = (int)tempFloat[0];
+ gMessageParms.g1 = (int)tempFloat[1];
+ gMessageParms.b1 = (int)tempFloat[2];
+ }
+ }
+ else if ( IsToken( pText, "fadein" ) )
+ {
+ if ( ParseFloats( pText, tempFloat, 1 ) )
+ {
+ gMessageParms.fadein = tempFloat[0];
+ }
+ }
+ else if ( IsToken( pText, "fadeout" ) )
+ {
+ if ( ParseFloats( pText, tempFloat, 3 ) )
+ {
+ gMessageParms.fadeout = tempFloat[0];
+ }
+ }
+ else if ( IsToken( pText, "holdtime" ) )
+ {
+ if ( ParseFloats( pText, tempFloat, 3 ) )
+ {
+ gMessageParms.holdtime = tempFloat[0];
+ }
+ }
+ else if ( IsToken( pText, "boxsize" ) )
+ {
+ if ( ParseFloats( pText, tempFloat, 1 ) )
+ {
+ gMessageParms.bRoundedRectBackdropBox = tempFloat[0] != 0.0f;
+ gMessageParms.flBoxSize = tempFloat[0];
+ }
+ }
+ else if ( IsToken( pText, "boxcolor" ) )
+ {
+ if ( ParseFloats( pText, tempFloat, 4 ) )
+ {
+ for ( int i = 0; i < 4; ++i )
+ {
+ gMessageParms.boxcolor[ i ] = (byte)(int)tempFloat[ i ];
+ }
+ }
+ }
+ else if ( IsToken( pText, "clearmessage" ) )
+ {
+ if ( ParseString( pText, g_pchSkipName, sizeof( g_pchSkipName ) ) )
+ {
+ if ( !g_pchSkipName[ 0 ] || !Q_stricmp( g_pchSkipName, "0" ) )
+ {
+ gMessageParms.pClearMessage = NULL;
+ }
+ else
+ {
+ gMessageParms.pClearMessage = g_pchSkipName;
+ }
+ }
+ }
+ else
+ {
+ ConDMsg("Unknown token: %s\n", pText );
+ }
+
+ return 1;
+ }
+ return 0;
+}
+
+#define NAME_HEAP_SIZE 16384
+
+void TextMessageParse( byte *pMemFile, int fileSize )
+{
+ char buf[512], trim[512];
+ char *pCurrentText=0, *pNameHeap;
+ char currentName[512], nameHeap[ NAME_HEAP_SIZE ];
+ int lastNamePos;
+
+ int mode = MSGFILE_NAME; // Searching for a message name
+ int lineNumber, filePos, lastLinePos;
+ int messageCount;
+
+ client_textmessage_t textMessages[ MAX_MESSAGES ];
+
+ int i, nameHeapSize, textHeapSize, messageSize, nameOffset;
+
+ lastNamePos = 0;
+ lineNumber = 0;
+ filePos = 0;
+ lastLinePos = 0;
+ messageCount = 0;
+
+ bool bSpew = CommandLine()->FindParm( "-textmessagedebug" ) ? true : false;
+
+ CharacterSetBuild( &g_WhiteSpace, " \r\n\t" );
+
+ while( memfgets( pMemFile, fileSize, &filePos, buf, 512 ) != NULL )
+ {
+ if(messageCount>=MAX_MESSAGES)
+ {
+ Sys_Error("tmessage::TextMessageParse : messageCount>=MAX_MESSAGES");
+ }
+
+ TrimSpace( buf, trim );
+ switch( mode )
+ {
+ case MSGFILE_NAME:
+ if ( IsComment( trim ) ) // Skip comment lines
+ break;
+
+ if ( ParseDirective( trim ) ) // Is this a directive "$command"?, if so parse it and break
+ break;
+
+ if ( IsStartOfText( trim ) )
+ {
+ mode = MSGFILE_TEXT;
+ pCurrentText = (char*)(pMemFile + filePos);
+ break;
+ }
+ if ( IsEndOfText( trim ) )
+ {
+ ConDMsg("Unexpected '}' found, line %d\n", lineNumber );
+ return;
+ }
+ Q_strncpy( currentName, trim, sizeof( currentName ) );
+ break;
+
+ case MSGFILE_TEXT:
+ if ( IsEndOfText( trim ) )
+ {
+ int length = strlen(currentName);
+
+ // Save name on name heap
+ if ( lastNamePos + length > 8192 )
+ {
+ ConDMsg("Error parsing file!\n" );
+ return;
+ }
+ Q_strcpy( nameHeap + lastNamePos, currentName );
+
+ // Terminate text in-place in the memory file (it's temporary memory that will be deleted)
+ // If the string starts with #, it's a localization string and we don't
+ // want the \n (or \r) on the end or the Find() lookup will fail (so subtract 2)
+ if ( pCurrentText && pCurrentText[0] && pCurrentText[0] == '#' && lastLinePos > 1 &&
+ ( ( pMemFile[lastLinePos - 2] == '\n' ) || ( pMemFile[lastLinePos - 2] == '\r' ) ) )
+ {
+ pMemFile[ lastLinePos - 2 ] = 0;
+ }
+ else
+ {
+ pMemFile[ lastLinePos - 1 ] = 0;
+ }
+
+ // Save name/text on heap
+ textMessages[ messageCount ] = gMessageParms;
+ textMessages[ messageCount ].pName = nameHeap + lastNamePos;
+ lastNamePos += strlen(currentName) + 1;
+ if ( gMessageParms.pClearMessage )
+ {
+ Q_strncpy( nameHeap + lastNamePos, textMessages[ messageCount ].pClearMessage, Q_strlen( textMessages[ messageCount ].pClearMessage ) + 1 );
+ textMessages[ messageCount ].pClearMessage = nameHeap + lastNamePos;
+ lastNamePos += Q_strlen( textMessages[ messageCount ].pClearMessage ) + 1;
+ }
+ textMessages[ messageCount ].pMessage = pCurrentText;
+
+ if ( bSpew )
+ {
+ client_textmessage_t *m = &textMessages[ messageCount ];
+ Msg( "%d %s\n",
+ messageCount, m->pName ? m->pName : "(null)" );
+ Msg( " effect %d, color1(%d,%d,%d,%d), color2(%d,%d,%d,%d)\n",
+ m->effect, m->r1, m->g1, m->b1, m->a1, m->r2, m->g2, m->b2, m->a2 );
+ Msg( " pos %f,%f, fadein %f fadeout %f hold %f fxtime %f\n",
+ m->x, m->y, m->fadein, m->fadeout, m->holdtime, m->fxtime );
+ Msg( " '%s'\n", m->pMessage ? m->pMessage : "(null)" );
+
+ Msg( " box %s, size %f, color(%d,%d,%d,%d)\n",
+ m->bRoundedRectBackdropBox ? "yes" : "no", m->flBoxSize, m->boxcolor[ 0 ], m->boxcolor[ 1 ], m->boxcolor[ 2 ], m->boxcolor[ 3 ] );
+
+ if ( m->pClearMessage )
+ {
+ Msg( " will clear '%s'\n", m->pClearMessage );
+ }
+ }
+
+ messageCount++;
+
+ // Reset parser to search for names
+ mode = MSGFILE_NAME;
+ break;
+ }
+ if ( IsStartOfText( trim ) )
+ {
+ ConDMsg("Unexpected '{' found, line %d\n", lineNumber );
+ return;
+ }
+ break;
+ }
+ lineNumber++;
+ lastLinePos = filePos;
+
+ if ( messageCount >= MAX_MESSAGES )
+ {
+ ConMsg("WARNING: TOO MANY MESSAGES IN TITLES.TXT, MAX IS %d\n", MAX_MESSAGES );
+ break;
+ }
+ }
+
+ ConDMsg("Parsed %d text messages\n", messageCount );
+ nameHeapSize = lastNamePos;
+ textHeapSize = 0;
+ for ( i = 0; i < messageCount; i++ )
+ textHeapSize += strlen( textMessages[i].pMessage ) + 1;
+
+
+ messageSize = (messageCount * sizeof(client_textmessage_t));
+
+ // Must malloc because we need to be able to clear it after initialization
+ gMessageTable = (client_textmessage_t *)malloc( textHeapSize + nameHeapSize + messageSize );
+
+ // Copy table over
+ memcpy( gMessageTable, textMessages, messageSize );
+
+ // Copy Name heap
+ pNameHeap = ((char *)gMessageTable) + messageSize;
+ memcpy( pNameHeap, nameHeap, nameHeapSize );
+ nameOffset = pNameHeap - gMessageTable[0].pName;
+
+ // Copy text & fixup pointers
+ pCurrentText = pNameHeap + nameHeapSize;
+
+ for ( i = 0; i < messageCount; i++ )
+ {
+ gMessageTable[i].pName += nameOffset; // Adjust name pointer (parallel buffer)
+ if ( gMessageTable[ i ].pClearMessage )
+ {
+ gMessageTable[ i ].pClearMessage += nameOffset;
+ }
+ Q_strcpy( pCurrentText, gMessageTable[i].pMessage ); // Copy text over
+ gMessageTable[i].pMessage = pCurrentText;
+ pCurrentText += strlen( pCurrentText ) + 1;
+ }
+
+#if _DEBUG
+ if ( (pCurrentText - (char *)gMessageTable) != (textHeapSize + nameHeapSize + messageSize) )
+ ConMsg("Overflow text message buffer!!!!!\n");
+#endif
+ gMessageTableCount = messageCount;
+}
+
+void TextMessageShutdown( void )
+{
+ // Clear out any old data that's sitting around.
+ if ( gMessageTable )
+ {
+ free( gMessageTable );
+ gMessageTable = NULL;
+ }
+}
+
+void TextMessageInit( void )
+{
+ int fileSize;
+ byte *pMemFile;
+
+ // Clear out any old data that's sitting around.
+ if ( gMessageTable )
+ {
+ free( gMessageTable );
+ gMessageTable = NULL;
+ }
+
+ pMemFile = COM_LoadFile( "scripts/titles.txt", 5, &fileSize );
+
+ if ( pMemFile )
+ {
+ TextMessageParse( pMemFile, fileSize );
+ free( pMemFile );
+ }
+
+ int i;
+
+ for ( i = 0; i < MAX_NETMESSAGE; i++ )
+ {
+ gNetworkTextMessage[ i ].pMessage =
+ gNetworkTextMessageBuffer[ i ];
+ }
+}
+
+void TextMessage_DemoMessage( const char *pszMessage, float fFadeInTime, float fFadeOutTime, float fHoldTime )
+{
+ if ( !pszMessage || !pszMessage[0] )
+ return;
+
+ // Restore
+ tm_demomessage = orig_demo_message;
+
+ Q_strncpy( gDemoMessageBuffer, (char *)pszMessage, sizeof( gDemoMessageBuffer ) );
+ tm_demomessage.fadein = fFadeInTime;
+ tm_demomessage.fadeout = fFadeOutTime;
+ tm_demomessage.holdtime = fHoldTime;
+}
+
+//-----------------------------------------------------------------------------
+// Purpose:
+// Input : *pszMessage -
+// *message -
+//-----------------------------------------------------------------------------
+void TextMessage_DemoMessageFull( const char *pszMessage, client_textmessage_t const *message )
+{
+ Assert( message );
+ if ( !message )
+ return;
+
+ if ( !pszMessage || !pszMessage[0] )
+ return;
+
+ memcpy( &tm_demomessage, message, sizeof( tm_demomessage ) );
+ tm_demomessage.pMessage = orig_demo_message.pMessage;
+ tm_demomessage.pName = orig_demo_message.pName;
+ Q_strncpy( gDemoMessageBuffer, pszMessage, sizeof( gDemoMessageBuffer ) );
+}
+
+
+client_textmessage_t *TextMessageGet( const char *pName )
+{
+ if (!Q_stricmp( pName, DEMO_MESSAGE ))
+ return &tm_demomessage;
+
+ // HACKHACK -- add 4 "channels" of network text
+ if (!Q_stricmp( pName, NETWORK_MESSAGE1 ))
+ return gNetworkTextMessage;
+ else if (!Q_stricmp( pName, NETWORK_MESSAGE2 ))
+ return gNetworkTextMessage + 1;
+ else if (!Q_stricmp( pName, NETWORK_MESSAGE3 ))
+ return gNetworkTextMessage + 2;
+ else if (!Q_stricmp( pName, NETWORK_MESSAGE4 ))
+ return gNetworkTextMessage + 3;
+ else if (!Q_stricmp( pName, NETWORK_MESSAGE5 ))
+ return gNetworkTextMessage + 4;
+ else if (!Q_stricmp( pName, NETWORK_MESSAGE6 ))
+ return gNetworkTextMessage + 5;
+
+ for ( int i = 0; i < gMessageTableCount; i++ )
+ {
+ if ( !Q_stricmp( pName, gMessageTable[i].pName ) )
+ return &gMessageTable[i];
+ }
+
+ return NULL;
+} \ No newline at end of file