]>
Commit | Line | Data |
---|---|---|
ccd6aacd VZ |
1 | /////////////////////////////////////////////////////////////////////////////// |
2 | // Name: tests/datetime/datetime.cpp | |
3 | // Purpose: wxDateTime unit test | |
4 | // Author: Vadim Zeitlin | |
5 | // Created: 2004-06-23 (extracted from samples/console/console.cpp) | |
6 | // RCS-ID: $Id$ | |
7 | // Copyright: (c) 2004 Vadim Zeitlin <vadim@wxwindows.org> | |
8 | /////////////////////////////////////////////////////////////////////////////// | |
9 | ||
10 | // ---------------------------------------------------------------------------- | |
11 | // headers | |
12 | // ---------------------------------------------------------------------------- | |
13 | ||
14 | #include "wx/wxprec.h" | |
15 | ||
16 | #ifdef __BORLANDC__ | |
17 | #pragma hdrstop | |
18 | #endif | |
19 | ||
20 | #ifndef WX_PRECOMP | |
21 | #endif // WX_PRECOMP | |
22 | ||
23 | #include "wx/datetime.h" | |
24 | ||
25 | #include "wx/cppunit.h" | |
26 | ||
27 | // ---------------------------------------------------------------------------- | |
28 | // broken down date representation used for testing | |
29 | // ---------------------------------------------------------------------------- | |
30 | ||
31 | struct Date | |
32 | { | |
33 | wxDateTime::wxDateTime_t day; | |
34 | wxDateTime::Month month; | |
35 | int year; | |
36 | wxDateTime::wxDateTime_t hour, min, sec; | |
37 | double jdn; | |
38 | wxDateTime::WeekDay wday; | |
39 | time_t gmticks, ticks; | |
40 | ||
41 | void Init(const wxDateTime::Tm& tm) | |
42 | { | |
43 | day = tm.mday; | |
44 | month = tm.mon; | |
45 | year = tm.year; | |
46 | hour = tm.hour; | |
47 | min = tm.min; | |
48 | sec = tm.sec; | |
49 | jdn = 0.0; | |
50 | gmticks = ticks = -1; | |
51 | } | |
52 | ||
53 | wxDateTime DT() const | |
54 | { return wxDateTime(day, month, year, hour, min, sec); } | |
55 | ||
56 | bool SameDay(const wxDateTime::Tm& tm) const | |
57 | { | |
58 | return day == tm.mday && month == tm.mon && year == tm.year; | |
59 | } | |
60 | ||
61 | wxString Format() const | |
62 | { | |
63 | wxString s; | |
64 | s.Printf(_T("%02d:%02d:%02d %10s %02d, %4d%s"), | |
65 | hour, min, sec, | |
66 | wxDateTime::GetMonthName(month).c_str(), | |
67 | day, | |
68 | abs(wxDateTime::ConvertYearToBC(year)), | |
69 | year > 0 ? _T("AD") : _T("BC")); | |
70 | return s; | |
71 | } | |
72 | ||
73 | wxString FormatDate() const | |
74 | { | |
75 | wxString s; | |
76 | s.Printf(_T("%02d-%s-%4d%s"), | |
77 | day, | |
78 | wxDateTime::GetMonthName(month, wxDateTime::Name_Abbr).c_str(), | |
79 | abs(wxDateTime::ConvertYearToBC(year)), | |
80 | year > 0 ? _T("AD") : _T("BC")); | |
81 | return s; | |
82 | } | |
83 | }; | |
84 | ||
85 | // ---------------------------------------------------------------------------- | |
86 | // test data | |
87 | // ---------------------------------------------------------------------------- | |
88 | ||
89 | static const Date testDates[] = | |
90 | { | |
91 | { 1, wxDateTime::Jan, 1970, 00, 00, 00, 2440587.5, wxDateTime::Thu, 0, -3600 }, | |
92 | { 7, wxDateTime::Feb, 2036, 00, 00, 00, 2464730.5, wxDateTime::Thu, -1, -1 }, | |
93 | { 8, wxDateTime::Feb, 2036, 00, 00, 00, 2464731.5, wxDateTime::Fri, -1, -1 }, | |
94 | { 1, wxDateTime::Jan, 2037, 00, 00, 00, 2465059.5, wxDateTime::Thu, -1, -1 }, | |
95 | { 1, wxDateTime::Jan, 2038, 00, 00, 00, 2465424.5, wxDateTime::Fri, -1, -1 }, | |
96 | { 21, wxDateTime::Jan, 2222, 00, 00, 00, 2532648.5, wxDateTime::Mon, -1, -1 }, | |
97 | { 29, wxDateTime::May, 1976, 12, 00, 00, 2442928.0, wxDateTime::Sat, 202219200, 202212000 }, | |
98 | { 29, wxDateTime::Feb, 1976, 00, 00, 00, 2442837.5, wxDateTime::Sun, 194400000, 194396400 }, | |
99 | { 1, wxDateTime::Jan, 1900, 12, 00, 00, 2415021.0, wxDateTime::Mon, -1, -1 }, | |
100 | { 1, wxDateTime::Jan, 1900, 00, 00, 00, 2415020.5, wxDateTime::Mon, -1, -1 }, | |
101 | { 15, wxDateTime::Oct, 1582, 00, 00, 00, 2299160.5, wxDateTime::Fri, -1, -1 }, | |
102 | { 4, wxDateTime::Oct, 1582, 00, 00, 00, 2299149.5, wxDateTime::Mon, -1, -1 }, | |
103 | { 1, wxDateTime::Mar, 1, 00, 00, 00, 1721484.5, wxDateTime::Thu, -1, -1 }, | |
104 | { 1, wxDateTime::Jan, 1, 00, 00, 00, 1721425.5, wxDateTime::Mon, -1, -1 }, | |
105 | { 31, wxDateTime::Dec, 0, 00, 00, 00, 1721424.5, wxDateTime::Sun, -1, -1 }, | |
106 | { 1, wxDateTime::Jan, 0, 00, 00, 00, 1721059.5, wxDateTime::Sat, -1, -1 }, | |
107 | { 12, wxDateTime::Aug, -1234, 00, 00, 00, 1270573.5, wxDateTime::Fri, -1, -1 }, | |
108 | { 12, wxDateTime::Aug, -4000, 00, 00, 00, 260313.5, wxDateTime::Sat, -1, -1 }, | |
109 | { 24, wxDateTime::Nov, -4713, 00, 00, 00, -0.5, wxDateTime::Mon, -1, -1 }, | |
110 | }; | |
111 | ||
112 | ||
113 | // ---------------------------------------------------------------------------- | |
114 | // test class | |
115 | // ---------------------------------------------------------------------------- | |
116 | ||
117 | class DateTimeTestCase : public CppUnit::TestCase | |
118 | { | |
119 | public: | |
120 | DateTimeTestCase() { } | |
121 | ||
122 | private: | |
123 | CPPUNIT_TEST_SUITE( DateTimeTestCase ); | |
124 | CPPUNIT_TEST( TestLeapYears ); | |
125 | CPPUNIT_TEST( TestTimeSet ); | |
126 | CPPUNIT_TEST( TestTimeJDN ); | |
127 | CPPUNIT_TEST( TestTimeWNumber ); | |
128 | CPPUNIT_TEST( TestTimeWDays ); | |
129 | CPPUNIT_TEST( TestTimeDST ); | |
130 | CPPUNIT_TEST( TestTimeFormat ); | |
131 | CPPUNIT_TEST( TestTimeTicks ); | |
132 | CPPUNIT_TEST( TestTimeParse ); | |
133 | CPPUNIT_TEST( TestTimeArithmetics ); | |
134 | CPPUNIT_TEST_SUITE_END(); | |
135 | ||
136 | void TestLeapYears(); | |
137 | void TestTimeSet(); | |
138 | void TestTimeJDN(); | |
139 | void TestTimeWNumber(); | |
140 | void TestTimeWDays(); | |
141 | void TestTimeDST(); | |
142 | void TestTimeFormat(); | |
143 | void TestTimeTicks(); | |
144 | void TestTimeParse(); | |
145 | void TestTimeArithmetics(); | |
146 | ||
147 | DECLARE_NO_COPY_CLASS(DateTimeTestCase) | |
148 | }; | |
149 | ||
150 | // register in the unnamed registry so that these tests are run by default | |
151 | CPPUNIT_TEST_SUITE_REGISTRATION( DateTimeTestCase ); | |
152 | ||
153 | // also include in it's own registry so that these tests can be run alone | |
154 | CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( DateTimeTestCase, "DateTimeTestCase" ); | |
155 | ||
156 | // ============================================================================ | |
157 | // implementation | |
158 | // ============================================================================ | |
159 | ||
160 | // test leap years detection | |
161 | void DateTimeTestCase::TestLeapYears() | |
162 | { | |
163 | static const struct LeapYearTestData | |
164 | { | |
165 | int year; | |
166 | bool isLeap; | |
167 | } years[] = | |
168 | { | |
169 | { 1900, false }, | |
170 | { 1990, false }, | |
171 | { 1976, true }, | |
172 | { 2000, true }, | |
173 | { 2030, false }, | |
174 | { 1984, true }, | |
175 | { 2100, false }, | |
176 | { 2400, true }, | |
177 | }; | |
178 | ||
179 | for ( size_t n = 0; n < WXSIZEOF(years); n++ ) | |
180 | { | |
181 | const LeapYearTestData& y = years[n]; | |
182 | ||
183 | CPPUNIT_ASSERT( wxDateTime::IsLeapYear(y.year) == y.isLeap ); | |
184 | } | |
185 | } | |
186 | ||
187 | // test constructing wxDateTime objects | |
188 | void DateTimeTestCase::TestTimeSet() | |
189 | { | |
190 | for ( size_t n = 0; n < WXSIZEOF(testDates); n++ ) | |
191 | { | |
192 | const Date& d1 = testDates[n]; | |
193 | wxDateTime dt = d1.DT(); | |
194 | ||
195 | Date d2; | |
196 | d2.Init(dt.GetTm()); | |
197 | ||
198 | wxString s1 = d1.Format(), | |
199 | s2 = d2.Format(); | |
200 | ||
201 | CPPUNIT_ASSERT( s1 == s2 ); | |
202 | } | |
203 | } | |
204 | ||
205 | // test conversions to JDN &c | |
206 | void DateTimeTestCase::TestTimeJDN() | |
207 | { | |
208 | for ( size_t n = 0; n < WXSIZEOF(testDates); n++ ) | |
209 | { | |
210 | const Date& d = testDates[n]; | |
211 | wxDateTime dt(d.day, d.month, d.year, d.hour, d.min, d.sec); | |
212 | double jdn = dt.GetJulianDayNumber(); | |
213 | ||
214 | CPPUNIT_ASSERT( jdn == d.jdn ); | |
e3559425 VZ |
215 | |
216 | dt.Set(jdn); | |
217 | CPPUNIT_ASSERT( dt.GetJulianDayNumber() == jdn ); | |
ccd6aacd VZ |
218 | } |
219 | } | |
220 | ||
221 | // test week days computation | |
222 | void DateTimeTestCase::TestTimeWDays() | |
223 | { | |
224 | // test GetWeekDay() | |
225 | size_t n; | |
226 | for ( n = 0; n < WXSIZEOF(testDates); n++ ) | |
227 | { | |
228 | const Date& d = testDates[n]; | |
229 | wxDateTime dt(d.day, d.month, d.year, d.hour, d.min, d.sec); | |
230 | ||
231 | wxDateTime::WeekDay wday = dt.GetWeekDay(); | |
232 | CPPUNIT_ASSERT( wday == d.wday ); | |
233 | } | |
234 | ||
235 | // test SetToWeekDay() | |
236 | struct WeekDateTestData | |
237 | { | |
238 | Date date; // the real date (precomputed) | |
239 | int nWeek; // its week index in the month | |
240 | wxDateTime::WeekDay wday; // the weekday | |
241 | wxDateTime::Month month; // the month | |
242 | int year; // and the year | |
243 | ||
244 | wxString Format() const | |
245 | { | |
246 | wxString s, which; | |
247 | switch ( nWeek < -1 ? -nWeek : nWeek ) | |
248 | { | |
249 | case 1: which = _T("first"); break; | |
250 | case 2: which = _T("second"); break; | |
251 | case 3: which = _T("third"); break; | |
252 | case 4: which = _T("fourth"); break; | |
253 | case 5: which = _T("fifth"); break; | |
254 | ||
255 | case -1: which = _T("last"); break; | |
256 | } | |
257 | ||
258 | if ( nWeek < -1 ) | |
259 | { | |
260 | which += _T(" from end"); | |
261 | } | |
262 | ||
263 | s.Printf(_T("The %s %s of %s in %d"), | |
264 | which.c_str(), | |
265 | wxDateTime::GetWeekDayName(wday).c_str(), | |
266 | wxDateTime::GetMonthName(month).c_str(), | |
267 | year); | |
268 | ||
269 | return s; | |
270 | } | |
271 | }; | |
272 | ||
273 | // the array data was generated by the following python program | |
274 | /* | |
275 | from DateTime import * | |
276 | from whrandom import * | |
277 | from string import * | |
278 | ||
279 | monthNames = [ 'Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec' ] | |
280 | wdayNames = [ 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun' ] | |
281 | ||
282 | week = DateTimeDelta(7) | |
283 | ||
284 | for n in range(20): | |
285 | year = randint(1900, 2100) | |
286 | month = randint(1, 12) | |
287 | day = randint(1, 28) | |
288 | dt = DateTime(year, month, day) | |
289 | wday = dt.day_of_week | |
290 | ||
291 | countFromEnd = choice([-1, 1]) | |
292 | weekNum = 0; | |
293 | ||
294 | while dt.month is month: | |
295 | dt = dt - countFromEnd * week | |
296 | weekNum = weekNum + countFromEnd | |
297 | ||
298 | data = { 'day': rjust(`day`, 2), 'month': monthNames[month - 1], 'year': year, 'weekNum': rjust(`weekNum`, 2), 'wday': wdayNames[wday] } | |
299 | ||
300 | print "{ { %(day)s, wxDateTime::%(month)s, %(year)d }, %(weekNum)d, "\ | |
301 | "wxDateTime::%(wday)s, wxDateTime::%(month)s, %(year)d }," % data | |
302 | */ | |
303 | ||
304 | static const WeekDateTestData weekDatesTestData[] = | |
305 | { | |
306 | { { 20, wxDateTime::Mar, 2045 }, 3, wxDateTime::Mon, wxDateTime::Mar, 2045 }, | |
307 | { { 5, wxDateTime::Jun, 1985 }, -4, wxDateTime::Wed, wxDateTime::Jun, 1985 }, | |
308 | { { 12, wxDateTime::Nov, 1961 }, -3, wxDateTime::Sun, wxDateTime::Nov, 1961 }, | |
309 | { { 27, wxDateTime::Feb, 2093 }, -1, wxDateTime::Fri, wxDateTime::Feb, 2093 }, | |
310 | { { 4, wxDateTime::Jul, 2070 }, -4, wxDateTime::Fri, wxDateTime::Jul, 2070 }, | |
311 | { { 2, wxDateTime::Apr, 1906 }, -5, wxDateTime::Mon, wxDateTime::Apr, 1906 }, | |
312 | { { 19, wxDateTime::Jul, 2023 }, -2, wxDateTime::Wed, wxDateTime::Jul, 2023 }, | |
313 | { { 5, wxDateTime::May, 1958 }, -4, wxDateTime::Mon, wxDateTime::May, 1958 }, | |
314 | { { 11, wxDateTime::Aug, 1900 }, 2, wxDateTime::Sat, wxDateTime::Aug, 1900 }, | |
315 | { { 14, wxDateTime::Feb, 1945 }, 2, wxDateTime::Wed, wxDateTime::Feb, 1945 }, | |
316 | { { 25, wxDateTime::Jul, 1967 }, -1, wxDateTime::Tue, wxDateTime::Jul, 1967 }, | |
317 | { { 9, wxDateTime::May, 1916 }, -4, wxDateTime::Tue, wxDateTime::May, 1916 }, | |
318 | { { 20, wxDateTime::Jun, 1927 }, 3, wxDateTime::Mon, wxDateTime::Jun, 1927 }, | |
319 | { { 2, wxDateTime::Aug, 2000 }, 1, wxDateTime::Wed, wxDateTime::Aug, 2000 }, | |
320 | { { 20, wxDateTime::Apr, 2044 }, 3, wxDateTime::Wed, wxDateTime::Apr, 2044 }, | |
321 | { { 20, wxDateTime::Feb, 1932 }, -2, wxDateTime::Sat, wxDateTime::Feb, 1932 }, | |
322 | { { 25, wxDateTime::Jul, 2069 }, 4, wxDateTime::Thu, wxDateTime::Jul, 2069 }, | |
323 | { { 3, wxDateTime::Apr, 1925 }, 1, wxDateTime::Fri, wxDateTime::Apr, 1925 }, | |
324 | { { 21, wxDateTime::Mar, 2093 }, 3, wxDateTime::Sat, wxDateTime::Mar, 2093 }, | |
325 | { { 3, wxDateTime::Dec, 2074 }, -5, wxDateTime::Mon, wxDateTime::Dec, 2074 }, | |
1c5d27e2 | 326 | |
ccd6aacd VZ |
327 | }; |
328 | ||
ccd6aacd VZ |
329 | wxDateTime dt; |
330 | for ( n = 0; n < WXSIZEOF(weekDatesTestData); n++ ) | |
331 | { | |
332 | const WeekDateTestData& wd = weekDatesTestData[n]; | |
333 | ||
334 | dt.SetToWeekDay(wd.wday, wd.nWeek, wd.month, wd.year); | |
335 | ||
336 | const Date& d = wd.date; | |
337 | CPPUNIT_ASSERT( d.SameDay(dt.GetTm()) ); | |
338 | } | |
339 | } | |
340 | ||
341 | // test the computation of (ISO) week numbers | |
342 | void DateTimeTestCase::TestTimeWNumber() | |
343 | { | |
344 | struct WeekNumberTestData | |
345 | { | |
346 | Date date; // the date | |
347 | wxDateTime::wxDateTime_t week; // the week number in the year | |
348 | wxDateTime::wxDateTime_t wmon; // the week number in the month | |
349 | wxDateTime::wxDateTime_t wmon2; // same but week starts with Sun | |
350 | wxDateTime::wxDateTime_t dnum; // day number in the year | |
351 | }; | |
352 | ||
353 | // data generated with the following python script: | |
354 | /* | |
355 | from DateTime import * | |
356 | from whrandom import * | |
357 | from string import * | |
358 | ||
359 | monthNames = [ 'Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec' ] | |
360 | wdayNames = [ 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun' ] | |
361 | ||
362 | def GetMonthWeek(dt): | |
363 | weekNumMonth = dt.iso_week[1] - DateTime(dt.year, dt.month, 1).iso_week[1] + 1 | |
364 | if weekNumMonth < 0: | |
365 | weekNumMonth = weekNumMonth + 53 | |
366 | return weekNumMonth | |
367 | ||
368 | def GetLastSundayBefore(dt): | |
369 | if dt.iso_week[2] == 7: | |
370 | return dt | |
371 | else: | |
372 | return dt - DateTimeDelta(dt.iso_week[2]) | |
373 | ||
374 | for n in range(20): | |
375 | year = randint(1900, 2100) | |
376 | month = randint(1, 12) | |
377 | day = randint(1, 28) | |
378 | dt = DateTime(year, month, day) | |
379 | dayNum = dt.day_of_year | |
380 | weekNum = dt.iso_week[1] | |
381 | weekNumMonth = GetMonthWeek(dt) | |
382 | ||
383 | weekNumMonth2 = 0 | |
384 | dtSunday = GetLastSundayBefore(dt) | |
385 | ||
386 | while dtSunday >= GetLastSundayBefore(DateTime(dt.year, dt.month, 1)): | |
387 | weekNumMonth2 = weekNumMonth2 + 1 | |
388 | dtSunday = dtSunday - DateTimeDelta(7) | |
389 | ||
390 | data = { 'day': rjust(`day`, 2), \ | |
391 | 'month': monthNames[month - 1], \ | |
392 | 'year': year, \ | |
393 | 'weekNum': rjust(`weekNum`, 2), \ | |
394 | 'weekNumMonth': weekNumMonth, \ | |
395 | 'weekNumMonth2': weekNumMonth2, \ | |
396 | 'dayNum': rjust(`dayNum`, 3) } | |
397 | ||
398 | print " { { %(day)s, "\ | |
399 | "wxDateTime::%(month)s, "\ | |
400 | "%(year)d }, "\ | |
401 | "%(weekNum)s, "\ | |
402 | "%(weekNumMonth)s, "\ | |
403 | "%(weekNumMonth2)s, "\ | |
404 | "%(dayNum)s }," % data | |
405 | ||
406 | */ | |
407 | static const WeekNumberTestData weekNumberTestDates[] = | |
408 | { | |
409 | { { 27, wxDateTime::Dec, 1966 }, 52, 5, 5, 361 }, | |
410 | { { 22, wxDateTime::Jul, 1926 }, 29, 4, 4, 203 }, | |
411 | { { 22, wxDateTime::Oct, 2076 }, 43, 4, 4, 296 }, | |
412 | { { 1, wxDateTime::Jul, 1967 }, 26, 1, 1, 182 }, | |
413 | { { 8, wxDateTime::Nov, 2004 }, 46, 2, 2, 313 }, | |
414 | { { 21, wxDateTime::Mar, 1920 }, 12, 3, 4, 81 }, | |
415 | { { 7, wxDateTime::Jan, 1965 }, 1, 2, 2, 7 }, | |
416 | { { 19, wxDateTime::Oct, 1999 }, 42, 4, 4, 292 }, | |
417 | { { 13, wxDateTime::Aug, 1955 }, 32, 2, 2, 225 }, | |
418 | { { 18, wxDateTime::Jul, 2087 }, 29, 3, 3, 199 }, | |
419 | { { 2, wxDateTime::Sep, 2028 }, 35, 1, 1, 246 }, | |
420 | { { 28, wxDateTime::Jul, 1945 }, 30, 5, 4, 209 }, | |
421 | { { 15, wxDateTime::Jun, 1901 }, 24, 3, 3, 166 }, | |
422 | { { 10, wxDateTime::Oct, 1939 }, 41, 3, 2, 283 }, | |
423 | { { 3, wxDateTime::Dec, 1965 }, 48, 1, 1, 337 }, | |
424 | { { 23, wxDateTime::Feb, 1940 }, 8, 4, 4, 54 }, | |
425 | { { 2, wxDateTime::Jan, 1987 }, 1, 1, 1, 2 }, | |
426 | { { 11, wxDateTime::Aug, 2079 }, 32, 2, 2, 223 }, | |
427 | { { 2, wxDateTime::Feb, 2063 }, 5, 1, 1, 33 }, | |
428 | { { 16, wxDateTime::Oct, 1942 }, 42, 3, 3, 289 }, | |
1c5d27e2 VZ |
429 | { { 30, wxDateTime::Dec, 2003 }, 1, 5, 5, 364 }, |
430 | { { 2, wxDateTime::Jan, 2004 }, 1, 1, 1, 2 }, | |
ccd6aacd VZ |
431 | }; |
432 | ||
433 | for ( size_t n = 0; n < WXSIZEOF(weekNumberTestDates); n++ ) | |
434 | { | |
435 | const WeekNumberTestData& wn = weekNumberTestDates[n]; | |
436 | const Date& d = wn.date; | |
437 | ||
438 | wxDateTime dt = d.DT(); | |
439 | ||
440 | wxDateTime::wxDateTime_t | |
441 | week = dt.GetWeekOfYear(wxDateTime::Monday_First), | |
442 | wmon = dt.GetWeekOfMonth(wxDateTime::Monday_First), | |
443 | wmon2 = dt.GetWeekOfMonth(wxDateTime::Sunday_First), | |
444 | dnum = dt.GetDayOfYear(); | |
445 | ||
446 | CPPUNIT_ASSERT( dnum == wn.dnum ); | |
447 | CPPUNIT_ASSERT( wmon == wn.wmon ); | |
448 | CPPUNIT_ASSERT( wmon2 == wn.wmon2 ); | |
449 | CPPUNIT_ASSERT( week == wn.week ); | |
450 | ||
4c27e2fa VZ |
451 | int year = d.year; |
452 | if ( week == 1 && d.month != wxDateTime::Jan ) | |
453 | { | |
454 | // this means we're in the first week of the next year | |
455 | year++; | |
456 | } | |
457 | ||
458 | wxDateTime | |
459 | dt2 = wxDateTime::SetToWeekOfYear(year, week, dt.GetWeekDay()); | |
ccd6aacd VZ |
460 | CPPUNIT_ASSERT( dt2 == dt ); |
461 | } | |
462 | } | |
463 | ||
464 | // test DST applicability | |
465 | void DateTimeTestCase::TestTimeDST() | |
466 | { | |
467 | // taken from http://www.energy.ca.gov/daylightsaving.html | |
468 | static const Date datesDST[2][2004 - 1900 + 1] = | |
469 | { | |
470 | { | |
471 | { 1, wxDateTime::Apr, 1990 }, | |
472 | { 7, wxDateTime::Apr, 1991 }, | |
473 | { 5, wxDateTime::Apr, 1992 }, | |
474 | { 4, wxDateTime::Apr, 1993 }, | |
475 | { 3, wxDateTime::Apr, 1994 }, | |
476 | { 2, wxDateTime::Apr, 1995 }, | |
477 | { 7, wxDateTime::Apr, 1996 }, | |
478 | { 6, wxDateTime::Apr, 1997 }, | |
479 | { 5, wxDateTime::Apr, 1998 }, | |
480 | { 4, wxDateTime::Apr, 1999 }, | |
481 | { 2, wxDateTime::Apr, 2000 }, | |
482 | { 1, wxDateTime::Apr, 2001 }, | |
483 | { 7, wxDateTime::Apr, 2002 }, | |
484 | { 6, wxDateTime::Apr, 2003 }, | |
485 | { 4, wxDateTime::Apr, 2004 }, | |
486 | }, | |
487 | { | |
488 | { 28, wxDateTime::Oct, 1990 }, | |
489 | { 27, wxDateTime::Oct, 1991 }, | |
490 | { 25, wxDateTime::Oct, 1992 }, | |
491 | { 31, wxDateTime::Oct, 1993 }, | |
492 | { 30, wxDateTime::Oct, 1994 }, | |
493 | { 29, wxDateTime::Oct, 1995 }, | |
494 | { 27, wxDateTime::Oct, 1996 }, | |
495 | { 26, wxDateTime::Oct, 1997 }, | |
496 | { 25, wxDateTime::Oct, 1998 }, | |
497 | { 31, wxDateTime::Oct, 1999 }, | |
498 | { 29, wxDateTime::Oct, 2000 }, | |
499 | { 28, wxDateTime::Oct, 2001 }, | |
500 | { 27, wxDateTime::Oct, 2002 }, | |
501 | { 26, wxDateTime::Oct, 2003 }, | |
502 | { 31, wxDateTime::Oct, 2004 }, | |
503 | } | |
504 | }; | |
505 | ||
506 | for ( int year = 1990; year < 2005; year++ ) | |
507 | { | |
508 | wxDateTime dtBegin = wxDateTime::GetBeginDST(year, wxDateTime::USA), | |
509 | dtEnd = wxDateTime::GetEndDST(year, wxDateTime::USA); | |
510 | ||
511 | size_t n = year - 1990; | |
512 | const Date& dBegin = datesDST[0][n]; | |
513 | const Date& dEnd = datesDST[1][n]; | |
514 | ||
515 | CPPUNIT_ASSERT( dBegin.SameDay(dtBegin.GetTm()) ); | |
516 | CPPUNIT_ASSERT( dEnd.SameDay(dtEnd.GetTm()) ); | |
517 | } | |
518 | } | |
519 | ||
520 | // test wxDateTime -> text conversion | |
521 | void DateTimeTestCase::TestTimeFormat() | |
522 | { | |
523 | // some information may be lost during conversion, so store what kind | |
524 | // of info should we recover after a round trip | |
525 | enum CompareKind | |
526 | { | |
527 | CompareNone, // don't try comparing | |
528 | CompareBoth, // dates and times should be identical | |
529 | CompareDate, // dates only | |
530 | CompareTime // time only | |
531 | }; | |
532 | ||
533 | static const struct | |
534 | { | |
535 | CompareKind compareKind; | |
536 | const wxChar *format; | |
537 | } formatTestFormats[] = | |
538 | { | |
539 | { CompareBoth, _T("---> %c") }, | |
540 | { CompareDate, _T("Date is %A, %d of %B, in year %Y") }, | |
541 | { CompareBoth, _T("Date is %x, time is %X") }, | |
542 | { CompareTime, _T("Time is %H:%M:%S or %I:%M:%S %p") }, | |
543 | { CompareNone, _T("The day of year: %j, the week of year: %W") }, | |
544 | { CompareDate, _T("ISO date without separators: %Y%m%d") }, | |
545 | }; | |
546 | ||
547 | static const Date formatTestDates[] = | |
548 | { | |
549 | { 29, wxDateTime::May, 1976, 18, 30, 00 }, | |
550 | { 31, wxDateTime::Dec, 1999, 23, 30, 00 }, | |
551 | #if 0 | |
552 | // this test can't work for other centuries because it uses two digit | |
553 | // years in formats, so don't even try it | |
554 | { 29, wxDateTime::May, 2076, 18, 30, 00 }, | |
555 | { 29, wxDateTime::Feb, 2400, 02, 15, 25 }, | |
556 | { 01, wxDateTime::Jan, -52, 03, 16, 47 }, | |
557 | #endif | |
558 | }; | |
559 | ||
560 | for ( size_t d = 0; d < WXSIZEOF(formatTestDates) + 1; d++ ) | |
561 | { | |
562 | wxDateTime dt = d == 0 ? wxDateTime::Now() : formatTestDates[d - 1].DT(); | |
563 | for ( size_t n = 0; n < WXSIZEOF(formatTestFormats); n++ ) | |
564 | { | |
565 | wxString s = dt.Format(formatTestFormats[n].format); | |
566 | ||
567 | // what can we recover? | |
568 | int kind = formatTestFormats[n].compareKind; | |
569 | ||
570 | // convert back | |
571 | wxDateTime dt2; | |
572 | const wxChar *result = dt2.ParseFormat(s, formatTestFormats[n].format); | |
573 | if ( !result ) | |
574 | { | |
575 | // converion failed - should it have? | |
576 | CPPUNIT_ASSERT( kind == CompareNone ); | |
577 | } | |
578 | else // conversion succeeded | |
579 | { | |
580 | // should have parsed the entire string | |
581 | CPPUNIT_ASSERT( !*result ); | |
582 | ||
583 | switch ( kind ) | |
584 | { | |
585 | case CompareBoth: | |
586 | CPPUNIT_ASSERT( dt2 == dt ); | |
587 | break; | |
588 | ||
589 | case CompareDate: | |
590 | CPPUNIT_ASSERT( dt.IsSameDate(dt2) ); | |
591 | break; | |
592 | ||
593 | case CompareTime: | |
594 | CPPUNIT_ASSERT( dt.IsSameTime(dt2) ); | |
595 | break; | |
596 | } | |
597 | } | |
598 | } | |
599 | } | |
600 | } | |
601 | ||
602 | void DateTimeTestCase::TestTimeTicks() | |
603 | { | |
604 | for ( size_t n = 0; n < WXSIZEOF(testDates); n++ ) | |
605 | { | |
606 | const Date& d = testDates[n]; | |
607 | if ( d.ticks == -1 ) | |
608 | continue; | |
609 | ||
610 | wxDateTime dt = d.DT(); | |
aa3698d5 RN |
611 | //RN: Translate according to test's time zone |
612 | //2nd param is to ignore DST - it's already factored | |
613 | //into Vadim's tests | |
614 | dt.MakeTimezone(wxDateTime::WEST, true); | |
ccd6aacd VZ |
615 | long ticks = (dt.GetValue() / 1000).ToLong(); |
616 | CPPUNIT_ASSERT( ticks == d.ticks ); | |
617 | ||
618 | dt = d.DT().ToTimezone(wxDateTime::GMT0); | |
619 | ticks = (dt.GetValue() / 1000).ToLong(); | |
620 | CPPUNIT_ASSERT( ticks == d.gmticks ); | |
621 | } | |
622 | } | |
623 | ||
624 | // test text -> wxDateTime conversion | |
625 | void DateTimeTestCase::TestTimeParse() | |
626 | { | |
627 | static const struct ParseTestData | |
628 | { | |
629 | const wxChar *format; | |
630 | Date date; | |
631 | bool good; | |
632 | } parseTestDates[] = | |
633 | { | |
634 | { _T("Sat, 18 Dec 1999 00:46:40 +0100"), { 18, wxDateTime::Dec, 1999, 00, 46, 40 }, true }, | |
635 | { _T("Wed, 1 Dec 1999 05:17:20 +0300"), { 1, wxDateTime::Dec, 1999, 03, 17, 20 }, true }, | |
636 | }; | |
637 | ||
638 | for ( size_t n = 0; n < WXSIZEOF(parseTestDates); n++ ) | |
639 | { | |
640 | const wxChar *format = parseTestDates[n].format; | |
641 | ||
642 | wxDateTime dt; | |
643 | if ( dt.ParseRfc822Date(format) ) | |
644 | { | |
645 | CPPUNIT_ASSERT( parseTestDates[n].good ); | |
646 | ||
647 | wxDateTime dtReal = parseTestDates[n].date.DT(); | |
aa3698d5 RN |
648 | //RN: We need this because the tests are based on |
649 | //a non-GMT time zone | |
650 | dtReal.MakeTimezone(wxDateTime::WEST, true); | |
ccd6aacd VZ |
651 | CPPUNIT_ASSERT( dt == dtReal ); |
652 | } | |
653 | else // failed to parse | |
654 | { | |
655 | CPPUNIT_ASSERT( !parseTestDates[n].good ); | |
656 | } | |
657 | } | |
658 | } | |
659 | ||
660 | void DateTimeTestCase::TestTimeArithmetics() | |
661 | { | |
662 | static const wxDateSpan testArithmData[] = | |
663 | { | |
664 | wxDateSpan::Day(), | |
665 | wxDateSpan::Week(), | |
666 | wxDateSpan::Month(), | |
667 | wxDateSpan::Year(), | |
668 | }; | |
669 | ||
670 | // the test will *not* work with arbitrary date! | |
671 | wxDateTime dt(2, wxDateTime::Dec, 1999), | |
672 | dt1, | |
673 | dt2; | |
674 | ||
675 | for ( size_t n = 0; n < WXSIZEOF(testArithmData); n++ ) | |
676 | { | |
677 | const wxDateSpan& span = testArithmData[n]; | |
678 | dt1 = dt + span; | |
679 | dt2 = dt - span; | |
680 | ||
681 | CPPUNIT_ASSERT( dt1 - span == dt ); | |
682 | CPPUNIT_ASSERT( dt2 + span == dt ); | |
683 | CPPUNIT_ASSERT( dt2 + 2*span == dt1 ); | |
684 | } | |
685 | } | |
686 |