1 /********************************************************************
2  * COPYRIGHT:
3  * Copyright (c) 1997-2015, International Business Machines Corporation and
4  * others. All Rights Reserved.
5  ********************************************************************/
6 /* Modification History:
7 *   Date        Name        Description
8 *   07/15/99    helena      Ported to HPUX 10/11 CC.
9 */
10 
11 #include "unicode/utypes.h"
12 
13 #if !UCONFIG_NO_FORMATTING
14 
15 #include "numfmtst.h"
16 #include "unicode/dcfmtsym.h"
17 #include "unicode/decimfmt.h"
18 #include "unicode/localpointer.h"
19 #include "unicode/ucurr.h"
20 #include "unicode/ustring.h"
21 #include "unicode/measfmt.h"
22 #include "unicode/curramt.h"
23 #include "digitlst.h"
24 #include "textfile.h"
25 #include "tokiter.h"
26 #include "charstr.h"
27 #include "putilimp.h"
28 #include "winnmtst.h"
29 #include <float.h>
30 #include <string.h>
31 #include <stdlib.h>
32 #include "cstring.h"
33 #include "unicode/numsys.h"
34 #include "fmtableimp.h"
35 
36 //#define NUMFMTST_CACHE_DEBUG 1
37 #include "stdio.h" /* for sprintf */
38 // #include "iostream"   // for cout
39 
40 //#define NUMFMTST_DEBUG 1
41 
42 static const UChar EUR[] = {69,85,82,0}; // "EUR"
43 static const UChar ISO_CURRENCY_USD[] = {0x55, 0x53, 0x44, 0}; // "USD"
44 
45 
46 // *****************************************************************************
47 // class NumberFormatTest
48 // *****************************************************************************
49 
50 #define CHECK(status,str) if (U_FAILURE(status)) { errcheckln(status, UnicodeString("FAIL: ") + str + " - " + u_errorName(status)); return; }
51 #define CHECK_DATA(status,str) if (U_FAILURE(status)) { dataerrln(UnicodeString("FAIL: ") + str + " - " + u_errorName(status)); return; }
52 
runIndexedTest(int32_t index,UBool exec,const char * & name,char *)53 void NumberFormatTest::runIndexedTest( int32_t index, UBool exec, const char* &name, char* /*par*/ )
54 {
55   TESTCASE_AUTO_BEGIN;
56   TESTCASE_AUTO(TestCurrencySign);
57   TESTCASE_AUTO(TestCurrency);
58   TESTCASE_AUTO(TestParse);
59   TESTCASE_AUTO(TestRounding487);
60   TESTCASE_AUTO(TestQuotes);
61   TESTCASE_AUTO(TestExponential);
62   TESTCASE_AUTO(TestPatterns);
63 
64   // Upgrade to alphaWorks - liu 5/99
65   TESTCASE_AUTO(TestExponent);
66   TESTCASE_AUTO(TestScientific);
67   TESTCASE_AUTO(TestPad);
68   TESTCASE_AUTO(TestPatterns2);
69   TESTCASE_AUTO(TestSecondaryGrouping);
70   TESTCASE_AUTO(TestSurrogateSupport);
71   TESTCASE_AUTO(TestAPI);
72 
73   TESTCASE_AUTO(TestCurrencyObject);
74   TESTCASE_AUTO(TestCurrencyPatterns);
75   //TESTCASE_AUTO(TestDigitList);
76   TESTCASE_AUTO(TestWhiteSpaceParsing);
77   TESTCASE_AUTO(TestComplexCurrency);  // This test removed because CLDR no longer uses choice formats in currency symbols.
78   TESTCASE_AUTO(TestRegCurrency);
79   TESTCASE_AUTO(TestSymbolsWithBadLocale);
80   TESTCASE_AUTO(TestAdoptDecimalFormatSymbols);
81 
82   TESTCASE_AUTO(TestScientific2);
83   TESTCASE_AUTO(TestScientificGrouping);
84   TESTCASE_AUTO(TestInt64);
85 
86   TESTCASE_AUTO(TestPerMill);
87   TESTCASE_AUTO(TestIllegalPatterns);
88   TESTCASE_AUTO(TestCases);
89 
90   TESTCASE_AUTO(TestCurrencyNames);
91   TESTCASE_AUTO(TestCurrencyAmount);
92   TESTCASE_AUTO(TestCurrencyUnit);
93   TESTCASE_AUTO(TestCoverage);
94   TESTCASE_AUTO(TestJB3832);
95   TESTCASE_AUTO(TestHost);
96   TESTCASE_AUTO(TestHostClone);
97   TESTCASE_AUTO(TestCurrencyFormat);
98   TESTCASE_AUTO(TestRounding);
99   TESTCASE_AUTO(TestNonpositiveMultiplier);
100   TESTCASE_AUTO(TestNumberingSystems);
101   TESTCASE_AUTO(TestSpaceParsing);
102   TESTCASE_AUTO(TestMultiCurrencySign);
103   TESTCASE_AUTO(TestCurrencyFormatForMixParsing);
104   TESTCASE_AUTO(TestDecimalFormatCurrencyParse);
105   TESTCASE_AUTO(TestCurrencyIsoPluralFormat);
106   TESTCASE_AUTO(TestCurrencyParsing);
107   TESTCASE_AUTO(TestParseCurrencyInUCurr);
108   TESTCASE_AUTO(TestFormatAttributes);
109   TESTCASE_AUTO(TestFieldPositionIterator);
110   TESTCASE_AUTO(TestDecimal);
111   TESTCASE_AUTO(TestCurrencyFractionDigits);
112   TESTCASE_AUTO(TestExponentParse);
113   TESTCASE_AUTO(TestExplicitParents);
114   TESTCASE_AUTO(TestLenientParse);
115   TESTCASE_AUTO(TestAvailableNumberingSystems);
116   TESTCASE_AUTO(TestRoundingPattern);
117   TESTCASE_AUTO(Test9087);
118   TESTCASE_AUTO(TestFormatFastpaths);
119   TESTCASE_AUTO(TestFormattableSize);
120   TESTCASE_AUTO(TestUFormattable);
121   TESTCASE_AUTO(TestSignificantDigits);
122   TESTCASE_AUTO(TestShowZero);
123   TESTCASE_AUTO(TestCompatibleCurrencies);
124   TESTCASE_AUTO(TestBug9936);
125   TESTCASE_AUTO(TestParseNegativeWithFaLocale);
126   TESTCASE_AUTO(TestParseNegativeWithAlternateMinusSign);
127   TESTCASE_AUTO(TestCustomCurrencySignAndSeparator);
128   TESTCASE_AUTO(TestParseSignsAndMarks);
129   TESTCASE_AUTO(Test10419RoundingWith0FractionDigits);
130   TESTCASE_AUTO(Test10468ApplyPattern);
131   TESTCASE_AUTO(TestRoundingScientific10542);
132   TESTCASE_AUTO(TestZeroScientific10547);
133   TESTCASE_AUTO(TestAccountingCurrency);
134   TESTCASE_AUTO(TestEquality);
135   TESTCASE_AUTO(TestCurrencyUsage);
136   TESTCASE_AUTO(TestDoubleLimit11439);
137   TESTCASE_AUTO(TestFastPathConsistent11524);
138   TESTCASE_AUTO_END;
139 }
140 
141 // -------------------------------------
142 
143 // Test API (increase code coverage)
144 void
TestAPI(void)145 NumberFormatTest::TestAPI(void)
146 {
147   logln("Test API");
148   UErrorCode status = U_ZERO_ERROR;
149   NumberFormat *test = NumberFormat::createInstance("root", status);
150   if(U_FAILURE(status)) {
151     dataerrln("unable to create format object - %s", u_errorName(status));
152   }
153   if(test != NULL) {
154     test->setMinimumIntegerDigits(10);
155     test->setMaximumIntegerDigits(2);
156 
157     test->setMinimumFractionDigits(10);
158     test->setMaximumFractionDigits(2);
159 
160     UnicodeString result;
161     FieldPosition pos;
162     Formattable bla("Paja Patak"); // Donald Duck for non Serbian speakers
163     test->format(bla, result, pos, status);
164     if(U_SUCCESS(status)) {
165       errln("Yuck... Formatted a duck... As a number!");
166     } else {
167       status = U_ZERO_ERROR;
168     }
169 
170     result.remove();
171     int64_t ll = 12;
172     test->format(ll, result);
173     if (result != "12.00"){
174         errln("format int64_t error");
175     }
176 
177     ParsePosition ppos;
178     LocalPointer<CurrencyAmount> currAmt(test->parseCurrency("",ppos));
179     // old test for (U_FAILURE(status)) was bogus here, method does not set status!
180     if (ppos.getIndex()) {
181         errln("Parsed empty string as currency");
182     }
183 
184     delete test;
185   }
186 }
187 
188 class StubNumberFormat :public NumberFormat{
189 public:
StubNumberFormat()190     StubNumberFormat(){};
format(double,UnicodeString & appendTo,FieldPosition &) const191     virtual UnicodeString& format(double ,UnicodeString& appendTo,FieldPosition& ) const {
192         return appendTo;
193     }
format(int32_t,UnicodeString & appendTo,FieldPosition &) const194     virtual UnicodeString& format(int32_t ,UnicodeString& appendTo,FieldPosition& ) const {
195         return appendTo.append((UChar)0x0033);
196     }
format(int64_t number,UnicodeString & appendTo,FieldPosition & pos) const197     virtual UnicodeString& format(int64_t number,UnicodeString& appendTo,FieldPosition& pos) const {
198         return NumberFormat::format(number, appendTo, pos);
199     }
format(const Formattable &,UnicodeString & appendTo,FieldPosition &,UErrorCode &) const200     virtual UnicodeString& format(const Formattable& , UnicodeString& appendTo, FieldPosition& , UErrorCode& ) const {
201         return appendTo;
202     }
parse(const UnicodeString &,Formattable &,ParsePosition &) const203     virtual void parse(const UnicodeString& ,
204                     Formattable& ,
205                     ParsePosition& ) const {}
parse(const UnicodeString &,Formattable &,UErrorCode &) const206     virtual void parse( const UnicodeString& ,
207                         Formattable& ,
208                         UErrorCode& ) const {}
getDynamicClassID(void) const209     virtual UClassID getDynamicClassID(void) const {
210         static char classID = 0;
211         return (UClassID)&classID;
212     }
clone() const213     virtual Format* clone() const {return NULL;}
214 };
215 
216 void
TestCoverage(void)217 NumberFormatTest::TestCoverage(void){
218     StubNumberFormat stub;
219     UnicodeString agent("agent");
220     FieldPosition pos;
221     int64_t num = 4;
222     if (stub.format(num, agent, pos) != UnicodeString("agent3")){
223         errln("NumberFormat::format(int64, UnicodString&, FieldPosition&) should delegate to (int32, ,)");
224     };
225 }
226 
227 // Test various patterns
228 void
TestPatterns(void)229 NumberFormatTest::TestPatterns(void)
230 {
231     UErrorCode status = U_ZERO_ERROR;
232     DecimalFormatSymbols sym(Locale::getUS(), status);
233     if (U_FAILURE(status)) { errcheckln(status, "FAIL: Could not construct DecimalFormatSymbols - %s", u_errorName(status)); return; }
234 
235     const char* pat[]    = { "#.#", "#.", ".#", "#" };
236     int32_t pat_length = (int32_t)(sizeof(pat) / sizeof(pat[0]));
237     const char* newpat[] = { "#0.#", "#0.", "#.0", "#" };
238     const char* num[]    = { "0",   "0.", ".0", "0" };
239     for (int32_t i=0; i<pat_length; ++i)
240     {
241         status = U_ZERO_ERROR;
242         DecimalFormat fmt(pat[i], sym, status);
243         if (U_FAILURE(status)) { errln((UnicodeString)"FAIL: DecimalFormat constructor failed for " + pat[i]); continue; }
244         UnicodeString newp; fmt.toPattern(newp);
245         if (!(newp == newpat[i]))
246             errln((UnicodeString)"FAIL: Pattern " + pat[i] + " should transmute to " + newpat[i] +
247                   "; " + newp + " seen instead");
248 
249         UnicodeString s; (*(NumberFormat*)&fmt).format((int32_t)0, s);
250         if (!(s == num[i]))
251         {
252             errln((UnicodeString)"FAIL: Pattern " + pat[i] + " should format zero as " + num[i] +
253                   "; " + s + " seen instead");
254             logln((UnicodeString)"Min integer digits = " + fmt.getMinimumIntegerDigits());
255         }
256     }
257 }
258 
259 /*
260 icu_2_4::DigitList::operator== 0 0 2 icuuc24d.dll digitlst.cpp Doug
261 icu_2_4::DigitList::append 0 0 4 icuin24d.dll digitlst.h Doug
262 icu_2_4::DigitList::operator!= 0 0 1 icuuc24d.dll digitlst.h Doug
263 */
264 /*
265 void
266 NumberFormatTest::TestDigitList(void)
267 {
268   // API coverage for DigitList
269   DigitList list1;
270   list1.append('1');
271   list1.fDecimalAt = 1;
272   DigitList list2;
273   list2.set((int32_t)1);
274   if (list1 != list2) {
275     errln("digitlist append, operator!= or set failed ");
276   }
277   if (!(list1 == list2)) {
278     errln("digitlist append, operator== or set failed ");
279   }
280 }
281 */
282 
283 // -------------------------------------
284 
285 // Test exponential pattern
286 void
TestExponential(void)287 NumberFormatTest::TestExponential(void)
288 {
289     UErrorCode status = U_ZERO_ERROR;
290     DecimalFormatSymbols sym(Locale::getUS(), status);
291     if (U_FAILURE(status)) { errcheckln(status, "FAIL: Bad status returned by DecimalFormatSymbols ct - %s", u_errorName(status)); return; }
292     const char* pat[] = { "0.####E0", "00.000E00", "##0.######E000", "0.###E0;[0.###E0]"  };
293     int32_t pat_length = (int32_t)(sizeof(pat) / sizeof(pat[0]));
294 
295 // The following #if statements allow this test to be built and run on
296 // platforms that do not have standard IEEE numerics.  For example,
297 // S/390 doubles have an exponent range of -78 to +75.  For the
298 // following #if statements to work, float.h must define
299 // DBL_MAX_10_EXP to be a compile-time constant.
300 
301 // This section may be expanded as needed.
302 
303 #if DBL_MAX_10_EXP > 300
304     double val[] = { 0.01234, 123456789, 1.23e300, -3.141592653e-271 };
305     int32_t val_length = (int32_t)(sizeof(val) / sizeof(val[0]));
306     const char* valFormat[] =
307     {
308         // 0.####E0
309         "1.234E-2", "1.2346E8", "1.23E300", "-3.1416E-271",
310         // 00.000E00
311         "12.340E-03", "12.346E07", "12.300E299", "-31.416E-272",
312         // ##0.######E000
313         "12.34E-003", "123.4568E006", "1.23E300", "-314.1593E-273",
314         // 0.###E0;[0.###E0]
315         "1.234E-2", "1.235E8", "1.23E300", "[3.142E-271]"
316     };
317     double valParse[] =
318     {
319         0.01234, 123460000, 1.23E300, -3.1416E-271,
320         0.01234, 123460000, 1.23E300, -3.1416E-271,
321         0.01234, 123456800, 1.23E300, -3.141593E-271,
322         0.01234, 123500000, 1.23E300, -3.142E-271,
323     };
324 #elif DBL_MAX_10_EXP > 70
325     double val[] = { 0.01234, 123456789, 1.23e70, -3.141592653e-71 };
326     int32_t val_length = sizeof(val) / sizeof(val[0]);
327     char* valFormat[] =
328     {
329         // 0.####E0
330         "1.234E-2", "1.2346E8", "1.23E70", "-3.1416E-71",
331         // 00.000E00
332         "12.340E-03", "12.346E07", "12.300E69", "-31.416E-72",
333         // ##0.######E000
334         "12.34E-003", "123.4568E006", "12.3E069", "-31.41593E-072",
335         // 0.###E0;[0.###E0]
336         "1.234E-2", "1.235E8", "1.23E70", "[3.142E-71]"
337     };
338     double valParse[] =
339     {
340         0.01234, 123460000, 1.23E70, -3.1416E-71,
341         0.01234, 123460000, 1.23E70, -3.1416E-71,
342         0.01234, 123456800, 1.23E70, -3.141593E-71,
343         0.01234, 123500000, 1.23E70, -3.142E-71,
344     };
345 #else
346     // Don't test double conversion
347     double* val = 0;
348     int32_t val_length = 0;
349     char** valFormat = 0;
350     double* valParse = 0;
351     logln("Warning: Skipping double conversion tests");
352 #endif
353 
354     int32_t lval[] = { 0, -1, 1, 123456789 };
355     int32_t lval_length = (int32_t)(sizeof(lval) / sizeof(lval[0]));
356     const char* lvalFormat[] =
357     {
358         // 0.####E0
359         "0E0", "-1E0", "1E0", "1.2346E8",
360         // 00.000E00
361         "00.000E00", "-10.000E-01", "10.000E-01", "12.346E07",
362         // ##0.######E000
363         "0E000", "-1E000", "1E000", "123.4568E006",
364         // 0.###E0;[0.###E0]
365         "0E0", "[1E0]", "1E0", "1.235E8"
366     };
367     int32_t lvalParse[] =
368     {
369         0, -1, 1, 123460000,
370         0, -1, 1, 123460000,
371         0, -1, 1, 123456800,
372         0, -1, 1, 123500000,
373     };
374     int32_t ival = 0, ilval = 0;
375     for (int32_t p=0; p<pat_length; ++p)
376     {
377         DecimalFormat fmt(pat[p], sym, status);
378         if (U_FAILURE(status)) { errln("FAIL: Bad status returned by DecimalFormat ct"); continue; }
379         UnicodeString pattern;
380         logln((UnicodeString)"Pattern \"" + pat[p] + "\" -toPattern-> \"" +
381           fmt.toPattern(pattern) + "\"");
382         int32_t v;
383         for (v=0; v<val_length; ++v)
384         {
385             UnicodeString s; (*(NumberFormat*)&fmt).format(val[v], s);
386             logln((UnicodeString)" " + val[v] + " -format-> " + s);
387             if (s != valFormat[v+ival])
388                 errln((UnicodeString)"FAIL: Expected " + valFormat[v+ival]);
389 
390             ParsePosition pos(0);
391             Formattable af;
392             fmt.parse(s, af, pos);
393             double a;
394             UBool useEpsilon = FALSE;
395             if (af.getType() == Formattable::kLong)
396                 a = af.getLong();
397             else if (af.getType() == Formattable::kDouble) {
398                 a = af.getDouble();
399 #if U_PF_OS390 <= U_PLATFORM && U_PLATFORM <= U_PF_OS400
400                 // S/390 will show a failure like this:
401                 //| -3.141592652999999e-271 -format-> -3.1416E-271
402                 //|                          -parse-> -3.1416e-271
403                 //| FAIL: Expected -3.141599999999999e-271
404                 // To compensate, we use an epsilon-based equality
405                 // test on S/390 only.  We don't want to do this in
406                 // general because it's less exacting.
407                 useEpsilon = TRUE;
408 #endif
409             }
410             else {
411                 errln((UnicodeString)"FAIL: Non-numeric Formattable returned");
412                 continue;
413             }
414             if (pos.getIndex() == s.length())
415             {
416                 logln((UnicodeString)"  -parse-> " + a);
417                 // Use epsilon comparison as necessary
418                 if ((useEpsilon &&
419                     (uprv_fabs(a - valParse[v+ival]) / a > (2*DBL_EPSILON))) ||
420                     (!useEpsilon && a != valParse[v+ival]))
421                 {
422                     errln((UnicodeString)"FAIL: Expected " + valParse[v+ival]);
423                 }
424             }
425             else {
426                 errln((UnicodeString)"FAIL: Partial parse (" + pos.getIndex() + " chars) -> " + a);
427                 errln((UnicodeString)"  should be (" + s.length() + " chars) -> " + valParse[v+ival]);
428             }
429         }
430         for (v=0; v<lval_length; ++v)
431         {
432             UnicodeString s;
433             (*(NumberFormat*)&fmt).format(lval[v], s);
434             logln((UnicodeString)" " + lval[v] + "L -format-> " + s);
435             if (s != lvalFormat[v+ilval])
436                 errln((UnicodeString)"ERROR: Expected " + lvalFormat[v+ilval] + " Got: " + s);
437 
438             ParsePosition pos(0);
439             Formattable af;
440             fmt.parse(s, af, pos);
441             if (af.getType() == Formattable::kLong ||
442                 af.getType() == Formattable::kInt64) {
443                 UErrorCode status = U_ZERO_ERROR;
444                 int32_t a = af.getLong(status);
445                 if (pos.getIndex() == s.length())
446                 {
447                     logln((UnicodeString)"  -parse-> " + a);
448                     if (a != lvalParse[v+ilval])
449                         errln((UnicodeString)"FAIL: Expected " + lvalParse[v+ilval]);
450                 }
451                 else
452                     errln((UnicodeString)"FAIL: Partial parse (" + pos.getIndex() + " chars) -> " + a);
453             }
454             else
455                 errln((UnicodeString)"FAIL: Non-long Formattable returned for " + s
456                     + " Double: " + af.getDouble()
457                     + ", Long: " + af.getLong());
458         }
459         ival += val_length;
460         ilval += lval_length;
461     }
462 }
463 
464 void
TestScientific2()465 NumberFormatTest::TestScientific2() {
466     // jb 2552
467     UErrorCode status = U_ZERO_ERROR;
468     DecimalFormat* fmt = (DecimalFormat*)NumberFormat::createCurrencyInstance("en_US", status);
469     if (U_SUCCESS(status)) {
470         double num = 12.34;
471         expect(*fmt, num, "$12.34");
472         fmt->setScientificNotation(TRUE);
473         expect(*fmt, num, "$1.23E1");
474         fmt->setScientificNotation(FALSE);
475         expect(*fmt, num, "$12.34");
476     }
477     delete fmt;
478 }
479 
480 void
TestScientificGrouping()481 NumberFormatTest::TestScientificGrouping() {
482     // jb 2552
483     UErrorCode status = U_ZERO_ERROR;
484     DecimalFormat fmt("##0.00E0",status);
485     if (U_SUCCESS(status)) {
486         expect(fmt, .01234, "12.3E-3");
487         expect(fmt, .1234, "123E-3");
488         expect(fmt, 1.234, "1.23E0");
489         expect(fmt, 12.34, "12.3E0");
490         expect(fmt, 123.4, "123E0");
491         expect(fmt, 1234., "1.23E3");
492     }
493 }
494 
495 /*static void setFromString(DigitList& dl, const char* str) {
496     char c;
497     UBool decimalSet = FALSE;
498     dl.clear();
499     while ((c = *str++)) {
500         if (c == '-') {
501             dl.fIsPositive = FALSE;
502         } else if (c == '+') {
503             dl.fIsPositive = TRUE;
504         } else if (c == '.') {
505             dl.fDecimalAt = dl.fCount;
506             decimalSet = TRUE;
507         } else {
508             dl.append(c);
509         }
510     }
511     if (!decimalSet) {
512         dl.fDecimalAt = dl.fCount;
513     }
514 }*/
515 
516 void
TestInt64()517 NumberFormatTest::TestInt64() {
518     UErrorCode status = U_ZERO_ERROR;
519     DecimalFormat fmt("#.#E0",status);
520     fmt.setMaximumFractionDigits(20);
521     if (U_SUCCESS(status)) {
522         expect(fmt, (Formattable)(int64_t)0, "0E0");
523         expect(fmt, (Formattable)(int64_t)-1, "-1E0");
524         expect(fmt, (Formattable)(int64_t)1, "1E0");
525         expect(fmt, (Formattable)(int64_t)2147483647, "2.147483647E9");
526         expect(fmt, (Formattable)((int64_t)-2147483647-1), "-2.147483648E9");
527         expect(fmt, (Formattable)(int64_t)U_INT64_MAX, "9.223372036854775807E18");
528         expect(fmt, (Formattable)(int64_t)U_INT64_MIN, "-9.223372036854775808E18");
529     }
530 
531     // also test digitlist
532 /*    int64_t int64max = U_INT64_MAX;
533     int64_t int64min = U_INT64_MIN;
534     const char* int64maxstr = "9223372036854775807";
535     const char* int64minstr = "-9223372036854775808";
536     UnicodeString fail("fail: ");
537 
538     // test max int64 value
539     DigitList dl;
540     setFromString(dl, int64maxstr);
541     {
542         if (!dl.fitsIntoInt64(FALSE)) {
543             errln(fail + int64maxstr + " didn't fit");
544         }
545         int64_t int64Value = dl.getInt64();
546         if (int64Value != int64max) {
547             errln(fail + int64maxstr);
548         }
549         dl.set(int64Value);
550         int64Value = dl.getInt64();
551         if (int64Value != int64max) {
552             errln(fail + int64maxstr);
553         }
554     }
555     // test negative of max int64 value (1 shy of min int64 value)
556     dl.fIsPositive = FALSE;
557     {
558         if (!dl.fitsIntoInt64(FALSE)) {
559             errln(fail + "-" + int64maxstr + " didn't fit");
560         }
561         int64_t int64Value = dl.getInt64();
562         if (int64Value != -int64max) {
563             errln(fail + "-" + int64maxstr);
564         }
565         dl.set(int64Value);
566         int64Value = dl.getInt64();
567         if (int64Value != -int64max) {
568             errln(fail + "-" + int64maxstr);
569         }
570     }
571     // test min int64 value
572     setFromString(dl, int64minstr);
573     {
574         if (!dl.fitsIntoInt64(FALSE)) {
575             errln(fail + "-" + int64minstr + " didn't fit");
576         }
577         int64_t int64Value = dl.getInt64();
578         if (int64Value != int64min) {
579             errln(fail + int64minstr);
580         }
581         dl.set(int64Value);
582         int64Value = dl.getInt64();
583         if (int64Value != int64min) {
584             errln(fail + int64minstr);
585         }
586     }
587     // test negative of min int 64 value (1 more than max int64 value)
588     dl.fIsPositive = TRUE; // won't fit
589     {
590         if (dl.fitsIntoInt64(FALSE)) {
591             errln(fail + "-(" + int64minstr + ") didn't fit");
592         }
593     }*/
594 }
595 
596 // -------------------------------------
597 
598 // Test the handling of quotes
599 void
TestQuotes(void)600 NumberFormatTest::TestQuotes(void)
601 {
602     UErrorCode status = U_ZERO_ERROR;
603     UnicodeString *pat;
604     DecimalFormatSymbols *sym = new DecimalFormatSymbols(Locale::getUS(), status);
605     if (U_FAILURE(status)) {
606         errcheckln(status, "Fail to create DecimalFormatSymbols - %s", u_errorName(status));
607         delete sym;
608         return;
609     }
610     pat = new UnicodeString("a'fo''o'b#");
611     DecimalFormat *fmt = new DecimalFormat(*pat, *sym, status);
612     UnicodeString s;
613     ((NumberFormat*)fmt)->format((int32_t)123, s);
614     logln((UnicodeString)"Pattern \"" + *pat + "\"");
615     logln((UnicodeString)" Format 123 -> " + escape(s));
616     if (!(s=="afo'ob123"))
617         errln((UnicodeString)"FAIL: Expected afo'ob123");
618 
619     s.truncate(0);
620     delete fmt;
621     delete pat;
622 
623     pat = new UnicodeString("a''b#");
624     fmt = new DecimalFormat(*pat, *sym, status);
625     ((NumberFormat*)fmt)->format((int32_t)123, s);
626     logln((UnicodeString)"Pattern \"" + *pat + "\"");
627     logln((UnicodeString)" Format 123 -> " + escape(s));
628     if (!(s=="a'b123"))
629         errln((UnicodeString)"FAIL: Expected a'b123");
630     delete fmt;
631     delete pat;
632     delete sym;
633 }
634 
635 /**
636  * Test the handling of the currency symbol in patterns.
637  */
638 void
TestCurrencySign(void)639 NumberFormatTest::TestCurrencySign(void)
640 {
641     UErrorCode status = U_ZERO_ERROR;
642     DecimalFormatSymbols* sym = new DecimalFormatSymbols(Locale::getUS(), status);
643     UnicodeString pat;
644     UChar currency = 0x00A4;
645     if (U_FAILURE(status)) {
646         errcheckln(status, "Fail to create DecimalFormatSymbols - %s", u_errorName(status));
647         delete sym;
648         return;
649     }
650     // "\xA4#,##0.00;-\xA4#,##0.00"
651     pat.append(currency).append("#,##0.00;-").
652         append(currency).append("#,##0.00");
653     DecimalFormat *fmt = new DecimalFormat(pat, *sym, status);
654     UnicodeString s; ((NumberFormat*)fmt)->format(1234.56, s);
655     pat.truncate(0);
656     logln((UnicodeString)"Pattern \"" + fmt->toPattern(pat) + "\"");
657     logln((UnicodeString)" Format " + 1234.56 + " -> " + escape(s));
658     if (s != "$1,234.56") dataerrln((UnicodeString)"FAIL: Expected $1,234.56");
659     s.truncate(0);
660     ((NumberFormat*)fmt)->format(- 1234.56, s);
661     logln((UnicodeString)" Format " + (-1234.56) + " -> " + escape(s));
662     if (s != "-$1,234.56") dataerrln((UnicodeString)"FAIL: Expected -$1,234.56");
663     delete fmt;
664     pat.truncate(0);
665     // "\xA4\xA4 #,##0.00;\xA4\xA4 -#,##0.00"
666     pat.append(currency).append(currency).
667         append(" #,##0.00;").
668         append(currency).append(currency).
669         append(" -#,##0.00");
670     fmt = new DecimalFormat(pat, *sym, status);
671     s.truncate(0);
672     ((NumberFormat*)fmt)->format(1234.56, s);
673     logln((UnicodeString)"Pattern \"" + fmt->toPattern(pat) + "\"");
674     logln((UnicodeString)" Format " + 1234.56 + " -> " + escape(s));
675     if (s != "USD 1,234.56") dataerrln((UnicodeString)"FAIL: Expected USD 1,234.56");
676     s.truncate(0);
677     ((NumberFormat*)fmt)->format(-1234.56, s);
678     logln((UnicodeString)" Format " + (-1234.56) + " -> " + escape(s));
679     if (s != "USD -1,234.56") dataerrln((UnicodeString)"FAIL: Expected USD -1,234.56");
680     delete fmt;
681     delete sym;
682     if (U_FAILURE(status)) errln((UnicodeString)"FAIL: Status " + u_errorName(status));
683 }
684 
685 // -------------------------------------
686 
toHexString(int32_t i)687 static UChar toHexString(int32_t i) { return (UChar)(i + (i < 10 ? 0x30 : (0x41 - 10))); }
688 
689 UnicodeString&
escape(UnicodeString & s)690 NumberFormatTest::escape(UnicodeString& s)
691 {
692     UnicodeString buf;
693     for (int32_t i=0; i<s.length(); ++i)
694     {
695         UChar c = s[(int32_t)i];
696         if (c <= (UChar)0x7F) buf += c;
697         else {
698             buf += (UChar)0x5c; buf += (UChar)0x55;
699             buf += toHexString((c & 0xF000) >> 12);
700             buf += toHexString((c & 0x0F00) >> 8);
701             buf += toHexString((c & 0x00F0) >> 4);
702             buf += toHexString(c & 0x000F);
703         }
704     }
705     return (s = buf);
706 }
707 
708 
709 // -------------------------------------
710 static const char* testCases[][2]= {
711      /* locale ID */  /* expected */
712     {"ca_ES_PREEURO", "\\u20A7\\u00A01.150" },
713     {"de_LU_PREEURO", "1,150\\u00A0F" },
714     {"el_GR_PREEURO", "1.150,50\\u00A0\\u0394\\u03C1\\u03C7" },
715     {"en_BE_PREEURO", "1.150,50\\u00A0BEF" },
716     {"es_ES_PREEURO", "1.150\\u00A0\\u20A7" },
717     {"eu_ES_PREEURO", "\\u20A7\\u00A01.150" },
718     {"gl_ES_PREEURO", "1.150\\u00A0\\u20A7" },
719     {"it_IT_PREEURO", "ITL\\u00A01.150" },
720     {"pt_PT_PREEURO", "1,150$50\\u00A0\\u200B"}, // per cldrbug 7670
721     {"en_US@currency=JPY", "\\u00A51,150"},
722     {"en_US@currency=jpy", "\\u00A51,150"},
723     {"en-US-u-cu-jpy", "\\u00A51,150"}
724 };
725 /**
726  * Test localized currency patterns.
727  */
728 void
TestCurrency(void)729 NumberFormatTest::TestCurrency(void)
730 {
731     UErrorCode status = U_ZERO_ERROR;
732     NumberFormat* currencyFmt = NumberFormat::createCurrencyInstance(Locale::getCanadaFrench(), status);
733     if (U_FAILURE(status)) {
734         dataerrln("Error calling NumberFormat::createCurrencyInstance()");
735         return;
736     }
737 
738     UnicodeString s; currencyFmt->format(1.50, s);
739     logln((UnicodeString)"Un pauvre ici a..........." + s);
740     if (!(s==CharsToUnicodeString("1,50\\u00A0$")))
741         errln((UnicodeString)"FAIL: Expected 1,50<nbsp>$");
742     delete currencyFmt;
743     s.truncate(0);
744     char loc[256]={0};
745     int len = uloc_canonicalize("de_DE_PREEURO", loc, 256, &status);
746     (void)len;  // Suppress unused variable warning.
747     currencyFmt = NumberFormat::createCurrencyInstance(Locale(loc),status);
748     currencyFmt->format(1.50, s);
749     logln((UnicodeString)"Un pauvre en Allemagne a.." + s);
750     if (!(s==CharsToUnicodeString("1,50\\u00A0DM")))
751         errln((UnicodeString)"FAIL: Expected 1,50<nbsp>DM");
752     delete currencyFmt;
753     s.truncate(0);
754     len = uloc_canonicalize("fr_FR_PREEURO", loc, 256, &status);
755     currencyFmt = NumberFormat::createCurrencyInstance(Locale(loc), status);
756     currencyFmt->format(1.50, s);
757     logln((UnicodeString)"Un pauvre en France a....." + s);
758     if (!(s==CharsToUnicodeString("1,50\\u00A0F")))
759         errln((UnicodeString)"FAIL: Expected 1,50<nbsp>F");
760     delete currencyFmt;
761     if (U_FAILURE(status))
762         errln((UnicodeString)"FAIL: Status " + (int32_t)status);
763 
764     for(int i=0; i < (int)(sizeof(testCases)/sizeof(testCases[i])); i++){
765         status = U_ZERO_ERROR;
766         const char *localeID = testCases[i][0];
767         UnicodeString expected(testCases[i][1], -1, US_INV);
768         expected = expected.unescape();
769         s.truncate(0);
770         char loc[256]={0};
771         uloc_canonicalize(localeID, loc, 256, &status);
772         currencyFmt = NumberFormat::createCurrencyInstance(Locale(loc), status);
773         if(U_FAILURE(status)){
774             errln("Could not create currency formatter for locale %s",localeID);
775             continue;
776         }
777         currencyFmt->format(1150.50, s);
778         if(s!=expected){
779             errln(UnicodeString("FAIL: Expected: ")+expected
780                     + UnicodeString(" Got: ") + s
781                     + UnicodeString( " for locale: ")+ UnicodeString(localeID) );
782         }
783         if (U_FAILURE(status)){
784             errln((UnicodeString)"FAIL: Status " + (int32_t)status);
785         }
786         delete currencyFmt;
787     }
788 }
789 
790 // -------------------------------------
791 
792 /**
793  * Test the Currency object handling, new as of ICU 2.2.
794  */
TestCurrencyObject()795 void NumberFormatTest::TestCurrencyObject() {
796     UErrorCode ec = U_ZERO_ERROR;
797     NumberFormat* fmt =
798         NumberFormat::createCurrencyInstance(Locale::getUS(), ec);
799 
800     if (U_FAILURE(ec)) {
801         dataerrln("FAIL: getCurrencyInstance(US) - %s", u_errorName(ec));
802         delete fmt;
803         return;
804     }
805 
806     Locale null("", "", "");
807 
808     expectCurrency(*fmt, null, 1234.56, "$1,234.56");
809 
810     expectCurrency(*fmt, Locale::getFrance(),
811                    1234.56, CharsToUnicodeString("\\u20AC1,234.56")); // Euro
812 
813     expectCurrency(*fmt, Locale::getJapan(),
814                    1234.56, CharsToUnicodeString("\\u00A51,235")); // Yen
815 
816     expectCurrency(*fmt, Locale("fr", "CH", ""),
817                    1234.56, "CHF1,234.56"); // no more 0.05 rounding here, see cldrbug 5548
818 
819     expectCurrency(*fmt, Locale::getUS(),
820                    1234.56, "$1,234.56");
821 
822     delete fmt;
823     fmt = NumberFormat::createCurrencyInstance(Locale::getFrance(), ec);
824 
825     if (U_FAILURE(ec)) {
826         errln("FAIL: getCurrencyInstance(FRANCE)");
827         delete fmt;
828         return;
829     }
830 
831     expectCurrency(*fmt, null, 1234.56, CharsToUnicodeString("1 234,56 \\u20AC"));
832 
833     expectCurrency(*fmt, Locale::getJapan(),
834                    1234.56, CharsToUnicodeString("1 235 JPY")); // Yen
835 
836     expectCurrency(*fmt, Locale("fr", "CH", ""),
837                    1234.56, "1 234,56 CHF"); // no more 0.05 rounding here, see cldrbug 5548
838 
839     expectCurrency(*fmt, Locale::getUS(),
840                    1234.56, "1 234,56 $US");
841 
842     expectCurrency(*fmt, Locale::getFrance(),
843                    1234.56, CharsToUnicodeString("1 234,56 \\u20AC")); // Euro
844 
845     delete fmt;
846 }
847 
848 // -------------------------------------
849 
850 /**
851  * Do rudimentary testing of parsing.
852  */
853 void
TestParse(void)854 NumberFormatTest::TestParse(void)
855 {
856     UErrorCode status = U_ZERO_ERROR;
857     UnicodeString arg("0");
858     DecimalFormat* format = new DecimalFormat("00", status);
859     //try {
860         Formattable n; format->parse(arg, n, status);
861         logln((UnicodeString)"parse(" + arg + ") = " + n.getLong());
862         if (n.getType() != Formattable::kLong ||
863             n.getLong() != 0) errln((UnicodeString)"FAIL: Expected 0");
864     delete format;
865     if (U_FAILURE(status)) errcheckln(status, (UnicodeString)"FAIL: Status " + u_errorName(status));
866     //}
867     //catch(Exception e) {
868     //    errln((UnicodeString)"Exception caught: " + e);
869     //}
870 }
871 
872 // -------------------------------------
873 
874 static const char *lenientAffixTestCases[] = {
875         "(1)",
876         "( 1)",
877         "(1 )",
878         "( 1 )"
879 };
880 
881 static const char *lenientMinusTestCases[] = {
882     "-5",
883     "\\u22125",
884     "\\u20105"
885 };
886 
887 static const char *lenientCurrencyTestCases[] = {
888         "$1,000",
889         "$ 1,000",
890         "$1000",
891         "$ 1000",
892         "$1 000.00",
893         "$ 1 000.00",
894         "$ 1\\u00A0000.00",
895         "1000.00"
896 };
897 
898 // changed from () to - per cldrbug 5674
899 static const char *lenientNegativeCurrencyTestCases[] = {
900         "-$1,000",
901         "-$ 1,000",
902         "-$1000",
903         "-$ 1000",
904         "-$1 000.00",
905         "-$ 1 000.00",
906         "- $ 1,000.00 ",
907         "-$ 1\\u00A0000.00",
908         "-1000.00"
909 };
910 
911 static const char *lenientPercentTestCases[] = {
912         "25%",
913         " 25%",
914         " 25 %",
915     	"25 %",
916 		"25\\u00A0%",
917 		"25"
918 };
919 
920 static const char *lenientNegativePercentTestCases[] = {
921 		"-25%",
922 		" -25%",
923 		" - 25%",
924 		"- 25 %",
925 		" - 25 %",
926 		"-25 %",
927 		"-25\\u00A0%",
928 		"-25",
929 		"- 25"
930 };
931 
932 static const char *strictFailureTestCases[] = {
933 		" 1000",
934 		"10,00",
935 		"1,000,.0"
936 };
937 
938 #define ARRAY_SIZE(array) ((int32_t) (sizeof (array) / sizeof(array[0])))
939 
940 /**
941  * Test lenient parsing.
942  */
943 void
TestLenientParse(void)944 NumberFormatTest::TestLenientParse(void)
945 {
946     UErrorCode status = U_ZERO_ERROR;
947     DecimalFormat *format = new DecimalFormat("(#,##0)", status);
948     Formattable n;
949 
950     if (format == NULL || U_FAILURE(status)) {
951         dataerrln("Unable to create DecimalFormat (#,##0) - %s", u_errorName(status));
952     } else {
953         format->setLenient(TRUE);
954         for (int32_t t = 0; t < ARRAY_SIZE (lenientAffixTestCases); t += 1) {
955         	UnicodeString testCase = ctou(lenientAffixTestCases[t]);
956 
957             format->parse(testCase, n, status);
958             logln((UnicodeString)"parse(" + testCase + ") = " + n.getLong());
959 
960             if (U_FAILURE(status) || n.getType() != Formattable::kLong ||
961             	n.getLong() != 1) {
962             	errln((UnicodeString)"Lenient parse failed for \"" + (UnicodeString) lenientAffixTestCases[t] + (UnicodeString) "\"");
963             	status = U_ZERO_ERROR;
964             }
965        }
966        delete format;
967     }
968 
969     Locale en_US("en_US");
970     Locale sv_SE("sv_SE");
971 
972     NumberFormat *mFormat = NumberFormat::createInstance(sv_SE, UNUM_DECIMAL, status);
973 
974     if (mFormat == NULL || U_FAILURE(status)) {
975         dataerrln("Unable to create NumberFormat (sv_SE, UNUM_DECIMAL) - %s", u_errorName(status));
976     } else {
977         mFormat->setLenient(TRUE);
978         for (int32_t t = 0; t < ARRAY_SIZE(lenientMinusTestCases); t += 1) {
979             UnicodeString testCase = ctou(lenientMinusTestCases[t]);
980 
981             mFormat->parse(testCase, n, status);
982             logln((UnicodeString)"parse(" + testCase + ") = " + n.getLong());
983 
984             if (U_FAILURE(status) || n.getType() != Formattable::kLong || n.getLong() != -5) {
985                 errln((UnicodeString)"Lenient parse failed for \"" + (UnicodeString) lenientMinusTestCases[t] + (UnicodeString) "\"");
986                 status = U_ZERO_ERROR;
987             }
988         }
989         delete mFormat;
990     }
991 
992     mFormat = NumberFormat::createInstance(en_US, UNUM_DECIMAL, status);
993 
994     if (mFormat == NULL || U_FAILURE(status)) {
995         dataerrln("Unable to create NumberFormat (en_US, UNUM_DECIMAL) - %s", u_errorName(status));
996     } else {
997         mFormat->setLenient(TRUE);
998         for (int32_t t = 0; t < ARRAY_SIZE(lenientMinusTestCases); t += 1) {
999             UnicodeString testCase = ctou(lenientMinusTestCases[t]);
1000 
1001             mFormat->parse(testCase, n, status);
1002             logln((UnicodeString)"parse(" + testCase + ") = " + n.getLong());
1003 
1004             if (U_FAILURE(status) || n.getType() != Formattable::kLong || n.getLong() != -5) {
1005                 errln((UnicodeString)"Lenient parse failed for \"" + (UnicodeString) lenientMinusTestCases[t] + (UnicodeString) "\"");
1006                 status = U_ZERO_ERROR;
1007             }
1008         }
1009         delete mFormat;
1010     }
1011 
1012     NumberFormat *cFormat = NumberFormat::createInstance(en_US, UNUM_CURRENCY, status);
1013 
1014     if (cFormat == NULL || U_FAILURE(status)) {
1015         dataerrln("Unable to create NumberFormat (en_US, UNUM_CURRENCY) - %s", u_errorName(status));
1016     } else {
1017         cFormat->setLenient(TRUE);
1018         for (int32_t t = 0; t < ARRAY_SIZE (lenientCurrencyTestCases); t += 1) {
1019         	UnicodeString testCase = ctou(lenientCurrencyTestCases[t]);
1020 
1021             cFormat->parse(testCase, n, status);
1022             logln((UnicodeString)"parse(" + testCase + ") = " + n.getLong());
1023 
1024             if (U_FAILURE(status) ||n.getType() != Formattable::kLong ||
1025             	n.getLong() != 1000) {
1026             	errln((UnicodeString)"Lenient parse failed for \"" + (UnicodeString) lenientCurrencyTestCases[t] + (UnicodeString) "\"");
1027             	status = U_ZERO_ERROR;
1028             }
1029         }
1030 
1031         for (int32_t t = 0; t < ARRAY_SIZE (lenientNegativeCurrencyTestCases); t += 1) {
1032         	UnicodeString testCase = ctou(lenientNegativeCurrencyTestCases[t]);
1033 
1034             cFormat->parse(testCase, n, status);
1035             logln((UnicodeString)"parse(" + testCase + ") = " + n.getLong());
1036 
1037             if (U_FAILURE(status) ||n.getType() != Formattable::kLong ||
1038             	n.getLong() != -1000) {
1039             	errln((UnicodeString)"Lenient parse failed for \"" + (UnicodeString) lenientNegativeCurrencyTestCases[t] + (UnicodeString) "\"");
1040             	status = U_ZERO_ERROR;
1041             }
1042         }
1043 
1044         delete cFormat;
1045     }
1046 
1047     NumberFormat *pFormat = NumberFormat::createPercentInstance(en_US, status);
1048 
1049     if (pFormat == NULL || U_FAILURE(status)) {
1050         dataerrln("Unable to create NumberFormat::createPercentInstance (en_US) - %s", u_errorName(status));
1051     } else {
1052         pFormat->setLenient(TRUE);
1053         for (int32_t t = 0; t < ARRAY_SIZE (lenientPercentTestCases); t += 1) {
1054         	UnicodeString testCase = ctou(lenientPercentTestCases[t]);
1055 
1056         	pFormat->parse(testCase, n, status);
1057             logln((UnicodeString)"parse(" + testCase + ") = " + n.getDouble());
1058 
1059             if (U_FAILURE(status) ||n.getType() != Formattable::kDouble ||
1060             	n.getDouble() != 0.25) {
1061             	errln((UnicodeString)"Lenient parse failed for \"" + (UnicodeString) lenientPercentTestCases[t] + (UnicodeString) "\"");
1062             	status = U_ZERO_ERROR;
1063             }
1064         }
1065 
1066         for (int32_t t = 0; t < ARRAY_SIZE (lenientNegativePercentTestCases); t += 1) {
1067         	UnicodeString testCase = ctou(lenientNegativePercentTestCases[t]);
1068 
1069         	pFormat->parse(testCase, n, status);
1070             logln((UnicodeString)"parse(" + testCase + ") = " + n.getDouble());
1071 
1072             if (U_FAILURE(status) ||n.getType() != Formattable::kDouble ||
1073             	n.getDouble() != -0.25) {
1074             	errln((UnicodeString)"Lenient parse failed for \"" + (UnicodeString) lenientNegativePercentTestCases[t] + (UnicodeString) "\"");
1075             	status = U_ZERO_ERROR;
1076             }
1077         }
1078 
1079         delete pFormat;
1080     }
1081 
1082    // Test cases that should fail with a strict parse and pass with a
1083    // lenient parse.
1084    NumberFormat *nFormat = NumberFormat::createInstance(en_US, status);
1085 
1086    if (nFormat == NULL || U_FAILURE(status)) {
1087        dataerrln("Unable to create NumberFormat (en_US) - %s", u_errorName(status));
1088    } else {
1089        // first, make sure that they fail with a strict parse
1090        for (int32_t t = 0; t < ARRAY_SIZE(strictFailureTestCases); t += 1) {
1091 	       UnicodeString testCase = ctou(strictFailureTestCases[t]);
1092 
1093 	       nFormat->parse(testCase, n, status);
1094 	       logln((UnicodeString)"parse(" + testCase + ") = " + n.getLong());
1095 
1096 	       if (! U_FAILURE(status)) {
1097 		       errln((UnicodeString)"Strict Parse succeeded for \"" + (UnicodeString) strictFailureTestCases[t] + (UnicodeString) "\"");
1098 	       }
1099 
1100 	       status = U_ZERO_ERROR;
1101        }
1102 
1103        // then, make sure that they pass with a lenient parse
1104        nFormat->setLenient(TRUE);
1105        for (int32_t t = 0; t < ARRAY_SIZE(strictFailureTestCases); t += 1) {
1106 	       UnicodeString testCase = ctou(strictFailureTestCases[t]);
1107 
1108 	       nFormat->parse(testCase, n, status);
1109 	       logln((UnicodeString)"parse(" + testCase + ") = " + n.getLong());
1110 
1111 	       if (U_FAILURE(status) ||n.getType() != Formattable::kLong ||
1112 	            	n.getLong() != 1000) {
1113 		       errln((UnicodeString)"Lenient parse failed for \"" + (UnicodeString) strictFailureTestCases[t] + (UnicodeString) "\"");
1114 		       status = U_ZERO_ERROR;
1115 	       }
1116        }
1117 
1118        delete nFormat;
1119    }
1120 }
1121 
1122 // -------------------------------------
1123 
1124 /**
1125  * Test proper rounding by the format method.
1126  */
1127 void
TestRounding487(void)1128 NumberFormatTest::TestRounding487(void)
1129 {
1130     UErrorCode status = U_ZERO_ERROR;
1131     NumberFormat *nf = NumberFormat::createInstance(status);
1132     if (U_FAILURE(status)) {
1133         dataerrln("Error calling NumberFormat::createInstance()");
1134         return;
1135     }
1136 
1137     roundingTest(*nf, 0.00159999, 4, "0.0016");
1138     roundingTest(*nf, 0.00995, 4, "0.01");
1139 
1140     roundingTest(*nf, 12.3995, 3, "12.4");
1141 
1142     roundingTest(*nf, 12.4999, 0, "12");
1143     roundingTest(*nf, - 19.5, 0, "-20");
1144     delete nf;
1145     if (U_FAILURE(status)) errln((UnicodeString)"FAIL: Status " + (int32_t)status);
1146 }
1147 
1148 /**
1149  * Test the functioning of the secondary grouping value.
1150  */
TestSecondaryGrouping(void)1151 void NumberFormatTest::TestSecondaryGrouping(void) {
1152     UErrorCode status = U_ZERO_ERROR;
1153     DecimalFormatSymbols US(Locale::getUS(), status);
1154     CHECK(status, "DecimalFormatSymbols ct");
1155 
1156     DecimalFormat f("#,##,###", US, status);
1157     CHECK(status, "DecimalFormat ct");
1158 
1159     expect2(f, (int32_t)123456789L, "12,34,56,789");
1160     expectPat(f, "#,##,###");
1161     f.applyPattern("#,###", status);
1162     CHECK(status, "applyPattern");
1163 
1164     f.setSecondaryGroupingSize(4);
1165     expect2(f, (int32_t)123456789L, "12,3456,789");
1166     expectPat(f, "#,####,###");
1167     NumberFormat *g = NumberFormat::createInstance(Locale("hi", "IN"), status);
1168     CHECK_DATA(status, "createInstance(hi_IN)");
1169 
1170     UnicodeString out;
1171     int32_t l = (int32_t)1876543210L;
1172     g->format(l, out);
1173     delete g;
1174     // expect "1,87,65,43,210", but with Hindi digits
1175     //         01234567890123
1176     UBool ok = TRUE;
1177     if (out.length() != 14) {
1178         ok = FALSE;
1179     } else {
1180         for (int32_t i=0; i<out.length(); ++i) {
1181             UBool expectGroup = FALSE;
1182             switch (i) {
1183             case 1:
1184             case 4:
1185             case 7:
1186             case 10:
1187                 expectGroup = TRUE;
1188                 break;
1189             }
1190             // Later -- fix this to get the actual grouping
1191             // character from the resource bundle.
1192             UBool isGroup = (out.charAt(i) == 0x002C);
1193             if (isGroup != expectGroup) {
1194                 ok = FALSE;
1195                 break;
1196             }
1197         }
1198     }
1199     if (!ok) {
1200         errln((UnicodeString)"FAIL  Expected " + l +
1201               " x hi_IN -> \"1,87,65,43,210\" (with Hindi digits), got \"" +
1202               escape(out) + "\"");
1203     } else {
1204         logln((UnicodeString)"Ok    " + l +
1205               " x hi_IN -> \"" +
1206               escape(out) + "\"");
1207     }
1208 }
1209 
TestWhiteSpaceParsing(void)1210 void NumberFormatTest::TestWhiteSpaceParsing(void) {
1211     UErrorCode ec = U_ZERO_ERROR;
1212     DecimalFormatSymbols US(Locale::getUS(), ec);
1213     DecimalFormat fmt("a  b#0c  ", US, ec);
1214     if (U_FAILURE(ec)) {
1215         errcheckln(ec, "FAIL: Constructor - %s", u_errorName(ec));
1216         return;
1217     }
1218     int32_t n = 1234;
1219     expect(fmt, "a b1234c ", n);
1220     expect(fmt, "a   b1234c   ", n);
1221 }
1222 
1223 /**
1224  * Test currencies whose display name is a ChoiceFormat.
1225  */
TestComplexCurrency()1226 void NumberFormatTest::TestComplexCurrency() {
1227 
1228 //    UErrorCode ec = U_ZERO_ERROR;
1229 //    Locale loc("kn", "IN", "");
1230 //    NumberFormat* fmt = NumberFormat::createCurrencyInstance(loc, ec);
1231 //    if (U_SUCCESS(ec)) {
1232 //        expect2(*fmt, 1.0, CharsToUnicodeString("Re.\\u00A01.00"));
1233 //        Use .00392625 because that's 2^-8.  Any value less than 0.005 is fine.
1234 //        expect(*fmt, 1.00390625, CharsToUnicodeString("Re.\\u00A01.00")); // tricky
1235 //        expect2(*fmt, 12345678.0, CharsToUnicodeString("Rs.\\u00A01,23,45,678.00"));
1236 //        expect2(*fmt, 0.5, CharsToUnicodeString("Rs.\\u00A00.50"));
1237 //        expect2(*fmt, -1.0, CharsToUnicodeString("-Re.\\u00A01.00"));
1238 //        expect2(*fmt, -10.0, CharsToUnicodeString("-Rs.\\u00A010.00"));
1239 //    } else {
1240 //        errln("FAIL: getCurrencyInstance(kn_IN)");
1241 //    }
1242 //    delete fmt;
1243 
1244 }
1245 
1246 // -------------------------------------
1247 
1248 void
roundingTest(NumberFormat & nf,double x,int32_t maxFractionDigits,const char * expected)1249 NumberFormatTest::roundingTest(NumberFormat& nf, double x, int32_t maxFractionDigits, const char* expected)
1250 {
1251     nf.setMaximumFractionDigits(maxFractionDigits);
1252     UnicodeString out; nf.format(x, out);
1253     logln((UnicodeString)"" + x + " formats with " + maxFractionDigits + " fractional digits to " + out);
1254     if (!(out==expected)) errln((UnicodeString)"FAIL: Expected " + expected);
1255 }
1256 
1257 /**
1258  * Upgrade to alphaWorks
1259  */
TestExponent(void)1260 void NumberFormatTest::TestExponent(void) {
1261     UErrorCode status = U_ZERO_ERROR;
1262     DecimalFormatSymbols US(Locale::getUS(), status);
1263     CHECK(status, "DecimalFormatSymbols constructor");
1264     DecimalFormat fmt1(UnicodeString("0.###E0"), US, status);
1265     CHECK(status, "DecimalFormat(0.###E0)");
1266     DecimalFormat fmt2(UnicodeString("0.###E+0"), US, status);
1267     CHECK(status, "DecimalFormat(0.###E+0)");
1268     int32_t n = 1234;
1269     expect2(fmt1, n, "1.234E3");
1270     expect2(fmt2, n, "1.234E+3");
1271     expect(fmt1, "1.234E+3", n); // Either format should parse "E+3"
1272 }
1273 
1274 /**
1275  * Upgrade to alphaWorks
1276  */
TestScientific(void)1277 void NumberFormatTest::TestScientific(void) {
1278     UErrorCode status = U_ZERO_ERROR;
1279     DecimalFormatSymbols US(Locale::getUS(), status);
1280     CHECK(status, "DecimalFormatSymbols constructor");
1281 
1282     // Test pattern round-trip
1283     const char* PAT[] = { "#E0", "0.####E0", "00.000E00", "##0.####E000",
1284                           "0.###E0;[0.###E0]" };
1285     int32_t PAT_length = (int32_t)(sizeof(PAT) / sizeof(PAT[0]));
1286     int32_t DIGITS[] = {
1287         // min int, max int, min frac, max frac
1288         0, 1, 0, 0, // "#E0"
1289         1, 1, 0, 4, // "0.####E0"
1290         2, 2, 3, 3, // "00.000E00"
1291         1, 3, 0, 4, // "##0.####E000"
1292         1, 1, 0, 3, // "0.###E0;[0.###E0]"
1293     };
1294     for (int32_t i=0; i<PAT_length; ++i) {
1295         UnicodeString pat(PAT[i]);
1296         DecimalFormat df(pat, US, status);
1297         CHECK(status, "DecimalFormat constructor");
1298         UnicodeString pat2;
1299         df.toPattern(pat2);
1300         if (pat == pat2) {
1301             logln(UnicodeString("Ok   Pattern rt \"") +
1302                   pat + "\" -> \"" +
1303                   pat2 + "\"");
1304         } else {
1305             errln(UnicodeString("FAIL Pattern rt \"") +
1306                   pat + "\" -> \"" +
1307                   pat2 + "\"");
1308         }
1309         // Make sure digit counts match what we expect
1310         if (df.getMinimumIntegerDigits() != DIGITS[4*i] ||
1311             df.getMaximumIntegerDigits() != DIGITS[4*i+1] ||
1312             df.getMinimumFractionDigits() != DIGITS[4*i+2] ||
1313             df.getMaximumFractionDigits() != DIGITS[4*i+3]) {
1314             errln(UnicodeString("FAIL \"" + pat +
1315                                 "\" min/max int; min/max frac = ") +
1316                   df.getMinimumIntegerDigits() + "/" +
1317                   df.getMaximumIntegerDigits() + ";" +
1318                   df.getMinimumFractionDigits() + "/" +
1319                   df.getMaximumFractionDigits() + ", expect " +
1320                   DIGITS[4*i] + "/" +
1321                   DIGITS[4*i+1] + ";" +
1322                   DIGITS[4*i+2] + "/" +
1323                   DIGITS[4*i+3]);
1324         }
1325     }
1326 
1327 
1328     // Test the constructor for default locale. We have to
1329     // manually set the default locale, as there is no
1330     // guarantee that the default locale has the same
1331     // scientific format.
1332     Locale def = Locale::getDefault();
1333     Locale::setDefault(Locale::getUS(), status);
1334     expect2(NumberFormat::createScientificInstance(status),
1335            12345.678901,
1336            "1.2345678901E4", status);
1337     Locale::setDefault(def, status);
1338 
1339     expect2(new DecimalFormat("#E0", US, status),
1340            12345.0,
1341            "1.2345E4", status);
1342     expect(new DecimalFormat("0E0", US, status),
1343            12345.0,
1344            "1E4", status);
1345     expect2(NumberFormat::createScientificInstance(Locale::getUS(), status),
1346            12345.678901,
1347            "1.2345678901E4", status);
1348     expect(new DecimalFormat("##0.###E0", US, status),
1349            12345.0,
1350            "12.34E3", status);
1351     expect(new DecimalFormat("##0.###E0", US, status),
1352            12345.00001,
1353            "12.35E3", status);
1354     expect2(new DecimalFormat("##0.####E0", US, status),
1355            (int32_t) 12345,
1356            "12.345E3", status);
1357     expect2(NumberFormat::createScientificInstance(Locale::getFrance(), status),
1358            12345.678901,
1359            "1,2345678901E4", status);
1360     expect(new DecimalFormat("##0.####E0", US, status),
1361            789.12345e-9,
1362            "789.12E-9", status);
1363     expect2(new DecimalFormat("##0.####E0", US, status),
1364            780.e-9,
1365            "780E-9", status);
1366     expect(new DecimalFormat(".###E0", US, status),
1367            45678.0,
1368            ".457E5", status);
1369     expect2(new DecimalFormat(".###E0", US, status),
1370            (int32_t) 0,
1371            ".0E0", status);
1372     /*
1373     expect(new DecimalFormat[] { new DecimalFormat("#E0", US),
1374                                  new DecimalFormat("##E0", US),
1375                                  new DecimalFormat("####E0", US),
1376                                  new DecimalFormat("0E0", US),
1377                                  new DecimalFormat("00E0", US),
1378                                  new DecimalFormat("000E0", US),
1379                                },
1380            new Long(45678000),
1381            new String[] { "4.5678E7",
1382                           "45.678E6",
1383                           "4567.8E4",
1384                           "5E7",
1385                           "46E6",
1386                           "457E5",
1387                         }
1388            );
1389     !
1390     ! Unroll this test into individual tests below...
1391     !
1392     */
1393     expect2(new DecimalFormat("#E0", US, status),
1394            (int32_t) 45678000, "4.5678E7", status);
1395     expect2(new DecimalFormat("##E0", US, status),
1396            (int32_t) 45678000, "45.678E6", status);
1397     expect2(new DecimalFormat("####E0", US, status),
1398            (int32_t) 45678000, "4567.8E4", status);
1399     expect(new DecimalFormat("0E0", US, status),
1400            (int32_t) 45678000, "5E7", status);
1401     expect(new DecimalFormat("00E0", US, status),
1402            (int32_t) 45678000, "46E6", status);
1403     expect(new DecimalFormat("000E0", US, status),
1404            (int32_t) 45678000, "457E5", status);
1405     /*
1406     expect(new DecimalFormat("###E0", US, status),
1407            new Object[] { new Double(0.0000123), "12.3E-6",
1408                           new Double(0.000123), "123E-6",
1409                           new Double(0.00123), "1.23E-3",
1410                           new Double(0.0123), "12.3E-3",
1411                           new Double(0.123), "123E-3",
1412                           new Double(1.23), "1.23E0",
1413                           new Double(12.3), "12.3E0",
1414                           new Double(123), "123E0",
1415                           new Double(1230), "1.23E3",
1416                          });
1417     !
1418     ! Unroll this test into individual tests below...
1419     !
1420     */
1421     expect2(new DecimalFormat("###E0", US, status),
1422            0.0000123, "12.3E-6", status);
1423     expect2(new DecimalFormat("###E0", US, status),
1424            0.000123, "123E-6", status);
1425     expect2(new DecimalFormat("###E0", US, status),
1426            0.00123, "1.23E-3", status);
1427     expect2(new DecimalFormat("###E0", US, status),
1428            0.0123, "12.3E-3", status);
1429     expect2(new DecimalFormat("###E0", US, status),
1430            0.123, "123E-3", status);
1431     expect2(new DecimalFormat("###E0", US, status),
1432            1.23, "1.23E0", status);
1433     expect2(new DecimalFormat("###E0", US, status),
1434            12.3, "12.3E0", status);
1435     expect2(new DecimalFormat("###E0", US, status),
1436            123.0, "123E0", status);
1437     expect2(new DecimalFormat("###E0", US, status),
1438            1230.0, "1.23E3", status);
1439     /*
1440     expect(new DecimalFormat("0.#E+00", US, status),
1441            new Object[] { new Double(0.00012), "1.2E-04",
1442                           new Long(12000),     "1.2E+04",
1443                          });
1444     !
1445     ! Unroll this test into individual tests below...
1446     !
1447     */
1448     expect2(new DecimalFormat("0.#E+00", US, status),
1449            0.00012, "1.2E-04", status);
1450     expect2(new DecimalFormat("0.#E+00", US, status),
1451            (int32_t) 12000, "1.2E+04", status);
1452 }
1453 
1454 /**
1455  * Upgrade to alphaWorks
1456  */
TestPad(void)1457 void NumberFormatTest::TestPad(void) {
1458     UErrorCode status = U_ZERO_ERROR;
1459     DecimalFormatSymbols US(Locale::getUS(), status);
1460     CHECK(status, "DecimalFormatSymbols constructor");
1461 
1462     expect2(new DecimalFormat("*^##.##", US, status),
1463            int32_t(0), "^^^^0", status);
1464     expect2(new DecimalFormat("*^##.##", US, status),
1465            -1.3, "^-1.3", status);
1466     expect2(new DecimalFormat("##0.0####E0*_ 'g-m/s^2'", US, status),
1467            int32_t(0), "0.0E0______ g-m/s^2", status);
1468     expect(new DecimalFormat("##0.0####E0*_ 'g-m/s^2'", US, status),
1469            1.0/3, "333.333E-3_ g-m/s^2", status);
1470     expect2(new DecimalFormat("##0.0####*_ 'g-m/s^2'", US, status),
1471            int32_t(0), "0.0______ g-m/s^2", status);
1472     expect(new DecimalFormat("##0.0####*_ 'g-m/s^2'", US, status),
1473            1.0/3, "0.33333__ g-m/s^2", status);
1474 
1475     // Test padding before a sign
1476     const char *formatStr = "*x#,###,###,##0.0#;*x(###,###,##0.0#)";
1477     expect2(new DecimalFormat(formatStr, US, status),
1478            int32_t(-10),  "xxxxxxxxxx(10.0)", status);
1479     expect2(new DecimalFormat(formatStr, US, status),
1480            int32_t(-1000),"xxxxxxx(1,000.0)", status);
1481     expect2(new DecimalFormat(formatStr, US, status),
1482            int32_t(-1000000),"xxx(1,000,000.0)", status);
1483     expect2(new DecimalFormat(formatStr, US, status),
1484            -100.37,       "xxxxxxxx(100.37)", status);
1485     expect2(new DecimalFormat(formatStr, US, status),
1486            -10456.37,     "xxxxx(10,456.37)", status);
1487     expect2(new DecimalFormat(formatStr, US, status),
1488            -1120456.37,   "xx(1,120,456.37)", status);
1489     expect2(new DecimalFormat(formatStr, US, status),
1490            -112045600.37, "(112,045,600.37)", status);
1491     expect2(new DecimalFormat(formatStr, US, status),
1492            -1252045600.37,"(1,252,045,600.37)", status);
1493 
1494     expect2(new DecimalFormat(formatStr, US, status),
1495            int32_t(10),  "xxxxxxxxxxxx10.0", status);
1496     expect2(new DecimalFormat(formatStr, US, status),
1497            int32_t(1000),"xxxxxxxxx1,000.0", status);
1498     expect2(new DecimalFormat(formatStr, US, status),
1499            int32_t(1000000),"xxxxx1,000,000.0", status);
1500     expect2(new DecimalFormat(formatStr, US, status),
1501            100.37,       "xxxxxxxxxx100.37", status);
1502     expect2(new DecimalFormat(formatStr, US, status),
1503            10456.37,     "xxxxxxx10,456.37", status);
1504     expect2(new DecimalFormat(formatStr, US, status),
1505            1120456.37,   "xxxx1,120,456.37", status);
1506     expect2(new DecimalFormat(formatStr, US, status),
1507            112045600.37, "xx112,045,600.37", status);
1508     expect2(new DecimalFormat(formatStr, US, status),
1509            10252045600.37,"10,252,045,600.37", status);
1510 
1511 
1512     // Test padding between a sign and a number
1513     const char *formatStr2 = "#,###,###,##0.0#*x;(###,###,##0.0#*x)";
1514     expect2(new DecimalFormat(formatStr2, US, status),
1515            int32_t(-10),  "(10.0xxxxxxxxxx)", status);
1516     expect2(new DecimalFormat(formatStr2, US, status),
1517            int32_t(-1000),"(1,000.0xxxxxxx)", status);
1518     expect2(new DecimalFormat(formatStr2, US, status),
1519            int32_t(-1000000),"(1,000,000.0xxx)", status);
1520     expect2(new DecimalFormat(formatStr2, US, status),
1521            -100.37,       "(100.37xxxxxxxx)", status);
1522     expect2(new DecimalFormat(formatStr2, US, status),
1523            -10456.37,     "(10,456.37xxxxx)", status);
1524     expect2(new DecimalFormat(formatStr2, US, status),
1525            -1120456.37,   "(1,120,456.37xx)", status);
1526     expect2(new DecimalFormat(formatStr2, US, status),
1527            -112045600.37, "(112,045,600.37)", status);
1528     expect2(new DecimalFormat(formatStr2, US, status),
1529            -1252045600.37,"(1,252,045,600.37)", status);
1530 
1531     expect2(new DecimalFormat(formatStr2, US, status),
1532            int32_t(10),  "10.0xxxxxxxxxxxx", status);
1533     expect2(new DecimalFormat(formatStr2, US, status),
1534            int32_t(1000),"1,000.0xxxxxxxxx", status);
1535     expect2(new DecimalFormat(formatStr2, US, status),
1536            int32_t(1000000),"1,000,000.0xxxxx", status);
1537     expect2(new DecimalFormat(formatStr2, US, status),
1538            100.37,       "100.37xxxxxxxxxx", status);
1539     expect2(new DecimalFormat(formatStr2, US, status),
1540            10456.37,     "10,456.37xxxxxxx", status);
1541     expect2(new DecimalFormat(formatStr2, US, status),
1542            1120456.37,   "1,120,456.37xxxx", status);
1543     expect2(new DecimalFormat(formatStr2, US, status),
1544            112045600.37, "112,045,600.37xx", status);
1545     expect2(new DecimalFormat(formatStr2, US, status),
1546            10252045600.37,"10,252,045,600.37", status);
1547 
1548     //testing the setPadCharacter(UnicodeString) and getPadCharacterString()
1549     DecimalFormat fmt("#", US, status);
1550     CHECK(status, "DecimalFormat constructor");
1551     UnicodeString padString("P");
1552     fmt.setPadCharacter(padString);
1553     expectPad(fmt, "*P##.##", DecimalFormat::kPadBeforePrefix, 5, padString);
1554     fmt.setPadCharacter((UnicodeString)"^");
1555     expectPad(fmt, "*^#", DecimalFormat::kPadBeforePrefix, 1, (UnicodeString)"^");
1556     //commented untill implementation is complete
1557   /*  fmt.setPadCharacter((UnicodeString)"^^^");
1558     expectPad(fmt, "*^^^#", DecimalFormat::kPadBeforePrefix, 3, (UnicodeString)"^^^");
1559     padString.remove();
1560     padString.append((UChar)0x0061);
1561     padString.append((UChar)0x0302);
1562     fmt.setPadCharacter(padString);
1563     UChar patternChars[]={0x002a, 0x0061, 0x0302, 0x0061, 0x0302, 0x0023, 0x0000};
1564     UnicodeString pattern(patternChars);
1565     expectPad(fmt, pattern , DecimalFormat::kPadBeforePrefix, 4, padString);
1566  */
1567 
1568 }
1569 
1570 /**
1571  * Upgrade to alphaWorks
1572  */
TestPatterns2(void)1573 void NumberFormatTest::TestPatterns2(void) {
1574     UErrorCode status = U_ZERO_ERROR;
1575     DecimalFormatSymbols US(Locale::getUS(), status);
1576     CHECK(status, "DecimalFormatSymbols constructor");
1577 
1578     DecimalFormat fmt("#", US, status);
1579     CHECK(status, "DecimalFormat constructor");
1580 
1581     UChar hat = 0x005E; /*^*/
1582 
1583     expectPad(fmt, "*^#", DecimalFormat::kPadBeforePrefix, 1, hat);
1584     expectPad(fmt, "$*^#", DecimalFormat::kPadAfterPrefix, 2, hat);
1585     expectPad(fmt, "#*^", DecimalFormat::kPadBeforeSuffix, 1, hat);
1586     expectPad(fmt, "#$*^", DecimalFormat::kPadAfterSuffix, 2, hat);
1587     expectPad(fmt, "$*^$#", ILLEGAL);
1588     expectPad(fmt, "#$*^$", ILLEGAL);
1589     expectPad(fmt, "'pre'#,##0*x'post'", DecimalFormat::kPadBeforeSuffix,
1590               12, (UChar)0x0078 /*x*/);
1591     expectPad(fmt, "''#0*x", DecimalFormat::kPadBeforeSuffix,
1592               3, (UChar)0x0078 /*x*/);
1593     expectPad(fmt, "'I''ll'*a###.##", DecimalFormat::kPadAfterPrefix,
1594               10, (UChar)0x0061 /*a*/);
1595 
1596     fmt.applyPattern("AA#,##0.00ZZ", status);
1597     CHECK(status, "applyPattern");
1598     fmt.setPadCharacter(hat);
1599 
1600     fmt.setFormatWidth(10);
1601 
1602     fmt.setPadPosition(DecimalFormat::kPadBeforePrefix);
1603     expectPat(fmt, "*^AA#,##0.00ZZ");
1604 
1605     fmt.setPadPosition(DecimalFormat::kPadBeforeSuffix);
1606     expectPat(fmt, "AA#,##0.00*^ZZ");
1607 
1608     fmt.setPadPosition(DecimalFormat::kPadAfterSuffix);
1609     expectPat(fmt, "AA#,##0.00ZZ*^");
1610 
1611     //            12  3456789012
1612     UnicodeString exp("AA*^#,##0.00ZZ", "");
1613     fmt.setFormatWidth(12);
1614     fmt.setPadPosition(DecimalFormat::kPadAfterPrefix);
1615     expectPat(fmt, exp);
1616 
1617     fmt.setFormatWidth(13);
1618     //              12  34567890123
1619     expectPat(fmt, "AA*^##,##0.00ZZ");
1620 
1621     fmt.setFormatWidth(14);
1622     //              12  345678901234
1623     expectPat(fmt, "AA*^###,##0.00ZZ");
1624 
1625     fmt.setFormatWidth(15);
1626     //              12  3456789012345
1627     expectPat(fmt, "AA*^####,##0.00ZZ"); // This is the interesting case
1628 
1629     fmt.setFormatWidth(16);
1630     //              12  34567890123456
1631     expectPat(fmt, "AA*^#,###,##0.00ZZ");
1632 }
1633 
TestSurrogateSupport(void)1634 void NumberFormatTest::TestSurrogateSupport(void) {
1635     UErrorCode status = U_ZERO_ERROR;
1636     DecimalFormatSymbols custom(Locale::getUS(), status);
1637     CHECK(status, "DecimalFormatSymbols constructor");
1638 
1639     custom.setSymbol(DecimalFormatSymbols::kDecimalSeparatorSymbol, "decimal");
1640     custom.setSymbol(DecimalFormatSymbols::kPlusSignSymbol, "plus");
1641     custom.setSymbol(DecimalFormatSymbols::kMinusSignSymbol, " minus ");
1642     custom.setSymbol(DecimalFormatSymbols::kExponentialSymbol, "exponent");
1643 
1644     UnicodeString patternStr("*\\U00010000##.##", "");
1645     patternStr = patternStr.unescape();
1646     UnicodeString expStr("\\U00010000\\U00010000\\U00010000\\U000100000", "");
1647     expStr = expStr.unescape();
1648     expect2(new DecimalFormat(patternStr, custom, status),
1649            int32_t(0), expStr, status);
1650 
1651     status = U_ZERO_ERROR;
1652     expect2(new DecimalFormat("*^##.##", custom, status),
1653            int32_t(0), "^^^^0", status);
1654     status = U_ZERO_ERROR;
1655     expect2(new DecimalFormat("##.##", custom, status),
1656            -1.3, " minus 1decimal3", status);
1657     status = U_ZERO_ERROR;
1658     expect2(new DecimalFormat("##0.0####E0 'g-m/s^2'", custom, status),
1659            int32_t(0), "0decimal0exponent0 g-m/s^2", status);
1660     status = U_ZERO_ERROR;
1661     expect(new DecimalFormat("##0.0####E0 'g-m/s^2'", custom, status),
1662            1.0/3, "333decimal333exponent minus 3 g-m/s^2", status);
1663     status = U_ZERO_ERROR;
1664     expect2(new DecimalFormat("##0.0#### 'g-m/s^2'", custom, status),
1665            int32_t(0), "0decimal0 g-m/s^2", status);
1666     status = U_ZERO_ERROR;
1667     expect(new DecimalFormat("##0.0#### 'g-m/s^2'", custom, status),
1668            1.0/3, "0decimal33333 g-m/s^2", status);
1669 
1670     UnicodeString zero((UChar32)0x10000);
1671     UnicodeString one((UChar32)0x10001);
1672     UnicodeString two((UChar32)0x10002);
1673     UnicodeString five((UChar32)0x10005);
1674     custom.setSymbol(DecimalFormatSymbols::kZeroDigitSymbol, zero);
1675     custom.setSymbol(DecimalFormatSymbols::kOneDigitSymbol, one);
1676     custom.setSymbol(DecimalFormatSymbols::kTwoDigitSymbol, two);
1677     custom.setSymbol(DecimalFormatSymbols::kFiveDigitSymbol, five);
1678     expStr = UnicodeString("\\U00010001decimal\\U00010002\\U00010005\\U00010000", "");
1679     expStr = expStr.unescape();
1680     status = U_ZERO_ERROR;
1681     expect2(new DecimalFormat("##0.000", custom, status),
1682            1.25, expStr, status);
1683 
1684     custom.setSymbol(DecimalFormatSymbols::kZeroDigitSymbol, (UChar)0x30);
1685     custom.setSymbol(DecimalFormatSymbols::kCurrencySymbol, "units of money");
1686     custom.setSymbol(DecimalFormatSymbols::kMonetarySeparatorSymbol, "money separator");
1687     patternStr = UNICODE_STRING_SIMPLE("0.00 \\u00A4' in your bank account'");
1688     patternStr = patternStr.unescape();
1689     expStr = UnicodeString(" minus 20money separator00 units of money in your bank account", "");
1690     status = U_ZERO_ERROR;
1691     expect2(new DecimalFormat(patternStr, custom, status),
1692            int32_t(-20), expStr, status);
1693 
1694     custom.setSymbol(DecimalFormatSymbols::kPercentSymbol, "percent");
1695     patternStr = "'You''ve lost ' -0.00 %' of your money today'";
1696     patternStr = patternStr.unescape();
1697     expStr = UnicodeString(" minus You've lost   minus 2000decimal00 percent of your money today", "");
1698     status = U_ZERO_ERROR;
1699     expect2(new DecimalFormat(patternStr, custom, status),
1700            int32_t(-20), expStr, status);
1701 }
1702 
TestCurrencyPatterns(void)1703 void NumberFormatTest::TestCurrencyPatterns(void) {
1704     int32_t i, locCount;
1705     const Locale* locs = NumberFormat::getAvailableLocales(locCount);
1706     for (i=0; i<locCount; ++i) {
1707         UErrorCode ec = U_ZERO_ERROR;
1708         NumberFormat* nf = NumberFormat::createCurrencyInstance(locs[i], ec);
1709         if (U_FAILURE(ec)) {
1710             errln("FAIL: Can't create NumberFormat(%s) - %s", locs[i].getName(), u_errorName(ec));
1711         } else {
1712             // Make sure currency formats do not have a variable number
1713             // of fraction digits
1714             int32_t min = nf->getMinimumFractionDigits();
1715             int32_t max = nf->getMaximumFractionDigits();
1716             if (min != max) {
1717                 UnicodeString a, b;
1718                 nf->format(1.0, a);
1719                 nf->format(1.125, b);
1720                 errln((UnicodeString)"FAIL: " + locs[i].getName() +
1721                       " min fraction digits != max fraction digits; "
1722                       "x 1.0 => " + escape(a) +
1723                       "; x 1.125 => " + escape(b));
1724             }
1725 
1726             // Make sure EURO currency formats have exactly 2 fraction digits
1727             DecimalFormat* df = dynamic_cast<DecimalFormat*>(nf);
1728             if (df != NULL) {
1729                 if (u_strcmp(EUR, df->getCurrency()) == 0) {
1730                     if (min != 2 || max != 2) {
1731                         UnicodeString a;
1732                         nf->format(1.0, a);
1733                         errln((UnicodeString)"FAIL: " + locs[i].getName() +
1734                               " is a EURO format but it does not have 2 fraction digits; "
1735                               "x 1.0 => " +
1736                               escape(a));
1737                     }
1738                 }
1739             }
1740         }
1741         delete nf;
1742     }
1743 }
1744 
TestRegCurrency(void)1745 void NumberFormatTest::TestRegCurrency(void) {
1746 #if !UCONFIG_NO_SERVICE
1747     UErrorCode status = U_ZERO_ERROR;
1748     UChar USD[4];
1749     ucurr_forLocale("en_US", USD, 4, &status);
1750     UChar YEN[4];
1751     ucurr_forLocale("ja_JP", YEN, 4, &status);
1752     UChar TMP[4];
1753     static const UChar QQQ[] = {0x51, 0x51, 0x51, 0};
1754     if(U_FAILURE(status)) {
1755         errcheckln(status, "Unable to get currency for locale, error %s", u_errorName(status));
1756         return;
1757     }
1758 
1759     UCurrRegistryKey enkey = ucurr_register(YEN, "en_US", &status);
1760     UCurrRegistryKey enUSEUROkey = ucurr_register(QQQ, "en_US_EURO", &status);
1761 
1762     ucurr_forLocale("en_US", TMP, 4, &status);
1763     if (u_strcmp(YEN, TMP) != 0) {
1764         errln("FAIL: didn't return YEN registered for en_US");
1765     }
1766 
1767     ucurr_forLocale("en_US_EURO", TMP, 4, &status);
1768     if (u_strcmp(QQQ, TMP) != 0) {
1769         errln("FAIL: didn't return QQQ for en_US_EURO");
1770     }
1771 
1772     int32_t fallbackLen = ucurr_forLocale("en_XX_BAR", TMP, 4, &status);
1773     if (fallbackLen) {
1774         errln("FAIL: tried to fallback en_XX_BAR");
1775     }
1776     status = U_ZERO_ERROR; // reset
1777 
1778     if (!ucurr_unregister(enkey, &status)) {
1779         errln("FAIL: couldn't unregister enkey");
1780     }
1781 
1782     ucurr_forLocale("en_US", TMP, 4, &status);
1783     if (u_strcmp(USD, TMP) != 0) {
1784         errln("FAIL: didn't return USD for en_US after unregister of en_US");
1785     }
1786     status = U_ZERO_ERROR; // reset
1787 
1788     ucurr_forLocale("en_US_EURO", TMP, 4, &status);
1789     if (u_strcmp(QQQ, TMP) != 0) {
1790         errln("FAIL: didn't return QQQ for en_US_EURO after unregister of en_US");
1791     }
1792 
1793     ucurr_forLocale("en_US_BLAH", TMP, 4, &status);
1794     if (u_strcmp(USD, TMP) != 0) {
1795         errln("FAIL: could not find USD for en_US_BLAH after unregister of en");
1796     }
1797     status = U_ZERO_ERROR; // reset
1798 
1799     if (!ucurr_unregister(enUSEUROkey, &status)) {
1800         errln("FAIL: couldn't unregister enUSEUROkey");
1801     }
1802 
1803     ucurr_forLocale("en_US_EURO", TMP, 4, &status);
1804     if (u_strcmp(EUR, TMP) != 0) {
1805         errln("FAIL: didn't return EUR for en_US_EURO after unregister of en_US_EURO");
1806     }
1807     status = U_ZERO_ERROR; // reset
1808 #endif
1809 }
1810 
TestCurrencyNames(void)1811 void NumberFormatTest::TestCurrencyNames(void) {
1812     // Do a basic check of getName()
1813     // USD { "US$", "US Dollar"            } // 04/04/1792-
1814     UErrorCode ec = U_ZERO_ERROR;
1815     static const UChar USD[] = {0x55, 0x53, 0x44, 0}; /*USD*/
1816     static const UChar USX[] = {0x55, 0x53, 0x58, 0}; /*USX*/
1817     static const UChar CAD[] = {0x43, 0x41, 0x44, 0}; /*CAD*/
1818     static const UChar ITL[] = {0x49, 0x54, 0x4C, 0}; /*ITL*/
1819     UBool isChoiceFormat;
1820     int32_t len;
1821     const UBool possibleDataError = TRUE;
1822     // Warning: HARD-CODED LOCALE DATA in this test.  If it fails, CHECK
1823     // THE LOCALE DATA before diving into the code.
1824     assertEquals("USD.getName(SYMBOL_NAME)",
1825                  UnicodeString("$"),
1826                  UnicodeString(ucurr_getName(USD, "en",
1827                                              UCURR_SYMBOL_NAME,
1828                                              &isChoiceFormat, &len, &ec)),
1829                                              possibleDataError);
1830     assertEquals("USD.getName(LONG_NAME)",
1831                  UnicodeString("US Dollar"),
1832                  UnicodeString(ucurr_getName(USD, "en",
1833                                              UCURR_LONG_NAME,
1834                                              &isChoiceFormat, &len, &ec)),
1835                                              possibleDataError);
1836     assertEquals("CAD.getName(SYMBOL_NAME)",
1837                  UnicodeString("CA$"),
1838                  UnicodeString(ucurr_getName(CAD, "en",
1839                                              UCURR_SYMBOL_NAME,
1840                                              &isChoiceFormat, &len, &ec)),
1841                                              possibleDataError);
1842     assertEquals("CAD.getName(SYMBOL_NAME)",
1843                  UnicodeString("$"),
1844                  UnicodeString(ucurr_getName(CAD, "en_CA",
1845                                              UCURR_SYMBOL_NAME,
1846                                              &isChoiceFormat, &len, &ec)),
1847                                              possibleDataError);
1848     assertEquals("USD.getName(SYMBOL_NAME) in en_AU",
1849                  UnicodeString("US$"),
1850                  UnicodeString(ucurr_getName(USD, "en_AU",
1851                                              UCURR_SYMBOL_NAME,
1852                                              &isChoiceFormat, &len, &ec)),
1853                                              possibleDataError);
1854     assertEquals("CAD.getName(SYMBOL_NAME)",
1855                  UnicodeString("CA$"),
1856                  UnicodeString(ucurr_getName(CAD, "en_AU",
1857                                              UCURR_SYMBOL_NAME,
1858                                              &isChoiceFormat, &len, &ec)),
1859                                              possibleDataError);
1860     assertEquals("USX.getName(LONG_NAME)",
1861                  UnicodeString("USX"),
1862                  UnicodeString(ucurr_getName(USX, "en_US",
1863                                              UCURR_LONG_NAME,
1864                                              &isChoiceFormat, &len, &ec)),
1865                                              possibleDataError);
1866     assertSuccess("ucurr_getName", ec);
1867 
1868     ec = U_ZERO_ERROR;
1869 
1870     // Test that a default or fallback warning is being returned. JB 4239.
1871     ucurr_getName(CAD, "es_ES", UCURR_LONG_NAME, &isChoiceFormat,
1872                             &len, &ec);
1873     assertTrue("ucurr_getName (es_ES fallback)",
1874                     U_USING_FALLBACK_WARNING == ec, TRUE, possibleDataError);
1875 
1876     ucurr_getName(CAD, "zh_TW", UCURR_LONG_NAME, &isChoiceFormat,
1877                             &len, &ec);
1878     assertTrue("ucurr_getName (zh_TW fallback)",
1879                     U_USING_FALLBACK_WARNING == ec, TRUE, possibleDataError);
1880 
1881     ucurr_getName(CAD, "en_US", UCURR_LONG_NAME, &isChoiceFormat,
1882                             &len, &ec);
1883     assertTrue("ucurr_getName (en_US default)",
1884                     U_USING_DEFAULT_WARNING == ec || U_USING_FALLBACK_WARNING == ec, TRUE);
1885 
1886     ucurr_getName(CAD, "ti", UCURR_LONG_NAME, &isChoiceFormat,
1887                             &len, &ec);
1888     assertTrue("ucurr_getName (ti default)",
1889                     U_USING_DEFAULT_WARNING == ec, TRUE);
1890 
1891     // Test that a default warning is being returned when falling back to root. JB 4536.
1892     ucurr_getName(ITL, "cy", UCURR_LONG_NAME, &isChoiceFormat,
1893                             &len, &ec);
1894     assertTrue("ucurr_getName (cy default to root)",
1895                     U_USING_DEFAULT_WARNING == ec, TRUE);
1896 
1897     // TODO add more tests later
1898 }
1899 
TestCurrencyUnit(void)1900 void NumberFormatTest::TestCurrencyUnit(void){
1901     UErrorCode ec = U_ZERO_ERROR;
1902     static const UChar USD[] = {85, 83, 68, 0}; /*USD*/
1903     static const UChar BAD[] = {63, 63, 63, 0}; /*???*/
1904     static const UChar BAD2[] = {63, 63, 65, 0}; /*???*/
1905     CurrencyUnit cu(USD, ec);
1906     assertSuccess("CurrencyUnit", ec);
1907 
1908     const UChar * r = cu.getISOCurrency(); // who is the buffer owner ?
1909     assertEquals("getISOCurrency()", USD, r);
1910 
1911     CurrencyUnit cu2(cu);
1912     if (!(cu2 == cu)){
1913         errln("CurrencyUnit copy constructed object should be same");
1914     }
1915 
1916     CurrencyUnit * cu3 = (CurrencyUnit *)cu.clone();
1917     if (!(*cu3 == cu)){
1918         errln("CurrencyUnit cloned object should be same");
1919     }
1920     CurrencyUnit bad(BAD, ec);
1921     assertSuccess("CurrencyUnit", ec);
1922     if (cu.getIndex() == bad.getIndex()) {
1923         errln("Indexes of different currencies should differ.");
1924     }
1925     CurrencyUnit bad2(BAD2, ec);
1926     assertSuccess("CurrencyUnit", ec);
1927     if (bad2.getIndex() != bad.getIndex()) {
1928         errln("Indexes of unrecognized currencies should be the same.");
1929     }
1930     if (bad == bad2) {
1931         errln("Different unrecognized currencies should not be equal.");
1932     }
1933     bad = bad2;
1934     if (bad != bad2) {
1935         errln("Currency unit assignment should be the same.");
1936     }
1937     delete cu3;
1938 }
1939 
TestCurrencyAmount(void)1940 void NumberFormatTest::TestCurrencyAmount(void){
1941     UErrorCode ec = U_ZERO_ERROR;
1942     static const UChar USD[] = {85, 83, 68, 0}; /*USD*/
1943     CurrencyAmount ca(9, USD, ec);
1944     assertSuccess("CurrencyAmount", ec);
1945 
1946     CurrencyAmount ca2(ca);
1947     if (!(ca2 == ca)){
1948         errln("CurrencyAmount copy constructed object should be same");
1949     }
1950 
1951     ca2=ca;
1952     if (!(ca2 == ca)){
1953         errln("CurrencyAmount assigned object should be same");
1954     }
1955 
1956     CurrencyAmount *ca3 = (CurrencyAmount *)ca.clone();
1957     if (!(*ca3 == ca)){
1958         errln("CurrencyAmount cloned object should be same");
1959     }
1960     delete ca3;
1961 }
1962 
TestSymbolsWithBadLocale(void)1963 void NumberFormatTest::TestSymbolsWithBadLocale(void) {
1964     Locale locDefault;
1965     static const char *badLocales[] = {
1966         // length < ULOC_FULLNAME_CAPACITY
1967         "x-crazy_ZZ_MY_SPECIAL_ADMINISTRATION_REGION_NEEDS_A_SPECIAL_VARIANT_WITH_A_REALLY_REALLY_REALLY_REALLY_REALLY_REALLY_REALLY_LONG_NAME",
1968 
1969         // length > ULOC_FULLNAME_CAPACITY
1970         "x-crazy_ZZ_MY_SPECIAL_ADMINISTRATION_REGION_NEEDS_A_SPECIAL_VARIANT_WITH_A_REALLY_REALLY_REALLY_REALLY_REALLY_REALLY_REALLY_REALLY_REALLY_REALLY_REALLY_LONG_NAME"
1971     }; // expect U_USING_DEFAULT_WARNING for both
1972 
1973     unsigned int i;
1974     for (i = 0; i < sizeof(badLocales) / sizeof(char*); i++) {
1975         const char *localeName = badLocales[i];
1976         Locale locBad(localeName);
1977         UErrorCode status = U_ZERO_ERROR;
1978         UnicodeString intlCurrencySymbol((UChar)0xa4);
1979 
1980         intlCurrencySymbol.append((UChar)0xa4);
1981 
1982         logln("Current locale is %s", Locale::getDefault().getName());
1983         Locale::setDefault(locBad, status);
1984         logln("Current locale is %s", Locale::getDefault().getName());
1985         DecimalFormatSymbols mySymbols(status);
1986         if (status != U_USING_DEFAULT_WARNING) {
1987             errln("DecimalFormatSymbols should return U_USING_DEFAULT_WARNING.");
1988         }
1989         if (strcmp(mySymbols.getLocale().getName(), locBad.getName()) != 0) {
1990             errln("DecimalFormatSymbols does not have the right locale.", locBad.getName());
1991         }
1992         int symbolEnum = (int)DecimalFormatSymbols::kDecimalSeparatorSymbol;
1993         for (; symbolEnum < (int)DecimalFormatSymbols::kFormatSymbolCount; symbolEnum++) {
1994             UnicodeString symbolString = mySymbols.getSymbol((DecimalFormatSymbols::ENumberFormatSymbol)symbolEnum);
1995             logln(UnicodeString("DecimalFormatSymbols[") + symbolEnum + UnicodeString("] = ") + prettify(symbolString));
1996             if (symbolString.length() == 0
1997                 && symbolEnum != (int)DecimalFormatSymbols::kGroupingSeparatorSymbol
1998                 && symbolEnum != (int)DecimalFormatSymbols::kMonetaryGroupingSeparatorSymbol)
1999             {
2000                 errln("DecimalFormatSymbols has an empty string at index %d.", symbolEnum);
2001             }
2002         }
2003 
2004         status = U_ZERO_ERROR;
2005         Locale::setDefault(locDefault, status);
2006         logln("Current locale is %s", Locale::getDefault().getName());
2007     }
2008 }
2009 
2010 /**
2011  * Check that adoptDecimalFormatSymbols and setDecimalFormatSymbols
2012  * behave the same, except for memory ownership semantics. (No
2013  * version of this test on Java, since Java has only one method.)
2014  */
TestAdoptDecimalFormatSymbols(void)2015 void NumberFormatTest::TestAdoptDecimalFormatSymbols(void) {
2016     UErrorCode ec = U_ZERO_ERROR;
2017     DecimalFormatSymbols *sym = new DecimalFormatSymbols(Locale::getUS(), ec);
2018     if (U_FAILURE(ec)) {
2019         errcheckln(ec, "Fail: DecimalFormatSymbols constructor - %s", u_errorName(ec));
2020         delete sym;
2021         return;
2022     }
2023     UnicodeString pat(" #,##0.00");
2024     pat.insert(0, (UChar)0x00A4);
2025     DecimalFormat fmt(pat, sym, ec);
2026     if (U_FAILURE(ec)) {
2027         errln("Fail: DecimalFormat constructor");
2028         return;
2029     }
2030 
2031     UnicodeString str;
2032     fmt.format(2350.75, str);
2033     if (str == "$ 2,350.75") {
2034         logln(str);
2035     } else {
2036         dataerrln("Fail: " + str + ", expected $ 2,350.75");
2037     }
2038 
2039     sym = new DecimalFormatSymbols(Locale::getUS(), ec);
2040     if (U_FAILURE(ec)) {
2041         errln("Fail: DecimalFormatSymbols constructor");
2042         delete sym;
2043         return;
2044     }
2045     sym->setSymbol(DecimalFormatSymbols::kCurrencySymbol, "Q");
2046     fmt.adoptDecimalFormatSymbols(sym);
2047 
2048     str.truncate(0);
2049     fmt.format(2350.75, str);
2050     if (str == "Q 2,350.75") {
2051         logln(str);
2052     } else {
2053         dataerrln("Fail: adoptDecimalFormatSymbols -> " + str + ", expected Q 2,350.75");
2054     }
2055 
2056     sym = new DecimalFormatSymbols(Locale::getUS(), ec);
2057     if (U_FAILURE(ec)) {
2058         errln("Fail: DecimalFormatSymbols constructor");
2059         delete sym;
2060         return;
2061     }
2062     DecimalFormat fmt2(pat, sym, ec);
2063     if (U_FAILURE(ec)) {
2064         errln("Fail: DecimalFormat constructor");
2065         return;
2066     }
2067 
2068     DecimalFormatSymbols sym2(Locale::getUS(), ec);
2069     if (U_FAILURE(ec)) {
2070         errln("Fail: DecimalFormatSymbols constructor");
2071         return;
2072     }
2073     sym2.setSymbol(DecimalFormatSymbols::kCurrencySymbol, "Q");
2074     fmt2.setDecimalFormatSymbols(sym2);
2075 
2076     str.truncate(0);
2077     fmt2.format(2350.75, str);
2078     if (str == "Q 2,350.75") {
2079         logln(str);
2080     } else {
2081         dataerrln("Fail: setDecimalFormatSymbols -> " + str + ", expected Q 2,350.75");
2082     }
2083 }
2084 
TestPerMill()2085 void NumberFormatTest::TestPerMill() {
2086     UErrorCode ec = U_ZERO_ERROR;
2087     UnicodeString str;
2088     DecimalFormat fmt(ctou("###.###\\u2030"), ec);
2089     if (!assertSuccess("DecimalFormat ct", ec)) return;
2090     assertEquals("0.4857 x ###.###\\u2030",
2091                  ctou("485.7\\u2030"), fmt.format(0.4857, str));
2092 
2093     DecimalFormatSymbols sym(Locale::getUS(), ec);
2094     sym.setSymbol(DecimalFormatSymbols::kPerMillSymbol, ctou("m"));
2095     DecimalFormat fmt2("", sym, ec);
2096     fmt2.applyLocalizedPattern("###.###m", ec);
2097     if (!assertSuccess("setup", ec)) return;
2098     str.truncate(0);
2099     assertEquals("0.4857 x ###.###m",
2100                  "485.7m", fmt2.format(0.4857, str));
2101 }
2102 
2103 /**
2104  * Generic test for patterns that should be legal/illegal.
2105  */
TestIllegalPatterns()2106 void NumberFormatTest::TestIllegalPatterns() {
2107     // Test cases:
2108     // Prefix with "-:" for illegal patterns
2109     // Prefix with "+:" for legal patterns
2110     const char* DATA[] = {
2111         // Unquoted special characters in the suffix are illegal
2112         "-:000.000|###",
2113         "+:000.000'|###'",
2114         0
2115     };
2116     for (int32_t i=0; DATA[i]; ++i) {
2117         const char* pat=DATA[i];
2118         UBool valid = (*pat) == '+';
2119         pat += 2;
2120         UErrorCode ec = U_ZERO_ERROR;
2121         DecimalFormat fmt(pat, ec); // locale doesn't matter here
2122         if (U_SUCCESS(ec) == valid) {
2123             logln("Ok: pattern \"%s\": %s",
2124                   pat, u_errorName(ec));
2125         } else {
2126             errcheckln(ec, "FAIL: pattern \"%s\" should have %s; got %s",
2127                   pat, (valid?"succeeded":"failed"),
2128                   u_errorName(ec));
2129         }
2130     }
2131 }
2132 
2133 //----------------------------------------------------------------------
2134 
2135 static const char* KEYWORDS[] = {
2136     /*0*/ "ref=", // <reference pattern to parse numbers>
2137     /*1*/ "loc=", // <locale for formats>
2138     /*2*/ "f:",   // <pattern or '-'> <number> <exp. string>
2139     /*3*/ "fp:",  // <pattern or '-'> <number> <exp. string> <exp. number>
2140     /*4*/ "rt:",  // <pattern or '-'> <(exp.) number> <(exp.) string>
2141     /*5*/ "p:",   // <pattern or '-'> <string> <exp. number>
2142     /*6*/ "perr:", // <pattern or '-'> <invalid string>
2143     /*7*/ "pat:", // <pattern or '-'> <exp. toPattern or '-' or 'err'>
2144     /*8*/ "fpc:", // <pattern or '-'> <curr.amt> <exp. string> <exp. curr.amt>
2145     0
2146 };
2147 
2148 /**
2149  * Return an integer representing the next token from this
2150  * iterator.  The integer will be an index into the given list, or
2151  * -1 if there are no more tokens, or -2 if the token is not on
2152  * the list.
2153  */
keywordIndex(const UnicodeString & tok)2154 static int32_t keywordIndex(const UnicodeString& tok) {
2155     for (int32_t i=0; KEYWORDS[i]!=0; ++i) {
2156         if (tok==KEYWORDS[i]) {
2157             return i;
2158         }
2159     }
2160     return -1;
2161 }
2162 
2163 /**
2164  * Parse a CurrencyAmount using the given NumberFormat, with
2165  * the 'delim' character separating the number and the currency.
2166  */
parseCurrencyAmount(const UnicodeString & str,const NumberFormat & fmt,UChar delim,Formattable & result,UErrorCode & ec)2167 static void parseCurrencyAmount(const UnicodeString& str,
2168                                 const NumberFormat& fmt,
2169                                 UChar delim,
2170                                 Formattable& result,
2171                                 UErrorCode& ec) {
2172     UnicodeString num, cur;
2173     int32_t i = str.indexOf(delim);
2174     str.extractBetween(0, i, num);
2175     str.extractBetween(i+1, INT32_MAX, cur);
2176     Formattable n;
2177     fmt.parse(num, n, ec);
2178     result.adoptObject(new CurrencyAmount(n, cur.getTerminatedBuffer(), ec));
2179 }
2180 
TestCases()2181 void NumberFormatTest::TestCases() {
2182     UErrorCode ec = U_ZERO_ERROR;
2183     TextFile reader("NumberFormatTestCases.txt", "UTF8", ec);
2184     if (U_FAILURE(ec)) {
2185         dataerrln("Couldn't open NumberFormatTestCases.txt");
2186         return;
2187     }
2188     TokenIterator tokens(&reader);
2189 
2190     Locale loc("en", "US", "");
2191     DecimalFormat *ref = 0, *fmt = 0;
2192     MeasureFormat *mfmt = 0;
2193     UnicodeString pat, tok, mloc, str, out, where, currAmt;
2194     Formattable n;
2195 
2196     for (;;) {
2197         ec = U_ZERO_ERROR;
2198         if (!tokens.next(tok, ec)) {
2199             break;
2200         }
2201         where = UnicodeString("(") + tokens.getLineNumber() + ") ";
2202         int32_t cmd = keywordIndex(tok);
2203         switch (cmd) {
2204         case 0:
2205             // ref= <reference pattern>
2206             if (!tokens.next(tok, ec)) goto error;
2207             delete ref;
2208             ref = new DecimalFormat(tok,
2209                       new DecimalFormatSymbols(Locale::getUS(), ec), ec);
2210             if (U_FAILURE(ec)) {
2211                 dataerrln("Error constructing DecimalFormat");
2212                 goto error;
2213             }
2214             break;
2215         case 1:
2216             // loc= <locale>
2217             if (!tokens.next(tok, ec)) goto error;
2218             loc = Locale::createFromName(CharString().appendInvariantChars(tok, ec).data());
2219             break;
2220         case 2: // f:
2221         case 3: // fp:
2222         case 4: // rt:
2223         case 5: // p:
2224             if (!tokens.next(tok, ec)) goto error;
2225             if (tok != "-") {
2226                 pat = tok;
2227                 delete fmt;
2228                 fmt = new DecimalFormat(pat, new DecimalFormatSymbols(loc, ec), ec);
2229                 if (U_FAILURE(ec)) {
2230                     errln("FAIL: " + where + "Pattern \"" + pat + "\": " + u_errorName(ec));
2231                     ec = U_ZERO_ERROR;
2232                     if (!tokens.next(tok, ec)) goto error;
2233                     if (!tokens.next(tok, ec)) goto error;
2234                     if (cmd == 3) {
2235                         if (!tokens.next(tok, ec)) goto error;
2236                     }
2237                     continue;
2238                 }
2239             }
2240             if (cmd == 2 || cmd == 3 || cmd == 4) {
2241                 // f: <pattern or '-'> <number> <exp. string>
2242                 // fp: <pattern or '-'> <number> <exp. string> <exp. number>
2243                 // rt: <pattern or '-'> <number> <string>
2244                 UnicodeString num;
2245                 if (!tokens.next(num, ec)) goto error;
2246                 if (!tokens.next(str, ec)) goto error;
2247                 ref->parse(num, n, ec);
2248                 assertSuccess("parse", ec);
2249                 assertEquals(where + "\"" + pat + "\".format(" + num + ")",
2250                              str, fmt->format(n, out.remove(), ec));
2251                 assertSuccess("format", ec);
2252                 if (cmd == 3) { // fp:
2253                     if (!tokens.next(num, ec)) goto error;
2254                     ref->parse(num, n, ec);
2255                     assertSuccess("parse", ec);
2256                 }
2257                 if (cmd != 2) { // != f:
2258                     Formattable m;
2259                     fmt->parse(str, m, ec);
2260                     assertSuccess("parse", ec);
2261                     assertEquals(where + "\"" + pat + "\".parse(\"" + str + "\")",
2262                                  n, m);
2263                 }
2264             }
2265             // p: <pattern or '-'> <string to parse> <exp. number>
2266             else {
2267                 UnicodeString expstr;
2268                 if (!tokens.next(str, ec)) goto error;
2269                 if (!tokens.next(expstr, ec)) goto error;
2270                 Formattable exp, n;
2271                 ref->parse(expstr, exp, ec);
2272                 assertSuccess("parse", ec);
2273                 fmt->parse(str, n, ec);
2274                 assertSuccess("parse", ec);
2275                 assertEquals(where + "\"" + pat + "\".parse(\"" + str + "\")",
2276                              exp, n);
2277             }
2278             break;
2279         case 8: // fpc:
2280             if (!tokens.next(tok, ec)) goto error;
2281             if (tok != "-") {
2282                 mloc = tok;
2283                 delete mfmt;
2284                 mfmt = MeasureFormat::createCurrencyFormat(
2285                     Locale::createFromName(
2286                         CharString().appendInvariantChars(mloc, ec).data()), ec);
2287                 if (U_FAILURE(ec)) {
2288                     errln("FAIL: " + where + "Loc \"" + mloc + "\": " + u_errorName(ec));
2289                     ec = U_ZERO_ERROR;
2290                     if (!tokens.next(tok, ec)) goto error;
2291                     if (!tokens.next(tok, ec)) goto error;
2292                     if (!tokens.next(tok, ec)) goto error;
2293                     continue;
2294                 }
2295             } else if (mfmt == NULL) {
2296                 errln("FAIL: " + where + "Loc \"" + mloc + "\": skip case using previous locale, no valid MeasureFormat");
2297                 if (!tokens.next(tok, ec)) goto error;
2298                 if (!tokens.next(tok, ec)) goto error;
2299                 if (!tokens.next(tok, ec)) goto error;
2300                 continue;
2301             }
2302             // fpc: <loc or '-'> <curr.amt> <exp. string> <exp. curr.amt>
2303             if (!tokens.next(currAmt, ec)) goto error;
2304             if (!tokens.next(str, ec)) goto error;
2305             parseCurrencyAmount(currAmt, *ref, (UChar)0x2F/*'/'*/, n, ec);
2306             if (assertSuccess("parseCurrencyAmount", ec)) {
2307                 assertEquals(where + "getCurrencyFormat(" + mloc + ").format(" + currAmt + ")",
2308                              str, mfmt->format(n, out.remove(), ec));
2309                 assertSuccess("format", ec);
2310             }
2311             if (!tokens.next(currAmt, ec)) goto error;
2312             parseCurrencyAmount(currAmt, *ref, (UChar)0x2F/*'/'*/, n, ec);
2313             if (assertSuccess("parseCurrencyAmount", ec)) {
2314                 Formattable m;
2315 
2316                 mfmt->parseObject(str, m, ec);
2317                 if (assertSuccess("parseCurrency", ec)) {
2318                     assertEquals(where + "getCurrencyFormat(" + mloc + ").parse(\"" + str + "\")",
2319                                  n, m);
2320                 } else {
2321                     errln("FAIL: source " + str);
2322                 }
2323             }
2324             break;
2325         case 6:
2326             // perr: <pattern or '-'> <invalid string>
2327             errln("FAIL: Under construction");
2328             goto done;
2329         case 7: {
2330             // pat: <pattern> <exp. toPattern, or '-' or 'err'>
2331             UnicodeString testpat;
2332             UnicodeString exppat;
2333             if (!tokens.next(testpat, ec)) goto error;
2334             if (!tokens.next(exppat, ec)) goto error;
2335             UBool err = exppat == "err";
2336             UBool existingPat = FALSE;
2337             if (testpat == "-") {
2338                 if (err) {
2339                     errln("FAIL: " + where + "Invalid command \"pat: - err\"");
2340                     continue;
2341                 }
2342                 existingPat = TRUE;
2343                 testpat = pat;
2344             }
2345             if (exppat == "-") exppat = testpat;
2346             DecimalFormat* f = 0;
2347             UErrorCode ec2 = U_ZERO_ERROR;
2348             if (existingPat) {
2349                 f = fmt;
2350             } else {
2351                 f = new DecimalFormat(testpat, ec2);
2352             }
2353             if (U_SUCCESS(ec2)) {
2354                 if (err) {
2355                     errln("FAIL: " + where + "Invalid pattern \"" + testpat +
2356                           "\" was accepted");
2357                 } else {
2358                     UnicodeString pat2;
2359                     assertEquals(where + "\"" + testpat + "\".toPattern()",
2360                                  exppat, f->toPattern(pat2));
2361                 }
2362             } else {
2363                 if (err) {
2364                     logln("Ok: " + where + "Invalid pattern \"" + testpat +
2365                           "\" failed: " + u_errorName(ec2));
2366                 } else {
2367                     errln("FAIL: " + where + "Valid pattern \"" + testpat +
2368                           "\" failed: " + u_errorName(ec2));
2369                 }
2370             }
2371             if (!existingPat) delete f;
2372             } break;
2373         case -1:
2374             errln("FAIL: " + where + "Unknown command \"" + tok + "\"");
2375             goto done;
2376         }
2377     }
2378     goto done;
2379 
2380  error:
2381     if (U_SUCCESS(ec)) {
2382         errln("FAIL: Unexpected EOF");
2383     } else {
2384         errcheckln(ec, "FAIL: " + where + "Unexpected " + u_errorName(ec));
2385     }
2386 
2387  done:
2388     delete mfmt;
2389     delete fmt;
2390     delete ref;
2391 }
2392 
2393 
2394 //----------------------------------------------------------------------
2395 // Support methods
2396 //----------------------------------------------------------------------
2397 
equalValue(const Formattable & a,const Formattable & b)2398 UBool NumberFormatTest::equalValue(const Formattable& a, const Formattable& b) {
2399     if (a.getType() == b.getType()) {
2400         return a == b;
2401     }
2402 
2403     if (a.getType() == Formattable::kLong) {
2404         if (b.getType() == Formattable::kInt64) {
2405             return a.getLong() == b.getLong();
2406         } else if (b.getType() == Formattable::kDouble) {
2407             return (double) a.getLong() == b.getDouble(); // TODO check use of double instead of long
2408         }
2409     } else if (a.getType() == Formattable::kDouble) {
2410         if (b.getType() == Formattable::kLong) {
2411             return a.getDouble() == (double) b.getLong();
2412         } else if (b.getType() == Formattable::kInt64) {
2413             return a.getDouble() == (double)b.getInt64();
2414         }
2415     } else if (a.getType() == Formattable::kInt64) {
2416         if (b.getType() == Formattable::kLong) {
2417                 return a.getInt64() == (int64_t)b.getLong();
2418         } else if (b.getType() == Formattable::kDouble) {
2419             return a.getInt64() == (int64_t)b.getDouble();
2420         }
2421     }
2422     return FALSE;
2423 }
2424 
expect3(NumberFormat & fmt,const Formattable & n,const UnicodeString & str)2425 void NumberFormatTest::expect3(NumberFormat& fmt, const Formattable& n, const UnicodeString& str) {
2426     // Don't round-trip format test, since we explicitly do it
2427     expect_rbnf(fmt, n, str, FALSE);
2428     expect_rbnf(fmt, str, n);
2429 }
2430 
expect2(NumberFormat & fmt,const Formattable & n,const UnicodeString & str)2431 void NumberFormatTest::expect2(NumberFormat& fmt, const Formattable& n, const UnicodeString& str) {
2432     // Don't round-trip format test, since we explicitly do it
2433     expect(fmt, n, str, FALSE);
2434     expect(fmt, str, n);
2435 }
2436 
expect2(NumberFormat * fmt,const Formattable & n,const UnicodeString & exp,UErrorCode status)2437 void NumberFormatTest::expect2(NumberFormat* fmt, const Formattable& n,
2438                                const UnicodeString& exp,
2439                                UErrorCode status) {
2440     if (fmt == NULL || U_FAILURE(status)) {
2441         dataerrln("FAIL: NumberFormat constructor");
2442     } else {
2443         expect2(*fmt, n, exp);
2444     }
2445     delete fmt;
2446 }
2447 
expect(NumberFormat & fmt,const UnicodeString & str,const Formattable & n)2448 void NumberFormatTest::expect(NumberFormat& fmt, const UnicodeString& str, const Formattable& n) {
2449     UErrorCode status = U_ZERO_ERROR;
2450     Formattable num;
2451     fmt.parse(str, num, status);
2452     if (U_FAILURE(status)) {
2453         dataerrln(UnicodeString("FAIL: Parse failed for \"") + str + "\" - " + u_errorName(status));
2454         return;
2455     }
2456     UnicodeString pat;
2457     ((DecimalFormat*) &fmt)->toPattern(pat);
2458     if (equalValue(num, n)) {
2459         logln(UnicodeString("Ok   \"") + str + "\" x " +
2460               pat + " = " +
2461               toString(num));
2462     } else {
2463         dataerrln(UnicodeString("FAIL \"") + str + "\" x " +
2464               pat + " = " +
2465               toString(num) + ", expected " + toString(n));
2466     }
2467 }
2468 
expect_rbnf(NumberFormat & fmt,const UnicodeString & str,const Formattable & n)2469 void NumberFormatTest::expect_rbnf(NumberFormat& fmt, const UnicodeString& str, const Formattable& n) {
2470     UErrorCode status = U_ZERO_ERROR;
2471     Formattable num;
2472     fmt.parse(str, num, status);
2473     if (U_FAILURE(status)) {
2474         errln(UnicodeString("FAIL: Parse failed for \"") + str + "\"");
2475         return;
2476     }
2477     if (equalValue(num, n)) {
2478         logln(UnicodeString("Ok   \"") + str + " = " +
2479               toString(num));
2480     } else {
2481         errln(UnicodeString("FAIL \"") + str + " = " +
2482               toString(num) + ", expected " + toString(n));
2483     }
2484 }
2485 
expect_rbnf(NumberFormat & fmt,const Formattable & n,const UnicodeString & exp,UBool rt)2486 void NumberFormatTest::expect_rbnf(NumberFormat& fmt, const Formattable& n,
2487                               const UnicodeString& exp, UBool rt) {
2488     UnicodeString saw;
2489     FieldPosition pos;
2490     UErrorCode status = U_ZERO_ERROR;
2491     fmt.format(n, saw, pos, status);
2492     CHECK(status, "NumberFormat::format");
2493     if (saw == exp) {
2494         logln(UnicodeString("Ok   ") + toString(n) +
2495               " = \"" +
2496               escape(saw) + "\"");
2497         // We should be able to round-trip the formatted string =>
2498         // number => string (but not the other way around: number
2499         // => string => number2, might have number2 != number):
2500         if (rt) {
2501             Formattable n2;
2502             fmt.parse(exp, n2, status);
2503             if (U_FAILURE(status)) {
2504                 errln(UnicodeString("FAIL: Parse failed for \"") + exp + "\"");
2505                 return;
2506             }
2507             UnicodeString saw2;
2508             fmt.format(n2, saw2, pos, status);
2509             CHECK(status, "NumberFormat::format");
2510             if (saw2 != exp) {
2511                 errln((UnicodeString)"FAIL \"" + exp + "\" => " + toString(n2) +
2512                       " => \"" + saw2 + "\"");
2513             }
2514         }
2515     } else {
2516         errln(UnicodeString("FAIL ") + toString(n) +
2517               " = \"" +
2518               escape(saw) + "\", expected \"" + exp + "\"");
2519     }
2520 }
2521 
expect(NumberFormat & fmt,const Formattable & n,const UnicodeString & exp,UBool rt)2522 void NumberFormatTest::expect(NumberFormat& fmt, const Formattable& n,
2523                               const UnicodeString& exp, UBool rt) {
2524     UnicodeString saw;
2525     FieldPosition pos;
2526     UErrorCode status = U_ZERO_ERROR;
2527     fmt.format(n, saw, pos, status);
2528     CHECK(status, "NumberFormat::format");
2529     UnicodeString pat;
2530     ((DecimalFormat*) &fmt)->toPattern(pat);
2531     if (saw == exp) {
2532         logln(UnicodeString("Ok   ") + toString(n) + " x " +
2533               escape(pat) + " = \"" +
2534               escape(saw) + "\"");
2535         // We should be able to round-trip the formatted string =>
2536         // number => string (but not the other way around: number
2537         // => string => number2, might have number2 != number):
2538         if (rt) {
2539             Formattable n2;
2540             fmt.parse(exp, n2, status);
2541             if (U_FAILURE(status)) {
2542                 errln(UnicodeString("FAIL: Parse failed for \"") + exp + "\" - " + u_errorName(status));
2543                 return;
2544             }
2545             UnicodeString saw2;
2546             fmt.format(n2, saw2, pos, status);
2547             CHECK(status, "NumberFormat::format");
2548             if (saw2 != exp) {
2549                 errln((UnicodeString)"FAIL \"" + exp + "\" => " + toString(n2) +
2550                       " => \"" + saw2 + "\"");
2551             }
2552         }
2553     } else {
2554         dataerrln(UnicodeString("FAIL ") + toString(n) + " x " +
2555               escape(pat) + " = \"" +
2556               escape(saw) + "\", expected \"" + exp + "\"");
2557     }
2558 }
2559 
expect(NumberFormat * fmt,const Formattable & n,const UnicodeString & exp,UBool rt,UErrorCode status)2560 void NumberFormatTest::expect(NumberFormat* fmt, const Formattable& n,
2561                               const UnicodeString& exp, UBool rt,
2562                               UErrorCode status) {
2563     if (fmt == NULL || U_FAILURE(status)) {
2564         dataerrln("FAIL: NumberFormat constructor");
2565     } else {
2566         expect(*fmt, n, exp, rt);
2567     }
2568     delete fmt;
2569 }
2570 
expectCurrency(NumberFormat & nf,const Locale & locale,double value,const UnicodeString & string)2571 void NumberFormatTest::expectCurrency(NumberFormat& nf, const Locale& locale,
2572                                       double value, const UnicodeString& string) {
2573     UErrorCode ec = U_ZERO_ERROR;
2574     DecimalFormat& fmt = * (DecimalFormat*) &nf;
2575     const UChar DEFAULT_CURR[] = {45/*-*/,0};
2576     UChar curr[4];
2577     u_strcpy(curr, DEFAULT_CURR);
2578     if (*locale.getLanguage() != 0) {
2579         ucurr_forLocale(locale.getName(), curr, 4, &ec);
2580         assertSuccess("ucurr_forLocale", ec);
2581         fmt.setCurrency(curr, ec);
2582         assertSuccess("DecimalFormat::setCurrency", ec);
2583         fmt.setCurrency(curr); //Deprecated variant, for coverage only
2584     }
2585     UnicodeString s;
2586     fmt.format(value, s);
2587     s.findAndReplace((UChar32)0x00A0, (UChar32)0x0020);
2588 
2589     // Default display of the number yields "1234.5599999999999"
2590     // instead of "1234.56".  Use a formatter to fix this.
2591     NumberFormat* f =
2592         NumberFormat::createInstance(Locale::getUS(), ec);
2593     UnicodeString v;
2594     if (U_FAILURE(ec)) {
2595         // Oops; bad formatter.  Use default op+= display.
2596         v = (UnicodeString)"" + value;
2597     } else {
2598         f->setMaximumFractionDigits(4);
2599         f->setGroupingUsed(FALSE);
2600         f->format(value, v);
2601     }
2602     delete f;
2603 
2604     if (s == string) {
2605         logln((UnicodeString)"Ok: " + v + " x " + curr + " => " + prettify(s));
2606     } else {
2607         errln((UnicodeString)"FAIL: " + v + " x " + curr + " => " + prettify(s) +
2608               ", expected " + prettify(string));
2609     }
2610 }
2611 
expectPat(DecimalFormat & fmt,const UnicodeString & exp)2612 void NumberFormatTest::expectPat(DecimalFormat& fmt, const UnicodeString& exp) {
2613     UnicodeString pat;
2614     fmt.toPattern(pat);
2615     if (pat == exp) {
2616         logln(UnicodeString("Ok   \"") + pat + "\"");
2617     } else {
2618         errln(UnicodeString("FAIL \"") + pat + "\", expected \"" + exp + "\"");
2619     }
2620 }
2621 
expectPad(DecimalFormat & fmt,const UnicodeString & pat,int32_t pos)2622 void NumberFormatTest::expectPad(DecimalFormat& fmt, const UnicodeString& pat,
2623                                  int32_t pos) {
2624     expectPad(fmt, pat, pos, 0, (UnicodeString)"");
2625 }
expectPad(DecimalFormat & fmt,const UnicodeString & pat,int32_t pos,int32_t width,UChar pad)2626 void NumberFormatTest::expectPad(DecimalFormat& fmt, const UnicodeString& pat,
2627                                  int32_t pos, int32_t width, UChar pad) {
2628     expectPad(fmt, pat, pos, width, UnicodeString(pad));
2629 }
expectPad(DecimalFormat & fmt,const UnicodeString & pat,int32_t pos,int32_t width,const UnicodeString & pad)2630 void NumberFormatTest::expectPad(DecimalFormat& fmt, const UnicodeString& pat,
2631                                  int32_t pos, int32_t width, const UnicodeString& pad) {
2632     int32_t apos = 0, awidth = 0;
2633     UnicodeString apadStr;
2634     UErrorCode status = U_ZERO_ERROR;
2635     fmt.applyPattern(pat, status);
2636     if (U_SUCCESS(status)) {
2637         apos = fmt.getPadPosition();
2638         awidth = fmt.getFormatWidth();
2639         apadStr=fmt.getPadCharacterString();
2640     } else {
2641         apos = -1;
2642         awidth = width;
2643         apadStr = pad;
2644     }
2645     if (apos == pos && awidth == width && apadStr == pad) {
2646         UnicodeString infoStr;
2647         if (pos == ILLEGAL) {
2648             infoStr = UnicodeString(" width=", "") + awidth + UnicodeString(" pad=", "") + apadStr;
2649         }
2650         logln(UnicodeString("Ok   \"") + pat + "\" pos=" + apos + infoStr);
2651     } else {
2652         errln(UnicodeString("FAIL \"") + pat + "\" pos=" + apos +
2653               " width=" + awidth + " pad=" + apadStr +
2654               ", expected " + pos + " " + width + " " + pad);
2655     }
2656 }
2657 
2658 // This test is flaky b/c the symbols for CNY and JPY are equivalent in this locale  - FIXME
TestCompatibleCurrencies()2659 void NumberFormatTest::TestCompatibleCurrencies() {
2660 /*
2661     static const UChar JPY[] = {0x4A, 0x50, 0x59, 0};
2662     static const UChar CNY[] = {0x43, 0x4E, 0x59, 0};
2663     UErrorCode status = U_ZERO_ERROR;
2664     LocalPointer<NumberFormat> fmt(
2665         NumberFormat::createCurrencyInstance(Locale::getUS(), status));
2666     if (U_FAILURE(status)) {
2667         errln("Could not create number format instance.");
2668         return;
2669     }
2670     logln("%s:%d - testing parse of halfwidth yen sign\n", __FILE__, __LINE__);
2671     expectParseCurrency(*fmt, JPY, 1235,  "\\u00A51,235");
2672     logln("%s:%d - testing parse of fullwidth yen sign\n", __FILE__, __LINE__);
2673     expectParseCurrency(*fmt, JPY, 1235,  "\\uFFE51,235");
2674     logln("%s:%d - testing parse of halfwidth yen sign\n", __FILE__, __LINE__);
2675     expectParseCurrency(*fmt, CNY, 1235,  "CN\\u00A51,235");
2676 
2677     LocalPointer<NumberFormat> fmtTW(
2678         NumberFormat::createCurrencyInstance(Locale::getTaiwan(), status));
2679 
2680     logln("%s:%d - testing parse of halfwidth yen sign in TW\n", __FILE__, __LINE__);
2681     expectParseCurrency(*fmtTW, CNY, 1235,  "\\u00A51,235");
2682     logln("%s:%d - testing parse of fullwidth yen sign in TW\n", __FILE__, __LINE__);
2683     expectParseCurrency(*fmtTW, CNY, 1235,  "\\uFFE51,235");
2684 
2685     LocalPointer<NumberFormat> fmtJP(
2686         NumberFormat::createCurrencyInstance(Locale::getJapan(), status));
2687 
2688     logln("%s:%d - testing parse of halfwidth yen sign in JP\n", __FILE__, __LINE__);
2689     expectParseCurrency(*fmtJP, JPY, 1235,  "\\u00A51,235");
2690     logln("%s:%d - testing parse of fullwidth yen sign in JP\n", __FILE__, __LINE__);
2691     expectParseCurrency(*fmtJP, JPY, 1235,  "\\uFFE51,235");
2692 
2693     // more..
2694 */
2695 }
2696 
expectParseCurrency(const NumberFormat & fmt,const UChar * currency,double amount,const char * text)2697 void NumberFormatTest::expectParseCurrency(const NumberFormat &fmt, const UChar* currency, double amount, const char *text) {
2698     ParsePosition ppos;
2699     UnicodeString utext = ctou(text);
2700     LocalPointer<CurrencyAmount> currencyAmount(fmt.parseCurrency(utext, ppos));
2701     if (!ppos.getIndex()) {
2702         errln(UnicodeString("Parse of ") + utext + " should have succeeded.");
2703         return;
2704     }
2705     UErrorCode status = U_ZERO_ERROR;
2706 
2707     char theInfo[100];
2708     sprintf(theInfo, "For locale %s, string \"%s\", currency ",
2709             fmt.getLocale(ULOC_ACTUAL_LOCALE, status).getBaseName(),
2710             text);
2711     u_austrcpy(theInfo+uprv_strlen(theInfo), currency);
2712 
2713     char theOperation[100];
2714 
2715     uprv_strcpy(theOperation, theInfo);
2716     uprv_strcat(theOperation, ", check amount:");
2717     assertTrue(theOperation, amount ==  currencyAmount->getNumber().getDouble(status));
2718 
2719     uprv_strcpy(theOperation, theInfo);
2720     uprv_strcat(theOperation, ", check currency:");
2721     assertEquals(theOperation, currency, currencyAmount->getISOCurrency());
2722 }
2723 
2724 
TestJB3832()2725 void NumberFormatTest::TestJB3832(){
2726     const char* localeID = "pt_PT@currency=PTE";
2727     Locale loc(localeID);
2728     UErrorCode status = U_ZERO_ERROR;
2729     UnicodeString expected(CharsToUnicodeString("1,150$50\\u00A0\\u200B")); // per cldrbug 7670
2730     UnicodeString s;
2731     NumberFormat* currencyFmt = NumberFormat::createCurrencyInstance(loc, status);
2732     if(U_FAILURE(status)){
2733         dataerrln("Could not create currency formatter for locale %s - %s", localeID, u_errorName(status));
2734         return;
2735     }
2736     currencyFmt->format(1150.50, s);
2737     if(s!=expected){
2738         errln(UnicodeString("FAIL: Expected: ")+expected
2739                 + UnicodeString(" Got: ") + s
2740                 + UnicodeString( " for locale: ")+ UnicodeString(localeID) );
2741     }
2742     if (U_FAILURE(status)){
2743         errln("FAIL: Status %s", u_errorName(status));
2744     }
2745     delete currencyFmt;
2746 }
2747 
TestHost()2748 void NumberFormatTest::TestHost()
2749 {
2750 #if U_PLATFORM_USES_ONLY_WIN32_API
2751     Win32NumberTest::testLocales(this);
2752 #endif
2753     Locale loc("en_US@compat=host");
2754     for (UNumberFormatStyle k = UNUM_DECIMAL;
2755          k < UNUM_FORMAT_STYLE_COUNT; k = (UNumberFormatStyle)(k+1)) {
2756         UErrorCode status = U_ZERO_ERROR;
2757         LocalPointer<NumberFormat> full(NumberFormat::createInstance(loc, k, status));
2758         if (!NumberFormat::isStyleSupported(k)) {
2759             if (status != U_UNSUPPORTED_ERROR) {
2760                 errln("FAIL: expected style %d to be unsupported - %s",
2761                       k, u_errorName(status));
2762             }
2763             continue;
2764         }
2765         if (full.isNull() || U_FAILURE(status)) {
2766             dataerrln("FAIL: Can't create number instance of style %d for host - %s",
2767                       k, u_errorName(status));
2768             return;
2769         }
2770         UnicodeString result1;
2771         Formattable number(10.00);
2772         full->format(number, result1, status);
2773         if (U_FAILURE(status)) {
2774             errln("FAIL: Can't format for host");
2775             return;
2776         }
2777         Formattable formattable;
2778         full->parse(result1, formattable, status);
2779         if (U_FAILURE(status)) {
2780             errln("FAIL: Can't parse for host");
2781             return;
2782         }
2783     }
2784 }
2785 
TestHostClone()2786 void NumberFormatTest::TestHostClone()
2787 {
2788     /*
2789     Verify that a cloned formatter gives the same results
2790     and is useable after the original has been deleted.
2791     */
2792     // This is mainly important on Windows.
2793     UErrorCode status = U_ZERO_ERROR;
2794     Locale loc("en_US@compat=host");
2795     UDate now = Calendar::getNow();
2796     NumberFormat *full = NumberFormat::createInstance(loc, status);
2797     if (full == NULL || U_FAILURE(status)) {
2798         dataerrln("FAIL: Can't create Relative date instance - %s", u_errorName(status));
2799         return;
2800     }
2801     UnicodeString result1;
2802     full->format(now, result1, status);
2803     Format *fullClone = full->clone();
2804     delete full;
2805     full = NULL;
2806 
2807     UnicodeString result2;
2808     fullClone->format(now, result2, status);
2809     if (U_FAILURE(status)) {
2810         errln("FAIL: format failure.");
2811     }
2812     if (result1 != result2) {
2813         errln("FAIL: Clone returned different result from non-clone.");
2814     }
2815     delete fullClone;
2816 }
2817 
TestCurrencyFormat()2818 void NumberFormatTest::TestCurrencyFormat()
2819 {
2820     // This test is here to increase code coverage.
2821     UErrorCode status = U_ZERO_ERROR;
2822     MeasureFormat *cloneObj;
2823     UnicodeString str;
2824     Formattable toFormat, result;
2825     static const UChar ISO_CODE[4] = {0x0047, 0x0042, 0x0050, 0};
2826 
2827     Locale  saveDefaultLocale = Locale::getDefault();
2828     Locale::setDefault( Locale::getUK(), status );
2829     if (U_FAILURE(status)) {
2830         errln("couldn't set default Locale!");
2831         return;
2832     }
2833 
2834     MeasureFormat *measureObj = MeasureFormat::createCurrencyFormat(status);
2835     Locale::setDefault( saveDefaultLocale, status );
2836     if (U_FAILURE(status)){
2837         dataerrln("FAIL: Status %s", u_errorName(status));
2838         return;
2839     }
2840     cloneObj = (MeasureFormat *)measureObj->clone();
2841     if (cloneObj == NULL) {
2842         errln("Clone doesn't work");
2843         return;
2844     }
2845     toFormat.adoptObject(new CurrencyAmount(1234.56, ISO_CODE, status));
2846     measureObj->format(toFormat, str, status);
2847     measureObj->parseObject(str, result, status);
2848     if (U_FAILURE(status)){
2849         errln("FAIL: Status %s", u_errorName(status));
2850     }
2851     if (result != toFormat) {
2852         errln("measureObj does not round trip. Formatted string was \"" + str + "\" Got: " + toString(result) + " Expected: " + toString(toFormat));
2853     }
2854     status = U_ZERO_ERROR;
2855     str.truncate(0);
2856     cloneObj->format(toFormat, str, status);
2857     cloneObj->parseObject(str, result, status);
2858     if (U_FAILURE(status)){
2859         errln("FAIL: Status %s", u_errorName(status));
2860     }
2861     if (result != toFormat) {
2862         errln("Clone does not round trip. Formatted string was \"" + str + "\" Got: " + toString(result) + " Expected: " + toString(toFormat));
2863     }
2864     if (*measureObj != *cloneObj) {
2865         errln("Cloned object is not equal to the original object");
2866     }
2867     delete measureObj;
2868     delete cloneObj;
2869 
2870     status = U_USELESS_COLLATOR_ERROR;
2871     if (MeasureFormat::createCurrencyFormat(status) != NULL) {
2872         errln("createCurrencyFormat should have returned NULL.");
2873     }
2874 }
2875 
2876 /* Port of ICU4J rounding test. */
TestRounding()2877 void NumberFormatTest::TestRounding() {
2878     UErrorCode status = U_ZERO_ERROR;
2879     DecimalFormat *df = (DecimalFormat*)NumberFormat::createCurrencyInstance(Locale::getEnglish(), status);
2880 
2881     if (U_FAILURE(status)) {
2882         dataerrln("Unable to create decimal formatter. - %s", u_errorName(status));
2883         return;
2884     }
2885 
2886     int roundingIncrements[]={1, 2, 5, 20, 50, 100};
2887     int testValues[]={0, 300};
2888 
2889     for (int j=0; j<2; j++) {
2890         for (int mode=DecimalFormat::kRoundUp;mode<DecimalFormat::kRoundHalfEven;mode++) {
2891             df->setRoundingMode((DecimalFormat::ERoundingMode)mode);
2892             for (int increment=0; increment<6; increment++) {
2893                 double base=testValues[j];
2894                 double rInc=roundingIncrements[increment];
2895                 checkRounding(df, base, 20, rInc);
2896                 rInc=1.000000000/rInc;
2897                 checkRounding(df, base, 20, rInc);
2898             }
2899         }
2900     }
2901     delete df;
2902 }
2903 
TestRoundingPattern()2904 void NumberFormatTest::TestRoundingPattern() {
2905     UErrorCode status = U_ZERO_ERROR;
2906     struct {
2907         UnicodeString  pattern;
2908         double        testCase;
2909         UnicodeString expected;
2910     } tests[] = {
2911             { (UnicodeString)"##0.65", 1.234, (UnicodeString)"1.30" },
2912             { (UnicodeString)"#50",    1230,  (UnicodeString)"1250" }
2913     };
2914     int32_t numOfTests = (sizeof(tests)/sizeof(tests[0]));
2915     UnicodeString result;
2916 
2917     DecimalFormat *df = (DecimalFormat*)NumberFormat::createCurrencyInstance(Locale::getEnglish(), status);
2918     if (U_FAILURE(status)) {
2919         dataerrln("Unable to create decimal formatter. - %s", u_errorName(status));
2920         return;
2921     }
2922 
2923     for (int32_t i = 0; i < numOfTests; i++) {
2924         result.remove();
2925 
2926         df->applyPattern(tests[i].pattern, status);
2927         if (U_FAILURE(status)) {
2928             errln("Unable to apply pattern to decimal formatter. - %s", u_errorName(status));
2929         }
2930 
2931         df->format(tests[i].testCase, result);
2932 
2933         if (result != tests[i].expected) {
2934             errln("String Pattern Rounding Test Failed: Pattern: \"" + tests[i].pattern + "\" Number: " + tests[i].testCase + " - Got: " + result + " Expected: " + tests[i].expected);
2935         }
2936     }
2937 
2938     delete df;
2939 }
2940 
checkRounding(DecimalFormat * df,double base,int iterations,double increment)2941 void NumberFormatTest::checkRounding(DecimalFormat* df, double base, int iterations, double increment) {
2942     df->setRoundingIncrement(increment);
2943     double lastParsed=INT32_MIN; //Intger.MIN_VALUE
2944     for (int i=-iterations; i<=iterations;i++) {
2945         double iValue=base+(increment*(i*0.1));
2946         double smallIncrement=0.00000001;
2947         if (iValue!=0) {
2948             smallIncrement*=iValue;
2949         }
2950         //we not only test the value, but some values in a small range around it
2951         lastParsed=checkRound(df, iValue-smallIncrement, lastParsed);
2952         lastParsed=checkRound(df, iValue, lastParsed);
2953         lastParsed=checkRound(df, iValue+smallIncrement, lastParsed);
2954     }
2955 }
2956 
checkRound(DecimalFormat * df,double iValue,double lastParsed)2957 double NumberFormatTest::checkRound(DecimalFormat* df, double iValue, double lastParsed) {
2958     UErrorCode status=U_ZERO_ERROR;
2959     UnicodeString formattedDecimal;
2960     double parsed;
2961     Formattable result;
2962     df->format(iValue, formattedDecimal, status);
2963 
2964     if (U_FAILURE(status)) {
2965         errln("Error formatting number.");
2966     }
2967 
2968     df->parse(formattedDecimal, result, status);
2969 
2970     if (U_FAILURE(status)) {
2971         errln("Error parsing number.");
2972     }
2973 
2974     parsed=result.getDouble();
2975 
2976     if (lastParsed>parsed) {
2977         errln("Rounding wrong direction! %d > %d", lastParsed, parsed);
2978     }
2979 
2980     return lastParsed;
2981 }
2982 
TestNonpositiveMultiplier()2983 void NumberFormatTest::TestNonpositiveMultiplier() {
2984     UErrorCode status = U_ZERO_ERROR;
2985     DecimalFormatSymbols US(Locale::getUS(), status);
2986     CHECK(status, "DecimalFormatSymbols constructor");
2987     DecimalFormat df(UnicodeString("0"), US, status);
2988     CHECK(status, "DecimalFormat(0)");
2989 
2990     // test zero multiplier
2991 
2992     int32_t mult = df.getMultiplier();
2993     df.setMultiplier(0);
2994     if (df.getMultiplier() != mult) {
2995         errln("DecimalFormat.setMultiplier(0) did not ignore its zero input");
2996     }
2997 
2998     // test negative multiplier
2999 
3000     df.setMultiplier(-1);
3001     if (df.getMultiplier() != -1) {
3002         errln("DecimalFormat.setMultiplier(-1) ignored its negative input");
3003         return;
3004     }
3005 
3006     expect(df, "1122.123", -1122.123);
3007     expect(df, "-1122.123", 1122.123);
3008     expect(df, "1.2", -1.2);
3009     expect(df, "-1.2", 1.2);
3010 
3011     // Note:  the tests with the final parameter of FALSE will not round trip.
3012     //        The initial numeric value will format correctly, after the multiplier.
3013     //        Parsing the formatted text will be out-of-range for an int64, however.
3014     //        The expect() function could be modified to detect this and fall back
3015     //        to looking at the decimal parsed value, but it doesn't.
3016     expect(df, U_INT64_MIN,    "9223372036854775808", FALSE);
3017     expect(df, U_INT64_MIN+1,  "9223372036854775807");
3018     expect(df, (int64_t)-123,                  "123");
3019     expect(df, (int64_t)123,                  "-123");
3020     expect(df, U_INT64_MAX-1, "-9223372036854775806");
3021     expect(df, U_INT64_MAX,   "-9223372036854775807");
3022 
3023     df.setMultiplier(-2);
3024     expect(df, -(U_INT64_MIN/2)-1, "-9223372036854775806");
3025     expect(df, -(U_INT64_MIN/2),   "-9223372036854775808");
3026     expect(df, -(U_INT64_MIN/2)+1, "-9223372036854775810", FALSE);
3027 
3028     df.setMultiplier(-7);
3029     expect(df, -(U_INT64_MAX/7)-1, "9223372036854775814", FALSE);
3030     expect(df, -(U_INT64_MAX/7),   "9223372036854775807");
3031     expect(df, -(U_INT64_MAX/7)+1, "9223372036854775800");
3032 
3033     // TODO: uncomment (and fix up) all the following int64_t tests once BigInteger is ported
3034     // (right now the big numbers get turned into doubles and lose tons of accuracy)
3035     //expect2(df, U_INT64_MAX, Int64ToUnicodeString(-U_INT64_MAX));
3036     //expect2(df, U_INT64_MIN, UnicodeString(Int64ToUnicodeString(U_INT64_MIN), 1));
3037     //expect2(df, U_INT64_MAX / 2, Int64ToUnicodeString(-(U_INT64_MAX / 2)));
3038     //expect2(df, U_INT64_MIN / 2, Int64ToUnicodeString(-(U_INT64_MIN / 2)));
3039 
3040     // TODO: uncomment (and fix up) once BigDecimal is ported and DecimalFormat can handle it
3041     //expect2(df, BigDecimal.valueOf(Long.MAX_VALUE), BigDecimal.valueOf(Long.MAX_VALUE).negate().toString());
3042     //expect2(df, BigDecimal.valueOf(Long.MIN_VALUE), BigDecimal.valueOf(Long.MIN_VALUE).negate().toString());
3043     //expect2(df, java.math.BigDecimal.valueOf(Long.MAX_VALUE), java.math.BigDecimal.valueOf(Long.MAX_VALUE).negate().toString());
3044     //expect2(df, java.math.BigDecimal.valueOf(Long.MIN_VALUE), java.math.BigDecimal.valueOf(Long.MIN_VALUE).negate().toString());
3045 }
3046 
3047 typedef struct {
3048     const char * stringToParse;
3049     int          parsedPos;
3050     int          errorIndex;
3051     UBool        lenient;
3052 } TestSpaceParsingItem;
3053 
3054 void
TestSpaceParsing()3055 NumberFormatTest::TestSpaceParsing() {
3056     // the data are:
3057     // the string to be parsed, parsed position, parsed error index
3058     const TestSpaceParsingItem DATA[] = {
3059         // TOTO: Update the following TODOs, some may be handled now
3060         {"$124",           4, -1, FALSE},
3061         {"$124 $124",      4, -1, FALSE},
3062         {"$124 ",          4, -1, FALSE},
3063         //{"$ 124 ",       5, -1, FALSE}, // TODO: need to handle space correctly
3064         //{"$\\u00A0124 ", 5, -1, FALSE}, // TODO: need to handle space correctly
3065         {"$ 124 ",         0,  1, FALSE}, // errorIndex used to be 0, now 1 (better)
3066         {"$\\u00A0124 ",   0,  1, FALSE}, // errorIndex used to be 0, now 1 (better)
3067         {" $ 124 ",        0,  0, FALSE}, // TODO: need to handle space correctly
3068         {"124$",           0,  3, FALSE}, // TODO: need to handle space correctly
3069         // {"124 $",       5, -1, FALSE}, // TODO: OK or not, need currency spacing rule
3070         {"124 $",          0,  3, FALSE},
3071         {"$124",           4, -1, TRUE},
3072         {"$124 $124",      4, -1, TRUE},
3073         {"$124 ",          4, -1, TRUE},
3074         {"$ 124 ",         5, -1, TRUE},
3075         {"$\\u00A0124 ",   5, -1, TRUE},
3076         {" $ 124 ",        6, -1, TRUE},
3077         //{"124$",         4, -1, TRUE}, // TODO: need to handle trailing currency correctly
3078         {"124$",           3, -1, TRUE},
3079         //{"124 $",        5, -1, TRUE}, // TODO: OK or not, need currency spacing rule
3080         {"124 $",          4, -1, TRUE},
3081     };
3082     UErrorCode status = U_ZERO_ERROR;
3083     Locale locale("en_US");
3084     NumberFormat* foo = NumberFormat::createCurrencyInstance(locale, status);
3085 
3086     if (U_FAILURE(status)) {
3087         delete foo;
3088         return;
3089     }
3090     for (uint32_t i = 0; i < sizeof(DATA)/sizeof(DATA[0]); ++i) {
3091         ParsePosition parsePosition(0);
3092         UnicodeString stringToBeParsed = ctou(DATA[i].stringToParse);
3093         int parsedPosition = DATA[i].parsedPos;
3094         int errorIndex = DATA[i].errorIndex;
3095         foo->setLenient(DATA[i].lenient);
3096         Formattable result;
3097         foo->parse(stringToBeParsed, result, parsePosition);
3098         if (parsePosition.getIndex() != parsedPosition ||
3099             parsePosition.getErrorIndex() != errorIndex) {
3100             errln("FAILED parse " + stringToBeParsed + "; lenient: " + DATA[i].lenient + "; wrong position, expected: (" + parsedPosition + ", " + errorIndex + "); got (" + parsePosition.getIndex() + ", " + parsePosition.getErrorIndex() + ")");
3101         }
3102         if (parsePosition.getErrorIndex() == -1 &&
3103             result.getType() == Formattable::kLong &&
3104             result.getLong() != 124) {
3105             errln("FAILED parse " + stringToBeParsed + "; wrong number, expect: 124, got " + result.getLong());
3106         }
3107     }
3108     delete foo;
3109 }
3110 
3111 /**
3112  * Test using various numbering systems and numbering system keyword.
3113  */
3114 typedef struct {
3115     const char *localeName;
3116     double      value;
3117     UBool        isRBNF;
3118     const char *expectedResult;
3119 } TestNumberingSystemItem;
3120 
TestNumberingSystems()3121 void NumberFormatTest::TestNumberingSystems() {
3122 
3123     const TestNumberingSystemItem DATA[] = {
3124         { "en_US@numbers=thai", 1234.567, FALSE, "\\u0E51,\\u0E52\\u0E53\\u0E54.\\u0E55\\u0E56\\u0E57" },
3125         { "en_US@numbers=hebr", 5678.0, TRUE, "\\u05D4\\u05F3\\u05EA\\u05E8\\u05E2\\u05F4\\u05D7" },
3126         { "en_US@numbers=arabext", 1234.567, FALSE, "\\u06F1\\u066c\\u06F2\\u06F3\\u06F4\\u066b\\u06F5\\u06F6\\u06F7" },
3127         { "ar_EG", 1234.567, FALSE, "\\u0661\\u066C\\u0662\\u0663\\u0664\\u066b\\u0665\\u0666\\u0667" },
3128         { "th_TH@numbers=traditional", 1234.567, FALSE, "\\u0E51,\\u0E52\\u0E53\\u0E54.\\u0E55\\u0E56\\u0E57" }, // fall back to native per TR35
3129         { "ar_MA", 1234.567, FALSE, "1.234,567" },
3130         { "en_US@numbers=hanidec", 1234.567, FALSE, "\\u4e00,\\u4e8c\\u4e09\\u56db.\\u4e94\\u516d\\u4e03" },
3131         { "ta_IN@numbers=native", 1234.567, FALSE, "\\u0BE7,\\u0BE8\\u0BE9\\u0BEA.\\u0BEB\\u0BEC\\u0BED" },
3132         { "ta_IN@numbers=traditional", 1235.0, TRUE, "\\u0BF2\\u0BE8\\u0BF1\\u0BE9\\u0BF0\\u0BEB" },
3133         { "ta_IN@numbers=finance", 1234.567, FALSE, "1,234.567" }, // fall back to default per TR35
3134         { "zh_TW@numbers=native", 1234.567, FALSE, "\\u4e00,\\u4e8c\\u4e09\\u56db.\\u4e94\\u516d\\u4e03" },
3135         { "zh_TW@numbers=traditional", 1234.567, TRUE, "\\u4E00\\u5343\\u4E8C\\u767E\\u4E09\\u5341\\u56DB\\u9EDE\\u4E94\\u516D\\u4E03" },
3136         { "zh_TW@numbers=finance", 1234.567, TRUE, "\\u58F9\\u4EDF\\u8CB3\\u4F70\\u53C3\\u62FE\\u8086\\u9EDE\\u4F0D\\u9678\\u67D2" },
3137         { NULL, 0, FALSE, NULL }
3138     };
3139 
3140     UErrorCode ec;
3141 
3142     const TestNumberingSystemItem *item;
3143     for (item = DATA; item->localeName != NULL; item++) {
3144         ec = U_ZERO_ERROR;
3145         Locale loc = Locale::createFromName(item->localeName);
3146 
3147         NumberFormat *origFmt = NumberFormat::createInstance(loc,ec);
3148         if (U_FAILURE(ec)) {
3149             dataerrln("FAIL: getInstance(%s) - %s", item->localeName, u_errorName(ec));
3150             continue;
3151         }
3152         // Clone to test ticket #10682
3153         NumberFormat *fmt = (NumberFormat *) origFmt->clone();
3154         delete origFmt;
3155 
3156 
3157         if (item->isRBNF) {
3158             expect3(*fmt,item->value,CharsToUnicodeString(item->expectedResult));
3159         } else {
3160             expect2(*fmt,item->value,CharsToUnicodeString(item->expectedResult));
3161         }
3162         delete fmt;
3163     }
3164 
3165 
3166     // Test bogus keyword value
3167     ec = U_ZERO_ERROR;
3168     Locale loc4 = Locale::createFromName("en_US@numbers=foobar");
3169     NumberFormat* fmt4= NumberFormat::createInstance(loc4, ec);
3170     if ( ec != U_UNSUPPORTED_ERROR ) {
3171         errln("FAIL: getInstance(en_US@numbers=foobar) should have returned U_UNSUPPORTED_ERROR");
3172         delete fmt4;
3173     }
3174 
3175     ec = U_ZERO_ERROR;
3176     NumberingSystem *ns = NumberingSystem::createInstance(ec);
3177     if (U_FAILURE(ec)) {
3178         dataerrln("FAIL: NumberingSystem::createInstance(ec); - %s", u_errorName(ec));
3179     }
3180 
3181     if ( ns != NULL ) {
3182         ns->getDynamicClassID();
3183         ns->getStaticClassID();
3184     } else {
3185         errln("FAIL: getInstance() returned NULL.");
3186     }
3187 
3188     NumberingSystem *ns1 = new NumberingSystem(*ns);
3189     if (ns1 == NULL) {
3190         errln("FAIL: NumberSystem copy constructor returned NULL.");
3191     }
3192 
3193     delete ns1;
3194     delete ns;
3195 
3196 }
3197 
3198 
3199 void
TestMultiCurrencySign()3200 NumberFormatTest::TestMultiCurrencySign() {
3201     const char* DATA[][6] = {
3202         // the fields in the following test are:
3203         // locale,
3204         // currency pattern (with negative pattern),
3205         // currency number to be formatted,
3206         // currency format using currency symbol name, such as "$" for USD,
3207         // currency format using currency ISO name, such as "USD",
3208         // currency format using plural name, such as "US dollars".
3209         // for US locale
3210         {"en_US", "\\u00A4#,##0.00;-\\u00A4#,##0.00", "1234.56", "$1,234.56", "USD1,234.56", "US dollars1,234.56"},
3211         {"en_US", "\\u00A4#,##0.00;-\\u00A4#,##0.00", "-1234.56", "-$1,234.56", "-USD1,234.56", "-US dollars1,234.56"},
3212         {"en_US", "\\u00A4#,##0.00;-\\u00A4#,##0.00", "1", "$1.00", "USD1.00", "US dollars1.00"},
3213         // for CHINA locale
3214         {"zh_CN", "\\u00A4#,##0.00;(\\u00A4#,##0.00)", "1234.56", "\\uFFE51,234.56", "CNY1,234.56", "\\u4EBA\\u6C11\\u5E011,234.56"},
3215         {"zh_CN", "\\u00A4#,##0.00;(\\u00A4#,##0.00)", "-1234.56", "(\\uFFE51,234.56)", "(CNY1,234.56)", "(\\u4EBA\\u6C11\\u5E011,234.56)"},
3216         {"zh_CN", "\\u00A4#,##0.00;(\\u00A4#,##0.00)", "1", "\\uFFE51.00", "CNY1.00", "\\u4EBA\\u6C11\\u5E011.00"}
3217     };
3218 
3219     const UChar doubleCurrencySign[] = {0xA4, 0xA4, 0};
3220     UnicodeString doubleCurrencyStr(doubleCurrencySign);
3221     const UChar tripleCurrencySign[] = {0xA4, 0xA4, 0xA4, 0};
3222     UnicodeString tripleCurrencyStr(tripleCurrencySign);
3223 
3224     for (uint32_t i=0; i<sizeof(DATA)/sizeof(DATA[0]); ++i) {
3225         const char* locale = DATA[i][0];
3226         UnicodeString pat = ctou(DATA[i][1]);
3227         double numberToBeFormat = atof(DATA[i][2]);
3228         UErrorCode status = U_ZERO_ERROR;
3229         DecimalFormatSymbols* sym = new DecimalFormatSymbols(Locale(locale), status);
3230         if (U_FAILURE(status)) {
3231             delete sym;
3232             continue;
3233         }
3234         for (int j=1; j<=3; ++j) {
3235             // j represents the number of currency sign in the pattern.
3236             if (j == 2) {
3237                 pat = pat.findAndReplace(ctou("\\u00A4"), doubleCurrencyStr);
3238             } else if (j == 3) {
3239                 pat = pat.findAndReplace(ctou("\\u00A4\\u00A4"), tripleCurrencyStr);
3240             }
3241 
3242             DecimalFormat* fmt = new DecimalFormat(pat, new DecimalFormatSymbols(*sym), status);
3243             if (U_FAILURE(status)) {
3244                 errln("FAILED init DecimalFormat ");
3245                 delete fmt;
3246                 continue;
3247             }
3248             UnicodeString s;
3249             ((NumberFormat*) fmt)->format(numberToBeFormat, s);
3250             // DATA[i][3] is the currency format result using a
3251             // single currency sign.
3252             // DATA[i][4] is the currency format result using
3253             // double currency sign.
3254             // DATA[i][5] is the currency format result using
3255             // triple currency sign.
3256             // DATA[i][j+2] is the currency format result using
3257             // 'j' number of currency sign.
3258             UnicodeString currencyFormatResult = ctou(DATA[i][2+j]);
3259             if (s.compare(currencyFormatResult)) {
3260                 errln("FAIL format: Expected " + currencyFormatResult + "; Got " + s);
3261             }
3262             // mix style parsing
3263             for (int k=3; k<=5; ++k) {
3264               // DATA[i][3] is the currency format result using a
3265               // single currency sign.
3266               // DATA[i][4] is the currency format result using
3267               // double currency sign.
3268               // DATA[i][5] is the currency format result using
3269               // triple currency sign.
3270               UnicodeString oneCurrencyFormat = ctou(DATA[i][k]);
3271               UErrorCode status = U_ZERO_ERROR;
3272               Formattable parseRes;
3273               fmt->parse(oneCurrencyFormat, parseRes, status);
3274               if (U_FAILURE(status) ||
3275                   (parseRes.getType() == Formattable::kDouble &&
3276                    parseRes.getDouble() != numberToBeFormat) ||
3277                   (parseRes.getType() == Formattable::kLong &&
3278                    parseRes.getLong() != numberToBeFormat)) {
3279                   errln("FAILED parse " + oneCurrencyFormat + "; (i, j, k): " +
3280                         i + ", " + j + ", " + k);
3281               }
3282             }
3283             delete fmt;
3284         }
3285         delete sym;
3286     }
3287 }
3288 
3289 
3290 void
TestCurrencyFormatForMixParsing()3291 NumberFormatTest::TestCurrencyFormatForMixParsing() {
3292     UErrorCode status = U_ZERO_ERROR;
3293     MeasureFormat* curFmt = MeasureFormat::createCurrencyFormat(Locale("en_US"), status);
3294     if (U_FAILURE(status)) {
3295         delete curFmt;
3296         return;
3297     }
3298     const char* formats[] = {
3299         "$1,234.56",  // string to be parsed
3300         "USD1,234.56",
3301         "US dollars1,234.56",
3302         "1,234.56 US dollars"
3303     };
3304     const CurrencyAmount* curramt = NULL;
3305     for (uint32_t i = 0; i < sizeof(formats)/sizeof(formats[0]); ++i) {
3306         UnicodeString stringToBeParsed = ctou(formats[i]);
3307         logln(UnicodeString("stringToBeParsed: ") + stringToBeParsed);
3308         Formattable result;
3309         UErrorCode status = U_ZERO_ERROR;
3310         curFmt->parseObject(stringToBeParsed, result, status);
3311         if (U_FAILURE(status)) {
3312           errln("FAIL: measure format parsing: '%s' ec: %s", formats[i], u_errorName(status));
3313         } else if (result.getType() != Formattable::kObject ||
3314             (curramt = dynamic_cast<const CurrencyAmount*>(result.getObject())) == NULL ||
3315             curramt->getNumber().getDouble() != 1234.56 ||
3316             UnicodeString(curramt->getISOCurrency()).compare(ISO_CURRENCY_USD)
3317         ) {
3318             errln("FAIL: getCurrencyFormat of default locale (en_US) failed roundtripping the number ");
3319             if (curramt->getNumber().getDouble() != 1234.56) {
3320                 errln((UnicodeString)"wong number, expect: 1234.56" + ", got: " + curramt->getNumber().getDouble());
3321             }
3322             if (curramt->getISOCurrency() != ISO_CURRENCY_USD) {
3323                 errln((UnicodeString)"wong currency, expect: USD" + ", got: " + curramt->getISOCurrency());
3324             }
3325         }
3326     }
3327     delete curFmt;
3328 }
3329 
3330 
3331 void
TestDecimalFormatCurrencyParse()3332 NumberFormatTest::TestDecimalFormatCurrencyParse() {
3333     // Locale.US
3334     UErrorCode status = U_ZERO_ERROR;
3335     DecimalFormatSymbols* sym = new DecimalFormatSymbols(Locale("en_US"), status);
3336     if (U_FAILURE(status)) {
3337         delete sym;
3338         return;
3339     }
3340     UnicodeString pat;
3341     UChar currency = 0x00A4;
3342     // "\xA4#,##0.00;-\xA4#,##0.00"
3343     pat.append(currency).append(currency).append(currency).append("#,##0.00;-").append(currency).append(currency).append(currency).append("#,##0.00");
3344     DecimalFormat* fmt = new DecimalFormat(pat, sym, status);
3345     if (U_FAILURE(status)) {
3346         delete fmt;
3347         errln("failed to new DecimalFormat in TestDecimalFormatCurrencyParse");
3348         return;
3349     }
3350     const char* DATA[][2] = {
3351         // the data are:
3352         // string to be parsed, the parsed result (number)
3353         {"$1.00", "1"},
3354         {"USD1.00", "1"},
3355         {"1.00 US dollar", "1"},
3356         {"$1,234.56", "1234.56"},
3357         {"USD1,234.56", "1234.56"},
3358         {"1,234.56 US dollar", "1234.56"},
3359     };
3360     for (uint32_t i = 0; i < sizeof(DATA)/sizeof(DATA[0]); ++i) {
3361         UnicodeString stringToBeParsed = ctou(DATA[i][0]);
3362         double parsedResult = atof(DATA[i][1]);
3363         UErrorCode status = U_ZERO_ERROR;
3364         Formattable result;
3365         fmt->parse(stringToBeParsed, result, status);
3366         if (U_FAILURE(status) ||
3367             (result.getType() == Formattable::kDouble &&
3368             result.getDouble() != parsedResult) ||
3369             (result.getType() == Formattable::kLong &&
3370             result.getLong() != parsedResult)) {
3371             errln((UnicodeString)"FAIL parse: Expected " + parsedResult);
3372         }
3373     }
3374     delete fmt;
3375 }
3376 
3377 
3378 void
TestCurrencyIsoPluralFormat()3379 NumberFormatTest::TestCurrencyIsoPluralFormat() {
3380     static const char* DATA[][6] = {
3381         // the data are:
3382         // locale,
3383         // currency amount to be formatted,
3384         // currency ISO code to be formatted,
3385         // format result using CURRENCYSTYLE,
3386         // format result using ISOCURRENCYSTYLE,
3387         // format result using PLURALCURRENCYSTYLE,
3388 
3389         {"en_US", "1", "USD", "$1.00", "USD1.00", "1.00 US dollars"},
3390         {"en_US", "1234.56", "USD", "$1,234.56", "USD1,234.56", "1,234.56 US dollars"},
3391         {"en_US", "-1234.56", "USD", "-$1,234.56", "-USD1,234.56", "-1,234.56 US dollars"},
3392         {"zh_CN", "1", "USD", "US$\\u00A01.00", "USD\\u00A01.00", "1.00\\u7F8E\\u5143"},
3393         {"zh_CN", "1234.56", "USD", "US$\\u00A01,234.56", "USD\\u00A01,234.56", "1,234.56\\u7F8E\\u5143"},
3394         // wrong ISO code {"zh_CN", "1", "CHY", "CHY1.00", "CHY1.00", "1.00 CHY"},
3395         // wrong ISO code {"zh_CN", "1234.56", "CHY", "CHY1,234.56", "CHY1,234.56", "1,234.56 CHY"},
3396         {"zh_CN", "1", "CNY", "\\uFFE5\\u00A01.00", "CNY\\u00A01.00", "1.00\\u4EBA\\u6C11\\u5E01"},
3397         {"zh_CN", "1234.56", "CNY", "\\uFFE5\\u00A01,234.56", "CNY\\u00A01,234.56", "1,234.56\\u4EBA\\u6C11\\u5E01"},
3398         {"ru_RU", "1", "RUB", "1,00\\u00A0\\u0440\\u0443\\u0431.", "1,00\\u00A0RUB", "1,00 \\u0440\\u043E\\u0441\\u0441\\u0438\\u0439\\u0441\\u043A\\u043E\\u0433\\u043E \\u0440\\u0443\\u0431\\u043B\\u044F"},
3399         {"ru_RU", "2", "RUB", "2,00\\u00A0\\u0440\\u0443\\u0431.", "2,00\\u00A0RUB", "2,00 \\u0440\\u043E\\u0441\\u0441\\u0438\\u0439\\u0441\\u043A\\u043E\\u0433\\u043E \\u0440\\u0443\\u0431\\u043B\\u044F"},
3400         {"ru_RU", "5", "RUB", "5,00\\u00A0\\u0440\\u0443\\u0431.", "5,00\\u00A0RUB", "5,00 \\u0440\\u043E\\u0441\\u0441\\u0438\\u0439\\u0441\\u043A\\u043E\\u0433\\u043E \\u0440\\u0443\\u0431\\u043B\\u044F"},
3401         // test locale without currency information
3402         {"root", "-1.23", "USD", "-US$\\u00A01.23", "-USD\\u00A01.23", "-1.23 USD"},
3403         // test choice format
3404         {"es_AR", "1", "INR", "INR1,00", "INR1,00", "1,00 rupia india"},
3405     };
3406     static const UNumberFormatStyle currencyStyles[] = {
3407         UNUM_CURRENCY,
3408         UNUM_CURRENCY_ISO,
3409         UNUM_CURRENCY_PLURAL
3410     };
3411 
3412     for (int32_t i=0; i<UPRV_LENGTHOF(DATA); ++i) {
3413       for (int32_t kIndex = 0; kIndex < UPRV_LENGTHOF(currencyStyles); ++kIndex) {
3414         UNumberFormatStyle k = currencyStyles[kIndex];
3415         const char* localeString = DATA[i][0];
3416         double numberToBeFormat = atof(DATA[i][1]);
3417         const char* currencyISOCode = DATA[i][2];
3418         Locale locale(localeString);
3419         UErrorCode status = U_ZERO_ERROR;
3420         NumberFormat* numFmt = NumberFormat::createInstance(locale, k, status);
3421         if (U_FAILURE(status)) {
3422             delete numFmt;
3423             dataerrln((UnicodeString)"can not create instance, locale:" + localeString + ", style: " + k + " - " + u_errorName(status));
3424             continue;
3425         }
3426         UChar currencyCode[4];
3427         u_charsToUChars(currencyISOCode, currencyCode, 4);
3428         numFmt->setCurrency(currencyCode, status);
3429         if (U_FAILURE(status)) {
3430             delete numFmt;
3431             errln((UnicodeString)"can not set currency:" + currencyISOCode);
3432             continue;
3433         }
3434 
3435         UnicodeString strBuf;
3436         numFmt->format(numberToBeFormat, strBuf);
3437         int resultDataIndex = 3 + kIndex;
3438         // DATA[i][resultDataIndex] is the currency format result
3439         // using 'k' currency style.
3440         UnicodeString formatResult = ctou(DATA[i][resultDataIndex]);
3441         if (strBuf.compare(formatResult)) {
3442             errln("FAIL: Expected " + formatResult + " actual: " + strBuf);
3443         }
3444         // test parsing, and test parsing for all currency formats.
3445         for (int j = 3; j < 6; ++j) {
3446             // DATA[i][3] is the currency format result using
3447             // CURRENCYSTYLE formatter.
3448             // DATA[i][4] is the currency format result using
3449             // ISOCURRENCYSTYLE formatter.
3450             // DATA[i][5] is the currency format result using
3451             // PLURALCURRENCYSTYLE formatter.
3452             UnicodeString oneCurrencyFormatResult = ctou(DATA[i][j]);
3453             UErrorCode status = U_ZERO_ERROR;
3454             Formattable parseResult;
3455             numFmt->parse(oneCurrencyFormatResult, parseResult, status);
3456             if (U_FAILURE(status) ||
3457                 (parseResult.getType() == Formattable::kDouble &&
3458                  parseResult.getDouble() != numberToBeFormat) ||
3459                 (parseResult.getType() == Formattable::kLong &&
3460                  parseResult.getLong() != numberToBeFormat)) {
3461                 errln((UnicodeString)"FAIL: getCurrencyFormat of locale " +
3462                       localeString + " failed roundtripping the number");
3463                 if (parseResult.getType() == Formattable::kDouble) {
3464                     errln((UnicodeString)"expected: " + numberToBeFormat + "; actual: " +parseResult.getDouble());
3465                 } else {
3466                     errln((UnicodeString)"expected: " + numberToBeFormat + "; actual: " +parseResult.getLong());
3467                 }
3468             }
3469         }
3470         delete numFmt;
3471       }
3472     }
3473 }
3474 
3475 void
TestCurrencyParsing()3476 NumberFormatTest::TestCurrencyParsing() {
3477     static const char* DATA[][6] = {
3478         // the data are:
3479         // locale,
3480         // currency amount to be formatted,
3481         // currency ISO code to be formatted,
3482         // format result using CURRENCYSTYLE,
3483         // format result using ISOCURRENCYSTYLE,
3484         // format result using PLURALCURRENCYSTYLE,
3485         {"en_US", "1", "USD", "$1.00", "USD1.00", "1.00 US dollar"},
3486         {"pa_IN", "1", "USD", "US$\\u00A01.00", "USD\\u00A01.00", "1.00 \\u0a2f\\u0a42.\\u0a10\\u0a38. \\u0a21\\u0a3e\\u0a32\\u0a30"},
3487         {"es_AR", "1", "USD", "US$1,00", "USD1,00", "1,00 d\\u00f3lar estadounidense"},
3488         {"ar_EG", "1", "USD", "US$\\u00a0\\u0661\\u066b\\u0660\\u0660", "USD\\u00a0\\u0661\\u066b\\u0660\\u0660", "\\u0661\\u066b\\u0660\\u0660 \\u062f\\u0648\\u0644\\u0627\\u0631 \\u0623\\u0645\\u0631\\u064a\\u0643\\u064a"},
3489         {"fa_CA", "1", "USD", "\\u200e$\\u06f1\\u066b\\u06f0\\u06f0", "\\u200eUSD\\u06f1\\u066b\\u06f0\\u06f0", "\\u200e\\u062f\\u0644\\u0627\\u0631 \\u0627\\u0645\\u0631\\u06cc\\u06a9\\u0627\\u06f1\\u066b\\u06f0\\u06f0"},
3490         {"he_IL", "1", "USD", "1.00\\u00a0$", "1.00\\u00a0USD", "1.00 \\u05d3\\u05d5\\u05dc\\u05e8 \\u05d0\\u05de\\u05e8\\u05d9\\u05e7\\u05d0\\u05d9"},
3491         {"hr_HR", "1", "USD", "1,00\\u00a0USD", "1,00\\u00a0USD", "1,00 Ameri\\u010dki dolar"},
3492         {"id_ID", "1", "USD", "US$1,00", "USD1,00", "1,00 Dolar Amerika Serikat"},
3493         {"it_IT", "1", "USD", "1,00\\u00a0US$", "1,00\\u00a0USD", "1,00 Dollaro Statunitense"},
3494         {"ko_KR", "1", "USD", "US$1.00", "USD1.00", "1.00\\ubbf8\\uad6d \\ub2ec\\ub7ec"},
3495         {"ja_JP", "1", "USD", "$1.00", "USD1.00", "1.00\\u7c73\\u30c9\\u30eb"},
3496         {"zh_CN", "1", "CNY", "\\uFFE5\\u00a01.00", "CNY\\u00a01.00", "1.00\\u4EBA\\u6C11\\u5E01"},
3497         {"zh_TW", "1", "CNY", "CN\\u00A51.00", "CNY1.00", "1.00 \\u4eba\\u6c11\\u5e63"},
3498         {"zh_Hant", "1", "CNY", "CN\\u00A51.00", "CNY1.00", "1.00 \\u4eba\\u6c11\\u5e63"},
3499         {"zh_Hant", "1", "JPY", "\\u00A51.00", "JPY1.00", "1.00 \\u65e5\\u5713"},
3500         {"ja_JP", "1", "JPY", "\\uFFE51.00", "JPY1.00", "1.00\\u65e5\\u672c\\u5186"},
3501         {"ja_JP", "1", "JPY", "\\u00A51.00", "JPY1.00", "1.00\\u65e5\\u672c\\u5186"},
3502         {"ru_RU", "1", "RUB", "1,00\\u00A0\\u0440\\u0443\\u0431.", "1,00\\u00A0RUB", "1,00 \\u0420\\u043E\\u0441\\u0441\\u0438\\u0439\\u0441\\u043A\\u0438\\u0439 \\u0440\\u0443\\u0431\\u043B\\u044C"}
3503     };
3504     static const UNumberFormatStyle currencyStyles[] = {
3505         UNUM_CURRENCY,
3506         UNUM_CURRENCY_ISO,
3507         UNUM_CURRENCY_PLURAL
3508     };
3509     static const char* currencyStyleNames[] = {
3510       "UNUM_CURRENCY",
3511       "UNUM_CURRENCY_ISO",
3512       "UNUM_CURRENCY_PLURAL"
3513     };
3514 
3515 #ifdef NUMFMTST_CACHE_DEBUG
3516 int deadloop = 0;
3517 for (;;) {
3518     printf("loop: %d\n", deadloop++);
3519 #endif
3520     for (uint32_t i=0; i< sizeof(DATA)/sizeof(DATA[0]); ++i) {  /* i = test case #  - should be i=0*/
3521       for (int32_t kIndex = 2; kIndex < UPRV_LENGTHOF(currencyStyles); ++kIndex) {
3522         UNumberFormatStyle k = currencyStyles[kIndex]; /* k = style */
3523         const char* localeString = DATA[i][0];
3524         double numberToBeFormat = atof(DATA[i][1]);
3525         const char* currencyISOCode = DATA[i][2];
3526         Locale locale(localeString);
3527         UErrorCode status = U_ZERO_ERROR;
3528         NumberFormat* numFmt = NumberFormat::createInstance(locale, k, status);
3529         logln("#%d NumberFormat(%s, %s) Currency=%s\n",
3530               i, localeString, currencyStyleNames[kIndex],
3531               currencyISOCode);
3532 
3533         if (U_FAILURE(status)) {
3534             delete numFmt;
3535             dataerrln((UnicodeString)"can not create instance, locale:" + localeString + ", style: " + k + " - " + u_errorName(status));
3536             continue;
3537         }
3538         UChar currencyCode[4];
3539         u_charsToUChars(currencyISOCode, currencyCode, 4);
3540         numFmt->setCurrency(currencyCode, status);
3541         if (U_FAILURE(status)) {
3542             delete numFmt;
3543             errln((UnicodeString)"can not set currency:" + currencyISOCode);
3544             continue;
3545         }
3546 
3547         UnicodeString strBuf;
3548         numFmt->format(numberToBeFormat, strBuf);
3549         /*
3550         int resultDataIndex = 3 + kIndex;
3551         // DATA[i][resultDataIndex] is the currency format result
3552         // using 'k' currency style.
3553         UnicodeString formatResult = ctou(DATA[i][resultDataIndex]);
3554         if (strBuf.compare(formatResult)) {
3555             errln("FAIL: Expected " + formatResult + " actual: " + strBuf);
3556         }
3557         */
3558         // test parsing, and test parsing for all currency formats.
3559         for (int j = 3; j < 6; ++j) {
3560             // DATA[i][3] is the currency format result using
3561             // CURRENCYSTYLE formatter.
3562             // DATA[i][4] is the currency format result using
3563             // ISOCURRENCYSTYLE formatter.
3564             // DATA[i][5] is the currency format result using
3565             // PLURALCURRENCYSTYLE formatter.
3566             UnicodeString oneCurrencyFormatResult = ctou(DATA[i][j]);
3567             UErrorCode status = U_ZERO_ERROR;
3568             Formattable parseResult;
3569             logln("parse(%s)", DATA[i][j]);
3570             numFmt->parse(oneCurrencyFormatResult, parseResult, status);
3571             if (U_FAILURE(status) ||
3572                 (parseResult.getType() == Formattable::kDouble &&
3573                  parseResult.getDouble() != numberToBeFormat) ||
3574                 (parseResult.getType() == Formattable::kLong &&
3575                  parseResult.getLong() != numberToBeFormat)) {
3576                 errln((UnicodeString)"FAIL: NumberFormat(" + localeString +", " + currencyStyleNames[kIndex] +
3577                       "), Currency="+currencyISOCode+", parse("+DATA[i][j]+") returned error " + (UnicodeString)u_errorName(status)+".  Testcase: data[" + i + "][" + currencyStyleNames[j-3] +"="+j+"]");
3578                 if (parseResult.getType() == Formattable::kDouble) {
3579                     errln((UnicodeString)"expected: " + numberToBeFormat + "; actual (double): " +parseResult.getDouble());
3580                 } else {
3581                     errln((UnicodeString)"expected: " + numberToBeFormat + "; actual (long): " +parseResult.getLong());
3582                 }
3583                 errln((UnicodeString)" round-trip would be: " + strBuf);
3584             }
3585         }
3586         delete numFmt;
3587       }
3588     }
3589 #ifdef NUMFMTST_CACHE_DEBUG
3590 }
3591 #endif
3592 }
3593 
3594 
3595 void
TestParseCurrencyInUCurr()3596 NumberFormatTest::TestParseCurrencyInUCurr() {
3597     const char* DATA[] = {
3598         "1.00 US DOLLAR",  // case in-sensitive
3599         "$1.00",
3600         "USD1.00",
3601         "US dollar1.00",
3602         "US dollars1.00",
3603         "$1.00",
3604         "A$1.00",
3605         "ADP1.00",
3606         "ADP1.00",
3607         "AED1.00",
3608         "AED1.00",
3609         "AFA1.00",
3610         "AFA1.00",
3611         "AFN1.00",
3612         "ALL1.00",
3613         "AMD1.00",
3614         "ANG1.00",
3615         "AOA1.00",
3616         "AOK1.00",
3617         "AOK1.00",
3618         "AON1.00",
3619         "AON1.00",
3620         "AOR1.00",
3621         "AOR1.00",
3622         "ARS1.00",
3623         "ARA1.00",
3624         "ARA1.00",
3625         "ARP1.00",
3626         "ARP1.00",
3627         "ARS1.00",
3628         "ATS1.00",
3629         "ATS1.00",
3630         "AUD1.00",
3631         "AWG1.00",
3632         "AZM1.00",
3633         "AZM1.00",
3634         "AZN1.00",
3635         "Afghan Afghani (1927\\u20132002)1.00",
3636         "Afghan afghani (1927\\u20132002)1.00",
3637         "Afghan Afghani1.00",
3638         "Afghan Afghanis1.00",
3639         "Albanian Lek1.00",
3640         "Albanian lek1.00",
3641         "Albanian lek\\u00eb1.00",
3642         "Algerian Dinar1.00",
3643         "Algerian dinar1.00",
3644         "Algerian dinars1.00",
3645         "Andorran Peseta1.00",
3646         "Andorran peseta1.00",
3647         "Andorran pesetas1.00",
3648         "Angolan Kwanza (1977\\u20131991)1.00",
3649         "Angolan Readjusted Kwanza (1995\\u20131999)1.00",
3650         "Angolan Kwanza1.00",
3651         "Angolan New Kwanza (1990\\u20132000)1.00",
3652         "Angolan kwanza (1977\\u20131991)1.00",
3653         "Angolan readjusted kwanza (1995\\u20131999)1.00",
3654         "Angolan kwanza1.00",
3655         "Angolan kwanzas (1977\\u20131991)1.00",
3656         "Angolan readjusted kwanzas (1995\\u20131999)1.00",
3657         "Angolan kwanzas1.00",
3658         "Angolan new kwanza (1990\\u20132000)1.00",
3659         "Angolan new kwanzas (1990\\u20132000)1.00",
3660         "Argentine Austral1.00",
3661         "Argentine Peso (1983\\u20131985)1.00",
3662         "Argentine Peso1.00",
3663         "Argentine austral1.00",
3664         "Argentine australs1.00",
3665         "Argentine peso (1983\\u20131985)1.00",
3666         "Argentine peso1.00",
3667         "Argentine pesos (1983\\u20131985)1.00",
3668         "Argentine pesos1.00",
3669         "Armenian Dram1.00",
3670         "Armenian dram1.00",
3671         "Armenian drams1.00",
3672         "Aruban Florin1.00",
3673         "Aruban florin1.00",
3674         "Australian Dollar1.00",
3675         "Australian dollar1.00",
3676         "Australian dollars1.00",
3677         "Austrian Schilling1.00",
3678         "Austrian schilling1.00",
3679         "Austrian schillings1.00",
3680         "Azerbaijani Manat (1993\\u20132006)1.00",
3681         "Azerbaijani Manat1.00",
3682         "Azerbaijani manat (1993\\u20132006)1.00",
3683         "Azerbaijani manat1.00",
3684         "Azerbaijani manats (1993\\u20132006)1.00",
3685         "Azerbaijani manats1.00",
3686         "BAD1.00",
3687         "BAD1.00",
3688         "BAM1.00",
3689         "BBD1.00",
3690         "BDT1.00",
3691         "BEC1.00",
3692         "BEC1.00",
3693         "BEF1.00",
3694         "BEL1.00",
3695         "BEL1.00",
3696         "BGL1.00",
3697         "BGN1.00",
3698         "BGN1.00",
3699         "BHD1.00",
3700         "BIF1.00",
3701         "BMD1.00",
3702         "BND1.00",
3703         "BOB1.00",
3704         "BOP1.00",
3705         "BOP1.00",
3706         "BOV1.00",
3707         "BOV1.00",
3708         "BRB1.00",
3709         "BRB1.00",
3710         "BRC1.00",
3711         "BRC1.00",
3712         "BRE1.00",
3713         "BRE1.00",
3714         "BRL1.00",
3715         "BRN1.00",
3716         "BRN1.00",
3717         "BRR1.00",
3718         "BRR1.00",
3719         "BSD1.00",
3720         "BSD1.00",
3721         "BTN1.00",
3722         "BUK1.00",
3723         "BUK1.00",
3724         "BWP1.00",
3725         "BYB1.00",
3726         "BYB1.00",
3727         "BYR1.00",
3728         "BZD1.00",
3729         "Bahamian Dollar1.00",
3730         "Bahamian dollar1.00",
3731         "Bahamian dollars1.00",
3732         "Bahraini Dinar1.00",
3733         "Bahraini dinar1.00",
3734         "Bahraini dinars1.00",
3735         "Bangladeshi Taka1.00",
3736         "Bangladeshi taka1.00",
3737         "Bangladeshi takas1.00",
3738         "Barbadian Dollar1.00",
3739         "Barbadian dollar1.00",
3740         "Barbadian dollars1.00",
3741         "Belarusian New Ruble (1994\\u20131999)1.00",
3742         "Belarusian Ruble1.00",
3743         "Belarusian new ruble (1994\\u20131999)1.00",
3744         "Belarusian new rubles (1994\\u20131999)1.00",
3745         "Belarusian ruble1.00",
3746         "Belarusian rubles1.00",
3747         "Belgian Franc (convertible)1.00",
3748         "Belgian Franc (financial)1.00",
3749         "Belgian Franc1.00",
3750         "Belgian franc (convertible)1.00",
3751         "Belgian franc (financial)1.00",
3752         "Belgian franc1.00",
3753         "Belgian francs (convertible)1.00",
3754         "Belgian francs (financial)1.00",
3755         "Belgian francs1.00",
3756         "Belize Dollar1.00",
3757         "Belize dollar1.00",
3758         "Belize dollars1.00",
3759         "Bermudan Dollar1.00",
3760         "Bermudan dollar1.00",
3761         "Bermudan dollars1.00",
3762         "Bhutanese Ngultrum1.00",
3763         "Bhutanese ngultrum1.00",
3764         "Bhutanese ngultrums1.00",
3765         "Bolivian Mvdol1.00",
3766         "Bolivian Peso1.00",
3767         "Bolivian mvdol1.00",
3768         "Bolivian mvdols1.00",
3769         "Bolivian peso1.00",
3770         "Bolivian pesos1.00",
3771         "Bolivian Boliviano1.00",
3772         "Bolivian Boliviano1.00",
3773         "Bolivian Bolivianos1.00",
3774         "Bosnia-Herzegovina Convertible Mark1.00",
3775         "Bosnia-Herzegovina Dinar (1992\\u20131994)1.00",
3776         "Bosnia-Herzegovina convertible mark1.00",
3777         "Bosnia-Herzegovina convertible marks1.00",
3778         "Bosnia-Herzegovina dinar (1992\\u20131994)1.00",
3779         "Bosnia-Herzegovina dinars (1992\\u20131994)1.00",
3780         "Botswanan Pula1.00",
3781         "Botswanan pula1.00",
3782         "Botswanan pulas1.00",
3783         "Brazilian New Cruzado (1989\\u20131990)1.00",
3784         "Brazilian Cruzado (1986\\u20131989)1.00",
3785         "Brazilian Cruzeiro (1990\\u20131993)1.00",
3786         "Brazilian New Cruzeiro (1967\\u20131986)1.00",
3787         "Brazilian Cruzeiro (1993\\u20131994)1.00",
3788         "Brazilian Real1.00",
3789         "Brazilian new cruzado (1989\\u20131990)1.00",
3790         "Brazilian new cruzados (1989\\u20131990)1.00",
3791         "Brazilian cruzado (1986\\u20131989)1.00",
3792         "Brazilian cruzados (1986\\u20131989)1.00",
3793         "Brazilian cruzeiro (1990\\u20131993)1.00",
3794         "Brazilian new cruzeiro (1967\\u20131986)1.00",
3795         "Brazilian cruzeiro (1993\\u20131994)1.00",
3796         "Brazilian cruzeiros (1990\\u20131993)1.00",
3797         "Brazilian new cruzeiros (1967\\u20131986)1.00",
3798         "Brazilian cruzeiros (1993\\u20131994)1.00",
3799         "Brazilian real1.00",
3800         "Brazilian reals1.00",
3801         "British Pound1.00",
3802         "British pound1.00",
3803         "British pounds1.00",
3804         "Brunei Dollar1.00",
3805         "Brunei dollar1.00",
3806         "Brunei dollars1.00",
3807         "Bulgarian Hard Lev1.00",
3808         "Bulgarian Lev1.00",
3809         "Bulgarian Leva1.00",
3810         "Bulgarian hard lev1.00",
3811         "Bulgarian hard leva1.00",
3812         "Bulgarian lev1.00",
3813         "Burmese Kyat1.00",
3814         "Burmese kyat1.00",
3815         "Burmese kyats1.00",
3816         "Burundian Franc1.00",
3817         "Burundian franc1.00",
3818         "Burundian francs1.00",
3819         "CA$1.00",
3820         "CAD1.00",
3821         "CDF1.00",
3822         "CDF1.00",
3823         "West African CFA Franc1.00",
3824         "Central African CFA Franc1.00",
3825         "West African CFA franc1.00",
3826         "Central African CFA franc1.00",
3827         "West African CFA francs1.00",
3828         "Central African CFA francs1.00",
3829         "CFP Franc1.00",
3830         "CFP franc1.00",
3831         "CFP francs1.00",
3832         "CFPF1.00",
3833         "CHE1.00",
3834         "CHE1.00",
3835         "CHF1.00",
3836         "CHW1.00",
3837         "CHW1.00",
3838         "CLF1.00",
3839         "CLF1.00",
3840         "CLP1.00",
3841         "CNY1.00",
3842         "COP1.00",
3843         "COU1.00",
3844         "COU1.00",
3845         "CRC1.00",
3846         "CSD1.00",
3847         "CSD1.00",
3848         "CSK1.00",
3849         "CSK1.00",
3850         "CUP1.00",
3851         "CUP1.00",
3852         "CVE1.00",
3853         "CYP1.00",
3854         "CZK1.00",
3855         "Cambodian Riel1.00",
3856         "Cambodian riel1.00",
3857         "Cambodian riels1.00",
3858         "Canadian Dollar1.00",
3859         "Canadian dollar1.00",
3860         "Canadian dollars1.00",
3861         "Cape Verdean Escudo1.00",
3862         "Cape Verdean escudo1.00",
3863         "Cape Verdean escudos1.00",
3864         "Cayman Islands Dollar1.00",
3865         "Cayman Islands dollar1.00",
3866         "Cayman Islands dollars1.00",
3867         "Chilean Peso1.00",
3868         "Chilean Unit of Account (UF)1.00",
3869         "Chilean peso1.00",
3870         "Chilean pesos1.00",
3871         "Chilean unit of account (UF)1.00",
3872         "Chilean units of account (UF)1.00",
3873         "Chinese Yuan1.00",
3874         "Chinese yuan1.00",
3875         "Colombian Peso1.00",
3876         "Colombian peso1.00",
3877         "Colombian pesos1.00",
3878         "Comorian Franc1.00",
3879         "Comorian franc1.00",
3880         "Comorian francs1.00",
3881         "Congolese Franc1.00",
3882         "Congolese franc1.00",
3883         "Congolese francs1.00",
3884         "Costa Rican Col\\u00f3n1.00",
3885         "Costa Rican col\\u00f3n1.00",
3886         "Costa Rican col\\u00f3ns1.00",
3887         "Croatian Dinar1.00",
3888         "Croatian Kuna1.00",
3889         "Croatian dinar1.00",
3890         "Croatian dinars1.00",
3891         "Croatian kuna1.00",
3892         "Croatian kunas1.00",
3893         "Cuban Peso1.00",
3894         "Cuban peso1.00",
3895         "Cuban pesos1.00",
3896         "Cypriot Pound1.00",
3897         "Cypriot pound1.00",
3898         "Cypriot pounds1.00",
3899         "Czech Republic Koruna1.00",
3900         "Czech Republic koruna1.00",
3901         "Czech Republic korunas1.00",
3902         "Czechoslovak Hard Koruna1.00",
3903         "Czechoslovak hard koruna1.00",
3904         "Czechoslovak hard korunas1.00",
3905         "DDM1.00",
3906         "DDM1.00",
3907         "DEM1.00",
3908         "DEM1.00",
3909         "DJF1.00",
3910         "DKK1.00",
3911         "DOP1.00",
3912         "DZD1.00",
3913         "Danish Krone1.00",
3914         "Danish krone1.00",
3915         "Danish kroner1.00",
3916         "German Mark1.00",
3917         "German mark1.00",
3918         "German marks1.00",
3919         "Djiboutian Franc1.00",
3920         "Djiboutian franc1.00",
3921         "Djiboutian francs1.00",
3922         "Dominican Peso1.00",
3923         "Dominican peso1.00",
3924         "Dominican pesos1.00",
3925         "EC$1.00",
3926         "ECS1.00",
3927         "ECS1.00",
3928         "ECV1.00",
3929         "ECV1.00",
3930         "EEK1.00",
3931         "EEK1.00",
3932         "EGP1.00",
3933         "EGP1.00",
3934         "ERN1.00",
3935         "ERN1.00",
3936         "ESA1.00",
3937         "ESA1.00",
3938         "ESB1.00",
3939         "ESB1.00",
3940         "ESP1.00",
3941         "ETB1.00",
3942         "EUR1.00",
3943         "East Caribbean Dollar1.00",
3944         "East Caribbean dollar1.00",
3945         "East Caribbean dollars1.00",
3946         "East German Mark1.00",
3947         "East German mark1.00",
3948         "East German marks1.00",
3949         "Ecuadorian Sucre1.00",
3950         "Ecuadorian Unit of Constant Value1.00",
3951         "Ecuadorian sucre1.00",
3952         "Ecuadorian sucres1.00",
3953         "Ecuadorian unit of constant value1.00",
3954         "Ecuadorian units of constant value1.00",
3955         "Egyptian Pound1.00",
3956         "Egyptian pound1.00",
3957         "Egyptian pounds1.00",
3958         "Salvadoran Col\\u00f3n1.00",
3959         "Salvadoran col\\u00f3n1.00",
3960         "Salvadoran colones1.00",
3961         "Equatorial Guinean Ekwele1.00",
3962         "Equatorial Guinean ekwele1.00",
3963         "Eritrean Nakfa1.00",
3964         "Eritrean nakfa1.00",
3965         "Eritrean nakfas1.00",
3966         "Estonian Kroon1.00",
3967         "Estonian kroon1.00",
3968         "Estonian kroons1.00",
3969         "Ethiopian Birr1.00",
3970         "Ethiopian birr1.00",
3971         "Ethiopian birrs1.00",
3972         "Euro1.00",
3973         "European Composite Unit1.00",
3974         "European Currency Unit1.00",
3975         "European Monetary Unit1.00",
3976         "European Unit of Account (XBC)1.00",
3977         "European Unit of Account (XBD)1.00",
3978         "European composite unit1.00",
3979         "European composite units1.00",
3980         "European currency unit1.00",
3981         "European currency units1.00",
3982         "European monetary unit1.00",
3983         "European monetary units1.00",
3984         "European unit of account (XBC)1.00",
3985         "European unit of account (XBD)1.00",
3986         "European units of account (XBC)1.00",
3987         "European units of account (XBD)1.00",
3988         "FIM1.00",
3989         "FIM1.00",
3990         "FJD1.00",
3991         "FKP1.00",
3992         "FKP1.00",
3993         "FRF1.00",
3994         "FRF1.00",
3995         "Falkland Islands Pound1.00",
3996         "Falkland Islands pound1.00",
3997         "Falkland Islands pounds1.00",
3998         "Fijian Dollar1.00",
3999         "Fijian dollar1.00",
4000         "Fijian dollars1.00",
4001         "Finnish Markka1.00",
4002         "Finnish markka1.00",
4003         "Finnish markkas1.00",
4004         "CHF1.00",
4005         "French Franc1.00",
4006         "French Gold Franc1.00",
4007         "French UIC-Franc1.00",
4008         "French UIC-franc1.00",
4009         "French UIC-francs1.00",
4010         "French franc1.00",
4011         "French francs1.00",
4012         "French gold franc1.00",
4013         "French gold francs1.00",
4014         "GBP1.00",
4015         "GEK1.00",
4016         "GEK1.00",
4017         "GEL1.00",
4018         "GHC1.00",
4019         "GHC1.00",
4020         "GHS1.00",
4021         "GIP1.00",
4022         "GIP1.00",
4023         "GMD1.00",
4024         "GMD1.00",
4025         "GNF1.00",
4026         "GNS1.00",
4027         "GNS1.00",
4028         "GQE1.00",
4029         "GQE1.00",
4030         "GRD1.00",
4031         "GRD1.00",
4032         "GTQ1.00",
4033         "GWE1.00",
4034         "GWE1.00",
4035         "GWP1.00",
4036         "GWP1.00",
4037         "GYD1.00",
4038         "Gambian Dalasi1.00",
4039         "Gambian dalasi1.00",
4040         "Gambian dalasis1.00",
4041         "Georgian Kupon Larit1.00",
4042         "Georgian Lari1.00",
4043         "Georgian kupon larit1.00",
4044         "Georgian kupon larits1.00",
4045         "Georgian lari1.00",
4046         "Georgian laris1.00",
4047         "Ghanaian Cedi (1979\\u20132007)1.00",
4048         "Ghanaian Cedi1.00",
4049         "Ghanaian cedi (1979\\u20132007)1.00",
4050         "Ghanaian cedi1.00",
4051         "Ghanaian cedis (1979\\u20132007)1.00",
4052         "Ghanaian cedis1.00",
4053         "Gibraltar Pound1.00",
4054         "Gibraltar pound1.00",
4055         "Gibraltar pounds1.00",
4056         "Gold1.00",
4057         "Gold1.00",
4058         "Greek Drachma1.00",
4059         "Greek drachma1.00",
4060         "Greek drachmas1.00",
4061         "Guatemalan Quetzal1.00",
4062         "Guatemalan quetzal1.00",
4063         "Guatemalan quetzals1.00",
4064         "Guinean Franc1.00",
4065         "Guinean Syli1.00",
4066         "Guinean franc1.00",
4067         "Guinean francs1.00",
4068         "Guinean syli1.00",
4069         "Guinean sylis1.00",
4070         "Guinea-Bissau Peso1.00",
4071         "Guinea-Bissau peso1.00",
4072         "Guinea-Bissau pesos1.00",
4073         "Guyanaese Dollar1.00",
4074         "Guyanaese dollar1.00",
4075         "Guyanaese dollars1.00",
4076         "HK$1.00",
4077         "HKD1.00",
4078         "HNL1.00",
4079         "HRD1.00",
4080         "HRD1.00",
4081         "HRK1.00",
4082         "HRK1.00",
4083         "HTG1.00",
4084         "HTG1.00",
4085         "HUF1.00",
4086         "Haitian Gourde1.00",
4087         "Haitian gourde1.00",
4088         "Haitian gourdes1.00",
4089         "Honduran Lempira1.00",
4090         "Honduran lempira1.00",
4091         "Honduran lempiras1.00",
4092         "Hong Kong Dollar1.00",
4093         "Hong Kong dollar1.00",
4094         "Hong Kong dollars1.00",
4095         "Hungarian Forint1.00",
4096         "Hungarian forint1.00",
4097         "Hungarian forints1.00",
4098         "IDR1.00",
4099         "IEP1.00",
4100         "ILP1.00",
4101         "ILP1.00",
4102         "ILS1.00",
4103         "INR1.00",
4104         "IQD1.00",
4105         "IRR1.00",
4106         "ISK1.00",
4107         "ISK1.00",
4108         "ITL1.00",
4109         "Icelandic Kr\\u00f3na1.00",
4110         "Icelandic kr\\u00f3na1.00",
4111         "Icelandic kr\\u00f3nur1.00",
4112         "Indian Rupee1.00",
4113         "Indian rupee1.00",
4114         "Indian rupees1.00",
4115         "Indonesian Rupiah1.00",
4116         "Indonesian rupiah1.00",
4117         "Indonesian rupiahs1.00",
4118         "Iranian Rial1.00",
4119         "Iranian rial1.00",
4120         "Iranian rials1.00",
4121         "Iraqi Dinar1.00",
4122         "Iraqi dinar1.00",
4123         "Iraqi dinars1.00",
4124         "Irish Pound1.00",
4125         "Irish pound1.00",
4126         "Irish pounds1.00",
4127         "Israeli Pound1.00",
4128         "Israeli new sheqel1.00",
4129         "Israeli pound1.00",
4130         "Israeli pounds1.00",
4131         "Italian Lira1.00",
4132         "Italian lira1.00",
4133         "Italian liras1.00",
4134         "JMD1.00",
4135         "JOD1.00",
4136         "JPY1.00",
4137         "Jamaican Dollar1.00",
4138         "Jamaican dollar1.00",
4139         "Jamaican dollars1.00",
4140         "Japanese Yen1.00",
4141         "Japanese yen1.00",
4142         "Jordanian Dinar1.00",
4143         "Jordanian dinar1.00",
4144         "Jordanian dinars1.00",
4145         "KES1.00",
4146         "KGS1.00",
4147         "KHR1.00",
4148         "KMF1.00",
4149         "KPW1.00",
4150         "KPW1.00",
4151         "KRW1.00",
4152         "KWD1.00",
4153         "KYD1.00",
4154         "KYD1.00",
4155         "KZT1.00",
4156         "Kazakhstani Tenge1.00",
4157         "Kazakhstani tenge1.00",
4158         "Kazakhstani tenges1.00",
4159         "Kenyan Shilling1.00",
4160         "Kenyan shilling1.00",
4161         "Kenyan shillings1.00",
4162         "Kuwaiti Dinar1.00",
4163         "Kuwaiti dinar1.00",
4164         "Kuwaiti dinars1.00",
4165         "Kyrgystani Som1.00",
4166         "Kyrgystani som1.00",
4167         "Kyrgystani soms1.00",
4168         "HNL1.00",
4169         "LAK1.00",
4170         "LAK1.00",
4171         "LBP1.00",
4172         "LKR1.00",
4173         "LRD1.00",
4174         "LRD1.00",
4175         "LSL1.00",
4176         "LTL1.00",
4177         "LTL1.00",
4178         "LTT1.00",
4179         "LTT1.00",
4180         "LUC1.00",
4181         "LUC1.00",
4182         "LUF1.00",
4183         "LUF1.00",
4184         "LUL1.00",
4185         "LUL1.00",
4186         "LVL1.00",
4187         "LVL1.00",
4188         "LVR1.00",
4189         "LVR1.00",
4190         "LYD1.00",
4191         "Laotian Kip1.00",
4192         "Laotian kip1.00",
4193         "Laotian kips1.00",
4194         "Latvian Lats1.00",
4195         "Latvian Ruble1.00",
4196         "Latvian lats1.00",
4197         "Latvian lati1.00",
4198         "Latvian ruble1.00",
4199         "Latvian rubles1.00",
4200         "Lebanese Pound1.00",
4201         "Lebanese pound1.00",
4202         "Lebanese pounds1.00",
4203         "Lesotho Loti1.00",
4204         "Lesotho loti1.00",
4205         "Lesotho lotis1.00",
4206         "Liberian Dollar1.00",
4207         "Liberian dollar1.00",
4208         "Liberian dollars1.00",
4209         "Libyan Dinar1.00",
4210         "Libyan dinar1.00",
4211         "Libyan dinars1.00",
4212         "Lithuanian Litas1.00",
4213         "Lithuanian Talonas1.00",
4214         "Lithuanian litas1.00",
4215         "Lithuanian litai1.00",
4216         "Lithuanian talonas1.00",
4217         "Lithuanian talonases1.00",
4218         "Luxembourgian Convertible Franc1.00",
4219         "Luxembourg Financial Franc1.00",
4220         "Luxembourgian Franc1.00",
4221         "Luxembourgian convertible franc1.00",
4222         "Luxembourgian convertible francs1.00",
4223         "Luxembourg financial franc1.00",
4224         "Luxembourg financial francs1.00",
4225         "Luxembourgian franc1.00",
4226         "Luxembourgian francs1.00",
4227         "MAD1.00",
4228         "MAD1.00",
4229         "MAF1.00",
4230         "MAF1.00",
4231         "MDL1.00",
4232         "MDL1.00",
4233         "MX$1.00",
4234         "MGA1.00",
4235         "MGA1.00",
4236         "MGF1.00",
4237         "MGF1.00",
4238         "MKD1.00",
4239         "MLF1.00",
4240         "MLF1.00",
4241         "MMK1.00",
4242         "MMK1.00",
4243         "MNT1.00",
4244         "MOP1.00",
4245         "MOP1.00",
4246         "MRO1.00",
4247         "MTL1.00",
4248         "MTP1.00",
4249         "MTP1.00",
4250         "MUR1.00",
4251         "MUR1.00",
4252         "MVR1.00",
4253         "MVR1.00",
4254         "MWK1.00",
4255         "MXN1.00",
4256         "MXP1.00",
4257         "MXP1.00",
4258         "MXV1.00",
4259         "MXV1.00",
4260         "MYR1.00",
4261         "MZE1.00",
4262         "MZE1.00",
4263         "MZM1.00",
4264         "MZN1.00",
4265         "Macanese Pataca1.00",
4266         "Macanese pataca1.00",
4267         "Macanese patacas1.00",
4268         "Macedonian Denar1.00",
4269         "Macedonian denar1.00",
4270         "Macedonian denari1.00",
4271         "Malagasy Ariaries1.00",
4272         "Malagasy Ariary1.00",
4273         "Malagasy Ariary1.00",
4274         "Malagasy Franc1.00",
4275         "Malagasy franc1.00",
4276         "Malagasy francs1.00",
4277         "Malawian Kwacha1.00",
4278         "Malawian Kwacha1.00",
4279         "Malawian Kwachas1.00",
4280         "Malaysian Ringgit1.00",
4281         "Malaysian ringgit1.00",
4282         "Malaysian ringgits1.00",
4283         "Maldivian Rufiyaa1.00",
4284         "Maldivian rufiyaa1.00",
4285         "Maldivian rufiyaas1.00",
4286         "Malian Franc1.00",
4287         "Malian franc1.00",
4288         "Malian francs1.00",
4289         "Maltese Lira1.00",
4290         "Maltese Pound1.00",
4291         "Maltese lira1.00",
4292         "Maltese lira1.00",
4293         "Maltese pound1.00",
4294         "Maltese pounds1.00",
4295         "Mauritanian Ouguiya1.00",
4296         "Mauritanian ouguiya1.00",
4297         "Mauritanian ouguiyas1.00",
4298         "Mauritian Rupee1.00",
4299         "Mauritian rupee1.00",
4300         "Mauritian rupees1.00",
4301         "Mexican Peso1.00",
4302         "Mexican Silver Peso (1861\\u20131992)1.00",
4303         "Mexican Investment Unit1.00",
4304         "Mexican peso1.00",
4305         "Mexican pesos1.00",
4306         "Mexican silver peso (1861\\u20131992)1.00",
4307         "Mexican silver pesos (1861\\u20131992)1.00",
4308         "Mexican investment unit1.00",
4309         "Mexican investment units1.00",
4310         "Moldovan Leu1.00",
4311         "Moldovan leu1.00",
4312         "Moldovan lei1.00",
4313         "Mongolian Tugrik1.00",
4314         "Mongolian tugrik1.00",
4315         "Mongolian tugriks1.00",
4316         "Moroccan Dirham1.00",
4317         "Moroccan Franc1.00",
4318         "Moroccan dirham1.00",
4319         "Moroccan dirhams1.00",
4320         "Moroccan franc1.00",
4321         "Moroccan francs1.00",
4322         "Mozambican Escudo1.00",
4323         "Mozambican Metical1.00",
4324         "Mozambican escudo1.00",
4325         "Mozambican escudos1.00",
4326         "Mozambican metical1.00",
4327         "Mozambican meticals1.00",
4328         "Myanmar Kyat1.00",
4329         "Myanmar kyat1.00",
4330         "Myanmar kyats1.00",
4331         "NAD1.00",
4332         "NGN1.00",
4333         "NIC1.00",
4334         "NIO1.00",
4335         "NIO1.00",
4336         "NLG1.00",
4337         "NLG1.00",
4338         "NOK1.00",
4339         "NPR1.00",
4340         "NT$1.00",
4341         "NZ$1.00",
4342         "NZD1.00",
4343         "Namibian Dollar1.00",
4344         "Namibian dollar1.00",
4345         "Namibian dollars1.00",
4346         "Nepalese Rupee1.00",
4347         "Nepalese rupee1.00",
4348         "Nepalese rupees1.00",
4349         "Netherlands Antillean Guilder1.00",
4350         "Netherlands Antillean guilder1.00",
4351         "Netherlands Antillean guilders1.00",
4352         "Dutch Guilder1.00",
4353         "Dutch guilder1.00",
4354         "Dutch guilders1.00",
4355         "Israeli New Sheqel1.00",
4356         "Israeli New Sheqels1.00",
4357         "New Zealand Dollar1.00",
4358         "New Zealand dollar1.00",
4359         "New Zealand dollars1.00",
4360         "Nicaraguan C\\u00f3rdoba1.00",
4361         "Nicaraguan C\\u00f3rdoba (1988\\u20131991)1.00",
4362         "Nicaraguan c\\u00f3rdoba1.00",
4363         "Nicaraguan c\\u00f3rdobas1.00",
4364         "Nicaraguan c\\u00f3rdoba (1988\\u20131991)1.00",
4365         "Nicaraguan c\\u00f3rdobas (1988\\u20131991)1.00",
4366         "Nigerian Naira1.00",
4367         "Nigerian naira1.00",
4368         "Nigerian nairas1.00",
4369         "North Korean Won1.00",
4370         "North Korean won1.00",
4371         "North Korean won1.00",
4372         "Norwegian Krone1.00",
4373         "Norwegian krone1.00",
4374         "Norwegian kroner1.00",
4375         "OMR1.00",
4376         "Mozambican Metical (1980\\u20132006)1.00",
4377         "Mozambican metical (1980\\u20132006)1.00",
4378         "Mozambican meticals (1980\\u20132006)1.00",
4379         "Romanian Lei (1952\\u20132006)1.00",
4380         "Romanian Leu (1952\\u20132006)1.00",
4381         "Romanian leu (1952\\u20132006)1.00",
4382         "Serbian Dinar (2002\\u20132006)1.00",
4383         "Serbian dinar (2002\\u20132006)1.00",
4384         "Serbian dinars (2002\\u20132006)1.00",
4385         "Sudanese Dinar (1992\\u20132007)1.00",
4386         "Sudanese Pound (1957\\u20131998)1.00",
4387         "Sudanese dinar (1992\\u20132007)1.00",
4388         "Sudanese dinars (1992\\u20132007)1.00",
4389         "Sudanese pound (1957\\u20131998)1.00",
4390         "Sudanese pounds (1957\\u20131998)1.00",
4391         "Turkish Lira (1922\\u20132005)1.00",
4392         "Turkish Lira (1922\\u20132005)1.00",
4393         "Omani Rial1.00",
4394         "Omani rial1.00",
4395         "Omani rials1.00",
4396         "PAB1.00",
4397         "PAB1.00",
4398         "PEI1.00",
4399         "PEI1.00",
4400         "PEN1.00",
4401         "PEN1.00",
4402         "PES1.00",
4403         "PES1.00",
4404         "PGK1.00",
4405         "PGK1.00",
4406         "PHP1.00",
4407         "PKR1.00",
4408         "PLN1.00",
4409         "PLZ1.00",
4410         "PLZ1.00",
4411         "PTE1.00",
4412         "PTE1.00",
4413         "PYG1.00",
4414         "Pakistani Rupee1.00",
4415         "Pakistani rupee1.00",
4416         "Pakistani rupees1.00",
4417         "Palladium1.00",
4418         "Palladium1.00",
4419         "Panamanian Balboa1.00",
4420         "Panamanian balboa1.00",
4421         "Panamanian balboas1.00",
4422         "Papua New Guinean Kina1.00",
4423         "Papua New Guinean kina1.00",
4424         "Papua New Guinean kina1.00",
4425         "Paraguayan Guarani1.00",
4426         "Paraguayan guarani1.00",
4427         "Paraguayan guaranis1.00",
4428         "Peruvian Inti1.00",
4429         "Peruvian Nuevo Sol1.00",
4430         "Peruvian Sol (1863\\u20131965)1.00",
4431         "Peruvian inti1.00",
4432         "Peruvian intis1.00",
4433         "Peruvian nuevo sol1.00",
4434         "Peruvian nuevos soles1.00",
4435         "Peruvian sol (1863\\u20131965)1.00",
4436         "Peruvian soles (1863\\u20131965)1.00",
4437         "Philippine Peso1.00",
4438         "Philippine peso1.00",
4439         "Philippine pesos1.00",
4440         "Platinum1.00",
4441         "Platinum1.00",
4442         "Polish Zloty (1950\\u20131995)1.00",
4443         "Polish Zloty1.00",
4444         "Polish zlotys1.00",
4445         "Polish zloty (PLZ)1.00",
4446         "Polish zloty1.00",
4447         "Polish zlotys (PLZ)1.00",
4448         "Portuguese Escudo1.00",
4449         "Portuguese Guinea Escudo1.00",
4450         "Portuguese Guinea escudo1.00",
4451         "Portuguese Guinea escudos1.00",
4452         "Portuguese escudo1.00",
4453         "Portuguese escudos1.00",
4454         "GTQ1.00",
4455         "QAR1.00",
4456         "Qatari Rial1.00",
4457         "Qatari rial1.00",
4458         "Qatari rials1.00",
4459         "RHD1.00",
4460         "RHD1.00",
4461         "RINET Funds1.00",
4462         "RINET Funds1.00",
4463         "CN\\u00a51.00",
4464         "ROL1.00",
4465         "ROL1.00",
4466         "RON1.00",
4467         "RON1.00",
4468         "RSD1.00",
4469         "RSD1.00",
4470         "RUB1.00",
4471         "RUB1.00",
4472         "RUR1.00",
4473         "RUR1.00",
4474         "RWF1.00",
4475         "RWF1.00",
4476         "Rhodesian Dollar1.00",
4477         "Rhodesian dollar1.00",
4478         "Rhodesian dollars1.00",
4479         "Romanian Leu1.00",
4480         "Romanian lei1.00",
4481         "Romanian leu1.00",
4482         "Russian Ruble (1991\\u20131998)1.00",
4483         "Russian Ruble1.00",
4484         "Russian ruble (1991\\u20131998)1.00",
4485         "Russian ruble1.00",
4486         "Russian rubles (1991\\u20131998)1.00",
4487         "Russian rubles1.00",
4488         "Rwandan Franc1.00",
4489         "Rwandan franc1.00",
4490         "Rwandan francs1.00",
4491         "SAR1.00",
4492         "SBD1.00",
4493         "SCR1.00",
4494         "SDD1.00",
4495         "SDD1.00",
4496         "SDG1.00",
4497         "SDG1.00",
4498         "SDP1.00",
4499         "SDP1.00",
4500         "SEK1.00",
4501         "SGD1.00",
4502         "SHP1.00",
4503         "SHP1.00",
4504         "SIT1.00",
4505         "SIT1.00",
4506         "SKK1.00",
4507         "SLL1.00",
4508         "SLL1.00",
4509         "SOS1.00",
4510         "SRD1.00",
4511         "SRD1.00",
4512         "SRG1.00",
4513         "STD1.00",
4514         "SUR1.00",
4515         "SUR1.00",
4516         "SVC1.00",
4517         "SVC1.00",
4518         "SYP1.00",
4519         "SZL1.00",
4520         "St. Helena Pound1.00",
4521         "St. Helena pound1.00",
4522         "St. Helena pounds1.00",
4523         "S\\u00e3o Tom\\u00e9 & Pr\\u00edncipe Dobra1.00",
4524         "S\\u00e3o Tom\\u00e9 & Pr\\u00edncipe dobra1.00",
4525         "S\\u00e3o Tom\\u00e9 & Pr\\u00edncipe dobras1.00",
4526         "Saudi Riyal1.00",
4527         "Saudi riyal1.00",
4528         "Saudi riyals1.00",
4529         "Serbian Dinar1.00",
4530         "Serbian dinar1.00",
4531         "Serbian dinars1.00",
4532         "Seychellois Rupee1.00",
4533         "Seychellois rupee1.00",
4534         "Seychellois rupees1.00",
4535         "Sierra Leonean Leone1.00",
4536         "Sierra Leonean leone1.00",
4537         "Sierra Leonean leones1.00",
4538         "Silver1.00",
4539         "Silver1.00",
4540         "Singapore Dollar1.00",
4541         "Singapore dollar1.00",
4542         "Singapore dollars1.00",
4543         "Slovak Koruna1.00",
4544         "Slovak koruna1.00",
4545         "Slovak korunas1.00",
4546         "Slovenian Tolar1.00",
4547         "Slovenian tolar1.00",
4548         "Slovenian tolars1.00",
4549         "Solomon Islands Dollar1.00",
4550         "Solomon Islands dollar1.00",
4551         "Solomon Islands dollars1.00",
4552         "Somali Shilling1.00",
4553         "Somali shilling1.00",
4554         "Somali shillings1.00",
4555         "South African Rand (financial)1.00",
4556         "South African Rand1.00",
4557         "South African rand (financial)1.00",
4558         "South African rand1.00",
4559         "South African rands (financial)1.00",
4560         "South African rand1.00",
4561         "South Korean Won1.00",
4562         "South Korean won1.00",
4563         "South Korean won1.00",
4564         "Soviet Rouble1.00",
4565         "Soviet rouble1.00",
4566         "Soviet roubles1.00",
4567         "Spanish Peseta (A account)1.00",
4568         "Spanish Peseta (convertible account)1.00",
4569         "Spanish Peseta1.00",
4570         "Spanish peseta (A account)1.00",
4571         "Spanish peseta (convertible account)1.00",
4572         "Spanish peseta1.00",
4573         "Spanish pesetas (A account)1.00",
4574         "Spanish pesetas (convertible account)1.00",
4575         "Spanish pesetas1.00",
4576         "Special Drawing Rights1.00",
4577         "Sri Lankan Rupee1.00",
4578         "Sri Lankan rupee1.00",
4579         "Sri Lankan rupees1.00",
4580         "Sudanese Pound1.00",
4581         "Sudanese pound1.00",
4582         "Sudanese pounds1.00",
4583         "Surinamese Dollar1.00",
4584         "Surinamese dollar1.00",
4585         "Surinamese dollars1.00",
4586         "Surinamese Guilder1.00",
4587         "Surinamese guilder1.00",
4588         "Surinamese guilders1.00",
4589         "Swazi Lilangeni1.00",
4590         "Swazi lilangeni1.00",
4591         "Swazi emalangeni1.00",
4592         "Swedish Krona1.00",
4593         "Swedish krona1.00",
4594         "Swedish kronor1.00",
4595         "Swiss Franc1.00",
4596         "Swiss franc1.00",
4597         "Swiss francs1.00",
4598         "Syrian Pound1.00",
4599         "Syrian pound1.00",
4600         "Syrian pounds1.00",
4601         "THB1.00",
4602         "TJR1.00",
4603         "TJR1.00",
4604         "TJS1.00",
4605         "TJS1.00",
4606         "TMM1.00",
4607         "TMM1.00",
4608         "TND1.00",
4609         "TND1.00",
4610         "TOP1.00",
4611         "TPE1.00",
4612         "TPE1.00",
4613         "TRL1.00",
4614         "TRY1.00",
4615         "TRY1.00",
4616         "TTD1.00",
4617         "TWD1.00",
4618         "TZS1.00",
4619         "New Taiwan Dollar1.00",
4620         "New Taiwan dollar1.00",
4621         "New Taiwan dollars1.00",
4622         "Tajikistani Ruble1.00",
4623         "Tajikistani Somoni1.00",
4624         "Tajikistani ruble1.00",
4625         "Tajikistani rubles1.00",
4626         "Tajikistani somoni1.00",
4627         "Tajikistani somonis1.00",
4628         "Tanzanian Shilling1.00",
4629         "Tanzanian shilling1.00",
4630         "Tanzanian shillings1.00",
4631         "Testing Currency Code1.00",
4632         "Testing Currency Code1.00",
4633         "Thai Baht1.00",
4634         "Thai baht1.00",
4635         "Thai baht1.00",
4636         "Timorese Escudo1.00",
4637         "Timorese escudo1.00",
4638         "Timorese escudos1.00",
4639         "Tongan Pa\\u02bbanga1.00",
4640         "Tongan pa\\u02bbanga1.00",
4641         "Tongan pa\\u02bbanga1.00",
4642         "Trinidad & Tobago Dollar1.00",
4643         "Trinidad & Tobago dollar1.00",
4644         "Trinidad & Tobago dollars1.00",
4645         "Tunisian Dinar1.00",
4646         "Tunisian dinar1.00",
4647         "Tunisian dinars1.00",
4648         "Turkish Lira1.00",
4649         "Turkish Lira1.00",
4650         "Turkish lira1.00",
4651         "Turkmenistani Manat1.00",
4652         "Turkmenistani manat1.00",
4653         "Turkmenistani manat1.00",
4654         "UAE dirham1.00",
4655         "UAE dirhams1.00",
4656         "UAH1.00",
4657         "UAK1.00",
4658         "UAK1.00",
4659         "UGS1.00",
4660         "UGS1.00",
4661         "UGX1.00",
4662         "US Dollar (Next day)1.00",
4663         "US Dollar (Same day)1.00",
4664         "US Dollar1.00",
4665         "US dollar (next day)1.00",
4666         "US dollar (same day)1.00",
4667         "US dollar1.00",
4668         "US dollars (next day)1.00",
4669         "US dollars (same day)1.00",
4670         "US dollars1.00",
4671         "USD1.00",
4672         "USN1.00",
4673         "USN1.00",
4674         "USS1.00",
4675         "USS1.00",
4676         "UYI1.00",
4677         "UYI1.00",
4678         "UYP1.00",
4679         "UYP1.00",
4680         "UYU1.00",
4681         "UZS1.00",
4682         "UZS1.00",
4683         "Ugandan Shilling (1966\\u20131987)1.00",
4684         "Ugandan Shilling1.00",
4685         "Ugandan shilling (1966\\u20131987)1.00",
4686         "Ugandan shilling1.00",
4687         "Ugandan shillings (1966\\u20131987)1.00",
4688         "Ugandan shillings1.00",
4689         "Ukrainian Hryvnia1.00",
4690         "Ukrainian Karbovanets1.00",
4691         "Ukrainian hryvnia1.00",
4692         "Ukrainian hryvnias1.00",
4693         "Ukrainian karbovanets1.00",
4694         "Ukrainian karbovantsiv1.00",
4695         "Colombian Real Value Unit1.00",
4696         "United Arab Emirates Dirham1.00",
4697         "Unknown Currency1.00",
4698         "Uruguayan Peso (1975\\u20131993)1.00",
4699         "Uruguayan Peso1.00",
4700         "Uruguayan Peso (Indexed Units)1.00",
4701         "Uruguayan peso (1975\\u20131993)1.00",
4702         "Uruguayan peso (indexed units)1.00",
4703         "Uruguayan peso1.00",
4704         "Uruguayan pesos (1975\\u20131993)1.00",
4705         "Uruguayan pesos (indexed units)1.00",
4706         "Uruguayan pesos1.00",
4707         "Uzbekistani Som1.00",
4708         "Uzbekistani som1.00",
4709         "Uzbekistani som1.00",
4710         "VEB1.00",
4711         "VEF1.00",
4712         "VND1.00",
4713         "VUV1.00",
4714         "Vanuatu Vatu1.00",
4715         "Vanuatu vatu1.00",
4716         "Vanuatu vatus1.00",
4717         "Venezuelan Bol\\u00edvar1.00",
4718         "Venezuelan Bol\\u00edvar (1871\\u20132008)1.00",
4719         "Venezuelan bol\\u00edvar1.00",
4720         "Venezuelan bol\\u00edvars1.00",
4721         "Venezuelan bol\\u00edvar (1871\\u20132008)1.00",
4722         "Venezuelan bol\\u00edvars (1871\\u20132008)1.00",
4723         "Vietnamese Dong1.00",
4724         "Vietnamese dong1.00",
4725         "Vietnamese dong1.00",
4726         "WIR Euro1.00",
4727         "WIR Franc1.00",
4728         "WIR euro1.00",
4729         "WIR euros1.00",
4730         "WIR franc1.00",
4731         "WIR francs1.00",
4732         "WST1.00",
4733         "WST1.00",
4734         "Samoan Tala1.00",
4735         "Samoan tala1.00",
4736         "Samoan tala1.00",
4737         "XAF1.00",
4738         "XAF1.00",
4739         "XAG1.00",
4740         "XAG1.00",
4741         "XAU1.00",
4742         "XAU1.00",
4743         "XBA1.00",
4744         "XBA1.00",
4745         "XBB1.00",
4746         "XBB1.00",
4747         "XBC1.00",
4748         "XBC1.00",
4749         "XBD1.00",
4750         "XBD1.00",
4751         "XCD1.00",
4752         "XDR1.00",
4753         "XDR1.00",
4754         "XEU1.00",
4755         "XEU1.00",
4756         "XFO1.00",
4757         "XFO1.00",
4758         "XFU1.00",
4759         "XFU1.00",
4760         "XOF1.00",
4761         "XOF1.00",
4762         "XPD1.00",
4763         "XPD1.00",
4764         "XPF1.00",
4765         "XPT1.00",
4766         "XPT1.00",
4767         "XRE1.00",
4768         "XRE1.00",
4769         "XTS1.00",
4770         "XTS1.00",
4771         "XXX1.00",
4772         "XXX1.00",
4773         "YDD1.00",
4774         "YDD1.00",
4775         "YER1.00",
4776         "YUD1.00",
4777         "YUD1.00",
4778         "YUM1.00",
4779         "YUM1.00",
4780         "YUN1.00",
4781         "YUN1.00",
4782         "Yemeni Dinar1.00",
4783         "Yemeni Rial1.00",
4784         "Yemeni dinar1.00",
4785         "Yemeni dinars1.00",
4786         "Yemeni rial1.00",
4787         "Yemeni rials1.00",
4788         "Yugoslavian Convertible Dinar (1990\\u20131992)1.00",
4789         "Yugoslavian Hard Dinar (1966\\u20131990)1.00",
4790         "Yugoslavian New Dinar (1994\\u20132002)1.00",
4791         "Yugoslavian convertible dinar (1990\\u20131992)1.00",
4792         "Yugoslavian convertible dinars (1990\\u20131992)1.00",
4793         "Yugoslavian hard dinar (1966\\u20131990)1.00",
4794         "Yugoslavian hard dinars (1966\\u20131990)1.00",
4795         "Yugoslavian new dinar (1994\\u20132002)1.00",
4796         "Yugoslavian new dinars (1994\\u20132002)1.00",
4797         "ZAL1.00",
4798         "ZAL1.00",
4799         "ZAR1.00",
4800         "ZMK1.00",
4801         "ZMK1.00",
4802         "ZRN1.00",
4803         "ZRN1.00",
4804         "ZRZ1.00",
4805         "ZRZ1.00",
4806         "ZWD1.00",
4807         "Zairean New Zaire (1993\\u20131998)1.00",
4808         "Zairean Zaire (1971\\u20131993)1.00",
4809         "Zairean new zaire (1993\\u20131998)1.00",
4810         "Zairean new zaires (1993\\u20131998)1.00",
4811         "Zairean zaire (1971\\u20131993)1.00",
4812         "Zairean zaires (1971\\u20131993)1.00",
4813         "Zambian Kwacha1.00",
4814         "Zambian kwacha1.00",
4815         "Zambian kwachas1.00",
4816         "Zimbabwean Dollar (1980\\u20132008)1.00",
4817         "Zimbabwean dollar (1980\\u20132008)1.00",
4818         "Zimbabwean dollars (1980\\u20132008)1.00",
4819         "euro1.00",
4820         "euros1.00",
4821         "Turkish lira (1922\\u20132005)1.00",
4822         "special drawing rights1.00",
4823         "Colombian real value unit1.00",
4824         "Colombian real value units1.00",
4825         "unknown currency1.00",
4826         "\\u00a31.00",
4827         "\\u00a51.00",
4828         "\\u20ab1.00",
4829         "\\u20aa1.00",
4830         "\\u20ac1.00",
4831         "\\u20b91.00",
4832         //
4833         // Following has extra text, should be parsed correctly too
4834         "$1.00 random",
4835         "USD1.00 random",
4836         "1.00 US dollar random",
4837         "1.00 US dollars random",
4838         "1.00 Afghan Afghani random",
4839         "1.00 Afghan Afghani random",
4840         "1.00 Afghan Afghanis (1927\\u20131992) random",
4841         "1.00 Afghan Afghanis random",
4842         "1.00 Albanian Lek random",
4843         "1.00 Albanian lek random",
4844         "1.00 Albanian lek\\u00eb random",
4845         "1.00 Algerian Dinar random",
4846         "1.00 Algerian dinar random",
4847         "1.00 Algerian dinars random",
4848         "1.00 Andorran Peseta random",
4849         "1.00 Andorran peseta random",
4850         "1.00 Andorran pesetas random",
4851         "1.00 Angolan Kwanza (1977\\u20131990) random",
4852         "1.00 Angolan Readjusted Kwanza (1995\\u20131999) random",
4853         "1.00 Angolan Kwanza random",
4854         "1.00 Angolan New Kwanza (1990\\u20132000) random",
4855         "1.00 Angolan kwanza (1977\\u20131991) random",
4856         "1.00 Angolan readjusted kwanza (1995\\u20131999) random",
4857         "1.00 Angolan kwanza random",
4858         "1.00 Angolan kwanzas (1977\\u20131991) random",
4859         "1.00 Angolan readjusted kwanzas (1995\\u20131999) random",
4860         "1.00 Angolan kwanzas random",
4861         "1.00 Angolan new kwanza (1990\\u20132000) random",
4862         "1.00 Angolan new kwanzas (1990\\u20132000) random",
4863         "1.00 Argentine Austral random",
4864         "1.00 Argentine Peso (1983\\u20131985) random",
4865         "1.00 Argentine Peso random",
4866         "1.00 Argentine austral random",
4867         "1.00 Argentine australs random",
4868         "1.00 Argentine peso (1983\\u20131985) random",
4869         "1.00 Argentine peso random",
4870         "1.00 Argentine pesos (1983\\u20131985) random",
4871         "1.00 Argentine pesos random",
4872         "1.00 Armenian Dram random",
4873         "1.00 Armenian dram random",
4874         "1.00 Armenian drams random",
4875         "1.00 Aruban Florin random",
4876         "1.00 Aruban florin random",
4877         "1.00 Australian Dollar random",
4878         "1.00 Australian dollar random",
4879         "1.00 Australian dollars random",
4880         "1.00 Austrian Schilling random",
4881         "1.00 Austrian schilling random",
4882         "1.00 Austrian schillings random",
4883         "1.00 Azerbaijani Manat (1993\\u20132006) random",
4884         "1.00 Azerbaijani Manat random",
4885         "1.00 Azerbaijani manat (1993\\u20132006) random",
4886         "1.00 Azerbaijani manat random",
4887         "1.00 Azerbaijani manats (1993\\u20132006) random",
4888         "1.00 Azerbaijani manats random",
4889         "1.00 Bahamian Dollar random",
4890         "1.00 Bahamian dollar random",
4891         "1.00 Bahamian dollars random",
4892         "1.00 Bahraini Dinar random",
4893         "1.00 Bahraini dinar random",
4894         "1.00 Bahraini dinars random",
4895         "1.00 Bangladeshi Taka random",
4896         "1.00 Bangladeshi taka random",
4897         "1.00 Bangladeshi takas random",
4898         "1.00 Barbadian Dollar random",
4899         "1.00 Barbadian dollar random",
4900         "1.00 Barbadian dollars random",
4901         "1.00 Belarusian New Ruble (1994\\u20131999) random",
4902         "1.00 Belarusian Ruble random",
4903         "1.00 Belarusian new ruble (1994\\u20131999) random",
4904         "1.00 Belarusian new rubles (1994\\u20131999) random",
4905         "1.00 Belarusian ruble random",
4906         "1.00 Belarusian rubles random",
4907         "1.00 Belgian Franc (convertible) random",
4908         "1.00 Belgian Franc (financial) random",
4909         "1.00 Belgian Franc random",
4910         "1.00 Belgian franc (convertible) random",
4911         "1.00 Belgian franc (financial) random",
4912         "1.00 Belgian franc random",
4913         "1.00 Belgian francs (convertible) random",
4914         "1.00 Belgian francs (financial) random",
4915         "1.00 Belgian francs random",
4916         "1.00 Belize Dollar random",
4917         "1.00 Belize dollar random",
4918         "1.00 Belize dollars random",
4919         "1.00 Bermudan Dollar random",
4920         "1.00 Bermudan dollar random",
4921         "1.00 Bermudan dollars random",
4922         "1.00 Bhutanese Ngultrum random",
4923         "1.00 Bhutanese ngultrum random",
4924         "1.00 Bhutanese ngultrums random",
4925         "1.00 Bolivian Mvdol random",
4926         "1.00 Bolivian Peso random",
4927         "1.00 Bolivian mvdol random",
4928         "1.00 Bolivian mvdols random",
4929         "1.00 Bolivian peso random",
4930         "1.00 Bolivian pesos random",
4931         "1.00 Bolivian Boliviano random",
4932         "1.00 Bolivian Boliviano random",
4933         "1.00 Bolivian Bolivianos random",
4934         "1.00 Bosnia-Herzegovina Convertible Mark random",
4935         "1.00 Bosnia-Herzegovina Dinar (1992\\u20131994) random",
4936         "1.00 Bosnia-Herzegovina convertible mark random",
4937         "1.00 Bosnia-Herzegovina convertible marks random",
4938         "1.00 Bosnia-Herzegovina dinar (1992\\u20131994) random",
4939         "1.00 Bosnia-Herzegovina dinars (1992\\u20131994) random",
4940         "1.00 Botswanan Pula random",
4941         "1.00 Botswanan pula random",
4942         "1.00 Botswanan pulas random",
4943         "1.00 Brazilian New Cruzado (1989\\u20131990) random",
4944         "1.00 Brazilian Cruzado (1986\\u20131989) random",
4945         "1.00 Brazilian Cruzeiro (1990\\u20131993) random",
4946         "1.00 Brazilian New Cruzeiro (1967\\u20131986) random",
4947         "1.00 Brazilian Cruzeiro (1993\\u20131994) random",
4948         "1.00 Brazilian Real random",
4949         "1.00 Brazilian new cruzado (1989\\u20131990) random",
4950         "1.00 Brazilian new cruzados (1989\\u20131990) random",
4951         "1.00 Brazilian cruzado (1986\\u20131989) random",
4952         "1.00 Brazilian cruzados (1986\\u20131989) random",
4953         "1.00 Brazilian cruzeiro (1990\\u20131993) random",
4954         "1.00 Brazilian new cruzeiro (1967\\u20131986) random",
4955         "1.00 Brazilian cruzeiro (1993\\u20131994) random",
4956         "1.00 Brazilian cruzeiros (1990\\u20131993) random",
4957         "1.00 Brazilian new cruzeiros (1967\\u20131986) random",
4958         "1.00 Brazilian cruzeiros (1993\\u20131994) random",
4959         "1.00 Brazilian real random",
4960         "1.00 Brazilian reals random",
4961         "1.00 British Pound random",
4962         "1.00 British pound random",
4963         "1.00 British pounds random",
4964         "1.00 Brunei Dollar random",
4965         "1.00 Brunei dollar random",
4966         "1.00 Brunei dollars random",
4967         "1.00 Bulgarian Hard Lev random",
4968         "1.00 Bulgarian Lev random",
4969         "1.00 Bulgarian Leva random",
4970         "1.00 Bulgarian hard lev random",
4971         "1.00 Bulgarian hard leva random",
4972         "1.00 Bulgarian lev random",
4973         "1.00 Burmese Kyat random",
4974         "1.00 Burmese kyat random",
4975         "1.00 Burmese kyats random",
4976         "1.00 Burundian Franc random",
4977         "1.00 Burundian franc random",
4978         "1.00 Burundian francs random",
4979         "1.00 Cambodian Riel random",
4980         "1.00 Cambodian riel random",
4981         "1.00 Cambodian riels random",
4982         "1.00 Canadian Dollar random",
4983         "1.00 Canadian dollar random",
4984         "1.00 Canadian dollars random",
4985         "1.00 Cape Verdean Escudo random",
4986         "1.00 Cape Verdean escudo random",
4987         "1.00 Cape Verdean escudos random",
4988         "1.00 Cayman Islands Dollar random",
4989         "1.00 Cayman Islands dollar random",
4990         "1.00 Cayman Islands dollars random",
4991         "1.00 Chilean Peso random",
4992         "1.00 Chilean Unit of Account (UF) random",
4993         "1.00 Chilean peso random",
4994         "1.00 Chilean pesos random",
4995         "1.00 Chilean unit of account (UF) random",
4996         "1.00 Chilean units of account (UF) random",
4997         "1.00 Chinese Yuan random",
4998         "1.00 Chinese yuan random",
4999         "1.00 Colombian Peso random",
5000         "1.00 Colombian peso random",
5001         "1.00 Colombian pesos random",
5002         "1.00 Comorian Franc random",
5003         "1.00 Comorian franc random",
5004         "1.00 Comorian francs random",
5005         "1.00 Congolese Franc Congolais random",
5006         "1.00 Congolese franc Congolais random",
5007         "1.00 Congolese francs Congolais random",
5008         "1.00 Costa Rican Col\\u00f3n random",
5009         "1.00 Costa Rican col\\u00f3n random",
5010         "1.00 Costa Rican col\\u00f3ns random",
5011         "1.00 Croatian Dinar random",
5012         "1.00 Croatian Kuna random",
5013         "1.00 Croatian dinar random",
5014         "1.00 Croatian dinars random",
5015         "1.00 Croatian kuna random",
5016         "1.00 Croatian kunas random",
5017         "1.00 Cuban Peso random",
5018         "1.00 Cuban peso random",
5019         "1.00 Cuban pesos random",
5020         "1.00 Cypriot Pound random",
5021         "1.00 Cypriot pound random",
5022         "1.00 Cypriot pounds random",
5023         "1.00 Czech Republic Koruna random",
5024         "1.00 Czech Republic koruna random",
5025         "1.00 Czech Republic korunas random",
5026         "1.00 Czechoslovak Hard Koruna random",
5027         "1.00 Czechoslovak hard koruna random",
5028         "1.00 Czechoslovak hard korunas random",
5029         "1.00 Danish Krone random",
5030         "1.00 Danish krone random",
5031         "1.00 Danish kroner random",
5032         "1.00 German Mark random",
5033         "1.00 German mark random",
5034         "1.00 German marks random",
5035         "1.00 Djiboutian Franc random",
5036         "1.00 Djiboutian franc random",
5037         "1.00 Djiboutian francs random",
5038         "1.00 Dominican Peso random",
5039         "1.00 Dominican peso random",
5040         "1.00 Dominican pesos random",
5041         "1.00 East Caribbean Dollar random",
5042         "1.00 East Caribbean dollar random",
5043         "1.00 East Caribbean dollars random",
5044         "1.00 East German Mark random",
5045         "1.00 East German mark random",
5046         "1.00 East German marks random",
5047         "1.00 Ecuadorian Sucre random",
5048         "1.00 Ecuadorian Unit of Constant Value random",
5049         "1.00 Ecuadorian sucre random",
5050         "1.00 Ecuadorian sucres random",
5051         "1.00 Ecuadorian unit of constant value random",
5052         "1.00 Ecuadorian units of constant value random",
5053         "1.00 Egyptian Pound random",
5054         "1.00 Egyptian pound random",
5055         "1.00 Egyptian pounds random",
5056         "1.00 Salvadoran Col\\u00f3n random",
5057         "1.00 Salvadoran col\\u00f3n random",
5058         "1.00 Salvadoran colones random",
5059         "1.00 Equatorial Guinean Ekwele random",
5060         "1.00 Equatorial Guinean ekwele random",
5061         "1.00 Eritrean Nakfa random",
5062         "1.00 Eritrean nakfa random",
5063         "1.00 Eritrean nakfas random",
5064         "1.00 Estonian Kroon random",
5065         "1.00 Estonian kroon random",
5066         "1.00 Estonian kroons random",
5067         "1.00 Ethiopian Birr random",
5068         "1.00 Ethiopian birr random",
5069         "1.00 Ethiopian birrs random",
5070         "1.00 European Composite Unit random",
5071         "1.00 European Currency Unit random",
5072         "1.00 European Monetary Unit random",
5073         "1.00 European Unit of Account (XBC) random",
5074         "1.00 European Unit of Account (XBD) random",
5075         "1.00 European composite unit random",
5076         "1.00 European composite units random",
5077         "1.00 European currency unit random",
5078         "1.00 European currency units random",
5079         "1.00 European monetary unit random",
5080         "1.00 European monetary units random",
5081         "1.00 European unit of account (XBC) random",
5082         "1.00 European unit of account (XBD) random",
5083         "1.00 European units of account (XBC) random",
5084         "1.00 European units of account (XBD) random",
5085         "1.00 Falkland Islands Pound random",
5086         "1.00 Falkland Islands pound random",
5087         "1.00 Falkland Islands pounds random",
5088         "1.00 Fijian Dollar random",
5089         "1.00 Fijian dollar random",
5090         "1.00 Fijian dollars random",
5091         "1.00 Finnish Markka random",
5092         "1.00 Finnish markka random",
5093         "1.00 Finnish markkas random",
5094         "1.00 French Franc random",
5095         "1.00 French Gold Franc random",
5096         "1.00 French UIC-Franc random",
5097         "1.00 French UIC-franc random",
5098         "1.00 French UIC-francs random",
5099         "1.00 French franc random",
5100         "1.00 French francs random",
5101         "1.00 French gold franc random",
5102         "1.00 French gold francs random",
5103         "1.00 Gambian Dalasi random",
5104         "1.00 Gambian dalasi random",
5105         "1.00 Gambian dalasis random",
5106         "1.00 Georgian Kupon Larit random",
5107         "1.00 Georgian Lari random",
5108         "1.00 Georgian kupon larit random",
5109         "1.00 Georgian kupon larits random",
5110         "1.00 Georgian lari random",
5111         "1.00 Georgian laris random",
5112         "1.00 Ghanaian Cedi (1979\\u20132007) random",
5113         "1.00 Ghanaian Cedi random",
5114         "1.00 Ghanaian cedi (1979\\u20132007) random",
5115         "1.00 Ghanaian cedi random",
5116         "1.00 Ghanaian cedis (1979\\u20132007) random",
5117         "1.00 Ghanaian cedis random",
5118         "1.00 Gibraltar Pound random",
5119         "1.00 Gibraltar pound random",
5120         "1.00 Gibraltar pounds random",
5121         "1.00 Gold random",
5122         "1.00 Gold random",
5123         "1.00 Greek Drachma random",
5124         "1.00 Greek drachma random",
5125         "1.00 Greek drachmas random",
5126         "1.00 Guatemalan Quetzal random",
5127         "1.00 Guatemalan quetzal random",
5128         "1.00 Guatemalan quetzals random",
5129         "1.00 Guinean Franc random",
5130         "1.00 Guinean Syli random",
5131         "1.00 Guinean franc random",
5132         "1.00 Guinean francs random",
5133         "1.00 Guinean syli random",
5134         "1.00 Guinean sylis random",
5135         "1.00 Guinea-Bissau Peso random",
5136         "1.00 Guinea-Bissau peso random",
5137         "1.00 Guinea-Bissau pesos random",
5138         "1.00 Guyanaese Dollar random",
5139         "1.00 Guyanaese dollar random",
5140         "1.00 Guyanaese dollars random",
5141         "1.00 Haitian Gourde random",
5142         "1.00 Haitian gourde random",
5143         "1.00 Haitian gourdes random",
5144         "1.00 Honduran Lempira random",
5145         "1.00 Honduran lempira random",
5146         "1.00 Honduran lempiras random",
5147         "1.00 Hong Kong Dollar random",
5148         "1.00 Hong Kong dollar random",
5149         "1.00 Hong Kong dollars random",
5150         "1.00 Hungarian Forint random",
5151         "1.00 Hungarian forint random",
5152         "1.00 Hungarian forints random",
5153         "1.00 Icelandic Kr\\u00f3na random",
5154         "1.00 Icelandic kr\\u00f3na random",
5155         "1.00 Icelandic kr\\u00f3nur random",
5156         "1.00 Indian Rupee random",
5157         "1.00 Indian rupee random",
5158         "1.00 Indian rupees random",
5159         "1.00 Indonesian Rupiah random",
5160         "1.00 Indonesian rupiah random",
5161         "1.00 Indonesian rupiahs random",
5162         "1.00 Iranian Rial random",
5163         "1.00 Iranian rial random",
5164         "1.00 Iranian rials random",
5165         "1.00 Iraqi Dinar random",
5166         "1.00 Iraqi dinar random",
5167         "1.00 Iraqi dinars random",
5168         "1.00 Irish Pound random",
5169         "1.00 Irish pound random",
5170         "1.00 Irish pounds random",
5171         "1.00 Israeli Pound random",
5172         "1.00 Israeli new sheqel random",
5173         "1.00 Israeli pound random",
5174         "1.00 Israeli pounds random",
5175         "1.00 Italian Lira random",
5176         "1.00 Italian lira random",
5177         "1.00 Italian liras random",
5178         "1.00 Jamaican Dollar random",
5179         "1.00 Jamaican dollar random",
5180         "1.00 Jamaican dollars random",
5181         "1.00 Japanese Yen random",
5182         "1.00 Japanese yen random",
5183         "1.00 Jordanian Dinar random",
5184         "1.00 Jordanian dinar random",
5185         "1.00 Jordanian dinars random",
5186         "1.00 Kazakhstani Tenge random",
5187         "1.00 Kazakhstani tenge random",
5188         "1.00 Kazakhstani tenges random",
5189         "1.00 Kenyan Shilling random",
5190         "1.00 Kenyan shilling random",
5191         "1.00 Kenyan shillings random",
5192         "1.00 Kuwaiti Dinar random",
5193         "1.00 Kuwaiti dinar random",
5194         "1.00 Kuwaiti dinars random",
5195         "1.00 Kyrgystani Som random",
5196         "1.00 Kyrgystani som random",
5197         "1.00 Kyrgystani soms random",
5198         "1.00 Laotian Kip random",
5199         "1.00 Laotian kip random",
5200         "1.00 Laotian kips random",
5201         "1.00 Latvian Lats random",
5202         "1.00 Latvian Ruble random",
5203         "1.00 Latvian lats random",
5204         "1.00 Latvian lati random",
5205         "1.00 Latvian ruble random",
5206         "1.00 Latvian rubles random",
5207         "1.00 Lebanese Pound random",
5208         "1.00 Lebanese pound random",
5209         "1.00 Lebanese pounds random",
5210         "1.00 Lesotho Loti random",
5211         "1.00 Lesotho loti random",
5212         "1.00 Lesotho lotis random",
5213         "1.00 Liberian Dollar random",
5214         "1.00 Liberian dollar random",
5215         "1.00 Liberian dollars random",
5216         "1.00 Libyan Dinar random",
5217         "1.00 Libyan dinar random",
5218         "1.00 Libyan dinars random",
5219         "1.00 Lithuanian Litas random",
5220         "1.00 Lithuanian Talonas random",
5221         "1.00 Lithuanian litas random",
5222         "1.00 Lithuanian litai random",
5223         "1.00 Lithuanian talonas random",
5224         "1.00 Lithuanian talonases random",
5225         "1.00 Luxembourgian Convertible Franc random",
5226         "1.00 Luxembourg Financial Franc random",
5227         "1.00 Luxembourgian Franc random",
5228         "1.00 Luxembourgian convertible franc random",
5229         "1.00 Luxembourgian convertible francs random",
5230         "1.00 Luxembourg financial franc random",
5231         "1.00 Luxembourg financial francs random",
5232         "1.00 Luxembourgian franc random",
5233         "1.00 Luxembourgian francs random",
5234         "1.00 Macanese Pataca random",
5235         "1.00 Macanese pataca random",
5236         "1.00 Macanese patacas random",
5237         "1.00 Macedonian Denar random",
5238         "1.00 Macedonian denar random",
5239         "1.00 Macedonian denari random",
5240         "1.00 Malagasy Ariaries random",
5241         "1.00 Malagasy Ariary random",
5242         "1.00 Malagasy Ariary random",
5243         "1.00 Malagasy Franc random",
5244         "1.00 Malagasy franc random",
5245         "1.00 Malagasy francs random",
5246         "1.00 Malawian Kwacha random",
5247         "1.00 Malawian Kwacha random",
5248         "1.00 Malawian Kwachas random",
5249         "1.00 Malaysian Ringgit random",
5250         "1.00 Malaysian ringgit random",
5251         "1.00 Malaysian ringgits random",
5252         "1.00 Maldivian Rufiyaa random",
5253         "1.00 Maldivian rufiyaa random",
5254         "1.00 Maldivian rufiyaas random",
5255         "1.00 Malian Franc random",
5256         "1.00 Malian franc random",
5257         "1.00 Malian francs random",
5258         "1.00 Maltese Lira random",
5259         "1.00 Maltese Pound random",
5260         "1.00 Maltese lira random",
5261         "1.00 Maltese liras random",
5262         "1.00 Maltese pound random",
5263         "1.00 Maltese pounds random",
5264         "1.00 Mauritanian Ouguiya random",
5265         "1.00 Mauritanian ouguiya random",
5266         "1.00 Mauritanian ouguiyas random",
5267         "1.00 Mauritian Rupee random",
5268         "1.00 Mauritian rupee random",
5269         "1.00 Mauritian rupees random",
5270         "1.00 Mexican Peso random",
5271         "1.00 Mexican Silver Peso (1861\\u20131992) random",
5272         "1.00 Mexican Investment Unit random",
5273         "1.00 Mexican peso random",
5274         "1.00 Mexican pesos random",
5275         "1.00 Mexican silver peso (1861\\u20131992) random",
5276         "1.00 Mexican silver pesos (1861\\u20131992) random",
5277         "1.00 Mexican investment unit random",
5278         "1.00 Mexican investment units random",
5279         "1.00 Moldovan Leu random",
5280         "1.00 Moldovan leu random",
5281         "1.00 Moldovan lei random",
5282         "1.00 Mongolian Tugrik random",
5283         "1.00 Mongolian tugrik random",
5284         "1.00 Mongolian tugriks random",
5285         "1.00 Moroccan Dirham random",
5286         "1.00 Moroccan Franc random",
5287         "1.00 Moroccan dirham random",
5288         "1.00 Moroccan dirhams random",
5289         "1.00 Moroccan franc random",
5290         "1.00 Moroccan francs random",
5291         "1.00 Mozambican Escudo random",
5292         "1.00 Mozambican Metical random",
5293         "1.00 Mozambican escudo random",
5294         "1.00 Mozambican escudos random",
5295         "1.00 Mozambican metical random",
5296         "1.00 Mozambican meticals random",
5297         "1.00 Myanmar Kyat random",
5298         "1.00 Myanmar kyat random",
5299         "1.00 Myanmar kyats random",
5300         "1.00 Namibian Dollar random",
5301         "1.00 Namibian dollar random",
5302         "1.00 Namibian dollars random",
5303         "1.00 Nepalese Rupee random",
5304         "1.00 Nepalese rupee random",
5305         "1.00 Nepalese rupees random",
5306         "1.00 Netherlands Antillean Guilder random",
5307         "1.00 Netherlands Antillean guilder random",
5308         "1.00 Netherlands Antillean guilders random",
5309         "1.00 Dutch Guilder random",
5310         "1.00 Dutch guilder random",
5311         "1.00 Dutch guilders random",
5312         "1.00 Israeli New Sheqel random",
5313         "1.00 Israeli new sheqels random",
5314         "1.00 New Zealand Dollar random",
5315         "1.00 New Zealand dollar random",
5316         "1.00 New Zealand dollars random",
5317         "1.00 Nicaraguan C\\u00f3rdoba random",
5318         "1.00 Nicaraguan C\\u00f3rdoba (1988\\u20131991) random",
5319         "1.00 Nicaraguan c\\u00f3rdoba random",
5320         "1.00 Nicaraguan c\\u00f3rdoba random",
5321         "1.00 Nicaraguan c\\u00f3rdoba (1988\\u20131991) random",
5322         "1.00 Nicaraguan c\\u00f3rdobas (1988\\u20131991) random",
5323         "1.00 Nigerian Naira random",
5324         "1.00 Nigerian naira random",
5325         "1.00 Nigerian nairas random",
5326         "1.00 North Korean Won random",
5327         "1.00 North Korean won random",
5328         "1.00 North Korean won random",
5329         "1.00 Norwegian Krone random",
5330         "1.00 Norwegian krone random",
5331         "1.00 Norwegian kroner random",
5332         "1.00 Mozambican Metical (1980\\u20132006) random",
5333         "1.00 Mozambican metical (1980\\u20132006) random",
5334         "1.00 Mozambican meticals (1980\\u20132006) random",
5335         "1.00 Romanian Lei (1952\\u20132006) random",
5336         "1.00 Romanian Leu (1952\\u20132006) random",
5337         "1.00 Romanian leu (1952\\u20132006) random",
5338         "1.00 Serbian Dinar (2002\\u20132006) random",
5339         "1.00 Serbian dinar (2002\\u20132006) random",
5340         "1.00 Serbian dinars (2002\\u20132006) random",
5341         "1.00 Sudanese Dinar (1992\\u20132007) random",
5342         "1.00 Sudanese Pound (1957\\u20131998) random",
5343         "1.00 Sudanese dinar (1992\\u20132007) random",
5344         "1.00 Sudanese dinars (1992\\u20132007) random",
5345         "1.00 Sudanese pound (1957\\u20131998) random",
5346         "1.00 Sudanese pounds (1957\\u20131998) random",
5347         "1.00 Turkish Lira (1922\\u20132005) random",
5348         "1.00 Turkish Lira (1922\\u20132005) random",
5349         "1.00 Omani Rial random",
5350         "1.00 Omani rial random",
5351         "1.00 Omani rials random",
5352         "1.00 Pakistani Rupee random",
5353         "1.00 Pakistani rupee random",
5354         "1.00 Pakistani rupees random",
5355         "1.00 Palladium random",
5356         "1.00 Palladium random",
5357         "1.00 Panamanian Balboa random",
5358         "1.00 Panamanian balboa random",
5359         "1.00 Panamanian balboas random",
5360         "1.00 Papua New Guinean Kina random",
5361         "1.00 Papua New Guinean kina random",
5362         "1.00 Papua New Guinean kina random",
5363         "1.00 Paraguayan Guarani random",
5364         "1.00 Paraguayan guarani random",
5365         "1.00 Paraguayan guaranis random",
5366         "1.00 Peruvian Inti random",
5367         "1.00 Peruvian Nuevo Sol random",
5368         "1.00 Peruvian Sol (1863\\u20131965) random",
5369         "1.00 Peruvian inti random",
5370         "1.00 Peruvian intis random",
5371         "1.00 Peruvian nuevo sol random",
5372         "1.00 Peruvian nuevos soles random",
5373         "1.00 Peruvian sol (1863\\u20131965) random",
5374         "1.00 Peruvian soles (1863\\u20131965) random",
5375         "1.00 Philippine Peso random",
5376         "1.00 Philippine peso random",
5377         "1.00 Philippine pesos random",
5378         "1.00 Platinum random",
5379         "1.00 Platinum random",
5380         "1.00 Polish Zloty (1950\\u20131995) random",
5381         "1.00 Polish Zloty random",
5382         "1.00 Polish zlotys random",
5383         "1.00 Polish zloty (PLZ) random",
5384         "1.00 Polish zloty random",
5385         "1.00 Polish zlotys (PLZ) random",
5386         "1.00 Portuguese Escudo random",
5387         "1.00 Portuguese Guinea Escudo random",
5388         "1.00 Portuguese Guinea escudo random",
5389         "1.00 Portuguese Guinea escudos random",
5390         "1.00 Portuguese escudo random",
5391         "1.00 Portuguese escudos random",
5392         "1.00 Qatari Rial random",
5393         "1.00 Qatari rial random",
5394         "1.00 Qatari rials random",
5395         "1.00 RINET Funds random",
5396         "1.00 RINET Funds random",
5397         "1.00 Rhodesian Dollar random",
5398         "1.00 Rhodesian dollar random",
5399         "1.00 Rhodesian dollars random",
5400         "1.00 Romanian Leu random",
5401         "1.00 Romanian lei random",
5402         "1.00 Romanian leu random",
5403         "1.00 Russian Ruble (1991\\u20131998) random",
5404         "1.00 Russian Ruble random",
5405         "1.00 Russian ruble (1991\\u20131998) random",
5406         "1.00 Russian ruble random",
5407         "1.00 Russian rubles (1991\\u20131998) random",
5408         "1.00 Russian rubles random",
5409         "1.00 Rwandan Franc random",
5410         "1.00 Rwandan franc random",
5411         "1.00 Rwandan francs random",
5412         "1.00 St. Helena Pound random",
5413         "1.00 St. Helena pound random",
5414         "1.00 St. Helena pounds random",
5415         "1.00 S\\u00e3o Tom\\u00e9 & Pr\\u00edncipe Dobra random",
5416         "1.00 S\\u00e3o Tom\\u00e9 & Pr\\u00edncipe dobra random",
5417         "1.00 S\\u00e3o Tom\\u00e9 & Pr\\u00edncipe dobras random",
5418         "1.00 Saudi Riyal random",
5419         "1.00 Saudi riyal random",
5420         "1.00 Saudi riyals random",
5421         "1.00 Serbian Dinar random",
5422         "1.00 Serbian dinar random",
5423         "1.00 Serbian dinars random",
5424         "1.00 Seychellois Rupee random",
5425         "1.00 Seychellois rupee random",
5426         "1.00 Seychellois rupees random",
5427         "1.00 Sierra Leonean Leone random",
5428         "1.00 Sierra Leonean leone random",
5429         "1.00 Sierra Leonean leones random",
5430         "1.00 Singapore Dollar random",
5431         "1.00 Singapore dollar random",
5432         "1.00 Singapore dollars random",
5433         "1.00 Slovak Koruna random",
5434         "1.00 Slovak koruna random",
5435         "1.00 Slovak korunas random",
5436         "1.00 Slovenian Tolar random",
5437         "1.00 Slovenian tolar random",
5438         "1.00 Slovenian tolars random",
5439         "1.00 Solomon Islands Dollar random",
5440         "1.00 Solomon Islands dollar random",
5441         "1.00 Solomon Islands dollars random",
5442         "1.00 Somali Shilling random",
5443         "1.00 Somali shilling random",
5444         "1.00 Somali shillings random",
5445         "1.00 South African Rand (financial) random",
5446         "1.00 South African Rand random",
5447         "1.00 South African rand (financial) random",
5448         "1.00 South African rand random",
5449         "1.00 South African rands (financial) random",
5450         "1.00 South African rand random",
5451         "1.00 South Korean Won random",
5452         "1.00 South Korean won random",
5453         "1.00 South Korean won random",
5454         "1.00 Soviet Rouble random",
5455         "1.00 Soviet rouble random",
5456         "1.00 Soviet roubles random",
5457         "1.00 Spanish Peseta (A account) random",
5458         "1.00 Spanish Peseta (convertible account) random",
5459         "1.00 Spanish Peseta random",
5460         "1.00 Spanish peseta (A account) random",
5461         "1.00 Spanish peseta (convertible account) random",
5462         "1.00 Spanish peseta random",
5463         "1.00 Spanish pesetas (A account) random",
5464         "1.00 Spanish pesetas (convertible account) random",
5465         "1.00 Spanish pesetas random",
5466         "1.00 Special Drawing Rights random",
5467         "1.00 Sri Lankan Rupee random",
5468         "1.00 Sri Lankan rupee random",
5469         "1.00 Sri Lankan rupees random",
5470         "1.00 Sudanese Pound random",
5471         "1.00 Sudanese pound random",
5472         "1.00 Sudanese pounds random",
5473         "1.00 Surinamese Dollar random",
5474         "1.00 Surinamese dollar random",
5475         "1.00 Surinamese dollars random",
5476         "1.00 Surinamese Guilder random",
5477         "1.00 Surinamese guilder random",
5478         "1.00 Surinamese guilders random",
5479         "1.00 Swazi Lilangeni random",
5480         "1.00 Swazi lilangeni random",
5481         "1.00 Swazi emalangeni random",
5482         "1.00 Swedish Krona random",
5483         "1.00 Swedish krona random",
5484         "1.00 Swedish kronor random",
5485         "1.00 Swiss Franc random",
5486         "1.00 Swiss franc random",
5487         "1.00 Swiss francs random",
5488         "1.00 Syrian Pound random",
5489         "1.00 Syrian pound random",
5490         "1.00 Syrian pounds random",
5491         "1.00 New Taiwan Dollar random",
5492         "1.00 New Taiwan dollar random",
5493         "1.00 New Taiwan dollars random",
5494         "1.00 Tajikistani Ruble random",
5495         "1.00 Tajikistani Somoni random",
5496         "1.00 Tajikistani ruble random",
5497         "1.00 Tajikistani rubles random",
5498         "1.00 Tajikistani somoni random",
5499         "1.00 Tajikistani somonis random",
5500         "1.00 Tanzanian Shilling random",
5501         "1.00 Tanzanian shilling random",
5502         "1.00 Tanzanian shillings random",
5503         "1.00 Testing Currency Code random",
5504         "1.00 Testing Currency Code random",
5505         "1.00 Thai Baht random",
5506         "1.00 Thai baht random",
5507         "1.00 Thai baht random",
5508         "1.00 Timorese Escudo random",
5509         "1.00 Timorese escudo random",
5510         "1.00 Timorese escudos random",
5511         "1.00 Trinidad & Tobago Dollar random",
5512         "1.00 Trinidad & Tobago dollar random",
5513         "1.00 Trinidad & Tobago dollars random",
5514         "1.00 Tunisian Dinar random",
5515         "1.00 Tunisian dinar random",
5516         "1.00 Tunisian dinars random",
5517         "1.00 Turkish Lira random",
5518         "1.00 Turkish Lira random",
5519         "1.00 Turkish lira random",
5520         "1.00 Turkmenistani Manat random",
5521         "1.00 Turkmenistani manat random",
5522         "1.00 Turkmenistani manat random",
5523         "1.00 US Dollar (Next day) random",
5524         "1.00 US Dollar (Same day) random",
5525         "1.00 US Dollar random",
5526         "1.00 US dollar (next day) random",
5527         "1.00 US dollar (same day) random",
5528         "1.00 US dollar random",
5529         "1.00 US dollars (next day) random",
5530         "1.00 US dollars (same day) random",
5531         "1.00 US dollars random",
5532         "1.00 Ugandan Shilling (1966\\u20131987) random",
5533         "1.00 Ugandan Shilling random",
5534         "1.00 Ugandan shilling (1966\\u20131987) random",
5535         "1.00 Ugandan shilling random",
5536         "1.00 Ugandan shillings (1966\\u20131987) random",
5537         "1.00 Ugandan shillings random",
5538         "1.00 Ukrainian Hryvnia random",
5539         "1.00 Ukrainian Karbovanets random",
5540         "1.00 Ukrainian hryvnia random",
5541         "1.00 Ukrainian hryvnias random",
5542         "1.00 Ukrainian karbovanets random",
5543         "1.00 Ukrainian karbovantsiv random",
5544         "1.00 Colombian Real Value Unit random",
5545         "1.00 United Arab Emirates Dirham random",
5546         "1.00 Unknown Currency random",
5547         "1.00 Uruguayan Peso (1975\\u20131993) random",
5548         "1.00 Uruguayan Peso random",
5549         "1.00 Uruguayan Peso (Indexed Units) random",
5550         "1.00 Uruguayan peso (1975\\u20131993) random",
5551         "1.00 Uruguayan peso (indexed units) random",
5552         "1.00 Uruguayan peso random",
5553         "1.00 Uruguayan pesos (1975\\u20131993) random",
5554         "1.00 Uruguayan pesos (indexed units) random",
5555         "1.00 Uzbekistani Som random",
5556         "1.00 Uzbekistani som random",
5557         "1.00 Uzbekistani som random",
5558         "1.00 Vanuatu Vatu random",
5559         "1.00 Vanuatu vatu random",
5560         "1.00 Vanuatu vatus random",
5561         "1.00 Venezuelan Bol\\u00edvar random",
5562         "1.00 Venezuelan Bol\\u00edvar (1871\\u20132008) random",
5563         "1.00 Venezuelan bol\\u00edvar random",
5564         "1.00 Venezuelan bol\\u00edvars random",
5565         "1.00 Venezuelan bol\\u00edvar (1871\\u20132008) random",
5566         "1.00 Venezuelan bol\\u00edvars (1871\\u20132008) random",
5567         "1.00 Vietnamese Dong random",
5568         "1.00 Vietnamese dong random",
5569         "1.00 Vietnamese dong random",
5570         "1.00 WIR Euro random",
5571         "1.00 WIR Franc random",
5572         "1.00 WIR euro random",
5573         "1.00 WIR euros random",
5574         "1.00 WIR franc random",
5575         "1.00 WIR francs random",
5576         "1.00 Samoan Tala random",
5577         "1.00 Samoan tala random",
5578         "1.00 Samoan tala random",
5579         "1.00 Yemeni Dinar random",
5580         "1.00 Yemeni Rial random",
5581         "1.00 Yemeni dinar random",
5582         "1.00 Yemeni dinars random",
5583         "1.00 Yemeni rial random",
5584         "1.00 Yemeni rials random",
5585         "1.00 Yugoslavian Convertible Dinar (1990\\u20131992) random",
5586         "1.00 Yugoslavian Hard Dinar (1966\\u20131990) random",
5587         "1.00 Yugoslavian New Dinar (1994\\u20132002) random",
5588         "1.00 Yugoslavian convertible dinar (1990\\u20131992) random",
5589         "1.00 Yugoslavian convertible dinars (1990\\u20131992) random",
5590         "1.00 Yugoslavian hard dinar (1966\\u20131990) random",
5591         "1.00 Yugoslavian hard dinars (1966\\u20131990) random",
5592         "1.00 Yugoslavian new dinar (1994\\u20132002) random",
5593         "1.00 Yugoslavian new dinars (1994\\u20132002) random",
5594         "1.00 Zairean New Zaire (1993\\u20131998) random",
5595         "1.00 Zairean Zaire (1971\\u20131993) random",
5596         "1.00 Zairean new zaire (1993\\u20131998) random",
5597         "1.00 Zairean new zaires (1993\\u20131998) random",
5598         "1.00 Zairean zaire (1971\\u20131993) random",
5599         "1.00 Zairean zaires (1971\\u20131993) random",
5600         "1.00 Zambian Kwacha random",
5601         "1.00 Zambian kwacha random",
5602         "1.00 Zambian kwachas random",
5603         "1.00 Zimbabwean Dollar (1980\\u20132008) random",
5604         "1.00 Zimbabwean dollar (1980\\u20132008) random",
5605         "1.00 Zimbabwean dollars (1980\\u20132008) random",
5606         "1.00 euro random",
5607         "1.00 euros random",
5608         "1.00 Turkish lira (1922\\u20132005) random",
5609         "1.00 special drawing rights random",
5610         "1.00 Colombian real value unit random",
5611         "1.00 Colombian real value units random",
5612         "1.00 unknown currency random",
5613     };
5614 
5615     const char* WRONG_DATA[] = {
5616         // Following are missing one last char in the currency name
5617         "usd1.00", // case sensitive
5618         "1.00 Nicaraguan Cordob",
5619         "1.00 Namibian Dolla",
5620         "1.00 Namibian dolla",
5621         "1.00 Nepalese Rupe",
5622         "1.00 Nepalese rupe",
5623         "1.00 Netherlands Antillean Guilde",
5624         "1.00 Netherlands Antillean guilde",
5625         "1.00 Dutch Guilde",
5626         "1.00 Dutch guilde",
5627         "1.00 Israeli New Sheqe",
5628         "1.00 New Zealand Dolla",
5629         "1.00 New Zealand dolla",
5630         "1.00 Nicaraguan cordob",
5631         "1.00 Nigerian Nair",
5632         "1.00 Nigerian nair",
5633         "1.00 North Korean Wo",
5634         "1.00 North Korean wo",
5635         "1.00 Norwegian Kron",
5636         "1.00 Norwegian kron",
5637         "1.00 US dolla",
5638         "1.00",
5639         "A1.00",
5640         "AD1.00",
5641         "AE1.00",
5642         "AF1.00",
5643         "AL1.00",
5644         "AM1.00",
5645         "AN1.00",
5646         "AO1.00",
5647         "AR1.00",
5648         "AT1.00",
5649         "AU1.00",
5650         "AW1.00",
5651         "AZ1.00",
5652         "Afghan Afghan1.00",
5653         "Afghan Afghani (1927\\u201320021.00",
5654         "Afl1.00",
5655         "Albanian Le1.00",
5656         "Algerian Dina1.00",
5657         "Andorran Peset1.00",
5658         "Angolan Kwanz1.00",
5659         "Angolan Kwanza (1977\\u201319901.00",
5660         "Angolan Readjusted Kwanza (1995\\u201319991.00",
5661         "Angolan New Kwanza (1990\\u201320001.00",
5662         "Argentine Austra1.00",
5663         "Argentine Pes1.00",
5664         "Argentine Peso (1983\\u201319851.00",
5665         "Armenian Dra1.00",
5666         "Aruban Flori1.00",
5667         "Australian Dolla1.00",
5668         "Austrian Schillin1.00",
5669         "Azerbaijani Mana1.00",
5670         "Azerbaijani Manat (1993\\u201320061.00",
5671         "B1.00",
5672         "BA1.00",
5673         "BB1.00",
5674         "BE1.00",
5675         "BG1.00",
5676         "BH1.00",
5677         "BI1.00",
5678         "BM1.00",
5679         "BN1.00",
5680         "BO1.00",
5681         "BR1.00",
5682         "BS1.00",
5683         "BT1.00",
5684         "BU1.00",
5685         "BW1.00",
5686         "BY1.00",
5687         "BZ1.00",
5688         "Bahamian Dolla1.00",
5689         "Bahraini Dina1.00",
5690         "Bangladeshi Tak1.00",
5691         "Barbadian Dolla1.00",
5692         "Bds1.00",
5693         "Belarusian New Ruble (1994\\u201319991.00",
5694         "Belarusian Rubl1.00",
5695         "Belgian Fran1.00",
5696         "Belgian Franc (convertible1.00",
5697         "Belgian Franc (financial1.00",
5698         "Belize Dolla1.00",
5699         "Bermudan Dolla1.00",
5700         "Bhutanese Ngultru1.00",
5701         "Bolivian Mvdo1.00",
5702         "Bolivian Pes1.00",
5703         "Bolivian Bolivian1.00",
5704         "Bosnia-Herzegovina Convertible Mar1.00",
5705         "Bosnia-Herzegovina Dina1.00",
5706         "Botswanan Pul1.00",
5707         "Brazilian Cruzad1.00",
5708         "Brazilian Cruzado Nov1.00",
5709         "Brazilian Cruzeir1.00",
5710         "Brazilian Cruzeiro (1990\\u201319931.00",
5711         "Brazilian New Cruzeiro (1967\\u201319861.00",
5712         "Brazilian Rea1.00",
5713         "British Pound Sterlin1.00",
5714         "Brunei Dolla1.00",
5715         "Bulgarian Hard Le1.00",
5716         "Bulgarian Le1.00",
5717         "Burmese Kya1.00",
5718         "Burundian Fran1.00",
5719         "C1.00",
5720         "CA1.00",
5721         "CD1.00",
5722         "CFP Fran1.00",
5723         "CFP1.00",
5724         "CH1.00",
5725         "CL1.00",
5726         "CN1.00",
5727         "CO1.00",
5728         "CS1.00",
5729         "CU1.00",
5730         "CV1.00",
5731         "CY1.00",
5732         "CZ1.00",
5733         "Cambodian Rie1.00",
5734         "Canadian Dolla1.00",
5735         "Cape Verdean Escud1.00",
5736         "Cayman Islands Dolla1.00",
5737         "Chilean Pes1.00",
5738         "Chilean Unit of Accoun1.00",
5739         "Chinese Yua1.00",
5740         "Colombian Pes1.00",
5741         "Comoro Fran1.00",
5742         "Congolese Fran1.00",
5743         "Costa Rican Col\\u00f31.00",
5744         "Croatian Dina1.00",
5745         "Croatian Kun1.00",
5746         "Cuban Pes1.00",
5747         "Cypriot Poun1.00",
5748         "Czech Republic Korun1.00",
5749         "Czechoslovak Hard Korun1.00",
5750         "D1.00",
5751         "DD1.00",
5752         "DE1.00",
5753         "DJ1.00",
5754         "DK1.00",
5755         "DO1.00",
5756         "DZ1.00",
5757         "Danish Kron1.00",
5758         "German Mar1.00",
5759         "Djiboutian Fran1.00",
5760         "Dk1.00",
5761         "Dominican Pes1.00",
5762         "EC1.00",
5763         "EE1.00",
5764         "EG1.00",
5765         "EQ1.00",
5766         "ER1.00",
5767         "ES1.00",
5768         "ET1.00",
5769         "EU1.00",
5770         "East Caribbean Dolla1.00",
5771         "East German Ostmar1.00",
5772         "Ecuadorian Sucr1.00",
5773         "Ecuadorian Unit of Constant Valu1.00",
5774         "Egyptian Poun1.00",
5775         "Ekwel1.00",
5776         "Salvadoran Col\\u00f31.00",
5777         "Equatorial Guinean Ekwel1.00",
5778         "Eritrean Nakf1.00",
5779         "Es1.00",
5780         "Estonian Kroo1.00",
5781         "Ethiopian Bir1.00",
5782         "Eur1.00",
5783         "European Composite Uni1.00",
5784         "European Currency Uni1.00",
5785         "European Monetary Uni1.00",
5786         "European Unit of Account (XBC1.00",
5787         "European Unit of Account (XBD1.00",
5788         "F1.00",
5789         "FB1.00",
5790         "FI1.00",
5791         "FJ1.00",
5792         "FK1.00",
5793         "FR1.00",
5794         "Falkland Islands Poun1.00",
5795         "Fd1.00",
5796         "Fijian Dolla1.00",
5797         "Finnish Markk1.00",
5798         "Fr1.00",
5799         "French Fran1.00",
5800         "French Gold Fran1.00",
5801         "French UIC-Fran1.00",
5802         "G1.00",
5803         "GB1.00",
5804         "GE1.00",
5805         "GH1.00",
5806         "GI1.00",
5807         "GM1.00",
5808         "GN1.00",
5809         "GQ1.00",
5810         "GR1.00",
5811         "GT1.00",
5812         "GW1.00",
5813         "GY1.00",
5814         "Gambian Dalas1.00",
5815         "Georgian Kupon Lari1.00",
5816         "Georgian Lar1.00",
5817         "Ghanaian Ced1.00",
5818         "Ghanaian Cedi (1979\\u201320071.00",
5819         "Gibraltar Poun1.00",
5820         "Gol1.00",
5821         "Greek Drachm1.00",
5822         "Guatemalan Quetza1.00",
5823         "Guinean Fran1.00",
5824         "Guinean Syl1.00",
5825         "Guinea-Bissau Pes1.00",
5826         "Guyanaese Dolla1.00",
5827         "HK1.00",
5828         "HN1.00",
5829         "HR1.00",
5830         "HT1.00",
5831         "HU1.00",
5832         "Haitian Gourd1.00",
5833         "Honduran Lempir1.00",
5834         "Hong Kong Dolla1.00",
5835         "Hungarian Forin1.00",
5836         "I1.00",
5837         "IE1.00",
5838         "IL1.00",
5839         "IN1.00",
5840         "IQ1.00",
5841         "IR1.00",
5842         "IS1.00",
5843         "IT1.00",
5844         "Icelandic Kron1.00",
5845         "Indian Rupe1.00",
5846         "Indonesian Rupia1.00",
5847         "Iranian Ria1.00",
5848         "Iraqi Dina1.00",
5849         "Irish Poun1.00",
5850         "Israeli Poun1.00",
5851         "Italian Lir1.00",
5852         "J1.00",
5853         "JM1.00",
5854         "JO1.00",
5855         "JP1.00",
5856         "Jamaican Dolla1.00",
5857         "Japanese Ye1.00",
5858         "Jordanian Dina1.00",
5859         "K S1.00",
5860         "K1.00",
5861         "KE1.00",
5862         "KG1.00",
5863         "KH1.00",
5864         "KP1.00",
5865         "KR1.00",
5866         "KW1.00",
5867         "KY1.00",
5868         "KZ1.00",
5869         "Kazakhstani Teng1.00",
5870         "Kenyan Shillin1.00",
5871         "Kuwaiti Dina1.00",
5872         "Kyrgystani So1.00",
5873         "LA1.00",
5874         "LB1.00",
5875         "LK1.00",
5876         "LR1.00",
5877         "LT1.00",
5878         "LU1.00",
5879         "LV1.00",
5880         "LY1.00",
5881         "Laotian Ki1.00",
5882         "Latvian Lat1.00",
5883         "Latvian Rubl1.00",
5884         "Lebanese Poun1.00",
5885         "Lesotho Lot1.00",
5886         "Liberian Dolla1.00",
5887         "Libyan Dina1.00",
5888         "Lithuanian Lit1.00",
5889         "Lithuanian Talona1.00",
5890         "Luxembourgian Convertible Fran1.00",
5891         "Luxembourg Financial Fran1.00",
5892         "Luxembourgian Fran1.00",
5893         "MA1.00",
5894         "MD1.00",
5895         "MDe1.00",
5896         "MEX1.00",
5897         "MG1.00",
5898         "ML1.00",
5899         "MM1.00",
5900         "MN1.00",
5901         "MO1.00",
5902         "MR1.00",
5903         "MT1.00",
5904         "MU1.00",
5905         "MV1.00",
5906         "MW1.00",
5907         "MX1.00",
5908         "MY1.00",
5909         "MZ1.00",
5910         "Macanese Patac1.00",
5911         "Macedonian Dena1.00",
5912         "Malagasy Ariar1.00",
5913         "Malagasy Fran1.00",
5914         "Malawian Kwach1.00",
5915         "Malaysian Ringgi1.00",
5916         "Maldivian Rufiya1.00",
5917         "Malian Fran1.00",
5918         "Malot1.00",
5919         "Maltese Lir1.00",
5920         "Maltese Poun1.00",
5921         "Mauritanian Ouguiy1.00",
5922         "Mauritian Rupe1.00",
5923         "Mexican Pes1.00",
5924         "Mexican Silver Peso (1861\\u201319921.00",
5925         "Mexican Investment Uni1.00",
5926         "Moldovan Le1.00",
5927         "Mongolian Tugri1.00",
5928         "Moroccan Dirha1.00",
5929         "Moroccan Fran1.00",
5930         "Mozambican Escud1.00",
5931         "Mozambican Metica1.00",
5932         "Myanmar Kya1.00",
5933         "N1.00",
5934         "NA1.00",
5935         "NAf1.00",
5936         "NG1.00",
5937         "NI1.00",
5938         "NK1.00",
5939         "NL1.00",
5940         "NO1.00",
5941         "NP1.00",
5942         "NT1.00",
5943         "Namibian Dolla1.00",
5944         "Nepalese Rupe1.00",
5945         "Netherlands Antillean Guilde1.00",
5946         "Dutch Guilde1.00",
5947         "Israeli New Sheqe1.00",
5948         "New Zealand Dolla1.00",
5949         "Nicaraguan C\\u00f3rdoba (1988\\u201319911.00",
5950         "Nicaraguan C\\u00f3rdob1.00",
5951         "Nigerian Nair1.00",
5952         "North Korean Wo1.00",
5953         "Norwegian Kron1.00",
5954         "Nr1.00",
5955         "OM1.00",
5956         "Old Mozambican Metica1.00",
5957         "Romanian Leu (1952\\u201320061.00",
5958         "Serbian Dinar (2002\\u201320061.00",
5959         "Sudanese Dinar (1992\\u201320071.00",
5960         "Sudanese Pound (1957\\u201319981.00",
5961         "Turkish Lira (1922\\u201320051.00",
5962         "Omani Ria1.00",
5963         "PA1.00",
5964         "PE1.00",
5965         "PG1.00",
5966         "PH1.00",
5967         "PK1.00",
5968         "PL1.00",
5969         "PT1.00",
5970         "PY1.00",
5971         "Pakistani Rupe1.00",
5972         "Palladiu1.00",
5973         "Panamanian Balbo1.00",
5974         "Papua New Guinean Kin1.00",
5975         "Paraguayan Guaran1.00",
5976         "Peruvian Int1.00",
5977         "Peruvian Sol (1863\\u201319651.00",
5978         "Peruvian Sol Nuev1.00",
5979         "Philippine Pes1.00",
5980         "Platinu1.00",
5981         "Polish Zlot1.00",
5982         "Polish Zloty (1950\\u201319951.00",
5983         "Portuguese Escud1.00",
5984         "Portuguese Guinea Escud1.00",
5985         "Pr1.00",
5986         "QA1.00",
5987         "Qatari Ria1.00",
5988         "RD1.00",
5989         "RH1.00",
5990         "RINET Fund1.00",
5991         "RS1.00",
5992         "RU1.00",
5993         "RW1.00",
5994         "Rb1.00",
5995         "Rhodesian Dolla1.00",
5996         "Romanian Le1.00",
5997         "Russian Rubl1.00",
5998         "Russian Ruble (1991\\u201319981.00",
5999         "Rwandan Fran1.00",
6000         "S1.00",
6001         "SA1.00",
6002         "SB1.00",
6003         "SC1.00",
6004         "SD1.00",
6005         "SE1.00",
6006         "SG1.00",
6007         "SH1.00",
6008         "SI1.00",
6009         "SK1.00",
6010         "SL R1.00",
6011         "SL1.00",
6012         "SO1.00",
6013         "ST1.00",
6014         "SU1.00",
6015         "SV1.00",
6016         "SY1.00",
6017         "SZ1.00",
6018         "St. Helena Poun1.00",
6019         "S\\u00e3o Tom\\u00e9 & Pr\\u00edncipe Dobr1.00",
6020         "Saudi Riya1.00",
6021         "Serbian Dina1.00",
6022         "Seychellois Rupe1.00",
6023         "Sh1.00",
6024         "Sierra Leonean Leon1.00",
6025         "Silve1.00",
6026         "Singapore Dolla1.00",
6027         "Slovak Korun1.00",
6028         "Slovenian Tola1.00",
6029         "Solomon Islands Dolla1.00",
6030         "Somali Shillin1.00",
6031         "South African Ran1.00",
6032         "South African Rand (financial1.00",
6033         "South Korean Wo1.00",
6034         "Soviet Roubl1.00",
6035         "Spanish Peset1.00",
6036         "Spanish Peseta (A account1.00",
6037         "Spanish Peseta (convertible account1.00",
6038         "Special Drawing Right1.00",
6039         "Sri Lankan Rupe1.00",
6040         "Sudanese Poun1.00",
6041         "Surinamese Dolla1.00",
6042         "Surinamese Guilde1.00",
6043         "Swazi Lilangen1.00",
6044         "Swedish Kron1.00",
6045         "Swiss Fran1.00",
6046         "Syrian Poun1.00",
6047         "T S1.00",
6048         "TH1.00",
6049         "TJ1.00",
6050         "TM1.00",
6051         "TN1.00",
6052         "TO1.00",
6053         "TP1.00",
6054         "TR1.00",
6055         "TT1.00",
6056         "TW1.00",
6057         "TZ1.00",
6058         "New Taiwan Dolla1.00",
6059         "Tajikistani Rubl1.00",
6060         "Tajikistani Somon1.00",
6061         "Tanzanian Shillin1.00",
6062         "Testing Currency Cod1.00",
6063         "Thai Bah1.00",
6064         "Timorese Escud1.00",
6065         "Tongan Pa\\u20bbang1.00",
6066         "Trinidad & Tobago Dolla1.00",
6067         "Tunisian Dina1.00",
6068         "Turkish Lir1.00",
6069         "Turkmenistani Mana1.00",
6070         "U S1.00",
6071         "U1.00",
6072         "UA1.00",
6073         "UG1.00",
6074         "US Dolla1.00",
6075         "US Dollar (Next day1.00",
6076         "US Dollar (Same day1.00",
6077         "US1.00",
6078         "UY1.00",
6079         "UZ1.00",
6080         "Ugandan Shillin1.00",
6081         "Ugandan Shilling (1966\\u201319871.00",
6082         "Ukrainian Hryvni1.00",
6083         "Ukrainian Karbovanet1.00",
6084         "Colombian Real Value Uni1.00",
6085         "United Arab Emirates Dirha1.00",
6086         "Unknown Currenc1.00",
6087         "Ur1.00",
6088         "Uruguay Peso (1975\\u201319931.00",
6089         "Uruguay Peso Uruguay1.00",
6090         "Uruguay Peso (Indexed Units1.00",
6091         "Uzbekistani So1.00",
6092         "V1.00",
6093         "VE1.00",
6094         "VN1.00",
6095         "VU1.00",
6096         "Vanuatu Vat1.00",
6097         "Venezuelan Bol\\u00edva1.00",
6098         "Venezuelan Bol\\u00edvar Fuert1.00",
6099         "Vietnamese Don1.00",
6100         "West African CFA Fran1.00",
6101         "Central African CFA Fran1.00",
6102         "WIR Eur1.00",
6103         "WIR Fran1.00",
6104         "WS1.00",
6105         "Samoa Tal1.00",
6106         "XA1.00",
6107         "XB1.00",
6108         "XC1.00",
6109         "XD1.00",
6110         "XE1.00",
6111         "XF1.00",
6112         "XO1.00",
6113         "XP1.00",
6114         "XR1.00",
6115         "XT1.00",
6116         "XX1.00",
6117         "YD1.00",
6118         "YE1.00",
6119         "YU1.00",
6120         "Yemeni Dina1.00",
6121         "Yemeni Ria1.00",
6122         "Yugoslavian Convertible Dina1.00",
6123         "Yugoslavian Hard Dinar (1966\\u201319901.00",
6124         "Yugoslavian New Dina1.00",
6125         "Z1.00",
6126         "ZA1.00",
6127         "ZM1.00",
6128         "ZR1.00",
6129         "ZW1.00",
6130         "Zairean New Zaire (1993\\u201319981.00",
6131         "Zairean Zair1.00",
6132         "Zambian Kwach1.00",
6133         "Zimbabwean Dollar (1980\\u201320081.00",
6134         "dra1.00",
6135         "lar1.00",
6136         "le1.00",
6137         "man1.00",
6138         "so1.00",
6139     };
6140 
6141     Locale locale("en_US");
6142     for (uint32_t i=0; i<sizeof(DATA)/sizeof(DATA[0]); ++i) {
6143       UnicodeString formatted = ctou(DATA[i]);
6144       UErrorCode status = U_ZERO_ERROR;
6145       NumberFormat* numFmt = NumberFormat::createInstance(locale, UNUM_CURRENCY, status);
6146       if (numFmt != NULL && U_SUCCESS(status)) {
6147           ParsePosition parsePos;
6148           LocalPointer<CurrencyAmount> currAmt(numFmt->parseCurrency(formatted, parsePos));
6149           if (parsePos.getIndex() > 0) {
6150               double doubleVal = currAmt->getNumber().getDouble(status);
6151               if ( doubleVal != 1.0 ) {
6152                   errln("Parsed as currency value other than 1.0: " + formatted + " -> " + doubleVal);
6153               }
6154           } else {
6155               errln("Failed to parse as currency: " + formatted);
6156           }
6157       } else {
6158           dataerrln("Unable to create NumberFormat. - %s", u_errorName(status));
6159           delete numFmt;
6160           break;
6161       }
6162       delete numFmt;
6163     }
6164 
6165     for (uint32_t i=0; i<sizeof(WRONG_DATA)/sizeof(WRONG_DATA[0]); ++i) {
6166       UnicodeString formatted = ctou(WRONG_DATA[i]);
6167       UErrorCode status = U_ZERO_ERROR;
6168       NumberFormat* numFmt = NumberFormat::createInstance(locale, UNUM_CURRENCY, status);
6169       if (numFmt != NULL && U_SUCCESS(status)) {
6170           ParsePosition parsePos;
6171           LocalPointer<CurrencyAmount> currAmt(numFmt->parseCurrency(formatted, parsePos));
6172           if (parsePos.getIndex() > 0) {
6173               double doubleVal = currAmt->getNumber().getDouble(status);
6174               errln("Parsed as currency, should not have: " + formatted + " -> " + doubleVal);
6175           }
6176       } else {
6177           dataerrln("Unable to create NumberFormat. - %s", u_errorName(status));
6178           delete numFmt;
6179           break;
6180       }
6181       delete numFmt;
6182     }
6183 }
6184 
6185 const char* attrString(int32_t);
6186 
6187 // UnicodeString s;
6188 //  std::string ss;
6189 //  std::cout << s.toUTF8String(ss)
expectPositions(FieldPositionIterator & iter,int32_t * values,int32_t tupleCount,const UnicodeString & str)6190 void NumberFormatTest::expectPositions(FieldPositionIterator& iter, int32_t *values, int32_t tupleCount,
6191                                        const UnicodeString& str)  {
6192   UBool found[10];
6193   FieldPosition fp;
6194 
6195   if (tupleCount > 10) {
6196     assertTrue("internal error, tupleCount too large", FALSE);
6197   } else {
6198     for (int i = 0; i < tupleCount; ++i) {
6199       found[i] = FALSE;
6200     }
6201   }
6202 
6203   logln(str);
6204   while (iter.next(fp)) {
6205     UBool ok = FALSE;
6206     int32_t id = fp.getField();
6207     int32_t start = fp.getBeginIndex();
6208     int32_t limit = fp.getEndIndex();
6209 
6210     // is there a logln using printf?
6211     char buf[128];
6212     sprintf(buf, "%24s %3d %3d %3d", attrString(id), id, start, limit);
6213     logln(buf);
6214 
6215     for (int i = 0; i < tupleCount; ++i) {
6216       if (found[i]) {
6217         continue;
6218       }
6219       if (values[i*3] == id &&
6220           values[i*3+1] == start &&
6221           values[i*3+2] == limit) {
6222         found[i] = ok = TRUE;
6223         break;
6224       }
6225     }
6226 
6227     assertTrue((UnicodeString)"found [" + id + "," + start + "," + limit + "]", ok);
6228   }
6229 
6230   // check that all were found
6231   UBool ok = TRUE;
6232   for (int i = 0; i < tupleCount; ++i) {
6233     if (!found[i]) {
6234       ok = FALSE;
6235       assertTrue((UnicodeString) "missing [" + values[i*3] + "," + values[i*3+1] + "," + values[i*3+2] + "]", found[i]);
6236     }
6237   }
6238   assertTrue("no expected values were missing", ok);
6239 }
6240 
expectPosition(FieldPosition & pos,int32_t id,int32_t start,int32_t limit,const UnicodeString & str)6241 void NumberFormatTest::expectPosition(FieldPosition& pos, int32_t id, int32_t start, int32_t limit,
6242                                        const UnicodeString& str)  {
6243   logln(str);
6244   assertTrue((UnicodeString)"id " + id + " == " + pos.getField(), id == pos.getField());
6245   assertTrue((UnicodeString)"begin " + start + " == " + pos.getBeginIndex(), start == pos.getBeginIndex());
6246   assertTrue((UnicodeString)"end " + limit + " == " + pos.getEndIndex(), limit == pos.getEndIndex());
6247 }
6248 
TestFieldPositionIterator()6249 void NumberFormatTest::TestFieldPositionIterator() {
6250   // bug 7372
6251   UErrorCode status = U_ZERO_ERROR;
6252   FieldPositionIterator iter1;
6253   FieldPositionIterator iter2;
6254   FieldPosition pos;
6255 
6256   DecimalFormat *decFmt = (DecimalFormat *) NumberFormat::createInstance(status);
6257   if (failure(status, "NumberFormat::createInstance", TRUE)) return;
6258 
6259   double num = 1234.56;
6260   UnicodeString str1;
6261   UnicodeString str2;
6262 
6263   assertTrue((UnicodeString)"self==", iter1 == iter1);
6264   assertTrue((UnicodeString)"iter1==iter2", iter1 == iter2);
6265 
6266   decFmt->format(num, str1, &iter1, status);
6267   assertTrue((UnicodeString)"iter1 != iter2", iter1 != iter2);
6268   decFmt->format(num, str2, &iter2, status);
6269   assertTrue((UnicodeString)"iter1 == iter2 (2)", iter1 == iter2);
6270   iter1.next(pos);
6271   assertTrue((UnicodeString)"iter1 != iter2 (2)", iter1 != iter2);
6272   iter2.next(pos);
6273   assertTrue((UnicodeString)"iter1 == iter2 (3)", iter1 == iter2);
6274 
6275   // should format ok with no iterator
6276   str2.remove();
6277   decFmt->format(num, str2, NULL, status);
6278   assertEquals("null fpiter", str1, str2);
6279 
6280   delete decFmt;
6281 }
6282 
TestFormatAttributes()6283 void NumberFormatTest::TestFormatAttributes() {
6284   Locale locale("en_US");
6285   UErrorCode status = U_ZERO_ERROR;
6286   DecimalFormat *decFmt = (DecimalFormat *) NumberFormat::createInstance(locale, UNUM_CURRENCY, status);
6287     if (failure(status, "NumberFormat::createInstance", TRUE)) return;
6288   double val = 12345.67;
6289 
6290   {
6291     int32_t expected[] = {
6292       UNUM_CURRENCY_FIELD, 0, 1,
6293       UNUM_GROUPING_SEPARATOR_FIELD, 3, 4,
6294       UNUM_INTEGER_FIELD, 1, 7,
6295       UNUM_DECIMAL_SEPARATOR_FIELD, 7, 8,
6296       UNUM_FRACTION_FIELD, 8, 10,
6297     };
6298     int32_t tupleCount = sizeof(expected)/(3 * sizeof(*expected));
6299 
6300     FieldPositionIterator posIter;
6301     UnicodeString result;
6302     decFmt->format(val, result, &posIter, status);
6303     expectPositions(posIter, expected, tupleCount, result);
6304   }
6305   {
6306     FieldPosition fp(UNUM_INTEGER_FIELD);
6307     UnicodeString result;
6308     decFmt->format(val, result, fp);
6309     expectPosition(fp, UNUM_INTEGER_FIELD, 1, 7, result);
6310   }
6311   {
6312     FieldPosition fp(UNUM_FRACTION_FIELD);
6313     UnicodeString result;
6314     decFmt->format(val, result, fp);
6315     expectPosition(fp, UNUM_FRACTION_FIELD, 8, 10, result);
6316   }
6317   delete decFmt;
6318 
6319   decFmt = (DecimalFormat *) NumberFormat::createInstance(locale, UNUM_SCIENTIFIC, status);
6320   val = -0.0000123;
6321   {
6322     int32_t expected[] = {
6323       UNUM_SIGN_FIELD, 0, 1,
6324       UNUM_INTEGER_FIELD, 1, 2,
6325       UNUM_DECIMAL_SEPARATOR_FIELD, 2, 3,
6326       UNUM_FRACTION_FIELD, 3, 5,
6327       UNUM_EXPONENT_SYMBOL_FIELD, 5, 6,
6328       UNUM_EXPONENT_SIGN_FIELD, 6, 7,
6329       UNUM_EXPONENT_FIELD, 7, 8
6330     };
6331     int32_t tupleCount = sizeof(expected)/(3 * sizeof(*expected));
6332 
6333     FieldPositionIterator posIter;
6334     UnicodeString result;
6335     decFmt->format(val, result, &posIter, status);
6336     expectPositions(posIter, expected, tupleCount, result);
6337   }
6338   {
6339     FieldPosition fp(UNUM_INTEGER_FIELD);
6340     UnicodeString result;
6341     decFmt->format(val, result, fp);
6342     expectPosition(fp, UNUM_INTEGER_FIELD, 1, 2, result);
6343   }
6344   {
6345     FieldPosition fp(UNUM_FRACTION_FIELD);
6346     UnicodeString result;
6347     decFmt->format(val, result, fp);
6348     expectPosition(fp, UNUM_FRACTION_FIELD, 3, 5, result);
6349   }
6350   delete decFmt;
6351 
6352   fflush(stderr);
6353 }
6354 
attrString(int32_t attrId)6355 const char* attrString(int32_t attrId) {
6356   switch (attrId) {
6357     case UNUM_INTEGER_FIELD: return "integer";
6358     case UNUM_FRACTION_FIELD: return "fraction";
6359     case UNUM_DECIMAL_SEPARATOR_FIELD: return "decimal separator";
6360     case UNUM_EXPONENT_SYMBOL_FIELD: return "exponent symbol";
6361     case UNUM_EXPONENT_SIGN_FIELD: return "exponent sign";
6362     case UNUM_EXPONENT_FIELD: return "exponent";
6363     case UNUM_GROUPING_SEPARATOR_FIELD: return "grouping separator";
6364     case UNUM_CURRENCY_FIELD: return "currency";
6365     case UNUM_PERCENT_FIELD: return "percent";
6366     case UNUM_PERMILL_FIELD: return "permille";
6367     case UNUM_SIGN_FIELD: return "sign";
6368     default: return "";
6369   }
6370 }
6371 
6372 //
6373 //   Test formatting & parsing of big decimals.
6374 //      API test, not a comprehensive test.
6375 //      See DecimalFormatTest/DataDrivenTests
6376 //
6377 #define ASSERT_SUCCESS(status) {if (U_FAILURE(status)) errln("file %s, line %d: status: %s", \
6378                                                 __FILE__, __LINE__, u_errorName(status));}
6379 #define ASSERT_EQUALS(expected, actual) {if ((expected) != (actual)) \
6380                   errln("file %s, line %d: %s != %s", __FILE__, __LINE__, #expected, #actual);}
6381 
operator !=(const char * s1,UnicodeString & s2)6382 static UBool operator != (const char *s1, UnicodeString &s2) {
6383     // This function lets ASSERT_EQUALS("literal", UnicodeString) work.
6384     UnicodeString us1(s1);
6385     return us1 != s2;
6386 }
6387 
TestDecimal()6388 void NumberFormatTest::TestDecimal() {
6389     {
6390         UErrorCode  status = U_ZERO_ERROR;
6391         Formattable f("12.345678999987654321E666", status);
6392         ASSERT_SUCCESS(status);
6393         StringPiece s = f.getDecimalNumber(status);
6394         ASSERT_SUCCESS(status);
6395         ASSERT_EQUALS("1.2345678999987654321E+667", s);
6396         //printf("%s\n", s.data());
6397     }
6398 
6399     {
6400         UErrorCode status = U_ZERO_ERROR;
6401         Formattable f1("this is not a number", status);
6402         ASSERT_EQUALS(U_DECIMAL_NUMBER_SYNTAX_ERROR, status);
6403     }
6404 
6405     {
6406         UErrorCode status = U_ZERO_ERROR;
6407         Formattable f;
6408         f.setDecimalNumber("123.45", status);
6409         ASSERT_SUCCESS(status);
6410         ASSERT_EQUALS( Formattable::kDouble, f.getType());
6411         ASSERT_EQUALS(123.45, f.getDouble());
6412         ASSERT_EQUALS(123.45, f.getDouble(status));
6413         ASSERT_SUCCESS(status);
6414         ASSERT_EQUALS("123.45", f.getDecimalNumber(status));
6415         ASSERT_SUCCESS(status);
6416 
6417         f.setDecimalNumber("4.5678E7", status);
6418         int32_t n;
6419         n = f.getLong();
6420         ASSERT_EQUALS(45678000, n);
6421 
6422         status = U_ZERO_ERROR;
6423         f.setDecimalNumber("-123", status);
6424         ASSERT_SUCCESS(status);
6425         ASSERT_EQUALS( Formattable::kLong, f.getType());
6426         ASSERT_EQUALS(-123, f.getLong());
6427         ASSERT_EQUALS(-123, f.getLong(status));
6428         ASSERT_SUCCESS(status);
6429         ASSERT_EQUALS("-123", f.getDecimalNumber(status));
6430         ASSERT_SUCCESS(status);
6431 
6432         status = U_ZERO_ERROR;
6433         f.setDecimalNumber("1234567890123", status);  // Number too big for 32 bits
6434         ASSERT_SUCCESS(status);
6435         ASSERT_EQUALS( Formattable::kInt64, f.getType());
6436         ASSERT_EQUALS(1234567890123LL, f.getInt64());
6437         ASSERT_EQUALS(1234567890123LL, f.getInt64(status));
6438         ASSERT_SUCCESS(status);
6439         ASSERT_EQUALS("1234567890123", f.getDecimalNumber(status));
6440         ASSERT_SUCCESS(status);
6441     }
6442 
6443     {
6444         UErrorCode status = U_ZERO_ERROR;
6445         NumberFormat *fmtr = NumberFormat::createInstance(Locale::getUS(), UNUM_DECIMAL, status);
6446         if (U_FAILURE(status) || fmtr == NULL) {
6447             dataerrln("Unable to create NumberFormat");
6448         } else {
6449             UnicodeString formattedResult;
6450             StringPiece num("244444444444444444444444444444444444446.4");
6451             fmtr->format(num, formattedResult, NULL, status);
6452             ASSERT_SUCCESS(status);
6453             ASSERT_EQUALS("244,444,444,444,444,444,444,444,444,444,444,444,446.4", formattedResult);
6454             //std::string ss; std::cout << formattedResult.toUTF8String(ss);
6455             delete fmtr;
6456         }
6457     }
6458 
6459     {
6460         // Check formatting a DigitList.  DigitList is internal, but this is
6461         // a critical interface that must work.
6462         UErrorCode status = U_ZERO_ERROR;
6463         NumberFormat *fmtr = NumberFormat::createInstance(Locale::getUS(), UNUM_DECIMAL, status);
6464         if (U_FAILURE(status) || fmtr == NULL) {
6465             dataerrln("Unable to create NumberFormat");
6466         } else {
6467             UnicodeString formattedResult;
6468             DigitList dl;
6469             StringPiece num("123.4566666666666666666666666666666666621E+40");
6470             dl.set(num, status);
6471             ASSERT_SUCCESS(status);
6472             fmtr->format(dl, formattedResult, NULL, status);
6473             ASSERT_SUCCESS(status);
6474             ASSERT_EQUALS("1,234,566,666,666,666,666,666,666,666,666,666,666,621,000", formattedResult);
6475 
6476             status = U_ZERO_ERROR;
6477             num.set("666.666");
6478             dl.set(num, status);
6479             FieldPosition pos(NumberFormat::FRACTION_FIELD);
6480             ASSERT_SUCCESS(status);
6481             formattedResult.remove();
6482             fmtr->format(dl, formattedResult, pos, status);
6483             ASSERT_SUCCESS(status);
6484             ASSERT_EQUALS("666.666", formattedResult);
6485             ASSERT_EQUALS(4, pos.getBeginIndex());
6486             ASSERT_EQUALS(7, pos.getEndIndex());
6487             delete fmtr;
6488         }
6489     }
6490 
6491     {
6492         // Check a parse with a formatter with a multiplier.
6493         UErrorCode status = U_ZERO_ERROR;
6494         NumberFormat *fmtr = NumberFormat::createInstance(Locale::getUS(), UNUM_PERCENT, status);
6495         if (U_FAILURE(status) || fmtr == NULL) {
6496             dataerrln("Unable to create NumberFormat");
6497         } else {
6498             UnicodeString input = "1.84%";
6499             Formattable result;
6500             fmtr->parse(input, result, status);
6501             ASSERT_SUCCESS(status);
6502             ASSERT_EQUALS(0, strcmp("0.0184", result.getDecimalNumber(status).data()));
6503             //std::cout << result.getDecimalNumber(status).data();
6504             delete fmtr;
6505         }
6506     }
6507 
6508 #if U_PLATFORM != U_PF_CYGWIN || defined(CYGWINMSVC)
6509     /*
6510      * This test fails on Cygwin (1.7.16) using GCC because of a rounding issue with strtod().
6511      * See #9463
6512      */
6513     {
6514         // Check that a parse returns a decimal number with full accuracy
6515         UErrorCode status = U_ZERO_ERROR;
6516         NumberFormat *fmtr = NumberFormat::createInstance(Locale::getUS(), UNUM_DECIMAL, status);
6517         if (U_FAILURE(status) || fmtr == NULL) {
6518             dataerrln("Unable to create NumberFormat");
6519         } else {
6520             UnicodeString input = "1.002200044400088880000070000";
6521             Formattable result;
6522             fmtr->parse(input, result, status);
6523             ASSERT_SUCCESS(status);
6524             ASSERT_EQUALS(0, strcmp("1.00220004440008888000007", result.getDecimalNumber(status).data()));
6525             ASSERT_EQUALS(1.00220004440008888,   result.getDouble());
6526             //std::cout << result.getDecimalNumber(status).data();
6527             delete fmtr;
6528         }
6529     }
6530 #endif
6531 
6532 }
6533 
TestCurrencyFractionDigits()6534 void NumberFormatTest::TestCurrencyFractionDigits() {
6535     UErrorCode status = U_ZERO_ERROR;
6536     UnicodeString text1, text2;
6537     double value = 99.12345;
6538 
6539     // Create currenct instance
6540     NumberFormat* fmt = NumberFormat::createCurrencyInstance("ja_JP", status);
6541     if (U_FAILURE(status) || fmt == NULL) {
6542         dataerrln("Unable to create NumberFormat");
6543     } else {
6544         fmt->format(value, text1);
6545 
6546         // Reset the same currency and format the test value again
6547         fmt->setCurrency(fmt->getCurrency(), status);
6548         ASSERT_SUCCESS(status);
6549         fmt->format(value, text2);
6550 
6551         if (text1 != text2) {
6552             errln((UnicodeString)"NumberFormat::format() should return the same result - text1="
6553                 + text1 + " text2=" + text2);
6554         }
6555         delete fmt;
6556     }
6557 }
6558 
TestExponentParse()6559 void NumberFormatTest::TestExponentParse() {
6560 
6561     UErrorCode status = U_ZERO_ERROR;
6562     Formattable result;
6563     ParsePosition parsePos(0);
6564 
6565     // set the exponent symbol
6566     status = U_ZERO_ERROR;
6567     DecimalFormatSymbols *symbols = new DecimalFormatSymbols(Locale::getDefault(), status);
6568     if(U_FAILURE(status)) {
6569         dataerrln((UnicodeString)"ERROR: Could not create DecimalFormatSymbols (Default)");
6570         return;
6571     }
6572 
6573     // create format instance
6574     status = U_ZERO_ERROR;
6575     DecimalFormat fmt("#####", symbols, status);
6576     if(U_FAILURE(status)) {
6577         errln((UnicodeString)"ERROR: Could not create DecimalFormat (pattern, symbols*)");
6578     }
6579 
6580     // parse the text
6581     fmt.parse("5.06e-27", result, parsePos);
6582     if(result.getType() != Formattable::kDouble &&
6583        result.getDouble() != 5.06E-27 &&
6584        parsePos.getIndex() != 8
6585        )
6586     {
6587         errln("ERROR: parse failed - expected 5.06E-27, 8  - returned %d, %i",
6588               result.getDouble(), parsePos.getIndex());
6589     }
6590 }
6591 
TestExplicitParents()6592 void NumberFormatTest::TestExplicitParents() {
6593 
6594     /* Test that number formats are properly inherited from es_419 */
6595     /* These could be subject to change if the CLDR data changes */
6596     static const char* parentLocaleTests[][2]= {
6597     /* locale ID */  /* expected */
6598     {"es_CO", "1.250,75" },
6599     {"es_CR", "1.250,75" },
6600     {"es_ES", "1.250,75" },
6601     {"es_GQ", "1.250,75" },
6602     {"es_MX", "1,250.75" },
6603     {"es_US", "1,250.75" },
6604     {"es_VE", "1.250,75" },
6605     };
6606 
6607     UnicodeString s;
6608 
6609     for(int i=0; i < (int)(sizeof(parentLocaleTests)/sizeof(parentLocaleTests[i])); i++){
6610         UErrorCode status = U_ZERO_ERROR;
6611         const char *localeID = parentLocaleTests[i][0];
6612         UnicodeString expected(parentLocaleTests[i][1], -1, US_INV);
6613         expected = expected.unescape();
6614         char loc[256]={0};
6615         uloc_canonicalize(localeID, loc, 256, &status);
6616         NumberFormat *fmt= NumberFormat::createInstance(Locale(loc), status);
6617         if(U_FAILURE(status)){
6618             dataerrln("Could not create number formatter for locale %s - %s",localeID, u_errorName(status));
6619             continue;
6620         }
6621         s.remove();
6622         fmt->format(1250.75, s);
6623         if(s!=expected){
6624             errln(UnicodeString("FAIL: Expected: ")+expected
6625                     + UnicodeString(" Got: ") + s
6626                     + UnicodeString( " for locale: ")+ UnicodeString(localeID) );
6627         }
6628         if (U_FAILURE(status)){
6629             errln((UnicodeString)"FAIL: Status " + (int32_t)status);
6630         }
6631         delete fmt;
6632     }
6633 
6634 }
6635 
6636 /**
6637  * Test available numbering systems API.
6638  */
TestAvailableNumberingSystems()6639 void NumberFormatTest::TestAvailableNumberingSystems() {
6640     UErrorCode status = U_ZERO_ERROR;
6641     StringEnumeration *availableNumberingSystems = NumberingSystem::getAvailableNames(status);
6642     CHECK_DATA(status, "NumberingSystem::getAvailableNames()")
6643 
6644     int32_t nsCount = availableNumberingSystems->count(status);
6645     if ( nsCount < 36 ) {
6646         errln("FAIL: Didn't get as many numbering systems as we had hoped for. Need at least 36, got %d",nsCount);
6647     }
6648 
6649     /* A relatively simple test of the API.  We call getAvailableNames() and cycle through */
6650     /* each name returned, attempting to create a numbering system based on that name and  */
6651     /* verifying that the name returned from the resulting numbering system is the same    */
6652     /* one that we initially thought.                                                      */
6653 
6654     int32_t len;
6655     for ( int32_t i = 0 ; i < nsCount ; i++ ) {
6656         const char *nsname = availableNumberingSystems->next(&len,status);
6657         NumberingSystem* ns = NumberingSystem::createInstanceByName(nsname,status);
6658         if ( uprv_strcmp(nsname,ns->getName()) ) {
6659             errln("FAIL: Numbering system name didn't match for name = %s\n",nsname);
6660         }
6661 
6662         delete ns;
6663     }
6664 
6665     delete availableNumberingSystems;
6666 }
6667 
6668 void
Test9087(void)6669 NumberFormatTest::Test9087(void)
6670 {
6671     U_STRING_DECL(pattern,"#",1);
6672     U_STRING_INIT(pattern,"#",1);
6673 
6674     U_STRING_DECL(infstr,"INF",3);
6675     U_STRING_INIT(infstr,"INF",3);
6676 
6677     U_STRING_DECL(nanstr,"NAN",3);
6678     U_STRING_INIT(nanstr,"NAN",3);
6679 
6680     UChar outputbuf[50] = {0};
6681     UErrorCode status = U_ZERO_ERROR;
6682     UNumberFormat* fmt = unum_open(UNUM_PATTERN_DECIMAL,pattern,1,NULL,NULL,&status);
6683     if ( U_FAILURE(status) ) {
6684         dataerrln("FAIL: error in unum_open() - %s", u_errorName(status));
6685         return;
6686     }
6687 
6688     unum_setSymbol(fmt,UNUM_INFINITY_SYMBOL,infstr,3,&status);
6689     unum_setSymbol(fmt,UNUM_NAN_SYMBOL,nanstr,3,&status);
6690     if ( U_FAILURE(status) ) {
6691         errln("FAIL: error setting symbols");
6692     }
6693 
6694     double inf = uprv_getInfinity();
6695 
6696     unum_setAttribute(fmt,UNUM_ROUNDING_MODE,UNUM_ROUND_HALFEVEN);
6697     unum_setDoubleAttribute(fmt,UNUM_ROUNDING_INCREMENT,0);
6698 
6699     UFieldPosition position = { 0, 0, 0};
6700     unum_formatDouble(fmt,inf,outputbuf,50,&position,&status);
6701 
6702     if ( u_strcmp(infstr, outputbuf)) {
6703         errln((UnicodeString)"FAIL: unexpected result for infinity - expected " + infstr + " got " + outputbuf);
6704     }
6705 
6706     unum_close(fmt);
6707 }
6708 
6709 #include "dcfmtimp.h"
6710 
TestFormatFastpaths()6711 void NumberFormatTest::TestFormatFastpaths() {
6712 #if UCONFIG_FORMAT_FASTPATHS_49
6713     logln("Sizeof DecimalFormat = %d, Sizeof DecimalFormatInternal=%d, UNUM_DECIMALFORMAT_INTERNAL_SIZE=%d\n",
6714         sizeof(DecimalFormat), sizeof(DecimalFormatInternal), UNUM_DECIMALFORMAT_INTERNAL_SIZE);
6715     if(UNUM_DECIMALFORMAT_INTERNAL_SIZE < sizeof(DecimalFormatInternal)) {
6716         errln("Error: sizeof(DecimalFormatInternal)=%d but UNUM_DECIMALFORMAT_INTERNAL_SIZE is only %d. Increase the #define?\n", sizeof(DecimalFormatInternal), UNUM_DECIMALFORMAT_INTERNAL_SIZE);
6717     } else if(UNUM_DECIMALFORMAT_INTERNAL_SIZE > (sizeof(DecimalFormatInternal)+16)) {
6718         infoln("Note: sizeof(DecimalFormatInternal)=%d but UNUM_DECIMALFORMAT_INTERNAL_SIZE is %d. Decrease the #define? sizeof(DecimalFormat)=%d\n", sizeof(DecimalFormatInternal), UNUM_DECIMALFORMAT_INTERNAL_SIZE, sizeof(DecimalFormat));
6719     }
6720 #else
6721     infoln("NOTE: UCONFIG_FORMAT_FASTPATHS not set, test skipped.");
6722 #endif
6723 
6724     // get some additional case
6725     {
6726         UErrorCode status=U_ZERO_ERROR;
6727         DecimalFormat df(UnicodeString("0000",""),status);
6728         int64_t long_number = 1;
6729         UnicodeString expect = "0001";
6730         UnicodeString result;
6731         FieldPosition pos;
6732         df.format(long_number, result, pos);
6733         if(U_FAILURE(status)||expect!=result) {
6734             errcheckln(status, "FAIL: expected '"+expect+"' got '"+result+"' status "+UnicodeString(u_errorName(status),""));
6735         } else {
6736             logln("OK:  got expected '"+result+"' status "+UnicodeString(u_errorName(status),""));
6737         }
6738     }
6739     {
6740         UErrorCode status=U_ZERO_ERROR;
6741         DecimalFormat df(UnicodeString("0000000000000000000",""),status);
6742         int64_t long_number = U_INT64_MIN; // -9223372036854775808L;
6743         // uint8_t bits[8];
6744         // memcpy(bits,&long_number,8);
6745         // for(int i=0;i<8;i++) {
6746         //   logln("bits: %02X", (unsigned int)bits[i]);
6747         // }
6748         UnicodeString expect = "-9223372036854775808";
6749         UnicodeString result;
6750         FieldPosition pos;
6751         df.format(long_number, result, pos);
6752         if(U_FAILURE(status)||expect!=result) {
6753             errcheckln(status, "FAIL: expected '"+expect+"' got '"+result+"' status "+UnicodeString(u_errorName(status),"")+" on -9223372036854775808");
6754         } else {
6755             logln("OK:  got expected '"+result+"' status "+UnicodeString(u_errorName(status),"")+" on -9223372036854775808");
6756         }
6757     }
6758     {
6759         UErrorCode status=U_ZERO_ERROR;
6760         DecimalFormat df(UnicodeString("0000000000000000000",""),status);
6761         int64_t long_number = U_INT64_MAX; // -9223372036854775808L;
6762         // uint8_t bits[8];
6763         // memcpy(bits,&long_number,8);
6764         // for(int i=0;i<8;i++) {
6765         //   logln("bits: %02X", (unsigned int)bits[i]);
6766         // }
6767         UnicodeString expect = "9223372036854775807";
6768         UnicodeString result;
6769         FieldPosition pos;
6770         df.format(long_number, result, pos);
6771         if(U_FAILURE(status)||expect!=result) {
6772             errcheckln(status, "FAIL: expected '"+expect+"' got '"+result+"' status "+UnicodeString(u_errorName(status),"")+" on U_INT64_MAX");
6773         } else {
6774             logln("OK:  got expected '"+result+"' status "+UnicodeString(u_errorName(status),"")+" on U_INT64_MAX");
6775         }
6776     }
6777     {
6778         UErrorCode status=U_ZERO_ERROR;
6779         DecimalFormat df(UnicodeString("0000000000000000000",""),status);
6780         int64_t long_number = 0;
6781         // uint8_t bits[8];
6782         // memcpy(bits,&long_number,8);
6783         // for(int i=0;i<8;i++) {
6784         //   logln("bits: %02X", (unsigned int)bits[i]);
6785         // }
6786         UnicodeString expect = "0000000000000000000";
6787         UnicodeString result;
6788         FieldPosition pos;
6789         df.format(long_number, result, pos);
6790         if(U_FAILURE(status)||expect!=result) {
6791             errcheckln(status, "FAIL: expected '"+expect+"' got '"+result+"' status "+UnicodeString(u_errorName(status),"")+" on 0");
6792         } else {
6793             logln("OK:  got expected '"+result+"' status "+UnicodeString(u_errorName(status),"")+" on 0");
6794         }
6795     }
6796     {
6797         UErrorCode status=U_ZERO_ERROR;
6798         DecimalFormat df(UnicodeString("0000000000000000000",""),status);
6799         int64_t long_number = U_INT64_MIN + 1;
6800         UnicodeString expect = "-9223372036854775807";
6801         UnicodeString result;
6802         FieldPosition pos;
6803         df.format(long_number, result, pos);
6804         if(U_FAILURE(status)||expect!=result) {
6805             errcheckln(status, "FAIL: expected '"+expect+"' got '"+result+"' status "+UnicodeString(u_errorName(status),"")+" on -9223372036854775807");
6806         } else {
6807             logln("OK:  got expected '"+result+"' status "+UnicodeString(u_errorName(status),"")+" on -9223372036854775807");
6808         }
6809     }
6810 }
6811 
6812 
TestFormattableSize(void)6813 void NumberFormatTest::TestFormattableSize(void) {
6814   if(sizeof(FmtStackData) > UNUM_INTERNAL_STACKARRAY_SIZE) {
6815     errln("Error: sizeof(FmtStackData)=%d, UNUM_INTERNAL_STACKARRAY_SIZE=%d\n",
6816           sizeof(FmtStackData), UNUM_INTERNAL_STACKARRAY_SIZE);
6817   } else if(sizeof(FmtStackData) < UNUM_INTERNAL_STACKARRAY_SIZE) {
6818     logln("Warning: sizeof(FmtStackData)=%d, UNUM_INTERNAL_STACKARRAY_SIZE=%d\n",
6819         sizeof(FmtStackData), UNUM_INTERNAL_STACKARRAY_SIZE);
6820   } else {
6821     logln("sizeof(FmtStackData)=%d, UNUM_INTERNAL_STACKARRAY_SIZE=%d\n",
6822         sizeof(FmtStackData), UNUM_INTERNAL_STACKARRAY_SIZE);
6823   }
6824 }
6825 
testFormattableAsUFormattable(const char * file,int line,Formattable & f)6826 UBool NumberFormatTest::testFormattableAsUFormattable(const char *file, int line, Formattable &f) {
6827   UnicodeString fileLine = UnicodeString(file)+UnicodeString(":")+line+UnicodeString(": ");
6828 
6829   UFormattable *u = f.toUFormattable();
6830   logln();
6831   if (u == NULL) {
6832     errln("%s:%d: Error: f.toUFormattable() retuned NULL.");
6833     return FALSE;
6834   }
6835   logln("%s:%d: comparing Formattable with UFormattable", file, line);
6836   logln(fileLine + toString(f));
6837 
6838   UErrorCode status = U_ZERO_ERROR;
6839   UErrorCode valueStatus = U_ZERO_ERROR;
6840   UFormattableType expectUType = UFMT_COUNT; // invalid
6841 
6842   UBool triedExact = FALSE; // did we attempt an exact comparison?
6843   UBool exactMatch = FALSE; // was the exact comparison true?
6844 
6845   switch( f.getType() ) {
6846   case Formattable::kDate:
6847     expectUType = UFMT_DATE;
6848     exactMatch = (f.getDate()==ufmt_getDate(u, &valueStatus));
6849     triedExact = TRUE;
6850     break;
6851   case Formattable::kDouble:
6852     expectUType = UFMT_DOUBLE;
6853     exactMatch = (f.getDouble()==ufmt_getDouble(u, &valueStatus));
6854     triedExact = TRUE;
6855     break;
6856   case Formattable::kLong:
6857     expectUType = UFMT_LONG;
6858     exactMatch = (f.getLong()==ufmt_getLong(u, &valueStatus));
6859     triedExact = TRUE;
6860     break;
6861   case Formattable::kString:
6862     expectUType = UFMT_STRING;
6863     {
6864       UnicodeString str;
6865       f.getString(str);
6866       int32_t len;
6867       const UChar* uch = ufmt_getUChars(u, &len, &valueStatus);
6868       if(U_SUCCESS(valueStatus)) {
6869         UnicodeString str2(uch, len);
6870         assertTrue("UChar* NULL-terminated", uch[len]==0);
6871         exactMatch = (str == str2);
6872       }
6873       triedExact = TRUE;
6874     }
6875     break;
6876   case Formattable::kArray:
6877     expectUType = UFMT_ARRAY;
6878     triedExact = TRUE;
6879     {
6880       int32_t count = ufmt_getArrayLength(u, &valueStatus);
6881       int32_t count2;
6882       const Formattable *array2 = f.getArray(count2);
6883       exactMatch = assertEquals(fileLine + " array count", count, count2);
6884 
6885       if(exactMatch) {
6886         for(int i=0;U_SUCCESS(valueStatus) && i<count;i++) {
6887           UFormattable *uu = ufmt_getArrayItemByIndex(u, i, &valueStatus);
6888           if(*Formattable::fromUFormattable(uu) != (array2[i])) {
6889             errln("%s:%d: operator== did not match at index[%d] - %p vs %p", file, line, i,
6890                   (const void*)Formattable::fromUFormattable(uu), (const void*)&(array2[i]));
6891             exactMatch = FALSE;
6892           } else {
6893             if(!testFormattableAsUFormattable("(sub item)",i,*Formattable::fromUFormattable(uu))) {
6894               exactMatch = FALSE;
6895             }
6896           }
6897         }
6898       }
6899     }
6900     break;
6901   case Formattable::kInt64:
6902     expectUType = UFMT_INT64;
6903     exactMatch = (f.getInt64()==ufmt_getInt64(u, &valueStatus));
6904     triedExact = TRUE;
6905     break;
6906   case Formattable::kObject:
6907     expectUType = UFMT_OBJECT;
6908     exactMatch = (f.getObject()==ufmt_getObject(u, &valueStatus));
6909     triedExact = TRUE;
6910     break;
6911   }
6912   UFormattableType uType = ufmt_getType(u, &status);
6913 
6914   if(U_FAILURE(status)) {
6915     errln("%s:%d: Error calling ufmt_getType - %s", file, line, u_errorName(status));
6916     return FALSE;
6917   }
6918 
6919   if(uType != expectUType) {
6920     errln("%s:%d: got type (%d) expected (%d) from ufmt_getType", file, line, (int) uType, (int) expectUType);
6921   }
6922 
6923   if(triedExact) {
6924     if(U_FAILURE(valueStatus)) {
6925       errln("%s:%d: got err %s trying to ufmt_get...() for exact match check", file, line, u_errorName(valueStatus));
6926     } else if(!exactMatch) {
6927      errln("%s:%d: failed exact match for the Formattable type", file, line);
6928     } else {
6929       logln("%s:%d: exact match OK", file, line);
6930     }
6931   } else {
6932     logln("%s:%d: note, did not attempt exact match for this formattable type", file, line);
6933   }
6934 
6935   if( assertEquals(fileLine + " isNumeric()", f.isNumeric(), ufmt_isNumeric(u))
6936       && f.isNumeric()) {
6937     UErrorCode convStatus = U_ZERO_ERROR;
6938 
6939     if(uType != UFMT_INT64) { // may fail to compare
6940       assertTrue(fileLine + " as doubles ==", f.getDouble(convStatus)==ufmt_getDouble(u, &convStatus));
6941     }
6942 
6943     if( assertSuccess(fileLine + " (numeric conversion status)", convStatus) ) {
6944       StringPiece fDecNum = f.getDecimalNumber(convStatus);
6945 #if 1
6946       int32_t len;
6947       const char *decNumChars = ufmt_getDecNumChars(u, &len, &convStatus);
6948 #else
6949       // copy version
6950       char decNumChars[200];
6951       int32_t len = ufmt_getDecNumChars(u, decNumChars, 200, &convStatus);
6952 #endif
6953 
6954       if( assertSuccess(fileLine + " (decNumbers conversion)", convStatus) ) {
6955         logln(fileLine + decNumChars);
6956         assertEquals(fileLine + " decNumChars length==", len, fDecNum.length());
6957         assertEquals(fileLine + " decNumChars digits", decNumChars, fDecNum.data());
6958       }
6959 
6960       UErrorCode int64ConversionF = U_ZERO_ERROR;
6961       int64_t l = f.getInt64(int64ConversionF);
6962       UErrorCode int64ConversionU = U_ZERO_ERROR;
6963       int64_t r = ufmt_getInt64(u, &int64ConversionU);
6964 
6965       if( (l==r)
6966           && ( uType != UFMT_INT64 ) // int64 better not overflow
6967           && (U_INVALID_FORMAT_ERROR==int64ConversionU)
6968           && (U_INVALID_FORMAT_ERROR==int64ConversionF) ) {
6969         logln("%s:%d: OK: 64 bit overflow", file, line);
6970       } else {
6971         assertEquals(fileLine + " as int64 ==", l, r);
6972         assertSuccess(fileLine + " Formattable.getnt64()", int64ConversionF);
6973         assertSuccess(fileLine + " ufmt_getInt64()", int64ConversionU);
6974       }
6975     }
6976   }
6977   return exactMatch || !triedExact;
6978 }
6979 
TestUFormattable(void)6980 void NumberFormatTest::TestUFormattable(void) {
6981   {
6982     // test that a default formattable is equal to Formattable()
6983     UErrorCode status = U_ZERO_ERROR;
6984     LocalUFormattablePointer defaultUFormattable(ufmt_open(&status));
6985     assertSuccess("calling umt_open", status);
6986     Formattable defaultFormattable;
6987     assertTrue((UnicodeString)"comparing ufmt_open() with Formattable()",
6988                (defaultFormattable
6989                 == *(Formattable::fromUFormattable(defaultUFormattable.getAlias()))));
6990     assertTrue((UnicodeString)"comparing ufmt_open() with Formattable()",
6991                (defaultFormattable
6992                 == *(Formattable::fromUFormattable(defaultUFormattable.getAlias()))));
6993     assertTrue((UnicodeString)"comparing Formattable() round tripped through UFormattable",
6994                (defaultFormattable
6995                 == *(Formattable::fromUFormattable(defaultFormattable.toUFormattable()))));
6996     assertTrue((UnicodeString)"comparing &Formattable() round tripped through UFormattable",
6997                ((&defaultFormattable)
6998                 == Formattable::fromUFormattable(defaultFormattable.toUFormattable())));
6999     assertFalse((UnicodeString)"comparing &Formattable() with ufmt_open()",
7000                ((&defaultFormattable)
7001                 == Formattable::fromUFormattable(defaultUFormattable.getAlias())));
7002     testFormattableAsUFormattable(__FILE__, __LINE__, defaultFormattable);
7003   }
7004   // test some random Formattables
7005   {
7006     Formattable f(ucal_getNow(), Formattable::kIsDate);
7007     testFormattableAsUFormattable(__FILE__, __LINE__,  f);
7008   }
7009   {
7010     Formattable f((double)1.61803398874989484820); // golden ratio
7011     testFormattableAsUFormattable(__FILE__, __LINE__,  f);
7012   }
7013   {
7014     Formattable f((int64_t)80994231587905127LL); // weight of the moon, in kilotons http://solarsystem.nasa.gov/planets/profile.cfm?Display=Facts&Object=Moon
7015     testFormattableAsUFormattable(__FILE__, __LINE__,  f);
7016   }
7017   {
7018     Formattable f((int32_t)4); // random number, source: http://www.xkcd.com/221/
7019     testFormattableAsUFormattable(__FILE__, __LINE__,  f);
7020   }
7021   {
7022     Formattable f("Hello world."); // should be invariant?
7023     testFormattableAsUFormattable(__FILE__, __LINE__,  f);
7024   }
7025   {
7026     UErrorCode status2 = U_ZERO_ERROR;
7027     Formattable f(StringPiece("73476730924573500000000.0"), status2); // weight of the moon, kg
7028     assertSuccess("Constructing a StringPiece", status2);
7029     testFormattableAsUFormattable(__FILE__, __LINE__,  f);
7030   }
7031   {
7032     UErrorCode status2 = U_ZERO_ERROR;
7033     UObject *obj = new Locale();
7034     Formattable f(obj);
7035     assertSuccess("Constructing a Formattable from a default constructed Locale()", status2);
7036     testFormattableAsUFormattable(__FILE__, __LINE__,  f);
7037   }
7038   {
7039     const Formattable array[] = {
7040       Formattable(ucal_getNow(), Formattable::kIsDate),
7041       Formattable((int32_t)4),
7042       Formattable((double)1.234),
7043     };
7044 
7045     Formattable fa(array, 3);
7046     testFormattableAsUFormattable(__FILE__, __LINE__, fa);
7047   }
7048 }
7049 
TestSignificantDigits(void)7050 void NumberFormatTest::TestSignificantDigits(void) {
7051   double input[] = {
7052         0, 0,
7053         0.1, -0.1,
7054         123, -123,
7055         12345, -12345,
7056         123.45, -123.45,
7057         123.44501, -123.44501,
7058         0.001234, -0.001234,
7059         0.00000000123, -0.00000000123,
7060         0.0000000000000000000123, -0.0000000000000000000123,
7061         1.2, -1.2,
7062         0.0000000012344501, -0.0000000012344501,
7063         123445.01, -123445.01,
7064         12344501000000000000000000000000000.0, -12344501000000000000000000000000000.0,
7065     };
7066     const char* expected[] = {
7067         "0.00", "0.00",
7068         "0.100", "-0.100",
7069         "123", "-123",
7070         "12345", "-12345",
7071         "123.45", "-123.45",
7072         "123.45", "-123.45",
7073         "0.001234", "-0.001234",
7074         "0.00000000123", "-0.00000000123",
7075         "0.0000000000000000000123", "-0.0000000000000000000123",
7076         "1.20", "-1.20",
7077         "0.0000000012345", "-0.0000000012345",
7078         "123450", "-123450",
7079         "12345000000000000000000000000000000", "-12345000000000000000000000000000000",
7080     };
7081 
7082     UErrorCode status = U_ZERO_ERROR;
7083     Locale locale("en_US");
7084     LocalPointer<DecimalFormat> numberFormat(static_cast<DecimalFormat*>(
7085             NumberFormat::createInstance(locale, status)));
7086     CHECK_DATA(status,"NumberFormat::createInstance")
7087 
7088     numberFormat->setSignificantDigitsUsed(TRUE);
7089     numberFormat->setMinimumSignificantDigits(3);
7090     numberFormat->setMaximumSignificantDigits(5);
7091     numberFormat->setGroupingUsed(false);
7092 
7093     UnicodeString result;
7094     UnicodeString expectedResult;
7095     for (unsigned int i = 0; i < sizeof(input)/sizeof(double); ++i) {
7096         numberFormat->format(input[i], result);
7097         UnicodeString expectedResult(expected[i]);
7098         if (result != expectedResult) {
7099           errln((UnicodeString)"Expected: '" + expectedResult + "' got '" + result);
7100         }
7101         result.remove();
7102     }
7103 }
7104 
TestShowZero()7105 void NumberFormatTest::TestShowZero() {
7106     UErrorCode status = U_ZERO_ERROR;
7107     Locale locale("en_US");
7108     LocalPointer<DecimalFormat> numberFormat(static_cast<DecimalFormat*>(
7109             NumberFormat::createInstance(locale, status)));
7110     CHECK_DATA(status, "NumberFormat::createInstance")
7111 
7112     numberFormat->setSignificantDigitsUsed(TRUE);
7113     numberFormat->setMaximumSignificantDigits(3);
7114 
7115     UnicodeString result;
7116     numberFormat->format(0.0, result);
7117     if (result != "0") {
7118         errln((UnicodeString)"Expected: 0, got " + result);
7119     }
7120 }
7121 
TestBug9936()7122 void NumberFormatTest::TestBug9936() {
7123     UErrorCode status = U_ZERO_ERROR;
7124     Locale locale("en_US");
7125     LocalPointer<DecimalFormat> numberFormat(static_cast<DecimalFormat*>(
7126             NumberFormat::createInstance(locale, status)));
7127     if (U_FAILURE(status)) {
7128         dataerrln("File %s, Line %d: status = %s.\n", __FILE__, __LINE__, u_errorName(status));
7129         return;
7130     }
7131 
7132     if (numberFormat->areSignificantDigitsUsed() == TRUE) {
7133         errln("File %s, Line %d: areSignificantDigitsUsed() was TRUE, expected FALSE.\n", __FILE__, __LINE__);
7134     }
7135     numberFormat->setSignificantDigitsUsed(TRUE);
7136     if (numberFormat->areSignificantDigitsUsed() == FALSE) {
7137         errln("File %s, Line %d: areSignificantDigitsUsed() was FALSE, expected TRUE.\n", __FILE__, __LINE__);
7138     }
7139 
7140     numberFormat->setSignificantDigitsUsed(FALSE);
7141     if (numberFormat->areSignificantDigitsUsed() == TRUE) {
7142         errln("File %s, Line %d: areSignificantDigitsUsed() was TRUE, expected FALSE.\n", __FILE__, __LINE__);
7143     }
7144 
7145     numberFormat->setMinimumSignificantDigits(3);
7146     if (numberFormat->areSignificantDigitsUsed() == FALSE) {
7147         errln("File %s, Line %d: areSignificantDigitsUsed() was FALSE, expected TRUE.\n", __FILE__, __LINE__);
7148     }
7149 
7150     numberFormat->setSignificantDigitsUsed(FALSE);
7151     numberFormat->setMaximumSignificantDigits(6);
7152     if (numberFormat->areSignificantDigitsUsed() == FALSE) {
7153         errln("File %s, Line %d: areSignificantDigitsUsed() was FALSE, expected TRUE.\n", __FILE__, __LINE__);
7154     }
7155 
7156 }
7157 
TestParseNegativeWithFaLocale()7158 void NumberFormatTest::TestParseNegativeWithFaLocale() {
7159     UErrorCode status = U_ZERO_ERROR;
7160     DecimalFormat *test = (DecimalFormat *) NumberFormat::createInstance("fa", status);
7161     CHECK_DATA(status, "NumberFormat::createInstance")
7162     test->setLenient(TRUE);
7163     Formattable af;
7164     ParsePosition ppos;
7165     UnicodeString value("\\u200e-0,5");
7166     value = value.unescape();
7167     test->parse(value, af, ppos);
7168     if (ppos.getIndex() == 0) {
7169         errln("Expected -0,5 to parse for Farsi.");
7170     }
7171     delete test;
7172 }
7173 
TestParseNegativeWithAlternateMinusSign()7174 void NumberFormatTest::TestParseNegativeWithAlternateMinusSign() {
7175     UErrorCode status = U_ZERO_ERROR;
7176     DecimalFormat *test = (DecimalFormat *) NumberFormat::createInstance("en", status);
7177     CHECK_DATA(status, "NumberFormat::createInstance")
7178     test->setLenient(TRUE);
7179     Formattable af;
7180     ParsePosition ppos;
7181     UnicodeString value("\\u208B0.5");
7182     value = value.unescape();
7183     test->parse(value, af, ppos);
7184     if (ppos.getIndex() == 0) {
7185         errln(UnicodeString("Expected ") + value + UnicodeString(" to parse."));
7186     }
7187     delete test;
7188 }
7189 
TestCustomCurrencySignAndSeparator()7190 void NumberFormatTest::TestCustomCurrencySignAndSeparator() {
7191     UErrorCode status = U_ZERO_ERROR;
7192     DecimalFormatSymbols custom(Locale::getUS(), status);
7193     CHECK(status, "DecimalFormatSymbols constructor");
7194 
7195     custom.setSymbol(DecimalFormatSymbols::kCurrencySymbol, "*");
7196     custom.setSymbol(DecimalFormatSymbols::kMonetaryGroupingSeparatorSymbol, "^");
7197     custom.setSymbol(DecimalFormatSymbols::kMonetarySeparatorSymbol, ":");
7198 
7199     UnicodeString pat(" #,##0.00");
7200     pat.insert(0, (UChar)0x00A4);
7201 
7202     DecimalFormat fmt(pat, custom, status);
7203     CHECK(status, "DecimalFormat constructor");
7204 
7205     UnicodeString numstr("* 1^234:56");
7206     expect2(fmt, (Formattable)((double)1234.56), numstr);
7207 }
7208 
7209 typedef struct {
7210     const char *   locale;
7211     UBool          lenient;
7212     UnicodeString  numString;
7213     double         value;
7214 } SignsAndMarksItem;
7215 
7216 
TestParseSignsAndMarks()7217 void NumberFormatTest::TestParseSignsAndMarks() {
7218     const SignsAndMarksItem items[] = {
7219         // locale               lenient numString                                                       value
7220         { "en",                 FALSE,  CharsToUnicodeString("12"),                                      12 },
7221         { "en",                 TRUE,   CharsToUnicodeString("12"),                                      12 },
7222         { "en",                 FALSE,  CharsToUnicodeString("-23"),                                    -23 },
7223         { "en",                 TRUE,   CharsToUnicodeString("-23"),                                    -23 },
7224         { "en",                 TRUE,   CharsToUnicodeString("- 23"),                                   -23 },
7225         { "en",                 FALSE,  CharsToUnicodeString("\\u200E-23"),                             -23 },
7226         { "en",                 TRUE,   CharsToUnicodeString("\\u200E-23"),                             -23 },
7227         { "en",                 TRUE,   CharsToUnicodeString("\\u200E- 23"),                            -23 },
7228 
7229         { "en@numbers=arab",    FALSE,  CharsToUnicodeString("\\u0663\\u0664"),                          34 },
7230         { "en@numbers=arab",    TRUE,   CharsToUnicodeString("\\u0663\\u0664"),                          34 },
7231         { "en@numbers=arab",    FALSE,  CharsToUnicodeString("-\\u0664\\u0665"),                        -45 },
7232         { "en@numbers=arab",    TRUE,   CharsToUnicodeString("-\\u0664\\u0665"),                        -45 },
7233         { "en@numbers=arab",    TRUE,   CharsToUnicodeString("- \\u0664\\u0665"),                       -45 },
7234         { "en@numbers=arab",    FALSE,  CharsToUnicodeString("\\u200F-\\u0664\\u0665"),                 -45 },
7235         { "en@numbers=arab",    TRUE,   CharsToUnicodeString("\\u200F-\\u0664\\u0665"),                 -45 },
7236         { "en@numbers=arab",    TRUE,   CharsToUnicodeString("\\u200F- \\u0664\\u0665"),                -45 },
7237 
7238         { "en@numbers=arabext", FALSE,  CharsToUnicodeString("\\u06F5\\u06F6"),                          56 },
7239         { "en@numbers=arabext", TRUE,   CharsToUnicodeString("\\u06F5\\u06F6"),                          56 },
7240         { "en@numbers=arabext", FALSE,  CharsToUnicodeString("-\\u06F6\\u06F7"),                        -67 },
7241         { "en@numbers=arabext", TRUE,   CharsToUnicodeString("-\\u06F6\\u06F7"),                        -67 },
7242         { "en@numbers=arabext", TRUE,   CharsToUnicodeString("- \\u06F6\\u06F7"),                       -67 },
7243         { "en@numbers=arabext", FALSE,  CharsToUnicodeString("\\u200E-\\u200E\\u06F6\\u06F7"),          -67 },
7244         { "en@numbers=arabext", TRUE,   CharsToUnicodeString("\\u200E-\\u200E\\u06F6\\u06F7"),          -67 },
7245         { "en@numbers=arabext", TRUE,   CharsToUnicodeString("\\u200E-\\u200E \\u06F6\\u06F7"),         -67 },
7246 
7247         { "he",                 FALSE,  CharsToUnicodeString("12"),                                      12 },
7248         { "he",                 TRUE,   CharsToUnicodeString("12"),                                      12 },
7249         { "he",                 FALSE,  CharsToUnicodeString("-23"),                                    -23 },
7250         { "he",                 TRUE,   CharsToUnicodeString("-23"),                                    -23 },
7251         { "he",                 TRUE,   CharsToUnicodeString("- 23"),                                   -23 },
7252         { "he",                 FALSE,  CharsToUnicodeString("\\u200E-23"),                             -23 },
7253         { "he",                 TRUE,   CharsToUnicodeString("\\u200E-23"),                             -23 },
7254         { "he",                 TRUE,   CharsToUnicodeString("\\u200E- 23"),                            -23 },
7255 
7256         { "ar",                 FALSE,  CharsToUnicodeString("\\u0663\\u0664"),                          34 },
7257         { "ar",                 TRUE,   CharsToUnicodeString("\\u0663\\u0664"),                          34 },
7258         { "ar",                 FALSE,  CharsToUnicodeString("-\\u0664\\u0665"),                        -45 },
7259         { "ar",                 TRUE,   CharsToUnicodeString("-\\u0664\\u0665"),                        -45 },
7260         { "ar",                 TRUE,   CharsToUnicodeString("- \\u0664\\u0665"),                       -45 },
7261         { "ar",                 FALSE,  CharsToUnicodeString("\\u200F-\\u0664\\u0665"),                 -45 },
7262         { "ar",                 TRUE,   CharsToUnicodeString("\\u200F-\\u0664\\u0665"),                 -45 },
7263         { "ar",                 TRUE,   CharsToUnicodeString("\\u200F- \\u0664\\u0665"),                -45 },
7264 
7265         { "ar_MA",              FALSE,  CharsToUnicodeString("12"),                                      12 },
7266         { "ar_MA",              TRUE,   CharsToUnicodeString("12"),                                      12 },
7267         { "ar_MA",              FALSE,  CharsToUnicodeString("-23"),                                    -23 },
7268         { "ar_MA",              TRUE,   CharsToUnicodeString("-23"),                                    -23 },
7269         { "ar_MA",              TRUE,   CharsToUnicodeString("- 23"),                                   -23 },
7270         { "ar_MA",              FALSE,  CharsToUnicodeString("\\u200E-23"),                             -23 },
7271         { "ar_MA",              TRUE,   CharsToUnicodeString("\\u200E-23"),                             -23 },
7272         { "ar_MA",              TRUE,   CharsToUnicodeString("\\u200E- 23"),                            -23 },
7273 
7274         { "fa",                 FALSE,  CharsToUnicodeString("\\u06F5\\u06F6"),                          56 },
7275         { "fa",                 TRUE,   CharsToUnicodeString("\\u06F5\\u06F6"),                          56 },
7276         { "fa",                 FALSE,  CharsToUnicodeString("\\u2212\\u06F6\\u06F7"),                  -67 },
7277         { "fa",                 TRUE,   CharsToUnicodeString("\\u2212\\u06F6\\u06F7"),                  -67 },
7278         { "fa",                 TRUE,   CharsToUnicodeString("\\u2212 \\u06F6\\u06F7"),                 -67 },
7279         { "fa",                 FALSE,  CharsToUnicodeString("\\u200E\\u2212\\u200E\\u06F6\\u06F7"),    -67 },
7280         { "fa",                 TRUE,   CharsToUnicodeString("\\u200E\\u2212\\u200E\\u06F6\\u06F7"),    -67 },
7281         { "fa",                 TRUE,   CharsToUnicodeString("\\u200E\\u2212\\u200E \\u06F6\\u06F7"),   -67 },
7282 
7283         { "ps",                 FALSE,  CharsToUnicodeString("\\u06F5\\u06F6"),                          56 },
7284         { "ps",                 TRUE,   CharsToUnicodeString("\\u06F5\\u06F6"),                          56 },
7285         { "ps",                 FALSE,  CharsToUnicodeString("-\\u06F6\\u06F7"),                        -67 },
7286         { "ps",                 TRUE,   CharsToUnicodeString("-\\u06F6\\u06F7"),                        -67 },
7287         { "ps",                 TRUE,   CharsToUnicodeString("- \\u06F6\\u06F7"),                       -67 },
7288         { "ps",                 FALSE,  CharsToUnicodeString("\\u200E-\\u200E\\u06F6\\u06F7"),          -67 },
7289         { "ps",                 TRUE,   CharsToUnicodeString("\\u200E-\\u200E\\u06F6\\u06F7"),          -67 },
7290         { "ps",                 TRUE,   CharsToUnicodeString("\\u200E-\\u200E \\u06F6\\u06F7"),         -67 },
7291         { "ps",                 FALSE,  CharsToUnicodeString("-\\u200E\\u06F6\\u06F7"),                 -67 },
7292         { "ps",                 TRUE,   CharsToUnicodeString("-\\u200E\\u06F6\\u06F7"),                 -67 },
7293         { "ps",                 TRUE,   CharsToUnicodeString("-\\u200E \\u06F6\\u06F7"),                -67 },
7294         // terminator
7295         { NULL,                 0,      UnicodeString(""),                                                0 },
7296     };
7297 
7298     const SignsAndMarksItem * itemPtr;
7299     for (itemPtr = items; itemPtr->locale != NULL; itemPtr++ ) {
7300         UErrorCode status = U_ZERO_ERROR;
7301         NumberFormat *numfmt = NumberFormat::createInstance(Locale(itemPtr->locale), status);
7302         if (U_SUCCESS(status)) {
7303             numfmt->setLenient(itemPtr->lenient);
7304             Formattable fmtobj;
7305             ParsePosition ppos;
7306             numfmt->parse(itemPtr->numString, fmtobj, ppos);
7307             if (ppos.getIndex() == itemPtr->numString.length()) {
7308                 double parsedValue = fmtobj.getDouble(status);
7309                 if (U_FAILURE(status) || parsedValue != itemPtr->value) {
7310                     errln((UnicodeString)"FAIL: locale " + itemPtr->locale + ", lenient " + itemPtr->lenient + ", parse of \"" + itemPtr->numString + "\" gives value " + parsedValue);
7311                 }
7312             } else {
7313                 errln((UnicodeString)"FAIL: locale " + itemPtr->locale + ", lenient " + itemPtr->lenient + ", parse of \"" + itemPtr->numString + "\" gives position " + ppos.getIndex());
7314             }
7315         } else {
7316             dataerrln("FAIL: NumberFormat::createInstance for locale % gives error %s", itemPtr->locale, u_errorName(status));
7317         }
7318         delete numfmt;
7319     }
7320 }
7321 
7322 typedef struct {
7323   DecimalFormat::ERoundingMode mode;
7324   double value;
7325   UnicodeString expected;
7326 } Test10419Data;
7327 
7328 
7329 // Tests that rounding works right when fractional digits is set to 0.
Test10419RoundingWith0FractionDigits()7330 void NumberFormatTest::Test10419RoundingWith0FractionDigits() {
7331     const Test10419Data items[] = {
7332         { DecimalFormat::kRoundCeiling, 1.488,  "2"},
7333         { DecimalFormat::kRoundDown, 1.588,  "1"},
7334         { DecimalFormat::kRoundFloor, 1.888,  "1"},
7335         { DecimalFormat::kRoundHalfDown, 1.5,  "1"},
7336         { DecimalFormat::kRoundHalfEven, 2.5,  "2"},
7337         { DecimalFormat::kRoundHalfUp, 2.5,  "3"},
7338         { DecimalFormat::kRoundUp, 1.5,  "2"},
7339     };
7340     UErrorCode status = U_ZERO_ERROR;
7341     LocalPointer<DecimalFormat> decfmt((DecimalFormat *) NumberFormat::createInstance(Locale("en_US"), status));
7342     if (U_FAILURE(status)) {
7343         dataerrln("Failure creating DecimalFormat %s", u_errorName(status));
7344         return;
7345     }
7346     for (int32_t i = 0; i < (int32_t) (sizeof(items) / sizeof(items[0])); ++i) {
7347         decfmt->setRoundingMode(items[i].mode);
7348         decfmt->setMaximumFractionDigits(0);
7349         UnicodeString actual;
7350         if (items[i].expected != decfmt->format(items[i].value, actual)) {
7351             errln("Expected " + items[i].expected + ", got " + actual);
7352         }
7353     }
7354 }
7355 
Test10468ApplyPattern()7356 void NumberFormatTest::Test10468ApplyPattern() {
7357     // Padding char of fmt is now 'a'
7358     UErrorCode status = U_ZERO_ERROR;
7359     DecimalFormat fmt("'I''ll'*a###.##", status);
7360 
7361     if (U_FAILURE(status)) {
7362         errcheckln(status, "DecimalFormat constructor failed - %s", u_errorName(status));
7363         return;
7364     }
7365 
7366     if (fmt.getPadCharacterString() != UnicodeString("a")) {
7367         errln("Padding character should be 'a'.");
7368         return;
7369     }
7370 
7371     // Padding char of fmt ought to be '*' since that is the default and no
7372     // explicit padding char is specified in the new pattern.
7373     fmt.applyPattern("AA#,##0.00ZZ", status);
7374 
7375     // Oops this still prints 'a' even though we changed the pattern.
7376     if (fmt.getPadCharacterString() != UnicodeString("*")) {
7377         errln("applyPattern did not clear padding character.");
7378     }
7379 }
7380 
TestRoundingScientific10542()7381 void NumberFormatTest::TestRoundingScientific10542() {
7382     UErrorCode status = U_ZERO_ERROR;
7383     DecimalFormat format("0.00E0", status);
7384     if (U_FAILURE(status)) {
7385         errcheckln(status, "DecimalFormat constructor failed - %s", u_errorName(status));
7386         return;
7387     }
7388 
7389     DecimalFormat::ERoundingMode roundingModes[] = {
7390             DecimalFormat::kRoundCeiling,
7391             DecimalFormat::kRoundDown,
7392             DecimalFormat::kRoundFloor,
7393             DecimalFormat::kRoundHalfDown,
7394             DecimalFormat::kRoundHalfEven,
7395             DecimalFormat::kRoundHalfUp,
7396             DecimalFormat::kRoundUp};
7397     const char *descriptions[] = {
7398             "Round Ceiling",
7399             "Round Down",
7400             "Round Floor",
7401             "Round half down",
7402             "Round half even",
7403             "Round half up",
7404             "Round up"};
7405 
7406     {
7407         double values[] = {-0.003006, -0.003005, -0.003004, 0.003014, 0.003015, 0.003016};
7408         // The order of these expected values correspond to the order of roundingModes and the order of values.
7409         const char *expected[] = {
7410                 "-3.00E-3", "-3.00E-3", "-3.00E-3", "3.02E-3", "3.02E-3", "3.02E-3",
7411                 "-3.00E-3", "-3.00E-3", "-3.00E-3", "3.01E-3", "3.01E-3", "3.01E-3",
7412                 "-3.01E-3", "-3.01E-3", "-3.01E-3", "3.01E-3", "3.01E-3", "3.01E-3",
7413                 "-3.01E-3", "-3.00E-3", "-3.00E-3", "3.01E-3", "3.01E-3", "3.02E-3",
7414                 "-3.01E-3", "-3.00E-3", "-3.00E-3", "3.01E-3", "3.02E-3", "3.02E-3",
7415                 "-3.01E-3", "-3.01E-3", "-3.00E-3", "3.01E-3", "3.02E-3", "3.02E-3",
7416                 "-3.01E-3", "-3.01E-3", "-3.01E-3", "3.02E-3", "3.02E-3", "3.02E-3"};
7417         verifyRounding(
7418                 format,
7419                 values,
7420                 expected,
7421                 roundingModes,
7422                 descriptions,
7423                 (int32_t) (sizeof(values) / sizeof(values[0])),
7424                 (int32_t) (sizeof(roundingModes) / sizeof(roundingModes[0])));
7425     }
7426     {
7427         double values[] = {-3006.0, -3005, -3004, 3014, 3015, 3016};
7428         // The order of these expected values correspond to the order of roundingModes and the order of values.
7429         const char *expected[] = {
7430                 "-3.00E3", "-3.00E3", "-3.00E3", "3.02E3", "3.02E3", "3.02E3",
7431                 "-3.00E3", "-3.00E3", "-3.00E3", "3.01E3", "3.01E3", "3.01E3",
7432                 "-3.01E3", "-3.01E3", "-3.01E3", "3.01E3", "3.01E3", "3.01E3",
7433                 "-3.01E3", "-3.00E3", "-3.00E3", "3.01E3", "3.01E3", "3.02E3",
7434                 "-3.01E3", "-3.00E3", "-3.00E3", "3.01E3", "3.02E3", "3.02E3",
7435                 "-3.01E3", "-3.01E3", "-3.00E3", "3.01E3", "3.02E3", "3.02E3",
7436                 "-3.01E3", "-3.01E3", "-3.01E3", "3.02E3", "3.02E3", "3.02E3"};
7437         verifyRounding(
7438                 format,
7439                 values,
7440                 expected,
7441                 roundingModes,
7442                 descriptions,
7443                 (int32_t) (sizeof(values) / sizeof(values[0])),
7444                 (int32_t) (sizeof(roundingModes) / sizeof(roundingModes[0])));
7445     }
7446 /* Commented out for now until we decide how rounding to zero should work, +0 vs. -0
7447     {
7448         double values[] = {0.0, -0.0};
7449         // The order of these expected values correspond to the order of roundingModes and the order of values.
7450         const char *expected[] = {
7451                 "0.00E0", "-0.00E0",
7452                 "0.00E0", "-0.00E0",
7453                 "0.00E0", "-0.00E0",
7454                 "0.00E0", "-0.00E0",
7455                 "0.00E0", "-0.00E0",
7456                 "0.00E0", "-0.00E0",
7457                 "0.00E0", "-0.00E0"};
7458         verifyRounding(
7459                 format,
7460                 values,
7461                 expected,
7462                 roundingModes,
7463                 descriptions,
7464                 (int32_t) (sizeof(values) / sizeof(values[0])),
7465                 (int32_t) (sizeof(roundingModes) / sizeof(roundingModes[0])));
7466     }
7467 */
7468     {
7469 
7470         double values[] = {1e25, 1e25 + 1e15, 1e25 - 1e15};
7471         // The order of these expected values correspond to the order of roundingModes and the order of values.
7472         const char *expected[] = {
7473                 "1.00E25", "1.01E25", "1.00E25",
7474                 "1.00E25", "1.00E25", "9.99E24",
7475                 "1.00E25", "1.00E25", "9.99E24",
7476                 "1.00E25", "1.00E25", "1.00E25",
7477                 "1.00E25", "1.00E25", "1.00E25",
7478                 "1.00E25", "1.00E25", "1.00E25",
7479                 "1.00E25", "1.01E25", "1.00E25"};
7480         verifyRounding(
7481                 format,
7482                 values,
7483                 expected,
7484                 roundingModes,
7485                 descriptions,
7486                 (int32_t) (sizeof(values) / sizeof(values[0])),
7487                 (int32_t) (sizeof(roundingModes) / sizeof(roundingModes[0])));
7488         }
7489     {
7490         double values[] = {-1e25, -1e25 + 1e15, -1e25 - 1e15};
7491         // The order of these expected values correspond to the order of roundingModes and the order of values.
7492         const char *expected[] = {
7493                 "-1.00E25", "-9.99E24", "-1.00E25",
7494                 "-1.00E25", "-9.99E24", "-1.00E25",
7495                 "-1.00E25", "-1.00E25", "-1.01E25",
7496                 "-1.00E25", "-1.00E25", "-1.00E25",
7497                 "-1.00E25", "-1.00E25", "-1.00E25",
7498                 "-1.00E25", "-1.00E25", "-1.00E25",
7499                 "-1.00E25", "-1.00E25", "-1.01E25"};
7500         verifyRounding(
7501                 format,
7502                 values,
7503                 expected,
7504                 roundingModes,
7505                 descriptions,
7506                 (int32_t) (sizeof(values) / sizeof(values[0])),
7507                 (int32_t) (sizeof(roundingModes) / sizeof(roundingModes[0])));
7508         }
7509     {
7510         double values[] = {1e-25, 1e-25 + 1e-35, 1e-25 - 1e-35};
7511         // The order of these expected values correspond to the order of roundingModes and the order of values.
7512         const char *expected[] = {
7513                 "1.00E-25", "1.01E-25", "1.00E-25",
7514                 "1.00E-25", "1.00E-25", "9.99E-26",
7515                 "1.00E-25", "1.00E-25", "9.99E-26",
7516                 "1.00E-25", "1.00E-25", "1.00E-25",
7517                 "1.00E-25", "1.00E-25", "1.00E-25",
7518                 "1.00E-25", "1.00E-25", "1.00E-25",
7519                 "1.00E-25", "1.01E-25", "1.00E-25"};
7520         verifyRounding(
7521                 format,
7522                 values,
7523                 expected,
7524                 roundingModes,
7525                 descriptions,
7526                 (int32_t) (sizeof(values) / sizeof(values[0])),
7527                 (int32_t) (sizeof(roundingModes) / sizeof(roundingModes[0])));
7528         }
7529     {
7530         double values[] = {-1e-25, -1e-25 + 1e-35, -1e-25 - 1e-35};
7531         // The order of these expected values correspond to the order of roundingModes and the order of values.
7532         const char *expected[] = {
7533                 "-1.00E-25", "-9.99E-26", "-1.00E-25",
7534                 "-1.00E-25", "-9.99E-26", "-1.00E-25",
7535                 "-1.00E-25", "-1.00E-25", "-1.01E-25",
7536                 "-1.00E-25", "-1.00E-25", "-1.00E-25",
7537                 "-1.00E-25", "-1.00E-25", "-1.00E-25",
7538                 "-1.00E-25", "-1.00E-25", "-1.00E-25",
7539                 "-1.00E-25", "-1.00E-25", "-1.01E-25"};
7540         verifyRounding(
7541                 format,
7542                 values,
7543                 expected,
7544                 roundingModes,
7545                 descriptions,
7546                 (int32_t) (sizeof(values) / sizeof(values[0])),
7547                 (int32_t) (sizeof(roundingModes) / sizeof(roundingModes[0])));
7548     }
7549 }
7550 
TestZeroScientific10547()7551 void NumberFormatTest::TestZeroScientific10547() {
7552     UErrorCode status = U_ZERO_ERROR;
7553     DecimalFormat fmt("0.00E0", status);
7554     if (!assertSuccess("Formt creation", status)) {
7555         return;
7556     }
7557     UnicodeString out;
7558     fmt.format(-0.0, out);
7559     assertEquals("format", "-0.00E0", out);
7560 }
7561 
verifyRounding(DecimalFormat & format,const double * values,const char * const * expected,const DecimalFormat::ERoundingMode * roundingModes,const char * const * descriptions,int32_t valueSize,int32_t roundingModeSize)7562 void NumberFormatTest::verifyRounding(
7563         DecimalFormat& format,
7564         const double *values,
7565         const char * const *expected,
7566         const DecimalFormat::ERoundingMode *roundingModes,
7567         const char * const *descriptions,
7568         int32_t valueSize,
7569         int32_t roundingModeSize) {
7570     for (int32_t i = 0; i < roundingModeSize; ++i) {
7571         format.setRoundingMode(roundingModes[i]);
7572         for (int32_t j = 0; j < valueSize; j++) {
7573             UnicodeString currentExpected(expected[i * valueSize + j]);
7574             currentExpected = currentExpected.unescape();
7575             UnicodeString actual;
7576             format.format(values[j], actual);
7577             if (currentExpected != actual) {
7578                 char buffer[256];
7579                 sprintf(
7580                         buffer,
7581                         "For %s value %f, expected ",
7582                         descriptions[i],
7583                         values[j]);
7584                 errln(UnicodeString(buffer) + currentExpected + ", got " + actual);
7585             }
7586         }
7587     }
7588 }
7589 
TestAccountingCurrency()7590 void NumberFormatTest::TestAccountingCurrency() {
7591     UErrorCode status = U_ZERO_ERROR;
7592     UNumberFormatStyle style = UNUM_CURRENCY_ACCOUNTING;
7593 
7594     expect(NumberFormat::createInstance("en_US", style, status),
7595         (Formattable)1234.5, "$1,234.50", TRUE, status);
7596     expect(NumberFormat::createInstance("en_US", style, status),
7597         (Formattable)-1234.5, "($1,234.50)", TRUE, status);
7598     expect(NumberFormat::createInstance("en_US", style, status),
7599         (Formattable)0, "$0.00", TRUE, status);
7600     expect(NumberFormat::createInstance("en_US", style, status),
7601         (Formattable)-0.2, "($0.20)", TRUE, status);
7602     expect(NumberFormat::createInstance("ja_JP", style, status),
7603         (Formattable)10000, UnicodeString("\\uFFE510,000").unescape(), TRUE, status);
7604     expect(NumberFormat::createInstance("ja_JP", style, status),
7605         (Formattable)-1000.5, UnicodeString("(\\uFFE51,000)").unescape(), FALSE, status);
7606     expect(NumberFormat::createInstance("de_DE", style, status),
7607         (Formattable)-23456.7, UnicodeString("-23.456,70\\u00A0\\u20AC").unescape(), TRUE, status);
7608 }
7609 
7610 // for #5186
TestEquality()7611 void NumberFormatTest::TestEquality() {
7612     UErrorCode status = U_ZERO_ERROR;
7613     DecimalFormatSymbols* symbols = new DecimalFormatSymbols(Locale("root"), status);
7614     if (U_FAILURE(status)) {
7615     	dataerrln("Fail: can't create DecimalFormatSymbols for root");
7616     	return;
7617     }
7618     UnicodeString pattern("#,##0.###");
7619     DecimalFormat* fmtBase = new DecimalFormat(pattern, symbols, status);
7620     if (U_FAILURE(status)) {
7621     	dataerrln("Fail: can't create DecimalFormat using root symbols");
7622     	return;
7623     }
7624 
7625     DecimalFormat* fmtClone = (DecimalFormat*)fmtBase->clone();
7626     fmtClone->setFormatWidth(fmtBase->getFormatWidth() + 32);
7627     if (*fmtClone == *fmtBase) {
7628         errln("Error: DecimalFormat == does not distinguish objects that differ only in FormatWidth");
7629     }
7630     delete fmtClone;
7631 
7632     delete fmtBase;
7633 }
7634 
TestCurrencyUsage()7635 void NumberFormatTest::TestCurrencyUsage() {
7636     double agent = 123.567;
7637 
7638     UErrorCode status;
7639     DecimalFormat *fmt;
7640 
7641     // compare the Currency and Currency Cash Digits
7642     // Note that as of CLDR 26:
7643     // * TWD switches from 0 decimals to 2; PKR still has 0, so change test to that
7644     // * CAD and all other currencies that rounded to .05 no longer do
7645     // 1st time for getter/setter, 2nd time for factory method
7646     Locale enUS_PKR("en_US@currency=PKR");
7647 
7648     for(int i=0; i<2; i++){
7649         status = U_ZERO_ERROR;
7650         if(i == 0){
7651             fmt = (DecimalFormat *) NumberFormat::createInstance(enUS_PKR, UNUM_CURRENCY, status);
7652             if (assertSuccess("en_US@currency=PKR/CURRENCY", status, TRUE) == FALSE) {
7653                 continue;
7654             }
7655 
7656             UnicodeString original;
7657             fmt->format(agent,original);
7658             assertEquals("Test Currency Usage 1", UnicodeString("PKR124"), original);
7659 
7660             // test the getter here
7661             UCurrencyUsage curUsage = fmt->getCurrencyUsage();
7662             assertEquals("Test usage getter - standard", curUsage, UCURR_USAGE_STANDARD);
7663 
7664             fmt->setCurrencyUsage(UCURR_USAGE_CASH, &status);
7665         }else{
7666             fmt = (DecimalFormat *) NumberFormat::createInstance(enUS_PKR, UNUM_CASH_CURRENCY, status);
7667             if (assertSuccess("en_US@currency=PKR/CASH", status, TRUE) == FALSE) {
7668                 continue;
7669             }
7670         }
7671 
7672         // must be usage = cash
7673         UCurrencyUsage curUsage = fmt->getCurrencyUsage();
7674         assertEquals("Test usage getter - cash", curUsage, UCURR_USAGE_CASH);
7675 
7676         UnicodeString cash_currency;
7677         fmt->format(agent,cash_currency);
7678         assertEquals("Test Currency Usage 2", UnicodeString("PKR124"), cash_currency);
7679         delete fmt;
7680     }
7681 
7682     // compare the Currency and Currency Cash Rounding
7683     // 1st time for getter/setter, 2nd time for factory method
7684     Locale enUS_CAD("en_US@currency=CAD");
7685     for(int i=0; i<2; i++){
7686         status = U_ZERO_ERROR;
7687         if(i == 0){
7688             fmt = (DecimalFormat *) NumberFormat::createInstance(enUS_CAD, UNUM_CURRENCY, status);
7689             if (assertSuccess("en_US@currency=CAD/CURRENCY", status, TRUE) == FALSE) {
7690                 continue;
7691             }
7692 
7693             UnicodeString original_rounding;
7694             fmt->format(agent, original_rounding);
7695             assertEquals("Test Currency Usage 3", UnicodeString("CA$123.57"), original_rounding);
7696             fmt->setCurrencyUsage(UCURR_USAGE_CASH, &status);
7697         }else{
7698             fmt = (DecimalFormat *) NumberFormat::createInstance(enUS_CAD, UNUM_CASH_CURRENCY, status);
7699             if (assertSuccess("en_US@currency=CAD/CASH", status, TRUE) == FALSE) {
7700                 continue;
7701             }
7702         }
7703 
7704         UnicodeString cash_rounding_currency;
7705         fmt->format(agent, cash_rounding_currency);
7706         assertEquals("Test Currency Usage 4", UnicodeString("CA$123.57"), cash_rounding_currency);
7707         delete fmt;
7708     }
7709 
7710     // Test the currency change
7711     // 1st time for getter/setter, 2nd time for factory method
7712     const UChar CUR_PKR[] = {0x50, 0x4B, 0x52, 0};
7713     for(int i=0; i<2; i++){
7714         status = U_ZERO_ERROR;
7715         if(i == 0){
7716             fmt = (DecimalFormat *) NumberFormat::createInstance(enUS_CAD, UNUM_CURRENCY, status);
7717             if (assertSuccess("en_US@currency=CAD/CURRENCY", status, TRUE) == FALSE) {
7718                 continue;
7719             }
7720             fmt->setCurrencyUsage(UCURR_USAGE_CASH, &status);
7721         }else{
7722             fmt = (DecimalFormat *) NumberFormat::createInstance(enUS_CAD, UNUM_CASH_CURRENCY, status);
7723             if (assertSuccess("en_US@currency=CAD/CASH", status, TRUE) == FALSE) {
7724                 continue;
7725             }
7726         }
7727 
7728         UnicodeString cur_original;
7729         fmt->format(agent, cur_original);
7730         assertEquals("Test Currency Usage 5", UnicodeString("CA$123.57"), cur_original);
7731 
7732         fmt->setCurrency(CUR_PKR, status);
7733         assertSuccess("Set currency to PKR", status);
7734 
7735         UnicodeString PKR_changed;
7736         fmt->format(agent, PKR_changed);
7737         assertEquals("Test Currency Usage 6", UnicodeString("PKR124"), PKR_changed);
7738         delete fmt;
7739     }
7740 }
7741 
7742 
7743 // Check the constant MAX_INT64_IN_DOUBLE.
7744 // The value should convert to a double with no loss of precision.
7745 // A failure may indicate a platform with a different double format, requiring
7746 // a revision to the constant.
7747 //
7748 // Note that this is actually hard to test, because the language standard gives
7749 //  compilers considerable flexibility to do unexpected things with rounding and
7750 //  with overflow in simple int to/from float conversions. Some compilers will completely optimize
7751 //  away a simple round-trip conversion from int64_t -> double -> int64_t.
7752 
TestDoubleLimit11439()7753 void NumberFormatTest::TestDoubleLimit11439() {
7754     char  buf[50];
7755     for (int64_t num = MAX_INT64_IN_DOUBLE-10; num<=MAX_INT64_IN_DOUBLE; num++) {
7756         sprintf(buf, "%lld", (long long)num);
7757         double fNum = 0.0;
7758         sscanf(buf, "%lf", &fNum);
7759         int64_t rtNum = fNum;
7760         if (num != rtNum) {
7761             errln("%s:%d MAX_INT64_IN_DOUBLE test, %lld did not round trip. Got %lld", __FILE__, __LINE__, (long long)num, (long long)rtNum);
7762             return;
7763         }
7764     }
7765     for (int64_t num = -MAX_INT64_IN_DOUBLE+10; num>=-MAX_INT64_IN_DOUBLE; num--) {
7766         sprintf(buf, "%lld", (long long)num);
7767         double fNum = 0.0;
7768         sscanf(buf, "%lf", &fNum);
7769         int64_t rtNum = fNum;
7770         if (num != rtNum) {
7771             errln("%s:%d MAX_INT64_IN_DOUBLE test, %lld did not round trip. Got %lld", __FILE__, __LINE__, (long long)num, (long long)rtNum);
7772             return;
7773         }
7774     }
7775 }
7776 
TestFastPathConsistent11524()7777 void NumberFormatTest::TestFastPathConsistent11524() {
7778     UErrorCode status = U_ZERO_ERROR;
7779     NumberFormat *fmt = NumberFormat::createInstance("en", status);
7780     if (U_FAILURE(status) || fmt == NULL) {
7781         dataerrln("Failed call to NumberFormat::createInstance() - %s", u_errorName(status));
7782         return;
7783     }
7784     fmt->setMaximumIntegerDigits(INT32_MIN);
7785     UnicodeString appendTo;
7786     assertEquals("", "0", fmt->format(123, appendTo));
7787     appendTo.remove();
7788     assertEquals("", "0", fmt->format(12345, appendTo));
7789     delete fmt;
7790 }
7791 
7792 
7793 #endif /* #if !UCONFIG_NO_FORMATTING */
7794