Windows2003-3790/base/win32/winnls/number.c

2359 lines
72 KiB
C
Raw Normal View History

2001-01-01 00:00:00 +01:00
/*++
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"
#include "nlssafe.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.
//
if(FAILED(StringCchCopyW(lpNumberStr, cchNumber, pFinal)))
{
//
// Failure should in theory be impossible, but if we ignore the
// return value, PREfast will complain.
//
SetLastError(ERROR_OUTOFMEMORY);
Length = 0;
}
}
//
// 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.
//
if(FAILED(StringCchCopyW(lpCurrencyStr, cchCurrency, pFinal)))
{
//
// Failure should in theory be impossible, but if we ignore the
// return value, PREfast will complain.
//
SetLastError(ERROR_OUTOFMEMORY);
Length = 0;
}
}
//
// 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));
}