1 /********************************************************************
2 * Copyright (c) 2008-2015, International Business Machines Corporation and
3 * others. All Rights Reserved.
4 ********************************************************************/
5
6 #include "unicode/utypes.h"
7
8 #if !UCONFIG_NO_FORMATTING
9
10 #include "unicode/decimfmt.h"
11 #include "unicode/tmunit.h"
12 #include "unicode/tmutamt.h"
13 #include "unicode/tmutfmt.h"
14 #include "unicode/ustring.h"
15 #include "cmemory.h"
16 #include "intltest.h"
17
18 //TODO: put as compilation flag
19 //#define TUFMTTS_DEBUG 1
20
21 #ifdef TUFMTTS_DEBUG
22 #include <iostream>
23 #endif
24
25 class TimeUnitTest : public IntlTest {
runIndexedTest(int32_t index,UBool exec,const char * & name,char *)26 void runIndexedTest(int32_t index, UBool exec, const char* &name, char* /*par*/ ) {
27 if (exec) logln("TestSuite TimeUnitTest");
28 TESTCASE_AUTO_BEGIN;
29 TESTCASE_AUTO(testBasic);
30 TESTCASE_AUTO(testAPI);
31 TESTCASE_AUTO(testGreekWithFallback);
32 TESTCASE_AUTO(testGreekWithSanitization);
33 TESTCASE_AUTO(test10219Plurals);
34 TESTCASE_AUTO(TestBritishShortHourFallback);
35 TESTCASE_AUTO_END;
36 }
37
38 public:
39 /**
40 * Performs basic tests
41 **/
42 void testBasic();
43
44 /**
45 * Performs API tests
46 **/
47 void testAPI();
48
49 /**
50 * Performs tests for Greek
51 * This tests that requests for short unit names correctly fall back
52 * to long unit names for a locale where the locale data does not
53 * provide short unit names. As of CLDR 1.9, Greek is one such language.
54 **/
55 void testGreekWithFallback();
56
57 /**
58 * Performs tests for Greek
59 * This tests that if the plural count listed in time unit format does not
60 * match those in the plural rules for the locale, those plural count in
61 * time unit format will be ingored and subsequently, fall back will kick in
62 * which is tested above.
63 * Without data sanitization, setNumberFormat() would crash.
64 * As of CLDR shiped in ICU4.8, Greek is one such language.
65 */
66 void testGreekWithSanitization();
67
68 /**
69 * Performs unit test for ticket 10219 making sure that plurals work
70 * correctly with rounding.
71 */
72 void test10219Plurals();
73
74 void TestBritishShortHourFallback();
75 };
76
createTimeUnitTest()77 extern IntlTest *createTimeUnitTest() {
78 return new TimeUnitTest();
79 }
80
81 // This function is more lenient than equals operator as it considers integer 3 hours and
82 // double 3.0 hours to be equal
tmaEqual(const TimeUnitAmount & left,const TimeUnitAmount & right)83 static UBool tmaEqual(const TimeUnitAmount& left, const TimeUnitAmount& right) {
84 if (left.getTimeUnitField() != right.getTimeUnitField()) {
85 return FALSE;
86 }
87 UErrorCode status = U_ZERO_ERROR;
88 if (!left.getNumber().isNumeric() || !right.getNumber().isNumeric()) {
89 return FALSE;
90 }
91 UBool result = left.getNumber().getDouble(status) == right.getNumber().getDouble(status);
92 if (U_FAILURE(status)) {
93 return FALSE;
94 }
95 return result;
96 }
97
98 /**
99 * Test basic
100 */
testBasic()101 void TimeUnitTest::testBasic() {
102 const char* locales[] = {"en", "sl", "fr", "zh", "ar", "ru", "zh_Hant", "pa"};
103 for ( unsigned int locIndex = 0;
104 locIndex < sizeof(locales)/sizeof(locales[0]);
105 ++locIndex ) {
106 UErrorCode status = U_ZERO_ERROR;
107 Locale loc(locales[locIndex]);
108 TimeUnitFormat** formats = new TimeUnitFormat*[2];
109 formats[UTMUTFMT_FULL_STYLE] = new TimeUnitFormat(loc, status);
110 if (!assertSuccess("TimeUnitFormat(full)", status, TRUE)) return;
111 formats[UTMUTFMT_ABBREVIATED_STYLE] = new TimeUnitFormat(loc, UTMUTFMT_ABBREVIATED_STYLE, status);
112 if (!assertSuccess("TimeUnitFormat(short)", status)) return;
113 #ifdef TUFMTTS_DEBUG
114 std::cout << "locale: " << locales[locIndex] << "\n";
115 #endif
116 for (int style = UTMUTFMT_FULL_STYLE;
117 style <= UTMUTFMT_ABBREVIATED_STYLE;
118 ++style) {
119 for (TimeUnit::UTimeUnitFields j = TimeUnit::UTIMEUNIT_YEAR;
120 j < TimeUnit::UTIMEUNIT_FIELD_COUNT;
121 j = (TimeUnit::UTimeUnitFields)(j+1)) {
122 #ifdef TUFMTTS_DEBUG
123 std::cout << "time unit: " << j << "\n";
124 #endif
125 double tests[] = {0, 0.5, 1, 1.5, 2, 2.5, 3, 3.5, 5, 10, 100, 101.35};
126 for (unsigned int i = 0; i < sizeof(tests)/sizeof(tests[0]); ++i) {
127 #ifdef TUFMTTS_DEBUG
128 std::cout << "number: " << tests[i] << "\n";
129 #endif
130 TimeUnitAmount* source = new TimeUnitAmount(tests[i], j, status);
131 if (!assertSuccess("TimeUnitAmount()", status)) return;
132 UnicodeString formatted;
133 Formattable formattable;
134 formattable.adoptObject(source);
135 formatted = ((Format*)formats[style])->format(formattable, formatted, status);
136 if (!assertSuccess("format()", status)) return;
137 #ifdef TUFMTTS_DEBUG
138 char formatResult[1000];
139 formatted.extract(0, formatted.length(), formatResult, "UTF-8");
140 std::cout << "format result: " << formatResult << "\n";
141 #endif
142 Formattable result;
143 ((Format*)formats[style])->parseObject(formatted, result, status);
144 if (!assertSuccess("parseObject()", status)) return;
145 if (!tmaEqual(*((TimeUnitAmount *)result.getObject()), *((TimeUnitAmount *) formattable.getObject()))) {
146 dataerrln("No round trip: ");
147 }
148 // other style parsing
149 Formattable result_1;
150 ((Format*)formats[1-style])->parseObject(formatted, result_1, status);
151 if (!assertSuccess("parseObject()", status)) return;
152 if (!tmaEqual(*((TimeUnitAmount *)result_1.getObject()), *((TimeUnitAmount *) formattable.getObject()))) {
153 dataerrln("No round trip: ");
154 }
155 }
156 }
157 }
158 delete formats[UTMUTFMT_FULL_STYLE];
159 delete formats[UTMUTFMT_ABBREVIATED_STYLE];
160 delete[] formats;
161 }
162 }
163
164
testAPI()165 void TimeUnitTest::testAPI() {
166 //================= TimeUnit =================
167 UErrorCode status = U_ZERO_ERROR;
168
169 TimeUnit* tmunit = TimeUnit::createInstance(TimeUnit::UTIMEUNIT_YEAR, status);
170 if (!assertSuccess("TimeUnit::createInstance", status)) return;
171
172 TimeUnit* another = (TimeUnit*)tmunit->clone();
173 TimeUnit third(*tmunit);
174 TimeUnit fourth = third;
175
176 assertTrue("orig and clone are equal", (*tmunit == *another));
177 assertTrue("copied and assigned are equal", (third == fourth));
178
179 TimeUnit* tmunit_m = TimeUnit::createInstance(TimeUnit::UTIMEUNIT_MONTH, status);
180 assertTrue("year != month", (*tmunit != *tmunit_m));
181
182 TimeUnit::UTimeUnitFields field = tmunit_m->getTimeUnitField();
183 assertTrue("field of month time unit is month", (field == TimeUnit::UTIMEUNIT_MONTH));
184
185 //===== Interoperability with MeasureUnit ======
186 MeasureUnit **ptrs = new MeasureUnit *[TimeUnit::UTIMEUNIT_FIELD_COUNT];
187
188 ptrs[TimeUnit::UTIMEUNIT_YEAR] = MeasureUnit::createYear(status);
189 ptrs[TimeUnit::UTIMEUNIT_MONTH] = MeasureUnit::createMonth(status);
190 ptrs[TimeUnit::UTIMEUNIT_DAY] = MeasureUnit::createDay(status);
191 ptrs[TimeUnit::UTIMEUNIT_WEEK] = MeasureUnit::createWeek(status);
192 ptrs[TimeUnit::UTIMEUNIT_HOUR] = MeasureUnit::createHour(status);
193 ptrs[TimeUnit::UTIMEUNIT_MINUTE] = MeasureUnit::createMinute(status);
194 ptrs[TimeUnit::UTIMEUNIT_SECOND] = MeasureUnit::createSecond(status);
195 if (!assertSuccess("TimeUnit::createInstance", status)) return;
196
197 for (TimeUnit::UTimeUnitFields j = TimeUnit::UTIMEUNIT_YEAR;
198 j < TimeUnit::UTIMEUNIT_FIELD_COUNT;
199 j = (TimeUnit::UTimeUnitFields)(j+1)) {
200 MeasureUnit *ptr = TimeUnit::createInstance(j, status);
201 if (!assertSuccess("TimeUnit::createInstance", status)) return;
202 // We have to convert *ptr to a MeasureUnit or else == will fail over
203 // differing types (TimeUnit vs. MeasureUnit).
204 assertTrue(
205 "Time unit should be equal to corresponding MeasureUnit",
206 MeasureUnit(*ptr) == *ptrs[j]);
207 delete ptr;
208 }
209 delete tmunit;
210 delete another;
211 delete tmunit_m;
212 for (int i = 0; i < TimeUnit::UTIMEUNIT_FIELD_COUNT; ++i) {
213 delete ptrs[i];
214 }
215 delete [] ptrs;
216
217 //
218 //================= TimeUnitAmount =================
219
220 Formattable formattable((int32_t)2);
221 TimeUnitAmount tma_long(formattable, TimeUnit::UTIMEUNIT_DAY, status);
222 if (!assertSuccess("TimeUnitAmount(formattable...)", status)) return;
223
224 formattable.setDouble(2);
225 TimeUnitAmount tma_double(formattable, TimeUnit::UTIMEUNIT_DAY, status);
226 if (!assertSuccess("TimeUnitAmount(formattable...)", status)) return;
227
228 formattable.setDouble(3);
229 TimeUnitAmount tma_double_3(formattable, TimeUnit::UTIMEUNIT_DAY, status);
230 if (!assertSuccess("TimeUnitAmount(formattable...)", status)) return;
231
232 TimeUnitAmount tma(2, TimeUnit::UTIMEUNIT_DAY, status);
233 if (!assertSuccess("TimeUnitAmount(number...)", status)) return;
234
235 TimeUnitAmount tma_h(2, TimeUnit::UTIMEUNIT_HOUR, status);
236 if (!assertSuccess("TimeUnitAmount(number...)", status)) return;
237
238 TimeUnitAmount second(tma);
239 TimeUnitAmount third_tma = tma;
240 TimeUnitAmount* fourth_tma = (TimeUnitAmount*)tma.clone();
241
242 assertTrue("orig and copy are equal", (second == tma));
243 assertTrue("clone and assigned are equal", (third_tma == *fourth_tma));
244 assertTrue("different if number diff", (tma_double != tma_double_3));
245 assertTrue("different if number type diff", (tma_double != tma_long));
246 assertTrue("different if time unit diff", (tma != tma_h));
247 assertTrue("same even different constructor", (tma_double == tma));
248
249 assertTrue("getTimeUnitField", (tma.getTimeUnitField() == TimeUnit::UTIMEUNIT_DAY));
250 delete fourth_tma;
251 //
252 //================= TimeUnitFormat =================
253 //
254 TimeUnitFormat* tmf_en = new TimeUnitFormat(Locale("en"), status);
255 if (!assertSuccess("TimeUnitFormat(en...)", status, TRUE)) return;
256 TimeUnitFormat tmf_fr(Locale("fr"), status);
257 if (!assertSuccess("TimeUnitFormat(fr...)", status)) return;
258
259 assertTrue("TimeUnitFormat: en and fr diff", (*tmf_en != tmf_fr));
260
261 TimeUnitFormat tmf_assign = *tmf_en;
262 assertTrue("TimeUnitFormat: orig and assign are equal", (*tmf_en == tmf_assign));
263
264 TimeUnitFormat tmf_copy(tmf_fr);
265 assertTrue("TimeUnitFormat: orig and copy are equal", (tmf_fr == tmf_copy));
266
267 TimeUnitFormat* tmf_clone = (TimeUnitFormat*)tmf_en->clone();
268 assertTrue("TimeUnitFormat: orig and clone are equal", (*tmf_en == *tmf_clone));
269 delete tmf_clone;
270
271 tmf_en->setLocale(Locale("fr"), status);
272 if (!assertSuccess("setLocale(fr...)", status)) return;
273
274 NumberFormat* numberFmt = NumberFormat::createInstance(
275 Locale("fr"), status);
276 if (!assertSuccess("NumberFormat::createInstance()", status)) return;
277 tmf_en->setNumberFormat(*numberFmt, status);
278 if (!assertSuccess("setNumberFormat(en...)", status)) return;
279 assertTrue("TimeUnitFormat: setLocale", (*tmf_en == tmf_fr));
280
281 delete tmf_en;
282
283 TimeUnitFormat* en_long = new TimeUnitFormat(Locale("en"), UTMUTFMT_FULL_STYLE, status);
284 if (!assertSuccess("TimeUnitFormat(en...)", status)) return;
285 delete en_long;
286
287 TimeUnitFormat* en_short = new TimeUnitFormat(Locale("en"), UTMUTFMT_ABBREVIATED_STYLE, status);
288 if (!assertSuccess("TimeUnitFormat(en...)", status)) return;
289 delete en_short;
290
291 TimeUnitFormat* format = new TimeUnitFormat(status);
292 format->setLocale(Locale("zh"), status);
293 format->setNumberFormat(*numberFmt, status);
294 if (!assertSuccess("TimeUnitFormat(en...)", status)) return;
295 delete numberFmt;
296 delete format;
297 }
298
299 /* @bug 7902
300 * Tests for Greek Language.
301 * This tests that requests for short unit names correctly fall back
302 * to long unit names for a locale where the locale data does not
303 * provide short unit names. As of CLDR 1.9, Greek is one such language.
304 */
testGreekWithFallback()305 void TimeUnitTest::testGreekWithFallback() {
306 UErrorCode status = U_ZERO_ERROR;
307
308 const char* locales[] = {"el-GR", "el"};
309 TimeUnit::UTimeUnitFields tunits[] = {TimeUnit::UTIMEUNIT_SECOND, TimeUnit::UTIMEUNIT_MINUTE, TimeUnit::UTIMEUNIT_HOUR, TimeUnit::UTIMEUNIT_DAY, TimeUnit::UTIMEUNIT_MONTH, TimeUnit::UTIMEUNIT_YEAR};
310 UTimeUnitFormatStyle styles[] = {UTMUTFMT_FULL_STYLE, UTMUTFMT_ABBREVIATED_STYLE};
311 const int numbers[] = {1, 7};
312
313 const UChar oneSecond[] = {0x0031, 0x0020, 0x03b4, 0x03b5, 0x03c5, 0x03c4, 0x03b5, 0x03c1, 0x03cc, 0x03bb, 0x03b5, 0x03c0, 0x03c4, 0x03bf, 0};
314 const UChar oneSecondShort[] = {0x0031, 0x0020, 0x03b4, 0x03b5, 0x03c5, 0x03c4, 0x002e, 0};
315 const UChar oneMinute[] = {0x0031, 0x0020, 0x03bb, 0x03b5, 0x03c0, 0x03c4, 0x03cc, 0};
316 const UChar oneMinuteShort[] = {0x0031, 0x0020, 0x03bb, 0x03b5, 0x03c0, 0x002e, 0};
317 const UChar oneHour[] = {0x0031, 0x0020, 0x03ce, 0x03c1, 0x03b1, 0};
318 const UChar oneDay[] = {0x0031, 0x0020, 0x03b7, 0x03bc, 0x03ad, 0x03c1, 0x03b1, 0};
319 const UChar oneMonth[] = {0x0031, 0x0020, 0x03bc, 0x03ae, 0x03bd, 0x03b1, 0x03c2, 0};
320 const UChar oneMonthShort[] = {0x0031, 0x0020, 0x03bc, 0x03ae, 0x03bd, 0x002e, 0};
321 const UChar oneYear[] = {0x0031, 0x0020, 0x03ad, 0x03c4, 0x03bf, 0x03c2, 0};
322 const UChar oneYearShort[] = {0x0031, 0x0020, 0x03ad, 0x03c4, 0x002e, 0};
323 const UChar sevenSeconds[] = {0x0037, 0x0020, 0x03b4, 0x03b5, 0x03c5, 0x03c4, 0x03b5, 0x03c1, 0x03cc, 0x03bb, 0x03b5, 0x03c0, 0x03c4, 0x03b1, 0};
324 const UChar sevenSecondsShort[] = {0x0037, 0x0020, 0x03b4, 0x03b5, 0x03c5, 0x03c4, 0x002e, 0};
325 const UChar sevenMinutes[] = {0x0037, 0x0020, 0x03bb, 0x03b5, 0x03c0, 0x03c4, 0x03ac, 0};
326 const UChar sevenMinutesShort[] = {0x0037, 0x0020, 0x03bb, 0x03b5, 0x03c0, 0x002e, 0};
327 const UChar sevenHours[] = {0x0037, 0x0020, 0x03ce, 0x03c1, 0x03b5, 0x03c2, 0};
328 const UChar sevenHoursShort[] = {0x0037, 0x0020, 0x03ce, 0x03c1, 0x002e, 0};
329 const UChar sevenDays[] = {0x0037, 0x0020, 0x03b7, 0x03bc, 0x03ad, 0x03c1, 0x03b5, 0x03c2, 0};
330 const UChar sevenMonths[] = {0x0037, 0x0020, 0x03bc, 0x03ae, 0x03bd, 0x03b5, 0x3c2, 0};
331 const UChar sevenMonthsShort[] = {0x0037, 0x0020, 0x03bc, 0x03ae, 0x03bd, 0x002e, 0};
332 const UChar sevenYears[] = {0x0037, 0x0020, 0x03ad, 0x03c4, 0x03b7, 0};
333 const UChar sevenYearsShort[] = {0x0037, 0x0020, 0x03ad, 0x03c4, 0x002e, 0};
334
335 const UnicodeString oneSecondStr(oneSecond);
336 const UnicodeString oneSecondShortStr(oneSecondShort);
337 const UnicodeString oneMinuteStr(oneMinute);
338 const UnicodeString oneMinuteShortStr(oneMinuteShort);
339 const UnicodeString oneHourStr(oneHour);
340 const UnicodeString oneDayStr(oneDay);
341 const UnicodeString oneMonthStr(oneMonth);
342 const UnicodeString oneMonthShortStr(oneMonthShort);
343 const UnicodeString oneYearStr(oneYear);
344 const UnicodeString oneYearShortStr(oneYearShort);
345 const UnicodeString sevenSecondsStr(sevenSeconds);
346 const UnicodeString sevenSecondsShortStr(sevenSecondsShort);
347 const UnicodeString sevenMinutesStr(sevenMinutes);
348 const UnicodeString sevenMinutesShortStr(sevenMinutesShort);
349 const UnicodeString sevenHoursStr(sevenHours);
350 const UnicodeString sevenHoursShortStr(sevenHoursShort);
351 const UnicodeString sevenDaysStr(sevenDays);
352 const UnicodeString sevenMonthsStr(sevenMonths);
353 const UnicodeString sevenMonthsShortStr(sevenMonthsShort);
354 const UnicodeString sevenYearsStr(sevenYears);
355 const UnicodeString sevenYearsShortStr(sevenYearsShort);
356
357 const UnicodeString expected[] = {
358 oneSecondStr, oneMinuteStr, oneHourStr, oneDayStr, oneMonthStr, oneYearStr,
359 oneSecondShortStr, oneMinuteShortStr, oneHourStr, oneDayStr, oneMonthShortStr, oneYearShortStr,
360 sevenSecondsStr, sevenMinutesStr, sevenHoursStr, sevenDaysStr, sevenMonthsStr, sevenYearsStr,
361 sevenSecondsShortStr, sevenMinutesShortStr, sevenHoursShortStr, sevenDaysStr, sevenMonthsShortStr, sevenYearsShortStr,
362
363 oneSecondStr, oneMinuteStr, oneHourStr, oneDayStr, oneMonthStr, oneYearStr,
364 oneSecondShortStr, oneMinuteShortStr, oneHourStr, oneDayStr, oneMonthShortStr, oneYearShortStr,
365 sevenSecondsStr, sevenMinutesStr, sevenHoursStr, sevenDaysStr, sevenMonthsStr, sevenYearsStr,
366 sevenSecondsShortStr, sevenMinutesShortStr, sevenHoursShortStr, sevenDaysStr, sevenMonthsShortStr, sevenYearsShortStr};
367
368 int counter = 0;
369 for ( unsigned int locIndex = 0;
370 locIndex < sizeof(locales)/sizeof(locales[0]);
371 ++locIndex ) {
372
373 Locale l = Locale::createFromName(locales[locIndex]);
374
375 for ( unsigned int numberIndex = 0;
376 numberIndex < sizeof(numbers)/sizeof(int);
377 ++numberIndex ) {
378
379 for ( unsigned int styleIndex = 0;
380 styleIndex < sizeof(styles)/sizeof(styles[0]);
381 ++styleIndex ) {
382
383 for ( unsigned int unitIndex = 0;
384 unitIndex < sizeof(tunits)/sizeof(tunits[0]);
385 ++unitIndex ) {
386
387 TimeUnitAmount *tamt = new TimeUnitAmount(numbers[numberIndex], tunits[unitIndex], status);
388 if (U_FAILURE(status)) {
389 dataerrln("generating TimeUnitAmount Object failed.");
390 #ifdef TUFMTTS_DEBUG
391 std::cout << "Failed to get TimeUnitAmount for " << tunits[unitIndex] << "\n";
392 #endif
393 return;
394 }
395
396 TimeUnitFormat *tfmt = new TimeUnitFormat(l, styles[styleIndex], status);
397 if (U_FAILURE(status)) {
398 dataerrln("generating TimeUnitAmount Object failed.");
399 #ifdef TUFMTTS_DEBUG
400 std::cout << "Failed to get TimeUnitFormat for " << locales[locIndex] << "\n";
401 #endif
402 return;
403 }
404
405 Formattable fmt;
406 UnicodeString str;
407
408 fmt.adoptObject(tamt);
409 str = ((Format *)tfmt)->format(fmt, str, status);
410 if (!assertSuccess("formatting relative time failed", status)) {
411 delete tfmt;
412 #ifdef TUFMTTS_DEBUG
413 std::cout << "Failed to format" << "\n";
414 #endif
415 return;
416 }
417
418 #ifdef TUFMTTS_DEBUG
419 char tmp[128]; //output
420 char tmp1[128]; //expected
421 int len = 0;
422 u_strToUTF8(tmp, 128, &len, str.getTerminatedBuffer(), str.length(), &status);
423 u_strToUTF8(tmp1, 128, &len, expected[counter].unescape().getTerminatedBuffer(), expected[counter].unescape().length(), &status);
424 std::cout << "Formatted string : " << tmp << " expected : " << tmp1 << "\n";
425 #endif
426 if (!assertEquals("formatted time string is not expected, locale: " + UnicodeString(locales[locIndex]) + " style: " + (int)styles[styleIndex] + " units: " + (int)tunits[unitIndex], expected[counter], str)) {
427 delete tfmt;
428 str.remove();
429 return;
430 }
431 delete tfmt;
432 str.remove();
433 ++counter;
434 }
435 }
436 }
437 }
438 }
439
440 // Test bug9042
testGreekWithSanitization()441 void TimeUnitTest::testGreekWithSanitization() {
442
443 UErrorCode status = U_ZERO_ERROR;
444 Locale elLoc("el");
445 NumberFormat* numberFmt = NumberFormat::createInstance(Locale("el"), status);
446 if (!assertSuccess("NumberFormat::createInstance for el locale", status, TRUE)) return;
447 numberFmt->setMaximumFractionDigits(1);
448
449 TimeUnitFormat* timeUnitFormat = new TimeUnitFormat(elLoc, status);
450 if (!assertSuccess("TimeUnitFormat::TimeUnitFormat for el locale", status)) return;
451
452 timeUnitFormat->setNumberFormat(*numberFmt, status);
453
454 delete numberFmt;
455 delete timeUnitFormat;
456 }
457
test10219Plurals()458 void TimeUnitTest::test10219Plurals() {
459 Locale usLocale("en_US");
460 double values[2] = {1.588, 1.011};
461 UnicodeString expected[2][3] = {
462 {"1 minute", "1.5 minutes", "1.58 minutes"},
463 {"1 minute", "1.0 minutes", "1.01 minutes"}
464 };
465 UErrorCode status = U_ZERO_ERROR;
466 TimeUnitFormat tuf(usLocale, status);
467 if (U_FAILURE(status)) {
468 dataerrln("generating TimeUnitFormat Object failed: %s", u_errorName(status));
469 return;
470 }
471 LocalPointer<DecimalFormat> nf((DecimalFormat *) NumberFormat::createInstance(usLocale, status));
472 if (U_FAILURE(status)) {
473 dataerrln("generating NumberFormat Object failed: %s", u_errorName(status));
474 return;
475 }
476 for (int32_t j = 0; j < UPRV_LENGTHOF(values); ++j) {
477 for (int32_t i = 0; i < UPRV_LENGTHOF(expected[j]); ++i) {
478 nf->setMinimumFractionDigits(i);
479 nf->setMaximumFractionDigits(i);
480 nf->setRoundingMode(DecimalFormat::kRoundDown);
481 tuf.setNumberFormat(*nf, status);
482 if (U_FAILURE(status)) {
483 dataerrln("setting NumberFormat failed: %s", u_errorName(status));
484 return;
485 }
486 UnicodeString actual;
487 Formattable fmt;
488 LocalPointer<TimeUnitAmount> tamt(
489 new TimeUnitAmount(values[j], TimeUnit::UTIMEUNIT_MINUTE, status), status);
490 if (U_FAILURE(status)) {
491 dataerrln("generating TimeUnitAmount Object failed: %s", u_errorName(status));
492 return;
493 }
494 fmt.adoptObject(tamt.orphan());
495 tuf.format(fmt, actual, status);
496 if (U_FAILURE(status)) {
497 dataerrln("Actual formatting failed: %s", u_errorName(status));
498 return;
499 }
500 if (expected[j][i] != actual) {
501 errln("Expected " + expected[j][i] + ", got " + actual);
502 }
503 }
504 }
505
506 // test parsing
507 Formattable result;
508 ParsePosition pos;
509 UnicodeString formattedString = "1 minutes";
510 tuf.parseObject(formattedString, result, pos);
511 if (formattedString.length() != pos.getIndex()) {
512 errln("Expect parsing to go all the way to the end of the string.");
513 }
514 }
515
TestBritishShortHourFallback()516 void TimeUnitTest::TestBritishShortHourFallback() {
517 // See ticket #11986 "incomplete fallback in MeasureFormat".
518 UErrorCode status = U_ZERO_ERROR;
519 Formattable oneHour(new TimeUnitAmount(1, TimeUnit::UTIMEUNIT_HOUR, status));
520 Locale en_GB("en_GB");
521 TimeUnitFormat formatter(en_GB, UTMUTFMT_ABBREVIATED_STYLE, status);
522 UnicodeString result;
523 formatter.format(oneHour, result, status);
524 assertSuccess("TestBritishShortHourFallback()", status);
525 assertEquals("TestBritishShortHourFallback()", UNICODE_STRING_SIMPLE("1 hr"), result);
526 }
527
528 #endif
529