• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /********************************************************************
2  * Copyright (c) 2008-2012, 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/tmunit.h"
11 #include "unicode/tmutamt.h"
12 #include "unicode/tmutfmt.h"
13 #include "tufmtts.h"
14 #include "unicode/ustring.h"
15 
16 //TODO: put as compilation flag
17 //#define TUFMTTS_DEBUG 1
18 
19 #ifdef TUFMTTS_DEBUG
20 #include <iostream>
21 #endif
22 
runIndexedTest(int32_t index,UBool exec,const char * & name,char *)23 void TimeUnitTest::runIndexedTest( int32_t index, UBool exec, const char* &name, char* /*par*/ ) {
24     if (exec) logln("TestSuite TimeUnitTest");
25     switch (index) {
26         TESTCASE(0, testBasic);
27         TESTCASE(1, testAPI);
28         TESTCASE(2, testGreekWithFallback);
29         TESTCASE(3, testGreekWithSanitization);
30         default: name = ""; break;
31     }
32 }
33 
34 /**
35  * Test basic
36  */
testBasic()37 void TimeUnitTest::testBasic() {
38     const char* locales[] = {"en", "sl", "fr", "zh", "ar", "ru", "zh_Hant", "pa"};
39     for ( unsigned int locIndex = 0;
40           locIndex < sizeof(locales)/sizeof(locales[0]);
41           ++locIndex ) {
42         UErrorCode status = U_ZERO_ERROR;
43         Locale loc(locales[locIndex]);
44         TimeUnitFormat** formats = new TimeUnitFormat*[2];
45         formats[UTMUTFMT_FULL_STYLE] = new TimeUnitFormat(loc, status);
46         if (!assertSuccess("TimeUnitFormat(full)", status, TRUE)) return;
47         formats[UTMUTFMT_ABBREVIATED_STYLE] = new TimeUnitFormat(loc, UTMUTFMT_ABBREVIATED_STYLE, status);
48         if (!assertSuccess("TimeUnitFormat(short)", status)) return;
49 #ifdef TUFMTTS_DEBUG
50         std::cout << "locale: " << locales[locIndex] << "\n";
51 #endif
52         for (int style = UTMUTFMT_FULL_STYLE;
53              style <= UTMUTFMT_ABBREVIATED_STYLE;
54              ++style) {
55           for (TimeUnit::UTimeUnitFields j = TimeUnit::UTIMEUNIT_YEAR;
56              j < TimeUnit::UTIMEUNIT_FIELD_COUNT;
57              j = (TimeUnit::UTimeUnitFields)(j+1)) {
58 #ifdef TUFMTTS_DEBUG
59             std::cout << "time unit: " << j << "\n";
60 #endif
61             double tests[] = {0, 0.5, 1, 1.5, 2, 2.5, 3, 3.5, 5, 10, 100, 101.35};
62             for (unsigned int i = 0; i < sizeof(tests)/sizeof(tests[0]); ++i) {
63 #ifdef TUFMTTS_DEBUG
64                 std::cout << "number: " << tests[i] << "\n";
65 #endif
66                 TimeUnitAmount* source = new TimeUnitAmount(tests[i], j, status);
67                 if (!assertSuccess("TimeUnitAmount()", status)) return;
68                 UnicodeString formatted;
69                 Formattable formattable;
70                 formattable.adoptObject(source);
71                 formatted = ((Format*)formats[style])->format(formattable, formatted, status);
72                 if (!assertSuccess("format()", status)) return;
73 #ifdef TUFMTTS_DEBUG
74                 char formatResult[1000];
75                 formatted.extract(0, formatted.length(), formatResult, "UTF-8");
76                 std::cout << "format result: " << formatResult << "\n";
77 #endif
78                 Formattable result;
79                 ((Format*)formats[style])->parseObject(formatted, result, status);
80                 if (!assertSuccess("parseObject()", status)) return;
81                 if (result != formattable) {
82                     dataerrln("No round trip: ");
83                 }
84                 // other style parsing
85                 Formattable result_1;
86                 ((Format*)formats[1-style])->parseObject(formatted, result_1, status);
87                 if (!assertSuccess("parseObject()", status)) return;
88                 if (result_1 != formattable) {
89                     dataerrln("No round trip: ");
90                 }
91             }
92           }
93         }
94         delete formats[UTMUTFMT_FULL_STYLE];
95         delete formats[UTMUTFMT_ABBREVIATED_STYLE];
96         delete[] formats;
97     }
98 }
99 
100 
testAPI()101 void TimeUnitTest::testAPI() {
102     //================= TimeUnit =================
103     UErrorCode status = U_ZERO_ERROR;
104 
105     TimeUnit* tmunit = TimeUnit::createInstance(TimeUnit::UTIMEUNIT_YEAR, status);
106     if (!assertSuccess("TimeUnit::createInstance", status)) return;
107 
108     TimeUnit* another = (TimeUnit*)tmunit->clone();
109     TimeUnit third(*tmunit);
110     TimeUnit fourth = third;
111 
112     assertTrue("orig and clone are equal", (*tmunit == *another));
113     assertTrue("copied and assigned are equal", (third == fourth));
114 
115     TimeUnit* tmunit_m = TimeUnit::createInstance(TimeUnit::UTIMEUNIT_MONTH, status);
116     assertTrue("year != month", (*tmunit != *tmunit_m));
117 
118     TimeUnit::UTimeUnitFields field = tmunit_m->getTimeUnitField();
119     assertTrue("field of month time unit is month", (field == TimeUnit::UTIMEUNIT_MONTH));
120 
121     delete tmunit;
122     delete another;
123     delete tmunit_m;
124     //
125     //================= TimeUnitAmount =================
126 
127     Formattable formattable((int32_t)2);
128     TimeUnitAmount tma_long(formattable, TimeUnit::UTIMEUNIT_DAY, status);
129     if (!assertSuccess("TimeUnitAmount(formattable...)", status)) return;
130 
131     formattable.setDouble(2);
132     TimeUnitAmount tma_double(formattable, TimeUnit::UTIMEUNIT_DAY, status);
133     if (!assertSuccess("TimeUnitAmount(formattable...)", status)) return;
134 
135     formattable.setDouble(3);
136     TimeUnitAmount tma_double_3(formattable, TimeUnit::UTIMEUNIT_DAY, status);
137     if (!assertSuccess("TimeUnitAmount(formattable...)", status)) return;
138 
139     TimeUnitAmount tma(2, TimeUnit::UTIMEUNIT_DAY, status);
140     if (!assertSuccess("TimeUnitAmount(number...)", status)) return;
141 
142     TimeUnitAmount tma_h(2, TimeUnit::UTIMEUNIT_HOUR, status);
143     if (!assertSuccess("TimeUnitAmount(number...)", status)) return;
144 
145     TimeUnitAmount second(tma);
146     TimeUnitAmount third_tma = tma;
147     TimeUnitAmount* fourth_tma = (TimeUnitAmount*)tma.clone();
148 
149     assertTrue("orig and copy are equal", (second == tma));
150     assertTrue("clone and assigned are equal", (third_tma == *fourth_tma));
151     assertTrue("different if number diff", (tma_double != tma_double_3));
152     assertTrue("different if number type diff", (tma_double != tma_long));
153     assertTrue("different if time unit diff", (tma != tma_h));
154     assertTrue("same even different constructor", (tma_double == tma));
155 
156     assertTrue("getTimeUnitField", (tma.getTimeUnitField() == TimeUnit::UTIMEUNIT_DAY));
157     delete fourth_tma;
158     //
159     //================= TimeUnitFormat =================
160     //
161     TimeUnitFormat* tmf_en = new TimeUnitFormat(Locale("en"), status);
162     if (!assertSuccess("TimeUnitFormat(en...)", status, TRUE)) return;
163     TimeUnitFormat tmf_fr(Locale("fr"), status);
164     if (!assertSuccess("TimeUnitFormat(fr...)", status)) return;
165 
166     assertTrue("TimeUnitFormat: en and fr diff", (*tmf_en != tmf_fr));
167 
168     TimeUnitFormat tmf_assign = *tmf_en;
169     assertTrue("TimeUnitFormat: orig and assign are equal", (*tmf_en == tmf_assign));
170 
171     TimeUnitFormat tmf_copy(tmf_fr);
172     assertTrue("TimeUnitFormat: orig and copy are equal", (tmf_fr == tmf_copy));
173 
174     TimeUnitFormat* tmf_clone = (TimeUnitFormat*)tmf_en->clone();
175     assertTrue("TimeUnitFormat: orig and clone are equal", (*tmf_en == *tmf_clone));
176     delete tmf_clone;
177 
178     tmf_en->setLocale(Locale("fr"), status);
179     if (!assertSuccess("setLocale(fr...)", status)) return;
180 
181     NumberFormat* numberFmt = NumberFormat::createInstance(
182                                  Locale("fr"), status);
183     if (!assertSuccess("NumberFormat::createInstance()", status)) return;
184     tmf_en->setNumberFormat(*numberFmt, status);
185     if (!assertSuccess("setNumberFormat(en...)", status)) return;
186     assertTrue("TimeUnitFormat: setLocale", (*tmf_en == tmf_fr));
187 
188     delete tmf_en;
189 
190     TimeUnitFormat* en_long = new TimeUnitFormat(Locale("en"), UTMUTFMT_FULL_STYLE, status);
191     if (!assertSuccess("TimeUnitFormat(en...)", status)) return;
192     delete en_long;
193 
194     TimeUnitFormat* en_short = new TimeUnitFormat(Locale("en"), UTMUTFMT_ABBREVIATED_STYLE, status);
195     if (!assertSuccess("TimeUnitFormat(en...)", status)) return;
196     delete en_short;
197 
198     TimeUnitFormat* format = new TimeUnitFormat(status);
199     format->setLocale(Locale("zh"), status);
200     format->setNumberFormat(*numberFmt, status);
201     if (!assertSuccess("TimeUnitFormat(en...)", status)) return;
202     delete numberFmt;
203     delete format;
204 }
205 
206 /* @bug 7902
207  * Tests for Greek Language.
208  * This tests that requests for short unit names correctly fall back
209  * to long unit names for a locale where the locale data does not
210  * provide short unit names. As of CLDR 1.9, Greek is one such language.
211  */
testGreekWithFallback()212 void TimeUnitTest::testGreekWithFallback() {
213     UErrorCode status = U_ZERO_ERROR;
214 
215     const char* locales[] = {"el-GR", "el"};
216     TimeUnit::UTimeUnitFields tunits[] = {TimeUnit::UTIMEUNIT_SECOND, TimeUnit::UTIMEUNIT_MINUTE, TimeUnit::UTIMEUNIT_HOUR, TimeUnit::UTIMEUNIT_DAY, TimeUnit::UTIMEUNIT_MONTH, TimeUnit::UTIMEUNIT_YEAR};
217     UTimeUnitFormatStyle styles[] = {UTMUTFMT_FULL_STYLE, UTMUTFMT_ABBREVIATED_STYLE};
218     const int numbers[] = {1, 7};
219 
220     const UChar oneSecond[] = {0x0031, 0x0020, 0x03b4, 0x03b5, 0x03c5, 0x03c4, 0x03b5, 0x03c1, 0x03cc, 0x03bb, 0x03b5, 0x03c0, 0x03c4, 0x03bf, 0};
221     const UChar oneMinute[] = {0x0031, 0x0020, 0x03bb, 0x03b5, 0x03c0, 0x03c4, 0x03cc, 0};
222     const UChar oneHour[] = {0x0031, 0x0020, 0x03ce, 0x03c1, 0x03b1, 0};
223     const UChar oneDay[] = {0x0031, 0x0020, 0x03b7, 0x03bc, 0x03ad, 0x03c1, 0x03b1, 0};
224     const UChar oneMonth[] = {0x0031, 0x0020, 0x03bc, 0x03ae, 0x03bd, 0x03b1, 0x03c2, 0};
225     const UChar oneYear[] = {0x0031, 0x0020, 0x03ad, 0x03c4, 0x03bf, 0x03c2, 0};
226     const UChar sevenSeconds[] = {0x0037, 0x0020, 0x03b4, 0x03b5, 0x03c5, 0x03c4, 0x03b5, 0x03c1, 0x03cc, 0x03bb, 0x03b5, 0x03c0, 0x03c4, 0x03b1, 0};
227     const UChar sevenMinutes[] = {0x0037, 0x0020, 0x03bb, 0x03b5, 0x03c0, 0x03c4, 0x03ac, 0};
228     const UChar sevenHours[] = {0x0037, 0x0020, 0x03ce, 0x03c1, 0x03b5, 0x03c2, 0};
229     const UChar sevenDays[] = {0x0037, 0x0020, 0x03b7, 0x03bc, 0x03ad, 0x03c1, 0x03b5, 0x03c2, 0};
230     const UChar sevenMonths[] = {0x0037, 0x0020, 0x03bc, 0x03ae, 0x03bd, 0x03b5, 0x3c2, 0};
231     const UChar sevenYears[] = {0x0037, 0x0020, 0x03ad, 0x03c4, 0x03b7, 0};
232 
233     const UnicodeString oneSecondStr(oneSecond);
234     const UnicodeString oneMinuteStr(oneMinute);
235     const UnicodeString oneHourStr(oneHour);
236     const UnicodeString oneDayStr(oneDay);
237     const UnicodeString oneMonthStr(oneMonth);
238     const UnicodeString oneYearStr(oneYear);
239     const UnicodeString sevenSecondsStr(sevenSeconds);
240     const UnicodeString sevenMinutesStr(sevenMinutes);
241     const UnicodeString sevenHoursStr(sevenHours);
242     const UnicodeString sevenDaysStr(sevenDays);
243     const UnicodeString sevenMonthsStr(sevenMonths);
244     const UnicodeString sevenYearsStr(sevenYears);
245 
246     const UnicodeString expected[] = {oneSecondStr, oneMinuteStr, oneHourStr, oneDayStr, oneMonthStr, oneYearStr,
247                               oneSecondStr, oneMinuteStr, oneHourStr, oneDayStr, oneMonthStr, oneYearStr,
248                               sevenSecondsStr, sevenMinutesStr, sevenHoursStr, sevenDaysStr, sevenMonthsStr, sevenYearsStr,
249                               sevenSecondsStr, sevenMinutesStr, sevenHoursStr, sevenDaysStr, sevenMonthsStr, sevenYearsStr,
250                               oneSecondStr, oneMinuteStr, oneHourStr, oneDayStr, oneMonthStr, oneYearStr,
251                               oneSecondStr, oneMinuteStr, oneHourStr, oneDayStr, oneMonthStr, oneYearStr,
252                               sevenSecondsStr, sevenMinutesStr, sevenHoursStr, sevenDaysStr, sevenMonthsStr, sevenYearsStr,
253                               sevenSecondsStr, sevenMinutesStr, sevenHoursStr, sevenDaysStr, sevenMonthsStr, sevenYearsStr};
254 
255     int counter = 0;
256     for ( unsigned int locIndex = 0;
257         locIndex < sizeof(locales)/sizeof(locales[0]);
258         ++locIndex ) {
259 
260         Locale l = Locale::createFromName(locales[locIndex]);
261 
262         for ( unsigned int numberIndex = 0;
263             numberIndex < sizeof(numbers)/sizeof(int);
264             ++numberIndex ) {
265 
266             for ( unsigned int styleIndex = 0;
267                 styleIndex < sizeof(styles)/sizeof(styles[0]);
268                 ++styleIndex ) {
269 
270                 for ( unsigned int unitIndex = 0;
271                     unitIndex < sizeof(tunits)/sizeof(tunits[0]);
272                     ++unitIndex ) {
273 
274                     TimeUnitAmount *tamt = new TimeUnitAmount(numbers[numberIndex], tunits[unitIndex], status);
275                     if (U_FAILURE(status)) {
276                         dataerrln("generating TimeUnitAmount Object failed.");
277 #ifdef TUFMTTS_DEBUG
278                         std::cout << "Failed to get TimeUnitAmount for " << tunits[unitIndex] << "\n";
279 #endif
280                         return;
281                     }
282 
283                     TimeUnitFormat *tfmt = new TimeUnitFormat(l, styles[styleIndex], status);
284                     if (U_FAILURE(status)) {
285                         dataerrln("generating TimeUnitAmount Object failed.");
286 #ifdef TUFMTTS_DEBUG
287                        std::cout <<  "Failed to get TimeUnitFormat for " << locales[locIndex] << "\n";
288 #endif
289                        return;
290                     }
291 
292                     Formattable fmt;
293                     UnicodeString str;
294 
295                     fmt.adoptObject(tamt);
296                     str = ((Format *)tfmt)->format(fmt, str, status);
297                     if (!assertSuccess("formatting relative time failed", status)) {
298                         delete tfmt;
299 #ifdef TUFMTTS_DEBUG
300                         std::cout <<  "Failed to format" << "\n";
301 #endif
302                         return;
303                     }
304 
305 #ifdef TUFMTTS_DEBUG
306                     char tmp[128];    //output
307                     char tmp1[128];    //expected
308                     int len = 0;
309                     u_strToUTF8(tmp, 128, &len, str.getTerminatedBuffer(), str.length(), &status);
310                     u_strToUTF8(tmp1, 128, &len, expected[counter].unescape().getTerminatedBuffer(), expected[counter].unescape().length(), &status);
311                     std::cout <<  "Formatted string : " << tmp << " expected : " << tmp1 << "\n";
312 #endif
313                     if (!assertEquals("formatted time string is not expected, locale: " + UnicodeString(locales[locIndex]) + " style: " + (int)styles[styleIndex] + " units: " + (int)tunits[unitIndex], expected[counter], str)) {
314                         delete tfmt;
315                         str.remove();
316                         return;
317                     }
318                     delete tfmt;
319                     str.remove();
320                     ++counter;
321                 }
322             }
323         }
324     }
325 }
326 
327 // Test bug9042
testGreekWithSanitization()328 void TimeUnitTest::testGreekWithSanitization() {
329 
330     UErrorCode status = U_ZERO_ERROR;
331     Locale elLoc("el");
332     NumberFormat* numberFmt = NumberFormat::createInstance(Locale("el"), status);
333     if (!assertSuccess("NumberFormat::createInstance for el locale", status, TRUE)) return;
334     numberFmt->setMaximumFractionDigits(1);
335 
336     TimeUnitFormat* timeUnitFormat = new TimeUnitFormat(elLoc, status);
337     if (!assertSuccess("TimeUnitFormat::TimeUnitFormat for el locale", status)) return;
338 
339     timeUnitFormat->setNumberFormat(*numberFmt, status);
340 
341     delete numberFmt;
342     delete timeUnitFormat;
343 }
344 
345 
346 #endif
347