2022-09-18 15:37:21 +00:00
/*
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
Copyright ( C ) 2000 - 2013 , Raven Software , Inc .
Copyright ( C ) 2001 - 2013 , Activision , Inc .
Copyright ( C ) 2013 - 2015 , OpenJK contributors
This file is part of the OpenJK source code .
OpenJK is free software ; you can redistribute it and / or modify it
under the terms of the GNU General Public License version 2 as
published by the Free Software Foundation .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program ; if not , see < http : //www.gnu.org/licenses/>.
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
*/
// cg_text.c --
// this line must stay at top so the whole PCH thing works...
# include "cg_headers.h"
# include "cg_media.h"
//int precacheWav_i; // Current high index of precacheWav array
//precacheWav_t precacheWav[MAX_PRECACHEWAV];
//int precacheText_i; // Current high index of precacheText array
//precacheText_t precacheText[MAX_PRECACHETEXT];
extern vec4_t textcolor_caption ;
extern vec4_t textcolor_center ;
extern vec4_t textcolor_scroll ;
// display text in a supplied box, start at top left and going down by however many pixels I feel like internally,
// return value is NULL if all fitted, else char * of next char to continue from that didn't fit.
//
// (coords are in the usual 640x480 virtual space)...
//
// ( if you get the same char * returned as what you passed in, then none of it fitted at all (box too small) )
//
// this is execrable, and should NOT have had to've been done now, but...
//
float gfAdvanceHack = 0.0f ; // MUST default to this
int giLinesOutput ; // hack-city after release, only used by one function
//
const char * CG_DisplayBoxedText ( int iBoxX , int iBoxY , int iBoxWidth , int iBoxHeight ,
const char * psText , int iFontHandle , float fScale ,
const vec4_t v4Color )
{
giLinesOutput = 0 ;
cgi_R_SetColor ( v4Color ) ;
// Setup a reasonable vertical spacing (taiwanese & japanese need 1.5 fontheight, so use that for all)...
//
const int iFontHeight = cgi_R_Font_HeightPixels ( iFontHandle , fScale ) ;
const int iFontHeightAdvance = ( int ) ( ( ( gfAdvanceHack = = 0.0f ) ? 1.5f : gfAdvanceHack ) * ( float ) iFontHeight ) ;
int iYpos = iBoxY ; // start print pos
// this could probably be simplified now, but it was converted from something else I didn't originally write,
// and it works anyway so wtf...
//
const char * psCurrentTextReadPos = psText ;
const char * psReadPosAtLineStart = psCurrentTextReadPos ;
const char * psBestLineBreakSrcPos = psCurrentTextReadPos ;
const char * psLastGood_s ; // needed if we get a full screen of chars with no punctuation or space (see usage notes)
while ( * psCurrentTextReadPos & & ( iYpos + iFontHeight < ( iBoxY + iBoxHeight ) ) )
{
char sLineForDisplay [ 2048 ] ; // ott
// construct a line...
//
psCurrentTextReadPos = psReadPosAtLineStart ;
sLineForDisplay [ 0 ] = ' \0 ' ;
while ( * psCurrentTextReadPos )
{
psLastGood_s = psCurrentTextReadPos ;
// read letter...
//
qboolean bIsTrailingPunctuation ;
int iAdvanceCount ;
unsigned int uiLetter = cgi_AnyLanguage_ReadCharFromString ( psCurrentTextReadPos , & iAdvanceCount , & bIsTrailingPunctuation ) ;
psCurrentTextReadPos + = iAdvanceCount ;
// concat onto string so far...
//
if ( uiLetter = = 32 & & sLineForDisplay [ 0 ] = = ' \0 ' )
{
psReadPosAtLineStart + + ;
continue ; // unless it's a space at the start of a line, in which case ignore it.
}
if ( uiLetter > 255 )
{
Q_strcat ( sLineForDisplay , sizeof ( sLineForDisplay ) , va ( " %c%c " , uiLetter > > 8 , uiLetter & 0xFF ) ) ;
}
else
{
Q_strcat ( sLineForDisplay , sizeof ( sLineForDisplay ) , va ( " %c " , uiLetter & 0xFF ) ) ;
}
if ( uiLetter = = ' \n ' )
{
// explicit new line...
//
sLineForDisplay [ strlen ( sLineForDisplay ) - 1 ] = ' \0 ' ; // kill the CR
psReadPosAtLineStart = psCurrentTextReadPos ;
psBestLineBreakSrcPos = psCurrentTextReadPos ;
break ; // print this line
}
else
if ( cgi_R_Font_StrLenPixels ( sLineForDisplay , iFontHandle , fScale ) > = iBoxWidth )
{
// reached screen edge, so cap off string at bytepos after last good position...
//
if ( uiLetter > 255 & & bIsTrailingPunctuation & & ! cgi_Language_UsesSpaces ( ) )
{
// Special case, don't consider line breaking if you're on an asian punctuation char of
// a language that doesn't use spaces...
//
}
else
{
if ( psBestLineBreakSrcPos = = psReadPosAtLineStart )
{
// aarrrggh!!!!! we'll only get here is someone has fed in a (probably) garbage string,
// since it doesn't have a single space or punctuation mark right the way across one line
// of the screen. So far, this has only happened in testing when I hardwired a taiwanese
// string into this function while the game was running in english (which should NEVER happen
// normally). On the other hand I suppose it'psCurrentTextReadPos entirely possible that some taiwanese string
// might have no punctuation at all, so...
//
psBestLineBreakSrcPos = psLastGood_s ; // force a break after last good letter
}
sLineForDisplay [ psBestLineBreakSrcPos - psReadPosAtLineStart ] = ' \0 ' ;
psReadPosAtLineStart = psCurrentTextReadPos = psBestLineBreakSrcPos ;
break ; // print this line
}
}
// record last-good linebreak pos... (ie if we've just concat'd a punctuation point (western or asian) or space)
//
if ( bIsTrailingPunctuation | | uiLetter = = ' ' | | ( uiLetter > 255 & & ! cgi_Language_UsesSpaces ( ) ) )
{
psBestLineBreakSrcPos = psCurrentTextReadPos ;
}
}
// ... and print it...
//
cgi_R_Font_DrawString ( iBoxX , iYpos , sLineForDisplay , v4Color , iFontHandle , - 1 , fScale ) ;
iYpos + = iFontHeightAdvance ;
giLinesOutput + + ;
// and echo to console in dev mode...
//
if ( cg_developer . integer )
{
// Com_Printf( "%psCurrentTextReadPos\n", sLineForDisplay );
}
}
return psReadPosAtLineStart ;
}
/*
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
CAPTION TEXT
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
*/
void CG_CaptionTextStop ( void )
{
cg . captionTextTime = 0 ;
}
// try and get the correct StripEd text (with retry) for a given reference...
//
// returns 0 if failed, else strlen...
//
static int cg_SP_GetStringTextStringWithRetry ( const char * psReference , char * psDest , int iSizeofDest )
{
int iReturn ;
if ( psReference [ 0 ] = = ' # ' )
{
// then we know the striped package name is already built in, so do NOT try prepending anything else...
//
return cgi_SP_GetStringTextString ( va ( " %s " , psReference + 1 ) , psDest , iSizeofDest ) ;
}
for ( int i = 0 ; i < STRIPED_LEVELNAME_VARIATIONS ; i + + )
{
if ( cgs . stripLevelName [ i ] [ 0 ] ) // entry present?
{
iReturn = cgi_SP_GetStringTextString ( va ( " %s_%s " , cgs . stripLevelName [ i ] , psReference ) , psDest , iSizeofDest ) ;
if ( iReturn )
{
return iReturn ;
}
}
}
return 0 ;
}
// slightly confusingly, the char arg for this function is an audio filename of the form "path/path/filename",
// the "filename" part of which should be the same as the StripEd reference we're looking for in the current
// level's string package...
//
void CG_CaptionText ( const char * str , int sound )
{
const char * s , * holds ;
int i ;
int holdTime ;
char text [ 8192 ] = { 0 } ;
const float fFontScale = cgi_Language_IsAsian ( ) ? 0.8f : 1.0f ;
holds = strrchr ( str , ' / ' ) ;
if ( ! holds )
{
# ifndef FINAL_BUILD
Com_Printf ( " WARNING: CG_CaptionText given audio filename with no '/':'%s' \n " , str ) ;
# endif
return ;
}
i = cg_SP_GetStringTextStringWithRetry ( holds + 1 , text , sizeof ( text ) ) ;
//ensure we found a match
if ( ! i )
{
# ifndef FINAL_BUILD
// we only care about some sound dirs...
if ( ! Q_stricmpn ( str , " sound/chars/ " , 12 ) ) // whichever language it is, it'll be pathed as english at this point
{
Com_Printf ( " WARNING: CG_CaptionText given invalid text key: '%s' \n " , str ) ;
}
else
{
// anything else is probably stuff we don't care about. It certainly shouldn't be speech, anyway
}
# endif
return ;
}
const int fontHeight = ( int ) ( ( cgi_Language_IsAsian ( ) ? 1.4f : 1.0f ) * ( float ) cgi_R_Font_HeightPixels ( cgs . media . qhFontMedium , fFontScale ) ) ; // taiwanese & japanese need 1.5 fontheight spacing
cg . captionTextTime = cg . time ;
if ( in_camera ) {
cg . captionTextY = SCREEN_HEIGHT - ( client_camera . bar_height_dest / 2 ) ; // ths is now a centre'd Y, not a start Y
} else { //get above the hud
2023-03-28 21:21:28 +00:00
cg . captionTextY = ( int ) ( 0.80f * ( ( float ) SCREEN_HEIGHT - ( float ) fontHeight * 1.5f ) ) ; // do NOT move this, it has to fit in between the weapon HUD and the datapad update.
2022-09-18 15:37:21 +00:00
}
cg . captionTextCurrentLine = 0 ;
// count the number of lines for centering
cg . scrollTextLines = 1 ;
memset ( cg . captionText , 0 , sizeof ( cg . captionText ) ) ;
// Break into individual lines
i = 0 ; // this could be completely removed and replace by "cg.scrollTextLines-1", but wtf?
s = ( const char * ) & text ;
// tai...
2023-03-07 22:52:55 +00:00
// s="<22> ɨ<EFBFBD> <C9A8> <EFBFBD> <EFBFBD> <EFBFBD> <EFBFBD> դh<D5A4> w<EFBFBD> g<EFBFBD> w<EFBFBD> <77> <EFBFBD> F<EFBFBD> A<EFBFBD> ڤ]<5D> <> Ҧ<EFBFBD> <D2A6> o<EFBFBD> {<7B> <> <EFBFBD> i<EFBFBD> <69> <EFBFBD> u<EFBFBD> ө<EFBFBD> <D3A9> v<EFBFBD> C<EFBFBD> ܤ<EFBFBD> <DCA4> <EFBFBD> <EFBFBD> a<EFBFBD> A<EFBFBD> <41> hĵ<68> <C4B5> <EFBFBD> <EFBFBD> <EFBFBD> ǥ<EFBFBD> è<EFBFBD> o<EFBFBD> {<7B> F<EFBFBD> @<40> Ǫ<EFBFBD> <C7AA> p<EFBFBD> A<EFBFBD> dzƦ b<C6A6> <62> <EFBFBD> <EFBFBD> <EFBFBD> e<EFBFBD> <65> <EFBFBD> Ⱦ<EFBFBD> <C8BE> <EFBFBD> E<EFBFBD> ǧJ<C7A7> <4A> o<EFBFBD> C<EFBFBD> L<EFBFBD> <4C> <EFBFBD> ˦<EFBFBD> <CBA6> ~<7E> <> ϸ`<60> A<EFBFBD> <41> L<EFBFBD> F<EFBFBD> h<EFBFBD> h<EFBFBD> <68> <EFBFBD> ơC<C6A1> {<7B> b<EFBFBD> L<EFBFBD> <4C> <EFBFBD> <EFBFBD> <EFBFBD> H<EFBFBD> <48> A<EFBFBD> åB<C3A5> ¯٠n<D9AD> <6E> <EFBFBD> <EFBFBD> <EFBFBD> f<EFBFBD> r<EFBFBD> C<EFBFBD> ھ ڳ̷s<CCB7> <73> <EFBFBD> <EFBFBD> <EFBFBD> i<EFBFBD> A<EFBFBD> ǧJ<C7A7> <4A> o<EFBFBD> H<EFBFBD> Υ L<CEA5> <4C> <EFBFBD> ҦФw<D0A4> g<EFBFBD> <67> <EFBFBD> <EFBFBD> <EFBFBD> <EFBFBD> <EFBFBD> ڤF<DAA4> <46> <EFBFBD> <EFBFBD> <EFBFBD> C<EFBFBD> ڨ<EFBFBD> <DAA8> R<EFBFBD> Ӱl<D3B0> <6C> <EFBFBD> ǧJ<C7A7> <4A> o<EFBFBD> H<EFBFBD> α ϥX<CFA5> Ҧ<EFBFBD> <D2A6> H<EFBFBD> <48> C<EFBFBD> o<EFBFBD> ä<EFBFBD> <C3A4> e<EFBFBD> <65> <EFBFBD> C";
2022-09-18 15:37:21 +00:00
// kor...
2023-03-07 22:52:55 +00:00
// s="Wp:<3A> <> Ÿ<EFBFBD> <C5B8> <EFBFBD> ̴<EFBFBD> <20> ָ<EFBFBD> . <20> ׵<EFBFBD> <D7B5> <EFBFBD> <20> <> <EFBFBD> Ѵ <EFBFBD> <D1B4> <20> װ<EFBFBD> <20> <> <EFBFBD> <EFBFBD> <EFBFBD> <EFBFBD> <20> <> <EFBFBD> <EFBFBD> ϰڴ<CFB0> .<2E> <> Ÿ<EFBFBD> <C5B8> <EFBFBD> ̴<EFBFBD> <20> ָ<EFBFBD> . <20> ׵<EFBFBD> <D7B5> <EFBFBD> <20> <> <EFBFBD> Ѵ <EFBFBD> <D1B4> <20> װ<EFBFBD> <20> <> <EFBFBD> <EFBFBD> <EFBFBD> <EFBFBD> <20> <> <EFBFBD> <EFBFBD> ϰڴ<CFB0> .";
2022-09-18 15:37:21 +00:00
holds = s ;
int iPlayingTimeMS = cgi_S_GetSampleLength ( sound ) ;
int iLengthInChars = strlen ( s ) ; //cgi_R_Font_StrLenChars(s); // strlen is also good for MBCS in this instance, since it's for timing
if ( iLengthInChars = = 0 )
{
iLengthInChars = 1 ;
}
cg . captionLetterTime = iPlayingTimeMS / iLengthInChars ;
const char * psBestLineBreakSrcPos = s ;
const char * psLastGood_s ; // needed if we get a full screen of chars with no punctuation or space (see usage notes)
while ( * s )
{
psLastGood_s = s ;
// read letter...
//
qboolean bIsTrailingPunctuation ;
int iAdvanceCount ;
unsigned int uiLetter = cgi_AnyLanguage_ReadCharFromString ( s , & iAdvanceCount , & bIsTrailingPunctuation ) ;
s + = iAdvanceCount ;
// concat onto string so far...
//
if ( uiLetter = = 32 & & cg . captionText [ i ] [ 0 ] = = ' \0 ' )
{
holds + + ;
continue ; // unless it's a space at the start of a line, in which case ignore it.
}
if ( uiLetter > 255 )
{
Q_strcat ( cg . captionText [ i ] , sizeof ( cg . captionText [ i ] ) , va ( " %c%c " , uiLetter > > 8 , uiLetter & 0xFF ) ) ;
}
else
{
Q_strcat ( cg . captionText [ i ] , sizeof ( cg . captionText [ i ] ) , va ( " %c " , uiLetter & 0xFF ) ) ;
}
if ( uiLetter = = ' \n ' )
{
// explicit new line...
//
cg . captionText [ i ] [ strlen ( cg . captionText [ i ] ) - 1 ] = ' \0 ' ; // kill the CR
i + + ;
holds = s ;
psBestLineBreakSrcPos = s ;
cg . scrollTextLines + + ;
}
else
if ( cgi_R_Font_StrLenPixels ( cg . captionText [ i ] , cgs . media . qhFontMedium , fFontScale ) > = SCREEN_WIDTH )
{
// reached screen edge, so cap off string at bytepos after last good position...
//
if ( uiLetter > 255 & & bIsTrailingPunctuation & & ! cgi_Language_UsesSpaces ( ) )
{
// Special case, don't consider line breaking if you're on an asian punctuation char of
// a language that doesn't use spaces...
//
}
else
{
if ( psBestLineBreakSrcPos = = holds )
{
// aarrrggh!!!!! we'll only get here is someone has fed in a (probably) garbage string,
// since it doesn't have a single space or punctuation mark right the way across one line
// of the screen. So far, this has only happened in testing when I hardwired a taiwanese
// string into this function while the game was running in english (which should NEVER happen
// normally). On the other hand I suppose it's entirely possible that some taiwanese string
// might have no punctuation at all, so...
//
psBestLineBreakSrcPos = psLastGood_s ; // force a break after last good letter
}
cg . captionText [ i ] [ psBestLineBreakSrcPos - holds ] = ' \0 ' ;
holds = s = psBestLineBreakSrcPos ;
i + + ;
cg . scrollTextLines + + ;
}
}
// record last-good linebreak pos... (ie if we've just concat'd a punctuation point (western or asian) or space)
//
if ( bIsTrailingPunctuation | | uiLetter = = ' ' | | ( uiLetter > 255 & & ! cgi_Language_UsesSpaces ( ) ) )
{
psBestLineBreakSrcPos = s ;
}
}
// calc the length of time to hold each 2 lines of text on the screen.... presumably this works?
//
holdTime = strlen ( cg . captionText [ 0 ] ) ;
if ( cg . scrollTextLines > 1 )
{
holdTime + = strlen ( cg . captionText [ 1 ] ) ; // strlen is also good for MBCS in this instance, since it's for timing
}
cg . captionNextTextTime = cg . time + ( holdTime * cg . captionLetterTime ) ;
cg . scrollTextTime = 0 ; // No scrolling during captions
//Echo to console in dev mode
if ( cg_developer . integer )
{
Com_Printf ( " %s \n " , cg . captionText [ 0 ] ) ; // ste: was [i], but surely sentence 0 is more useful than last?
}
}
2023-03-28 21:21:28 +00:00
void CG_DrawCaptionText ( bool inImmersiveCamera )
2022-09-18 15:37:21 +00:00
{
int i ;
int x , y , w ;
int holdTime ;
if ( ! cg . captionTextTime )
{
return ;
}
const float fFontScale = cgi_Language_IsAsian ( ) ? 0.8f : 1.0f ;
if ( cg_skippingcin . integer ! = 0 )
{
cg . captionTextTime = 0 ;
return ;
}
if ( cg . captionNextTextTime < cg . time )
{
cg . captionTextCurrentLine + = 2 ;
if ( cg . captionTextCurrentLine > = cg . scrollTextLines )
{
cg . captionTextTime = 0 ;
return ;
}
else
{
holdTime = strlen ( cg . captionText [ cg . captionTextCurrentLine ] ) ;
if ( cg . scrollTextLines > = cg . captionTextCurrentLine )
{
// ( strlen is also good for MBCS in this instance, since it's for timing -ste)
//
holdTime + = strlen ( cg . captionText [ cg . captionTextCurrentLine + 1 ] ) ;
}
cg . captionNextTextTime = cg . time + ( holdTime * cg . captionLetterTime ) ; //50);
}
}
// Give a color if one wasn't given
if ( ( textcolor_caption [ 0 ] = = 0 ) & & ( textcolor_caption [ 1 ] = = 0 ) & &
( textcolor_caption [ 2 ] = = 0 ) & & ( textcolor_caption [ 3 ] = = 0 ) )
{
VectorCopy4 ( colorTable [ CT_WHITE ] , textcolor_caption ) ;
}
cgi_R_SetColor ( textcolor_caption ) ;
// Set Y of the first line (varies if only printing one line of text)
// (this all works, please don't mess with it)
2023-03-28 21:21:28 +00:00
const int fontHeight = ( int ) ( 3.0f * ( float ) cgi_R_Font_HeightPixels ( cgs . media . qhFontSmall , fFontScale * FONT_SCALE ) ) ;
2022-09-18 15:37:21 +00:00
const bool bPrinting2Lines = ! ! ( cg . captionText [ cg . captionTextCurrentLine + 1 ] [ 0 ] ) ;
y = cg . captionTextY - ( ( float ) fontHeight * ( bPrinting2Lines ? 1 : 0.5f ) ) ; // captionTextY was a centered Y pos, not a top one
y - = cgi_Language_IsAsian ( ) ? 0 : 4 ;
2023-03-28 21:21:28 +00:00
if ( inImmersiveCamera ) {
y - = 100 ;
}
2022-09-18 15:37:21 +00:00
for ( i = cg . captionTextCurrentLine ; i < cg . captionTextCurrentLine + 2 ; + + i )
{
2022-10-08 09:43:18 +00:00
w = cgi_R_Font_StrLenPixels ( cg . captionText [ i ] , cgs . media . qhFontSmall , fFontScale * FONT_SCALE ) ;
2022-09-18 15:37:21 +00:00
if ( w )
{
2023-03-28 21:21:28 +00:00
int offset = w / 2 ;
int tempX = SCREEN_WIDTH / 2 ;
2022-10-08 09:43:18 +00:00
int tempY = y ;
2023-03-28 21:21:28 +00:00
cgi_R_Font_DrawString ( tempX - offset , tempY , cg . captionText [ i ] , textcolor_caption , cgs . media . qhFontSmall , - 1 , fFontScale * FONT_SCALE ) ;
2022-09-18 15:37:21 +00:00
y + = fontHeight ;
}
}
cgi_R_SetColor ( NULL ) ;
}
/*
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
SCROLL TEXT
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
CG_ScrollText - split text up into seperate lines
' str ' arg is StripEd string reference , eg " CREDITS_RAVEN "
*/
int giScrollTextPixelWidth = SCREEN_WIDTH ;
void CG_ScrollText ( const char * str , int iPixelWidth )
{
const char * s , * holds ;
int i ; //, len;//, numChars;
giScrollTextPixelWidth = iPixelWidth ;
// first, ask the strlen of the final string...
//
i = cgi_SP_GetStringTextString ( str , NULL , 0 ) ;
//ensure we found a match
if ( ! i )
{
# ifndef FINAL_BUILD
Com_Printf ( " WARNING: CG_ScrollText given invalid text key :'%s' \n " , str ) ;
# endif
return ;
}
//
// malloc space to hold it...
//
char * psText = ( char * ) cgi_Z_Malloc ( i + 1 , TAG_TEMP_WORKSPACE ) ;
//
// now get the string...
//
i = cgi_SP_GetStringTextString ( str , psText , i + 1 ) ;
//ensure we found a match
if ( ! i )
{
assert ( 0 ) ; // should never get here now, but wtf?
cgi_Z_Free ( psText ) ;
# ifndef FINAL_BUILD
Com_Printf ( " WARNING: CG_ScrollText given invalid text key :'%s' \n " , str ) ;
# endif
return ;
}
cg . scrollTextTime = cg . time ;
cg . printTextY = SCREEN_HEIGHT ;
cg . scrollTextLines = 1 ;
s = psText ;
i = 0 ;
holds = s ;
const char * psBestLineBreakSrcPos = s ;
const char * psLastGood_s ; // needed if we get a full screen of chars with no punctuation or space (see usage notes)
while ( * s )
{
psLastGood_s = s ;
// read letter...
//
qboolean bIsTrailingPunctuation ;
int iAdvanceCount ;
unsigned int uiLetter = cgi_AnyLanguage_ReadCharFromString ( s , & iAdvanceCount , & bIsTrailingPunctuation ) ;
s + = iAdvanceCount ;
// concat onto string so far...
//
if ( uiLetter = = 32 & & cg . printText [ i ] [ 0 ] = = ' \0 ' )
{
holds + + ;
continue ; // unless it's a space at the start of a line, in which case ignore it.
}
if ( uiLetter > 255 )
{
Q_strcat ( cg . printText [ i ] , sizeof ( cg . printText [ i ] ) , va ( " %c%c " , uiLetter > > 8 , uiLetter & 0xFF ) ) ;
}
else
{
Q_strcat ( cg . printText [ i ] , sizeof ( cg . printText [ i ] ) , va ( " %c " , uiLetter & 0xFF ) ) ;
}
// record last-good linebreak pos... (ie if we've just concat'd a punctuation point (western or asian) or space)
//
if ( bIsTrailingPunctuation | | uiLetter = = ' ' )
{
psBestLineBreakSrcPos = s ;
}
if ( uiLetter = = ' \n ' )
{
// explicit new line...
//
cg . printText [ i ] [ strlen ( cg . printText [ i ] ) - 1 ] = ' \0 ' ; // kill the CR
i + + ;
assert ( i < ( int ) ( sizeof ( cg . printText ) / sizeof ( cg . printText [ 0 ] ) ) ) ;
if ( i > = ( int ) ( sizeof ( cg . printText ) / sizeof ( cg . printText [ 0 ] ) ) )
{
break ;
}
holds = s ;
cg . scrollTextLines + + ;
}
else
if ( cgi_R_Font_StrLenPixels ( cg . printText [ i ] , cgs . media . qhFontMedium , 1.0f ) > = iPixelWidth )
{
// reached screen edge, so cap off string at bytepos after last good position...
//
if ( psBestLineBreakSrcPos = = holds )
{
// aarrrggh!!!!! we'll only get here is someone has fed in a (probably) garbage string,
// since it doesn't have a single space or punctuation mark right the way across one line
// of the screen. So far, this has only happened in testing when I hardwired a taiwanese
// string into this function while the game was running in english (which should NEVER happen
// normally). On the other hand I suppose it's entirely possible that some taiwanese string
// might have no punctuation at all, so...
//
psBestLineBreakSrcPos = psLastGood_s ; // force a break after last good letter
}
cg . printText [ i ] [ psBestLineBreakSrcPos - holds ] = ' \0 ' ;
holds = s = psBestLineBreakSrcPos ;
i + + ;
assert ( i < ( int ) ( sizeof ( cg . printText ) / sizeof ( cg . printText [ 0 ] ) ) ) ;
cg . scrollTextLines + + ;
}
}
cg . captionTextTime = 0 ; // No captions during scrolling
cgi_Z_Free ( psText ) ;
}
// draws using [textcolor_scroll]...
//
# define SCROLL_LPM (1 / 50.0) // 1 line per 50 ms
void CG_DrawScrollText ( void )
{
int i ;
int x , y ;
2022-10-08 09:43:18 +00:00
const int fontHeight = ( int ) ( 1.5f * ( float ) cgi_R_Font_HeightPixels ( cgs . media . qhFontSmall , FONT_SCALE ) ) ; // taiwanese & japanese need 1.5 fontheight spacing
2022-09-18 15:37:21 +00:00
if ( ! cg . scrollTextTime )
{
return ;
}
cgi_R_SetColor ( textcolor_scroll ) ;
y = cg . printTextY - ( cg . time - cg . scrollTextTime ) * SCROLL_LPM ;
// cgi_R_Font_DrawString(320, 200, va("Scrolltext printing @ %d",y), colorTable[CT_LTGOLD1], cgs.media.qhFontMedium, -1, 1.0f);
// See if text has finished scrolling off screen
if ( ( y + cg . scrollTextLines * fontHeight ) < 1 )
{
cg . scrollTextTime = 0 ;
return ;
}
for ( i = 0 ; i < cg . scrollTextLines ; + + i )
{
// Is this line off top of screen?
if ( ( y + ( ( i + 1 ) * fontHeight ) ) < 1 )
{
y + = fontHeight ;
continue ;
}
// or past bottom of screen?
else if ( y > SCREEN_HEIGHT )
{
break ;
}
// w = cgi_R_Font_StrLenPixels(cg.printText[i], cgs.media.qhFontMedium, 1.0f);
// if (w)
{
2023-03-28 21:21:28 +00:00
int offset = giScrollTextPixelWidth / 2 ;
int tempX = SCREEN_WIDTH / 2 ;
2022-10-08 09:43:18 +00:00
int tempY = y ;
2023-03-28 21:21:28 +00:00
cgi_R_Font_DrawString ( tempX - offset , tempY , cg . printText [ i ] , textcolor_scroll , cgs . media . qhFontSmall , - 1 , FONT_SCALE ) ;
2022-09-18 15:37:21 +00:00
y + = fontHeight ;
}
}
cgi_R_SetColor ( NULL ) ;
}
/*
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
CENTER PRINTING
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
*/
/*
= = = = = = = = = = = = = =
CG_CenterPrint
Called for important messages that should stay in the center of the screen
for a few moments
= = = = = = = = = = = = = =
*/
2023-05-14 21:50:24 +00:00
void CG_CenterPrint ( const char * str , int y , int delayOverride ) {
2022-09-18 15:37:21 +00:00
char * s ;
// Find text to match the str given
if ( * str = = ' @ ' )
{
int i ;
i = cgi_SP_GetStringTextString ( str + 1 , cg . centerPrint , sizeof ( cg . centerPrint ) ) ;
if ( ! i )
{
Com_Printf ( S_COLOR_RED " CG_CenterPrint: cannot find reference '%s' in StringPackage! \n " , str ) ;
Q_strncpyz ( cg . centerPrint , str , sizeof ( cg . centerPrint ) ) ;
}
}
else
{
Q_strncpyz ( cg . centerPrint , str , sizeof ( cg . centerPrint ) ) ;
}
cg . centerPrintTime = cg . time ;
2023-05-14 21:50:24 +00:00
cg . centerPrintDelayOverride = delayOverride ;
2022-09-18 15:37:21 +00:00
cg . centerPrintY = y ;
// count the number of lines for centering
cg . centerPrintLines = 1 ;
s = cg . centerPrint ;
while ( * s ) {
if ( * s = = ' \n ' )
cg . centerPrintLines + + ;
s + + ;
}
}
/*
= = = = = = = = = = = = = = = = = = =
CG_DrawCenterString
= = = = = = = = = = = = = = = = = = =
*/
void CG_DrawCenterString ( void )
{
char * start ;
unsigned int l ;
int x , y , w ;
float * color ;
if ( ! cg . centerPrintTime ) {
return ;
}
2023-05-14 21:50:24 +00:00
//Default time is 2 seconds
int printTime = cg . centerPrintDelayOverride = = - 1 ? 2000 : cg . centerPrintDelayOverride ;
color = CG_FadeColor ( cg . centerPrintTime , printTime ) ;
2022-09-18 15:37:21 +00:00
if ( ! color ) {
return ;
}
if ( ( textcolor_center [ 0 ] = = 0 ) & & ( textcolor_center [ 1 ] = = 0 ) & &
( textcolor_center [ 2 ] = = 0 ) & & ( textcolor_center [ 3 ] = = 0 ) )
{
VectorCopy4 ( colorTable [ CT_WHITE ] , textcolor_center ) ;
}
start = cg . centerPrint ;
2022-10-08 09:43:18 +00:00
const int fontHeight = cgi_R_Font_HeightPixels ( cgs . media . qhFontSmall , FONT_SCALE ) ;
2022-09-18 15:37:21 +00:00
y = cg . centerPrintY - ( cg . centerPrintLines * fontHeight ) / 2 ;
while ( 1 ) {
char linebuffer [ 1024 ] ;
// this is kind of unpleasant when dealing with MBCS, but...
//
const char * psString = start ;
int iOutIndex = 0 ;
for ( l = 0 ; l < sizeof ( linebuffer ) - 1 ; l + + ) {
int iAdvanceCount ;
unsigned int uiLetter = cgi_AnyLanguage_ReadCharFromString ( psString , & iAdvanceCount ) ;
psString + = iAdvanceCount ;
if ( ! uiLetter | | uiLetter = = ' \n ' ) {
break ;
}
if ( uiLetter > 255 )
{
linebuffer [ iOutIndex + + ] = uiLetter > > 8 ;
linebuffer [ iOutIndex + + ] = uiLetter & 0xFF ;
}
else
{
linebuffer [ iOutIndex + + ] = uiLetter & 0xFF ;
}
}
linebuffer [ iOutIndex + + ] = ' \0 ' ;
2022-10-08 09:43:18 +00:00
w = cgi_R_Font_StrLenPixels ( linebuffer , cgs . media . qhFontSmall , FONT_SCALE ) ;
2022-09-18 15:37:21 +00:00
2023-03-28 21:21:28 +00:00
int offset = w / 2 ;
int tempX = SCREEN_WIDTH / 2 ;
2022-10-08 09:43:18 +00:00
int tempY = y ;
2023-03-28 21:21:28 +00:00
cgi_R_Font_DrawString ( tempX - offset , tempY , linebuffer , textcolor_center , cgs . media . qhFontSmall , - 1 , FONT_SCALE ) ;
2022-09-18 15:37:21 +00:00
y + = fontHeight ;
while ( * start & & ( * start ! = ' \n ' ) ) {
start + + ;
}
if ( ! * start ) {
break ;
}
start + + ;
}
}