X-Git-Url: https://git.saurik.com/apple/icu.git/blobdiff_plain/374ca955a76ecab1204ca8bfa63ff9238d998416..151279e3792e85d0417b499c229886b3af724f55:/icuSources/test/intltest/numrgts.cpp?ds=sidebyside diff --git a/icuSources/test/intltest/numrgts.cpp b/icuSources/test/intltest/numrgts.cpp index bfc26475..633d4537 100644 --- a/icuSources/test/intltest/numrgts.cpp +++ b/icuSources/test/intltest/numrgts.cpp @@ -1,6 +1,5 @@ /*********************************************************************** - * COPYRIGHT: - * Copyright (c) 1997-2004, International Business Machines Corporation + * Copyright (c) 1997-2012, International Business Machines Corporation * and others. All Rights Reserved. ***********************************************************************/ @@ -19,8 +18,77 @@ #include "unicode/resbund.h" #include "unicode/calendar.h" #include "unicode/datefmt.h" +#include "unicode/ucurr.h" #include "putilimp.h" +class MyNumberFormatTest : public NumberFormat +{ +public: + + virtual UClassID getDynamicClassID(void) const; + + virtual UnicodeString& format( double number, + UnicodeString& toAppendTo, + FieldPositionIterator* posIter, + UErrorCode& status) const + { + return NumberFormat::format(number, toAppendTo, posIter, status); + } + + /* Just keep this here to make some of the compilers happy */ + virtual UnicodeString& format(const Formattable& obj, + UnicodeString& toAppendTo, + FieldPosition& pos, + UErrorCode& status) const + { + return NumberFormat::format(obj, toAppendTo, pos, status); + } + + /* Just use one of the format functions */ + virtual UnicodeString& format( double /* number */, + UnicodeString& toAppendTo, + FieldPosition& /* pos */) const + { + toAppendTo = ""; + return toAppendTo; + } + + /* + public Number parse(String text, ParsePosition parsePosition) + { return new Integer(0); } + */ + + /* Just use one of the parse functions */ + virtual void parse( const UnicodeString& /* text */, + Formattable& result, + ParsePosition& /* parsePosition */) const + { + result.setLong((int32_t)0); + } + + virtual void parse( const UnicodeString& text, + Formattable& result, + UErrorCode& status) const + { + NumberFormat::parse(text, result, status); + } + virtual Format* clone() const + { return NULL; } + + virtual UnicodeString& format(int32_t, + UnicodeString& foo, + FieldPosition&) const + { return foo.remove(); } + + virtual UnicodeString& format(int64_t, + UnicodeString& foo, + FieldPosition&) const + { return foo.remove(); } + + virtual void applyPattern(const UnicodeString&, UParseError&, UErrorCode&){ + } +}; + int32_t gMyNumberFormatTestClassID; UClassID MyNumberFormatTest::getDynamicClassID() const { @@ -100,17 +168,26 @@ NumberFormatRegressionTest::runIndexedTest( int32_t index, UBool exec, const cha CASE(58,Test4243011); CASE(59,Test4243108); CASE(60,TestJ691); + CASE(61,Test8199); + CASE(62,Test9109); + CASE(63,Test9780); + CASE(64,Test9677); default: name = ""; break; } } UBool -NumberFormatRegressionTest::failure(UErrorCode status, const UnicodeString& msg, const Locale& l) +NumberFormatRegressionTest::failure(UErrorCode status, const UnicodeString& msg, const Locale& l, UBool possibleDataError) { if(U_FAILURE(status)) { - errln(UnicodeString("FAIL: ", "") + msg - + UnicodeString(" failed, error ", "") + UnicodeString(u_errorName(status), "") + UnicodeString(l.getName(),"")); + if (possibleDataError) { + dataerrln(UnicodeString("FAIL: ", "") + msg + + UnicodeString(" failed, error ", "") + UnicodeString(u_errorName(status), "") + UnicodeString(l.getName(),"")); + } else { + errcheckln(status, UnicodeString("FAIL: ", "") + msg + + UnicodeString(" failed, error ", "") + UnicodeString(u_errorName(status), "") + UnicodeString(l.getName(),"")); + } return TRUE; } @@ -118,11 +195,16 @@ NumberFormatRegressionTest::failure(UErrorCode status, const UnicodeString& msg, } UBool -NumberFormatRegressionTest::failure(UErrorCode status, const UnicodeString& msg, const char *l) +NumberFormatRegressionTest::failure(UErrorCode status, const UnicodeString& msg, const char *l, UBool possibleDataError) { if(U_FAILURE(status)) { - errln(UnicodeString("FAIL: ", "") + msg - + UnicodeString(" failed, error ", "") + UnicodeString(u_errorName(status), "") + UnicodeString(l, "")); + if (possibleDataError) { + dataerrln(UnicodeString("FAIL: ", "") + msg + + UnicodeString(" failed, error ", "") + UnicodeString(u_errorName(status), "") + UnicodeString(l, "")); + } else { + errcheckln(status, UnicodeString("FAIL: ", "") + msg + + UnicodeString(" failed, error ", "") + UnicodeString(u_errorName(status), "") + UnicodeString(l, "")); + } return TRUE; } @@ -130,11 +212,16 @@ NumberFormatRegressionTest::failure(UErrorCode status, const UnicodeString& msg, } UBool -NumberFormatRegressionTest::failure(UErrorCode status, const UnicodeString& msg) +NumberFormatRegressionTest::failure(UErrorCode status, const UnicodeString& msg, UBool possibleDataError) { if(U_FAILURE(status)) { - errln(UnicodeString("FAIL: ", "") + msg - + UnicodeString(" failed, error ", "") + UnicodeString(u_errorName(status), "")); + if (possibleDataError) { + dataerrln(UnicodeString("FAIL: ", "") + msg + + UnicodeString(" failed, error ", "") + UnicodeString(u_errorName(status), "")); + } else { + errcheckln(status, UnicodeString("FAIL: ", "") + msg + + UnicodeString(" failed, error ", "") + UnicodeString(u_errorName(status), "")); + } return TRUE; } @@ -197,23 +284,24 @@ void NumberFormatRegressionTest::Test4088161 (void) { UErrorCode status = U_ZERO_ERROR; DecimalFormat *df = new DecimalFormat(status); - failure(status, "new DecimalFormat", ""); - double d = 100; - df->setMinimumFractionDigits(0); - df->setMaximumFractionDigits(16); - UnicodeString sBuf1; - FieldPosition fp1(0); - logln(UnicodeString("d = ") + d); - logln("maxFractionDigits = " + df->getMaximumFractionDigits()); - - logln(" format(d) = '" + df->format(d, sBuf1, fp1) + "'"); - df->setMaximumFractionDigits(17); - UnicodeString sBuf2; - FieldPosition fp2(0); - logln("maxFractionDigits = " + df->getMaximumFractionDigits()); - sBuf2 = df->format(d, sBuf2, fp2); - if(sBuf2 != "100") - errln(" format(d) = '" + sBuf2 + "'"); + if (!failure(status, "new DecimalFormat", "")) { + double d = 100; + df->setMinimumFractionDigits(0); + df->setMaximumFractionDigits(16); + UnicodeString sBuf1; + FieldPosition fp1(0); + logln(UnicodeString("d = ") + d); + logln(UnicodeString("maxFractionDigits = ") + df->getMaximumFractionDigits()); + + logln(" format(d) = '" + df->format(d, sBuf1, fp1) + "'"); + df->setMaximumFractionDigits(17); + UnicodeString sBuf2; + FieldPosition fp2(0); + logln(UnicodeString("maxFractionDigits = ") + df->getMaximumFractionDigits()); + sBuf2 = df->format(d, sBuf2, fp2); + if(sBuf2 != "100") + errln(" format(d) = '" + sBuf2 + "'"); + } delete df; } @@ -312,7 +400,10 @@ NumberFormatRegressionTest::assignFloatValue(float returnfloat) logln(UnicodeString(" VALUE ") + returnfloat); UErrorCode status = U_ZERO_ERROR; NumberFormat *nfcommon = NumberFormat::createCurrencyInstance(Locale::getUS(), status); - failure(status, "NumberFormat::createCurrencyInstance", Locale::getUS()); + if (failure(status, "NumberFormat::createCurrencyInstance", Locale::getUS(), TRUE)){ + delete nfcommon; + return returnfloat; + } nfcommon->setGroupingUsed(FALSE); UnicodeString stringValue; @@ -383,7 +474,10 @@ void NumberFormatRegressionTest::Test4071492 (void) double x = 0.00159999; UErrorCode status = U_ZERO_ERROR; NumberFormat *nf = NumberFormat::createInstance(status); - failure(status, "NumberFormat::createInstance", Locale::getUS()); + if (failure(status, "NumberFormat::createInstance", Locale::getUS(), TRUE)) { + delete nf; + return; + } nf->setMaximumFractionDigits(4); UnicodeString out; FieldPosition pos(FieldPosition::DONT_CARE); @@ -408,20 +502,19 @@ void NumberFormatRegressionTest::Test4086575(void) // TODO: There is not a good way to find out that the creation of this number format has // failed. Major rewiring of format construction proposed. if(U_FAILURE(status)) { - errln("Something is wrong with French number format - it should not fallback. Exitting"); + dataerrln("Something is wrong with French number format - it should not fallback. Exitting - %s", u_errorName(status)); delete nf1; return; } failure(status, "NumberFormat::createInstance", Locale::getFrance()); - + // C++ workaround to make sure cast works - // Wouldn't dynamic_cast be great? - if(nf1->getDynamicClassID() != DecimalFormat::getStaticClassID()) { + DecimalFormat *nf = dynamic_cast(nf1); + if(nf == NULL) { errln("NumberFormat::createInstance returned incorrect type."); return; } - DecimalFormat *nf = (DecimalFormat*) nf1; UnicodeString temp; logln("nf toPattern1: " + nf->toPattern(temp)); logln("nf toLocPattern1: " + nf->toLocalizedPattern(temp)); @@ -488,7 +581,7 @@ void NumberFormatRegressionTest::Test4068693(void) UErrorCode status = U_ZERO_ERROR; DecimalFormat *df = new DecimalFormat(status); if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + errcheckln(status, "Error creating DecimalFormat: %s", u_errorName(status)); delete df; return; } @@ -544,7 +637,7 @@ void NumberFormatRegressionTest::Test4087251 (void) UErrorCode status = U_ZERO_ERROR; DecimalFormat *df = new DecimalFormat(status); if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + errcheckln(status, "Error creating DecimalFormat: %s", u_errorName(status)); delete df; return; } @@ -610,7 +703,7 @@ void NumberFormatRegressionTest::Test4090504 (void) UErrorCode status = U_ZERO_ERROR; DecimalFormat *df = new DecimalFormat(status); if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + errcheckln(status, "Error creating DecimalFormat: %s", u_errorName(status)); delete df; return; } @@ -622,8 +715,8 @@ void NumberFormatRegressionTest::Test4090504 (void) df->setMaximumFractionDigits(i); //sb = new StringBuffer(""); fp.setField(0); - logln(" getMaximumFractionDigits() = " + i); - logln(" formated: " + df->format(d, sb, fp)); + logln(UnicodeString(" getMaximumFractionDigits() = ") + i); + logln(UnicodeString(" formated: ") + df->format(d, sb, fp)); } /*} catch (Exception foo) { errln("Bug 4090504 regression test failed. Message : " + foo.getMessage()); @@ -639,7 +732,7 @@ void NumberFormatRegressionTest::Test4095713 (void) UErrorCode status = U_ZERO_ERROR; DecimalFormat *df = new DecimalFormat(status); if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + errcheckln(status, "Error creating DecimalFormat: %s", u_errorName(status)); delete df; return; } @@ -666,7 +759,7 @@ void NumberFormatRegressionTest::Test4092561 (void) UErrorCode status = U_ZERO_ERROR; DecimalFormat *df = new DecimalFormat(status); if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + errcheckln(status, "Error creating DecimalFormat: %s", u_errorName(status)); delete df; return; } @@ -692,7 +785,7 @@ void NumberFormatRegressionTest::Test4092480 (void) UErrorCode status = U_ZERO_ERROR; DecimalFormat *dfFoo = new DecimalFormat(UnicodeString("000"), status); if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + errcheckln(status, "Error creating DecimalFormat: %s", u_errorName(status)); delete dfFoo; return; } @@ -748,15 +841,15 @@ void NumberFormatRegressionTest::Test4087244 (void) { Locale *de = new Locale(loc); NumberFormat *nf = NumberFormat::createCurrencyInstance(*de, status); if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + dataerrln("Error creating DecimalFormat: %s", u_errorName(status)); delete nf; return; } - if (nf->getDynamicClassID() != DecimalFormat::getStaticClassID()) { + DecimalFormat *df = dynamic_cast(nf); + if(df == NULL) { errln("expected DecimalFormat!"); return; } - DecimalFormat *df = (DecimalFormat*) nf; const DecimalFormatSymbols *sym = df->getDecimalFormatSymbols(); UnicodeString decSep = sym->getSymbol(DecimalFormatSymbols::kDecimalSeparatorSymbol); UnicodeString monSep = sym->getSymbol(DecimalFormatSymbols::kMonetarySeparatorSymbol); @@ -794,28 +887,28 @@ void NumberFormatRegressionTest::Test4070798 (void) /* User error : String expectedDefault = "-5\u00a0789,987"; - String expectedCurrency = "5\u00a0789,98 F"; + String expectedCurrency = "5\u00a0789,98\u00a0F"; String expectedPercent = "-578\u00a0998%"; */ UChar chars1 [] = { 0x2d, 0x35, 0x00a0, 0x37, 0x38, 0x39, 0x2c, 0x39, 0x38, 0x38 }; UChar chars2 [] = { - 0x35, 0x00a0, 0x37, 0x38, 0x39, 0x2c, 0x39, 0x39, 0x20, 0x46 + 0x35, 0x00a0, 0x37, 0x38, 0x39, 0x2c, 0x39, 0x39, 0x00a0, 0x46 }; UChar chars3 [] = { - 0x2d, 0x35, 0x37, 0x38, 0x00a0, 0x39, 0x39, 0x39, 0x25 + 0x2d, 0x35, 0x37, 0x38, 0x00a0, 0x39, 0x39, 0x39, 0x00a0, 0x25 }; UnicodeString expectedDefault(chars1, 10, 10); UnicodeString expectedCurrency(chars2, 10, 10); - UnicodeString expectedPercent(chars3, 9, 9); + UnicodeString expectedPercent(chars3, 10, 10); UErrorCode status = U_ZERO_ERROR; char loc[256]={0}; int len = uloc_canonicalize("fr_FR_PREEURO", loc, 256, &status); formatter = NumberFormat::createInstance(Locale(loc), status); if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + dataerrln("Error creating DecimalFormat: %s", u_errorName(status)); delete formatter; return; } @@ -870,25 +963,28 @@ void NumberFormatRegressionTest::Test4071005 (void) UnicodeString tempString; /* User error : String expectedDefault = "-5\u00a0789,987"; - String expectedCurrency = "5\u00a0789,98 $"; + String expectedCurrency = "5\u00a0789,98\u00a0$"; String expectedPercent = "-578\u00a0998%"; */ UChar chars1 [] = { 0x2d, 0x35, 0x00a0, 0x37, 0x38, 0x39, 0x2c, 0x39, 0x38, 0x38 }; UChar chars2 [] = { - 0x35, 0x00a0, 0x37, 0x38, 0x39, 0x2c, 0x39, 0x39, 0x20, 0x24 + 0x35, 0x00a0, 0x37, 0x38, 0x39, 0x2c, 0x39, 0x39, 0x00a0, 0x24 }; UChar chars3 [] = { - 0x2d, 0x35, 0x37, 0x38, 0x00a0, 0x39, 0x39, 0x39, 0x25 + 0x2d, 0x35, 0x37, 0x38, 0x00a0, 0x39, 0x39, 0x39, 0x00a0, 0x25 }; UnicodeString expectedDefault(chars1, 10, 10); UnicodeString expectedCurrency(chars2, 10, 10); - UnicodeString expectedPercent(chars3, 9, 9); + UnicodeString expectedPercent(chars3, 10, 10); UErrorCode status = U_ZERO_ERROR; formatter = NumberFormat::createInstance(Locale::getCanadaFrench(), status); - failure(status, "NumberFormat::createNumberInstance", Locale::getCanadaFrench()); + if (failure(status, "NumberFormat::createNumberInstance", Locale::getCanadaFrench(), TRUE)){ + delete formatter; + return; + }; tempString = formatter->format (-5789.9876, tempString); if (tempString == expectedDefault) { @@ -939,18 +1035,24 @@ void NumberFormatRegressionTest::Test4071014 (void) UnicodeString tempString; /* user error : String expectedDefault = "-5.789,987"; - String expectedCurrency = "5.789,98 DM"; + String expectedCurrency = "5.789,98 DEM"; String expectedPercent = "-578.998%"; */ UnicodeString expectedDefault("-5.789,988"); - UnicodeString expectedCurrency("5.789,99 DM"); - UnicodeString expectedPercent("-578.999%"); + UnicodeString expectedCurrency("5.789,99\\u00A0DEM"); + UnicodeString expectedPercent("-578.999\\u00A0%"); + + expectedCurrency = expectedCurrency.unescape(); + expectedPercent = expectedPercent.unescape(); UErrorCode status = U_ZERO_ERROR; char loc[256]={0}; uloc_canonicalize("de_DE_PREEURO", loc, 256, &status); formatter = NumberFormat::createInstance(Locale(loc), status); - failure(status, "NumberFormat::createNumberInstance", loc); + if (failure(status, "NumberFormat::createNumberInstance", loc, TRUE)){ + delete formatter; + return; + } tempString.remove(); tempString = formatter->format (-5789.9876, tempString); @@ -1001,11 +1103,11 @@ void NumberFormatRegressionTest::Test4071859 (void) UnicodeString tempString; /* user error : String expectedDefault = "-5.789,987"; - String expectedCurrency = "-L. 5.789,98"; + String expectedCurrency = "-L.\\u00A05.789,98"; String expectedPercent = "-578.998%"; */ UnicodeString expectedDefault("-5.789,988"); - UnicodeString expectedCurrency("-\\u20A4 5.790"); + UnicodeString expectedCurrency("-ITL\\u00A05.790", -1, US_INV); UnicodeString expectedPercent("-578.999%"); expectedCurrency = expectedCurrency.unescape(); @@ -1013,7 +1115,10 @@ void NumberFormatRegressionTest::Test4071859 (void) char loc[256]={0}; uloc_canonicalize("it_IT_PREEURO", loc, 256, &status); formatter = NumberFormat::createInstance(Locale(loc), status); - failure(status, "NumberFormat::createNumberInstance"); + if (failure(status, "NumberFormat::createNumberInstance", TRUE)){ + delete formatter; + return; + }; tempString = formatter->format (-5789.9876, tempString); if (tempString == expectedDefault) { @@ -1061,26 +1166,27 @@ void NumberFormatRegressionTest::Test4093610(void) { UErrorCode status = U_ZERO_ERROR; DecimalFormat *df = new DecimalFormat("#0.#", status); - failure(status, "new DecimalFormat"); - UnicodeString s("12.4"); - roundingTest(df, 12.35, s); - roundingTest(df, 12.45, s); - s = "12.5"; - roundingTest(df, 12.452,s); - s = "12.6"; - roundingTest(df, 12.55, s); - roundingTest(df, 12.65, s); - s = "12.7"; - roundingTest(df, 12.652,s); - s = "12.8"; - roundingTest(df, 12.75, s); - roundingTest(df, 12.752,s); - roundingTest(df, 12.85, s); - s = "12.9"; - roundingTest(df, 12.852,s); - s = "13"; - roundingTest(df, 12.95, s); - roundingTest(df, 12.952,s); + if (!failure(status, "new DecimalFormat")) { + UnicodeString s("12.4"); + roundingTest(df, 12.35, s); + roundingTest(df, 12.45, s); + s = "12.5"; + roundingTest(df, 12.452,s); + s = "12.6"; + roundingTest(df, 12.55, s); + roundingTest(df, 12.65, s); + s = "12.7"; + roundingTest(df, 12.652,s); + s = "12.8"; + roundingTest(df, 12.75, s); + roundingTest(df, 12.752,s); + roundingTest(df, 12.85, s); + s = "12.9"; + roundingTest(df, 12.852,s); + s = "13"; + roundingTest(df, 12.95, s); + roundingTest(df, 12.952,s); + } delete df; } @@ -1102,6 +1208,12 @@ void NumberFormatRegressionTest::Test4098741(void) //try { UErrorCode status = U_ZERO_ERROR; NumberFormat *fmt = NumberFormat::createPercentInstance(status); + if (U_FAILURE(status)) { + dataerrln("Error calling NumberFormat::createPercentInstance"); + delete fmt; + return; + } + fmt->setMaximumFractionDigits(20); UnicodeString temp; logln(fmt->format(.001, temp)); @@ -1121,7 +1233,7 @@ void NumberFormatRegressionTest::Test4074454(void) UErrorCode status = U_ZERO_ERROR; DecimalFormat *fmt = new DecimalFormat("#,#00.00;-#.#", status); if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + errcheckln(status, "Error creating DecimalFormat: %s", u_errorName(status)); delete fmt; return; } @@ -1133,7 +1245,7 @@ void NumberFormatRegressionTest::Test4074454(void) FieldPosition pos(FieldPosition::DONT_CARE); tempString = newFmt->format(3456.78, tempString, pos); if (tempString != UnicodeString("3,456.78 p'ieces")) - errln("Failed! 3456.78 p'ieces expected, but got : " + tempString); + dataerrln("Failed! 3456.78 p'ieces expected, but got : " + tempString); /*} catch (Exception foo) { errln("An exception was thrown for any inconsistent negative pattern."); }*/ @@ -1190,13 +1302,13 @@ void NumberFormatRegressionTest::Test4101481(void) UErrorCode status = U_ZERO_ERROR; DecimalFormat *sdf = new DecimalFormat(UnicodeString("#,##0"), status); if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + errcheckln(status, "Error creating DecimalFormat: %s", u_errorName(status)); delete sdf; return; } failure(status, "new DecimalFormat"); if (sdf->getMinimumIntegerDigits() != 1) - errln("Minimum integer digits : " + sdf->getMinimumIntegerDigits()); + errln(UnicodeString("Minimum integer digits : ") + sdf->getMinimumIntegerDigits()); delete sdf; } /* @bug 4052223 (API addition request A27) @@ -1208,7 +1320,7 @@ void NumberFormatRegressionTest::Test4052223(void) UErrorCode status = U_ZERO_ERROR; DecimalFormat *fmt = new DecimalFormat(UnicodeString("#,#00.00"), status); if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + errcheckln(status, "Error creating DecimalFormat: %s", u_errorName(status)); delete fmt; return; } @@ -1292,7 +1404,7 @@ void NumberFormatRegressionTest::Test4108738(void) failure(status, "new DecimalFormatSymbols"); DecimalFormat *df = new DecimalFormat("#,##0.###", syms, status); if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + errcheckln(status, "Error creating DecimalFormat: %s", u_errorName(status)); delete df; return; } @@ -1331,7 +1443,7 @@ void NumberFormatRegressionTest::Test4106658(void) UErrorCode status = U_ZERO_ERROR; DecimalFormat *df = new DecimalFormat(status); // Corrected; see 4147706 if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + errcheckln(status, "Error creating DecimalFormat: %s", u_errorName(status)); delete df; return; } @@ -1342,7 +1454,7 @@ void NumberFormatRegressionTest::Test4106658(void) UnicodeString temp; FieldPosition pos(FieldPosition::DONT_CARE); -#if defined(U_HPUX) +#if U_PLATFORM == U_PF_HPUX d1 = 0.0 * -1.0; // old HPUX compiler ignores volatile keyword #else d1 *= -1.0; // Some compilers have a problem with defining -0.0 @@ -1367,7 +1479,7 @@ void NumberFormatRegressionTest::Test4106662(void) UErrorCode status = U_ZERO_ERROR; DecimalFormat *df = new DecimalFormat(status); if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + errcheckln(status, "Error creating DecimalFormat: %s", u_errorName(status)); delete df; return; } @@ -1401,7 +1513,7 @@ void NumberFormatRegressionTest::Test4114639(void) UErrorCode status = U_ZERO_ERROR; NumberFormat *format = NumberFormat::createInstance(status); if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + dataerrln("Error creating DecimalFormat: %s", u_errorName(status)); delete format; return; } @@ -1426,7 +1538,7 @@ void NumberFormatRegressionTest::Test4106664(void) UErrorCode status = U_ZERO_ERROR; DecimalFormat *df = new DecimalFormat(status); if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + errcheckln(status, "Error creating DecimalFormat: %s", u_errorName(status)); delete df; return; } @@ -1460,7 +1572,7 @@ void NumberFormatRegressionTest::Test4106667(void) UErrorCode status = U_ZERO_ERROR; DecimalFormat *df = new DecimalFormat(status); if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + errcheckln(status, "Error creating DecimalFormat: %s", u_errorName(status)); delete df; return; } @@ -1473,7 +1585,7 @@ void NumberFormatRegressionTest::Test4106667(void) FieldPosition pos(FieldPosition::DONT_CARE); logln("pattern: \"" + df->toPattern(temp) + "\""); -#if defined(U_HPUX) +#if U_PLATFORM == U_PF_HPUX d = 0.0 * -1.0; // old HPUX compiler ignores volatile keyword #else d *= -1.0; // Some compilers have a problem with defining -0.0 @@ -1489,7 +1601,7 @@ void NumberFormatRegressionTest::Test4106667(void) /* @bug 4110936 * DecimalFormat.setMaximumIntegerDigits() works incorrectly. */ -#ifdef OS390 +#if U_PLATFORM == U_PF_OS390 # define MAX_INT_DIGITS 70 #else # define MAX_INT_DIGITS 128 @@ -1500,7 +1612,7 @@ void NumberFormatRegressionTest::Test4110936(void) UErrorCode status = U_ZERO_ERROR; NumberFormat *nf = NumberFormat::createInstance(status); if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + dataerrln("Error creating DecimalFormat: %s", u_errorName(status)); delete nf; return; } @@ -1508,7 +1620,7 @@ void NumberFormatRegressionTest::Test4110936(void) nf->setMaximumIntegerDigits(MAX_INT_DIGITS); logln("setMaximumIntegerDigits(MAX_INT_DIGITS)"); if (nf->getMaximumIntegerDigits() != MAX_INT_DIGITS) - errln("getMaximumIntegerDigits() returns " + + errln(UnicodeString("getMaximumIntegerDigits() returns ") + nf->getMaximumIntegerDigits()); delete nf; @@ -1532,15 +1644,21 @@ void NumberFormatRegressionTest::Test4122840(void) NULL/*"java.text.resources.LocaleElements"*/, locales[i], status); failure(status, "new ResourceBundle"); - ResourceBundle numPat = rb->get("NumberPatterns", status); - failure(status, "new ResourceBundle(NumberPatterns)"); - // + ResourceBundle numPat = rb->getWithFallback("NumberElements", status); + failure(status, "rb.get(NumberElements)"); + numPat = numPat.getWithFallback("latn",status); + failure(status, "rb.get(latn)"); + numPat = numPat.getWithFallback("patterns",status); + failure(status, "rb.get(patterns)"); + numPat = numPat.getWithFallback("currencyFormat",status); + failure(status, "rb.get(currencyFormat)"); + // // Get the currency pattern for this locale. We have to fish it // out of the ResourceBundle directly, since DecimalFormat.toPattern // will return the localized symbol, not \00a4 // - UnicodeString pattern = numPat.getStringEx(1, status); - failure(status, "rb->getStringArray"); + UnicodeString pattern = numPat.getString(status); + failure(status, "rb->getString()"); UChar fo[] = { 0x00A4 }; UnicodeString foo(fo, 1, 1); @@ -1594,6 +1712,21 @@ void NumberFormatRegressionTest::Test4122840(void) DecimalFormat *fmt2 = new DecimalFormat(buf, *symbols, status); failure(status, "new DecimalFormat"); + // Get the currency (if there is one) so we can set the rounding and fraction + const UChar *currency = fmt1->getCurrency(); + if (*currency != 0) { + double rounding = ucurr_getRoundingIncrement(currency, &status); + int32_t frac = ucurr_getDefaultFractionDigits(currency, &status); + if (U_SUCCESS(status)) { + fmt2->setRoundingIncrement(rounding); + fmt2->setMinimumFractionDigits(frac); + fmt2->setMaximumFractionDigits(frac); + } + else { + failure(status, "Fetching currency rounding/fractions"); + } + } + UnicodeString result2; fmt2->format(1.111, result2, pos); @@ -1620,7 +1753,7 @@ void NumberFormatRegressionTest::Test4125885(void) double rate = 12.34; DecimalFormat *formatDec = new DecimalFormat ("000.00", status); if(U_FAILURE(status)) { - errln("Error creating DecimalFormat: %s", u_errorName(status)); + errcheckln(status, "Error creating DecimalFormat: %s", u_errorName(status)); delete formatDec; return; } @@ -1653,22 +1786,22 @@ void NumberFormatRegressionTest::Test4134034(void) { UErrorCode status = U_ZERO_ERROR; DecimalFormat *nf = new DecimalFormat("##,###,###.00", status); - failure(status, "new DecimalFormat"); - - UnicodeString f; - FieldPosition pos(FieldPosition::DONT_CARE); - f = nf->format(9.02, f, pos); - if (f == UnicodeString("9.02")) - logln(f + " ok"); - else - errln("9.02 -> " + f + "; want 9.02"); + if (!failure(status, "new DecimalFormat")) { + UnicodeString f; + FieldPosition pos(FieldPosition::DONT_CARE); + f = nf->format(9.02, f, pos); + if (f == UnicodeString("9.02")) + logln(f + " ok"); + else + errln("9.02 -> " + f + "; want 9.02"); - f.remove(); - f = nf->format((int32_t)0, f, pos); - if (f == UnicodeString(".00")) - logln(f + " ok"); - else - errln("0 -> " + f + "; want .00"); + f.remove(); + f = nf->format((int32_t)0, f, pos); + if (f == UnicodeString(".00")) + logln(f + " ok"); + else + errln("0 -> " + f + "; want .00"); + } delete nf; } @@ -1712,15 +1845,16 @@ void NumberFormatRegressionTest::Test4134300(void) { UnicodeString result; UErrorCode status = U_ZERO_ERROR; DecimalFormat *df = new DecimalFormat(DATA[i], status); - failure(status, "new DecimalFormat"); - FieldPosition pos(FieldPosition::DONT_CARE); - result = df->format(1.2, result, pos); - if (result != DATA[i+1]) { - errln("Fail: 1.2 x " + DATA[i] + " = " + result + - "; want " + DATA[i+1]); - } - else { - logln("Ok: 1.2 x " + DATA[i] + " = " + result); + if (!failure(status, "new DecimalFormat")) { + FieldPosition pos(FieldPosition::DONT_CARE); + result = df->format(1.2, result, pos); + if (result != DATA[i+1]) { + errln("Fail: 1.2 x " + DATA[i] + " = " + result + + "; want " + DATA[i+1]); + } + else { + logln("Ok: 1.2 x " + DATA[i] + " = " + result); + } } delete df; @@ -1737,16 +1871,17 @@ void NumberFormatRegressionTest::Test4140009(void) DecimalFormatSymbols *syms = new DecimalFormatSymbols(Locale::getEnglish(), status); failure(status, "new DecimalFormatSymbols"); DecimalFormat *f = new DecimalFormat(UnicodeString(""), syms, status); - failure(status, "new DecimalFormat"); - UnicodeString s; - FieldPosition pos(FieldPosition::DONT_CARE); - s = f->format(123.456, s, pos); - if (s != UnicodeString("123.456")) - errln("Fail: Format empty pattern x 123.456 => " + s); - s.remove(); - s = f->format(-123.456, s, pos); - if (s != UnicodeString("-123.456")) - errln("Fail: Format empty pattern x -123.456 => " + s); + if (!failure(status, "new DecimalFormat")) { + UnicodeString s; + FieldPosition pos(FieldPosition::DONT_CARE); + s = f->format(123.456, s, pos); + if (s != UnicodeString("123.456")) + errln("Fail: Format empty pattern x 123.456 => " + s); + s.remove(); + s = f->format(-123.456, s, pos); + if (s != UnicodeString("-123.456")) + errln("Fail: Format empty pattern x -123.456 => " + s); + } delete f; } @@ -1777,13 +1912,16 @@ void NumberFormatRegressionTest::Test4145457() { //try { UErrorCode status = U_ZERO_ERROR; NumberFormat *nff = NumberFormat::createInstance(status); - failure(status, "NumberFormat::createInstance"); - if(nff->getDynamicClassID() != DecimalFormat::getStaticClassID()) { + if (failure(status, "NumberFormat::createInstance", TRUE)){ + delete nff; + return; + }; + DecimalFormat *nf = dynamic_cast(nff); + if(nf == NULL) { errln("DecimalFormat needed to continue"); return; } - DecimalFormat *nf = (DecimalFormat*)nff; DecimalFormatSymbols *sym = (DecimalFormatSymbols*) nf->getDecimalFormatSymbols(); sym->setSymbol(DecimalFormatSymbols::kDecimalSeparatorSymbol, (UChar)/*'\''*/0x0027); nf->setDecimalFormatSymbols(*sym); @@ -1812,6 +1950,7 @@ void NumberFormatRegressionTest::Test4145457() { out2 = nf->format(pi, out2, pos); UnicodeString pat2; pat2 = nf->toPattern(pat2); + pp.setIndex(0); nf->parse(out2, num, pp); double val2 = num.getDouble(); @@ -1854,16 +1993,17 @@ void NumberFormatRegressionTest::Test4147295(void) UnicodeString pattern("#,###"); logln("Applying pattern \"" + pattern + "\""); sdf->applyPattern(pattern, status); - failure(status, "sdf->applyPattern"); - int minIntDig = sdf->getMinimumIntegerDigits(); - if (minIntDig != 0) { - errln("Test failed"); - errln(" Minimum integer digits : " + minIntDig); - UnicodeString temp; - errln(" new pattern: " + sdf->toPattern(temp)); - } else { - logln("Test passed"); - logln(" Minimum integer digits : " + minIntDig); + if (!failure(status, "sdf->applyPattern")) { + int minIntDig = sdf->getMinimumIntegerDigits(); + if (minIntDig != 0) { + errln("Test failed"); + errln(UnicodeString(" Minimum integer digits : ") + minIntDig); + UnicodeString temp; + errln(UnicodeString(" new pattern: ") + sdf->toPattern(temp)); + } else { + logln("Test passed"); + logln(UnicodeString(" Minimum integer digits : ") + minIntDig); + } } delete sdf; } @@ -1879,28 +2019,29 @@ void NumberFormatRegressionTest::Test4147706(void) DecimalFormat *df = new DecimalFormat("#,##0.0##", status); failure(status, "new DecimalFormat"); DecimalFormatSymbols *syms = new DecimalFormatSymbols(Locale::getEnglish(), status); - failure(status, "new DecimalFormatSymbols"); - UnicodeString f1; - UnicodeString f2, temp; - FieldPosition pos(FieldPosition::DONT_CARE); - volatile double d1 = 0.0; // volatile to prevent code optimization - double d2 = -0.0001; + if (!failure(status, "new DecimalFormatSymbols")) { + UnicodeString f1; + UnicodeString f2, temp; + FieldPosition pos(FieldPosition::DONT_CARE); + volatile double d1 = 0.0; // volatile to prevent code optimization + double d2 = -0.0001; -#if defined(U_HPUX) - d1 = 0.0 * -1.0; // old HPUX compiler ignores volatile keyword +#if U_PLATFORM == U_PF_HPUX + d1 = 0.0 * -1.0; // old HPUX compiler ignores volatile keyword #else - d1 *= -1.0; // Some compilers have a problem with defining -0.0 + d1 *= -1.0; // Some compilers have a problem with defining -0.0 #endif - df->adoptDecimalFormatSymbols(syms); - f1 = df->format(d1, f1, pos); - f2 = df->format(d2, f2, pos); - if (f1 != UnicodeString("-0.0")) { - errln(UnicodeString("") + d1 + UnicodeString(" x \"") + df->toPattern(temp) + "\" is formatted as \"" + f1 + '"'); - } - if (f2 != UnicodeString("-0.0")) { - errln(UnicodeString("") + d2 + UnicodeString(" x \"") + df->toPattern(temp) + "\" is formatted as \"" + f2 + '"'); + df->adoptDecimalFormatSymbols(syms); + f1 = df->format(d1, f1, pos); + f2 = df->format(d2, f2, pos); + if (f1 != UnicodeString("-0.0")) { + errln(UnicodeString("") + d1 + UnicodeString(" x \"") + df->toPattern(temp) + "\" is formatted as \"" + f1 + '"'); + } + if (f2 != UnicodeString("-0.0")) { + errln(UnicodeString("") + d2 + UnicodeString(" x \"") + df->toPattern(temp) + "\" is formatted as \"" + f2 + '"'); + } } - + delete df; } @@ -1932,7 +2073,7 @@ NumberFormatRegressionTest::Test4162198(void) UErrorCode status = U_ZERO_ERROR; NumberFormat *f = NumberFormat::createInstance(status); if(U_FAILURE(status)) { - errln("Couldn't create number format"); + dataerrln("Couldn't create number format - %s", u_errorName(status)); return; } f->setMaximumFractionDigits(INT32_MAX); @@ -1976,7 +2117,7 @@ NumberFormatRegressionTest::Test4162852(void) NumberFormat *f = (i == 0) ? NumberFormat::createInstance(status) : NumberFormat::createPercentInstance(status); if(U_FAILURE(status)) { - errln("Couldn't create number format"); + dataerrln("Couldn't create number format - %s", u_errorName(status)); return; } double d = 0.0; @@ -1991,7 +2132,7 @@ NumberFormatRegressionTest::Test4162852(void) logln(UnicodeString("") + d + " -> " + '"' + s + '"' + " -> " + e); -#if (defined(OS390) && !defined(IEEE_754)) || defined(OS400) +#if (U_PLATFORM == U_PF_OS390 && !defined(IEEE_754)) || U_PLATFORM == U_PF_OS400 if (e != 0.0) { #else if (e != 0.0 || 1.0/e > 0.0) { @@ -2012,7 +2153,10 @@ static double _u_abs(double a) { return a<0?-a:a; } void NumberFormatRegressionTest::Test4167494(void) { UErrorCode status = U_ZERO_ERROR; NumberFormat *fmt = NumberFormat::createInstance(Locale::getUS(), status); - failure(status, "NumberFormat::createInstance"); + if (failure(status, "NumberFormat::createInstance", TRUE)){ + delete fmt; + return; + }; double a = DBL_MAX * 0.99; // DBL_MAX itself overflows to +Inf UnicodeString s; @@ -2045,12 +2189,15 @@ void NumberFormatRegressionTest::Test4167494(void) { void NumberFormatRegressionTest::Test4170798(void) { UErrorCode status = U_ZERO_ERROR; NumberFormat *nf = NumberFormat::createInstance(Locale::getUS(), status); - failure(status, "NumberFormat::createInstance"); - if(nf->getDynamicClassID() != DecimalFormat::getStaticClassID()) { + if (failure(status, "NumberFormat::createInstance", TRUE)){ + delete nf; + return; + }; + DecimalFormat *df = dynamic_cast(nf); + if(df == NULL) { errln("DecimalFormat needed to continue"); return; } - DecimalFormat *df = (DecimalFormat*) nf; df->setParseIntegerOnly(TRUE); Formattable n; ParsePosition pos(0); @@ -2083,12 +2230,13 @@ void NumberFormatRegressionTest::Test4176114(void) { UnicodeString s; for (int i=0; i " + - s + ", want " + exp); + if (!failure(status, "DecimalFormat constructor")) { + df.toPattern(s); + UnicodeString exp(DATA[i+1]); + if (s != exp) { + errln(UnicodeString("FAIL: ") + DATA[i] + " -> " + + s + ", want " + exp); + } } } } @@ -2115,21 +2263,22 @@ void NumberFormatRegressionTest::Test4179818(void) { DecimalFormatSymbols sym(Locale::getUS(), status); failure(status, "Construct DecimalFormatSymbols"); DecimalFormat fmt("#", sym, status); - failure(status, "Construct DecimalFormat"); - for (int i=0; i \"" + pat + "\" -> \"" + f2.toPattern(p) + "\""); + } else { + UnicodeString l, p; + logln(UnicodeString("PASS: ") + type[j] + avail[i].getDisplayName(l) + + " -> \"" + pat + + "\""); } // Test toLocalizedPattern/applyLocalizedPattern round trip @@ -2287,6 +2443,17 @@ void NumberFormatRegressionTest::Test4212072(void) { if (U_FAILURE(status)) { continue; } + + // Make sure we set the currency attributes appropriately + if (j == 1) { // Currency format + f2.setCurrency(f2.getCurrency(), status); + } + failure(status, + UnicodeString("setCurrency() for (") + pat + ")", avail[i]); + if (U_FAILURE(status)) { + continue; + } + if (*df != f2) { UnicodeString l, p; errln(UnicodeString("FAIL: ") + type[j] + avail[i].getDisplayName(l) + @@ -2309,7 +2476,10 @@ void NumberFormatRegressionTest::Test4212072(void) { void NumberFormatRegressionTest::Test4216742(void) { UErrorCode status = U_ZERO_ERROR; DecimalFormat *fmt = (DecimalFormat*) NumberFormat::createInstance(Locale::getUS(), status); - failure(status, "createInstance", Locale::getUS()); + if (failure(status, "createInstance", Locale::getUS(), TRUE)){ + delete fmt; + return; + }; int32_t DATA[] = { INT32_MIN, INT32_MAX, -100000000, 100000000 }; int DATA_length = (int)(sizeof(DATA) / sizeof(DATA[0])); for (int i=0; i 0 != DATA[i] > 0) { + if ((d > 0) != (DATA[i] > 0)) { errln(UnicodeString("\"") + str + "\" parse(x " + fmt->getMultiplier() + ") => " + toString(num)); @@ -2348,7 +2518,10 @@ void NumberFormatRegressionTest::Test4217661(void) { int D_length = (int)(sizeof(D) / sizeof(D[0])); UErrorCode status = U_ZERO_ERROR; NumberFormat *fmt = NumberFormat::createInstance(Locale::getUS(), status); - failure(status, "createInstance", Locale::getUS()); + if (failure(status, "createInstance", Locale::getUS(), TRUE)){ + delete fmt; + return; + }; fmt->setMaximumFractionDigits(2); for (int i=0; isetMinimumFractionDigits(1); nf->setMaximumFractionDigits(1); double a = -0.09; @@ -2390,22 +2566,23 @@ void NumberFormatRegressionTest::Test4243011(void) { DecimalFormatSymbols sym(Locale::getUS(), status); failure(status, "DecimalFormatSymbols ct", Locale::getUS()); DecimalFormat fmt(UnicodeString("0."), sym, status); - failure(status, "DecimalFormat ct", Locale::getUS()); - - const double NUM[] = { -2.5, -1.5, -0.5, 0.5, 1.5, 2.5, 3.5, 4.5 }; - const char* STR[] = { "-2.", "-2.", "-0.", "0.", "2.", "2.", "4.", "4." }; - int32_t N = (int32_t)(sizeof(NUM) / sizeof(NUM[0])); - - for (int32_t i=0; iparse(numStr, val, status); + TEST_CHECK_STATUS(status); + TEST_ASSERT(Formattable::kDouble == val.getType()); + TEST_ASSERT(1000000000 == val.getInt64(status)); + TEST_CHECK_STATUS(status); + TEST_ASSERT(1000000000.6 == val.getDouble(status)); + TEST_CHECK_STATUS(status); + + numStr = "100000000000000001.1"; // approx 1E17, parses as a double rather + // than int64 because of the fraction + // even though int64 is more precise. + nf->parse(numStr, val, status); + TEST_CHECK_STATUS(status); + TEST_ASSERT(Formattable::kDouble == val.getType()); + TEST_ASSERT(100000000000000001LL == val.getInt64(status)); + TEST_CHECK_STATUS(status); + TEST_ASSERT(100000000000000000.0 == val.getDouble(status)); + TEST_CHECK_STATUS(status); + + numStr = "1E17"; // Parses with the internal decimal number having non-zero exponent + nf->parse(numStr, val, status); + TEST_CHECK_STATUS(status); + TEST_ASSERT(Formattable::kInt64 == val.getType()); + TEST_ASSERT(100000000000000000LL == val.getInt64()); + TEST_ASSERT(1.0E17 == val.getDouble(status)); + TEST_CHECK_STATUS(status); + + numStr = "9223372036854775807"; // largest int64_t + nf->parse(numStr, val, status); + TEST_CHECK_STATUS(status); + TEST_ASSERT(Formattable::kInt64 == val.getType()); + TEST_ASSERT(9223372036854775807LL == val.getInt64()); + // In the following check, note that a substantial range of integers will + // convert to the same double value. There are also platform variations + // in the rounding at compile time of double constants. + TEST_ASSERT(9223372036854775808.0 >= val.getDouble(status)); + TEST_ASSERT(9223372036854774700.0 <= val.getDouble(status)); + TEST_CHECK_STATUS(status); + + numStr = "-9223372036854775808"; // smallest int64_t + nf->parse(numStr, val, status); + TEST_CHECK_STATUS(status); + TEST_ASSERT(Formattable::kInt64 == val.getType()); + // TEST_ASSERT(-9223372036854775808LL == val.getInt64()); // Compiler chokes on constant. + TEST_ASSERT((int64_t)0x8000000000000000LL == val.getInt64()); + TEST_ASSERT(-9223372036854775808.0 == val.getDouble(status)); + TEST_CHECK_STATUS(status); + + numStr = "9223372036854775808"; // largest int64_t + 1 + nf->parse(numStr, val, status); + TEST_CHECK_STATUS(status); + TEST_ASSERT(Formattable::kDouble == val.getType()); + TEST_ASSERT(9223372036854775807LL == val.getInt64(status)); + TEST_ASSERT(status == U_INVALID_FORMAT_ERROR); + status = U_ZERO_ERROR; + TEST_ASSERT(9223372036854775810.0 == val.getDouble(status)); + TEST_CHECK_STATUS(status); + + numStr = "-9223372036854775809"; // smallest int64_t - 1 + nf->parse(numStr, val, status); + TEST_CHECK_STATUS(status); + TEST_ASSERT(Formattable::kDouble == val.getType()); + // TEST_ASSERT(-9223372036854775808LL == val.getInt64(status)); // spurious compiler warnings + TEST_ASSERT((int64_t)0x8000000000000000LL == val.getInt64(status)); + TEST_ASSERT(status == U_INVALID_FORMAT_ERROR); + status = U_ZERO_ERROR; + TEST_ASSERT(-9223372036854775810.0 == val.getDouble(status)); + TEST_CHECK_STATUS(status); + + // Test values near the limit of where doubles can represent all integers. + // The implementation strategy of getInt64() changes at this boundary. + // Strings to be parsed include a decimal fraction to force them to be + // parsed as doubles rather than ints. The fraction is discarded + // from the parsed double value because it is beyond what can be represented. + + status = U_ZERO_ERROR; + numStr = "9007199254740991.1"; // largest 53 bit int + nf->parse(numStr, val, status); + TEST_CHECK_STATUS(status); + // printf("getInt64() returns %lld\n", val.getInt64(status)); + TEST_ASSERT(Formattable::kDouble == val.getType()); + TEST_ASSERT(9007199254740991LL == val.getInt64(status)); + TEST_ASSERT(9007199254740991.0 == val.getDouble(status)); + TEST_CHECK_STATUS(status); + + status = U_ZERO_ERROR; + numStr = "9007199254740992.1"; // 54 bits for the int part. + nf->parse(numStr, val, status); + TEST_CHECK_STATUS(status); + TEST_ASSERT(Formattable::kDouble == val.getType()); + TEST_ASSERT(9007199254740992LL == val.getInt64(status)); + TEST_ASSERT(9007199254740992.0 == val.getDouble(status)); + TEST_CHECK_STATUS(status); + + status = U_ZERO_ERROR; + numStr = "9007199254740993.1"; // 54 bits for the int part. Double will round + nf->parse(numStr, val, status); // the ones digit, putting it up to ...994 + TEST_CHECK_STATUS(status); + TEST_ASSERT(Formattable::kDouble == val.getType()); + TEST_ASSERT(9007199254740993LL == val.getInt64(status)); + TEST_ASSERT(9007199254740994.0 == val.getDouble(status)); + TEST_CHECK_STATUS(status); + + delete nf; +} + +void NumberFormatRegressionTest::Test9109(void) { + UErrorCode status = U_ZERO_ERROR; + Formattable val; + ParsePosition pos; + DecimalFormat fmt("+##", status); + fmt.setLenient(TRUE); + + if (U_FAILURE(status)) { + dataerrln("Failed to create DecimalFormat with pattern '+##' - %s", u_errorName(status)); + } + + UnicodeString text("123"); + int32_t expected = 123; + int32_t expos = 3; + + fmt.parse(text, val, pos); + if (pos.getErrorIndex() >= 0) { + errln(UnicodeString("Parse failure at ") + pos.getErrorIndex()); + } else if (val.getLong() != 123) { + errln(UnicodeString("Incorrect parse result: ") + val.getLong() + " expected: " + expected); + } else if (pos.getIndex() != 3) { + errln(UnicodeString("Incorrect parse position: ") + pos.getIndex() + " expected: " + expos); + } +} + + +void NumberFormatRegressionTest::Test9780(void) { + UErrorCode status = U_ZERO_ERROR; + NumberFormat *nf = NumberFormat::createInstance(Locale::getUS(), status); + if (failure(status, "NumberFormat::createInstance", TRUE)){ + delete nf; + return; + }; + DecimalFormat *df = dynamic_cast(nf); + if(df == NULL) { + errln("DecimalFormat needed to continue"); + return; + } + df->setParseIntegerOnly(TRUE); + + { + Formattable n; + ParsePosition pos(0); + UnicodeString toParse("1,234",""); + df->parse(toParse, n, pos); + if (n.getType() != Formattable::kLong + || n.getLong() != 1234) { + errln(UnicodeString("FAIL: parse(\"") + toParse + UnicodeString("\") returns ") + toString(n)); + } + } + // should still work in lenient mode, just won't get fastpath + df->setLenient(TRUE); + { + Formattable n; + ParsePosition pos(0); + UnicodeString toParse("1,234",""); + df->parse(toParse, n, pos); + if (n.getType() != Formattable::kLong + || n.getLong() != 1234) { + errln(UnicodeString("FAIL: parse(\"") + toParse + UnicodeString("\") returns ") + toString(n)); + } + } + delete nf; +} + + +void NumberFormatRegressionTest::Test9677(void) { + static const UChar pattern[] = { 0x23,0x23,0x23,0x23,0x2E,0x23,0x23,0x23,0x23,0 }; // "####.####" + static const UChar positivePrefix[] = { 0x40,0 }; // "@" + static const UChar negativePrefix[] = { 0x6E,0 }; // "n" + static const UChar text[] = { 0x31,0x32,0x33,0x34,0x35,0x36,0x37,0x38,0x39,0 }; // 123456789 + static const UChar text2[] = { 0x6E, 0x31,0x32,0x33,0x34,0x35,0x36,0x37,0x38,0x39,0 }; // n123456789 + + UErrorCode status = U_ZERO_ERROR; + LocalUNumberFormatPointer f(unum_open(UNUM_DEFAULT, NULL, 0, "en_US", NULL, &status)); + if (U_FAILURE(status)) { + dataerrln("Failure opening unum_open"); + return; + } + + if (U_SUCCESS(status)) { + unum_applyPattern(f.getAlias(), FALSE, pattern, -1, NULL, &status); + unum_setTextAttribute(f.getAlias(), UNUM_POSITIVE_PREFIX, positivePrefix, -1, &status); + assertSuccess("setting attributes", status); + } + + if(U_SUCCESS(status)) { + int32_t n = unum_parse(f.getAlias(), text, -1, NULL, &status); + logln("unum_parse status %s, result %d\n", u_errorName(status), n); + + if(U_FAILURE(status)) { + logln("Got expected parse error %s\n", u_errorName(status)); + status = U_ZERO_ERROR; + } else { + errln("FAIL: unum_parse status %s, result %d - expected failure\n", u_errorName(status), n); + } + } + + if (U_SUCCESS(status)) { + unum_setTextAttribute(f.getAlias(), UNUM_POSITIVE_PREFIX, NULL, 0, &status); + assertSuccess("setting attributes", status); + logln("removed positive prefix"); + } + + if(U_SUCCESS(status)) { + int32_t n = unum_parse(f.getAlias(), text, -1, NULL, &status); + logln("unum_parse status %s, result %d\n", u_errorName(status), n); + + if(U_FAILURE(status)) { + errln("FAIL: with pos prefix removed, parse error %s\n", u_errorName(status)); + status = U_ZERO_ERROR; + } else { + if(n!=123456789) { + errln("FAIL: with pos prefix removed , unum_parse status %s, result %d expected 123456789\n", u_errorName(status), n); + } else { + logln("PASS: with pos prefix removed , unum_parse status %s, result %d expected 123456789\n", u_errorName(status),n); + } + } + } + + if(U_SUCCESS(status)) { + int32_t n = unum_parse(f.getAlias(), text2, -1, NULL, &status); + logln("unum_parse status %s, result %d\n", u_errorName(status), n); + + if(U_FAILURE(status)) { + logln("text2: Got expected parse error %s\n", u_errorName(status)); + status = U_ZERO_ERROR; + } else { + errln("FAIL: text2: unum_parse status %s, result %d - expected failure\n", u_errorName(status), n); + } + } + + if (U_SUCCESS(status)) { + unum_setTextAttribute(f.getAlias(), UNUM_NEGATIVE_PREFIX, negativePrefix, -1, &status); + assertSuccess("setting attributes", status); + logln("Set a different neg prefix prefix"); + } + + if(U_SUCCESS(status)) { + int32_t n = unum_parse(f.getAlias(), text2, -1, NULL, &status); + logln("unum_parse status %s, result %d\n", u_errorName(status), n); + + if(U_FAILURE(status)) { + errln("FAIL: with different neg prefix , parse error %s\n", u_errorName(status)); + status = U_ZERO_ERROR; + } else { +; + if(n!=-123456789) { + errln("FAIL: with different neg prefix , unum_parse status %s, result %d expected -123456789\n", u_errorName(status), n); + } else { + logln("PASS: with different neg prefix , unum_parse status %s, result %d expected -123456789\n", u_errorName(status), n); + } + } + } +} + + #endif /* #if !UCONFIG_NO_FORMATTING */