|
|
/*++
Copyright (c) 1991-2000, Microsoft Corporation All rights reserved.
Module Name:
number.c
Abstract:
This file contains functions that form properly formatted number and currency strings for a given locale.
APIs found in this file: GetNumberFormatW GetCurrencyFormatW
Revision History:
07-28-93 JulieB Created.
--*/
//
// Include Files.
//
#include "nls.h"
//
// Constant Declarations.
//
#define MAX_NUMBER_BUFFER 256 // size of static buffer
#define MAX_GROUPS 5 // max number of groupings
#define MAX_GROUPING_NUMBER 9999 // max value for groupings
//
// Account for:
// - number of fractional digits
// - decimal seperator
// - negative sign
// - zero terminator
//
#define MAX_NON_INTEGER_PART ( MAX_VALUE_IDIGITS + \
MAX_SDECIMAL + \ MAX_SNEGSIGN + \ 1 ) //
// Account for:
// - negative sign
// - blank spaces
// - one extra number from rounding
// - one extra grouping separator from rounding
//
#define MAX_NUMBER_EXTRAS ( MAX_SNEGSIGN + \
MAX_BLANKS + \ 1 + \ MAX_STHOUSAND ) //
// Account for:
// - negative sign
// - currency sign
// - blank spaces
// - one extra number from rounding
// - one extra grouping separator from rounding
//
#define MAX_CURRENCY_EXTRAS ( MAX_SNEGSIGN + \
MAX_SCURRENCY + \ MAX_BLANKS + \ 1 + \ MAX_SMONTHOUSEP )
//
// Forward Declarations.
//
BOOL IsValidNumberFormat( CONST NUMBERFMTW *pFormat);
BOOL IsValidCurrencyFormat( CONST CURRENCYFMTW *pFormat);
UINT GetRegIntValue( LCID Locale, LCTYPE LCType, BOOL NoUserOverride, SIZE_T CacheOffset, LPWSTR pRegVal, LPWSTR pDefault, int DefaultVal, int UpperBound);
int ConvertGroupingStringToInt( LPWSTR pGroupingSrc, LPWSTR pGroupingDest);
UINT GetGroupingValue( LCID Locale, LCTYPE LCType, BOOL NoUserOverride, SIZE_T CacheOffset, LPWSTR pRegVal, LPWSTR pDefault, int DefaultVal);
int GetNumberString( PLOC_HASH pHashN, LPWSTR pValue, LPNUMBERFMTW pFormat, LPWSTR *ppBuf, int BufSize, BOOL *pfZeroValue, int *pNeededSizeToAllocate, BOOL fSetError);
int ParseNumber( PLOC_HASH pHashN, BOOL NoUserOverride, LPWSTR pValue, LPNUMBERFMTW pFormat, LPWSTR *ppBuf, int BufSize, int *pNeededSizeToAllocate, BOOL fSetError);
int ParseCurrency( PLOC_HASH pHashN, BOOL NoUserOverride, LPWSTR pValue, LPCURRENCYFMTW pFormat, LPWSTR *ppBuf, int BufSize, int *pNeededSizeToAllocate, BOOL fSetError);
//-------------------------------------------------------------------------//
// INTERNAL MACROS //
//-------------------------------------------------------------------------//
////////////////////////////////////////////////////////////////////////////
//
// NLS_COPY_UNICODE_STR
//
// Copies a zero terminated Unicode string from pSrc to the pDest buffer.
// The pDest pointer is advanced to the end of the string.
//
// DEFINED AS A MACRO.
//
// 07-28-93 JulieB Created.
////////////////////////////////////////////////////////////////////////////
#define NLS_COPY_UNICODE_STR( pDest, \
pSrc ) \ { \ LPWSTR pTmp; /* temp pointer to source */ \ \ \ pTmp = pSrc; \ while (*pTmp) \ { \ *pDest = *pTmp; \ pDest++; \ pTmp++; \ } \ }
////////////////////////////////////////////////////////////////////////////
//
// NLS_COPY_UNICODE_STR_NOADV
//
// Copies a zero terminated Unicode string from pSrc to the pDest buffer.
// The pDest pointer is NOT advanced to the end of the string.
//
// DEFINED AS A MACRO.
//
// 07-28-93 JulieB Created.
////////////////////////////////////////////////////////////////////////////
#define NLS_COPY_UNICODE_STR_TMP( pDest, \
pSrc ) \ { \ LPWSTR pSrcT; /* temp pointer to source */ \ LPWSTR pDestT; /* temp pointer to destination */ \ \ \ pSrcT = pSrc; \ pDestT = pDest; \ while (*pSrcT) \ { \ *pDestT = *pSrcT; \ pDestT++; \ pSrcT++; \ } \ }
////////////////////////////////////////////////////////////////////////////
//
// NLS_ROUND_IT
//
// Rounds the floating point number given as a string.
//
// NOTE: This function will reset the pBegin pointer if an
// extra character is added to the string.
//
// DEFINED AS A MACRO.
//
// 07-28-93 JulieB Created.
////////////////////////////////////////////////////////////////////////////
#define NLS_ROUND_IT( pBegin, \
pEnd, \ IntPartGroup, \ pSep ) \ { \ LPWSTR pRound = pEnd; /* ptr to position in string */ \ LPWSTR pEndSep; /* ptr to end of group separator */ \ \ \ /* \
* Round the digits in the string one by one, going backwards in \ * the string. Stop when either a value other than 9 is found, \ * or the beginning of the string is reached. \ */ \ while (pRound >= pBegin) \ { \ if ((*pRound < NLS_CHAR_ZERO) || (*pRound > NLS_CHAR_NINE)) \ { \ pRound--; \ } \ else if (*pRound == NLS_CHAR_NINE) \ { \ *pRound = NLS_CHAR_ZERO; \ pRound--; \ } \ else \ { \ (*pRound)++; \ break; \ } \ } \ \ /* \
* Make sure we don't have a number like 9.999, where we would need \ * to add an extra character to the string and make it 10.00. \ */ \ if (pRound < pBegin) \ { \ /* \
* All values to the right of the decimal are zero. All values \ * to the left of the decimal are either zero or the grouping \ * separator. \ */ \ if ((IntPartGroup) == 0) \ { \ /* \
* Adding another integer means we need to add another \ * grouping separator. \ */ \ pEndSep = pSep + NlsStrLenW(pSep) - 1; \ while (pEndSep >= pSep) \ { \ (pBegin)--; \ *(pBegin) = *pEndSep; \ pEndSep--; \ } \ } \ \ /* \
* Store a 1 at the beginning of the string and reset the \ * pointer to the beginning of the string. \ */ \ (pBegin)--; \ *(pBegin) = NLS_CHAR_ONE; \ } \ }
//-------------------------------------------------------------------------//
// API ROUTINES //
//-------------------------------------------------------------------------//
////////////////////////////////////////////////////////////////////////////
//
// GetNumberFormatW
//
// Returns a properly formatted number string for the given locale.
// This call also indicates how much memory is necessary to contain
// the desired information.
//
// 07-28-93 JulieB Created.
////////////////////////////////////////////////////////////////////////////
int WINAPI GetNumberFormatW( LCID Locale, DWORD dwFlags, LPCWSTR lpValue, CONST NUMBERFMTW *lpFormat, LPWSTR lpNumberStr, int cchNumber)
{ PLOC_HASH pHashN; // ptr to LOC hash node
int Length = 0; // number of characters written
LPNUMBERFMTW pFormat; // ptr to number format struct
NUMBERFMTW NumFmt; // number format
WCHAR pString[MAX_NUMBER_BUFFER]; // ptr to temporary buffer
LPWSTR pFinal; // ptr to the final string
BOOL NoUserOverride; // if no user override flag set
WCHAR pDecimal[MAX_REG_VAL_SIZE]; // temp buffer for decimal sep
WCHAR pThousand[MAX_REG_VAL_SIZE]; // temp buffer for thousand sep
int NeededSizeToAllocate = 0; // size of buffer needed
WCHAR *pTemp = NULL; // allocated temp storage buffer
//
// Initialize UserOverride.
//
NoUserOverride = dwFlags & LOCALE_NOUSEROVERRIDE;
//
// Invalid Parameter Check:
// - validate LCID
// - count is negative
// - NULL src string
// - NULL data pointer AND count is not zero
// - ptrs to string buffers same
//
VALIDATE_LOCALE(Locale, pHashN, FALSE); if ( (pHashN == NULL) || (cchNumber < 0) || (lpValue == NULL) || ((lpNumberStr == NULL) && (cchNumber != 0)) || (lpValue == lpNumberStr) ) { SetLastError(ERROR_INVALID_PARAMETER); return (0); }
//
// Invalid Flags Check:
// - flags other than valid ones
// - lpFormat not NULL AND NoUserOverride flag is set
//
if ( (dwFlags & GNF_INVALID_FLAG) || ((lpFormat != NULL) && (NoUserOverride)) ) { SetLastError(ERROR_INVALID_FLAGS); return (0); }
//
// Set pFormat to point at the proper format structure.
//
if (lpFormat != NULL) { //
// Use the format structure given by the caller.
//
pFormat = (LPNUMBERFMTW)lpFormat;
if (!IsValidNumberFormat(pFormat)) { SetLastError(ERROR_INVALID_PARAMETER); return (0); } } else { //
// Use the format structure defined here.
//
pFormat = &NumFmt;
//
// Get the number of decimal digits.
//
pFormat->NumDigits = GetRegIntValue( Locale, LOCALE_IDIGITS, NoUserOverride, FIELD_OFFSET(NLS_USER_INFO, iDigits), NLS_VALUE_IDIGITS, pHashN->pLocaleFixed->szIDigits, 2, MAX_VALUE_IDIGITS );
//
// Get the leading zero in decimal fields option.
//
pFormat->LeadingZero = GetRegIntValue( Locale, LOCALE_ILZERO, NoUserOverride, FIELD_OFFSET(NLS_USER_INFO, iLZero), NLS_VALUE_ILZERO, pHashN->pLocaleFixed->szILZero, 1, MAX_VALUE_ILZERO );
//
// Get the negative ordering.
//
pFormat->NegativeOrder = GetRegIntValue( Locale, LOCALE_INEGNUMBER, NoUserOverride, FIELD_OFFSET(NLS_USER_INFO, iNegNumber), NLS_VALUE_INEGNUMBER, pHashN->pLocaleFixed->szINegNumber, 1, MAX_VALUE_INEGNUMBER );
//
// Get the grouping left of the decimal.
//
pFormat->Grouping = GetGroupingValue( Locale, LOCALE_SGROUPING, NoUserOverride, FIELD_OFFSET(NLS_USER_INFO, sGrouping), NLS_VALUE_SGROUPING, (LPWORD)(pHashN->pLocaleHdr) + pHashN->pLocaleHdr->SGrouping, 3 );
//
// Get the decimal separator.
//
// NOTE: This must follow the above calls because
// pDecSep is used as a temporary buffer above.
//
if ( (!NoUserOverride) && GetUserInfo( Locale, LOCALE_SDECIMAL, FIELD_OFFSET(NLS_USER_INFO, sDecimal), NLS_VALUE_SDECIMAL, pDecimal, ARRAYSIZE(pDecimal), TRUE ) && IsValidSeparatorString( pDecimal, MAX_SDECIMAL, FALSE ) ) { pFormat->lpDecimalSep = pDecimal; } else { pFormat->lpDecimalSep = (LPWORD)(pHashN->pLocaleHdr) + pHashN->pLocaleHdr->SDecimal; }
//
// Get the thousand separator.
// This string may be a null string.
//
if ( (!NoUserOverride) && GetUserInfo( Locale, LOCALE_STHOUSAND, FIELD_OFFSET(NLS_USER_INFO, sThousand), NLS_VALUE_STHOUSAND, pThousand, ARRAYSIZE(pThousand), FALSE ) && IsValidSeparatorString( pThousand, MAX_STHOUSAND, FALSE ) ) { pFormat->lpThousandSep = pThousand; } else { pFormat->lpThousandSep = (LPWORD)(pHashN->pLocaleHdr) + pHashN->pLocaleHdr->SThousand; } }
//
// Parse the number format string.
//
pFinal = pString; Length = ParseNumber( pHashN, NoUserOverride, (LPWSTR)lpValue, pFormat, &pFinal, MAX_NUMBER_BUFFER, &NeededSizeToAllocate, FALSE );
//
// If the failure is due to a stack variable size limitation, then
// try to satisfy the request from the local process heap.
//
if ((Length == 0) && (NeededSizeToAllocate > 0)) { pTemp = RtlAllocateHeap( RtlProcessHeap(), 0, NeededSizeToAllocate * sizeof(TCHAR) ); if (pTemp) { pFinal = pTemp; Length = ParseNumber( pHashN, NoUserOverride, (LPWSTR)lpValue, pFormat, &pFinal, NeededSizeToAllocate, &NeededSizeToAllocate, TRUE ); } }
//
// Check cchNumber for size of given buffer.
//
if ((cchNumber == 0) || (Length == 0)) { //
// If cchNumber is 0, then we can't use lpNumberStr. In this
// case, we simply want to return the length (in characters) of
// the string to be copied.
//
Length = Length; } else if (cchNumber < Length) { //
// The buffer is too small for the string, so return an error
// and zero bytes written.
//
SetLastError(ERROR_INSUFFICIENT_BUFFER); Length = 0; } else { //
// Copy the number string to lpNumberStr and null terminate it.
// Return the number of characters copied.
//
NlsStrCpyW(lpNumberStr, pFinal); }
//
// Free any dynamically allocated memory.
//
if (pTemp != NULL) { RtlFreeHeap(RtlProcessHeap(), 0, pTemp); }
//
// Return the number of characters copied.
//
return (Length); }
////////////////////////////////////////////////////////////////////////////
//
// GetCurrencyFormatW
//
// Returns a properly formatted currency string for the given locale.
// This call also indicates how much memory is necessary to contain
// the desired information.
//
// 07-28-93 JulieB Created.
////////////////////////////////////////////////////////////////////////////
int WINAPI GetCurrencyFormatW( LCID Locale, DWORD dwFlags, LPCWSTR lpValue, CONST CURRENCYFMTW *lpFormat, LPWSTR lpCurrencyStr, int cchCurrency)
{ PLOC_HASH pHashN; // ptr to LOC hash node
int Length = 0; // number of characters written
LPCURRENCYFMTW pFormat; // ptr to currency format struct
CURRENCYFMTW CurrFmt; // currency format
WCHAR pString[MAX_NUMBER_BUFFER]; // ptr to temporary buffer
LPWSTR pFinal; // ptr to the final string
BOOL NoUserOverride; // if no user override flag set
WCHAR pDecimal[MAX_REG_VAL_SIZE]; // temp buffer for decimal sep
WCHAR pThousand[MAX_REG_VAL_SIZE]; // temp buffer for thousand sep
WCHAR pCurrency[MAX_REG_VAL_SIZE]; // temp buffer for currency symbol
int NeededSizeToAllocate = 0; // size of buffer needed
WCHAR *pTemp = NULL; // allocated temp storage buffer
//
// Initialize UserOverride.
//
NoUserOverride = dwFlags & LOCALE_NOUSEROVERRIDE;
//
// Invalid Parameter Check:
// - validate LCID
// - count is negative
// - NULL src string
// - NULL data pointer AND count is not zero
// - ptrs to string buffers same
//
VALIDATE_LOCALE(Locale, pHashN, FALSE); if ( (pHashN == NULL) || (cchCurrency < 0) || (lpValue == NULL) || ((lpCurrencyStr == NULL) && (cchCurrency != 0)) || (lpValue == lpCurrencyStr) ) { SetLastError(ERROR_INVALID_PARAMETER); return (0); }
//
// Invalid Flags Check:
// - flags other than valid ones
// - lpFormat not NULL AND NoUserOverride flag is set
//
if ( (dwFlags & GCF_INVALID_FLAG) || ((lpFormat != NULL) && (NoUserOverride)) ) { SetLastError(ERROR_INVALID_FLAGS); return (0); }
//
// Set pFormat to point at the proper format structure.
//
if (lpFormat != NULL) { //
// Use the format structure given by the caller.
//
pFormat = (LPCURRENCYFMTW)lpFormat;
if (!IsValidCurrencyFormat(pFormat)) { SetLastError(ERROR_INVALID_PARAMETER); return (0); } } else { //
// Use the format structure defined here.
//
pFormat = &CurrFmt;
//
// Get the number of decimal digits.
//
pFormat->NumDigits = GetRegIntValue( Locale, LOCALE_ICURRDIGITS, NoUserOverride, FIELD_OFFSET(NLS_USER_INFO, iCurrDigits), NLS_VALUE_ICURRDIGITS, pHashN->pLocaleFixed->szICurrDigits, 2, MAX_VALUE_ICURRDIGITS );
//
// Get the leading zero in decimal fields option.
//
pFormat->LeadingZero = GetRegIntValue( Locale, LOCALE_ILZERO, NoUserOverride, FIELD_OFFSET(NLS_USER_INFO, iLZero), NLS_VALUE_ILZERO, pHashN->pLocaleFixed->szILZero, 1, MAX_VALUE_ILZERO );
//
// Get the positive ordering.
//
pFormat->PositiveOrder = GetRegIntValue( Locale, LOCALE_ICURRENCY, NoUserOverride, FIELD_OFFSET(NLS_USER_INFO, iCurrency), NLS_VALUE_ICURRENCY, pHashN->pLocaleFixed->szICurrency, 0, MAX_VALUE_ICURRENCY );
//
// Get the negative ordering.
//
pFormat->NegativeOrder = GetRegIntValue( Locale, LOCALE_INEGCURR, NoUserOverride, FIELD_OFFSET(NLS_USER_INFO, iNegCurr), NLS_VALUE_INEGCURR, pHashN->pLocaleFixed->szINegCurr, 1, MAX_VALUE_INEGCURR );
//
// Get the grouping left of the decimal.
//
pFormat->Grouping = GetGroupingValue( Locale, LOCALE_SMONGROUPING, NoUserOverride, FIELD_OFFSET(NLS_USER_INFO, sMonGrouping), NLS_VALUE_SMONGROUPING, (LPWORD)(pHashN->pLocaleHdr) + pHashN->pLocaleHdr->SMonGrouping, 3 );
//
// Get the decimal separator.
//
// NOTE: This must follow the above calls because
// pDecSep is used as a temporary buffer.
//
if ( (!NoUserOverride) && GetUserInfo( Locale, LOCALE_SMONDECIMALSEP, FIELD_OFFSET(NLS_USER_INFO, sMonDecSep), NLS_VALUE_SMONDECIMALSEP, pDecimal, ARRAYSIZE(pDecimal), TRUE ) && IsValidSeparatorString( pDecimal, MAX_SDECIMAL, FALSE ) ) { pFormat->lpDecimalSep = pDecimal; } else { pFormat->lpDecimalSep = (LPWORD)(pHashN->pLocaleHdr) + pHashN->pLocaleHdr->SMonDecSep; }
//
// Get the thousand separator.
// This string may be a null string.
//
if ( (!NoUserOverride) && GetUserInfo( Locale, LOCALE_SMONTHOUSANDSEP, FIELD_OFFSET(NLS_USER_INFO, sMonThouSep), NLS_VALUE_SMONTHOUSANDSEP, pThousand, ARRAYSIZE(pThousand), FALSE ) && IsValidSeparatorString( pThousand, MAX_STHOUSAND, FALSE ) ) { pFormat->lpThousandSep = pThousand; } else { pFormat->lpThousandSep = (LPWORD)(pHashN->pLocaleHdr) + pHashN->pLocaleHdr->SMonThousSep; }
//
// Get the currency symbol.
// This string may be a null string.
//
if ( (!NoUserOverride) && GetUserInfo( Locale, LOCALE_SCURRENCY, FIELD_OFFSET(NLS_USER_INFO, sCurrency), NLS_VALUE_SCURRENCY, pCurrency, ARRAYSIZE(pCurrency), FALSE ) && IsValidSeparatorString( pCurrency, MAX_SCURRENCY, FALSE ) ) { pFormat->lpCurrencySymbol = pCurrency; } else { pFormat->lpCurrencySymbol = (LPWORD)(pHashN->pLocaleHdr) + pHashN->pLocaleHdr->SCurrency; } }
//
// Parse the currency format string.
//
pFinal = pString; Length = ParseCurrency( pHashN, NoUserOverride, (LPWSTR)lpValue, pFormat, &pFinal, MAX_NUMBER_BUFFER, &NeededSizeToAllocate, FALSE );
//
// If the failure is due to a stack variable size limitation, then
// try to satisfy the request from the local process heap.
//
if ((Length == 0) && (NeededSizeToAllocate > 0)) { pTemp = RtlAllocateHeap( RtlProcessHeap(), 0, NeededSizeToAllocate * sizeof(TCHAR) ); if (pTemp) { pFinal = pTemp; Length = ParseCurrency( pHashN, NoUserOverride, (LPWSTR)lpValue, pFormat, &pFinal, NeededSizeToAllocate, &NeededSizeToAllocate, TRUE ); } }
//
// Check cchCurrency for size of given buffer.
//
if ((cchCurrency == 0) || (Length == 0)) { //
// If cchCurrency is 0, then we can't use lpCurrencyStr. In this
// case, we simply want to return the length (in characters) of
// the string to be copied.
//
Length = Length; } else if (cchCurrency < Length) { //
// The buffer is too small for the string, so return an error
// and zero bytes written.
//
SetLastError(ERROR_INSUFFICIENT_BUFFER); Length = 0; } else { //
// Copy the currency string to lpCurrencyStr and null terminate it.
// Return the number of characters copied.
//
NlsStrCpyW(lpCurrencyStr, pFinal); }
//
// Free any dynamically allocated memory.
//
if (pTemp != NULL) { RtlFreeHeap(RtlProcessHeap(), 0, pTemp); }
//
// Return the number of characters copied.
//
return (Length); }
//-------------------------------------------------------------------------//
// INTERNAL ROUTINES //
//-------------------------------------------------------------------------//
////////////////////////////////////////////////////////////////////////////
//
// IsValidNumberFormat
//
// Returns TRUE if the given format is valid. Otherwise, it returns FALSE.
//
// 07-28-93 JulieB Created.
////////////////////////////////////////////////////////////////////////////
BOOL IsValidNumberFormat( CONST NUMBERFMTW *pFormat)
{ //
// Check for invalid values.
//
if ((pFormat->NumDigits > MAX_VALUE_IDIGITS) || (pFormat->LeadingZero > MAX_VALUE_ILZERO) || (pFormat->Grouping > MAX_GROUPING_NUMBER) || (pFormat->NegativeOrder > MAX_VALUE_INEGNUMBER) || (pFormat->lpDecimalSep == NULL) || (!IsValidSeparatorString( pFormat->lpDecimalSep, MAX_SDECIMAL, (pFormat->NumDigits) ? TRUE : FALSE)) || (pFormat->lpThousandSep == NULL) || (!IsValidSeparatorString( pFormat->lpThousandSep, MAX_STHOUSAND, FALSE ))) { return (FALSE); }
//
// Return success.
//
return (TRUE); }
////////////////////////////////////////////////////////////////////////////
//
// IsValidCurrencyFormat
//
// Returns TRUE if the given format is valid. Otherwise, it returns FALSE.
//
// 07-28-93 JulieB Created.
////////////////////////////////////////////////////////////////////////////
BOOL IsValidCurrencyFormat( CONST CURRENCYFMTW *pFormat)
{ //
// Check for invalid values.
//
if ((pFormat->NumDigits > MAX_VALUE_IDIGITS) || (pFormat->LeadingZero > MAX_VALUE_ILZERO) || (pFormat->Grouping > MAX_GROUPING_NUMBER) || (pFormat->lpDecimalSep == NULL) || (!IsValidSeparatorString( pFormat->lpDecimalSep, MAX_SMONDECSEP, (pFormat->NumDigits) ? TRUE : FALSE)) || (pFormat->lpThousandSep == NULL) || (!IsValidSeparatorString( pFormat->lpThousandSep, MAX_SMONTHOUSEP, FALSE )) || (pFormat->lpCurrencySymbol == NULL) || (!IsValidSeparatorString( pFormat->lpCurrencySymbol, MAX_SCURRENCY, FALSE )) || (pFormat->PositiveOrder > MAX_VALUE_ICURRENCY) || (pFormat->NegativeOrder > MAX_VALUE_INEGCURR)) { return (FALSE); }
//
// Return success.
//
return (TRUE); }
////////////////////////////////////////////////////////////////////////////
//
// GetRegIntValue
//
// Retrieves the specified locale information, converts the unicode string
// to an integer value, and returns the value.
//
// 07-28-93 JulieB Created.
////////////////////////////////////////////////////////////////////////////
UINT GetRegIntValue( LCID Locale, LCTYPE LCType, BOOL NoUserOverride, SIZE_T CacheOffset, LPWSTR pRegVal, LPWSTR pDefault, int DefaultVal, int UpperBound) { UNICODE_STRING ObUnicodeStr; // value string
int Value; // value
WCHAR pTemp[MAX_REG_VAL_SIZE]; // temp buffer
//
// Initialize values.
//
Value = -1;
//
// Try the user registry.
//
if ((!NoUserOverride) && GetUserInfo( Locale, LCType, CacheOffset, pRegVal, pTemp, ARRAYSIZE(pTemp), TRUE )) { //
// Convert the user data to an integer.
//
RtlInitUnicodeString(&ObUnicodeStr, pTemp); if ((RtlUnicodeStringToInteger(&ObUnicodeStr, 10, &Value)) || (Value < 0) || (Value > UpperBound)) { //
// Bad value, so store -1 so that the system default
// will be used.
//
Value = -1; } }
//
// See if the value obtained above is valid.
//
if (Value < 0) { //
// Convert system default data to an integer.
//
RtlInitUnicodeString(&ObUnicodeStr, pDefault); if ((RtlUnicodeStringToInteger(&ObUnicodeStr, 10, &Value)) || (Value < 0) || (Value > UpperBound)) { //
// Bad value, so use the chosen default value.
//
Value = DefaultVal; } }
return ((UINT)Value); }
////////////////////////////////////////////////////////////////////////////
//
// ConvertGroupingStringToInt
//
// Converts the given grouping string to an integer.
// For example, 3;2;0 becomes 32 and 3;0 becomes 3 and 3;2 becomes 320.
//
// NOTE: The pGrouping buffer will be modified.
//
// 01-05-98 JulieB Created.
////////////////////////////////////////////////////////////////////////////
int ConvertGroupingStringToInt( LPWSTR pGroupingSrc, LPWSTR pGroupingDest) { LPWSTR pSrc = pGroupingSrc; // temp ptr to src position
LPWSTR pDest = pGroupingDest; // temp ptr to dest position
UNICODE_STRING ObUnicodeStr; // value string
int Value; // value
//
// Filter out all non-numeric values and all zero values.
// Store the result in the destination buffer.
//
while (*pSrc) { if ((*pSrc < NLS_CHAR_ONE) || (*pSrc > NLS_CHAR_NINE)) { pSrc++; } else { if (pSrc != pDest) { *pDest = *pSrc; } pSrc++; pDest++; } }
//
// Make sure there is something in the destination buffer.
// Also, see if we need to add a zero in the case of 3;2 becomes 320.
//
if ((pDest == pGroupingDest) || (*(pSrc - 1) != NLS_CHAR_ZERO)) { *pDest = NLS_CHAR_ZERO; pDest++; }
//
// Null terminate the buffer.
//
*pDest = 0;
//
// Convert the string to an integer.
//
RtlInitUnicodeString(&ObUnicodeStr, pGroupingDest); RtlUnicodeStringToInteger(&ObUnicodeStr, 10, &Value);
//
// Return the integer value.
//
return (Value); }
////////////////////////////////////////////////////////////////////////////
//
// GetGroupingValue
//
// Retrieves the specified grouping information, converts the grouping
// string to an integer value (eg. 3;2;0 -> 32), and returns the value.
//
// 07-28-93 JulieB Created.
////////////////////////////////////////////////////////////////////////////
UINT GetGroupingValue( LCID Locale, LCTYPE LCType, BOOL NoUserOverride, SIZE_T CacheOffset, LPWSTR pRegVal, LPWSTR pDefault, int DefaultVal) { int Value; // value
WCHAR pTemp[MAX_REG_VAL_SIZE]; // temp buffer
//
// Initialize values.
//
Value = -1;
//
// Try the user registry.
//
if ((!NoUserOverride) && GetUserInfo( Locale, LCType, CacheOffset, pRegVal, pTemp, ARRAYSIZE(pTemp), TRUE )) { //
// Convert the grouping string to an integer.
// 3;0 becomes 3, 3;2;0 becomes 32, and 3;2 becomes 320.
//
Value = ConvertGroupingStringToInt(pTemp, pTemp); if (Value < 0) { //
// Bad value, so store -1 so that the system default
// will be used.
//
Value = -1; } }
//
// See if the value obtained above is valid.
//
if (Value < 0) { //
// Convert the grouping string to an integer.
// 3;0 becomes 3, 3;2;0 becomes 32, and 3;2 becomes 320.
//
Value = ConvertGroupingStringToInt(pDefault, pTemp); if (Value < 0) { //
// Bad value, so use the chosen default value.
//
Value = DefaultVal; } }
//
// Return the value.
//
return ((UINT)Value); }
////////////////////////////////////////////////////////////////////////////
//
// GetNumberString
//
// Puts the properly formatted number string into the given string buffer.
// It returns the number of characters written to the string buffer.
//
// 07-28-93 JulieB Created.
////////////////////////////////////////////////////////////////////////////
int GetNumberString( PLOC_HASH pHashN, LPWSTR pValue, LPNUMBERFMTW pFormat, LPWSTR *ppBuf, int BufSize, BOOL *pfZeroValue, int *pNeededSizeToAllocate, BOOL fSetError)
{ LPWSTR pDecPt; // ptr to decimal point in given buffer
LPWSTR pPos; // ptr to position in given buffer
LPWSTR pPos2; // ptr to position in given buffer
LPWSTR pPosBuf; // ptr to position in final buffer
int IntPartSize; // size of integer part of string
int GroupSize; // size of groupings left of decimal
int IntegerNum; // number of integers left of decimal
WCHAR wch; // wide character place holder
int pGroupArray[MAX_GROUPS]; // array of groups
int NumGroupings; // number of groupings
int NumSeparators; // number of separators
int NumDigits; // number of digits
int Ctr; // loop counter
UINT NumRound = 1; // # digits left before adding group separator
//
// Reset to indicate no need to allocate memory dynamically.
//
*pNeededSizeToAllocate = 0;
//
// Validate the string and find the decimal point in the string.
//
// The only valid characters within the string are:
// negative sign - in first position only
// decimal point
// Unicode code points for integers 0 - 9
//
pPos = pValue; while ((wch = *pPos) && (wch != NLS_CHAR_PERIOD)) { if ((wch < NLS_CHAR_ZERO) || (wch > NLS_CHAR_NINE)) { SetLastError(ERROR_INVALID_PARAMETER); return (0); } pPos++; } pDecPt = pPos;
if (*pPos) { pPos++; while (wch = *pPos) { if ((wch < NLS_CHAR_ZERO) || (wch > NLS_CHAR_NINE)) { SetLastError(ERROR_INVALID_PARAMETER); return (0); } pPos++; } }
//
// Remove any leading zeros in the integer part.
//
while (pValue < pDecPt) { if (*pValue != NLS_CHAR_ZERO) { break; } pValue++; }
//
// Save the number of integers to the left of the decimal.
//
IntegerNum = (int)(pDecPt - pValue);
//
// Make sure the value string passed in is not too large for
// the buffers.
//
IntPartSize = IntegerNum; NumGroupings = 0; NumSeparators = 0; if ((GroupSize = pFormat->Grouping) && (IntPartSize)) { //
// Count the number of groupings and save them in an array to be
// used later.
//
while (GroupSize && (NumGroupings < MAX_GROUPS)) { pGroupArray[NumGroupings] = GroupSize % 10; GroupSize /= 10; NumGroupings++; }
//
// Count the number of groupings that apply to the given number
// string.
//
NumDigits = IntegerNum; Ctr = (NumGroupings != 0) ? (NumGroupings - 1) : 0; while (Ctr) { if (NumDigits > pGroupArray[Ctr]) { NumDigits -= pGroupArray[Ctr]; NumSeparators++; } else { if (NumDigits == pGroupArray[Ctr]) { NumRound = 0; } break; } Ctr--; } if ((Ctr == 0) && pGroupArray[0]) { if (NumDigits > pGroupArray[0]) { NumSeparators += (NumDigits - 1) / pGroupArray[0]; } NumRound = NumDigits % pGroupArray[0]; }
IntPartSize += MAX_STHOUSAND * NumSeparators; }
//
// Make sure the buffer is large enough. If not, return the size
// needed.
//
if (IntPartSize > (BufSize - MAX_NON_INTEGER_PART)) { if (fSetError) { SetLastError(ERROR_INVALID_PARAMETER); } *pNeededSizeToAllocate = (IntPartSize + MAX_NON_INTEGER_PART); return (0); }
//
// Initialize pointers.
//
pPosBuf = *ppBuf; pPos = pValue; *pfZeroValue = FALSE;
//
// See if there are any digits before the decimal point.
//
if (pPos == pDecPt) { //
// Possibly a zero value. All leading zeros were removed, so
// there is no integer part.
//
*pfZeroValue = TRUE;
//
// No digits before decimal point, so add a leading zero
// to the final string if appropriate.
//
if (pFormat->LeadingZero) { *pPosBuf = NLS_CHAR_ZERO; pPosBuf++; } } else if (!NumSeparators) { //
// Grouping Size is zero or larger than the integer part of the
// string, so copy up to the decimal point (or end of string).
//
while (pPos < pDecPt) { *pPosBuf = *pPos; pPosBuf++; pPos++; } } else { //
// Copy up to where the first thousand separator should be.
// Use groupings of GroupSize numbers up to the decimal point.
//
NumDigits = IntegerNum; Ctr = (NumGroupings != 0) ? (NumGroupings - 1) : 0; while (Ctr) { if (NumDigits > pGroupArray[Ctr]) { NumDigits -= pGroupArray[Ctr]; } else { break; } Ctr--; } GroupSize = pGroupArray[Ctr];
pPos2 = GroupSize ? (pPos + (NumDigits % GroupSize)) : (pPos + NumDigits); if (pPos2 == pPos) { //
// Don't want to write thousand separator at the beginning
// of the string. There's at least GroupSize numbers
// in the string, so just advance pPos2 so that GroupSize
// numbers will be copied.
//
pPos2 = pPos + GroupSize; } while (pPos < pPos2) { *pPosBuf = *pPos; pPosBuf++; pPos++; NumDigits--; }
//
// Copy the thousand separator followed by GroupSize number of
// digits from the given string until the entire repeating
// GroupSize ends (or end of string).
//
while (NumDigits) { //
// Copy the localized thousand separator.
//
pPos2 = pFormat->lpThousandSep; while (*pPos2) { *pPosBuf = *pPos2; pPosBuf++; pPos2++; }
//
// Copy GroupSize number of digits.
//
pPos2 = pPos + GroupSize; while (pPos < pPos2) { *pPosBuf = *pPos; pPosBuf++; pPos++; NumDigits--; } }
//
// Copy the thousand separator followed by GroupSize number of
// digits from the given string - until the decimal point (or
// end of string) in the given string is reached.
//
if (pPos < pDecPt) { Ctr++; while (Ctr < NumGroupings) { //
// Copy the localized thousand separator.
//
pPos2 = pFormat->lpThousandSep; while (*pPos2) { *pPosBuf = *pPos2; pPosBuf++; pPos2++; }
//
// Copy GroupSize number of digits.
//
pPos2 = pPos + pGroupArray[Ctr]; while (pPos < pPos2) { *pPosBuf = *pPos; pPosBuf++; pPos++; }
//
// Go to the next grouping.
//
Ctr++; } } }
//
// See if there is a decimal separator in the given string.
//
if (pFormat->NumDigits > 0) { //
// Copy the localized decimal separator only if the number
// of digits right of the decimal is greater than zero.
//
pDecPt = pPosBuf; pPos2 = pFormat->lpDecimalSep; while (*pPos2) { *pPosBuf = *pPos2; pPosBuf++; pPos2++; } }
//
// Skip over the decimal point in the given string and
// copy the rest of the digits from the given string.
//
if (*pPos) { pPos++; } pPos2 = pPos + pFormat->NumDigits; while ((*pPos) && (pPos < pPos2)) { if (*pPos != NLS_CHAR_ZERO) { *pfZeroValue = FALSE; } *pPosBuf = *pPos; pPosBuf++; pPos++; }
//
// Make sure some value is in the buffer.
//
if (*ppBuf == pPosBuf) { *pPosBuf = NLS_CHAR_ZERO; pPosBuf++; }
//
// See if we need to round the number or pad it with zeros.
//
if (*pPos) { //
// Round the number if necessary.
//
if (*pPos2 > L'4') { *pfZeroValue = FALSE;
//
// Round the number. If GroupSize is 0, then we need to
// pass in a non-zero value so that the thousand separator
// will not be added to the front of the string (if it
// rounds that far).
//
pPosBuf--; NLS_ROUND_IT( *ppBuf, pPosBuf, NumRound, pFormat->lpThousandSep ); pPosBuf++; } } else { //
// Pad the string with the appropriate number of zeros.
//
while (pPos < pPos2) { *pPosBuf = NLS_CHAR_ZERO; pPosBuf++; pPos++; } }
//
// Zero terminate the string.
//
*pPosBuf = 0;
//
// Return the number of characters written to the buffer, including
// the null terminator.
//
return ((int)((pPosBuf - *ppBuf) + 1)); }
////////////////////////////////////////////////////////////////////////////
//
// ParseNumber
//
// Puts the properly formatted number string into the given string buffer.
// It returns the number of characters written to the string buffer.
//
// 07-28-93 JulieB Created.
////////////////////////////////////////////////////////////////////////////
int ParseNumber( PLOC_HASH pHashN, BOOL NoUserOverride, LPWSTR pValue, LPNUMBERFMTW pFormat, LPWSTR *ppBuf, int BufSize, int *pNeededSizeToAllocate, BOOL fSetError)
{ LPWSTR pBegin; // ptr to beginning of final buffer
LPWSTR pEnd; // ptr to end of final buffer
LPWSTR pNegSign; // ptr to negative sign string
BOOL IsNeg; // if negative sign in string
int Length; // length of number string
BOOL fZeroValue = FALSE; // if number is a zero value
int NegSignSize; // size of negative sign string
WCHAR pTemp[MAX_REG_VAL_SIZE]; // temp buffer
//
// Initialize pointer.
//
// Account for:
// - negative sign
// - blank spaces
// - one extra number from rounding
// - one extra grouping separator from rounding
//
pBegin = *ppBuf + MAX_NUMBER_EXTRAS;
//
// If the first value is a negative, then increment past it.
//
if (IsNeg = (*pValue == NLS_CHAR_HYPHEN)) { pValue++; }
//
// Get the appropriate number string and place it in the buffer.
//
Length = GetNumberString( pHashN, pValue, pFormat, &pBegin, BufSize - MAX_NUMBER_EXTRAS, &fZeroValue, pNeededSizeToAllocate, fSetError ); if (!Length) { if (*pNeededSizeToAllocate > 0) { *pNeededSizeToAllocate += MAX_NUMBER_EXTRAS; } return (0); }
//
// Advance pEnd position pointer to the end of the number string.
//
pEnd = pBegin + (Length - 1);
//
// See if any characters should be put in the buffer BEFORE and
// AFTER the properly formatted number string.
// - negative sign or opening/closing parenthesis
// - blank space
//
if (!fZeroValue && IsNeg) { //
// Get the negative sign string.
//
if (pFormat->NegativeOrder != 0) { if ( (!NoUserOverride) && GetUserInfo( pHashN->Locale, LOCALE_SNEGATIVESIGN, FIELD_OFFSET(NLS_USER_INFO, sNegSign), NLS_VALUE_SNEGATIVESIGN, pTemp, ARRAYSIZE(pTemp), TRUE ) && IsValidSeparatorString( pTemp, MAX_SNEGSIGN, FALSE ) ) { pNegSign = pTemp; } else { pNegSign = (LPWORD)(pHashN->pLocaleHdr) + pHashN->pLocaleHdr->SNegativeSign; } }
switch (pFormat->NegativeOrder) { case ( 0 ) : { //
// Put the opening parenthesis in the buffer.
//
pBegin--; *pBegin = NLS_CHAR_OPEN_PAREN;
//
// Put the closing parenthesis in the buffer.
//
*pEnd = NLS_CHAR_CLOSE_PAREN; pEnd++;
break; } case ( 2 ) : { //
// Put the space in the buffer.
//
pBegin--; *pBegin = NLS_CHAR_SPACE;
//
// Fall through to case 1.
//
} case ( 1 ) : default : { //
// Copy the negative sign to the buffer.
//
NegSignSize = NlsStrLenW(pNegSign); pBegin -= NegSignSize; NLS_COPY_UNICODE_STR_TMP(pBegin, pNegSign);
break; } case ( 4 ) : { //
// Put the space in the buffer.
//
*pEnd = NLS_CHAR_SPACE; pEnd++;
//
// Fall Through to case 3.
//
} case ( 3 ) : { //
// Copy the negative sign to the buffer.
//
NLS_COPY_UNICODE_STR(pEnd, pNegSign);
break; } } }
//
// Zero terminate the string.
//
*pEnd = 0;
//
// Return the pointer to the beginning of the string.
//
*ppBuf = pBegin;
//
// Return the number of characters written to the buffer, including
// the null terminator.
//
return ((int)((pEnd - pBegin) + 1)); }
////////////////////////////////////////////////////////////////////////////
//
// ParseCurrency
//
// Puts the properly formatted currency string into the given string buffer.
// It returns the number of characters written to the string buffer.
//
// 07-28-93 JulieB Created.
////////////////////////////////////////////////////////////////////////////
int ParseCurrency( PLOC_HASH pHashN, BOOL NoUserOverride, LPWSTR pValue, LPCURRENCYFMTW pFormat, LPWSTR *ppBuf, int BufSize, int *pNeededSizeToAllocate, BOOL fSetError)
{ LPWSTR pBegin; // ptr to beginning of final buffer
LPWSTR pEnd; // ptr to end of final buffer
LPWSTR pNegSign; // ptr to negative sign string
BOOL IsNeg; // if negative sign in string
int Length; // length of number string
BOOL fZeroValue = FALSE; // if number is a zero value
int NegSignSize; // size of negative sign string
UINT NegOrder; // negative ordering
int CurrSymSize; // size of currency symbol
WCHAR pTemp[MAX_REG_VAL_SIZE]; // temp buffer
//
// Initialize pointer.
//
// Account for:
// - negative sign
// - currency sign
// - blank spaces
// - one extra number from rounding
// - one extra grouping separator from rounding
//
pBegin = *ppBuf + MAX_CURRENCY_EXTRAS;
//
// If the first value is a negative, then increment past it.
//
if (IsNeg = (*pValue == NLS_CHAR_HYPHEN)) { pValue++; }
//
// Get the appropriate number string and place it in the buffer.
//
Length = GetNumberString( pHashN, pValue, (LPNUMBERFMTW)pFormat, &pBegin, BufSize - MAX_CURRENCY_EXTRAS, &fZeroValue, pNeededSizeToAllocate, fSetError ); if (!Length) { if (*pNeededSizeToAllocate > 0) { *pNeededSizeToAllocate += MAX_CURRENCY_EXTRAS; } return (0); }
//
// Advance pEnd position pointer to the end of the number string.
//
pEnd = pBegin + (Length - 1);
//
// Get the size of the currency symbol.
//
CurrSymSize = NlsStrLenW(pFormat->lpCurrencySymbol);
//
// See if any characters should be put in the buffer BEFORE and
// AFTER the properly formatted number string.
// - currency symbol
// - negative sign or opening/closing parenthesis
// - blank space
//
if (!fZeroValue && IsNeg) { //
// Get the negative sign string and the size of it.
//
NegOrder = pFormat->NegativeOrder; if ((NegOrder != 0) && (NegOrder != 4) && (NegOrder < 14)) { if ( (!NoUserOverride) && GetUserInfo( pHashN->Locale, LOCALE_SNEGATIVESIGN, FIELD_OFFSET(NLS_USER_INFO, sNegSign), NLS_VALUE_SNEGATIVESIGN, pTemp, ARRAYSIZE(pTemp), TRUE ) && IsValidSeparatorString( pTemp, MAX_SNEGSIGN, FALSE ) ) { pNegSign = pTemp; } else { pNegSign = (LPWORD)(pHashN->pLocaleHdr) + pHashN->pLocaleHdr->SNegativeSign; }
NegSignSize = NlsStrLenW(pNegSign); }
switch (NegOrder) { case ( 0 ) : { //
// Copy the currency symbol to the buffer.
//
pBegin -= CurrSymSize; NLS_COPY_UNICODE_STR_TMP(pBegin, pFormat->lpCurrencySymbol);
//
// Put the opening parenthesis in the buffer.
//
pBegin--; *pBegin = NLS_CHAR_OPEN_PAREN;
//
// Put the closing parenthesis in the buffer.
//
*pEnd = NLS_CHAR_CLOSE_PAREN; pEnd++;
break; } case ( 1 ) : default : { //
// Copy the currency symbol to the buffer.
//
pBegin -= CurrSymSize; NLS_COPY_UNICODE_STR_TMP(pBegin, pFormat->lpCurrencySymbol);
//
// Copy the negative sign to the buffer.
//
pBegin -= NegSignSize; NLS_COPY_UNICODE_STR_TMP(pBegin, pNegSign);
break; } case ( 2 ) : { //
// Copy the negative sign to the buffer.
//
pBegin -= NegSignSize; NLS_COPY_UNICODE_STR_TMP(pBegin, pNegSign);
//
// Copy the currency symbol to the buffer.
//
pBegin -= CurrSymSize; NLS_COPY_UNICODE_STR_TMP(pBegin, pFormat->lpCurrencySymbol);
break; } case ( 3 ) : { //
// Copy the currency symbol to the buffer.
//
pBegin -= CurrSymSize; NLS_COPY_UNICODE_STR_TMP(pBegin, pFormat->lpCurrencySymbol);
//
// Copy the negative sign to the buffer.
//
NLS_COPY_UNICODE_STR(pEnd, pNegSign);
break; } case ( 4 ) : { //
// Put the opening parenthesis in the buffer.
//
pBegin--; *pBegin = NLS_CHAR_OPEN_PAREN;
//
// Copy the currency symbol to the buffer.
//
NLS_COPY_UNICODE_STR(pEnd, pFormat->lpCurrencySymbol);
//
// Put the closing parenthesis in the buffer.
//
*pEnd = NLS_CHAR_CLOSE_PAREN; pEnd++;
break; } case ( 5 ) : { //
// Copy the negative sign to the buffer.
//
pBegin -= NegSignSize; NLS_COPY_UNICODE_STR_TMP(pBegin, pNegSign);
//
// Copy the currency symbol to the buffer.
//
NLS_COPY_UNICODE_STR(pEnd, pFormat->lpCurrencySymbol);
break; } case ( 6 ) : { //
// Copy the negative sign to the buffer.
//
NLS_COPY_UNICODE_STR(pEnd, pNegSign);
//
// Copy the currency symbol to the buffer.
//
NLS_COPY_UNICODE_STR(pEnd, pFormat->lpCurrencySymbol);
break; } case ( 7 ) : { //
// Copy the currency symbol to the buffer.
//
NLS_COPY_UNICODE_STR(pEnd, pFormat->lpCurrencySymbol);
//
// Copy the negative sign to the buffer.
//
NLS_COPY_UNICODE_STR(pEnd, pNegSign);
break; } case ( 8 ) : { //
// Copy the negative sign to the buffer.
//
pBegin -= NegSignSize; NLS_COPY_UNICODE_STR_TMP(pBegin, pNegSign);
//
// Put a space in the buffer.
//
*pEnd = NLS_CHAR_SPACE; pEnd++;
//
// Copy the currency symbol to the buffer.
//
NLS_COPY_UNICODE_STR(pEnd, pFormat->lpCurrencySymbol);
break; } case ( 9 ) : { //
// Put a space in the buffer.
//
pBegin--; *pBegin = NLS_CHAR_SPACE;
//
// Copy the currency symbol to the buffer.
//
pBegin -= CurrSymSize; NLS_COPY_UNICODE_STR_TMP(pBegin, pFormat->lpCurrencySymbol);
//
// Copy the negative sign to the buffer.
//
pBegin -= NegSignSize; NLS_COPY_UNICODE_STR_TMP(pBegin, pNegSign);
break; } case ( 10 ) : { //
// Put a space in the buffer.
//
*pEnd = NLS_CHAR_SPACE; pEnd++;
//
// Copy the currency symbol to the buffer.
//
NLS_COPY_UNICODE_STR(pEnd, pFormat->lpCurrencySymbol);
//
// Copy the negative sign to the buffer.
//
NLS_COPY_UNICODE_STR(pEnd, pNegSign);
break; } case ( 11 ) : { //
// Put a space in the buffer.
//
pBegin--; *pBegin = NLS_CHAR_SPACE;
//
// Copy the currency symbol to the buffer.
//
pBegin -= CurrSymSize; NLS_COPY_UNICODE_STR_TMP(pBegin, pFormat->lpCurrencySymbol);
//
// Copy the negative sign to the buffer.
//
NLS_COPY_UNICODE_STR(pEnd, pNegSign);
break; } case ( 12 ) : { //
// Copy the negative sign to the buffer.
//
pBegin -= NegSignSize; NLS_COPY_UNICODE_STR_TMP(pBegin, pNegSign);
//
// Put a space in the buffer.
//
pBegin--; *pBegin = NLS_CHAR_SPACE;
//
// Copy the currency symbol to the buffer.
//
pBegin -= CurrSymSize; NLS_COPY_UNICODE_STR_TMP(pBegin, pFormat->lpCurrencySymbol);
break; } case ( 13 ) : { //
// Copy the negative sign to the buffer.
//
NLS_COPY_UNICODE_STR(pEnd, pNegSign);
//
// Put a space in the buffer.
//
*pEnd = NLS_CHAR_SPACE; pEnd++;
//
// Copy the currency symbol to the buffer.
//
NLS_COPY_UNICODE_STR(pEnd, pFormat->lpCurrencySymbol);
break; } case ( 14 ) : { //
// Put a space in the buffer.
//
pBegin--; *pBegin = NLS_CHAR_SPACE;
//
// Copy the currency symbol to the buffer.
//
pBegin -= CurrSymSize; NLS_COPY_UNICODE_STR_TMP(pBegin, pFormat->lpCurrencySymbol);
//
// Put the opening parenthesis in the buffer.
//
pBegin--; *pBegin = NLS_CHAR_OPEN_PAREN;
//
// Put the closing parenthesis in the buffer.
//
*pEnd = NLS_CHAR_CLOSE_PAREN; pEnd++;
break; } case ( 15 ) : { //
// Put the opening parenthesis in the buffer.
//
pBegin--; *pBegin = NLS_CHAR_OPEN_PAREN;
//
// Put a space in the buffer.
//
*pEnd = NLS_CHAR_SPACE; pEnd++;
//
// Copy the currency symbol to the buffer.
//
NLS_COPY_UNICODE_STR(pEnd, pFormat->lpCurrencySymbol);
//
// Put the closing parenthesis in the buffer.
//
*pEnd = NLS_CHAR_CLOSE_PAREN; pEnd++;
break; } } } else { //
// Positive value. Store the currency symbol in the string
// if the positive order is either 0 or 2. Otherwise, wait
// till the end.
//
switch (pFormat->PositiveOrder) { case ( 2 ) : { //
// Put a space in the buffer.
//
pBegin--; *pBegin = NLS_CHAR_SPACE;
//
// Fall through to case 0.
//
} case ( 0 ) : default : { //
// Copy the currency symbol to the buffer.
//
pBegin -= CurrSymSize; NLS_COPY_UNICODE_STR_TMP(pBegin, pFormat->lpCurrencySymbol);
break; } case ( 3 ) : { //
// Put a space in the buffer.
//
*pEnd = NLS_CHAR_SPACE; pEnd++;
//
// Fall through to case 1.
//
} case ( 1 ) : { //
// Copy the currency symbol to the buffer.
//
NLS_COPY_UNICODE_STR(pEnd, pFormat->lpCurrencySymbol);
break; } } }
//
// Zero terminate the string.
//
*pEnd = 0;
//
// Return the pointer to the beginning of the string.
//
*ppBuf = pBegin;
//
// Return the number of characters written to the buffer, including
// the null terminator.
//
return ((int)((pEnd - pBegin) + 1)); }
|