]> git.saurik.com Git - apple/icu.git/blob - icuSources/test/cintltst/cdateintervalformattest.c
ICU-62123.0.1.tar.gz
[apple/icu.git] / icuSources / test / cintltst / cdateintervalformattest.c
1 // © 2016 and later: Unicode, Inc. and others.
2 // License & terms of use: http://www.unicode.org/copyright.html
3 /********************************************************************
4 * Copyright (c) 2011-2016, International Business Machines Corporation
5 * and others. All Rights Reserved.
6 ********************************************************************/
7 /* C API TEST FOR DATE INTERVAL FORMAT */
8
9 #include "unicode/utypes.h"
10
11 #if !UCONFIG_NO_FORMATTING
12
13 #include "unicode/udateintervalformat.h"
14 #include "unicode/udat.h"
15 #include "unicode/ucal.h"
16 #include "unicode/ustring.h"
17 #include "cintltst.h"
18 #include "cmemory.h"
19
20 static void TestDateIntervalFormat(void);
21 static void TestFPos_SkelWithSeconds(void);
22 static void TestOpen(void);
23
24 void addDateIntervalFormatTest(TestNode** root);
25
26 #define TESTCASE(x) addTest(root, &x, "tsformat/cdateintervalformattest/" #x)
27
28 void addDateIntervalFormatTest(TestNode** root)
29 {
30 TESTCASE(TestDateIntervalFormat);
31 TESTCASE(TestFPos_SkelWithSeconds);
32 TESTCASE(TestOpen);
33 }
34
35 static const char tzUSPacific[] = "US/Pacific";
36 static const char tzAsiaTokyo[] = "Asia/Tokyo";
37 #define Date201103021030 1299090600000.0 /* 2011-Mar-02 1030 in US/Pacific, 2011-Mar-03 0330 in Asia/Tokyo */
38 #define Date201009270800 1285599629000.0 /* 2010-Sep-27 0800 in US/Pacific */
39 #define Date201712300900 1514653200000.0 /* 2017-Dec-30 0900 in US/Pacific */
40 #define _MINUTE (60.0*1000.0)
41 #define _HOUR (60.0*60.0*1000.0)
42 #define _DAY (24.0*60.0*60.0*1000.0)
43
44 typedef struct {
45 const char * locale;
46 const char * skeleton;
47 const char * tzid;
48 UDateIntervalFormatAttributeValue minimizeType;
49 const UDate from;
50 const UDate to;
51 const char * resultExpected;
52 } DateIntervalFormatTestItem;
53
54 /* Just a small set of tests for now, the real functionality is tested in the C++ tests */
55 static const DateIntervalFormatTestItem testItems[] = {
56 { "en", "MMMdHHmm", tzUSPacific, UDTITVFMT_MINIMIZE_NONE, Date201103021030, Date201103021030 + 7.0*_HOUR, "Mar 2, 10:30\\u2009\\u2013\\u200917:30" },
57 { "en", "MMMdHHmm", tzAsiaTokyo, UDTITVFMT_MINIMIZE_NONE, Date201103021030, Date201103021030 + 7.0*_HOUR, "Mar 3, 03:30\\u2009\\u2013\\u200910:30" },
58 { "en", "yMMMEd", tzUSPacific, UDTITVFMT_MINIMIZE_NONE, Date201009270800, Date201009270800 + 12.0*_HOUR, "Mon, Sep 27, 2010" },
59 { "en", "yMMMEd", tzUSPacific, UDTITVFMT_MINIMIZE_NONE, Date201009270800, Date201009270800 + 31.0*_DAY, "Mon, Sep 27\\u2009\\u2013\\u2009Thu, Oct 28, 2010" },
60 { "en", "yMMMEd", tzUSPacific, UDTITVFMT_MINIMIZE_NONE, Date201009270800, Date201009270800 + 410.0*_DAY, "Mon, Sep 27, 2010\\u2009\\u2013\\u2009Fri, Nov 11, 2011" },
61 { "de", "Hm", tzUSPacific, UDTITVFMT_MINIMIZE_NONE, Date201009270800, Date201009270800 + 12.0*_HOUR, "08:00\\u201320:00 Uhr" },
62 { "de", "Hm", tzUSPacific, UDTITVFMT_MINIMIZE_NONE, Date201009270800, Date201009270800 + 31.0*_DAY, "27.9.2010, 08:00\\u2009\\u2013\\u200928.10.2010, 08:00" },
63 { "ja", "MMMd", tzUSPacific, UDTITVFMT_MINIMIZE_NONE, Date201009270800, Date201009270800 + 1.0*_DAY, "9\\u670827\\u65E5\\uFF5E28\\u65E5" },
64 { "en", "jm", tzUSPacific, UDTITVFMT_MINIMIZE_NONE, Date201103021030, Date201103021030 + 1.0*_HOUR, "10:30 AM\\u2009\\u2013\\u200911:30 AM" },
65 { "en", "jm", tzUSPacific, UDTITVFMT_MINIMIZE_NONE, Date201103021030, Date201103021030 + 12.0*_HOUR, "10:30 AM\\u2009\\u2013\\u200910:30 PM" },
66 { "it", "yMMMMd", tzUSPacific, UDTITVFMT_MINIMIZE_NONE, Date201103021030, Date201103021030 + 15.0*_DAY, "2\\u201317 marzo 2011" },
67 { "en_SA", "MMMd", tzUSPacific, UDTITVFMT_MINIMIZE_NONE, Date201009270800, Date201009270800 + 6.0*_DAY, "18\\u2009\\u2013\\u200924 Shaw." },
68 { "en@calendar=islamic-umalqura", "MMMd", tzUSPacific, UDTITVFMT_MINIMIZE_NONE, Date201009270800, Date201009270800 + 6.0*_DAY, "Shaw. 18\\u2009\\u2013\\u200924" },
69 // Apple-specific
70 { "en", "MMMd", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_MONTHS, Date201009270800, Date201009270800 + 6.0*_DAY, "Sep 27\\u2009\\u2013\\u20093" },
71 { "en", "MMMd", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_MONTHS, Date201009270800, Date201009270800 + 32.0*_DAY, "Sep 27\\u2009\\u2013\\u2009Oct 29" },
72 { "en", "MMMd", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_MONTHS, Date201712300900, Date201712300900 + 6.0*_DAY, "Dec 30\\u2009\\u2013\\u20095" }, // across year boundary
73 { "en", "MMMd", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_MONTHS, Date201712300900, Date201712300900 + 32.0*_DAY, "Dec 30, 2017\\u2009\\u2013\\u2009Jan 31, 2018" }, // across year boundary but > 1 month
74 { "fr", "MMMd", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_MONTHS, Date201009270800, Date201009270800 + 6.0*_DAY, "27\\u20133 oct." },
75 { "fr", "MMMd", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_MONTHS, Date201009270800, Date201009270800 + 32.0*_DAY, "27 sept.\\u2009\\u2013\\u200929 oct." },
76 { "fr", "MMMd", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_MONTHS, Date201712300900, Date201712300900 + 6.0*_DAY, "30\\u20135 janv." }, // across year boundary
77 { "fr", "MMMd", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_MONTHS, Date201712300900, Date201712300900 + 32.0*_DAY, "30 d\\u00E9c. 2017\\u2009\\u2013\\u200931 janv. 2018" }, // across year boundary but > 1 month
78
79 { "en", "yMMMd", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_MONTHS, Date201009270800, Date201009270800 + 6.0*_DAY, "Sep 27\\u2009\\u2013\\u20093, 2010" },
80 { "en", "yMMMd", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_MONTHS, Date201009270800, Date201009270800 + 32.0*_DAY, "Sep 27\\u2009\\u2013\\u2009Oct 29, 2010" },
81 { "en", "yMMMd", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_MONTHS, Date201712300900, Date201712300900 + 6.0*_DAY, "Dec 30, 2017\\u2009\\u2013\\u2009Jan 5, 2018" }, // across year boundary
82 { "en", "yMMMd", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_MONTHS, Date201712300900, Date201712300900 + 32.0*_DAY, "Dec 30, 2017\\u2009\\u2013\\u2009Jan 31, 2018" }, // across year boundary but > 1 month
83 { "fr", "yMMMd", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_MONTHS, Date201009270800, Date201009270800 + 6.0*_DAY, "27\\u20133 oct. 2010" },
84 { "fr", "yMMMd", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_MONTHS, Date201009270800, Date201009270800 + 32.0*_DAY, "27 sept.\\u2009\\u2013\\u200929 oct. 2010" },
85 { "fr", "yMMMd", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_MONTHS, Date201712300900, Date201712300900 + 6.0*_DAY, "30 d\\u00E9c. 2017\\u2009\\u2013\\u20095 janv. 2018" }, // across year boundary
86 { "fr", "yMMMd", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_MONTHS, Date201712300900, Date201712300900 + 32.0*_DAY, "30 d\\u00E9c. 2017\\u2009\\u2013\\u200931 janv. 2018" }, // across year boundary but > 1 month
87
88 { "en", "MMMdjmm", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_DAYS, Date201009270800, Date201009270800 + 10.0*_HOUR, "Sep 27, 8:00 AM\\u2009\\u2013\\u20096:00 PM" },
89 { "en", "MMMdjmm", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_DAYS, Date201009270800, Date201009270800 + 17.0*_HOUR, "Sep 27, 8:00 AM\\u2009\\u2013\\u2009Sep 28, 1:00 AM" },
90 { "en", "MMMdjmm", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_DAYS, Date201009270800 + 12.0*_HOUR, Date201009270800 + 17.0*_HOUR, "Sep 27, 8:00 PM\\u2009\\u2013\\u20091:00 AM" },
91 { "en", "MMMdjmm", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_DAYS, Date201009270800 + 12.0*_HOUR, Date201009270800 + 26.0*_HOUR, "Sep 27, 8:00 PM\\u2009\\u2013\\u2009Sep 28, 10:00 AM" },
92 { "en", "MMMdjmm", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_DAYS, Date201009270800 + 12.0*_HOUR, Date201009270800 + 35.0*_HOUR, "Sep 27, 8:00 PM\\u2009\\u2013\\u2009Sep 28, 7:00 PM" },
93 { "fr", "MMMdjmm", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_DAYS, Date201009270800, Date201009270800 + 10.0*_HOUR, "27 sept. \\u00E0 08:00\\u2009\\u2013\\u200918:00" },
94 { "fr", "MMMdjmm", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_DAYS, Date201009270800, Date201009270800 + 17.0*_HOUR, "27 sept. \\u00E0 08:00\\u2009\\u2013\\u200928 sept. \\u00E0 01:00" },
95 { "fr", "MMMdjmm", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_DAYS, Date201009270800 + 12.0*_HOUR, Date201009270800 + 17.0*_HOUR, "27 sept. \\u00E0 20:00\\u2009\\u2013\\u200901:00" },
96 { "fr", "MMMdjmm", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_DAYS, Date201009270800 + 12.0*_HOUR, Date201009270800 + 26.0*_HOUR, "27 sept. \\u00E0 20:00\\u2009\\u2013\\u200928 sept. \\u00E0 10:00" },
97 { "fr", "MMMdjmm", tzUSPacific, UDTITVFMT_MINIMIZE_ADJACENT_DAYS, Date201009270800 + 12.0*_HOUR, Date201009270800 + 35.0*_HOUR, "27 sept. \\u00E0 20:00\\u2009\\u2013\\u200928 sept. \\u00E0 19:00" },
98
99 { NULL, NULL, NULL, UDTITVFMT_MINIMIZE_NONE, 0, 0, NULL }
100 };
101
102 enum {
103 kSkelBufLen = 32,
104 kTZIDBufLen = 96,
105 kFormatBufLen = 128
106 };
107
108 static void TestDateIntervalFormat()
109 {
110 const DateIntervalFormatTestItem * testItemPtr;
111 UErrorCode status = U_ZERO_ERROR;
112 ctest_setTimeZone(NULL, &status);
113 log_verbose("\nTesting udtitvfmt_open() and udtitvfmt_format() with various parameters\n");
114 for ( testItemPtr = testItems; testItemPtr->locale != NULL; ++testItemPtr ) {
115 UDateIntervalFormat* udtitvfmt;
116 int32_t tzidLen;
117 UChar skelBuf[kSkelBufLen];
118 UChar tzidBuf[kTZIDBufLen];
119 const char * tzidForLog = (testItemPtr->tzid)? testItemPtr->tzid: "NULL";
120
121 status = U_ZERO_ERROR;
122 u_unescape(testItemPtr->skeleton, skelBuf, kSkelBufLen);
123 if ( testItemPtr->tzid ) {
124 u_unescape(testItemPtr->tzid, tzidBuf, kTZIDBufLen);
125 tzidLen = -1;
126 } else {
127 tzidLen = 0;
128 }
129 udtitvfmt = udtitvfmt_open(testItemPtr->locale, skelBuf, -1, tzidBuf, tzidLen, &status);
130 if ( U_SUCCESS(status) ) {
131 UChar result[kFormatBufLen];
132 UChar resultExpected[kFormatBufLen];
133 udtitvfmt_setAttribute(udtitvfmt, UDTITVFMT_MINIMIZE_TYPE, testItemPtr->minimizeType, &status);
134 if ( U_FAILURE(status) ) {
135 log_err("FAIL: udtitvfmt_setAttribute for locale %s, skeleton %s, tzid %s, minimizeType %d: %s\n",
136 testItemPtr->locale, testItemPtr->skeleton, tzidForLog, (int)testItemPtr->minimizeType, myErrorName(status) );
137 }
138 int32_t fmtLen = udtitvfmt_format(udtitvfmt, testItemPtr->from, testItemPtr->to, result, kFormatBufLen, NULL, &status);
139 if (fmtLen >= kFormatBufLen) {
140 result[kFormatBufLen-1] = 0;
141 }
142 if ( U_SUCCESS(status) ) {
143 u_unescape(testItemPtr->resultExpected, resultExpected, kFormatBufLen);
144 if ( u_strcmp(result, resultExpected) != 0 ) {
145 char bcharBuf[kFormatBufLen];
146 #if 0
147 log_err("ERROR: udtitvfmt_format for locale %s, skeleton %s, tzid %s, minimizeType %d, from %.1f, to %.1f: expect %s, get %s\n",
148 testItemPtr->locale, testItemPtr->skeleton, tzidForLog, (int)testItemPtr->minimizeType,
149 testItemPtr->from, testItemPtr->to, testItemPtr->resultExpected, u_austrcpy(bcharBuf,result) );
150 #else
151 // Apple-specific version
152 char bexpbuf[kFormatBufLen];
153 u_strToUTF8(bexpbuf, kFormatBufLen, NULL, resultExpected, -1, &status);
154 u_strToUTF8(bcharBuf, kFormatBufLen, NULL, result, fmtLen, &status);
155 log_err("ERROR: udtitvfmt_format for locale %s, skeleton %s, tzid %s, minimizeType %d, from %.1f, to %.1f: expect %s, get %s\n",
156 testItemPtr->locale, testItemPtr->skeleton, tzidForLog, (int)testItemPtr->minimizeType,
157 testItemPtr->from, testItemPtr->to, bexpbuf, bcharBuf );
158 #endif
159 }
160 } else {
161 log_err("FAIL: udtitvfmt_format for locale %s, skeleton %s, tzid %s, minimizeType %d, from %.1f, to %.1f: %s\n",
162 testItemPtr->locale, testItemPtr->skeleton, tzidForLog, (int)testItemPtr->minimizeType,
163 testItemPtr->from, testItemPtr->to, myErrorName(status) );
164 }
165 udtitvfmt_close(udtitvfmt);
166 } else {
167 log_data_err("FAIL: udtitvfmt_open for locale %s, skeleton %s, tzid %s - %s\n",
168 testItemPtr->locale, testItemPtr->skeleton, tzidForLog, myErrorName(status) );
169 }
170 }
171 ctest_resetTimeZone();
172 }
173
174 /********************************************************************
175 * TestFPos_SkelWithSeconds and related data
176 ********************************************************************
177 */
178
179 static UChar zoneGMT[] = { 0x47,0x4D,0x54,0 }; // GMT
180 static const UDate startTime = 1416474000000.0; // 2014 Nov 20 09:00 GMT
181
182 static const double deltas[] = {
183 0.0, // none
184 200.0, // 200 millisec
185 20000.0, // 20 sec
186 1200000.0, // 20 min
187 7200000.0, // 2 hrs
188 43200000.0, // 12 hrs
189 691200000.0, // 8 days
190 1382400000.0, // 16 days,
191 8640000000.0, // 100 days
192 -1.0
193 };
194 enum { kNumDeltas = UPRV_LENGTHOF(deltas) - 1 };
195
196 typedef struct {
197 int32_t posBegin;
198 int32_t posEnd;
199 const char * format;
200 } ExpectPosAndFormat;
201
202 static const ExpectPosAndFormat exp_en_HHmm[kNumDeltas] = {
203 { 3, 5, "09:00" },
204 { 3, 5, "09:00" },
205 { 3, 5, "09:00" },
206 { 3, 5, "09:00\\u2009\\u2013\\u200909:20" },
207 { 3, 5, "09:00\\u2009\\u2013\\u200911:00" },
208 { 3, 5, "09:00\\u2009\\u2013\\u200921:00" },
209 { 15, 17, "11/20/2014, 09:00\\u2009\\u2013\\u200911/28/2014, 09:00" },
210 { 15, 17, "11/20/2014, 09:00\\u2009\\u2013\\u200912/6/2014, 09:00" },
211 { 15, 17, "11/20/2014, 09:00\\u2009\\u2013\\u20092/28/2015, 09:00" }
212 };
213
214 static const ExpectPosAndFormat exp_en_HHmmss[kNumDeltas] = {
215 { 3, 5, "09:00:00" },
216 { 3, 5, "09:00:00" },
217 { 3, 5, "09:00:00\\u2009\\u2013\\u200909:00:20" },
218 { 3, 5, "09:00:00\\u2009\\u2013\\u200909:20:00" },
219 { 3, 5, "09:00:00\\u2009\\u2013\\u200911:00:00" },
220 { 3, 5, "09:00:00\\u2009\\u2013\\u200921:00:00" },
221 { 15, 17, "11/20/2014, 09:00:00\\u2009\\u2013\\u200911/28/2014, 09:00:00" },
222 { 15, 17, "11/20/2014, 09:00:00\\u2009\\u2013\\u200912/6/2014, 09:00:00" },
223 { 15, 17, "11/20/2014, 09:00:00\\u2009\\u2013\\u20092/28/2015, 09:00:00" }
224 };
225
226 static const ExpectPosAndFormat exp_en_yyMMdd[kNumDeltas] = {
227 { 0, 0, "11/20/14" },
228 { 0, 0, "11/20/14" },
229 { 0, 0, "11/20/14" },
230 { 0, 0, "11/20/14" },
231 { 0, 0, "11/20/14" },
232 { 0, 0, "11/20/14" },
233 { 0, 0, "11/20/14\\u2009\\u2013\\u200911/28/14" },
234 { 0, 0, "11/20/14\\u2009\\u2013\\u200912/6/14" },
235 { 0, 0, "11/20/14\\u2009\\u2013\\u20092/28/15" }
236 };
237
238 static const ExpectPosAndFormat exp_en_yyMMddHHmm[kNumDeltas] = {
239 { 13, 15, "11/20/14, 09:00" },
240 { 13, 15, "11/20/14, 09:00" },
241 { 13, 15, "11/20/14, 09:00" },
242 { 13, 15, "11/20/14, 09:00\\u2009\\u2013\\u200909:20" },
243 { 13, 15, "11/20/14, 09:00\\u2009\\u2013\\u200911:00" },
244 { 13, 15, "11/20/14, 09:00\\u2009\\u2013\\u200921:00" },
245 { 13, 15, "11/20/14, 09:00\\u2009\\u2013\\u200911/28/14, 09:00" },
246 { 13, 15, "11/20/14, 09:00\\u2009\\u2013\\u200912/06/14, 09:00" },
247 { 13, 15, "11/20/14, 09:00\\u2009\\u2013\\u200902/28/15, 09:00" }
248 };
249
250 static const ExpectPosAndFormat exp_en_yyMMddHHmmss[kNumDeltas] = {
251 { 13, 15, "11/20/14, 09:00:00" },
252 { 13, 15, "11/20/14, 09:00:00" },
253 { 13, 15, "11/20/14, 09:00:00\\u2009\\u2013\\u200909:00:20" },
254 { 13, 15, "11/20/14, 09:00:00\\u2009\\u2013\\u200909:20:00" },
255 { 13, 15, "11/20/14, 09:00:00\\u2009\\u2013\\u200911:00:00" },
256 { 13, 15, "11/20/14, 09:00:00\\u2009\\u2013\\u200921:00:00" },
257 { 13, 15, "11/20/14, 09:00:00\\u2009\\u2013\\u200911/28/14, 09:00:00" },
258 { 13, 15, "11/20/14, 09:00:00\\u2009\\u2013\\u200912/06/14, 09:00:00" },
259 { 13, 15, "11/20/14, 09:00:00\\u2009\\u2013\\u200902/28/15, 09:00:00" }
260 };
261
262 static const ExpectPosAndFormat exp_en_yMMMdhmmssz[kNumDeltas] = {
263 { 16, 18, "Nov 20, 2014, 9:00:00 AM GMT" },
264 { 16, 18, "Nov 20, 2014, 9:00:00 AM GMT" },
265 { 16, 18, "Nov 20, 2014, 9:00:00 AM GMT\\u2009\\u2013\\u20099:00:20 AM GMT" },
266 { 16, 18, "Nov 20, 2014, 9:00:00 AM GMT\\u2009\\u2013\\u20099:20:00 AM GMT" },
267 { 16, 18, "Nov 20, 2014, 9:00:00 AM GMT\\u2009\\u2013\\u200911:00:00 AM GMT" },
268 { 16, 18, "Nov 20, 2014, 9:00:00 AM GMT\\u2009\\u2013\\u20099:00:00 PM GMT" },
269 { 16, 18, "Nov 20, 2014, 9:00:00 AM GMT\\u2009\\u2013\\u2009Nov 28, 2014, 9:00:00 AM GMT" },
270 { 16, 18, "Nov 20, 2014, 9:00:00 AM GMT\\u2009\\u2013\\u2009Dec 6, 2014, 9:00:00 AM GMT" },
271 { 16, 18, "Nov 20, 2014, 9:00:00 AM GMT\\u2009\\u2013\\u2009Feb 28, 2015, 9:00:00 AM GMT" }
272 };
273
274 static const ExpectPosAndFormat exp_ja_yyMMddHHmm[kNumDeltas] = {
275 { 11, 13, "14/11/20 9:00" },
276 { 11, 13, "14/11/20 9:00" },
277 { 11, 13, "14/11/20 9:00" },
278 { 11, 13, "14/11/20 9\\u664200\\u5206\\uFF5E9\\u664220\\u5206" },
279 { 11, 13, "14/11/20 9\\u664200\\u5206\\uFF5E11\\u664200\\u5206" },
280 { 11, 13, "14/11/20 9\\u664200\\u5206\\uFF5E21\\u664200\\u5206" },
281 { 11, 13, "14/11/20 9:00\\uFF5E14/11/28 9:00" },
282 { 11, 13, "14/11/20 9:00\\uFF5E14/12/06 9:00" },
283 { 11, 13, "14/11/20 9:00\\uFF5E15/02/28 9:00" }
284 };
285
286 static const ExpectPosAndFormat exp_ja_yyMMddHHmmss[kNumDeltas] = {
287 { 11, 13, "14/11/20 9:00:00" },
288 { 11, 13, "14/11/20 9:00:00" },
289 { 11, 13, "14/11/20 9:00:00\\uFF5E9:00:20" },
290 { 11, 13, "14/11/20 9:00:00\\uFF5E9:20:00" },
291 { 11, 13, "14/11/20 9:00:00\\uFF5E11:00:00" },
292 { 11, 13, "14/11/20 9:00:00\\uFF5E21:00:00" },
293 { 11, 13, "14/11/20 9:00:00\\uFF5E14/11/28 9:00:00" },
294 { 11, 13, "14/11/20 9:00:00\\uFF5E14/12/06 9:00:00" },
295 { 11, 13, "14/11/20 9:00:00\\uFF5E15/02/28 9:00:00" }
296 };
297
298 static const ExpectPosAndFormat exp_ja_yMMMdHHmmss[kNumDeltas] = {
299 { 14, 16, "2014\\u5E7411\\u670820\\u65E5 9:00:00" },
300 { 14, 16, "2014\\u5E7411\\u670820\\u65E5 9:00:00" },
301 { 14, 16, "2014\\u5E7411\\u670820\\u65E5 9:00:00\\uFF5E9:00:20" },
302 { 14, 16, "2014\\u5E7411\\u670820\\u65E5 9:00:00\\uFF5E9:20:00" },
303 { 14, 16, "2014\\u5E7411\\u670820\\u65E5 9:00:00\\uFF5E11:00:00" },
304 { 14, 16, "2014\\u5E7411\\u670820\\u65E5 9:00:00\\uFF5E21:00:00" },
305 { 14, 16, "2014\\u5E7411\\u670820\\u65E5 9:00:00\\uFF5E2014\\u5E7411\\u670828\\u65E5 9:00:00" },
306 { 14, 16, "2014\\u5E7411\\u670820\\u65E5 9:00:00\\uFF5E2014\\u5E7412\\u67086\\u65E5 9:00:00" },
307 { 14, 16, "2014\\u5E7411\\u670820\\u65E5 9:00:00\\uFF5E2015\\u5E742\\u670828\\u65E5 9:00:00" }
308 };
309
310 typedef struct {
311 const char * locale;
312 const char * skeleton;
313 UDateFormatField fieldToCheck;
314 const ExpectPosAndFormat * expected;
315 } LocaleAndSkeletonItem;
316
317 static const LocaleAndSkeletonItem locSkelItems[] = {
318 { "en", "HHmm", UDAT_MINUTE_FIELD, exp_en_HHmm },
319 { "en", "HHmmss", UDAT_MINUTE_FIELD, exp_en_HHmmss },
320 { "en", "yyMMdd", UDAT_MINUTE_FIELD, exp_en_yyMMdd },
321 { "en", "yyMMddHHmm", UDAT_MINUTE_FIELD, exp_en_yyMMddHHmm },
322 { "en", "yyMMddHHmmss", UDAT_MINUTE_FIELD, exp_en_yyMMddHHmmss },
323 { "en", "yMMMdhmmssz", UDAT_MINUTE_FIELD, exp_en_yMMMdhmmssz },
324 { "ja", "yyMMddHHmm", UDAT_MINUTE_FIELD, exp_ja_yyMMddHHmm },
325 { "ja", "yyMMddHHmmss", UDAT_MINUTE_FIELD, exp_ja_yyMMddHHmmss },
326 { "ja", "yMMMdHHmmss", UDAT_MINUTE_FIELD, exp_ja_yMMMdHHmmss },
327 { NULL, NULL, (UDateFormatField)0, NULL }
328 };
329
330 enum { kSizeUBuf = 96, kSizeBBuf = 192 };
331
332 static void TestFPos_SkelWithSeconds()
333 {
334 const LocaleAndSkeletonItem * locSkelItemPtr;
335 for (locSkelItemPtr = locSkelItems; locSkelItemPtr->locale != NULL; locSkelItemPtr++) {
336 UDateIntervalFormat* udifmt;
337 UChar ubuf[kSizeUBuf];
338 int32_t ulen, uelen;
339 UErrorCode status = U_ZERO_ERROR;
340 ulen = u_unescape(locSkelItemPtr->skeleton, ubuf, kSizeUBuf);
341 udifmt = udtitvfmt_open(locSkelItemPtr->locale, ubuf, ulen, zoneGMT, -1, &status);
342 if ( U_FAILURE(status) ) {
343 log_data_err("FAIL: udtitvfmt_open for locale %s, skeleton %s: %s\n",
344 locSkelItemPtr->locale, locSkelItemPtr->skeleton, u_errorName(status));
345 } else {
346 const double * deltasPtr = deltas;
347 const ExpectPosAndFormat * expectedPtr = locSkelItemPtr->expected;
348 for (; *deltasPtr >= 0.0; deltasPtr++, expectedPtr++) {
349 UFieldPosition fpos = { locSkelItemPtr->fieldToCheck, 0, 0 };
350 UChar uebuf[kSizeUBuf];
351 char bbuf[kSizeBBuf];
352 char bebuf[kSizeBBuf];
353 status = U_ZERO_ERROR;
354 uelen = u_unescape(expectedPtr->format, uebuf, kSizeUBuf);
355 ulen = udtitvfmt_format(udifmt, startTime, startTime + *deltasPtr, ubuf, kSizeUBuf, &fpos, &status);
356 if ( U_FAILURE(status) ) {
357 log_err("FAIL: udtitvfmt_format for locale %s, skeleton %s, delta %.1f: %s\n",
358 locSkelItemPtr->locale, locSkelItemPtr->skeleton, *deltasPtr, u_errorName(status));
359 } else if ( ulen != uelen || u_strncmp(ubuf,uebuf,uelen) != 0 ||
360 fpos.beginIndex != expectedPtr->posBegin || fpos.endIndex != expectedPtr->posEnd ) {
361 u_strToUTF8(bbuf, kSizeBBuf, NULL, ubuf, ulen, &status);
362 u_strToUTF8(bebuf, kSizeBBuf, NULL, uebuf, uelen, &status); // convert back to get unescaped string
363 log_err("FAIL: udtitvfmt_format for locale %s, skeleton %s, delta %12.1f, expect %d-%d \"%s\", get %d-%d \"%s\"\n",
364 locSkelItemPtr->locale, locSkelItemPtr->skeleton, *deltasPtr,
365 expectedPtr->posBegin, expectedPtr->posEnd, bebuf,
366 fpos.beginIndex, fpos.endIndex, bbuf);
367 }
368 }
369 udtitvfmt_close(udifmt);
370 }
371 }
372 }
373
374 static const char* openLocales[] = {
375 "en",
376 "en@calendar=japanese",
377 "en@calendar=coptic",
378 "en@calendar=chinese",
379 "en_001",
380 "en_001@calendar=japanese",
381 "en_001@calendar=coptic",
382 "en_001@calendar=chinese",
383 "en_AU",
384 "en_AU@calendar=japanese", // had problems
385 "en_AU@calendar=coptic", // had problems
386 "en_AU@calendar=chinese",
387 "en_CA",
388 "en_CA@calendar=japanese", // had problems
389 "en_CA@calendar=coptic", // had problems
390 "en_CA@calendar=chinese",
391 "en_CN",
392 "en_CN@calendar=japanese", // had problems
393 "en_CN@calendar=coptic", // had problems
394 "en_CN@calendar=chinese",
395 "en_DE@calendar=japanese", // had problems
396 "en_DE@calendar=coptic", // had problems
397 "en_GB",
398 "en_GB@calendar=japanese", // had problems
399 "en_GB@calendar=coptic", // had problems
400 "en_GB@calendar=chinese",
401 "en_HK@calendar=japanese", // had problems
402 "en_HK@calendar=coptic", // had problems
403 "en_IE@calendar=japanese", // had problems
404 "en_IE@calendar=coptic", // had problems
405 "en_IN@calendar=japanese", // had problems
406 "en_IN@calendar=coptic", // had problems
407 "en_JP",
408 "en_JP@calendar=japanese",
409 "en_JP@calendar=coptic",
410 "en_JP@calendar=chinese",
411 "en_NZ",
412 "en_NZ@calendar=japanese", // had problems
413 "en_NZ@calendar=coptic", // had problems
414 "en_NZ@calendar=chinese",
415 "en_SG@calendar=japanese", // had problems
416 "en_SG@calendar=coptic", // had problems
417 "es",
418 "es@calendar=japanese",
419 "es@calendar=coptic",
420 "es@calendar=chinese",
421 "es_419",
422 "es_419@calendar=japanese", // had problems
423 "es_419@calendar=coptic", // had problems
424 "es_419@calendar=chinese",
425 "es_MX",
426 "es_MX@calendar=japanese",
427 "es_MX@calendar=coptic",
428 "es_MX@calendar=chinese",
429 "es_US",
430 "es_US@calendar=japanese",
431 "es_US@calendar=coptic",
432 "es_US@calendar=chinese",
433 "fr",
434 "fr@calendar=japanese",
435 "fr@calendar=coptic",
436 "fr@calendar=chinese",
437 "fr_CA",
438 "fr_CA@calendar=japanese", // had problems
439 "fr_CA@calendar=coptic", // had problems
440 "fr_CA@calendar=chinese",
441 "fr_CH",
442 "fr_CH@calendar=japanese",
443 "fr_CH@calendar=coptic",
444 "fr_CH@calendar=chinese",
445 "fr_BE",
446 "fr_BE@calendar=japanese",
447 "fr_BE@calendar=coptic",
448 "fr_BE@calendar=chinese",
449 "nl_BE@calendar=japanese", // had problems
450 "nl_BE@calendar=coptic", // had problems
451 "pt",
452 "pt@calendar=japanese",
453 "pt@calendar=coptic",
454 "pt@calendar=chinese",
455 "pt_PT",
456 "pt_PT@calendar=japanese", // had problems
457 "pt_PT@calendar=coptic", // had problems
458 "pt_PT@calendar=chinese",
459 "zh_Hant",
460 "zh_Hant@calendar=japanese",
461 "zh_Hant@calendar=coptic",
462 "zh_Hant@calendar=chinese",
463 "zh_Hant_HK",
464 "zh_Hant_HK@calendar=japanese", // had problems
465 "zh_Hant_HK@calendar=coptic", // had problems
466 "zh_Hant_HK@calendar=chinese",
467 NULL
468 };
469 static const UChar* openSkeleton = u"zzzzyMMMMEEEEdhmmss";
470 static const UChar* openZone = u"America/Vancouver";
471
472 static void TestOpen()
473 {
474 const char* locale;
475 const char** localesPtr = openLocales;
476 while ((locale = *localesPtr++) != NULL) {
477 UErrorCode status = U_ZERO_ERROR;
478 UDateIntervalFormat* udatintv = udtitvfmt_open(locale, openSkeleton, -1, openZone, -1, &status);
479 if ( U_FAILURE(status) ) {
480 log_err("FAIL: udtitvfmt_open for locale %s: %s\n", locale, u_errorName(status));
481 } else {
482 udtitvfmt_close(udatintv);
483 }
484 }
485 }
486
487 #endif /* #if !UCONFIG_NO_FORMATTING */