2013-06-26 15:22:04 -07:00
//========= Copyright Valve Corporation, All rights reserved. ============//
# if defined( WIN32 ) && !defined( _X360 )
# include <windows.h>
# elif defined( POSIX )
# include <iconv.h>
# endif
# include "tier1/ilocalize.h"
# include "utlstring.h"
2013-10-06 13:46:15 -04:00
# ifdef _WIN32
2013-06-26 15:22:04 -07:00
# pragma warning( disable: 4018 ) // '<' : signed/unsigned mismatch
2013-10-06 13:46:15 -04:00
# endif
2013-06-26 15:22:04 -07:00
//-----------------------------------------------------------------------------
// Purpose: converts an english string to unicode
//-----------------------------------------------------------------------------
int ILocalize : : ConvertANSIToUnicode ( const char * ansi , wchar_t * unicode , int unicodeBufferSizeInBytes )
{
# ifdef POSIX
return Q_UTF8ToUnicode ( ansi , unicode , unicodeBufferSizeInBytes ) ;
# else
int chars = MultiByteToWideChar ( CP_UTF8 , 0 , ansi , - 1 , unicode , unicodeBufferSizeInBytes / sizeof ( wchar_t ) ) ;
unicode [ ( unicodeBufferSizeInBytes / sizeof ( wchar_t ) ) - 1 ] = 0 ;
return chars ;
# endif
}
//-----------------------------------------------------------------------------
// Purpose: converts an unicode string to an english string
//-----------------------------------------------------------------------------
int ILocalize : : ConvertUnicodeToANSI ( const wchar_t * unicode , char * ansi , int ansiBufferSize )
{
# ifdef POSIX
return Q_UnicodeToUTF8 ( unicode , ansi , ansiBufferSize ) ;
# else
int result = WideCharToMultiByte ( CP_UTF8 , 0 , unicode , - 1 , ansi , ansiBufferSize , NULL , NULL ) ;
ansi [ ansiBufferSize - 1 ] = 0 ;
return result ;
# endif
}
//-----------------------------------------------------------------------------
// Purpose: construct string helper
//-----------------------------------------------------------------------------
template < typename T >
void ConstructStringVArgsInternal_Impl ( T * unicodeOutput , int unicodeBufferSizeInBytes , const T * formatString , int numFormatParameters , va_list argList )
{
static const int k_cMaxFormatStringArguments = 9 ; // We only look one character ahead and start at %s1
Assert ( numFormatParameters < = k_cMaxFormatStringArguments ) ;
// Safety check
if ( unicodeOutput = = NULL | | unicodeBufferSizeInBytes < 1 )
{
return ;
}
if ( ! formatString | | numFormatParameters > k_cMaxFormatStringArguments )
{
unicodeOutput [ 0 ] = 0 ;
return ;
}
int unicodeBufferSize = unicodeBufferSizeInBytes / sizeof ( T ) ;
const T * searchPos = formatString ;
T * outputPos = unicodeOutput ;
T * argParams [ k_cMaxFormatStringArguments ] ;
for ( int i = 0 ; i < numFormatParameters ; i + + )
{
argParams [ i ] = va_arg ( argList , T * ) ;
}
//assumes we can't have %s10
//assume both are 0 terminated?
int formatLength = StringFuncs < T > : : Length ( formatString ) ;
while ( searchPos [ 0 ] ! = ' \0 ' & & unicodeBufferSize > 1 )
{
if ( formatLength > = 3 & & searchPos [ 0 ] = = ' % ' & & searchPos [ 1 ] = = ' s ' )
{
//this is an escape sequence - %s1, %s2 etc, up to %s9
int argindex = ( searchPos [ 2 ] ) - ' 0 ' - 1 ; // 0 for %s1, 1 for %s2, etc.
if ( argindex < 0 | | argindex > k_cMaxFormatStringArguments )
{
Warning ( " Bad format string in CLocalizeStringTable::ConstructString \n " ) ;
* outputPos = ' \0 ' ;
return ;
}
if ( argindex < numFormatParameters )
{
T const * param = argParams [ argindex ] ;
if ( param = = NULL )
param = StringFuncs < T > : : NullDebugString ( ) ;
int paramSize = StringFuncs < T > : : Length ( param ) ;
if ( paramSize > = unicodeBufferSize )
{
paramSize = unicodeBufferSize - 1 ;
}
memcpy ( outputPos , param , paramSize * sizeof ( T ) ) ;
unicodeBufferSize - = paramSize ;
outputPos + = paramSize ;
searchPos + = 3 ;
formatLength - = 3 ;
}
else
{
AssertMsg ( argindex < numFormatParameters , " ConstructStringVArgsInternal_Impl() - Found a %s# escape sequence whose index was more than the number of args. " ) ;
//copy it over, char by char
* outputPos = * searchPos ;
outputPos + + ;
unicodeBufferSize - - ;
searchPos + + ;
formatLength - - ;
}
}
else
{
//copy it over, char by char
* outputPos = * searchPos ;
outputPos + + ;
unicodeBufferSize - - ;
searchPos + + ;
formatLength - - ;
}
}
// ensure null termination
Assert ( outputPos - unicodeOutput < unicodeBufferSizeInBytes / sizeof ( T ) ) ;
* outputPos = L ' \0 ' ;
}
void ILocalize : : ConstructStringVArgsInternal ( char * unicodeOutput , int unicodeBufferSizeInBytes , const char * formatString , int numFormatParameters , va_list argList )
{
ConstructStringVArgsInternal_Impl < char > ( unicodeOutput , unicodeBufferSizeInBytes , formatString , numFormatParameters , argList ) ;
}
void ILocalize : : ConstructStringVArgsInternal ( wchar_t * unicodeOutput , int unicodeBufferSizeInBytes , const wchar_t * formatString , int numFormatParameters , va_list argList )
{
ConstructStringVArgsInternal_Impl < wchar_t > ( unicodeOutput , unicodeBufferSizeInBytes , formatString , numFormatParameters , argList ) ;
}
//-----------------------------------------------------------------------------
// Purpose: construct string helper
//-----------------------------------------------------------------------------
template < typename T >
const T * GetTypedKeyValuesString ( KeyValues * pKeyValues , const char * pKeyName ) ;
template < >
const char * GetTypedKeyValuesString < char > ( KeyValues * pKeyValues , const char * pKeyName )
{
return pKeyValues - > GetString ( pKeyName , " [unknown] " ) ;
}
template < >
const wchar_t * GetTypedKeyValuesString < wchar_t > ( KeyValues * pKeyValues , const char * pKeyName )
{
return pKeyValues - > GetWString ( pKeyName , L " [unknown] " ) ;
}
template < typename T >
void ConstructStringKeyValuesInternal_Impl ( T * unicodeOutput , int unicodeBufferSizeInBytes , const T * formatString , KeyValues * localizationVariables )
{
T * outputPos = unicodeOutput ;
//assumes we can't have %s10
//assume both are 0 terminated?
int unicodeBufferSize = unicodeBufferSizeInBytes / sizeof ( T ) ;
while ( * formatString ! = ' \0 ' & & unicodeBufferSize > 1 )
{
bool shouldAdvance = true ;
if ( * formatString = = ' % ' )
{
// this is an escape sequence that specifies a variable name
if ( formatString [ 1 ] = = ' s ' & & formatString [ 2 ] > = ' 0 ' & & formatString [ 2 ] < = ' 9 ' )
{
// old style escape sequence, ignore
}
else if ( formatString [ 1 ] = = ' % ' )
{
// just a '%' char, just write the second one
formatString + + ;
}
else if ( localizationVariables )
{
// get out the variable name
const T * varStart = formatString + 1 ;
const T * varEnd = StringFuncs < T > : : FindChar ( varStart , ' % ' ) ;
if ( varEnd & & * varEnd = = ' % ' )
{
shouldAdvance = false ;
// assume variable names must be ascii, do a quick convert
char variableName [ 32 ] ;
char * vset = variableName ;
for ( const T * pws = varStart ; pws < varEnd & & ( vset < variableName + sizeof ( variableName ) - 1 ) ; + + pws , + + vset )
{
* vset = ( char ) * pws ;
}
* vset = 0 ;
// look up the variable name
const T * value = GetTypedKeyValuesString < T > ( localizationVariables , variableName ) ;
int paramSize = StringFuncs < T > : : Length ( value ) ;
if ( paramSize > = unicodeBufferSize )
{
paramSize = MAX ( 0 , unicodeBufferSize - 1 ) ;
}
StringFuncs < T > : : Copy ( outputPos , value , paramSize ) ;
unicodeBufferSize - = paramSize ;
outputPos + = paramSize ;
formatString = varEnd + 1 ;
}
}
}
if ( shouldAdvance )
{
//copy it over, char by char
* outputPos = * formatString ;
outputPos + + ;
unicodeBufferSize - - ;
formatString + + ;
}
}
// ensure null termination
* outputPos = ' \0 ' ;
}
void ILocalize : : ConstructStringKeyValuesInternal ( char * unicodeOutput , int unicodeBufferSizeInBytes , const char * formatString , KeyValues * localizationVariables )
{
ConstructStringKeyValuesInternal_Impl < char > ( unicodeOutput , unicodeBufferSizeInBytes , formatString , localizationVariables ) ;
}
void ILocalize : : ConstructStringKeyValuesInternal ( wchar_t * unicodeOutput , int unicodeBufferSizeInBytes , const wchar_t * formatString , KeyValues * localizationVariables )
{
ConstructStringKeyValuesInternal_Impl < wchar_t > ( unicodeOutput , unicodeBufferSizeInBytes , formatString , localizationVariables ) ;
}