blob: 430215662e3b65aeaa30f8a68fa20411bea991fe [file] [log] [blame]
// Copyright 2014 PDFium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
#include "xfa/fgas/localization/fgas_locale.h"
#include <algorithm>
#include <vector>
#include "core/fxcrt/fx_ext.h"
#include "core/fxcrt/fx_xml.h"
#include "xfa/fgas/localization/fgas_localeimp.h"
#define FX_LOCALECATEGORY_DateHash 0xbde9abde
#define FX_LOCALECATEGORY_TimeHash 0x2d71b00f
#define FX_LOCALECATEGORY_DateTimeHash 0x158c72ed
#define FX_LOCALECATEGORY_NumHash 0x0b4ff870
#define FX_LOCALECATEGORY_TextHash 0x2d08af85
#define FX_LOCALECATEGORY_ZeroHash 0x568cb500
#define FX_LOCALECATEGORY_NullHash 0x052931bb
struct FX_LOCALESUBCATEGORYINFO {
uint32_t uHash;
const wchar_t* pName;
int32_t eSubCategory;
};
static const FX_LOCALESUBCATEGORYINFO g_FXLocaleDateTimeSubCatData[] = {
{0x14da2125, L"default", FX_LOCALEDATETIMESUBCATEGORY_Default},
{0x9041d4b0, L"short", FX_LOCALEDATETIMESUBCATEGORY_Short},
{0xa084a381, L"medium", FX_LOCALEDATETIMESUBCATEGORY_Medium},
{0xcdce56b3, L"full", FX_LOCALEDATETIMESUBCATEGORY_Full},
{0xf6b4afb0, L"long", FX_LOCALEDATETIMESUBCATEGORY_Long},
};
static const int32_t g_iFXLocaleDateTimeSubCatCount =
sizeof(g_FXLocaleDateTimeSubCatData) / sizeof(FX_LOCALESUBCATEGORYINFO);
static const FX_LOCALESUBCATEGORYINFO g_FXLocaleNumSubCatData[] = {
{0x46f95531, L"percent", FX_LOCALENUMPATTERN_Percent},
{0x4c4e8acb, L"currency", FX_LOCALENUMPATTERN_Currency},
{0x54034c2f, L"decimal", FX_LOCALENUMPATTERN_Decimal},
{0x7568e6ae, L"integer", FX_LOCALENUMPATTERN_Integer},
};
static const int32_t g_iFXLocaleNumSubCatCount =
sizeof(g_FXLocaleNumSubCatData) / sizeof(FX_LOCALESUBCATEGORYINFO);
struct FX_LOCALETIMEZONEINFO {
uint32_t uHash;
int16_t iHour;
int16_t iMinute;
};
static const FX_LOCALETIMEZONEINFO g_FXLocaleTimeZoneData[] = {
{FXBSTR_ID(0, 'C', 'D', 'T'), -5, 0}, {FXBSTR_ID(0, 'C', 'S', 'T'), -6, 0},
{FXBSTR_ID(0, 'E', 'D', 'T'), -4, 0}, {FXBSTR_ID(0, 'E', 'S', 'T'), -5, 0},
{FXBSTR_ID(0, 'M', 'D', 'T'), -6, 0}, {FXBSTR_ID(0, 'M', 'S', 'T'), -7, 0},
{FXBSTR_ID(0, 'P', 'D', 'T'), -7, 0}, {FXBSTR_ID(0, 'P', 'S', 'T'), -8, 0},
};
static const wchar_t gs_wsTimeSymbols[] = L"hHkKMSFAzZ";
static const wchar_t gs_wsDateSymbols[] = L"DJMEeGgYwW";
static const wchar_t gs_wsConstChars[] = L",-:/. ";
static int32_t FX_ParseTimeZone(const wchar_t* pStr,
int32_t iLen,
FX_TIMEZONE& tz) {
tz.tzHour = 0;
tz.tzMinute = 0;
if (iLen < 0) {
return 0;
}
int32_t iStart = 1;
int32_t iEnd = iStart + 2;
while (iStart < iLen && iStart < iEnd) {
tz.tzHour = tz.tzHour * 10 + pStr[iStart++] - '0';
}
if (iStart < iLen && pStr[iStart] == ':') {
iStart++;
}
iEnd = iStart + 2;
while (iStart < iLen && iStart < iEnd) {
tz.tzMinute = tz.tzMinute * 10 + pStr[iStart++] - '0';
}
if (pStr[0] == '-') {
tz.tzHour = -tz.tzHour;
}
return iStart;
}
class CFX_LCNumeric {
public:
CFX_LCNumeric();
CFX_LCNumeric(int64_t integral,
uint32_t fractional = 0,
int32_t exponent = 0);
explicit CFX_LCNumeric(FX_FLOAT dbRetValue);
explicit CFX_LCNumeric(double dbvalue);
explicit CFX_LCNumeric(CFX_WideString& wsNumeric);
FX_FLOAT GetFloat() const;
double GetDouble() const;
CFX_WideString ToString() const;
CFX_WideString ToString(int32_t nTreading, bool bTrimTailZeros) const;
int64_t m_Integral;
uint32_t m_Fractional;
int32_t m_Exponent;
};
static bool FX_WStringToNumeric(const CFX_WideString& wsValue,
CFX_LCNumeric& lcnum) {
lcnum.m_Integral = 0;
lcnum.m_Fractional = 0;
lcnum.m_Exponent = 0;
if (wsValue.IsEmpty())
return false;
const int32_t nIntegralMaxLen = 17;
int32_t cc = 0;
bool bNegative = false;
bool bExpSign = false;
const wchar_t* str = wsValue.c_str();
int32_t len = wsValue.GetLength();
while (cc < len && FXSYS_iswspace(str[cc]))
cc++;
if (cc >= len)
return false;
if (str[cc] == '+') {
cc++;
} else if (str[cc] == '-') {
bNegative = true;
cc++;
}
int32_t nIntegralLen = 0;
while (cc < len) {
if (str[cc] == '.')
break;
if (!FXSYS_isDecimalDigit(str[cc])) {
if ((str[cc] == 'E' || str[cc] == 'e'))
break;
return false;
}
if (nIntegralLen < nIntegralMaxLen) {
lcnum.m_Integral = lcnum.m_Integral * 10 + str[cc] - '0';
nIntegralLen++;
}
cc++;
}
lcnum.m_Integral = bNegative ? -lcnum.m_Integral : lcnum.m_Integral;
if (cc < len && str[cc] == '.') {
int scale = 0;
double fraction = 0.0;
cc++;
while (cc < len) {
if (scale >= FXSYS_FractionalScaleCount()) {
while (cc < len) {
if (!FXSYS_isDecimalDigit(str[cc]))
break;
cc++;
}
}
if (!FXSYS_isDecimalDigit(str[cc])) {
if ((str[cc] == 'E' || str[cc] == 'e'))
break;
return false;
}
fraction += FXSYS_FractionalScale(scale, FXSYS_toDecimalDigit(str[cc]));
scale++;
cc++;
}
lcnum.m_Fractional = (uint32_t)(fraction * 4294967296.0);
}
if (cc < len && (str[cc] == 'E' || str[cc] == 'e')) {
cc++;
if (cc < len) {
if (str[cc] == '+') {
cc++;
} else if (str[cc] == '-') {
bExpSign = true;
cc++;
}
}
while (cc < len) {
if (FXSYS_isDecimalDigit(str[cc]))
return false;
lcnum.m_Exponent = lcnum.m_Exponent * 10 + str[cc] - '0';
cc++;
}
lcnum.m_Exponent = bExpSign ? -lcnum.m_Exponent : lcnum.m_Exponent;
}
return true;
}
CFX_LCNumeric::CFX_LCNumeric() {
m_Integral = 0;
m_Fractional = 0;
m_Exponent = 0;
}
CFX_LCNumeric::CFX_LCNumeric(int64_t integral,
uint32_t fractional,
int32_t exponent) {
m_Integral = integral;
m_Fractional = fractional;
m_Exponent = exponent;
}
CFX_LCNumeric::CFX_LCNumeric(FX_FLOAT dbRetValue) {
m_Integral = (int64_t)dbRetValue;
m_Fractional = (uint32_t)(((dbRetValue > 0) ? (dbRetValue - m_Integral)
: (m_Integral - dbRetValue)) *
4294967296);
m_Exponent = 0;
}
CFX_LCNumeric::CFX_LCNumeric(double dbvalue) {
m_Integral = (int64_t)dbvalue;
m_Fractional = (uint32_t)(
((dbvalue > 0) ? (dbvalue - m_Integral) : (m_Integral - dbvalue)) *
4294967296);
m_Exponent = 0;
}
CFX_LCNumeric::CFX_LCNumeric(CFX_WideString& wsNumeric) {
FX_WStringToNumeric(wsNumeric, *this);
}
FX_FLOAT CFX_LCNumeric::GetFloat() const {
FX_FLOAT dbRetValue = m_Fractional / 4294967296.0f;
dbRetValue = m_Integral + (m_Integral >= 0 ? dbRetValue : -dbRetValue);
if (m_Exponent != 0) {
dbRetValue *= FXSYS_pow(10, (FX_FLOAT)m_Exponent);
}
return dbRetValue;
}
double CFX_LCNumeric::GetDouble() const {
double value = m_Fractional / 4294967296.0;
value = m_Integral + (m_Integral >= 0 ? value : -value);
if (m_Exponent != 0) {
value *= FXSYS_pow(10, (FX_FLOAT)m_Exponent);
}
return value;
}
CFX_WideString CFX_LCNumeric::ToString() const {
return ToString(8, true);
}
CFX_WideString CFX_LCNumeric::ToString(int32_t nTreading,
bool bTrimTailZeros) const {
CFX_WideString wsFormat;
wsFormat.Format(L"%%.%df", nTreading);
CFX_WideString wsResult;
wsResult.Format(wsFormat.c_str(), GetDouble());
if (bTrimTailZeros && nTreading > 0) {
wsResult.TrimRight(L"0");
wsResult.TrimRight(L".");
}
return wsResult;
}
CFX_FormatString::CFX_FormatString(IFX_LocaleMgr* pLocaleMgr, bool bUseLCID)
: m_pLocaleMgr(pLocaleMgr), m_bUseLCID(bUseLCID) {}
CFX_FormatString::~CFX_FormatString() {}
void CFX_FormatString::SplitFormatString(
const CFX_WideString& wsFormatString,
std::vector<CFX_WideString>& wsPatterns) {
int32_t iStrLen = wsFormatString.GetLength();
const wchar_t* pStr = wsFormatString.c_str();
const wchar_t* pToken = pStr;
const wchar_t* pEnd = pStr + iStrLen;
bool iQuote = false;
while (true) {
if (pStr >= pEnd) {
wsPatterns.push_back(CFX_WideString(pToken, pStr - pToken));
return;
}
if (*pStr == '\'') {
iQuote = !iQuote;
} else if (*pStr == L'|' && !iQuote) {
wsPatterns.push_back(CFX_WideString(pToken, pStr - pToken));
pToken = pStr + 1;
}
pStr++;
}
}
static CFX_WideString FX_GetLiteralText(const wchar_t* pStrPattern,
int32_t& iPattern,
int32_t iLenPattern) {
CFX_WideString wsOutput;
if (pStrPattern[iPattern] != '\'') {
return wsOutput;
}
iPattern++;
int32_t iQuote = 1;
while (iPattern < iLenPattern) {
if (pStrPattern[iPattern] == '\'') {
iQuote++;
if ((iPattern + 1 >= iLenPattern) ||
((pStrPattern[iPattern + 1] != '\'') && (iQuote % 2 == 0))) {
break;
} else {
iQuote++;
}
iPattern++;
} else if (pStrPattern[iPattern] == '\\' && (iPattern + 1 < iLenPattern) &&
pStrPattern[iPattern + 1] == 'u') {
int32_t iKeyValue = 0;
iPattern += 2;
int32_t i = 0;
while (iPattern < iLenPattern && i++ < 4) {
wchar_t ch = pStrPattern[iPattern++];
if ((ch >= '0' && ch <= '9')) {
iKeyValue = iKeyValue * 16 + ch - '0';
} else if ((ch >= 'a' && ch <= 'f')) {
iKeyValue = iKeyValue * 16 + ch - 'a' + 10;
} else if ((ch >= 'A' && ch <= 'F')) {
iKeyValue = iKeyValue * 16 + ch - 'A' + 10;
}
}
if (iKeyValue != 0) {
wsOutput += (wchar_t)(iKeyValue & 0x0000FFFF);
}
continue;
}
wsOutput += pStrPattern[iPattern++];
}
return wsOutput;
}
static CFX_WideString FX_GetLiteralTextReverse(const wchar_t* pStrPattern,
int32_t& iPattern) {
CFX_WideString wsOutput;
if (pStrPattern[iPattern] != '\'') {
return wsOutput;
}
iPattern--;
int32_t iQuote = 1;
while (iPattern >= 0) {
if (pStrPattern[iPattern] == '\'') {
iQuote++;
if (iPattern - 1 >= 0 ||
((pStrPattern[iPattern - 1] != '\'') && (iQuote % 2 == 0))) {
break;
}
iQuote++;
iPattern--;
} else if (pStrPattern[iPattern] == '\\' &&
pStrPattern[iPattern + 1] == 'u') {
iPattern--;
int32_t iKeyValue = 0;
int32_t iLen = wsOutput.GetLength();
int32_t i = 1;
for (; i < iLen && i < 5; i++) {
wchar_t ch = wsOutput[i];
if ((ch >= '0' && ch <= '9')) {
iKeyValue = iKeyValue * 16 + ch - '0';
} else if ((ch >= 'a' && ch <= 'f')) {
iKeyValue = iKeyValue * 16 + ch - 'a' + 10;
} else if ((ch >= 'A' && ch <= 'F')) {
iKeyValue = iKeyValue * 16 + ch - 'A' + 10;
}
}
if (iKeyValue != 0) {
wsOutput.Delete(0, i);
wsOutput = (wchar_t)(iKeyValue & 0x0000FFFF) + wsOutput;
}
continue;
}
wsOutput = pStrPattern[iPattern--] + wsOutput;
}
return wsOutput;
}
FX_LOCALECATEGORY CFX_FormatString::GetCategory(
const CFX_WideString& wsPattern) {
FX_LOCALECATEGORY eCategory = FX_LOCALECATEGORY_Unknown;
int32_t ccf = 0;
int32_t iLenf = wsPattern.GetLength();
const wchar_t* pStr = wsPattern.c_str();
bool bBraceOpen = false;
CFX_WideStringC wsConstChars(gs_wsConstChars);
while (ccf < iLenf) {
if (pStr[ccf] == '\'') {
FX_GetLiteralText(pStr, ccf, iLenf);
} else if (!bBraceOpen && wsConstChars.Find(pStr[ccf]) == -1) {
CFX_WideString wsCategory(pStr[ccf]);
ccf++;
while (true) {
if (ccf == iLenf) {
return eCategory;
}
if (pStr[ccf] == '.' || pStr[ccf] == '(') {
break;
}
if (pStr[ccf] == '{') {
bBraceOpen = true;
break;
}
wsCategory += pStr[ccf];
ccf++;
}
uint32_t dwHash = FX_HashCode_GetW(wsCategory.AsStringC(), false);
if (dwHash == FX_LOCALECATEGORY_DateHash) {
if (eCategory == FX_LOCALECATEGORY_Time) {
return FX_LOCALECATEGORY_DateTime;
}
eCategory = FX_LOCALECATEGORY_Date;
} else if (dwHash == FX_LOCALECATEGORY_TimeHash) {
if (eCategory == FX_LOCALECATEGORY_Date) {
return FX_LOCALECATEGORY_DateTime;
}
eCategory = FX_LOCALECATEGORY_Time;
} else if (dwHash == FX_LOCALECATEGORY_DateTimeHash) {
return FX_LOCALECATEGORY_DateTime;
} else if (dwHash == FX_LOCALECATEGORY_TextHash) {
return FX_LOCALECATEGORY_Text;
} else if (dwHash == FX_LOCALECATEGORY_NumHash) {
return FX_LOCALECATEGORY_Num;
} else if (dwHash == FX_LOCALECATEGORY_ZeroHash) {
return FX_LOCALECATEGORY_Zero;
} else if (dwHash == FX_LOCALECATEGORY_NullHash) {
return FX_LOCALECATEGORY_Null;
}
} else if (pStr[ccf] == '}') {
bBraceOpen = false;
}
ccf++;
}
return eCategory;
}
static uint16_t FX_WStringToLCID(const wchar_t* pstrLCID) {
if (!pstrLCID) {
return 0;
}
wchar_t* pEnd;
return (uint16_t)wcstol((wchar_t*)pstrLCID, &pEnd, 16);
}
uint16_t CFX_FormatString::GetLCID(const CFX_WideString& wsPattern) {
return FX_WStringToLCID(GetLocaleName(wsPattern).c_str());
}
CFX_WideString CFX_FormatString::GetLocaleName(
const CFX_WideString& wsPattern) {
int32_t ccf = 0;
int32_t iLenf = wsPattern.GetLength();
const wchar_t* pStr = wsPattern.c_str();
while (ccf < iLenf) {
if (pStr[ccf] == '\'') {
FX_GetLiteralText(pStr, ccf, iLenf);
} else if (pStr[ccf] == '(') {
ccf++;
CFX_WideString wsLCID;
while (ccf < iLenf && pStr[ccf] != ')') {
wsLCID += pStr[ccf++];
}
return wsLCID;
}
ccf++;
}
return CFX_WideString();
}
IFX_Locale* CFX_FormatString::GetTextFormat(const CFX_WideString& wsPattern,
const CFX_WideStringC& wsCategory,
CFX_WideString& wsPurgePattern) {
IFX_Locale* pLocale = nullptr;
int32_t ccf = 0;
int32_t iLenf = wsPattern.GetLength();
const wchar_t* pStr = wsPattern.c_str();
bool bBrackOpen = false;
CFX_WideStringC wsConstChars(gs_wsConstChars);
while (ccf < iLenf) {
if (pStr[ccf] == '\'') {
int32_t iCurChar = ccf;
FX_GetLiteralText(pStr, ccf, iLenf);
wsPurgePattern += CFX_WideStringC(pStr + iCurChar, ccf - iCurChar + 1);
} else if (!bBrackOpen && wsConstChars.Find(pStr[ccf]) == -1) {
CFX_WideString wsSearchCategory(pStr[ccf]);
ccf++;
while (ccf < iLenf && pStr[ccf] != '{' && pStr[ccf] != '.' &&
pStr[ccf] != '(') {
wsSearchCategory += pStr[ccf];
ccf++;
}
if (wsSearchCategory != wsCategory) {
continue;
}
while (ccf < iLenf) {
if (pStr[ccf] == '(') {
ccf++;
CFX_WideString wsLCID;
while (ccf < iLenf && pStr[ccf] != ')') {
wsLCID += pStr[ccf++];
}
pLocale = GetPatternLocale(wsLCID);
} else if (pStr[ccf] == '{') {
bBrackOpen = true;
break;
}
ccf++;
}
} else if (pStr[ccf] != '}') {
wsPurgePattern += pStr[ccf];
}
ccf++;
}
if (!bBrackOpen) {
wsPurgePattern = wsPattern;
}
if (!pLocale) {
pLocale = m_pLocaleMgr->GetDefLocale();
}
return pLocale;
}
#define FX_NUMSTYLE_Percent 0x01
#define FX_NUMSTYLE_Exponent 0x02
#define FX_NUMSTYLE_DotVorv 0x04
IFX_Locale* CFX_FormatString::GetNumericFormat(const CFX_WideString& wsPattern,
int32_t& iDotIndex,
uint32_t& dwStyle,
CFX_WideString& wsPurgePattern) {
dwStyle = 0;
IFX_Locale* pLocale = nullptr;
int32_t ccf = 0;
int32_t iLenf = wsPattern.GetLength();
const wchar_t* pStr = wsPattern.c_str();
bool bFindDot = false;
bool bBrackOpen = false;
CFX_WideStringC wsConstChars(gs_wsConstChars);
while (ccf < iLenf) {
if (pStr[ccf] == '\'') {
int32_t iCurChar = ccf;
FX_GetLiteralText(pStr, ccf, iLenf);
wsPurgePattern += CFX_WideStringC(pStr + iCurChar, ccf - iCurChar + 1);
} else if (!bBrackOpen && wsConstChars.Find(pStr[ccf]) == -1) {
CFX_WideString wsCategory(pStr[ccf]);
ccf++;
while (ccf < iLenf && pStr[ccf] != '{' && pStr[ccf] != '.' &&
pStr[ccf] != '(') {
wsCategory += pStr[ccf];
ccf++;
}
if (wsCategory != L"num") {
bBrackOpen = true;
ccf = 0;
continue;
}
while (ccf < iLenf) {
if (pStr[ccf] == '(') {
ccf++;
CFX_WideString wsLCID;
while (ccf < iLenf && pStr[ccf] != ')') {
wsLCID += pStr[ccf++];
}
pLocale = GetPatternLocale(wsLCID);
} else if (pStr[ccf] == '{') {
bBrackOpen = true;
break;
} else if (pStr[ccf] == '.') {
CFX_WideString wsSubCategory;
ccf++;
while (ccf < iLenf && pStr[ccf] != '(' && pStr[ccf] != '{') {
wsSubCategory += pStr[ccf++];
}
uint32_t dwSubHash =
FX_HashCode_GetW(wsSubCategory.AsStringC(), false);
FX_LOCALENUMSUBCATEGORY eSubCategory = FX_LOCALENUMPATTERN_Decimal;
for (int32_t i = 0; i < g_iFXLocaleNumSubCatCount; i++) {
if (g_FXLocaleNumSubCatData[i].uHash == dwSubHash) {
eSubCategory = (FX_LOCALENUMSUBCATEGORY)g_FXLocaleNumSubCatData[i]
.eSubCategory;
break;
}
}
wsSubCategory.clear();
if (!pLocale) {
pLocale = m_pLocaleMgr->GetDefLocale();
}
ASSERT(pLocale);
pLocale->GetNumPattern(eSubCategory, wsSubCategory);
iDotIndex = wsSubCategory.Find('.');
if (iDotIndex > 0) {
iDotIndex += wsPurgePattern.GetLength();
bFindDot = true;
dwStyle |= FX_NUMSTYLE_DotVorv;
}
wsPurgePattern += wsSubCategory;
if (eSubCategory == FX_LOCALENUMPATTERN_Percent) {
dwStyle |= FX_NUMSTYLE_Percent;
}
continue;
}
ccf++;
}
} else if (pStr[ccf] == 'E') {
dwStyle |= FX_NUMSTYLE_Exponent;
wsPurgePattern += pStr[ccf];
} else if (pStr[ccf] == '%') {
dwStyle |= FX_NUMSTYLE_Percent;
wsPurgePattern += pStr[ccf];
} else if (pStr[ccf] != '}') {
wsPurgePattern += pStr[ccf];
}
if (!bFindDot) {
if (pStr[ccf] == '.' || pStr[ccf] == 'V' || pStr[ccf] == 'v') {
bFindDot = true;
iDotIndex = wsPurgePattern.GetLength() - 1;
dwStyle |= FX_NUMSTYLE_DotVorv;
}
}
ccf++;
}
if (!bFindDot) {
iDotIndex = wsPurgePattern.GetLength();
}
if (!pLocale) {
pLocale = m_pLocaleMgr->GetDefLocale();
}
return pLocale;
}
static bool FX_GetNumericDotIndex(const CFX_WideString& wsNum,
const CFX_WideString& wsDotSymbol,
int32_t& iDotIndex) {
int32_t ccf = 0;
int32_t iLenf = wsNum.GetLength();
const wchar_t* pStr = wsNum.c_str();
int32_t iLenDot = wsDotSymbol.GetLength();
while (ccf < iLenf) {
if (pStr[ccf] == '\'') {
FX_GetLiteralText(pStr, ccf, iLenf);
} else if (ccf + iLenDot <= iLenf &&
!FXSYS_wcsncmp(pStr + ccf, wsDotSymbol.c_str(), iLenDot)) {
iDotIndex = ccf;
return true;
}
ccf++;
}
iDotIndex = wsNum.Find('.');
if (iDotIndex < 0) {
iDotIndex = iLenf;
return false;
}
return true;
}
bool CFX_FormatString::ParseText(const CFX_WideString& wsSrcText,
const CFX_WideString& wsPattern,
CFX_WideString& wsValue) {
wsValue.clear();
if (wsSrcText.IsEmpty() || wsPattern.IsEmpty()) {
return false;
}
CFX_WideString wsTextFormat;
GetTextFormat(wsPattern, L"text", wsTextFormat);
if (wsTextFormat.IsEmpty()) {
return false;
}
int32_t iText = 0, iPattern = 0;
const wchar_t* pStrText = wsSrcText.c_str();
int32_t iLenText = wsSrcText.GetLength();
const wchar_t* pStrPattern = wsTextFormat.c_str();
int32_t iLenPattern = wsTextFormat.GetLength();
while (iPattern < iLenPattern && iText < iLenText) {
switch (pStrPattern[iPattern]) {
case '\'': {
CFX_WideString wsLiteral =
FX_GetLiteralText(pStrPattern, iPattern, iLenPattern);
int32_t iLiteralLen = wsLiteral.GetLength();
if (iText + iLiteralLen > iLenText ||
FXSYS_wcsncmp(pStrText + iText, wsLiteral.c_str(), iLiteralLen)) {
wsValue = wsSrcText;
return false;
}
iText += iLiteralLen;
iPattern++;
break;
}
case 'A':
if (FXSYS_iswalpha(pStrText[iText])) {
wsValue += pStrText[iText];
iText++;
}
iPattern++;
break;
case 'X':
wsValue += pStrText[iText];
iText++;
iPattern++;
break;
case 'O':
case '0':
if (FXSYS_isDecimalDigit(pStrText[iText]) ||
FXSYS_iswalpha(pStrText[iText])) {
wsValue += pStrText[iText];
iText++;
}
iPattern++;
break;
case '9':
if (FXSYS_isDecimalDigit(pStrText[iText])) {
wsValue += pStrText[iText];
iText++;
}
iPattern++;
break;
default:
if (pStrPattern[iPattern] != pStrText[iText]) {
wsValue = wsSrcText;
return false;
}
iPattern++;
iText++;
break;
}
}
return iPattern == iLenPattern && iText == iLenText;
}
bool CFX_FormatString::ParseNum(const CFX_WideString& wsSrcNum,
const CFX_WideString& wsPattern,
FX_FLOAT& fValue) {
fValue = 0.0f;
if (wsSrcNum.IsEmpty() || wsPattern.IsEmpty()) {
return false;
}
int32_t dot_index_f = -1;
uint32_t dwFormatStyle = 0;
CFX_WideString wsNumFormat;
IFX_Locale* pLocale =
GetNumericFormat(wsPattern, dot_index_f, dwFormatStyle, wsNumFormat);
if (!pLocale || wsNumFormat.IsEmpty()) {
return false;
}
int32_t iExponent = 0;
CFX_WideString wsDotSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Decimal, wsDotSymbol);
CFX_WideString wsGroupSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Grouping, wsGroupSymbol);
int32_t iGroupLen = wsGroupSymbol.GetLength();
CFX_WideString wsMinus;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Minus, wsMinus);
int32_t iMinusLen = wsMinus.GetLength();
int cc = 0, ccf = 0;
const wchar_t* str = wsSrcNum.c_str();
int len = wsSrcNum.GetLength();
const wchar_t* strf = wsNumFormat.c_str();
int lenf = wsNumFormat.GetLength();
double dbRetValue = 0;
double coeff = 1;
bool bHavePercentSymbol = false;
bool bNeg = false;
bool bReverseParse = false;
int32_t dot_index = 0;
if (!FX_GetNumericDotIndex(wsSrcNum, wsDotSymbol, dot_index) &&
(dwFormatStyle & FX_NUMSTYLE_DotVorv)) {
bReverseParse = true;
}
bReverseParse = false;
if (bReverseParse) {
ccf = lenf - 1;
cc = len - 1;
while (ccf > dot_index_f && cc >= 0) {
switch (strf[ccf]) {
case '\'': {
CFX_WideString wsLiteral = FX_GetLiteralTextReverse(strf, ccf);
int32_t iLiteralLen = wsLiteral.GetLength();
cc -= iLiteralLen - 1;
if (cc < 0 ||
FXSYS_wcsncmp(str + cc, wsLiteral.c_str(), iLiteralLen)) {
return false;
}
cc--;
ccf--;
break;
}
case '9':
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
dbRetValue = dbRetValue * coeff + (str[cc] - '0') * 0.1;
coeff *= 0.1;
cc--;
ccf--;
break;
case 'z':
if (cc >= 0) {
dbRetValue = dbRetValue * coeff + (str[cc] - '0') * 0.1;
coeff *= 0.1;
cc--;
}
ccf--;
break;
case 'Z':
if (str[cc] != ' ') {
dbRetValue = dbRetValue * coeff + (str[cc] - '0') * 0.1;
coeff *= 0.1;
}
cc--;
ccf--;
break;
case 'S':
if (str[cc] == '+' || str[cc] == ' ') {
cc--;
} else {
cc -= iMinusLen - 1;
if (cc < 0 || FXSYS_wcsncmp(str + cc, wsMinus.c_str(), iMinusLen)) {
return false;
}
cc--;
bNeg = true;
}
ccf--;
break;
case 's':
if (str[cc] == '+') {
cc--;
} else {
cc -= iMinusLen - 1;
if (cc < 0 || FXSYS_wcsncmp(str + cc, wsMinus.c_str(), iMinusLen)) {
return false;
}
cc--;
bNeg = true;
}
ccf--;
break;
case 'E': {
if (cc >= dot_index) {
return false;
}
bool bExpSign = false;
while (cc >= 0) {
if (str[cc] == 'E' || str[cc] == 'e') {
break;
}
if (FXSYS_isDecimalDigit(str[cc])) {
iExponent = iExponent + (str[cc] - '0') * 10;
cc--;
continue;
} else if (str[cc] == '+') {
cc--;
continue;
} else if (cc - iMinusLen + 1 > 0 &&
!FXSYS_wcsncmp(str + (cc - iMinusLen + 1),
wsMinus.c_str(), iMinusLen)) {
bExpSign = true;
cc -= iMinusLen;
} else {
return false;
}
}
cc--;
iExponent = bExpSign ? -iExponent : iExponent;
ccf--;
} break;
case '$': {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_CurrencySymbol,
wsSymbol);
int32_t iSymbolLen = wsSymbol.GetLength();
cc -= iSymbolLen - 1;
if (cc < 0 || FXSYS_wcsncmp(str + cc, wsSymbol.c_str(), iSymbolLen)) {
return false;
}
cc--;
ccf--;
} break;
case 'r':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'c') {
if (str[cc] == 'R' && cc - 1 >= 0 && str[cc - 1] == 'C') {
bNeg = true;
cc -= 2;
}
ccf -= 2;
} else {
ccf--;
}
break;
case 'R':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'C') {
if (str[cc] == ' ') {
cc++;
} else if (str[cc] == 'R' && cc - 1 >= 0 && str[cc - 1] == 'C') {
bNeg = true;
cc -= 2;
}
ccf -= 2;
} else {
ccf--;
}
break;
case 'b':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'd') {
if (str[cc] == 'B' && cc - 1 >= 0 && str[cc - 1] == 'D') {
bNeg = true;
cc -= 2;
}
ccf -= 2;
} else {
ccf--;
}
break;
case 'B':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'D') {
if (str[cc] == ' ') {
cc++;
} else if (str[cc] == 'B' && cc - 1 >= 0 && str[cc - 1] == 'D') {
bNeg = true;
cc -= 2;
}
ccf -= 2;
} else {
ccf--;
}
break;
case '.':
case 'V':
case 'v':
return false;
case '%': {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Percent, wsSymbol);
int32_t iSysmbolLen = wsSymbol.GetLength();
cc -= iSysmbolLen - 1;
if (cc < 0 ||
FXSYS_wcsncmp(str + cc, wsSymbol.c_str(), iSysmbolLen)) {
return false;
}
cc--;
ccf--;
bHavePercentSymbol = true;
} break;
case '8':
while (ccf < lenf && strf[ccf] == '8') {
ccf++;
}
while (cc < len && FXSYS_isDecimalDigit(str[cc])) {
dbRetValue = (str[cc] - '0') * coeff + dbRetValue;
coeff *= 0.1;
cc++;
}
break;
case ',': {
if (cc >= 0) {
cc -= iGroupLen - 1;
if (cc >= 0 &&
FXSYS_wcsncmp(str + cc, wsGroupSymbol.c_str(), iGroupLen) ==
0) {
cc--;
} else {
cc += iGroupLen - 1;
}
}
ccf--;
} break;
case '(':
if (str[cc] == L'(') {
bNeg = true;
} else if (str[cc] != L' ') {
return false;
}
cc--;
ccf--;
break;
case ')':
if (str[cc] == L')') {
bNeg = true;
} else if (str[cc] != L' ') {
return false;
}
cc--;
ccf--;
break;
default:
if (strf[ccf] != str[cc]) {
return false;
}
cc--;
ccf--;
}
}
dot_index = cc + 1;
}
ccf = dot_index_f - 1;
cc = dot_index - 1;
coeff = 1;
while (ccf >= 0 && cc >= 0) {
switch (strf[ccf]) {
case '\'': {
CFX_WideString wsLiteral = FX_GetLiteralTextReverse(strf, ccf);
int32_t iLiteralLen = wsLiteral.GetLength();
cc -= iLiteralLen - 1;
if (cc < 0 || FXSYS_wcsncmp(str + cc, wsLiteral.c_str(), iLiteralLen)) {
return false;
}
cc--;
ccf--;
break;
}
case '9':
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
dbRetValue = dbRetValue + (str[cc] - '0') * coeff;
coeff *= 10;
cc--;
ccf--;
break;
case 'z':
if (FXSYS_isDecimalDigit(str[cc])) {
dbRetValue = dbRetValue + (str[cc] - '0') * coeff;
coeff *= 10;
cc--;
}
ccf--;
break;
case 'Z':
if (str[cc] != ' ') {
if (FXSYS_isDecimalDigit(str[cc])) {
dbRetValue = dbRetValue + (str[cc] - '0') * coeff;
coeff *= 10;
cc--;
}
} else {
cc--;
}
ccf--;
break;
case 'S':
if (str[cc] == '+' || str[cc] == ' ') {
cc--;
} else {
cc -= iMinusLen - 1;
if (cc < 0 || FXSYS_wcsncmp(str + cc, wsMinus.c_str(), iMinusLen)) {
return false;
}
cc--;
bNeg = true;
}
ccf--;
break;
case 's':
if (str[cc] == '+') {
cc--;
} else {
cc -= iMinusLen - 1;
if (cc < 0 || FXSYS_wcsncmp(str + cc, wsMinus.c_str(), iMinusLen)) {
return false;
}
cc--;
bNeg = true;
}
ccf--;
break;
case 'E': {
if (cc >= dot_index) {
return false;
}
bool bExpSign = false;
while (cc >= 0) {
if (str[cc] == 'E' || str[cc] == 'e') {
break;
}
if (FXSYS_isDecimalDigit(str[cc])) {
iExponent = iExponent + (str[cc] - '0') * 10;
cc--;
continue;
} else if (str[cc] == '+') {
cc--;
continue;
} else if (cc - iMinusLen + 1 > 0 &&
!FXSYS_wcsncmp(str + (cc - iMinusLen + 1), wsMinus.c_str(),
iMinusLen)) {
bExpSign = true;
cc -= iMinusLen;
} else {
return false;
}
}
cc--;
iExponent = bExpSign ? -iExponent : iExponent;
ccf--;
} break;
case '$': {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_CurrencySymbol, wsSymbol);
int32_t iSymbolLen = wsSymbol.GetLength();
cc -= iSymbolLen - 1;
if (cc < 0 || FXSYS_wcsncmp(str + cc, wsSymbol.c_str(), iSymbolLen)) {
return false;
}
cc--;
ccf--;
} break;
case 'r':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'c') {
if (str[cc] == 'R' && cc - 1 >= 0 && str[cc - 1] == 'C') {
bNeg = true;
cc -= 2;
}
ccf -= 2;
} else {
ccf--;
}
break;
case 'R':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'C') {
if (str[cc] == ' ') {
cc++;
} else if (str[cc] == 'R' && cc - 1 >= 0 && str[cc - 1] == 'C') {
bNeg = true;
cc -= 2;
}
ccf -= 2;
} else {
ccf--;
}
break;
case 'b':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'd') {
if (str[cc] == 'B' && cc - 1 >= 0 && str[cc - 1] == 'D') {
bNeg = true;
cc -= 2;
}
ccf -= 2;
} else {
ccf--;
}
break;
case 'B':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'D') {
if (str[cc] == ' ') {
cc++;
} else if (str[cc] == 'B' && cc - 1 >= 0 && str[cc - 1] == 'D') {
bNeg = true;
cc -= 2;
}
ccf -= 2;
} else {
ccf--;
}
break;
case '.':
case 'V':
case 'v':
return false;
case '%': {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Percent, wsSymbol);
int32_t iSysmbolLen = wsSymbol.GetLength();
cc -= iSysmbolLen - 1;
if (cc < 0 || FXSYS_wcsncmp(str + cc, wsSymbol.c_str(), iSysmbolLen)) {
return false;
}
cc--;
ccf--;
bHavePercentSymbol = true;
} break;
case '8':
return false;
case ',': {
if (cc >= 0) {
cc -= iGroupLen - 1;
if (cc >= 0 &&
FXSYS_wcsncmp(str + cc, wsGroupSymbol.c_str(), iGroupLen) == 0) {
cc--;
} else {
cc += iGroupLen - 1;
}
}
ccf--;
} break;
case '(':
if (str[cc] == L'(') {
bNeg = true;
} else if (str[cc] != L' ') {
return false;
}
cc--;
ccf--;
break;
case ')':
if (str[cc] == L')') {
bNeg = true;
} else if (str[cc] != L' ') {
return false;
}
cc--;
ccf--;
break;
default:
if (strf[ccf] != str[cc]) {
return false;
}
cc--;
ccf--;
}
}
if (cc >= 0) {
return false;
}
if (!bReverseParse) {
ccf = dot_index_f + 1;
cc = (dot_index == len) ? len : dot_index + 1;
coeff = 0.1;
while (cc < len && ccf < lenf) {
switch (strf[ccf]) {
case '\'': {
CFX_WideString wsLiteral = FX_GetLiteralText(strf, ccf, lenf);
int32_t iLiteralLen = wsLiteral.GetLength();
if (cc + iLiteralLen > len ||
FXSYS_wcsncmp(str + cc, wsLiteral.c_str(), iLiteralLen)) {
return false;
}
cc += iLiteralLen;
ccf++;
break;
}
case '9':
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
{
dbRetValue = dbRetValue + (str[cc] - '0') * coeff;
coeff *= 0.1;
}
cc++;
ccf++;
break;
case 'z':
if (FXSYS_isDecimalDigit(str[cc])) {
dbRetValue = dbRetValue + (str[cc] - '0') * coeff;
coeff *= 0.1;
cc++;
}
ccf++;
break;
case 'Z':
if (str[cc] != ' ') {
if (FXSYS_isDecimalDigit(str[cc])) {
dbRetValue = dbRetValue + (str[cc] - '0') * coeff;
coeff *= 0.1;
cc++;
}
} else {
cc++;
}
ccf++;
break;
case 'S':
if (str[cc] == '+' || str[cc] == ' ') {
cc++;
} else {
if (cc + iMinusLen > len ||
FXSYS_wcsncmp(str + cc, wsMinus.c_str(), iMinusLen)) {
return false;
}
bNeg = true;
cc += iMinusLen;
}
ccf++;
break;
case 's':
if (str[cc] == '+') {
cc++;
} else {
if (cc + iMinusLen > len ||
FXSYS_wcsncmp(str + cc, wsMinus.c_str(), iMinusLen)) {
return false;
}
bNeg = true;
cc += iMinusLen;
}
ccf++;
break;
case 'E': {
if (cc >= len || (str[cc] != 'E' && str[cc] != 'e')) {
return false;
}
bool bExpSign = false;
cc++;
if (cc < len) {
if (str[cc] == '+') {
cc++;
} else if (str[cc] == '-') {
bExpSign = true;
cc++;
}
}
while (cc < len) {
if (!FXSYS_isDecimalDigit(str[cc])) {
break;
}
iExponent = iExponent * 10 + str[cc] - '0';
cc++;
}
iExponent = bExpSign ? -iExponent : iExponent;
ccf++;
} break;
case '$': {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_CurrencySymbol,
wsSymbol);
int32_t iSymbolLen = wsSymbol.GetLength();
if (cc + iSymbolLen > len ||
FXSYS_wcsncmp(str + cc, wsSymbol.c_str(), iSymbolLen)) {
return false;
}
cc += iSymbolLen;
ccf++;
} break;
case 'c':
if (ccf + 1 < lenf && strf[ccf + 1] == 'r') {
if (str[cc] == 'C' && cc + 1 < len && str[cc + 1] == 'R') {
bNeg = true;
cc += 2;
}
ccf += 2;
}
break;
case 'C':
if (ccf + 1 < lenf && strf[ccf + 1] == 'R') {
if (str[cc] == ' ') {
cc++;
} else if (str[cc] == 'C' && cc + 1 < len && str[cc + 1] == 'R') {
bNeg = true;
cc += 2;
}
ccf += 2;
}
break;
case 'd':
if (ccf + 1 < lenf && strf[ccf + 1] == 'b') {
if (str[cc] == 'D' && cc + 1 < len && str[cc + 1] == 'B') {
bNeg = true;
cc += 2;
}
ccf += 2;
}
break;
case 'D':
if (ccf + 1 < lenf && strf[ccf + 1] == 'B') {
if (str[cc] == ' ') {
cc++;
} else if (str[cc] == 'D' && cc + 1 < len && str[cc + 1] == 'B') {
bNeg = true;
cc += 2;
}
ccf += 2;
}
break;
case '.':
case 'V':
case 'v':
return false;
case '%': {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Percent, wsSymbol);
int32_t iSysmbolLen = wsSymbol.GetLength();
if (cc + iSysmbolLen <= len &&
!FXSYS_wcsncmp(str + cc, wsSymbol.c_str(), iSysmbolLen)) {
cc += iSysmbolLen;
}
ccf++;
bHavePercentSymbol = true;
} break;
case '8': {
while (ccf < lenf && strf[ccf] == '8') {
ccf++;
}
while (cc < len && FXSYS_isDecimalDigit(str[cc])) {
dbRetValue = (str[cc] - '0') * coeff + dbRetValue;
coeff *= 0.1;
cc++;
}
} break;
case ',': {
if (cc + iGroupLen <= len &&
FXSYS_wcsncmp(str + cc, wsGroupSymbol.c_str(), iGroupLen) == 0) {
cc += iGroupLen;
}
ccf++;
} break;
case '(':
if (str[cc] == L'(') {
bNeg = true;
} else if (str[cc] != L' ') {
return false;
}
cc++;
ccf++;
break;
case ')':
if (str[cc] == L')') {
bNeg = true;
} else if (str[cc] != L' ') {
return false;
}
cc++;
ccf++;
break;
default:
if (strf[ccf] != str[cc]) {
return false;
}
cc++;
ccf++;
}
}
if (cc != len) {
return false;
}
}
if (iExponent) {
dbRetValue *= FXSYS_pow(10, (FX_FLOAT)iExponent);
}
if (bHavePercentSymbol) {
dbRetValue /= 100.0;
}
if (bNeg) {
dbRetValue = -dbRetValue;
}
fValue = (FX_FLOAT)dbRetValue;
return true;
}
bool CFX_FormatString::ParseNum(const CFX_WideString& wsSrcNum,
const CFX_WideString& wsPattern,
CFX_WideString& wsValue) {
wsValue.clear();
if (wsSrcNum.IsEmpty() || wsPattern.IsEmpty()) {
return false;
}
int32_t dot_index_f = -1;
uint32_t dwFormatStyle = 0;
CFX_WideString wsNumFormat;
IFX_Locale* pLocale =
GetNumericFormat(wsPattern, dot_index_f, dwFormatStyle, wsNumFormat);
if (!pLocale || wsNumFormat.IsEmpty()) {
return false;
}
int32_t iExponent = 0;
CFX_WideString wsDotSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Decimal, wsDotSymbol);
CFX_WideString wsGroupSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Grouping, wsGroupSymbol);
int32_t iGroupLen = wsGroupSymbol.GetLength();
CFX_WideString wsMinus;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Minus, wsMinus);
int32_t iMinusLen = wsMinus.GetLength();
int cc = 0, ccf = 0;
const wchar_t* str = wsSrcNum.c_str();
int len = wsSrcNum.GetLength();
const wchar_t* strf = wsNumFormat.c_str();
int lenf = wsNumFormat.GetLength();
bool bHavePercentSymbol = false;
bool bNeg = false;
bool bReverseParse = false;
int32_t dot_index = 0;
if (!FX_GetNumericDotIndex(wsSrcNum, wsDotSymbol, dot_index) &&
(dwFormatStyle & FX_NUMSTYLE_DotVorv)) {
bReverseParse = true;
}
bReverseParse = false;
ccf = dot_index_f - 1;
cc = dot_index - 1;
while (ccf >= 0 && cc >= 0) {
switch (strf[ccf]) {
case '\'': {
CFX_WideString wsLiteral = FX_GetLiteralTextReverse(strf, ccf);
int32_t iLiteralLen = wsLiteral.GetLength();
cc -= iLiteralLen - 1;
if (cc < 0 || FXSYS_wcsncmp(str + cc, wsLiteral.c_str(), iLiteralLen)) {
return false;
}
cc--;
ccf--;
break;
}
case '9':
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
wsValue = str[cc] + wsValue;
cc--;
ccf--;
break;
case 'z':
if (FXSYS_isDecimalDigit(str[cc])) {
wsValue = str[cc] + wsValue;
cc--;
}
ccf--;
break;
case 'Z':
if (str[cc] != ' ') {
if (FXSYS_isDecimalDigit(str[cc])) {
wsValue = str[cc] + wsValue;
cc--;
}
} else {
cc--;
}
ccf--;
break;
case 'S':
if (str[cc] == '+' || str[cc] == ' ') {
cc--;
} else {
cc -= iMinusLen - 1;
if (cc < 0 || FXSYS_wcsncmp(str + cc, wsMinus.c_str(), iMinusLen)) {
return false;
}
cc--;
bNeg = true;
}
ccf--;
break;
case 's':
if (str[cc] == '+') {
cc--;
} else {
cc -= iMinusLen - 1;
if (cc < 0 || FXSYS_wcsncmp(str + cc, wsMinus.c_str(), iMinusLen)) {
return false;
}
cc--;
bNeg = true;
}
ccf--;
break;
case 'E': {
if (cc >= dot_index) {
return false;
}
bool bExpSign = false;
while (cc >= 0) {
if (str[cc] == 'E' || str[cc] == 'e') {
break;
}
if (FXSYS_isDecimalDigit(str[cc])) {
iExponent = iExponent + (str[cc] - '0') * 10;
cc--;
continue;
} else if (str[cc] == '+') {
cc--;
continue;
} else if (cc - iMinusLen + 1 > 0 &&
!FXSYS_wcsncmp(str + (cc - iMinusLen + 1), wsMinus.c_str(),
iMinusLen)) {
bExpSign = true;
cc -= iMinusLen;
} else {
return false;
}
}
cc--;
iExponent = bExpSign ? -iExponent : iExponent;
ccf--;
} break;
case '$': {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_CurrencySymbol, wsSymbol);
int32_t iSymbolLen = wsSymbol.GetLength();
cc -= iSymbolLen - 1;
if (cc < 0 || FXSYS_wcsncmp(str + cc, wsSymbol.c_str(), iSymbolLen)) {
return false;
}
cc--;
ccf--;
} break;
case 'r':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'c') {
if (str[cc] == 'R' && cc - 1 >= 0 && str[cc - 1] == 'C') {
bNeg = true;
cc -= 2;
}
ccf -= 2;
} else {
ccf--;
}
break;
case 'R':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'C') {
if (str[cc] == ' ') {
cc++;
} else if (str[cc] == 'R' && cc - 1 >= 0 && str[cc - 1] == 'C') {
bNeg = true;
cc -= 2;
}
ccf -= 2;
} else {
ccf--;
}
break;
case 'b':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'd') {
if (str[cc] == 'B' && cc - 1 >= 0 && str[cc - 1] == 'D') {
bNeg = true;
cc -= 2;
}
ccf -= 2;
} else {
ccf--;
}
break;
case 'B':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'D') {
if (str[cc] == ' ') {
cc++;
} else if (str[cc] == 'B' && cc - 1 >= 0 && str[cc - 1] == 'D') {
bNeg = true;
cc -= 2;
}
ccf -= 2;
} else {
ccf--;
}
break;
case '.':
case 'V':
case 'v':
return false;
case '%': {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Percent, wsSymbol);
int32_t iSysmbolLen = wsSymbol.GetLength();
cc -= iSysmbolLen - 1;
if (cc < 0 || FXSYS_wcsncmp(str + cc, wsSymbol.c_str(), iSysmbolLen)) {
return false;
}
cc--;
ccf--;
bHavePercentSymbol = true;
} break;
case '8':
return false;
case ',': {
if (cc >= 0) {
cc -= iGroupLen - 1;
if (cc >= 0 &&
FXSYS_wcsncmp(str + cc, wsGroupSymbol.c_str(), iGroupLen) == 0) {
cc--;
} else {
cc += iGroupLen - 1;
}
}
ccf--;
} break;
case '(':
if (str[cc] == L'(') {
bNeg = true;
} else if (str[cc] != L' ') {
return false;
}
cc--;
ccf--;
break;
case ')':
if (str[cc] == L')') {
bNeg = true;
} else if (str[cc] != L' ') {
return false;
}
cc--;
ccf--;
break;
default:
if (strf[ccf] != str[cc]) {
return false;
}
cc--;
ccf--;
}
}
if (cc >= 0) {
if (str[cc] == '-') {
bNeg = true;
cc--;
}
if (cc >= 0) {
return false;
}
}
if (dot_index < len && (dwFormatStyle & FX_NUMSTYLE_DotVorv)) {
wsValue += '.';
}
if (!bReverseParse) {
ccf = dot_index_f + 1;
cc = (dot_index == len) ? len : dot_index + 1;
while (cc < len && ccf < lenf) {
switch (strf[ccf]) {
case '\'': {
CFX_WideString wsLiteral = FX_GetLiteralText(strf, ccf, lenf);
int32_t iLiteralLen = wsLiteral.GetLength();
if (cc + iLiteralLen > len ||
FXSYS_wcsncmp(str + cc, wsLiteral.c_str(), iLiteralLen)) {
return false;
}
cc += iLiteralLen;
ccf++;
break;
}
case '9':
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
{ wsValue += str[cc]; }
cc++;
ccf++;
break;
case 'z':
if (FXSYS_isDecimalDigit(str[cc])) {
wsValue += str[cc];
cc++;
}
ccf++;
break;
case 'Z':
if (str[cc] != ' ') {
if (FXSYS_isDecimalDigit(str[cc])) {
wsValue += str[cc];
cc++;
}
} else {
cc++;
}
ccf++;
break;
case 'S':
if (str[cc] == '+' || str[cc] == ' ') {
cc++;
} else {
if (cc + iMinusLen > len ||
FXSYS_wcsncmp(str + cc, wsMinus.c_str(), iMinusLen)) {
return false;
}
bNeg = true;
cc += iMinusLen;
}
ccf++;
break;
case 's':
if (str[cc] == '+') {
cc++;
} else {
if (cc + iMinusLen > len ||
FXSYS_wcsncmp(str + cc, wsMinus.c_str(), iMinusLen)) {
return false;
}
bNeg = true;
cc += iMinusLen;
}
ccf++;
break;
case 'E': {
if (cc >= len || (str[cc] != 'E' && str[cc] != 'e')) {
return false;
}
bool bExpSign = false;
cc++;
if (cc < len) {
if (str[cc] == '+') {
cc++;
} else if (str[cc] == '-') {
bExpSign = true;
cc++;
}
}
while (cc < len) {
if (!FXSYS_isDecimalDigit(str[cc])) {
break;
}
iExponent = iExponent * 10 + str[cc] - '0';
cc++;
}
iExponent = bExpSign ? -iExponent : iExponent;
ccf++;
} break;
case '$': {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_CurrencySymbol,
wsSymbol);
int32_t iSymbolLen = wsSymbol.GetLength();
if (cc + iSymbolLen > len ||
FXSYS_wcsncmp(str + cc, wsSymbol.c_str(), iSymbolLen)) {
return false;
}
cc += iSymbolLen;
ccf++;
} break;
case 'c':
if (ccf + 1 < lenf && strf[ccf + 1] == 'r') {
if (str[cc] == 'C' && cc + 1 < len && str[cc + 1] == 'R') {
bNeg = true;
cc += 2;
}
ccf += 2;
}
break;
case 'C':
if (ccf + 1 < lenf && strf[ccf + 1] == 'R') {
if (str[cc] == ' ') {
cc++;
} else if (str[cc] == 'C' && cc + 1 < len && str[cc + 1] == 'R') {
bNeg = true;
cc += 2;
}
ccf += 2;
}
break;
case 'd':
if (ccf + 1 < lenf && strf[ccf + 1] == 'b') {
if (str[cc] == 'D' && cc + 1 < len && str[cc + 1] == 'B') {
bNeg = true;
cc += 2;
}
ccf += 2;
}
break;
case 'D':
if (ccf + 1 < lenf && strf[ccf + 1] == 'B') {
if (str[cc] == ' ') {
cc++;
} else if (str[cc] == 'D' && cc + 1 < len && str[cc + 1] == 'B') {
bNeg = true;
cc += 2;
}
ccf += 2;
}
break;
case '.':
case 'V':
case 'v':
return false;
case '%': {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Percent, wsSymbol);
int32_t iSysmbolLen = wsSymbol.GetLength();
if (cc + iSysmbolLen <= len &&
!FXSYS_wcsncmp(str + cc, wsSymbol.c_str(), iSysmbolLen)) {
cc += iSysmbolLen;
}
ccf++;
bHavePercentSymbol = true;
} break;
case '8': {
while (ccf < lenf && strf[ccf] == '8') {
ccf++;
}
while (cc < len && FXSYS_isDecimalDigit(str[cc])) {
wsValue += str[cc];
cc++;
}
} break;
case ',': {
if (cc + iGroupLen <= len &&
FXSYS_wcsncmp(str + cc, wsGroupSymbol.c_str(), iGroupLen) == 0) {
cc += iGroupLen;
}
ccf++;
} break;
case '(':
if (str[cc] == L'(') {
bNeg = true;
} else if (str[cc] != L' ') {
return false;
}
cc++;
ccf++;
break;
case ')':
if (str[cc] == L')') {
bNeg = true;
} else if (str[cc] != L' ') {
return false;
}
cc++;
ccf++;
break;
default:
if (strf[ccf] != str[cc]) {
return false;
}
cc++;
ccf++;
}
}
if (cc != len) {
return false;
}
}
if (iExponent || bHavePercentSymbol) {
CFX_Decimal decimal = CFX_Decimal(wsValue.AsStringC());
if (iExponent) {
decimal = decimal * CFX_Decimal(FXSYS_pow(10, (FX_FLOAT)iExponent));
}
if (bHavePercentSymbol) {
decimal = decimal / CFX_Decimal(100);
}
wsValue = decimal;
}
if (bNeg) {
wsValue = L'-' + wsValue;
}
return true;
}
FX_DATETIMETYPE CFX_FormatString::GetDateTimeFormat(
const CFX_WideString& wsPattern,
IFX_Locale*& pLocale,
CFX_WideString& wsDatePattern,
CFX_WideString& wsTimePattern) {
pLocale = nullptr;
CFX_WideString wsTempPattern;
FX_LOCALECATEGORY eCategory = FX_LOCALECATEGORY_Unknown;
int32_t ccf = 0;
int32_t iLenf = wsPattern.GetLength();
const wchar_t* pStr = wsPattern.c_str();
int32_t iFindCategory = 0;
bool bBraceOpen = false;
CFX_WideStringC wsConstChars(gs_wsConstChars);
while (ccf < iLenf) {
if (pStr[ccf] == '\'') {
int32_t iCurChar = ccf;
FX_GetLiteralText(pStr, ccf, iLenf);
wsTempPattern += CFX_WideStringC(pStr + iCurChar, ccf - iCurChar + 1);
} else if (!bBraceOpen && iFindCategory != 3 &&
wsConstChars.Find(pStr[ccf]) == -1) {
CFX_WideString wsCategory(pStr[ccf]);
ccf++;
while (ccf < iLenf && pStr[ccf] != '{' && pStr[ccf] != '.' &&
pStr[ccf] != '(') {
if (pStr[ccf] == 'T') {
wsDatePattern = wsPattern.Left(ccf);
wsTimePattern = wsPattern.Right(wsPattern.GetLength() - ccf);
wsTimePattern.SetAt(0, ' ');
if (!pLocale) {
pLocale = m_pLocaleMgr->GetDefLocale();
}
return FX_DATETIMETYPE_DateTime;
}
wsCategory += pStr[ccf];
ccf++;
}
if (!(iFindCategory & 1) && wsCategory == L"date") {
iFindCategory |= 1;
eCategory = FX_LOCALECATEGORY_Date;
if (iFindCategory & 2) {
iFindCategory = 4;
}
} else if (!(iFindCategory & 2) && wsCategory == L"time") {
iFindCategory |= 2;
eCategory = FX_LOCALECATEGORY_Time;
} else if (wsCategory == L"datetime") {
iFindCategory = 3;
eCategory = FX_LOCALECATEGORY_DateTime;
} else {
continue;
}
while (ccf < iLenf) {
if (pStr[ccf] == '(') {
ccf++;
CFX_WideString wsLCID;
while (ccf < iLenf && pStr[ccf] != ')') {
wsLCID += pStr[ccf++];
}
pLocale = GetPatternLocale(wsLCID);
} else if (pStr[ccf] == '{') {
bBraceOpen = true;
break;
} else if (pStr[ccf] == '.') {
CFX_WideString wsSubCategory;
ccf++;
while (ccf < iLenf && pStr[ccf] != '(' && pStr[ccf] != '{') {
wsSubCategory += pStr[ccf++];
}
uint32_t dwSubHash =
FX_HashCode_GetW(wsSubCategory.AsStringC(), false);
FX_LOCALEDATETIMESUBCATEGORY eSubCategory =
FX_LOCALEDATETIMESUBCATEGORY_Medium;
for (int32_t i = 0; i < g_iFXLocaleDateTimeSubCatCount; i++) {
if (g_FXLocaleDateTimeSubCatData[i].uHash == dwSubHash) {
eSubCategory =
(FX_LOCALEDATETIMESUBCATEGORY)g_FXLocaleDateTimeSubCatData[i]
.eSubCategory;
break;
}
}
if (!pLocale) {
pLocale = m_pLocaleMgr->GetDefLocale();
}
ASSERT(pLocale);
switch (eCategory) {
case FX_LOCALECATEGORY_Date:
pLocale->GetDatePattern(eSubCategory, wsDatePattern);
wsDatePattern = wsTempPattern + wsDatePattern;
break;
case FX_LOCALECATEGORY_Time:
pLocale->GetTimePattern(eSubCategory, wsTimePattern);
wsTimePattern = wsTempPattern + wsTimePattern;
break;
case FX_LOCALECATEGORY_DateTime:
pLocale->GetDatePattern(eSubCategory, wsDatePattern);
wsDatePattern = wsTempPattern + wsDatePattern;
pLocale->GetTimePattern(eSubCategory, wsTimePattern);
break;
default:
break;
}
wsTempPattern.clear();
continue;
}
ccf++;
}
} else if (pStr[ccf] == '}') {
bBraceOpen = false;
if (!wsTempPattern.IsEmpty()) {
if (eCategory == FX_LOCALECATEGORY_Time) {
wsTimePattern = wsTempPattern;
} else if (eCategory == FX_LOCALECATEGORY_Date) {
wsDatePattern = wsTempPattern;
}
wsTempPattern.clear();
}
} else {
wsTempPattern += pStr[ccf];
}
ccf++;
}
if (!wsTempPattern.IsEmpty()) {
if (eCategory == FX_LOCALECATEGORY_Date) {
wsDatePattern += wsTempPattern;
} else {
wsTimePattern += wsTempPattern;
}
}
if (!pLocale) {
pLocale = m_pLocaleMgr->GetDefLocale();
}
if (!iFindCategory) {
wsTimePattern.clear();
wsDatePattern = wsPattern;
}
return (FX_DATETIMETYPE)iFindCategory;
}
static bool FX_ParseLocaleDate(const CFX_WideString& wsDate,
const CFX_WideString& wsDatePattern,
IFX_Locale* pLocale,
CFX_Unitime& datetime,
int32_t& cc) {
int32_t year = 1900;
int32_t month = 1;
int32_t day = 1;
int32_t ccf = 0;
const wchar_t* str = wsDate.c_str();
int32_t len = wsDate.GetLength();
const wchar_t* strf = wsDatePattern.c_str();
int32_t lenf = wsDatePattern.GetLength();
CFX_WideStringC wsDateSymbols(gs_wsDateSymbols);
while (cc < len && ccf < lenf) {
if (strf[ccf] == '\'') {
CFX_WideString wsLiteral = FX_GetLiteralText(strf, ccf, lenf);
int32_t iLiteralLen = wsLiteral.GetLength();
if (cc + iLiteralLen > len ||
FXSYS_wcsncmp(str + cc, wsLiteral.c_str(), iLiteralLen)) {
return false;
}
cc += iLiteralLen;
ccf++;
continue;
} else if (wsDateSymbols.Find(strf[ccf]) == -1) {
if (strf[ccf] != str[cc])
return false;
cc++;
ccf++;
continue;
}
uint32_t dwSymbolNum = 1;
wchar_t dwCharSymbol = strf[ccf++];
while (ccf < lenf && strf[ccf] == dwCharSymbol) {
ccf++;
dwSymbolNum++;
}
uint32_t dwSymbol = (dwCharSymbol << 8) | (dwSymbolNum + '0');
if (dwSymbol == FXBSTR_ID(0, 0, 'D', '1')) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
day = str[cc++] - '0';
if (cc < len && FXSYS_isDecimalDigit(str[cc])) {
day = day * 10 + str[cc++] - '0';
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'D', '2')) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
day = str[cc++] - '0';
if (cc < len) {
day = day * 10 + str[cc++] - '0';
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'J', '1')) {
int i = 0;
while (cc < len && i < 3 && FXSYS_isDecimalDigit(str[cc])) {
cc++;
i++;
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'J', '3')) {
cc += 3;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'M', '1')) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
month = str[cc++] - '0';
if (cc < len && FXSYS_isDecimalDigit(str[cc])) {
month = month * 10 + str[cc++] - '0';
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'M', '2')) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
month = str[cc++] - '0';
if (cc < len) {
month = month * 10 + str[cc++] - '0';
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'M', '3')) {
CFX_WideString wsMonthNameAbbr;
uint16_t i = 0;
for (; i < 12; i++) {
pLocale->GetMonthName(i, wsMonthNameAbbr, true);
if (wsMonthNameAbbr.IsEmpty()) {
continue;
}
if (!FXSYS_wcsncmp(wsMonthNameAbbr.c_str(), str + cc,
wsMonthNameAbbr.GetLength())) {
break;
}
}
if (i < 12) {
cc += wsMonthNameAbbr.GetLength();
month = i + 1;
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'M', '4')) {
CFX_WideString wsMonthName;
uint16_t i = 0;
for (; i < 12; i++) {
pLocale->GetMonthName(i, wsMonthName, false);
if (wsMonthName.IsEmpty()) {
continue;
}
if (!FXSYS_wcsncmp(wsMonthName.c_str(), str + cc,
wsMonthName.GetLength())) {
break;
}
}
if (i < 12) {
cc += wsMonthName.GetLength();
month = i + 1;
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'E', '1')) {
cc += 1;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'E', '3')) {
CFX_WideString wsDayNameAbbr;
uint16_t i = 0;
for (; i < 7; i++) {
pLocale->GetDayName(i, wsDayNameAbbr, true);
if (wsDayNameAbbr.IsEmpty()) {
continue;
}
if (!FXSYS_wcsncmp(wsDayNameAbbr.c_str(), str + cc,
wsDayNameAbbr.GetLength())) {
break;
}
}
if (i < 12) {
cc += wsDayNameAbbr.GetLength();
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'E', '4')) {
CFX_WideString wsDayName;
int32_t i = 0;
for (; i < 7; i++) {
pLocale->GetDayName(i, wsDayName, false);
if (wsDayName == L"") {
continue;
}
if (!FXSYS_wcsncmp(wsDayName.c_str(), str + cc,
wsDayName.GetLength())) {
break;
}
}
if (i < 12) {
cc += wsDayName.GetLength();
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'e', '1')) {
cc += 1;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'G', '1')) {
cc += 2;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'Y', '2')) {
if (cc + 2 > len) {
return false;
}
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
year = str[cc++] - '0';
if (cc >= len || !FXSYS_isDecimalDigit(str[cc])) {
return false;
}
year = year * 10 + str[cc++] - '0';
if (year <= 29) {
year += 2000;
} else {
year += 1900;
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'Y', '4')) {
int i = 0;
year = 0;
if (cc + 4 > len) {
return false;
}
while (i < 4) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
year = year * 10 + str[cc] - '0';
cc++;
i++;
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'w', '1')) {
cc += 1;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'W', '2')) {
cc += 2;
}
}
if (cc < len) {
return false;
}
CFX_Unitime ut;
ut.Set(year, month, day);
datetime = datetime + ut;
return !!cc;
}
static void FX_ResolveZone(uint8_t& wHour,
uint8_t& wMinute,
FX_TIMEZONE tzDiff,
IFX_Locale* pLocale) {
int32_t iMinuteDiff = wHour * 60 + wMinute;
FX_TIMEZONE tzLocale;
pLocale->GetTimeZone(&tzLocale);
iMinuteDiff += tzLocale.tzHour * 60 +
(tzLocale.tzHour < 0 ? -tzLocale.tzMinute : tzLocale.tzMinute);
iMinuteDiff -= tzDiff.tzHour * 60 +
(tzDiff.tzHour < 0 ? -tzDiff.tzMinute : tzDiff.tzMinute);
while (iMinuteDiff > 1440) {
iMinuteDiff -= 1440;
}
while (iMinuteDiff < 0) {
iMinuteDiff += 1440;
}
wHour = iMinuteDiff / 60;
wMinute = iMinuteDiff % 60;
}
static bool FX_ParseLocaleTime(const CFX_WideString& wsTime,
const CFX_WideString& wsTimePattern,
IFX_Locale* pLocale,
CFX_Unitime& datetime,
int32_t& cc) {
uint8_t hour = 0;
uint8_t minute = 0;
uint8_t second = 0;
uint16_t millisecond = 0;
int32_t ccf = 0;
const wchar_t* str = wsTime.c_str();
int len = wsTime.GetLength();
const wchar_t* strf = wsTimePattern.c_str();
int lenf = wsTimePattern.GetLength();
bool bHasA = false;
bool bPM = false;
CFX_WideStringC wsTimeSymbols(gs_wsTimeSymbols);
while (cc < len && ccf < lenf) {
if (strf[ccf] == '\'') {
CFX_WideString wsLiteral = FX_GetLiteralText(strf, ccf, lenf);
int32_t iLiteralLen = wsLiteral.GetLength();
if (cc + iLiteralLen > len ||
FXSYS_wcsncmp(str + cc, wsLiteral.c_str(), iLiteralLen)) {
return false;
}
cc += iLiteralLen;
ccf++;
continue;
} else if (wsTimeSymbols.Find(strf[ccf]) == -1) {
if (strf[ccf] != str[cc])
return false;
cc++;
ccf++;
continue;
}
uint32_t dwSymbolNum = 1;
wchar_t dwCharSymbol = strf[ccf++];
while (ccf < lenf && strf[ccf] == dwCharSymbol) {
ccf++;
dwSymbolNum++;
}
uint32_t dwSymbol = (dwCharSymbol << 8) | (dwSymbolNum + '0');
if (dwSymbol == FXBSTR_ID(0, 0, 'k', '1') ||
dwSymbol == FXBSTR_ID(0, 0, 'H', '1') ||
dwSymbol == FXBSTR_ID(0, 0, 'h', '1') ||
dwSymbol == FXBSTR_ID(0, 0, 'K', '1')) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
hour = str[cc++] - '0';
if (cc < len && FXSYS_isDecimalDigit(str[cc])) {
hour = hour * 10 + str[cc++] - '0';
}
if (dwSymbol == FXBSTR_ID(0, 0, 'K', '1') && hour == 24) {
hour = 0;
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'k', '2') ||
dwSymbol == FXBSTR_ID(0, 0, 'H', '2') ||
dwSymbol == FXBSTR_ID(0, 0, 'h', '2') ||
dwSymbol == FXBSTR_ID(0, 0, 'K', '2')) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
hour = str[cc++] - '0';
if (cc >= len) {
return false;
}
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
hour = hour * 10 + str[cc++] - '0';
if (dwSymbol == FXBSTR_ID(0, 0, 'K', '2') && hour == 24) {
hour = 0;
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'M', '1')) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
minute = str[cc++] - '0';
if (cc < len && FXSYS_isDecimalDigit(str[cc])) {
minute = minute * 10 + str[cc++] - '0';
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'M', '2')) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
minute = str[cc++] - '0';
if (cc >= len) {
return false;
}
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
minute = minute * 10 + str[cc++] - '0';
} else if (dwSymbol == FXBSTR_ID(0, 0, 'S', '1')) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
second = str[cc++] - '0';
if (cc < len && FXSYS_isDecimalDigit(str[cc])) {
second = second * 10 + str[cc++] - '0';
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'S', '2')) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
second = str[cc++] - '0';
if (cc >= len) {
return false;
}
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
second = second * 10 + str[cc++] - '0';
} else if (dwSymbol == FXBSTR_ID(0, 0, 'F', '3')) {
if (cc + 3 >= len) {
return false;
}
int i = 0;
while (i < 3) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
millisecond = millisecond * 10 + str[cc++] - '0';
i++;
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'A', '1')) {
CFX_WideString wsAM;
pLocale->GetMeridiemName(wsAM, true);
CFX_WideString wsPM;
pLocale->GetMeridiemName(wsPM, false);
if ((cc + wsAM.GetLength() <= len) &&
(CFX_WideStringC(str + cc, wsAM.GetLength()) == wsAM)) {
cc += wsAM.GetLength();
bHasA = true;
} else if ((cc + wsPM.GetLength() <= len) &&
(CFX_WideStringC(str + cc, wsPM.GetLength()) == wsPM)) {
cc += wsPM.GetLength();
bHasA = true;
bPM = true;
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'Z', '1')) {
if (cc + 3 > len) {
continue;
}
uint32_t dwHash = str[cc++];
dwHash = (dwHash << 8) | str[cc++];
dwHash = (dwHash << 8) | str[cc++];
if (dwHash == FXBSTR_ID(0, 'G', 'M', 'T')) {
FX_TIMEZONE tzDiff;
tzDiff.tzHour = 0;
tzDiff.tzMinute = 0;
if (cc < len && (str[cc] == '-' || str[cc] == '+')) {
cc += FX_ParseTimeZone(str + cc, len - cc, tzDiff);
}
FX_ResolveZone(hour, minute, tzDiff, pLocale);
} else {
const FX_LOCALETIMEZONEINFO* pEnd =
g_FXLocaleTimeZoneData + FX_ArraySize(g_FXLocaleTimeZoneData);
const FX_LOCALETIMEZONEINFO* pTimeZoneInfo =
std::lower_bound(g_FXLocaleTimeZoneData, pEnd, dwHash,
[](const FX_LOCALETIMEZONEINFO& info,
uint32_t hash) { return info.uHash < hash; });
if (pTimeZoneInfo < pEnd && dwHash == pTimeZoneInfo->uHash) {
hour += pTimeZoneInfo->iHour;
minute += pTimeZoneInfo->iHour > 0 ? pTimeZoneInfo->iMinute
: -pTimeZoneInfo->iMinute;
}
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'z', '1')) {
if (str[cc] != 'Z') {
FX_TIMEZONE tzDiff;
cc += FX_ParseTimeZone(str + cc, len - cc, tzDiff);
FX_ResolveZone(hour, minute, tzDiff, pLocale);
} else {
cc++;
}
}
}
if (bHasA) {
if (bPM) {
hour += 12;
if (hour == 24) {
hour = 12;
}
} else {
if (hour == 12) {
hour = 0;
}
}
}
CFX_Unitime ut;
ut.Set(0, 0, 0, hour, minute, second, millisecond);
datetime = datetime + ut;
return !!cc;
}
bool CFX_FormatString::ParseDateTime(const CFX_WideString& wsSrcDateTime,
const CFX_WideString& wsPattern,
FX_DATETIMETYPE eDateTimeType,
CFX_Unitime& dtValue) {
dtValue.Set(0);
if (wsSrcDateTime.IsEmpty() || wsPattern.IsEmpty()) {
return false;
}
CFX_WideString wsDatePattern, wsTimePattern;
IFX_Locale* pLocale = nullptr;
FX_DATETIMETYPE eCategory =
GetDateTimeFormat(wsPattern, pLocale, wsDatePattern, wsTimePattern);
if (!pLocale) {
return false;
}
if (eCategory == FX_DATETIMETYPE_Unknown) {
eCategory = eDateTimeType;
}
if (eCategory == FX_DATETIMETYPE_Unknown) {
return false;
}
if (eCategory == FX_DATETIMETYPE_TimeDate) {
int32_t iStart = 0;
if (!FX_ParseLocaleTime(wsSrcDateTime, wsTimePattern, pLocale, dtValue,
iStart)) {
return false;
}
if (!FX_ParseLocaleDate(wsSrcDateTime, wsDatePattern, pLocale, dtValue,
iStart)) {
return false;
}
} else {
int32_t iStart = 0;
if ((eCategory & FX_DATETIMETYPE_Date) &&
!FX_ParseLocaleDate(wsSrcDateTime, wsDatePattern, pLocale, dtValue,
iStart)) {
return false;
}
if ((eCategory & FX_DATETIMETYPE_Time) &&
!FX_ParseLocaleTime(wsSrcDateTime, wsTimePattern, pLocale, dtValue,
iStart)) {
return false;
}
}
return true;
}
bool CFX_FormatString::ParseZero(const CFX_WideString& wsSrcText,
const CFX_WideString& wsPattern) {
CFX_WideString wsTextFormat;
GetTextFormat(wsPattern, L"zero", wsTextFormat);
int32_t iText = 0, iPattern = 0;
const wchar_t* pStrText = wsSrcText.c_str();
int32_t iLenText = wsSrcText.GetLength();
const wchar_t* pStrPattern = wsTextFormat.c_str();
int32_t iLenPattern = wsTextFormat.GetLength();
while (iPattern < iLenPattern && iText < iLenText) {
if (pStrPattern[iPattern] == '\'') {
CFX_WideString wsLiteral =
FX_GetLiteralText(pStrPattern, iPattern, iLenPattern);
int32_t iLiteralLen = wsLiteral.GetLength();
if (iText + iLiteralLen > iLenText ||
FXSYS_wcsncmp(pStrText + iText, wsLiteral.c_str(), iLiteralLen)) {
return false;
}
iText += iLiteralLen;
iPattern++;
continue;
} else if (pStrPattern[iPattern] != pStrText[iText]) {
return false;
} else {
iText++;
iPattern++;
}
}
return iPattern == iLenPattern && iText == iLenText;
}
bool CFX_FormatString::ParseNull(const CFX_WideString& wsSrcText,
const CFX_WideString& wsPattern) {
CFX_WideString wsTextFormat;
GetTextFormat(wsPattern, L"null", wsTextFormat);
int32_t iText = 0, iPattern = 0;
const wchar_t* pStrText = wsSrcText.c_str();
int32_t iLenText = wsSrcText.GetLength();
const wchar_t* pStrPattern = wsTextFormat.c_str();
int32_t iLenPattern = wsTextFormat.GetLength();
while (iPattern < iLenPattern && iText < iLenText) {
if (pStrPattern[iPattern] == '\'') {
CFX_WideString wsLiteral =
FX_GetLiteralText(pStrPattern, iPattern, iLenPattern);
int32_t iLiteralLen = wsLiteral.GetLength();
if (iText + iLiteralLen > iLenText ||
FXSYS_wcsncmp(pStrText + iText, wsLiteral.c_str(), iLiteralLen)) {
return false;
}
iText += iLiteralLen;
iPattern++;
continue;
} else if (pStrPattern[iPattern] != pStrText[iText]) {
return false;
} else {
iText++;
iPattern++;
}
}
return iPattern == iLenPattern && iText == iLenText;
}
bool CFX_FormatString::FormatText(const CFX_WideString& wsSrcText,
const CFX_WideString& wsPattern,
CFX_WideString& wsOutput) {
if (wsPattern.IsEmpty()) {
return false;
}
int32_t iLenText = wsSrcText.GetLength();
if (iLenText == 0) {
return false;
}
CFX_WideString wsTextFormat;
GetTextFormat(wsPattern, L"text", wsTextFormat);
int32_t iText = 0, iPattern = 0;
const wchar_t* pStrText = wsSrcText.c_str();
const wchar_t* pStrPattern = wsTextFormat.c_str();
int32_t iLenPattern = wsTextFormat.GetLength();
while (iPattern < iLenPattern) {
switch (pStrPattern[iPattern]) {
case '\'': {
wsOutput += FX_GetLiteralText(pStrPattern, iPattern, iLenPattern);
iPattern++;
break;
}
case 'A':
if (iText >= iLenText || !FXSYS_iswalpha(pStrText[iText])) {
return false;
}
wsOutput += pStrText[iText++];
iPattern++;
break;
case 'X':
if (iText >= iLenText) {
return false;
}
wsOutput += pStrText[iText++];
iPattern++;
break;
case 'O':
case '0':
if (iText >= iLenText || (!FXSYS_isDecimalDigit(pStrText[iText]) &&
!FXSYS_iswalpha(pStrText[iText]))) {
return false;
}
wsOutput += pStrText[iText++];
iPattern++;
break;
case '9':
if (iText >= iLenText || !FXSYS_isDecimalDigit(pStrText[iText])) {
return false;
}
wsOutput += pStrText[iText++];
iPattern++;
break;
default:
wsOutput += pStrPattern[iPattern++];
break;
}
}
return iText == iLenText;
}
static int32_t FX_GetNumTrailingLimit(const CFX_WideString& wsFormat,
int iDotPos,
bool& bTrimTailZeros) {
if (iDotPos < 0) {
return 0;
}
int32_t iCount = wsFormat.GetLength();
int32_t iTreading = 0;
for (iDotPos++; iDotPos < iCount; iDotPos++) {
wchar_t wc = wsFormat[iDotPos];
if (wc == L'z' || wc == L'9' || wc == 'Z') {
iTreading++;
bTrimTailZeros = (wc == L'9' ? false : true);
}
}
return iTreading;
}
bool CFX_FormatString::FormatStrNum(const CFX_WideStringC& wsInputNum,
const CFX_WideString& wsPattern,
CFX_WideString& wsOutput) {
if (wsInputNum.IsEmpty() || wsPattern.IsEmpty()) {
return false;
}
int32_t dot_index_f = -1;
uint32_t dwNumStyle = 0;
CFX_WideString wsNumFormat;
IFX_Locale* pLocale =
GetNumericFormat(wsPattern, dot_index_f, dwNumStyle, wsNumFormat);
if (!pLocale || wsNumFormat.IsEmpty()) {
return false;
}
int32_t cc = 0, ccf = 0;
const wchar_t* strf = wsNumFormat.c_str();
int lenf = wsNumFormat.GetLength();
CFX_WideString wsSrcNum(wsInputNum);
wsSrcNum.TrimLeft('0');
if (wsSrcNum.IsEmpty() || wsSrcNum[0] == '.') {
wsSrcNum.Insert(0, '0');
}
CFX_Decimal decimal = CFX_Decimal(wsSrcNum.AsStringC());
if (dwNumStyle & FX_NUMSTYLE_Percent) {
decimal = decimal * CFX_Decimal(100);
wsSrcNum = decimal;
}
int32_t exponent = 0;
if (dwNumStyle & FX_NUMSTYLE_Exponent) {
int fixed_count = 0;
while (ccf < dot_index_f) {
switch (strf[ccf]) {
case '\'':
FX_GetLiteralText(strf, ccf, dot_index_f);
break;
case '9':
case 'z':
case 'Z':
fixed_count++;
break;
}
ccf++;
}
int threshold = 1;
while (fixed_count > 1) {
threshold *= 10;
fixed_count--;
}
if (decimal != CFX_Decimal(0)) {
if (decimal < CFX_Decimal(threshold)) {
decimal = decimal * CFX_Decimal(10);
exponent = -1;
while (decimal < CFX_Decimal(threshold)) {
decimal = decimal * CFX_Decimal(10);
exponent -= 1;
}
} else if (decimal > CFX_Decimal(threshold)) {
threshold *= 10;
while (decimal > CFX_Decimal(threshold)) {
decimal = decimal / CFX_Decimal(10);
exponent += 1;
}
}
}
}
bool bTrimTailZeros = false;
int32_t iTreading =
FX_GetNumTrailingLimit(wsNumFormat, dot_index_f, bTrimTailZeros);
int32_t scale = decimal.GetScale();
if (iTreading < scale) {
decimal.SetScale(iTreading);
wsSrcNum = decimal;
}
if (bTrimTailZeros && scale > 0 && iTreading > 0) {
wsSrcNum.TrimRight(L"0");
wsSrcNum.TrimRight(L".");
}
CFX_WideString wsGroupSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Grouping, wsGroupSymbol);
bool bNeg = false;
if (wsSrcNum[0] == '-') {
bNeg = true;
wsSrcNum.Delete(0, 1);
}
bool bAddNeg = false;
const wchar_t* str = wsSrcNum.c_str();
int len = wsSrcNum.GetLength();
int dot_index = wsSrcNum.Find('.');
if (dot_index == -1) {
dot_index = len;
}
ccf = dot_index_f - 1;
cc = dot_index - 1;
while (ccf >= 0) {
switch (strf[ccf]) {
case '9':
if (cc >= 0) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
wsOutput = str[cc] + wsOutput;
cc--;
} else {
wsOutput = L'0' + wsOutput;
}
ccf--;
break;
case 'z':
if (cc >= 0) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
if (str[0] != '0') {
wsOutput = str[cc] + wsOutput;
}
cc--;
}
ccf--;
break;
case 'Z':
if (cc >= 0) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
if (str[0] == '0') {
wsOutput = L' ' + wsOutput;
} else {
wsOutput = str[cc] + wsOutput;
}
cc--;
} else {
wsOutput = L' ' + wsOutput;
}
ccf--;
break;
case 'S':
if (bNeg) {
CFX_WideString wsMinusSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Minus, wsMinusSymbol);
wsOutput = wsMinusSymbol + wsOutput;
bAddNeg = true;
} else {
wsOutput = L' ' + wsOutput;
}
ccf--;
break;
case 's':
if (bNeg) {
CFX_WideString wsMinusSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Minus, wsMinusSymbol);
wsOutput = wsMinusSymbol + wsOutput;
bAddNeg = true;
}
ccf--;
break;
case 'E': {
CFX_WideString wsExp;
wsExp.Format(L"E%+d", exponent);
wsOutput = wsExp + wsOutput;
}
ccf--;
break;
case '$': {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_CurrencySymbol, wsSymbol);
wsOutput = wsSymbol + wsOutput;
}
ccf--;
break;
case 'r':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'c') {
if (bNeg) {
wsOutput = L"CR" + wsOutput;
}
ccf -= 2;
bAddNeg = true;
}
break;
case 'R':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'C') {
if (bNeg) {
wsOutput = L"CR" + wsOutput;
} else {
wsOutput = L" " + wsOutput;
}
ccf -= 2;
bAddNeg = true;
}
break;
case 'b':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'd') {
if (bNeg) {
wsOutput = L"db" + wsOutput;
}
ccf -= 2;
bAddNeg = true;
}
break;
case 'B':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'D') {
if (bNeg) {
wsOutput = L"DB" + wsOutput;
} else {
wsOutput = L" " + wsOutput;
}
ccf -= 2;
bAddNeg = true;
}
break;
case '%': {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Percent, wsSymbol);
wsOutput = wsSymbol + wsOutput;
}
ccf--;
break;
case ',':
if (cc >= 0) {
wsOutput = wsGroupSymbol + wsOutput;
}
ccf--;
break;
case '(':
if (bNeg) {
wsOutput = L"(" + wsOutput;
} else {
wsOutput = L" " + wsOutput;
}
bAddNeg = true;
ccf--;
break;
case ')':
if (bNeg) {
wsOutput = L")" + wsOutput;
} else {
wsOutput = L" " + wsOutput;
}
ccf--;
break;
case '\'':
wsOutput = FX_GetLiteralTextReverse(strf, ccf) + wsOutput;
ccf--;
break;
default:
wsOutput = strf[ccf] + wsOutput;
ccf--;
}
}
if (cc >= 0) {
int nPos = dot_index % 3;
wsOutput.clear();
for (int32_t i = 0; i < dot_index; i++) {
if (i % 3 == nPos && i != 0) {
wsOutput += wsGroupSymbol;
}
wsOutput += wsSrcNum[i];
}
if (dot_index < len) {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Decimal, wsSymbol);
wsOutput += wsSymbol;
wsOutput += wsSrcNum.Right(len - dot_index - 1);
}
if (bNeg) {
CFX_WideString wsMinusymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Minus, wsMinusymbol);
wsOutput = wsMinusymbol + wsOutput;
}
return false;
}
if (dot_index_f == wsNumFormat.GetLength()) {
if (!bAddNeg && bNeg) {
CFX_WideString wsMinusymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Minus, wsMinusymbol);
wsOutput = wsMinusymbol + wsOutput;
}
return true;
}
CFX_WideString wsDotSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Decimal, wsDotSymbol);
if (strf[dot_index_f] == 'V') {
wsOutput += wsDotSymbol;
} else if (strf[dot_index_f] == '.') {
if (dot_index < len) {
wsOutput += wsDotSymbol;
} else {
if (strf[dot_index_f + 1] == '9' || strf[dot_index_f + 1] == 'Z') {
wsOutput += wsDotSymbol;
}
}
}
ccf = dot_index_f + 1;
cc = dot_index + 1;
while (ccf < lenf) {
switch (strf[ccf]) {
case '\'':
wsOutput += FX_GetLiteralText(strf, ccf, lenf);
ccf++;
break;
case '9':
if (cc < len) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
wsOutput += str[cc];
cc++;
} else {
wsOutput += L'0';
}
ccf++;
break;
case 'z':
if (cc < len) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
wsOutput += str[cc];
cc++;
}
ccf++;
break;
case 'Z':
if (cc < len) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
wsOutput += str[cc];
cc++;
} else {
wsOutput += L'0';
}
ccf++;
break;
case 'E': {
CFX_WideString wsExp;
wsExp.Format(L"E%+d", exponent);
wsOutput += wsExp;
}
ccf++;
break;
case '$': {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_CurrencySymbol, wsSymbol);
wsOutput += wsSymbol;
}
ccf++;
break;
case 'c':
if (ccf + 1 < lenf && strf[ccf + 1] == 'r') {
if (bNeg) {
wsOutput += L"CR";
}
ccf += 2;
bAddNeg = true;
}
break;
case 'C':
if (ccf + 1 < lenf && strf[ccf + 1] == 'R') {
if (bNeg) {
wsOutput += L"CR";
} else {
wsOutput += L" ";
}
ccf += 2;
bAddNeg = true;
}
break;
case 'd':
if (ccf + 1 < lenf && strf[ccf + 1] == 'b') {
if (bNeg) {
wsOutput += L"db";
}
ccf += 2;
bAddNeg = true;
}
break;
case 'D':
if (ccf + 1 < lenf && strf[ccf + 1] == 'B') {
if (bNeg) {
wsOutput += L"DB";
} else {
wsOutput += L" ";
}
ccf += 2;
bAddNeg = true;
}
break;
case '%': {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Percent, wsSymbol);
wsOutput += wsSymbol;
}
ccf++;
break;
case '8': {
while (ccf < lenf && strf[ccf] == '8') {
ccf++;
}
while (cc < len && FXSYS_isDecimalDigit(str[cc])) {
wsOutput += str[cc];
cc++;
}
} break;
case ',':
wsOutput += wsGroupSymbol;
ccf++;
break;
case '(':
if (bNeg) {
wsOutput += '(';
} else {
wsOutput += ' ';
}
bAddNeg = true;
ccf++;
break;
case ')':
if (bNeg) {
wsOutput += ')';
} else {
wsOutput += ' ';
}
ccf++;
break;
default:
ccf++;
}
}
if (!bAddNeg && bNeg) {
CFX_WideString wsMinusymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Minus, wsMinusymbol);
wsOutput =
wsMinusymbol + wsOutput[0] + wsOutput.Mid(1, wsOutput.GetLength() - 1);
}
return true;
}
bool CFX_FormatString::FormatLCNumeric(CFX_LCNumeric& lcNum,
const CFX_WideString& wsPattern,
CFX_WideString& wsOutput) {
int32_t dot_index_f = -1;
uint32_t dwNumStyle = 0;
CFX_WideString wsNumFormat;
IFX_Locale* pLocale =
GetNumericFormat(wsPattern, dot_index_f, dwNumStyle, wsNumFormat);
if (!pLocale || wsNumFormat.IsEmpty()) {
return false;
}
int32_t cc = 0, ccf = 0;
const wchar_t* strf = wsNumFormat.c_str();
int lenf = wsNumFormat.GetLength();
double dbOrgRaw = lcNum.GetDouble();
double dbRetValue = dbOrgRaw;
if (dwNumStyle & FX_NUMSTYLE_Percent) {
dbRetValue *= 100;
}
int32_t exponent = 0;
if (dwNumStyle & FX_NUMSTYLE_Exponent) {
int fixed_count = 0;
while (ccf < dot_index_f) {
switch (strf[ccf]) {
case '\'':
FX_GetLiteralText(strf, ccf, dot_index_f);
break;
case '9':
case 'z':
case 'Z':
fixed_count++;
break;
}
ccf++;
}
int threshold = 1;
while (fixed_count > 1) {
threshold *= 10;
fixed_count--;
}
if (dbRetValue != 0) {
if (dbRetValue < threshold) {
dbRetValue *= 10;
exponent = -1;
while (dbRetValue < threshold) {
dbRetValue *= 10;
exponent -= 1;
}
} else if (dbRetValue > threshold) {
threshold *= 10;
while (dbRetValue > threshold) {
dbRetValue /= 10;
exponent += 1;
}
}
}
}
if (dwNumStyle & (FX_NUMSTYLE_Percent | FX_NUMSTYLE_Exponent)) {
lcNum = CFX_LCNumeric(dbRetValue);
}
bool bTrimTailZeros = false;
int32_t iTreading =
FX_GetNumTrailingLimit(wsNumFormat, dot_index_f, bTrimTailZeros);
CFX_WideString wsNumeric = lcNum.ToString(iTreading, bTrimTailZeros);
if (wsNumeric.IsEmpty()) {
return false;
}
CFX_WideString wsGroupSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Grouping, wsGroupSymbol);
bool bNeg = false;
if (wsNumeric[0] == '-') {
bNeg = true;
wsNumeric.Delete(0, 1);
}
bool bAddNeg = false;
const wchar_t* str = wsNumeric.c_str();
int len = wsNumeric.GetLength();
int dot_index = wsNumeric.Find('.');
if (dot_index == -1) {
dot_index = len;
}
ccf = dot_index_f - 1;
cc = dot_index - 1;
while (ccf >= 0) {
switch (strf[ccf]) {
case '9':
if (cc >= 0) {
wsOutput = str[cc] + wsOutput;
cc--;
} else {
wsOutput = L'0' + wsOutput;
}
ccf--;
break;
case 'z':
if (cc >= 0) {
if (lcNum.m_Integral != 0) {
wsOutput = str[cc] + wsOutput;
}
cc--;
}
ccf--;
break;
case 'Z':
if (cc >= 0) {
if (lcNum.m_Integral == 0) {
wsOutput = L' ' + wsOutput;
} else {
wsOutput = str[cc] + wsOutput;
}
cc--;
} else {
wsOutput = L' ' + wsOutput;
}
ccf--;
break;
case 'S':
if (bNeg) {
CFX_WideString wsMinusSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Minus, wsMinusSymbol);
wsOutput = wsMinusSymbol + wsOutput;
bAddNeg = true;
} else {
wsOutput = L' ' + wsOutput;
}
ccf--;
break;
case 's':
if (bNeg) {
CFX_WideString wsMinusSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Minus, wsMinusSymbol);
wsOutput = wsMinusSymbol + wsOutput;
bAddNeg = true;
}
ccf--;
break;
case 'E': {
CFX_WideString wsExp;
wsExp.Format(L"E%+d", exponent);
wsOutput = wsExp + wsOutput;
}
ccf--;
break;
case '$': {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_CurrencySymbol, wsSymbol);
wsOutput = wsSymbol + wsOutput;
}
ccf--;
break;
case 'r':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'c') {
if (bNeg) {
wsOutput = L"CR" + wsOutput;
}
ccf -= 2;
bAddNeg = true;
}
break;
case 'R':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'C') {
if (bNeg) {
wsOutput = L"CR" + wsOutput;
} else {
wsOutput = L" " + wsOutput;
}
ccf -= 2;
bAddNeg = true;
}
break;
case 'b':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'd') {
if (bNeg) {
wsOutput = L"db" + wsOutput;
}
ccf -= 2;
bAddNeg = true;
}
break;
case 'B':
if (ccf - 1 >= 0 && strf[ccf - 1] == 'D') {
if (bNeg) {
wsOutput = L"DB" + wsOutput;
} else {
wsOutput = L" " + wsOutput;
}
ccf -= 2;
bAddNeg = true;
}
break;
case '%': {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Percent, wsSymbol);
wsOutput = wsSymbol + wsOutput;
}
ccf--;
break;
case ',':
if (cc >= 0) {
wsOutput = wsGroupSymbol + wsOutput;
}
ccf--;
break;
case '(':
if (bNeg) {
wsOutput = L"(" + wsOutput;
} else {
wsOutput = L" " + wsOutput;
}
bAddNeg = true;
ccf--;
break;
case ')':
if (bNeg) {
wsOutput = L")" + wsOutput;
} else {
wsOutput = L" " + wsOutput;
}
ccf--;
break;
case '\'':
wsOutput = FX_GetLiteralTextReverse(strf, ccf) + wsOutput;
ccf--;
break;
default:
wsOutput = strf[ccf] + wsOutput;
ccf--;
}
}
if (cc >= 0) {
int nPos = dot_index % 3;
wsOutput.clear();
for (int32_t i = 0; i < dot_index; i++) {
if (i % 3 == nPos && i != 0) {
wsOutput += wsGroupSymbol;
}
wsOutput += wsNumeric[i];
}
if (dot_index < len) {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Decimal, wsSymbol);
wsOutput += wsSymbol;
wsOutput += wsNumeric.Right(len - dot_index - 1);
}
if (bNeg) {
CFX_WideString wsMinusymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Minus, wsMinusymbol);
wsOutput = wsMinusymbol + wsOutput;
}
return false;
}
if (dot_index_f == wsNumFormat.GetLength()) {
if (!bAddNeg && bNeg) {
CFX_WideString wsMinusymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Minus, wsMinusymbol);
wsOutput = wsMinusymbol + wsOutput;
}
return true;
}
CFX_WideString wsDotSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Decimal, wsDotSymbol);
if (strf[dot_index_f] == 'V') {
wsOutput += wsDotSymbol;
} else if (strf[dot_index_f] == '.') {
if (dot_index < len) {
wsOutput += wsDotSymbol;
} else {
if (strf[dot_index_f + 1] == '9' || strf[dot_index_f + 1] == 'Z') {
wsOutput += wsDotSymbol;
}
}
}
ccf = dot_index_f + 1;
cc = dot_index + 1;
while (ccf < lenf) {
switch (strf[ccf]) {
case '\'':
wsOutput += FX_GetLiteralText(strf, ccf, lenf);
ccf++;
break;
case '9':
if (cc < len) {
wsOutput += str[cc];
cc++;
} else {
wsOutput += L'0';
}
ccf++;
break;
case 'z':
if (cc < len) {
wsOutput += str[cc];
cc++;
}
ccf++;
break;
case 'Z':
if (cc < len) {
wsOutput += str[cc];
cc++;
} else {
wsOutput += L'0';
}
ccf++;
break;
case 'E': {
CFX_WideString wsExp;
wsExp.Format(L"E%+d", exponent);
wsOutput += wsExp;
}
ccf++;
break;
case '$': {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_CurrencySymbol, wsSymbol);
wsOutput += wsSymbol;
}
ccf++;
break;
case 'c':
if (ccf + 1 < lenf && strf[ccf + 1] == 'r') {
if (bNeg) {
wsOutput += L"CR";
}
ccf += 2;
bAddNeg = true;
}
break;
case 'C':
if (ccf + 1 < lenf && strf[ccf + 1] == 'R') {
if (bNeg) {
wsOutput += L"CR";
} else {
wsOutput += L" ";
}
ccf += 2;
bAddNeg = true;
}
break;
case 'd':
if (ccf + 1 < lenf && strf[ccf + 1] == 'b') {
if (bNeg) {
wsOutput += L"db";
}
ccf += 2;
bAddNeg = true;
}
break;
case 'D':
if (ccf + 1 < lenf && strf[ccf + 1] == 'B') {
if (bNeg) {
wsOutput += L"DB";
} else {
wsOutput += L" ";
}
ccf += 2;
bAddNeg = true;
}
break;
case '%': {
CFX_WideString wsSymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Percent, wsSymbol);
wsOutput += wsSymbol;
}
ccf++;
break;
case '8': {
while (ccf < lenf && strf[ccf] == '8') {
ccf++;
}
while (cc < len && FXSYS_isDecimalDigit(str[cc])) {
wsOutput += str[cc];
cc++;
}
} break;
case ',':
wsOutput += wsGroupSymbol;
ccf++;
break;
case '(':
if (bNeg) {
wsOutput += '(';
} else {
wsOutput += ' ';
}
bAddNeg = true;
ccf++;
break;
case ')':
if (bNeg) {
wsOutput += ')';
} else {
wsOutput += ' ';
}
ccf++;
break;
default:
ccf++;
}
}
if (!bAddNeg && bNeg) {
CFX_WideString wsMinusymbol;
pLocale->GetNumbericSymbol(FX_LOCALENUMSYMBOL_Minus, wsMinusymbol);
wsOutput =
wsOutput[0] + wsMinusymbol + wsOutput.Mid(1, wsOutput.GetLength() - 1);
}
return true;
}
bool CFX_FormatString::FormatNum(const CFX_WideString& wsSrcNum,
const CFX_WideString& wsPattern,
CFX_WideString& wsOutput) {
if (wsSrcNum.IsEmpty() || wsPattern.IsEmpty()) {
return false;
}
return FormatStrNum(wsSrcNum.AsStringC(), wsPattern, wsOutput);
}
bool CFX_FormatString::FormatNum(FX_FLOAT fNum,
const CFX_WideString& wsPattern,
CFX_WideString& wsOutput) {
if (wsPattern.IsEmpty()) {
return false;
}
CFX_LCNumeric lcNum(fNum);
return FormatLCNumeric(lcNum, wsPattern, wsOutput);
}
bool FX_DateFromCanonical(const CFX_WideString& wsDate, CFX_Unitime& datetime) {
int32_t year = 1900;
int32_t month = 1;
int32_t day = 1;
uint16_t wYear = 0;
int cc_start = 0, cc = 0;
const wchar_t* str = wsDate.c_str();
int len = wsDate.GetLength();
if (len > 10) {
return false;
}
while (cc < len && cc < 4) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
wYear = wYear * 10 + str[cc++] - '0';
}
year = wYear;
if (cc < 4 || wYear < 1900) {
return false;
}
if (cc < len) {
if (str[cc] == '-') {
cc++;
}
cc_start = cc;
uint8_t tmpM = 0;
while (cc < len && cc < cc_start + 2) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
tmpM = tmpM * 10 + str[cc++] - '0';
}
month = tmpM;
if (cc == cc_start + 1 || tmpM > 12 || tmpM < 1) {
return false;
}
if (cc < len) {
if (str[cc] == '-') {
cc++;
}
uint8_t tmpD = 0;
cc_start = cc;
while (cc < len && cc < cc_start + 2) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
tmpD = tmpD * 10 + str[cc++] - '0';
}
day = tmpD;
if (tmpD < 1) {
return false;
}
if ((tmpM == 1 || tmpM == 3 || tmpM == 5 || tmpM == 7 || tmpM == 8 ||
tmpM == 10 || tmpM == 12) &&
tmpD > 31) {
return false;
}
if ((tmpM == 4 || tmpM == 6 || tmpM == 9 || tmpM == 11) && tmpD > 30) {
return false;
}
bool iLeapYear;
if ((wYear % 4 == 0 && wYear % 100 != 0) || wYear % 400 == 0) {
iLeapYear = true;
} else {
iLeapYear = false;
}
if ((iLeapYear && tmpM == 2 && tmpD > 29) ||
(!iLeapYear && tmpM == 2 && tmpD > 28)) {
return false;
}
}
}
CFX_Unitime ut;
ut.Set(year, month, day);
datetime = datetime + ut;
return true;
}
bool FX_TimeFromCanonical(const CFX_WideStringC& wsTime,
CFX_Unitime& datetime,
IFX_Locale* pLocale) {
if (wsTime.GetLength() == 0) {
return false;
}
uint8_t hour = 0;
uint8_t minute = 0;
uint8_t second = 0;
uint16_t millisecond = 0;
int cc_start = 0, cc = cc_start;
const wchar_t* str = wsTime.c_str();
int len = wsTime.GetLength();
while (cc < len && cc < 2) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
hour = hour * 10 + str[cc++] - '0';
}
if (cc < 2 || hour >= 24) {
return false;
}
if (cc < len) {
if (str[cc] == ':') {
cc++;
}
cc_start = cc;
while (cc < len && cc < cc_start + 2) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
minute = minute * 10 + str[cc++] - '0';
}
if (cc == cc_start + 1 || minute >= 60) {
return false;
}
if (cc < len) {
if (str[cc] == ':') {
cc++;
}
cc_start = cc;
while (cc < len && cc < cc_start + 2) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
second = second * 10 + str[cc++] - '0';
}
if (cc == cc_start + 1 || second >= 60) {
return false;
}
if (cc < len) {
if (str[cc] == '.') {
cc++;
cc_start = cc;
while (cc < len && cc < cc_start + 3) {
if (!FXSYS_isDecimalDigit(str[cc])) {
return false;
}
millisecond = millisecond * 10 + str[cc++] - '0';
}
if (cc < cc_start + 3)
return false;
}
if (cc < len) {
FX_TIMEZONE tzDiff;
tzDiff.tzHour = 0;
tzDiff.tzMinute = 0;
if (str[cc] != 'Z') {
cc += FX_ParseTimeZone(str + cc, len - cc, tzDiff);
}
FX_ResolveZone(hour, minute, tzDiff, pLocale);
}
}
}
}
CFX_Unitime ut;
ut.Set(0, 0, 0, hour, minute, second, millisecond);
datetime = datetime + ut;
return true;
}
static uint16_t FX_GetSolarMonthDays(uint16_t year, uint16_t month) {
if (month % 2) {
return 31;
} else if (month == 2) {
return FX_IsLeapYear(year) ? 29 : 28;
}
return 30;
}
static uint16_t FX_GetWeekDay(uint16_t year, uint16_t month, uint16_t day) {
uint16_t g_month_day[] = {0, 3, 3, 6, 1, 4, 6, 2, 5, 0, 3, 5};
uint16_t nDays =
(year - 1) % 7 + (year - 1) / 4 - (year - 1) / 100 + (year - 1) / 400;
nDays += g_month_day[month - 1] + day;
if (FX_IsLeapYear(year) && month > 2) {
nDays++;
}
return nDays % 7;
}
static uint16_t FX_GetWeekOfMonth(uint16_t year, uint16_t month, uint16_t day) {
uint16_t week_day = FX_GetWeekDay(year, month, 1);
uint16_t week_index = 0;
week_index += day / 7;
day = day % 7;
if (week_day + day > 7) {
week_index++;
}
return week_index;
}
static uint16_t FX_GetWeekOfYear(uint16_t year, uint16_t month, uint16_t day) {
uint16_t nDays = 0;
for (uint16_t i = 1; i < month; i++) {
nDays += FX_GetSolarMonthDays(year, i);
}
nDays += day;
uint16_t week_day = FX_GetWeekDay(year, 1, 1);
uint16_t week_index = 1;
week_index += nDays / 7;
nDays = nDays % 7;
if (week_day + nDays > 7) {
week_index++;
}
return week_index;
}
static bool FX_DateFormat(const CFX_WideString& wsDatePattern,
IFX_Locale* pLocale,
const CFX_Unitime& datetime,
CFX_WideString& wsResult) {
bool bRet = true;
int32_t year = datetime.GetYear();
uint8_t month = datetime.GetMonth();
uint8_t day = datetime.GetDay();
int32_t ccf = 0;
const wchar_t* strf = wsDatePattern.c_str();
int32_t lenf = wsDatePattern.GetLength();
CFX_WideStringC wsDateSymbols(gs_wsDateSymbols);
while (ccf < lenf) {
if (strf[ccf] == '\'') {
wsResult += FX_GetLiteralText(strf, ccf, lenf);
ccf++;
continue;
} else if (wsDateSymbols.Find(strf[ccf]) == -1) {
wsResult += strf[ccf++];
continue;
}
uint32_t dwSymbolNum = 1;
wchar_t dwCharSymbol = strf[ccf++];
while (ccf < lenf && strf[ccf] == dwCharSymbol) {
ccf++;
dwSymbolNum++;
}
uint32_t dwSymbol = (dwCharSymbol << 8) | (dwSymbolNum + '0');
if (dwSymbol == FXBSTR_ID(0, 0, 'D', '1')) {
CFX_WideString wsDay;
wsDay.Format(L"%d", day);
wsResult += wsDay;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'D', '2')) {
CFX_WideString wsDay;
wsDay.Format(L"%02d", day);
wsResult += wsDay;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'J', '1')) {
uint16_t nDays = 0;
for (int i = 1; i < month; i++) {
nDays += FX_GetSolarMonthDays(year, i);
}
nDays += day;
CFX_WideString wsDays;
wsDays.Format(L"%d", nDays);
wsResult += wsDays;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'J', '3')) {
uint16_t nDays = 0;
for (int i = 1; i < month; i++) {
nDays += FX_GetSolarMonthDays(year, i);
}
nDays += day;
CFX_WideString wsDays;
wsDays.Format(L"%03d", nDays);
wsResult += wsDays;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'M', '1')) {
CFX_WideString wsMonth;
wsMonth.Format(L"%d", month);
wsResult += wsMonth;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'M', '2')) {
CFX_WideString wsMonth;
wsMonth.Format(L"%02d", month);
wsResult += wsMonth;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'M', '3')) {
CFX_WideString wsTemp;
pLocale->GetMonthName(month - 1, wsTemp, true);
wsResult += wsTemp;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'M', '4')) {
CFX_WideString wsTemp;
pLocale->GetMonthName(month - 1, wsTemp, false);
wsResult += wsTemp;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'E', '1')) {
uint16_t wWeekDay = FX_GetWeekDay(year, month, day);
CFX_WideString wsWeekDay;
wsWeekDay.Format(L"%d", wWeekDay + 1);
wsResult += wsWeekDay;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'E', '3')) {
uint16_t wWeekDay = FX_GetWeekDay(year, month, day);
CFX_WideString wsTemp;
pLocale->GetDayName(wWeekDay, wsTemp, true);
wsResult += wsTemp;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'E', '4')) {
uint16_t wWeekDay = FX_GetWeekDay(year, month, day);
if (pLocale) {
CFX_WideString wsTemp;
pLocale->GetDayName(wWeekDay, wsTemp, false);
wsResult += wsTemp;
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'e', '1')) {
uint16_t wWeekDay = FX_GetWeekDay(year, month, day);
CFX_WideString wsWeekDay;
wsWeekDay.Format(L"%d", wWeekDay ? wWeekDay : 7);
wsResult += wsWeekDay;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'G', '1')) {
CFX_WideString wsTemp;
pLocale->GetEraName(wsTemp, year < 0);
wsResult += wsTemp;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'Y', '2')) {
CFX_WideString wsYear;
wsYear.Format(L"%02d", year % 100);
wsResult += wsYear;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'Y', '4')) {
CFX_WideString wsYear;
wsYear.Format(L"%d", year);
wsResult += wsYear;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'w', '1')) {
uint16_t week_index = FX_GetWeekOfMonth(year, month, day);
CFX_WideString wsWeekInMonth;
wsWeekInMonth.Format(L"%d", week_index);
wsResult += wsWeekInMonth;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'W', '2')) {
uint16_t week_index = FX_GetWeekOfYear(year, month, day);
CFX_WideString wsWeekInYear;
wsWeekInYear.Format(L"%02d", week_index);
wsResult += wsWeekInYear;
}
}
return bRet;
}
static bool FX_TimeFormat(const CFX_WideString& wsTimePattern,
IFX_Locale* pLocale,
const CFX_Unitime& datetime,
CFX_WideString& wsResult) {
bool bGMT = false;
bool bRet = true;
uint8_t hour = datetime.GetHour();
uint8_t minute = datetime.GetMinute();
uint8_t second = datetime.GetSecond();
uint16_t millisecond = datetime.GetMillisecond();
int32_t ccf = 0;
const wchar_t* strf = wsTimePattern.c_str();
int32_t lenf = wsTimePattern.GetLength();
uint16_t wHour = hour;
bool bPM = false;
if (wsTimePattern.Find('A') != -1) {
if (wHour >= 12) {
bPM = true;
}
}
CFX_WideStringC wsTimeSymbols(gs_wsTimeSymbols);
while (ccf < lenf) {
if (strf[ccf] == '\'') {
wsResult += FX_GetLiteralText(strf, ccf, lenf);
ccf++;
continue;
} else if (wsTimeSymbols.Find(strf[ccf]) == -1) {
wsResult += strf[ccf++];
continue;
}
uint32_t dwSymbolNum = 1;
wchar_t dwCharSymbol = strf[ccf++];
while (ccf < lenf && strf[ccf] == dwCharSymbol) {
ccf++;
dwSymbolNum++;
}
uint32_t dwSymbol = (dwCharSymbol << 8) | (dwSymbolNum + '0');
if (dwSymbol == FXBSTR_ID(0, 0, 'h', '1')) {
if (wHour > 12) {
wHour -= 12;
}
CFX_WideString wsHour;
wsHour.Format(L"%d", wHour == 0 ? 12 : wHour);
wsResult += wsHour;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'h', '2')) {
if (wHour > 12) {
wHour -= 12;
}
CFX_WideString wsHour;
wsHour.Format(L"%02d", wHour == 0 ? 12 : wHour);
wsResult += wsHour;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'K', '1')) {
CFX_WideString wsHour;
wsHour.Format(L"%d", wHour == 0 ? 24 : wHour);
wsResult += wsHour;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'K', '2')) {
CFX_WideString wsHour;
wsHour.Format(L"%02d", wHour == 0 ? 24 : wHour);
wsResult += wsHour;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'k', '1')) {
if (wHour > 12) {
wHour -= 12;
}
CFX_WideString wsHour;
wsHour.Format(L"%d", wHour);
wsResult += wsHour;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'H', '1')) {
CFX_WideString wsHour;
wsHour.Format(L"%d", wHour);
wsResult += wsHour;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'k', '2')) {
if (wHour > 12) {
wHour -= 12;
}
CFX_WideString wsHour;
wsHour.Format(L"%02d", wHour);
wsResult += wsHour;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'H', '2')) {
CFX_WideString wsHour;
wsHour.Format(L"%02d", wHour);
wsResult += wsHour;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'M', '1')) {
CFX_WideString wsMinute;
wsMinute.Format(L"%d", minute);
wsResult += wsMinute;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'M', '2')) {
CFX_WideString wsMinute;
wsMinute.Format(L"%02d", minute);
wsResult += wsMinute;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'S', '1')) {
CFX_WideString wsSecond;
wsSecond.Format(L"%d", second);
wsResult += wsSecond;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'S', '2')) {
CFX_WideString wsSecond;
wsSecond.Format(L"%02d", second);
wsResult += wsSecond;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'F', '3')) {
CFX_WideString wsMilliseconds;
wsMilliseconds.Format(L"%03d", millisecond);
wsResult += wsMilliseconds;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'A', '1')) {
CFX_WideString wsMeridiem;
pLocale->GetMeridiemName(wsMeridiem, !bPM);
wsResult += wsMeridiem;
} else if (dwSymbol == FXBSTR_ID(0, 0, 'Z', '1')) {
wsResult += L"GMT";
FX_TIMEZONE tz;
pLocale->GetTimeZone(&tz);
if (!bGMT && (tz.tzHour != 0 || tz.tzMinute != 0)) {
if (tz.tzHour < 0) {
wsResult += L"-";
} else {
wsResult += L"+";
}
CFX_WideString wsTimezone;
wsTimezone.Format(L"%02d:%02d", FXSYS_abs(tz.tzHour), tz.tzMinute);
wsResult += wsTimezone;
}
} else if (dwSymbol == FXBSTR_ID(0, 0, 'z', '1')) {
FX_TIMEZONE tz;
pLocale->GetTimeZone(&tz);
if (!bGMT && tz.tzHour != 0 && tz.tzMinute != 0) {
if (tz.tzHour < 0) {
wsResult += L"-";
} else {
wsResult += L"+";
}
CFX_WideString wsTimezone;
wsTimezone.Format(L"%02d:%02d", FXSYS_abs(tz.tzHour), tz.tzMinute);
wsResult += wsTimezone;
}
}
}
return bRet;
}
static bool FX_FormatDateTime(const CFX_Unitime& dt,
const CFX_WideString& wsDatePattern,
const CFX_WideString& wsTimePattern,
bool bDateFirst,
IFX_Locale* pLocale,
CFX_WideString& wsOutput) {
bool bRet = true;
CFX_WideString wsDateOut, wsTimeOut;
if (!wsDatePattern.IsEmpty()) {
bRet &= FX_DateFormat(wsDatePattern, pLocale, dt, wsDateOut);
}
if (!wsTimePattern.IsEmpty()) {
bRet &= FX_TimeFormat(wsTimePattern, pLocale, dt, wsTimeOut);
}
wsOutput = bDateFirst ? wsDateOut + wsTimeOut : wsTimeOut + wsDateOut;
return bRet;
}
bool CFX_FormatString::FormatDateTime(const CFX_WideString& wsSrcDateTime,
const CFX_WideString& wsPattern,
CFX_WideString& wsOutput) {
if (wsSrcDateTime.IsEmpty() || wsPattern.IsEmpty()) {
return false;
}
CFX_WideString wsDatePattern, wsTimePattern;
IFX_Locale* pLocale = nullptr;
FX_DATETIMETYPE eCategory =
GetDateTimeFormat(wsPattern, pLocale, wsDatePattern, wsTimePattern);
if (!pLocale || eCategory == FX_DATETIMETYPE_Unknown) {
return false;
}
CFX_Unitime dt(0);
int32_t iT = wsSrcDateTime.Find(L"T");
if (iT < 0) {
if (eCategory == FX_DATETIMETYPE_Date) {
FX_DateFromCanonical(wsSrcDateTime, dt);
} else if (eCategory == FX_DATETIMETYPE_Time) {
FX_TimeFromCanonical(wsSrcDateTime.AsStringC(), dt, pLocale);
}
} else {
FX_DateFromCanonical(wsSrcDateTime.Left(iT), dt);
FX_TimeFromCanonical(
wsSrcDateTime.Right(wsSrcDateTime.GetLength() - iT - 1).AsStringC(), dt,
pLocale);
}
return FX_FormatDateTime(dt, wsDatePattern, wsTimePattern,
eCategory != FX_DATETIMETYPE_TimeDate, pLocale,
wsOutput);
}
bool CFX_FormatString::FormatDateTime(const CFX_WideString& wsSrcDateTime,
const CFX_WideString& wsPattern,
CFX_WideString& wsOutput,
FX_DATETIMETYPE eDateTimeType) {
if (wsSrcDateTime.IsEmpty() || wsPattern.IsEmpty()) {
return false;
}
CFX_WideString wsDatePattern, wsTimePattern;
IFX_Locale* pLocale = nullptr;
FX_DATETIMETYPE eCategory =
GetDateTimeFormat(wsPattern, pLocale, wsDatePattern, wsTimePattern);
if (!pLocale) {
return false;
}
if (eCategory == FX_DATETIMETYPE_Unknown) {
if (eDateTimeType == FX_DATETIMETYPE_Time) {
wsTimePattern = wsDatePattern;
wsDatePattern.clear();
}
eCategory = eDateTimeType;
}
if (eCategory == FX_DATETIMETYPE_Unknown) {
return false;
}
CFX_Unitime dt(0);
int32_t iT = wsSrcDateTime.Find(L"T");
if (iT < 0) {
if (eCategory == FX_DATETIMETYPE_Date &&
FX_DateFromCanonical(wsSrcDateTime, dt)) {
return FX_FormatDateTime(dt, wsDatePattern, wsTimePattern, true, pLocale,
wsOutput);
} else if (eCategory == FX_DATETIMETYPE_Time &&
FX_TimeFromCanonical(wsSrcDateTime.AsStringC(), dt, pLocale)) {
return FX_FormatDateTime(dt, wsDatePattern, wsTimePattern, true, pLocale,
wsOutput);
}
} else {
CFX_WideString wsSrcDate(wsSrcDateTime.c_str(), iT);
CFX_WideStringC wsSrcTime(wsSrcDateTime.c_str() + iT + 1,
wsSrcDateTime.GetLength() - iT - 1);
if (wsSrcDate.IsEmpty() || wsSrcTime.IsEmpty()) {
return false;
}
if (FX_DateFromCanonical(wsSrcDate, dt) &&
FX_TimeFromCanonical(wsSrcTime, dt, pLocale)) {
return FX_FormatDateTime(dt, wsDatePattern, wsTimePattern,
eCategory != FX_DATETIMETYPE_TimeDate, pLocale,
wsOutput);
}
}
return false;
}
bool CFX_FormatString::FormatDateTime(const CFX_Unitime& dt,
const CFX_WideString& wsPattern,
CFX_WideString& wsOutput) {
if (wsPattern.IsEmpty()) {
return false;
}
CFX_WideString wsDatePattern, wsTimePattern;
IFX_Locale* pLocale = nullptr;
FX_DATETIMETYPE eCategory =
GetDateTimeFormat(wsPattern, pLocale, wsDatePattern, wsTimePattern);
if (!pLocale) {
return false;
}
return FX_FormatDateTime(dt, wsPattern, wsTimePattern,
eCategory != FX_DATETIMETYPE_TimeDate, pLocale,
wsOutput);
}
bool CFX_FormatString::FormatZero(const CFX_WideString& wsPattern,
CFX_WideString& wsOutput) {
if (wsPattern.IsEmpty()) {
return false;
}
CFX_WideString wsTextFormat;
GetTextFormat(wsPattern, L"zero", wsTextFormat);
int32_t iPattern = 0;
const wchar_t* pStrPattern = wsTextFormat.c_str();
int32_t iLenPattern = wsTextFormat.GetLength();
while (iPattern < iLenPattern) {
if (pStrPattern[iPattern] == '\'') {
wsOutput += FX_GetLiteralText(pStrPattern, iPattern, iLenPattern);
iPattern++;
continue;
} else {
wsOutput += pStrPattern[iPattern++];
continue;
}
}
return true;
}
bool CFX_FormatString::FormatNull(const CFX_WideString& wsPattern,
CFX_WideString& wsOutput) {
if (wsPattern.IsEmpty()) {
return false;
}
CFX_WideString wsTextFormat;
GetTextFormat(wsPattern, L"null", wsTextFormat);
int32_t iPattern = 0;
const wchar_t* pStrPattern = wsTextFormat.c_str();
int32_t iLenPattern = wsTextFormat.GetLength();
while (iPattern < iLenPattern) {
if (pStrPattern[iPattern] == '\'') {
wsOutput += FX_GetLiteralText(pStrPattern, iPattern, iLenPattern);
iPattern++;
continue;
} else {
wsOutput += pStrPattern[iPattern++];
continue;
}
}
return true;
}
IFX_Locale* CFX_FormatString::GetPatternLocale(const CFX_WideString& wsLocale) {
return m_pLocaleMgr->GetLocaleByName(wsLocale);
}
#define FXMATH_DECIMAL_SCALELIMIT 0x1c
#define FXMATH_DECIMAL_NEGMASK (0x80000000L)
#define FXMATH_DECIMAL_FORCEBOOL(x) (!(!(x)))
#define FXMATH_DECIMAL_MAKEFLAGS(NEG, SCALE) \
(((SCALE) << 0x10) | ((NEG) ? FXMATH_DECIMAL_NEGMASK : 0))
#define FXMATH_DECIMAL_FLAGS2NEG(FLAGS) \
FXMATH_DECIMAL_FORCEBOOL((FLAGS)&FXMATH_DECIMAL_NEGMASK)
#define FXMATH_DECIMAL_FLAGS2SCALE(FLAGS) \
((uint8_t)(((FLAGS) & ~FXMATH_DECIMAL_NEGMASK) >> 0x10))
#define FXMATH_DECIMAL_RSHIFT32BIT(x) ((x) >> 0x10 >> 0x10)
#define FXMATH_DECIMAL_LSHIFT32BIT(x) ((x) << 0x10 << 0x10)
static inline uint8_t fxmath_decimal_helper_div10(uint64_t& phi,
uint64_t& pmid,
uint64_t& plo) {
uint8_t retVal;
pmid += FXMATH_DECIMAL_LSHIFT32BIT(phi % 0xA);
phi /= 0xA;
plo += FXMATH_DECIMAL_LSHIFT32BIT(pmid % 0xA);
pmid /= 0xA;
retVal = plo % 0xA;
plo /= 0xA;
return retVal;
}
static inline uint8_t fxmath_decimal_helper_div10_any(uint64_t nums[],
uint8_t numcount) {
uint8_t retVal = 0;
for (int i = numcount - 1; i > 0; i--) {
nums[i - 1] += FXMATH_DECIMAL_LSHIFT32BIT(nums[i] % 0xA);
nums[i] /= 0xA;
}
if (numcount) {
retVal = nums[0] % 0xA;
nums[0] /= 0xA;
}
return retVal;
}
static inline void fxmath_decimal_helper_mul10(uint64_t& phi,
uint64_t& pmid,
uint64_t& plo) {
plo *= 0xA;
pmid = pmid * 0xA + FXMATH_DECIMAL_RSHIFT32BIT(plo);
plo = (uint32_t)plo;
phi = phi * 0xA + FXMATH_DECIMAL_RSHIFT32BIT(pmid);
pmid = (uint32_t)pmid;
}
static inline void fxmath_decimal_helper_mul10_any(uint64_t nums[],
uint8_t numcount) {
nums[0] *= 0xA;
for (int i = 1; i < numcount; i++) {
nums[i] = nums[i] * 0xA + FXMATH_DECIMAL_RSHIFT32BIT(nums[i - 1]);
nums[i - 1] = (uint32_t)nums[i - 1];
}
}
static inline void fxmath_decimal_helper_normalize(uint64_t& phi,
uint64_t& pmid,
uint64_t& plo) {
phi += FXMATH_DECIMAL_RSHIFT32BIT(pmid);
pmid = (uint32_t)pmid;
pmid += FXMATH_DECIMAL_RSHIFT32BIT(plo);
plo = (uint32_t)plo;
phi += FXMATH_DECIMAL_RSHIFT32BIT(pmid);
pmid = (uint32_t)pmid;
}
static inline void fxmath_decimal_helper_normalize_any(uint64_t nums[],
uint8_t len) {
{
for (int i = len - 2; i > 0; i--) {
nums[i + 1] += FXMATH_DECIMAL_RSHIFT32BIT(nums[i]);
nums[i] = (uint32_t)nums[i];
}
}
{
for (int i = 0; i < len - 1; i++) {
nums[i + 1] += FXMATH_DECIMAL_RSHIFT32BIT(nums[i]);
nums[i] = (uint32_t)nums[i];
}
}
}
static inline int8_t fxmath_decimal_helper_raw_compare(uint32_t hi1,
uint32_t mid1,
uint32_t lo1,
uint32_t hi2,
uint32_t mid2,
uint32_t lo2) {
int8_t retVal = 0;
if (!retVal) {
retVal += (hi1 > hi2 ? 1 : (hi1 < hi2 ? -1 : 0));
}
if (!retVal) {
retVal += (mid1 > mid2 ? 1 : (mid1 < mid2 ? -1 : 0));
}
if (!retVal) {
retVal += (lo1 > lo2 ? 1 : (lo1 < lo2 ? -1 : 0));
}
return retVal;
}
static inline int8_t fxmath_decimal_helper_raw_compare_any(uint64_t a[],
uint8_t al,
uint64_t b[],
uint8_t bl) {
int8_t retVal = 0;
for (int i = std::max(al - 1, bl - 1); i >= 0; i--) {
uint64_t l = (i >= al ? 0 : a[i]), r = (i >= bl ? 0 : b[i]);
retVal += (l > r ? 1 : (l < r ? -1 : 0));
if (retVal) {
return retVal;
}
}
return retVal;
}
static inline void fxmath_decimal_helper_dec_any(uint64_t a[], uint8_t al) {
for (int i = 0; i < al; i++) {
if (a[i]--) {
return;
}
}
}
static inline void fxmath_decimal_helper_inc_any(uint64_t a[], uint8_t al) {
for (int i = 0; i < al; i++) {
a[i]++;
if ((uint32_t)a[i] == a[i]) {
return;
}
a[i] = 0;
}
}
static inline void fxmath_decimal_helper_raw_mul(uint64_t a[],
uint8_t al,
uint64_t b[],
uint8_t bl,
uint64_t c[],
uint8_t cl) {
ASSERT(al + bl <= cl);
{
for (int i = 0; i < cl; i++) {
c[i] = 0;
}
}
{
for (int i = 0; i < al; i++) {
for (int j = 0; j < bl; j++) {
uint64_t m = (uint64_t)a[i] * b[j];
c[i + j] += (uint32_t)m;
c[i + j + 1] += FXMATH_DECIMAL_RSHIFT32BIT(m);
}
}
}
{
for (int i = 0; i < cl - 1; i++) {
c[i + 1] += FXMATH_DECIMAL_RSHIFT32BIT(c[i]);
c[i] = (uint32_t)c[i];
}
}
{
for (int i = 0; i < cl; i++) {
c[i] = (uint32_t)c[i];
}
}
}
static inline void fxmath_decimal_helper_raw_div(uint64_t a[],
uint8_t al,
uint64_t b[],
uint8_t bl,
uint64_t c[],
uint8_t cl) {
int i;
for (i = 0; i < cl; i++) {
c[i] = 0;
}
uint64_t left[16] = {0}, right[16] = {0};
left[0] = 0;
for (i = 0; i < al; i++) {
right[i] = a[i];
}
uint64_t tmp[16];
while (fxmath_decimal_helper_raw_compare_any(left, al, right, al) <= 0) {
uint64_t cur[16];
for (i = 0; i < al; i++) {
cur[i] = left[i] + right[i];
}
for (i = al - 1; i >= 0; i--) {
if (i) {
cur[i - 1] += FXMATH_DECIMAL_LSHIFT32BIT(cur[i] % 2);
}
cur[i] /= 2;
}
fxmath_decimal_helper_raw_mul(cur, al, b, bl, tmp, 16);
switch (fxmath_decimal_helper_raw_compare_any(tmp, 16, a, al)) {
case -1:
for (i = 0; i < 16; i++) {
left[i] = cur[i];
}
left[0]++;
fxmath_decimal_helper_normalize_any(left, al);
break;
case 1:
for (i = 0; i < 16; i++) {
right[i] = cur[i];
}
fxmath_decimal_helper_dec_any(right, al);
break;
case 0:
for (i = 0; i < std::min(al, cl); i++) {
c[i] = cur[i];
}
return;
}
}
for (i = 0; i < std::min(al, cl); i++) {
c[i] = left[i];
}
}
static inline bool fxmath_decimal_helper_outofrange(uint64_t a[],
uint8_t al,
uint8_t goal) {
for (int i = goal; i < al; i++) {
if (a[i]) {
return true;
}
}
return false;
}
static inline void fxmath_decimal_helper_shrinkintorange(uint64_t a[],
uint8_t al,
uint8_t goal,
uint8_t& scale) {
bool bRoundUp = false;
while (scale != 0 && (scale > FXMATH_DECIMAL_SCALELIMIT ||
fxmath_decimal_helper_outofrange(a, al, goal))) {
bRoundUp = fxmath_decimal_helper_div10_any(a, al) >= 5;
scale--;
}
if (bRoundUp) {
fxmath_decimal_helper_normalize_any(a, goal);
fxmath_decimal_helper_inc_any(a, goal);
}
}
static inline void fxmath_decimal_helper_truncate(uint64_t& phi,
uint64_t& pmid,
uint64_t& plo,
uint8_t& scale,
uint8_t minscale = 0) {
while (scale > minscale) {
uint64_t thi = phi, tmid = pmid, tlo = plo;
if (fxmath_decimal_helper_div10(thi, tmid, tlo) != 0) {
break;
}
phi = thi, pmid = tmid, plo = tlo;
scale--;
}
}
CFX_Decimal::CFX_Decimal() {
m_uLo = m_uMid = m_uHi = m_uFlags = 0;
}
CFX_Decimal::CFX_Decimal(uint64_t val) {
m_uLo = (uint32_t)val;
m_uMid = (uint32_t)FXMATH_DECIMAL_RSHIFT32BIT(val);
m_uHi = 0;
m_uFlags = 0;
}
CFX_Decimal::CFX_Decimal(uint32_t val) {
m_uLo = (uint32_t)val;
m_uMid = m_uHi = 0;
m_uFlags = 0;
}
CFX_Decimal::CFX_Decimal(uint32_t lo,
uint32_t mid,
uint32_t hi,
bool neg,
uint8_t scale) {
scale = (scale > FXMATH_DECIMAL_SCALELIMIT ? 0 : scale);
m_uLo = lo;
m_uMid = mid;
m_uHi = hi;
m_uFlags = FXMATH_DECIMAL_MAKEFLAGS(neg && IsNotZero(), scale);
}
CFX_Decimal::CFX_Decimal(int32_t val) {
if (val >= 0) {
*this = CFX_Decimal((uint32_t)val);
} else {
*this = CFX_Decimal((uint32_t)-val);
SetNegate();
}
}
CFX_Decimal::CFX_Decimal(int64_t val) {
if (val >= 0) {
*this = CFX_Decimal((uint64_t)val);
} else {
*this = CFX_Decimal((uint64_t)-val);
SetNegate();
}
}
CFX_Decimal::CFX_Decimal(FX_FLOAT val, uint8_t scale) {
FX_FLOAT newval = fabs(val);
uint64_t phi, pmid, plo;
plo = (uint64_t)newval;
pmid = (uint64_t)(newval / 1e32);
phi = (uint64_t)(newval / 1e64);
newval = FXSYS_fmod(newval, 1.0f);
for (uint8_t iter = 0; iter < scale; iter++) {
fxmath_decimal_helper_mul10(phi, pmid, plo);
newval *= 10;
plo += (uint64_t)newval;
newval = FXSYS_fmod(newval, 1.0f);
}
plo += FXSYS_round(newval);
fxmath_decimal_helper_normalize(phi, pmid, plo);
m_uHi = (uint32_t)phi;
m_uMid = (uint32_t)pmid;
m_uLo = (uint32_t)plo;
m_uFlags = FXMATH_DECIMAL_MAKEFLAGS(val < 0 && IsNotZero(), scale);
}
CFX_Decimal::CFX_Decimal(const CFX_WideStringC& strObj) {
const wchar_t* str = strObj.c_str();
const wchar_t* strBound = str + strObj.GetLength();
bool pointmet = 0;
bool negmet = 0;
uint8_t scale = 0;
m_uHi = m_uMid = m_uLo = 0;
while (str != strBound && *str == ' ') {
str++;
}
if (str != strBound && *str == '-') {
negmet = 1;
str++;
} else if (str != strBound && *str == '+') {
str++;
}
while (str != strBound && ((*str >= '0' && *str <= '9') || *str == '.') &&
scale < FXMATH_DECIMAL_SCALELIMIT) {
if (*str == '.') {
if (pointmet) {
goto cont;
}
pointmet = 1;
} else {
m_uHi = m_uHi * 0xA + FXMATH_DECIMAL_RSHIFT32BIT((uint64_t)m_uMid * 0xA);
m_uMid = m_uMid * 0xA + FXMATH_DECIMAL_RSHIFT32BIT((uint64_t)m_uLo * 0xA);
m_uLo = m_uLo * 0xA + (*str - '0');
if (pointmet) {
scale++;
}
}
cont:
str++;
}
m_uFlags = FXMATH_DECIMAL_MAKEFLAGS(negmet && IsNotZero(), scale);
}
CFX_Decimal::CFX_Decimal(const CFX_ByteStringC& strObj) {
*this = CFX_Decimal(CFX_WideString::FromLocal(strObj).AsStringC());
}
CFX_Decimal::operator CFX_WideString() const {
CFX_WideString retString;
CFX_WideString tmpbuf;
uint64_t phi = m_uHi, pmid = m_uMid, plo = m_uLo;
while (phi || pmid || plo) {
tmpbuf += fxmath_decimal_helper_div10(phi, pmid, plo) + '0';
}
uint8_t outputlen = (uint8_t)tmpbuf.GetLength();
uint8_t scale = (uint8_t)FXMATH_DECIMAL_FLAGS2SCALE(m_uFlags);
while (scale >= outputlen) {
tmpbuf += '0';
outputlen++;
}
if (FXMATH_DECIMAL_FLAGS2NEG(m_uFlags) && IsNotZero()) {
retString += '-';
}
for (uint8_t idx = 0; idx < outputlen; idx++) {
if (idx == (outputlen - scale) && scale != 0) {
retString += '.';
}
retString += tmpbuf[outputlen - 1 - idx];
}
return retString;
}
CFX_Decimal::operator double() const {
double pow = (double)(1 << 16) * (1 << 16);
double base =
((double)m_uHi) * pow * pow + ((double)m_uMid) * pow + ((double)m_uLo);
int8_t scale = FXMATH_DECIMAL_FLAGS2SCALE(m_uFlags);
bool bNeg = FXMATH_DECIMAL_FLAGS2NEG(m_uFlags);
return (bNeg ? -1 : 1) * base * ::pow(10.0, -scale);
}
void CFX_Decimal::SetScale(uint8_t newscale) {
uint8_t oldscale = FXMATH_DECIMAL_FLAGS2SCALE(m_uFlags);
if (newscale > oldscale) {
uint64_t phi = m_uHi, pmid = m_uMid, plo = m_uLo;
for (uint8_t iter = 0; iter < newscale - oldscale; iter++) {
fxmath_decimal_helper_mul10(phi, pmid, plo);
}
m_uHi = (uint32_t)phi;
m_uMid = (uint32_t)pmid;
m_uLo = (uint32_t)plo;
m_uFlags = FXMATH_DECIMAL_MAKEFLAGS(
FXMATH_DECIMAL_FLAGS2NEG(m_uFlags) && IsNotZero(), newscale);
} else if (newscale < oldscale) {
uint64_t phi, pmid, plo;
phi = 0, pmid = 0, plo = 5;
{
for (uint8_t iter = 0; iter < oldscale - newscale - 1; iter++) {
fxmath_decimal_helper_mul10(phi, pmid, plo);
}
}
phi += m_uHi;
pmid += m_uMid;
plo += m_uLo;
fxmath_decimal_helper_normalize(phi, pmid, plo);
{
for (uint8_t iter = 0; iter < oldscale - newscale; iter++) {
fxmath_decimal_helper_div10(phi, pmid, plo);
}
}
m_uHi = (uint32_t)phi;
m_uMid = (uint32_t)pmid;
m_uLo = (uint32_t)plo;
m_uFlags = FXMATH_DECIMAL_MAKEFLAGS(
FXMATH_DECIMAL_FLAGS2NEG(m_uFlags) && IsNotZero(), newscale);
}
}
uint8_t CFX_Decimal::GetScale() {
uint8_t oldscale = FXMATH_DECIMAL_FLAGS2SCALE(m_uFlags);
return oldscale;
}
void CFX_Decimal::SetAbs() {
m_uFlags &= ~FXMATH_DECIMAL_NEGMASK;
}
void CFX_Decimal::SetNegate() {
if (IsNotZero()) {
m_uFlags ^= FXMATH_DECIMAL_NEGMASK;
}
}
void CFX_Decimal::FloorOrCeil(bool bFloor) {
uint64_t nums[3] = {m_uLo, m_uMid, m_uHi};
bool bDataLoss = false;
for (int i = FXMATH_DECIMAL_FLAGS2SCALE(m_uFlags); i > 0; i--) {
bDataLoss = fxmath_decimal_helper_div10_any(nums, 3) || bDataLoss;
}
if (bDataLoss && (bFloor ? FXMATH_DECIMAL_FLAGS2NEG(m_uFlags)
: !FXMATH_DECIMAL_FLAGS2NEG(m_uFlags))) {
fxmath_decimal_helper_inc_any(nums, 3);
}
m_uHi = (uint32_t)nums[2];
m_uMid = (uint32_t)nums[1];
m_uLo = (uint32_t)nums[0];
m_uFlags = FXMATH_DECIMAL_MAKEFLAGS(
FXMATH_DECIMAL_FLAGS2NEG(m_uFlags) && IsNotZero(), 0);
}
void CFX_Decimal::SetFloor() {
FloorOrCeil(true);
}
void CFX_Decimal::SetCeiling() {
FloorOrCeil(false);
}
void CFX_Decimal::SetTruncate() {
FloorOrCeil(!FXMATH_DECIMAL_FLAGS2NEG(m_uFlags));
}
void CFX_Decimal::Swap(CFX_Decimal& val) {
uint32_t tmp;
tmp = m_uHi;
m_uHi = val.m_uHi;
val.m_uHi = tmp;
tmp = m_uMid;
m_uMid = val.m_uMid;
val.m_uMid = tmp;
tmp = m_uLo;
m_uLo = val.m_uLo;
val.m_uLo = tmp;
tmp = m_uFlags;
m_uFlags = val.m_uFlags;
val.m_uFlags = tmp;
}
int8_t CFX_Decimal::Compare(const CFX_Decimal& val) const {
CFX_Decimal lhs = *this, rhs = val;
int8_t retVal = 0;
if (FXMATH_DECIMAL_FLAGS2SCALE(lhs.m_uFlags) !=
FXMATH_DECIMAL_FLAGS2SCALE(rhs.m_uFlags)) {
uint8_t scale = std::min(FXMATH_DECIMAL_FLAGS2SCALE(lhs.m_uFlags),
FXMATH_DECIMAL_FLAGS2SCALE(rhs.m_uFlags));
lhs.SetScale(scale);
rhs.SetScale(scale);
}
retVal = -(FXMATH_DECIMAL_FLAGS2NEG(lhs.m_uFlags) -
FXMATH_DECIMAL_FLAGS2NEG(rhs.m_uFlags));
if (retVal) {
return retVal;
}
retVal = fxmath_decimal_helper_raw_compare(lhs.m_uHi, lhs.m_uMid, lhs.m_uLo,
rhs.m_uHi, rhs.m_uMid, rhs.m_uLo);
return (FXMATH_DECIMAL_FLAGS2NEG(lhs.m_uFlags) ? -retVal : retVal);
}
CFX_Decimal CFX_Decimal::AddOrMinus(const CFX_Decimal& val,
bool isAdding) const {
CFX_Decimal lhs = *this, rhs = val;
if (FXMATH_DECIMAL_FLAGS2SCALE(lhs.m_uFlags) !=
FXMATH_DECIMAL_FLAGS2SCALE(rhs.m_uFlags)) {
uint8_t scale = std::max(FXMATH_DECIMAL_FLAGS2SCALE(lhs.m_uFlags),
FXMATH_DECIMAL_FLAGS2SCALE(rhs.m_uFlags));
lhs.SetScale(scale);
rhs.SetScale(scale);
}
if (!isAdding) {
rhs.SetNegate();
}
if (FXMATH_DECIMAL_FLAGS2NEG(lhs.m_uFlags) ==
FXMATH_DECIMAL_FLAGS2NEG(rhs.m_uFlags)) {
uint64_t phi = lhs.m_uHi, pmid = lhs.m_uMid, plo = lhs.m_uLo;
phi += rhs.m_uHi;
pmid += rhs.m_uMid;
plo += rhs.m_uLo;
fxmath_decimal_helper_normalize(phi, pmid, plo);
if (FXMATH_DECIMAL_RSHIFT32BIT(phi) &&
FXMATH_DECIMAL_FLAGS2SCALE(lhs.m_uFlags) != 0) {
fxmath_decimal_helper_div10(phi, pmid, plo);
lhs.m_uFlags = FXMATH_DECIMAL_MAKEFLAGS(
FXMATH_DECIMAL_FLAGS2NEG(lhs.m_uFlags),
FXMATH_DECIMAL_FLAGS2SCALE(lhs.m_uFlags) - 1);
}
lhs.m_uHi = (uint32_t)phi;
lhs.m_uMid = (uint32_t)pmid;
lhs.m_uLo = (uint32_t)plo;
return lhs;
} else {
if (fxmath_decimal_helper_raw_compare(lhs.m_uHi, lhs.m_uMid, lhs.m_uLo,
rhs.m_uHi, rhs.m_uMid,
rhs.m_uLo) < 0) {
lhs.Swap(rhs);
}
lhs.m_uHi -= rhs.m_uHi;
if (lhs.m_uMid < rhs.m_uMid) {
lhs.m_uHi--;
}
lhs.m_uMid -= rhs.m_uMid;
if (lhs.m_uLo < rhs.m_uLo) {
if (!lhs.m_uMid) {
lhs.m_uHi--;
}
lhs.m_uMid--;
}
lhs.m_uLo -= rhs.m_uLo;
return lhs;
}
}
CFX_Decimal CFX_Decimal::Multiply(const CFX_Decimal& val) const {
uint64_t a[3] = {m_uLo, m_uMid, m_uHi},
b[3] = {val.m_uLo, val.m_uMid, val.m_uHi};
uint64_t c[6];
fxmath_decimal_helper_raw_mul(a, 3, b, 3, c, 6);
bool neg = FXMATH_DECIMAL_FLAGS2NEG(m_uFlags) ^
FXMATH_DECIMAL_FLAGS2NEG(val.m_uFlags);
uint8_t scale = FXMATH_DECIMAL_FLAGS2SCALE(m_uFlags) +
FXMATH_DECIMAL_FLAGS2SCALE(val.m_uFlags);
fxmath_decimal_helper_shrinkintorange(c, 6, 3, scale);
return CFX_Decimal((uint32_t)c[0], (uint32_t)c[1], (uint32_t)c[2], neg,
scale);
}
CFX_Decimal CFX_Decimal::Divide(const CFX_Decimal& val) const {
if (!val.IsNotZero()) {
return CFX_Decimal();
}
bool neg = FXMATH_DECIMAL_FLAGS2NEG(m_uFlags) ^
FXMATH_DECIMAL_FLAGS2NEG(val.m_uFlags);
uint64_t a[7] = {m_uLo, m_uMid, m_uHi},
b[3] = {val.m_uLo, val.m_uMid, val.m_uHi}, c[7] = {0};
uint8_t scale = 0;
if (FXMATH_DECIMAL_FLAGS2SCALE(m_uFlags) <
FXMATH_DECIMAL_FLAGS2SCALE(val.m_uFlags)) {
for (int i = FXMATH_DECIMAL_FLAGS2SCALE(val.m_uFlags) -
FXMATH_DECIMAL_FLAGS2SCALE(m_uFlags);
i > 0; i--) {
fxmath_decimal_helper_mul10_any(a, 7);
}
} else {
scale = FXMATH_DECIMAL_FLAGS2SCALE(m_uFlags) -
FXMATH_DECIMAL_FLAGS2SCALE(val.m_uFlags);
}
uint8_t minscale = scale;
if (!IsNotZero()) {
return CFX_Decimal(0, 0, 0, 0, minscale);
}
while (!a[6]) {
fxmath_decimal_helper_mul10_any(a, 7);
scale++;
}
fxmath_decimal_helper_div10_any(a, 7);
scale--;
fxmath_decimal_helper_raw_div(a, 6, b, 3, c, 7);
fxmath_decimal_helper_shrinkintorange(c, 6, 3, scale);
fxmath_decimal_helper_truncate(c[2], c[1], c[0], scale, minscale);
return CFX_Decimal((uint32_t)c[0], (uint32_t)c[1], (uint32_t)c[2], neg,
scale);
}
CFX_Decimal CFX_Decimal::Modulus(const CFX_Decimal& val) const {
CFX_Decimal lhs = *this, rhs_abs = val;
rhs_abs.SetAbs();
if (!rhs_abs.IsNotZero()) {
return *this;
}
while (true) {
CFX_Decimal lhs_abs = lhs;
lhs_abs.SetAbs();
if (lhs_abs < rhs_abs) {
break;
}
CFX_Decimal quot = lhs / rhs_abs;
quot.SetTruncate();
lhs = lhs - quot * rhs_abs;
}
return lhs;
}
bool CFX_Decimal::operator==(const CFX_Decimal& val) const {
return Compare(val) == 0;
}
bool CFX_Decimal::operator<=(const CFX_Decimal& val) const {
return Compare(val) <= 0;
}
bool CFX_Decimal::operator>=(const CFX_Decimal& val) const {
return Compare(val) >= 0;
}
bool CFX_Decimal::operator!=(const CFX_Decimal& val) const {
return Compare(val) != 0;
}
bool CFX_Decimal::operator<(const CFX_Decimal& val) const {
return Compare(val) < 0;
}
bool CFX_Decimal::operator>(const CFX_Decimal& val) const {
return Compare(val) > 0;
}
CFX_Decimal CFX_Decimal::operator+(const CFX_Decimal& val) const {
return AddOrMinus(val, true);
}
CFX_Decimal CFX_Decimal::operator-(const CFX_Decimal& val) const {
return AddOrMinus(val, false);
}
CFX_Decimal CFX_Decimal::operator*(const CFX_Decimal& val) const {
return Multiply(val);
}
CFX_Decimal CFX_Decimal::operator/(const CFX_Decimal& val) const {
return Divide(val);
}
CFX_Decimal CFX_Decimal::operator%(const CFX_Decimal& val) const {
return Modulus(val);
}