]> git.saurik.com Git - wxWidgets.git/blob - src/common/datetime.cpp
fixed GetWeekOfYear() for first/last week in some cases (patch 908793)
[wxWidgets.git] / src / common / datetime.cpp
1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: wx/datetime.h
3 // Purpose: implementation of time/date related classes
4 // Author: Vadim Zeitlin
5 // Modified by:
6 // Created: 11.05.99
7 // RCS-ID: $Id$
8 // Copyright: (c) 1999 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9 // parts of code taken from sndcal library by Scott E. Lee:
10 //
11 // Copyright 1993-1995, Scott E. Lee, all rights reserved.
12 // Permission granted to use, copy, modify, distribute and sell
13 // so long as the above copyright and this permission statement
14 // are retained in all copies.
15 //
16 // Licence: wxWindows licence
17 ///////////////////////////////////////////////////////////////////////////////
18
19 /*
20 * Implementation notes:
21 *
22 * 1. the time is stored as a 64bit integer containing the signed number of
23 * milliseconds since Jan 1. 1970 (the Unix Epoch) - so it is always
24 * expressed in GMT.
25 *
26 * 2. the range is thus something about 580 million years, but due to current
27 * algorithms limitations, only dates from Nov 24, 4714BC are handled
28 *
29 * 3. standard ANSI C functions are used to do time calculations whenever
30 * possible, i.e. when the date is in the range Jan 1, 1970 to 2038
31 *
32 * 4. otherwise, the calculations are done by converting the date to/from JDN
33 * first (the range limitation mentioned above comes from here: the
34 * algorithm used by Scott E. Lee's code only works for positive JDNs, more
35 * or less)
36 *
37 * 5. the object constructed for the given DD-MM-YYYY HH:MM:SS corresponds to
38 * this moment in local time and may be converted to the object
39 * corresponding to the same date/time in another time zone by using
40 * ToTimezone()
41 *
42 * 6. the conversions to the current (or any other) timezone are done when the
43 * internal time representation is converted to the broken-down one in
44 * wxDateTime::Tm.
45 */
46
47 // ============================================================================
48 // declarations
49 // ============================================================================
50
51 // ----------------------------------------------------------------------------
52 // headers
53 // ----------------------------------------------------------------------------
54
55 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
56 #pragma implementation "datetime.h"
57 #endif
58
59 // For compilers that support precompilation, includes "wx.h".
60 #include "wx/wxprec.h"
61
62 #ifdef __BORLANDC__
63 #pragma hdrstop
64 #endif
65
66 #if !defined(wxUSE_DATETIME) || wxUSE_DATETIME
67
68 #ifndef WX_PRECOMP
69 #include "wx/string.h"
70 #include "wx/log.h"
71 #endif // WX_PRECOMP
72
73 #include "wx/intl.h"
74 #include "wx/thread.h"
75 #include "wx/tokenzr.h"
76 #include "wx/module.h"
77
78 #include <ctype.h>
79
80 #include "wx/datetime.h"
81 #include "wx/stopwatch.h" // for wxGetLocalTimeMillis()
82
83 const long wxDateTime::TIME_T_FACTOR = 1000l;
84
85 #if wxUSE_EXTENDED_RTTI
86
87 template<> void wxStringReadValue(const wxString &s , wxDateTime &data )
88 {
89 data.ParseFormat(s,wxT("%Y-%m-%d %H:%M:%S")) ;
90 }
91
92 template<> void wxStringWriteValue(wxString &s , const wxDateTime &data )
93 {
94 s = data.Format(wxT("%Y-%m-%d %H:%M:%S")) ;
95 }
96
97 wxCUSTOM_TYPE_INFO(wxDateTime, wxToStringConverter<wxDateTime> , wxFromStringConverter<wxDateTime>)
98
99 #endif
100
101 //
102 // ----------------------------------------------------------------------------
103 // conditional compilation
104 // ----------------------------------------------------------------------------
105
106 #if defined(HAVE_STRPTIME) && defined(__GLIBC__) && \
107 ((__GLIBC__ == 2) && (__GLIBC_MINOR__ == 0))
108 // glibc 2.0.7 strptime() is broken - the following snippet causes it to
109 // crash (instead of just failing):
110 //
111 // strncpy(buf, "Tue Dec 21 20:25:40 1999", 128);
112 // strptime(buf, "%x", &tm);
113 //
114 // so don't use it
115 #undef HAVE_STRPTIME
116 #endif // broken strptime()
117
118 #if defined(__MWERKS__) && wxUSE_UNICODE
119 #include <wtime.h>
120 #endif
121
122 #if !defined(WX_TIMEZONE) && !defined(WX_GMTOFF_IN_TM)
123 #if defined(__BORLANDC__) || defined(__MINGW32__) || defined(__VISAGECPP__)
124 #define WX_TIMEZONE _timezone
125 #elif defined(__MWERKS__)
126 long wxmw_timezone = 28800;
127 #define WX_TIMEZONE wxmw_timezone
128 #elif defined(__DJGPP__) || defined(__WINE__)
129 #include <sys/timeb.h>
130 #include <values.h>
131 static long wxGetTimeZone()
132 {
133 static long timezone = MAXLONG; // invalid timezone
134 if (timezone == MAXLONG)
135 {
136 struct timeb tb;
137 ftime(&tb);
138 timezone = tb.timezone;
139 }
140 return timezone;
141 }
142 #define WX_TIMEZONE wxGetTimeZone()
143 #elif defined(__DARWIN__)
144 #define WX_GMTOFF_IN_TM
145 #else // unknown platform - try timezone
146 #define WX_TIMEZONE timezone
147 #endif
148 #endif // !WX_TIMEZONE && !WX_GMTOFF_IN_TM
149
150 // ----------------------------------------------------------------------------
151 // macros
152 // ----------------------------------------------------------------------------
153
154 // debugging helper: just a convenient replacement of wxCHECK()
155 #define wxDATETIME_CHECK(expr, msg) \
156 if ( !(expr) ) \
157 { \
158 wxFAIL_MSG(msg); \
159 *this = wxInvalidDateTime; \
160 return *this; \
161 }
162
163 // ----------------------------------------------------------------------------
164 // private classes
165 // ----------------------------------------------------------------------------
166
167 class wxDateTimeHolidaysModule : public wxModule
168 {
169 public:
170 virtual bool OnInit()
171 {
172 wxDateTimeHolidayAuthority::AddAuthority(new wxDateTimeWorkDays);
173
174 return true;
175 }
176
177 virtual void OnExit()
178 {
179 wxDateTimeHolidayAuthority::ClearAllAuthorities();
180 wxDateTimeHolidayAuthority::ms_authorities.clear();
181 }
182
183 private:
184 DECLARE_DYNAMIC_CLASS(wxDateTimeHolidaysModule)
185 };
186
187 IMPLEMENT_DYNAMIC_CLASS(wxDateTimeHolidaysModule, wxModule)
188
189 // ----------------------------------------------------------------------------
190 // constants
191 // ----------------------------------------------------------------------------
192
193 // some trivial ones
194 static const int MONTHS_IN_YEAR = 12;
195
196 static const int SEC_PER_MIN = 60;
197
198 static const int MIN_PER_HOUR = 60;
199
200 static const int HOURS_PER_DAY = 24;
201
202 static const long SECONDS_PER_DAY = 86400l;
203
204 static const int DAYS_PER_WEEK = 7;
205
206 static const long MILLISECONDS_PER_DAY = 86400000l;
207
208 // this is the integral part of JDN of the midnight of Jan 1, 1970
209 // (i.e. JDN(Jan 1, 1970) = 2440587.5)
210 static const long EPOCH_JDN = 2440587l;
211
212 // the date of JDN -0.5 (as we don't work with fractional parts, this is the
213 // reference date for us) is Nov 24, 4714BC
214 static const int JDN_0_YEAR = -4713;
215 static const int JDN_0_MONTH = wxDateTime::Nov;
216 static const int JDN_0_DAY = 24;
217
218 // the constants used for JDN calculations
219 static const long JDN_OFFSET = 32046l;
220 static const long DAYS_PER_5_MONTHS = 153l;
221 static const long DAYS_PER_4_YEARS = 1461l;
222 static const long DAYS_PER_400_YEARS = 146097l;
223
224 // this array contains the cumulated number of days in all previous months for
225 // normal and leap years
226 static const wxDateTime::wxDateTime_t gs_cumulatedDays[2][MONTHS_IN_YEAR] =
227 {
228 { 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334 },
229 { 0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335 }
230 };
231
232 // ----------------------------------------------------------------------------
233 // global data
234 // ----------------------------------------------------------------------------
235
236 // in the fine tradition of ANSI C we use our equivalent of (time_t)-1 to
237 // indicate an invalid wxDateTime object
238 const wxDateTime wxDefaultDateTime;
239
240 wxDateTime::Country wxDateTime::ms_country = wxDateTime::Country_Unknown;
241
242 // ----------------------------------------------------------------------------
243 // private functions
244 // ----------------------------------------------------------------------------
245
246 // debugger helper: shows what the date really is
247 #ifdef __WXDEBUG__
248 extern const wxChar *wxDumpDate(const wxDateTime* dt)
249 {
250 static wxChar buf[128];
251
252 wxStrcpy(buf, dt->Format(_T("%Y-%m-%d (%a) %H:%M:%S")));
253
254 return buf;
255 }
256 #endif // Debug
257
258 // get the number of days in the given month of the given year
259 static inline
260 wxDateTime::wxDateTime_t GetNumOfDaysInMonth(int year, wxDateTime::Month month)
261 {
262 // the number of days in month in Julian/Gregorian calendar: the first line
263 // is for normal years, the second one is for the leap ones
264 static wxDateTime::wxDateTime_t daysInMonth[2][MONTHS_IN_YEAR] =
265 {
266 { 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 },
267 { 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 }
268 };
269
270 return daysInMonth[wxDateTime::IsLeapYear(year)][month];
271 }
272
273 // returns the time zone in the C sense, i.e. the difference UTC - local
274 // (in seconds)
275 static int GetTimeZone()
276 {
277 #ifdef WX_GMTOFF_IN_TM
278 // set to true when the timezone is set
279 static bool s_timezoneSet = false;
280 static long gmtoffset = LONG_MAX; // invalid timezone
281
282 // ensure that the timezone variable is set by calling localtime
283 if ( !s_timezoneSet )
284 {
285 // just call localtime() instead of figuring out whether this system
286 // supports tzset(), _tzset() or something else
287 time_t t = 0;
288 struct tm *tm;
289
290 tm = localtime(&t);
291 s_timezoneSet = true;
292
293 // note that GMT offset is the opposite of time zone and so to return
294 // consistent results in both WX_GMTOFF_IN_TM and !WX_GMTOFF_IN_TM
295 // cases we have to negate it
296 gmtoffset = -tm->tm_gmtoff;
297 }
298
299 return (int)gmtoffset;
300 #else // !WX_GMTOFF_IN_TM
301 return (int)WX_TIMEZONE;
302 #endif // WX_GMTOFF_IN_TM/!WX_GMTOFF_IN_TM
303 }
304
305 // return the integral part of the JDN for the midnight of the given date (to
306 // get the real JDN you need to add 0.5, this is, in fact, JDN of the
307 // noon of the previous day)
308 static long GetTruncatedJDN(wxDateTime::wxDateTime_t day,
309 wxDateTime::Month mon,
310 int year)
311 {
312 // CREDIT: code below is by Scott E. Lee (but bugs are mine)
313
314 // check the date validity
315 wxASSERT_MSG(
316 (year > JDN_0_YEAR) ||
317 ((year == JDN_0_YEAR) && (mon > JDN_0_MONTH)) ||
318 ((year == JDN_0_YEAR) && (mon == JDN_0_MONTH) && (day >= JDN_0_DAY)),
319 _T("date out of range - can't convert to JDN")
320 );
321
322 // make the year positive to avoid problems with negative numbers division
323 year += 4800;
324
325 // months are counted from March here
326 int month;
327 if ( mon >= wxDateTime::Mar )
328 {
329 month = mon - 2;
330 }
331 else
332 {
333 month = mon + 10;
334 year--;
335 }
336
337 // now we can simply add all the contributions together
338 return ((year / 100) * DAYS_PER_400_YEARS) / 4
339 + ((year % 100) * DAYS_PER_4_YEARS) / 4
340 + (month * DAYS_PER_5_MONTHS + 2) / 5
341 + day
342 - JDN_OFFSET;
343 }
344
345 // this function is a wrapper around strftime(3) adding error checking
346 static wxString CallStrftime(const wxChar *format, const tm* tm)
347 {
348 wxChar buf[4096];
349 if ( !wxStrftime(buf, WXSIZEOF(buf), format, tm) )
350 {
351 // buffer is too small?
352 wxFAIL_MSG(_T("strftime() failed"));
353 }
354
355 return wxString(buf);
356 }
357
358 #ifdef HAVE_STRPTIME
359
360 // glibc2 doesn't define this in the headers unless _XOPEN_SOURCE is defined
361 // which, unfortunately, wreaks havoc elsewhere
362 #if defined(__GLIBC__) && (__GLIBC__ == 2)
363 extern "C" char *strptime(const char *, const char *, struct tm *);
364 #endif
365
366 // Unicode-friendly strptime() wrapper
367 static const wxChar *
368 CallStrptime(const wxChar *input, const char *fmt, tm *tm)
369 {
370 // the problem here is that strptime() returns pointer into the string we
371 // passed to it while we're really interested in the pointer into the
372 // original, Unicode, string so we try to transform the pointer back
373 #if wxUSE_UNICODE
374 wxCharBuffer inputMB(wxConvertWX2MB(input));
375 #else // ASCII
376 const char * const inputMB = input;
377 #endif // Unicode/Ascii
378
379 const char *result = strptime(inputMB, fmt, tm);
380 if ( !result )
381 return NULL;
382
383 #if wxUSE_UNICODE
384 // FIXME: this is wrong in presence of surrogates &c
385 return input + (result - inputMB.data());
386 #else // ASCII
387 return result;
388 #endif // Unicode/Ascii
389 }
390
391 #endif // HAVE_STRPTIME
392
393 // if year and/or month have invalid values, replace them with the current ones
394 static void ReplaceDefaultYearMonthWithCurrent(int *year,
395 wxDateTime::Month *month)
396 {
397 struct tm *tmNow = NULL;
398
399 if ( *year == wxDateTime::Inv_Year )
400 {
401 tmNow = wxDateTime::GetTmNow();
402
403 *year = 1900 + tmNow->tm_year;
404 }
405
406 if ( *month == wxDateTime::Inv_Month )
407 {
408 if ( !tmNow )
409 tmNow = wxDateTime::GetTmNow();
410
411 *month = (wxDateTime::Month)tmNow->tm_mon;
412 }
413 }
414
415 // fll the struct tm with default values
416 static void InitTm(struct tm& tm)
417 {
418 // struct tm may have etxra fields (undocumented and with unportable
419 // names) which, nevertheless, must be set to 0
420 memset(&tm, 0, sizeof(struct tm));
421
422 tm.tm_mday = 1; // mday 0 is invalid
423 tm.tm_year = 76; // any valid year
424 tm.tm_isdst = -1; // auto determine
425 }
426
427 // parsing helpers
428 // ---------------
429
430 // return the month if the string is a month name or Inv_Month otherwise
431 static wxDateTime::Month GetMonthFromName(const wxString& name, int flags)
432 {
433 wxDateTime::Month mon;
434 for ( mon = wxDateTime::Jan; mon < wxDateTime::Inv_Month; wxNextMonth(mon) )
435 {
436 // case-insensitive comparison either one of or with both abbreviated
437 // and not versions
438 if ( flags & wxDateTime::Name_Full )
439 {
440 if ( name.CmpNoCase(wxDateTime::
441 GetMonthName(mon, wxDateTime::Name_Full)) == 0 )
442 {
443 break;
444 }
445 }
446
447 if ( flags & wxDateTime::Name_Abbr )
448 {
449 if ( name.CmpNoCase(wxDateTime::
450 GetMonthName(mon, wxDateTime::Name_Abbr)) == 0 )
451 {
452 break;
453 }
454 }
455 }
456
457 return mon;
458 }
459
460 // return the weekday if the string is a weekday name or Inv_WeekDay otherwise
461 static wxDateTime::WeekDay GetWeekDayFromName(const wxString& name, int flags)
462 {
463 wxDateTime::WeekDay wd;
464 for ( wd = wxDateTime::Sun; wd < wxDateTime::Inv_WeekDay; wxNextWDay(wd) )
465 {
466 // case-insensitive comparison either one of or with both abbreviated
467 // and not versions
468 if ( flags & wxDateTime::Name_Full )
469 {
470 if ( name.CmpNoCase(wxDateTime::
471 GetWeekDayName(wd, wxDateTime::Name_Full)) == 0 )
472 {
473 break;
474 }
475 }
476
477 if ( flags & wxDateTime::Name_Abbr )
478 {
479 if ( name.CmpNoCase(wxDateTime::
480 GetWeekDayName(wd, wxDateTime::Name_Abbr)) == 0 )
481 {
482 break;
483 }
484 }
485 }
486
487 return wd;
488 }
489
490 // scans all digits (but no more than len) and returns the resulting number
491 static bool GetNumericToken(size_t len, const wxChar*& p, unsigned long *number)
492 {
493 size_t n = 1;
494 wxString s;
495 while ( wxIsdigit(*p) )
496 {
497 s += *p++;
498
499 if ( len && ++n > len )
500 break;
501 }
502
503 return !s.IsEmpty() && s.ToULong(number);
504 }
505
506 // scans all alphabetic characters and returns the resulting string
507 static wxString GetAlphaToken(const wxChar*& p)
508 {
509 wxString s;
510 while ( wxIsalpha(*p) )
511 {
512 s += *p++;
513 }
514
515 return s;
516 }
517
518 // ============================================================================
519 // implementation of wxDateTime
520 // ============================================================================
521
522 // ----------------------------------------------------------------------------
523 // struct Tm
524 // ----------------------------------------------------------------------------
525
526 wxDateTime::Tm::Tm()
527 {
528 year = (wxDateTime_t)wxDateTime::Inv_Year;
529 mon = wxDateTime::Inv_Month;
530 mday = 0;
531 hour = min = sec = msec = 0;
532 wday = wxDateTime::Inv_WeekDay;
533 }
534
535 wxDateTime::Tm::Tm(const struct tm& tm, const TimeZone& tz)
536 : m_tz(tz)
537 {
538 msec = 0;
539 sec = tm.tm_sec;
540 min = tm.tm_min;
541 hour = tm.tm_hour;
542 mday = tm.tm_mday;
543 mon = (wxDateTime::Month)tm.tm_mon;
544 year = 1900 + tm.tm_year;
545 wday = tm.tm_wday;
546 yday = tm.tm_yday;
547 }
548
549 bool wxDateTime::Tm::IsValid() const
550 {
551 // we allow for the leap seconds, although we don't use them (yet)
552 return (year != wxDateTime::Inv_Year) && (mon != wxDateTime::Inv_Month) &&
553 (mday <= GetNumOfDaysInMonth(year, mon)) &&
554 (hour < 24) && (min < 60) && (sec < 62) && (msec < 1000);
555 }
556
557 void wxDateTime::Tm::ComputeWeekDay()
558 {
559 // compute the week day from day/month/year: we use the dumbest algorithm
560 // possible: just compute our JDN and then use the (simple to derive)
561 // formula: weekday = (JDN + 1.5) % 7
562 wday = (wxDateTime::WeekDay)(GetTruncatedJDN(mday, mon, year) + 2) % 7;
563 }
564
565 void wxDateTime::Tm::AddMonths(int monDiff)
566 {
567 // normalize the months field
568 while ( monDiff < -mon )
569 {
570 year--;
571
572 monDiff += MONTHS_IN_YEAR;
573 }
574
575 while ( monDiff + mon >= MONTHS_IN_YEAR )
576 {
577 year++;
578
579 monDiff -= MONTHS_IN_YEAR;
580 }
581
582 mon = (wxDateTime::Month)(mon + monDiff);
583
584 wxASSERT_MSG( mon >= 0 && mon < MONTHS_IN_YEAR, _T("logic error") );
585
586 // NB: we don't check here that the resulting date is valid, this function
587 // is private and the caller must check it if needed
588 }
589
590 void wxDateTime::Tm::AddDays(int dayDiff)
591 {
592 // normalize the days field
593 while ( dayDiff + mday < 1 )
594 {
595 AddMonths(-1);
596
597 dayDiff += GetNumOfDaysInMonth(year, mon);
598 }
599
600 mday += dayDiff;
601 while ( mday > GetNumOfDaysInMonth(year, mon) )
602 {
603 mday -= GetNumOfDaysInMonth(year, mon);
604
605 AddMonths(1);
606 }
607
608 wxASSERT_MSG( mday > 0 && mday <= GetNumOfDaysInMonth(year, mon),
609 _T("logic error") );
610 }
611
612 // ----------------------------------------------------------------------------
613 // class TimeZone
614 // ----------------------------------------------------------------------------
615
616 wxDateTime::TimeZone::TimeZone(wxDateTime::TZ tz)
617 {
618 switch ( tz )
619 {
620 case wxDateTime::Local:
621 // get the offset from C RTL: it returns the difference GMT-local
622 // while we want to have the offset _from_ GMT, hence the '-'
623 m_offset = -GetTimeZone();
624 break;
625
626 case wxDateTime::GMT_12:
627 case wxDateTime::GMT_11:
628 case wxDateTime::GMT_10:
629 case wxDateTime::GMT_9:
630 case wxDateTime::GMT_8:
631 case wxDateTime::GMT_7:
632 case wxDateTime::GMT_6:
633 case wxDateTime::GMT_5:
634 case wxDateTime::GMT_4:
635 case wxDateTime::GMT_3:
636 case wxDateTime::GMT_2:
637 case wxDateTime::GMT_1:
638 m_offset = -3600*(wxDateTime::GMT0 - tz);
639 break;
640
641 case wxDateTime::GMT0:
642 case wxDateTime::GMT1:
643 case wxDateTime::GMT2:
644 case wxDateTime::GMT3:
645 case wxDateTime::GMT4:
646 case wxDateTime::GMT5:
647 case wxDateTime::GMT6:
648 case wxDateTime::GMT7:
649 case wxDateTime::GMT8:
650 case wxDateTime::GMT9:
651 case wxDateTime::GMT10:
652 case wxDateTime::GMT11:
653 case wxDateTime::GMT12:
654 m_offset = 3600*(tz - wxDateTime::GMT0);
655 break;
656
657 case wxDateTime::A_CST:
658 // Central Standard Time in use in Australia = UTC + 9.5
659 m_offset = 60l*(9*60 + 30);
660 break;
661
662 default:
663 wxFAIL_MSG( _T("unknown time zone") );
664 }
665 }
666
667 // ----------------------------------------------------------------------------
668 // static functions
669 // ----------------------------------------------------------------------------
670
671 /* static */
672 bool wxDateTime::IsLeapYear(int year, wxDateTime::Calendar cal)
673 {
674 if ( year == Inv_Year )
675 year = GetCurrentYear();
676
677 if ( cal == Gregorian )
678 {
679 // in Gregorian calendar leap years are those divisible by 4 except
680 // those divisible by 100 unless they're also divisible by 400
681 // (in some countries, like Russia and Greece, additional corrections
682 // exist, but they won't manifest themselves until 2700)
683 return (year % 4 == 0) && ((year % 100 != 0) || (year % 400 == 0));
684 }
685 else if ( cal == Julian )
686 {
687 // in Julian calendar the rule is simpler
688 return year % 4 == 0;
689 }
690 else
691 {
692 wxFAIL_MSG(_T("unknown calendar"));
693
694 return false;
695 }
696 }
697
698 /* static */
699 int wxDateTime::GetCentury(int year)
700 {
701 return year > 0 ? year / 100 : year / 100 - 1;
702 }
703
704 /* static */
705 int wxDateTime::ConvertYearToBC(int year)
706 {
707 // year 0 is BC 1
708 return year > 0 ? year : year - 1;
709 }
710
711 /* static */
712 int wxDateTime::GetCurrentYear(wxDateTime::Calendar cal)
713 {
714 switch ( cal )
715 {
716 case Gregorian:
717 return Now().GetYear();
718
719 case Julian:
720 wxFAIL_MSG(_T("TODO"));
721 break;
722
723 default:
724 wxFAIL_MSG(_T("unsupported calendar"));
725 break;
726 }
727
728 return Inv_Year;
729 }
730
731 /* static */
732 wxDateTime::Month wxDateTime::GetCurrentMonth(wxDateTime::Calendar cal)
733 {
734 switch ( cal )
735 {
736 case Gregorian:
737 return Now().GetMonth();
738
739 case Julian:
740 wxFAIL_MSG(_T("TODO"));
741 break;
742
743 default:
744 wxFAIL_MSG(_T("unsupported calendar"));
745 break;
746 }
747
748 return Inv_Month;
749 }
750
751 /* static */
752 wxDateTime::wxDateTime_t wxDateTime::GetNumberOfDays(int year, Calendar cal)
753 {
754 if ( year == Inv_Year )
755 {
756 // take the current year if none given
757 year = GetCurrentYear();
758 }
759
760 switch ( cal )
761 {
762 case Gregorian:
763 case Julian:
764 return IsLeapYear(year) ? 366 : 365;
765
766 default:
767 wxFAIL_MSG(_T("unsupported calendar"));
768 break;
769 }
770
771 return 0;
772 }
773
774 /* static */
775 wxDateTime::wxDateTime_t wxDateTime::GetNumberOfDays(wxDateTime::Month month,
776 int year,
777 wxDateTime::Calendar cal)
778 {
779 wxCHECK_MSG( month < MONTHS_IN_YEAR, 0, _T("invalid month") );
780
781 if ( cal == Gregorian || cal == Julian )
782 {
783 if ( year == Inv_Year )
784 {
785 // take the current year if none given
786 year = GetCurrentYear();
787 }
788
789 return GetNumOfDaysInMonth(year, month);
790 }
791 else
792 {
793 wxFAIL_MSG(_T("unsupported calendar"));
794
795 return 0;
796 }
797 }
798
799 /* static */
800 wxString wxDateTime::GetMonthName(wxDateTime::Month month,
801 wxDateTime::NameFlags flags)
802 {
803 wxCHECK_MSG( month != Inv_Month, _T(""), _T("invalid month") );
804
805 // notice that we must set all the fields to avoid confusing libc (GNU one
806 // gets confused to a crash if we don't do this)
807 tm tm;
808 InitTm(tm);
809 tm.tm_mon = month;
810
811 return CallStrftime(flags == Name_Abbr ? _T("%b") : _T("%B"), &tm);
812 }
813
814 /* static */
815 wxString wxDateTime::GetWeekDayName(wxDateTime::WeekDay wday,
816 wxDateTime::NameFlags flags)
817 {
818 wxCHECK_MSG( wday != Inv_WeekDay, _T(""), _T("invalid weekday") );
819
820 // take some arbitrary Sunday (but notice that the day should be such that
821 // after adding wday to it below we still have a valid date, e.g. don't
822 // take 28 here!)
823 tm tm;
824 InitTm(tm);
825 tm.tm_mday = 21;
826 tm.tm_mon = Nov;
827 tm.tm_year = 99;
828
829 // and offset it by the number of days needed to get the correct wday
830 tm.tm_mday += wday;
831
832 // call mktime() to normalize it...
833 (void)mktime(&tm);
834
835 // ... and call strftime()
836 return CallStrftime(flags == Name_Abbr ? _T("%a") : _T("%A"), &tm);
837 }
838
839 /* static */
840 void wxDateTime::GetAmPmStrings(wxString *am, wxString *pm)
841 {
842 tm tm;
843 InitTm(tm);
844 wxChar buffer[64];
845 // @Note: Do not call 'CallStrftime' here! CallStrftime checks the return code
846 // and causes an assertion failed if the buffer is to small (which is good) - OR -
847 // if strftime does not return anything because the format string is invalid - OR -
848 // if there are no 'am' / 'pm' tokens defined for the current locale (which is not good).
849 // wxDateTime::ParseTime will try several different formats to parse the time.
850 // As a result, GetAmPmStrings might get called, even if the current locale
851 // does not define any 'am' / 'pm' tokens. In this case, wxStrftime would
852 // assert, even though it is a perfectly legal use.
853 if ( am )
854 {
855 if (wxStrftime(buffer, sizeof buffer, _T("%p"), &tm) > 0)
856 *am = wxString(buffer);
857 else
858 *am = wxString();
859 }
860 if ( pm )
861 {
862 tm.tm_hour = 13;
863 if (wxStrftime(buffer, sizeof buffer, _T("%p"), &tm) > 0)
864 *pm = wxString(buffer);
865 else
866 *pm = wxString();
867 }
868 }
869
870 // ----------------------------------------------------------------------------
871 // Country stuff: date calculations depend on the country (DST, work days,
872 // ...), so we need to know which rules to follow.
873 // ----------------------------------------------------------------------------
874
875 /* static */
876 wxDateTime::Country wxDateTime::GetCountry()
877 {
878 // TODO use LOCALE_ICOUNTRY setting under Win32
879
880 if ( ms_country == Country_Unknown )
881 {
882 // try to guess from the time zone name
883 time_t t = time(NULL);
884 struct tm *tm = localtime(&t);
885
886 wxString tz = CallStrftime(_T("%Z"), tm);
887 if ( tz == _T("WET") || tz == _T("WEST") )
888 {
889 ms_country = UK;
890 }
891 else if ( tz == _T("CET") || tz == _T("CEST") )
892 {
893 ms_country = Country_EEC;
894 }
895 else if ( tz == _T("MSK") || tz == _T("MSD") )
896 {
897 ms_country = Russia;
898 }
899 else if ( tz == _T("AST") || tz == _T("ADT") ||
900 tz == _T("EST") || tz == _T("EDT") ||
901 tz == _T("CST") || tz == _T("CDT") ||
902 tz == _T("MST") || tz == _T("MDT") ||
903 tz == _T("PST") || tz == _T("PDT") )
904 {
905 ms_country = USA;
906 }
907 else
908 {
909 // well, choose a default one
910 ms_country = USA;
911 }
912 }
913
914 return ms_country;
915 }
916
917 /* static */
918 void wxDateTime::SetCountry(wxDateTime::Country country)
919 {
920 ms_country = country;
921 }
922
923 /* static */
924 bool wxDateTime::IsWestEuropeanCountry(Country country)
925 {
926 if ( country == Country_Default )
927 {
928 country = GetCountry();
929 }
930
931 return (Country_WesternEurope_Start <= country) &&
932 (country <= Country_WesternEurope_End);
933 }
934
935 // ----------------------------------------------------------------------------
936 // DST calculations: we use 3 different rules for the West European countries,
937 // USA and for the rest of the world. This is undoubtedly false for many
938 // countries, but I lack the necessary info (and the time to gather it),
939 // please add the other rules here!
940 // ----------------------------------------------------------------------------
941
942 /* static */
943 bool wxDateTime::IsDSTApplicable(int year, Country country)
944 {
945 if ( year == Inv_Year )
946 {
947 // take the current year if none given
948 year = GetCurrentYear();
949 }
950
951 if ( country == Country_Default )
952 {
953 country = GetCountry();
954 }
955
956 switch ( country )
957 {
958 case USA:
959 case UK:
960 // DST was first observed in the US and UK during WWI, reused
961 // during WWII and used again since 1966
962 return year >= 1966 ||
963 (year >= 1942 && year <= 1945) ||
964 (year == 1918 || year == 1919);
965
966 default:
967 // assume that it started after WWII
968 return year > 1950;
969 }
970 }
971
972 /* static */
973 wxDateTime wxDateTime::GetBeginDST(int year, Country country)
974 {
975 if ( year == Inv_Year )
976 {
977 // take the current year if none given
978 year = GetCurrentYear();
979 }
980
981 if ( country == Country_Default )
982 {
983 country = GetCountry();
984 }
985
986 if ( !IsDSTApplicable(year, country) )
987 {
988 return wxInvalidDateTime;
989 }
990
991 wxDateTime dt;
992
993 if ( IsWestEuropeanCountry(country) || (country == Russia) )
994 {
995 // DST begins at 1 a.m. GMT on the last Sunday of March
996 if ( !dt.SetToLastWeekDay(Sun, Mar, year) )
997 {
998 // weird...
999 wxFAIL_MSG( _T("no last Sunday in March?") );
1000 }
1001
1002 dt += wxTimeSpan::Hours(1);
1003
1004 // disable DST tests because it could result in an infinite recursion!
1005 dt.MakeGMT(true);
1006 }
1007 else switch ( country )
1008 {
1009 case USA:
1010 switch ( year )
1011 {
1012 case 1918:
1013 case 1919:
1014 // don't know for sure - assume it was in effect all year
1015
1016 case 1943:
1017 case 1944:
1018 case 1945:
1019 dt.Set(1, Jan, year);
1020 break;
1021
1022 case 1942:
1023 // DST was installed Feb 2, 1942 by the Congress
1024 dt.Set(2, Feb, year);
1025 break;
1026
1027 // Oil embargo changed the DST period in the US
1028 case 1974:
1029 dt.Set(6, Jan, 1974);
1030 break;
1031
1032 case 1975:
1033 dt.Set(23, Feb, 1975);
1034 break;
1035
1036 default:
1037 // before 1986, DST begun on the last Sunday of April, but
1038 // in 1986 Reagan changed it to begin at 2 a.m. of the
1039 // first Sunday in April
1040 if ( year < 1986 )
1041 {
1042 if ( !dt.SetToLastWeekDay(Sun, Apr, year) )
1043 {
1044 // weird...
1045 wxFAIL_MSG( _T("no first Sunday in April?") );
1046 }
1047 }
1048 else
1049 {
1050 if ( !dt.SetToWeekDay(Sun, 1, Apr, year) )
1051 {
1052 // weird...
1053 wxFAIL_MSG( _T("no first Sunday in April?") );
1054 }
1055 }
1056
1057 dt += wxTimeSpan::Hours(2);
1058
1059 // TODO what about timezone??
1060 }
1061
1062 break;
1063
1064 default:
1065 // assume Mar 30 as the start of the DST for the rest of the world
1066 // - totally bogus, of course
1067 dt.Set(30, Mar, year);
1068 }
1069
1070 return dt;
1071 }
1072
1073 /* static */
1074 wxDateTime wxDateTime::GetEndDST(int year, Country country)
1075 {
1076 if ( year == Inv_Year )
1077 {
1078 // take the current year if none given
1079 year = GetCurrentYear();
1080 }
1081
1082 if ( country == Country_Default )
1083 {
1084 country = GetCountry();
1085 }
1086
1087 if ( !IsDSTApplicable(year, country) )
1088 {
1089 return wxInvalidDateTime;
1090 }
1091
1092 wxDateTime dt;
1093
1094 if ( IsWestEuropeanCountry(country) || (country == Russia) )
1095 {
1096 // DST ends at 1 a.m. GMT on the last Sunday of October
1097 if ( !dt.SetToLastWeekDay(Sun, Oct, year) )
1098 {
1099 // weirder and weirder...
1100 wxFAIL_MSG( _T("no last Sunday in October?") );
1101 }
1102
1103 dt += wxTimeSpan::Hours(1);
1104
1105 // disable DST tests because it could result in an infinite recursion!
1106 dt.MakeGMT(true);
1107 }
1108 else switch ( country )
1109 {
1110 case USA:
1111 switch ( year )
1112 {
1113 case 1918:
1114 case 1919:
1115 // don't know for sure - assume it was in effect all year
1116
1117 case 1943:
1118 case 1944:
1119 dt.Set(31, Dec, year);
1120 break;
1121
1122 case 1945:
1123 // the time was reset after the end of the WWII
1124 dt.Set(30, Sep, year);
1125 break;
1126
1127 default:
1128 // DST ends at 2 a.m. on the last Sunday of October
1129 if ( !dt.SetToLastWeekDay(Sun, Oct, year) )
1130 {
1131 // weirder and weirder...
1132 wxFAIL_MSG( _T("no last Sunday in October?") );
1133 }
1134
1135 dt += wxTimeSpan::Hours(2);
1136
1137 // TODO what about timezone??
1138 }
1139 break;
1140
1141 default:
1142 // assume October 26th as the end of the DST - totally bogus too
1143 dt.Set(26, Oct, year);
1144 }
1145
1146 return dt;
1147 }
1148
1149 // ----------------------------------------------------------------------------
1150 // constructors and assignment operators
1151 // ----------------------------------------------------------------------------
1152
1153 // return the current time with ms precision
1154 /* static */ wxDateTime wxDateTime::UNow()
1155 {
1156 return wxDateTime(wxGetLocalTimeMillis());
1157 }
1158
1159 // the values in the tm structure contain the local time
1160 wxDateTime& wxDateTime::Set(const struct tm& tm)
1161 {
1162 struct tm tm2(tm);
1163 time_t timet = mktime(&tm2);
1164
1165 if ( timet == (time_t)-1 )
1166 {
1167 // mktime() rather unintuitively fails for Jan 1, 1970 if the hour is
1168 // less than timezone - try to make it work for this case
1169 if ( tm2.tm_year == 70 && tm2.tm_mon == 0 && tm2.tm_mday == 1 )
1170 {
1171 // add timezone to make sure that date is in range
1172 tm2.tm_sec -= GetTimeZone();
1173
1174 timet = mktime(&tm2);
1175 if ( timet != (time_t)-1 )
1176 {
1177 timet += GetTimeZone();
1178
1179 return Set(timet);
1180 }
1181 }
1182
1183 wxFAIL_MSG( _T("mktime() failed") );
1184
1185 *this = wxInvalidDateTime;
1186
1187 return *this;
1188 }
1189 else
1190 {
1191 return Set(timet);
1192 }
1193 }
1194
1195 wxDateTime& wxDateTime::Set(wxDateTime_t hour,
1196 wxDateTime_t minute,
1197 wxDateTime_t second,
1198 wxDateTime_t millisec)
1199 {
1200 // we allow seconds to be 61 to account for the leap seconds, even if we
1201 // don't use them really
1202 wxDATETIME_CHECK( hour < 24 &&
1203 second < 62 &&
1204 minute < 60 &&
1205 millisec < 1000,
1206 _T("Invalid time in wxDateTime::Set()") );
1207
1208 // get the current date from system
1209 struct tm *tm = GetTmNow();
1210
1211 wxDATETIME_CHECK( tm, _T("localtime() failed") );
1212
1213 // adjust the time
1214 tm->tm_hour = hour;
1215 tm->tm_min = minute;
1216 tm->tm_sec = second;
1217
1218 (void)Set(*tm);
1219
1220 // and finally adjust milliseconds
1221 return SetMillisecond(millisec);
1222 }
1223
1224 wxDateTime& wxDateTime::Set(wxDateTime_t day,
1225 Month month,
1226 int year,
1227 wxDateTime_t hour,
1228 wxDateTime_t minute,
1229 wxDateTime_t second,
1230 wxDateTime_t millisec)
1231 {
1232 wxDATETIME_CHECK( hour < 24 &&
1233 second < 62 &&
1234 minute < 60 &&
1235 millisec < 1000,
1236 _T("Invalid time in wxDateTime::Set()") );
1237
1238 ReplaceDefaultYearMonthWithCurrent(&year, &month);
1239
1240 wxDATETIME_CHECK( (0 < day) && (day <= GetNumberOfDays(month, year)),
1241 _T("Invalid date in wxDateTime::Set()") );
1242
1243 // the range of time_t type (inclusive)
1244 static const int yearMinInRange = 1970;
1245 static const int yearMaxInRange = 2037;
1246
1247 // test only the year instead of testing for the exact end of the Unix
1248 // time_t range - it doesn't bring anything to do more precise checks
1249 if ( year >= yearMinInRange && year <= yearMaxInRange )
1250 {
1251 // use the standard library version if the date is in range - this is
1252 // probably more efficient than our code
1253 struct tm tm;
1254 tm.tm_year = year - 1900;
1255 tm.tm_mon = month;
1256 tm.tm_mday = day;
1257 tm.tm_hour = hour;
1258 tm.tm_min = minute;
1259 tm.tm_sec = second;
1260 tm.tm_isdst = -1; // mktime() will guess it
1261
1262 (void)Set(tm);
1263
1264 // and finally adjust milliseconds
1265 return SetMillisecond(millisec);
1266 }
1267 else
1268 {
1269 // do time calculations ourselves: we want to calculate the number of
1270 // milliseconds between the given date and the epoch
1271
1272 // get the JDN for the midnight of this day
1273 m_time = GetTruncatedJDN(day, month, year);
1274 m_time -= EPOCH_JDN;
1275 m_time *= SECONDS_PER_DAY * TIME_T_FACTOR;
1276
1277 // JDN corresponds to GMT, we take localtime
1278 Add(wxTimeSpan(hour, minute, second + GetTimeZone(), millisec));
1279 }
1280
1281 return *this;
1282 }
1283
1284 wxDateTime& wxDateTime::Set(double jdn)
1285 {
1286 // so that m_time will be 0 for the midnight of Jan 1, 1970 which is jdn
1287 // EPOCH_JDN + 0.5
1288 jdn -= EPOCH_JDN + 0.5;
1289
1290 jdn *= MILLISECONDS_PER_DAY;
1291
1292 // JDNs always suppose an UTC date, so bring it back to local time zone
1293 // (also see GetJulianDayNumber() implementation)
1294 long tzDiff = GetTimeZone();
1295 if ( IsDST() == 1 )
1296 {
1297 // FIXME: again, we suppose that DST is always one hour
1298 tzDiff -= 3600;
1299 }
1300
1301 jdn += tzDiff*1000; // tzDiff is in seconds
1302
1303 m_time.Assign(jdn);
1304
1305 return *this;
1306 }
1307
1308 wxDateTime& wxDateTime::ResetTime()
1309 {
1310 Tm tm = GetTm();
1311
1312 if ( tm.hour || tm.min || tm.sec || tm.msec )
1313 {
1314 tm.msec =
1315 tm.sec =
1316 tm.min =
1317 tm.hour = 0;
1318
1319 Set(tm);
1320 }
1321
1322 return *this;
1323 }
1324
1325 // ----------------------------------------------------------------------------
1326 // DOS Date and Time Format functions
1327 // ----------------------------------------------------------------------------
1328 // the dos date and time value is an unsigned 32 bit value in the format:
1329 // YYYYYYYMMMMDDDDDhhhhhmmmmmmsssss
1330 //
1331 // Y = year offset from 1980 (0-127)
1332 // M = month (1-12)
1333 // D = day of month (1-31)
1334 // h = hour (0-23)
1335 // m = minute (0-59)
1336 // s = bisecond (0-29) each bisecond indicates two seconds
1337 // ----------------------------------------------------------------------------
1338
1339 wxDateTime& wxDateTime::SetFromDOS(unsigned long ddt)
1340 {
1341 struct tm tm;
1342 InitTm(tm);
1343
1344 long year = ddt & 0xFE000000;
1345 year >>= 25;
1346 year += 80;
1347 tm.tm_year = year;
1348
1349 long month = ddt & 0x1E00000;
1350 month >>= 21;
1351 month -= 1;
1352 tm.tm_mon = month;
1353
1354 long day = ddt & 0x1F0000;
1355 day >>= 16;
1356 tm.tm_mday = day;
1357
1358 long hour = ddt & 0xF800;
1359 hour >>= 11;
1360 tm.tm_hour = hour;
1361
1362 long minute = ddt & 0x7E0;
1363 minute >>= 5;
1364 tm.tm_min = minute;
1365
1366 long second = ddt & 0x1F;
1367 tm.tm_sec = second * 2;
1368
1369 return Set(mktime(&tm));
1370 }
1371
1372 unsigned long wxDateTime::GetAsDOS() const
1373 {
1374 unsigned long ddt;
1375 time_t ticks = GetTicks();
1376 struct tm *tm = localtime(&ticks);
1377
1378 long year = tm->tm_year;
1379 year -= 80;
1380 year <<= 25;
1381
1382 long month = tm->tm_mon;
1383 month += 1;
1384 month <<= 21;
1385
1386 long day = tm->tm_mday;
1387 day <<= 16;
1388
1389 long hour = tm->tm_hour;
1390 hour <<= 11;
1391
1392 long minute = tm->tm_min;
1393 minute <<= 5;
1394
1395 long second = tm->tm_sec;
1396 second /= 2;
1397
1398 ddt = year | month | day | hour | minute | second;
1399 return ddt;
1400 }
1401
1402 // ----------------------------------------------------------------------------
1403 // time_t <-> broken down time conversions
1404 // ----------------------------------------------------------------------------
1405
1406 wxDateTime::Tm wxDateTime::GetTm(const TimeZone& tz) const
1407 {
1408 wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") );
1409
1410 time_t time = GetTicks();
1411 if ( time != (time_t)-1 )
1412 {
1413 // use C RTL functions
1414 tm *tm;
1415 if ( tz.GetOffset() == -GetTimeZone() )
1416 {
1417 // we are working with local time
1418 tm = localtime(&time);
1419
1420 // should never happen
1421 wxCHECK_MSG( tm, Tm(), _T("localtime() failed") );
1422 }
1423 else
1424 {
1425 time += (time_t)tz.GetOffset();
1426 #if defined(__VMS__) || defined(__WATCOMC__) // time is unsigned so avoid warning
1427 int time2 = (int) time;
1428 if ( time2 >= 0 )
1429 #else
1430 if ( time >= 0 )
1431 #endif
1432 {
1433 tm = gmtime(&time);
1434
1435 // should never happen
1436 wxCHECK_MSG( tm, Tm(), _T("gmtime() failed") );
1437 }
1438 else
1439 {
1440 tm = (struct tm *)NULL;
1441 }
1442 }
1443
1444 if ( tm )
1445 {
1446 // adjust the milliseconds
1447 Tm tm2(*tm, tz);
1448 long timeOnly = (m_time % MILLISECONDS_PER_DAY).ToLong();
1449 tm2.msec = (wxDateTime_t)(timeOnly % 1000);
1450 return tm2;
1451 }
1452 //else: use generic code below
1453 }
1454
1455 // remember the time and do the calculations with the date only - this
1456 // eliminates rounding errors of the floating point arithmetics
1457
1458 wxLongLong timeMidnight = m_time + tz.GetOffset() * 1000;
1459
1460 long timeOnly = (timeMidnight % MILLISECONDS_PER_DAY).ToLong();
1461
1462 // we want to always have positive time and timeMidnight to be really
1463 // the midnight before it
1464 if ( timeOnly < 0 )
1465 {
1466 timeOnly = MILLISECONDS_PER_DAY + timeOnly;
1467 }
1468
1469 timeMidnight -= timeOnly;
1470
1471 // calculate the Gregorian date from JDN for the midnight of our date:
1472 // this will yield day, month (in 1..12 range) and year
1473
1474 // actually, this is the JDN for the noon of the previous day
1475 long jdn = (timeMidnight / MILLISECONDS_PER_DAY).ToLong() + EPOCH_JDN;
1476
1477 // CREDIT: code below is by Scott E. Lee (but bugs are mine)
1478
1479 wxASSERT_MSG( jdn > -2, _T("JDN out of range") );
1480
1481 // calculate the century
1482 long temp = (jdn + JDN_OFFSET) * 4 - 1;
1483 long century = temp / DAYS_PER_400_YEARS;
1484
1485 // then the year and day of year (1 <= dayOfYear <= 366)
1486 temp = ((temp % DAYS_PER_400_YEARS) / 4) * 4 + 3;
1487 long year = (century * 100) + (temp / DAYS_PER_4_YEARS);
1488 long dayOfYear = (temp % DAYS_PER_4_YEARS) / 4 + 1;
1489
1490 // and finally the month and day of the month
1491 temp = dayOfYear * 5 - 3;
1492 long month = temp / DAYS_PER_5_MONTHS;
1493 long day = (temp % DAYS_PER_5_MONTHS) / 5 + 1;
1494
1495 // month is counted from March - convert to normal
1496 if ( month < 10 )
1497 {
1498 month += 3;
1499 }
1500 else
1501 {
1502 year += 1;
1503 month -= 9;
1504 }
1505
1506 // year is offset by 4800
1507 year -= 4800;
1508
1509 // check that the algorithm gave us something reasonable
1510 wxASSERT_MSG( (0 < month) && (month <= 12), _T("invalid month") );
1511 wxASSERT_MSG( (1 <= day) && (day < 32), _T("invalid day") );
1512
1513 // construct Tm from these values
1514 Tm tm;
1515 tm.year = (int)year;
1516 tm.mon = (Month)(month - 1); // algorithm yields 1 for January, not 0
1517 tm.mday = (wxDateTime_t)day;
1518 tm.msec = (wxDateTime_t)(timeOnly % 1000);
1519 timeOnly -= tm.msec;
1520 timeOnly /= 1000; // now we have time in seconds
1521
1522 tm.sec = (wxDateTime_t)(timeOnly % 60);
1523 timeOnly -= tm.sec;
1524 timeOnly /= 60; // now we have time in minutes
1525
1526 tm.min = (wxDateTime_t)(timeOnly % 60);
1527 timeOnly -= tm.min;
1528
1529 tm.hour = (wxDateTime_t)(timeOnly / 60);
1530
1531 return tm;
1532 }
1533
1534 wxDateTime& wxDateTime::SetYear(int year)
1535 {
1536 wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") );
1537
1538 Tm tm(GetTm());
1539 tm.year = year;
1540 Set(tm);
1541
1542 return *this;
1543 }
1544
1545 wxDateTime& wxDateTime::SetMonth(Month month)
1546 {
1547 wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") );
1548
1549 Tm tm(GetTm());
1550 tm.mon = month;
1551 Set(tm);
1552
1553 return *this;
1554 }
1555
1556 wxDateTime& wxDateTime::SetDay(wxDateTime_t mday)
1557 {
1558 wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") );
1559
1560 Tm tm(GetTm());
1561 tm.mday = mday;
1562 Set(tm);
1563
1564 return *this;
1565 }
1566
1567 wxDateTime& wxDateTime::SetHour(wxDateTime_t hour)
1568 {
1569 wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") );
1570
1571 Tm tm(GetTm());
1572 tm.hour = hour;
1573 Set(tm);
1574
1575 return *this;
1576 }
1577
1578 wxDateTime& wxDateTime::SetMinute(wxDateTime_t min)
1579 {
1580 wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") );
1581
1582 Tm tm(GetTm());
1583 tm.min = min;
1584 Set(tm);
1585
1586 return *this;
1587 }
1588
1589 wxDateTime& wxDateTime::SetSecond(wxDateTime_t sec)
1590 {
1591 wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") );
1592
1593 Tm tm(GetTm());
1594 tm.sec = sec;
1595 Set(tm);
1596
1597 return *this;
1598 }
1599
1600 wxDateTime& wxDateTime::SetMillisecond(wxDateTime_t millisecond)
1601 {
1602 wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") );
1603
1604 // we don't need to use GetTm() for this one
1605 m_time -= m_time % 1000l;
1606 m_time += millisecond;
1607
1608 return *this;
1609 }
1610
1611 // ----------------------------------------------------------------------------
1612 // wxDateTime arithmetics
1613 // ----------------------------------------------------------------------------
1614
1615 wxDateTime& wxDateTime::Add(const wxDateSpan& diff)
1616 {
1617 Tm tm(GetTm());
1618
1619 tm.year += diff.GetYears();
1620 tm.AddMonths(diff.GetMonths());
1621
1622 // check that the resulting date is valid
1623 if ( tm.mday > GetNumOfDaysInMonth(tm.year, tm.mon) )
1624 {
1625 // We suppose that when adding one month to Jan 31 we want to get Feb
1626 // 28 (or 29), i.e. adding a month to the last day of the month should
1627 // give the last day of the next month which is quite logical.
1628 //
1629 // Unfortunately, there is no logic way to understand what should
1630 // Jan 30 + 1 month be - Feb 28 too or Feb 27 (assuming non leap year)?
1631 // We make it Feb 28 (last day too), but it is highly questionable.
1632 tm.mday = GetNumOfDaysInMonth(tm.year, tm.mon);
1633 }
1634
1635 tm.AddDays(diff.GetTotalDays());
1636
1637 Set(tm);
1638
1639 wxASSERT_MSG( IsSameTime(tm),
1640 _T("Add(wxDateSpan) shouldn't modify time") );
1641
1642 return *this;
1643 }
1644
1645 // ----------------------------------------------------------------------------
1646 // Weekday and monthday stuff
1647 // ----------------------------------------------------------------------------
1648
1649 bool wxDateTime::SetToTheWeek(wxDateTime_t numWeek,
1650 WeekDay weekday,
1651 WeekFlags flags)
1652 {
1653 wxASSERT_MSG( numWeek > 0,
1654 _T("invalid week number: weeks are counted from 1") );
1655
1656 int year = GetYear();
1657
1658 // Jan 4 always lies in the 1st week of the year
1659 Set(4, Jan, year);
1660 SetToWeekDayInSameWeek(weekday, flags) += wxDateSpan::Weeks(numWeek - 1);
1661
1662 if ( GetYear() != year )
1663 {
1664 // oops... numWeek was too big
1665 return false;
1666 }
1667
1668 return true;
1669 }
1670
1671 wxDateTime& wxDateTime::SetToLastMonthDay(Month month,
1672 int year)
1673 {
1674 // take the current month/year if none specified
1675 if ( year == Inv_Year )
1676 year = GetYear();
1677 if ( month == Inv_Month )
1678 month = GetMonth();
1679
1680 return Set(GetNumOfDaysInMonth(year, month), month, year);
1681 }
1682
1683 wxDateTime& wxDateTime::SetToWeekDayInSameWeek(WeekDay weekday, WeekFlags flags)
1684 {
1685 wxDATETIME_CHECK( weekday != Inv_WeekDay, _T("invalid weekday") );
1686
1687 int wdayThis = GetWeekDay();
1688 if ( weekday == wdayThis )
1689 {
1690 // nothing to do
1691 return *this;
1692 }
1693
1694 if ( flags == Default_First )
1695 {
1696 flags = GetCountry() == USA ? Sunday_First : Monday_First;
1697 }
1698
1699 // the logic below based on comparing weekday and wdayThis works if Sun (0)
1700 // is the first day in the week, but breaks down for Monday_First case so
1701 // we adjust the week days in this case
1702 if( flags == Monday_First )
1703 {
1704 if ( wdayThis == Sun )
1705 wdayThis += 7;
1706 }
1707 //else: Sunday_First, nothing to do
1708
1709 // go forward or back in time to the day we want
1710 if ( weekday < wdayThis )
1711 {
1712 return Subtract(wxDateSpan::Days(wdayThis - weekday));
1713 }
1714 else // weekday > wdayThis
1715 {
1716 return Add(wxDateSpan::Days(weekday - wdayThis));
1717 }
1718 }
1719
1720 wxDateTime& wxDateTime::SetToNextWeekDay(WeekDay weekday)
1721 {
1722 wxDATETIME_CHECK( weekday != Inv_WeekDay, _T("invalid weekday") );
1723
1724 int diff;
1725 WeekDay wdayThis = GetWeekDay();
1726 if ( weekday == wdayThis )
1727 {
1728 // nothing to do
1729 return *this;
1730 }
1731 else if ( weekday < wdayThis )
1732 {
1733 // need to advance a week
1734 diff = 7 - (wdayThis - weekday);
1735 }
1736 else // weekday > wdayThis
1737 {
1738 diff = weekday - wdayThis;
1739 }
1740
1741 return Add(wxDateSpan::Days(diff));
1742 }
1743
1744 wxDateTime& wxDateTime::SetToPrevWeekDay(WeekDay weekday)
1745 {
1746 wxDATETIME_CHECK( weekday != Inv_WeekDay, _T("invalid weekday") );
1747
1748 int diff;
1749 WeekDay wdayThis = GetWeekDay();
1750 if ( weekday == wdayThis )
1751 {
1752 // nothing to do
1753 return *this;
1754 }
1755 else if ( weekday > wdayThis )
1756 {
1757 // need to go to previous week
1758 diff = 7 - (weekday - wdayThis);
1759 }
1760 else // weekday < wdayThis
1761 {
1762 diff = wdayThis - weekday;
1763 }
1764
1765 return Subtract(wxDateSpan::Days(diff));
1766 }
1767
1768 bool wxDateTime::SetToWeekDay(WeekDay weekday,
1769 int n,
1770 Month month,
1771 int year)
1772 {
1773 wxCHECK_MSG( weekday != Inv_WeekDay, false, _T("invalid weekday") );
1774
1775 // we don't check explicitly that -5 <= n <= 5 because we will return false
1776 // anyhow in such case - but may be should still give an assert for it?
1777
1778 // take the current month/year if none specified
1779 ReplaceDefaultYearMonthWithCurrent(&year, &month);
1780
1781 wxDateTime dt;
1782
1783 // TODO this probably could be optimised somehow...
1784
1785 if ( n > 0 )
1786 {
1787 // get the first day of the month
1788 dt.Set(1, month, year);
1789
1790 // get its wday
1791 WeekDay wdayFirst = dt.GetWeekDay();
1792
1793 // go to the first weekday of the month
1794 int diff = weekday - wdayFirst;
1795 if ( diff < 0 )
1796 diff += 7;
1797
1798 // add advance n-1 weeks more
1799 diff += 7*(n - 1);
1800
1801 dt += wxDateSpan::Days(diff);
1802 }
1803 else // count from the end of the month
1804 {
1805 // get the last day of the month
1806 dt.SetToLastMonthDay(month, year);
1807
1808 // get its wday
1809 WeekDay wdayLast = dt.GetWeekDay();
1810
1811 // go to the last weekday of the month
1812 int diff = wdayLast - weekday;
1813 if ( diff < 0 )
1814 diff += 7;
1815
1816 // and rewind n-1 weeks from there
1817 diff += 7*(-n - 1);
1818
1819 dt -= wxDateSpan::Days(diff);
1820 }
1821
1822 // check that it is still in the same month
1823 if ( dt.GetMonth() == month )
1824 {
1825 *this = dt;
1826
1827 return true;
1828 }
1829 else
1830 {
1831 // no such day in this month
1832 return false;
1833 }
1834 }
1835
1836 static inline
1837 wxDateTime::wxDateTime_t GetDayOfYearFromTm(const wxDateTime::Tm& tm)
1838 {
1839 return gs_cumulatedDays[wxDateTime::IsLeapYear(tm.year)][tm.mon] + tm.mday;
1840 }
1841
1842 wxDateTime::wxDateTime_t wxDateTime::GetDayOfYear(const TimeZone& tz) const
1843 {
1844 return GetDayOfYearFromTm(GetTm(tz));
1845 }
1846
1847 // convert Sun, Mon, ..., Sat into 6, 0, ..., 5
1848 static inline int ConvertWeekDayToMondayBase(int wd)
1849 {
1850 return wd == wxDateTime::Sun ? 6 : wd - 1;
1851 }
1852
1853 wxDateTime::wxDateTime_t
1854 wxDateTime::GetWeekOfYear(wxDateTime::WeekFlags flags, const TimeZone& tz) const
1855 {
1856 if ( flags == Default_First )
1857 {
1858 flags = GetCountry() == USA ? Sunday_First : Monday_First;
1859 }
1860
1861 Tm tm(GetTm(tz));
1862 wxDateTime_t nDayInYear = GetDayOfYearFromTm(tm);
1863
1864 int wdTarget = GetWeekDay(tz);
1865 int wdYearStart = wxDateTime(1, Jan, GetYear()).GetWeekDay();
1866 int week;
1867 if ( flags == Sunday_First )
1868 {
1869 // FIXME: First week is not calculated correctly.
1870 week = (nDayInYear - wdTarget + 7) / 7;
1871 if ( wdYearStart == Wed || wdYearStart == Thu )
1872 week++;
1873 }
1874 else // week starts with monday
1875 {
1876 // adjust the weekdays to non-US style.
1877 wdYearStart = ConvertWeekDayToMondayBase(wdYearStart);
1878 wdTarget = ConvertWeekDayToMondayBase(wdTarget);
1879
1880 // quoting from http://www.cl.cam.ac.uk/~mgk25/iso-time.html:
1881 //
1882 // Week 01 of a year is per definition the first week that has the
1883 // Thursday in this year, which is equivalent to the week that
1884 // contains the fourth day of January. In other words, the first
1885 // week of a new year is the week that has the majority of its
1886 // days in the new year. Week 01 might also contain days from the
1887 // previous year and the week before week 01 of a year is the last
1888 // week (52 or 53) of the previous year even if it contains days
1889 // from the new year. A week starts with Monday (day 1) and ends
1890 // with Sunday (day 7).
1891 //
1892
1893 // if Jan 1 is Thursday or less, it is in the first week of this year
1894 if ( wdYearStart < 4 )
1895 {
1896 // count the number of entire weeks between Jan 1 and this date
1897 week = (nDayInYear + wdYearStart + 6 - wdTarget)/7;
1898
1899 // be careful to check for overflow in the next year
1900 if ( week == 53 && tm.mday - wdTarget > 28 )
1901 week = 1;
1902 }
1903 else // Jan 1 is in the last week of the previous year
1904 {
1905 // check if we happen to be at the last week of previous year:
1906 if ( tm.mon == Jan && tm.mday < 8 - wdYearStart )
1907 week = wxDateTime(31, Dec, GetYear()-1).GetWeekOfYear();
1908 else
1909 week = (nDayInYear + wdYearStart - 1 - wdTarget)/7;
1910 }
1911 }
1912
1913 return week;
1914 }
1915
1916 wxDateTime::wxDateTime_t wxDateTime::GetWeekOfMonth(wxDateTime::WeekFlags flags,
1917 const TimeZone& tz) const
1918 {
1919 Tm tm = GetTm(tz);
1920 wxDateTime dtMonthStart = wxDateTime(1, tm.mon, tm.year);
1921 int nWeek = GetWeekOfYear(flags) - dtMonthStart.GetWeekOfYear(flags) + 1;
1922 if ( nWeek < 0 )
1923 {
1924 // this may happen for January when Jan, 1 is the last week of the
1925 // previous year
1926 nWeek += IsLeapYear(tm.year - 1) ? 53 : 52;
1927 }
1928
1929 return (wxDateTime::wxDateTime_t)nWeek;
1930 }
1931
1932 wxDateTime& wxDateTime::SetToYearDay(wxDateTime::wxDateTime_t yday)
1933 {
1934 int year = GetYear();
1935 wxDATETIME_CHECK( (0 < yday) && (yday <= GetNumberOfDays(year)),
1936 _T("invalid year day") );
1937
1938 bool isLeap = IsLeapYear(year);
1939 for ( Month mon = Jan; mon < Inv_Month; wxNextMonth(mon) )
1940 {
1941 // for Dec, we can't compare with gs_cumulatedDays[mon + 1], but we
1942 // don't need it neither - because of the CHECK above we know that
1943 // yday lies in December then
1944 if ( (mon == Dec) || (yday <= gs_cumulatedDays[isLeap][mon + 1]) )
1945 {
1946 Set(yday - gs_cumulatedDays[isLeap][mon], mon, year);
1947
1948 break;
1949 }
1950 }
1951
1952 return *this;
1953 }
1954
1955 // ----------------------------------------------------------------------------
1956 // Julian day number conversion and related stuff
1957 // ----------------------------------------------------------------------------
1958
1959 double wxDateTime::GetJulianDayNumber() const
1960 {
1961 // JDN are always expressed for the UTC dates
1962 Tm tm(ToTimezone(UTC).GetTm(UTC));
1963
1964 double result = GetTruncatedJDN(tm.mday, tm.mon, tm.year);
1965
1966 // add the part GetTruncatedJDN() neglected
1967 result += 0.5;
1968
1969 // and now add the time: 86400 sec = 1 JDN
1970 return result + ((double)(60*(60*tm.hour + tm.min) + tm.sec)) / 86400;
1971 }
1972
1973 double wxDateTime::GetRataDie() const
1974 {
1975 // March 1 of the year 0 is Rata Die day -306 and JDN 1721119.5
1976 return GetJulianDayNumber() - 1721119.5 - 306;
1977 }
1978
1979 // ----------------------------------------------------------------------------
1980 // timezone and DST stuff
1981 // ----------------------------------------------------------------------------
1982
1983 int wxDateTime::IsDST(wxDateTime::Country country) const
1984 {
1985 wxCHECK_MSG( country == Country_Default, -1,
1986 _T("country support not implemented") );
1987
1988 // use the C RTL for the dates in the standard range
1989 time_t timet = GetTicks();
1990 if ( timet != (time_t)-1 )
1991 {
1992 tm *tm = localtime(&timet);
1993
1994 wxCHECK_MSG( tm, -1, _T("localtime() failed") );
1995
1996 return tm->tm_isdst;
1997 }
1998 else
1999 {
2000 int year = GetYear();
2001
2002 if ( !IsDSTApplicable(year, country) )
2003 {
2004 // no DST time in this year in this country
2005 return -1;
2006 }
2007
2008 return IsBetween(GetBeginDST(year, country), GetEndDST(year, country));
2009 }
2010 }
2011
2012 wxDateTime& wxDateTime::MakeTimezone(const TimeZone& tz, bool noDST)
2013 {
2014 long secDiff = GetTimeZone() + tz.GetOffset();
2015
2016 // we need to know whether DST is or not in effect for this date unless
2017 // the test disabled by the caller
2018 if ( !noDST && (IsDST() == 1) )
2019 {
2020 // FIXME we assume that the DST is always shifted by 1 hour
2021 secDiff -= 3600;
2022 }
2023
2024 return Subtract(wxTimeSpan::Seconds(secDiff));
2025 }
2026
2027 // ----------------------------------------------------------------------------
2028 // wxDateTime to/from text representations
2029 // ----------------------------------------------------------------------------
2030
2031 wxString wxDateTime::Format(const wxChar *format, const TimeZone& tz) const
2032 {
2033 wxCHECK_MSG( format, _T(""), _T("NULL format in wxDateTime::Format") );
2034
2035 // we have to use our own implementation if the date is out of range of
2036 // strftime() or if we use non standard specificators
2037 time_t time = GetTicks();
2038 if ( (time != (time_t)-1) && !wxStrstr(format, _T("%l")) )
2039 {
2040 // use strftime()
2041 tm *tm;
2042 if ( tz.GetOffset() == -GetTimeZone() )
2043 {
2044 // we are working with local time
2045 tm = localtime(&time);
2046
2047 // should never happen
2048 wxCHECK_MSG( tm, wxEmptyString, _T("localtime() failed") );
2049 }
2050 else
2051 {
2052 time += (int)tz.GetOffset();
2053
2054 #if defined(__VMS__) || defined(__WATCOMC__) // time is unsigned so avoid warning
2055 int time2 = (int) time;
2056 if ( time2 >= 0 )
2057 #else
2058 if ( time >= 0 )
2059 #endif
2060 {
2061 tm = gmtime(&time);
2062
2063 // should never happen
2064 wxCHECK_MSG( tm, wxEmptyString, _T("gmtime() failed") );
2065 }
2066 else
2067 {
2068 tm = (struct tm *)NULL;
2069 }
2070 }
2071
2072 if ( tm )
2073 {
2074 return CallStrftime(format, tm);
2075 }
2076 //else: use generic code below
2077 }
2078
2079 // we only parse ANSI C format specifications here, no POSIX 2
2080 // complications, no GNU extensions but we do add support for a "%l" format
2081 // specifier allowing to get the number of milliseconds
2082 Tm tm = GetTm(tz);
2083
2084 // used for calls to strftime() when we only deal with time
2085 struct tm tmTimeOnly;
2086 tmTimeOnly.tm_hour = tm.hour;
2087 tmTimeOnly.tm_min = tm.min;
2088 tmTimeOnly.tm_sec = tm.sec;
2089 tmTimeOnly.tm_wday = 0;
2090 tmTimeOnly.tm_yday = 0;
2091 tmTimeOnly.tm_mday = 1; // any date will do
2092 tmTimeOnly.tm_mon = 0;
2093 tmTimeOnly.tm_year = 76;
2094 tmTimeOnly.tm_isdst = 0; // no DST, we adjust for tz ourselves
2095
2096 wxString tmp, res, fmt;
2097 for ( const wxChar *p = format; *p; p++ )
2098 {
2099 if ( *p != _T('%') )
2100 {
2101 // copy as is
2102 res += *p;
2103
2104 continue;
2105 }
2106
2107 // set the default format
2108 switch ( *++p )
2109 {
2110 case _T('Y'): // year has 4 digits
2111 fmt = _T("%04d");
2112 break;
2113
2114 case _T('j'): // day of year has 3 digits
2115 case _T('l'): // milliseconds have 3 digits
2116 fmt = _T("%03d");
2117 break;
2118
2119 case _T('w'): // week day as number has only one
2120 fmt = _T("%d");
2121 break;
2122
2123 default:
2124 // it's either another valid format specifier in which case
2125 // the format is "%02d" (for all the rest) or we have the
2126 // field width preceding the format in which case it will
2127 // override the default format anyhow
2128 fmt = _T("%02d");
2129 }
2130
2131 bool restart = true;
2132 while ( restart )
2133 {
2134 restart = false;
2135
2136 // start of the format specification
2137 switch ( *p )
2138 {
2139 case _T('a'): // a weekday name
2140 case _T('A'):
2141 // second parameter should be true for abbreviated names
2142 res += GetWeekDayName(tm.GetWeekDay(),
2143 *p == _T('a') ? Name_Abbr : Name_Full);
2144 break;
2145
2146 case _T('b'): // a month name
2147 case _T('B'):
2148 res += GetMonthName(tm.mon,
2149 *p == _T('b') ? Name_Abbr : Name_Full);
2150 break;
2151
2152 case _T('c'): // locale default date and time representation
2153 case _T('x'): // locale default date representation
2154 //
2155 // the problem: there is no way to know what do these format
2156 // specifications correspond to for the current locale.
2157 //
2158 // the solution: use a hack and still use strftime(): first
2159 // find the YEAR which is a year in the strftime() range (1970
2160 // - 2038) whose Jan 1 falls on the same week day as the Jan 1
2161 // of the real year. Then make a copy of the format and
2162 // replace all occurences of YEAR in it with some unique
2163 // string not appearing anywhere else in it, then use
2164 // strftime() to format the date in year YEAR and then replace
2165 // YEAR back by the real year and the unique replacement
2166 // string back with YEAR. Notice that "all occurences of YEAR"
2167 // means all occurences of 4 digit as well as 2 digit form!
2168 //
2169 // the bugs: we assume that neither of %c nor %x contains any
2170 // fields which may change between the YEAR and real year. For
2171 // example, the week number (%U, %W) and the day number (%j)
2172 // will change if one of these years is leap and the other one
2173 // is not!
2174 {
2175 // find the YEAR: normally, for any year X, Jan 1 or the
2176 // year X + 28 is the same weekday as Jan 1 of X (because
2177 // the weekday advances by 1 for each normal X and by 2
2178 // for each leap X, hence by 5 every 4 years or by 35
2179 // which is 0 mod 7 every 28 years) but this rule breaks
2180 // down if there are years between X and Y which are
2181 // divisible by 4 but not leap (i.e. divisible by 100 but
2182 // not 400), hence the correction.
2183
2184 int yearReal = GetYear(tz);
2185 int mod28 = yearReal % 28;
2186
2187 // be careful to not go too far - we risk to leave the
2188 // supported range
2189 int year;
2190 if ( mod28 < 10 )
2191 {
2192 year = 1988 + mod28; // 1988 == 0 (mod 28)
2193 }
2194 else
2195 {
2196 year = 1970 + mod28 - 10; // 1970 == 10 (mod 28)
2197 }
2198
2199 int nCentury = year / 100,
2200 nCenturyReal = yearReal / 100;
2201
2202 // need to adjust for the years divisble by 400 which are
2203 // not leap but are counted like leap ones if we just take
2204 // the number of centuries in between for nLostWeekDays
2205 int nLostWeekDays = (nCentury - nCenturyReal) -
2206 (nCentury / 4 - nCenturyReal / 4);
2207
2208 // we have to gain back the "lost" weekdays: note that the
2209 // effect of this loop is to not do anything to
2210 // nLostWeekDays (which we won't use any more), but to
2211 // (indirectly) set the year correctly
2212 while ( (nLostWeekDays % 7) != 0 )
2213 {
2214 nLostWeekDays += year++ % 4 ? 1 : 2;
2215 }
2216
2217 // at any rate, we couldn't go further than 1988 + 9 + 28!
2218 wxASSERT_MSG( year < 2030,
2219 _T("logic error in wxDateTime::Format") );
2220
2221 wxString strYear, strYear2;
2222 strYear.Printf(_T("%d"), year);
2223 strYear2.Printf(_T("%d"), year % 100);
2224
2225 // find two strings not occuring in format (this is surely
2226 // not optimal way of doing it... improvements welcome!)
2227 wxString fmt = format;
2228 wxString replacement = (wxChar)-1;
2229 while ( fmt.Find(replacement) != wxNOT_FOUND )
2230 {
2231 replacement << (wxChar)-1;
2232 }
2233
2234 wxString replacement2 = (wxChar)-2;
2235 while ( fmt.Find(replacement) != wxNOT_FOUND )
2236 {
2237 replacement << (wxChar)-2;
2238 }
2239
2240 // replace all occurences of year with it
2241 bool wasReplaced = fmt.Replace(strYear, replacement) > 0;
2242 if ( !wasReplaced )
2243 wasReplaced = fmt.Replace(strYear2, replacement2) > 0;
2244
2245 // use strftime() to format the same date but in supported
2246 // year
2247 //
2248 // NB: we assume that strftime() doesn't check for the
2249 // date validity and will happily format the date
2250 // corresponding to Feb 29 of a non leap year (which
2251 // may happen if yearReal was leap and year is not)
2252 struct tm tmAdjusted;
2253 InitTm(tmAdjusted);
2254 tmAdjusted.tm_hour = tm.hour;
2255 tmAdjusted.tm_min = tm.min;
2256 tmAdjusted.tm_sec = tm.sec;
2257 tmAdjusted.tm_wday = tm.GetWeekDay();
2258 tmAdjusted.tm_yday = GetDayOfYear();
2259 tmAdjusted.tm_mday = tm.mday;
2260 tmAdjusted.tm_mon = tm.mon;
2261 tmAdjusted.tm_year = year - 1900;
2262 tmAdjusted.tm_isdst = 0; // no DST, already adjusted
2263 wxString str = CallStrftime(*p == _T('c') ? _T("%c")
2264 : _T("%x"),
2265 &tmAdjusted);
2266
2267 // now replace the occurence of 1999 with the real year
2268 wxString strYearReal, strYearReal2;
2269 strYearReal.Printf(_T("%04d"), yearReal);
2270 strYearReal2.Printf(_T("%02d"), yearReal % 100);
2271 str.Replace(strYear, strYearReal);
2272 str.Replace(strYear2, strYearReal2);
2273
2274 // and replace back all occurences of replacement string
2275 if ( wasReplaced )
2276 {
2277 str.Replace(replacement2, strYear2);
2278 str.Replace(replacement, strYear);
2279 }
2280
2281 res += str;
2282 }
2283 break;
2284
2285 case _T('d'): // day of a month (01-31)
2286 res += wxString::Format(fmt, tm.mday);
2287 break;
2288
2289 case _T('H'): // hour in 24h format (00-23)
2290 res += wxString::Format(fmt, tm.hour);
2291 break;
2292
2293 case _T('I'): // hour in 12h format (01-12)
2294 {
2295 // 24h -> 12h, 0h -> 12h too
2296 int hour12 = tm.hour > 12 ? tm.hour - 12
2297 : tm.hour ? tm.hour : 12;
2298 res += wxString::Format(fmt, hour12);
2299 }
2300 break;
2301
2302 case _T('j'): // day of the year
2303 res += wxString::Format(fmt, GetDayOfYear(tz));
2304 break;
2305
2306 case _T('l'): // milliseconds (NOT STANDARD)
2307 res += wxString::Format(fmt, GetMillisecond(tz));
2308 break;
2309
2310 case _T('m'): // month as a number (01-12)
2311 res += wxString::Format(fmt, tm.mon + 1);
2312 break;
2313
2314 case _T('M'): // minute as a decimal number (00-59)
2315 res += wxString::Format(fmt, tm.min);
2316 break;
2317
2318 case _T('p'): // AM or PM string
2319 res += CallStrftime(_T("%p"), &tmTimeOnly);
2320 break;
2321
2322 case _T('S'): // second as a decimal number (00-61)
2323 res += wxString::Format(fmt, tm.sec);
2324 break;
2325
2326 case _T('U'): // week number in the year (Sunday 1st week day)
2327 res += wxString::Format(fmt, GetWeekOfYear(Sunday_First, tz));
2328 break;
2329
2330 case _T('W'): // week number in the year (Monday 1st week day)
2331 res += wxString::Format(fmt, GetWeekOfYear(Monday_First, tz));
2332 break;
2333
2334 case _T('w'): // weekday as a number (0-6), Sunday = 0
2335 res += wxString::Format(fmt, tm.GetWeekDay());
2336 break;
2337
2338 // case _T('x'): -- handled with "%c"
2339
2340 case _T('X'): // locale default time representation
2341 // just use strftime() to format the time for us
2342 res += CallStrftime(_T("%X"), &tmTimeOnly);
2343 break;
2344
2345 case _T('y'): // year without century (00-99)
2346 res += wxString::Format(fmt, tm.year % 100);
2347 break;
2348
2349 case _T('Y'): // year with century
2350 res += wxString::Format(fmt, tm.year);
2351 break;
2352
2353 case _T('Z'): // timezone name
2354 res += CallStrftime(_T("%Z"), &tmTimeOnly);
2355 break;
2356
2357 default:
2358 // is it the format width?
2359 fmt.Empty();
2360 while ( *p == _T('-') || *p == _T('+') ||
2361 *p == _T(' ') || wxIsdigit(*p) )
2362 {
2363 fmt += *p;
2364 }
2365
2366 if ( !fmt.IsEmpty() )
2367 {
2368 // we've only got the flags and width so far in fmt
2369 fmt.Prepend(_T('%'));
2370 fmt.Append(_T('d'));
2371
2372 restart = true;
2373
2374 break;
2375 }
2376
2377 // no, it wasn't the width
2378 wxFAIL_MSG(_T("unknown format specificator"));
2379
2380 // fall through and just copy it nevertheless
2381
2382 case _T('%'): // a percent sign
2383 res += *p;
2384 break;
2385
2386 case 0: // the end of string
2387 wxFAIL_MSG(_T("missing format at the end of string"));
2388
2389 // just put the '%' which was the last char in format
2390 res += _T('%');
2391 break;
2392 }
2393 }
2394 }
2395
2396 return res;
2397 }
2398
2399 // this function parses a string in (strict) RFC 822 format: see the section 5
2400 // of the RFC for the detailed description, but briefly it's something of the
2401 // form "Sat, 18 Dec 1999 00:48:30 +0100"
2402 //
2403 // this function is "strict" by design - it must reject anything except true
2404 // RFC822 time specs.
2405 //
2406 // TODO a great candidate for using reg exps
2407 const wxChar *wxDateTime::ParseRfc822Date(const wxChar* date)
2408 {
2409 wxCHECK_MSG( date, (wxChar *)NULL, _T("NULL pointer in wxDateTime::Parse") );
2410
2411 const wxChar *p = date;
2412 const wxChar *comma = wxStrchr(p, _T(','));
2413 if ( comma )
2414 {
2415 // the part before comma is the weekday
2416
2417 // skip it for now - we don't use but might check that it really
2418 // corresponds to the specfied date
2419 p = comma + 1;
2420
2421 if ( *p != _T(' ') )
2422 {
2423 wxLogDebug(_T("no space after weekday in RFC822 time spec"));
2424
2425 return (wxChar *)NULL;
2426 }
2427
2428 p++; // skip space
2429 }
2430
2431 // the following 1 or 2 digits are the day number
2432 if ( !wxIsdigit(*p) )
2433 {
2434 wxLogDebug(_T("day number expected in RFC822 time spec, none found"));
2435
2436 return (wxChar *)NULL;
2437 }
2438
2439 wxDateTime_t day = *p++ - _T('0');
2440 if ( wxIsdigit(*p) )
2441 {
2442 day *= 10;
2443 day += *p++ - _T('0');
2444 }
2445
2446 if ( *p++ != _T(' ') )
2447 {
2448 return (wxChar *)NULL;
2449 }
2450
2451 // the following 3 letters specify the month
2452 wxString monName(p, 3);
2453 Month mon;
2454 if ( monName == _T("Jan") )
2455 mon = Jan;
2456 else if ( monName == _T("Feb") )
2457 mon = Feb;
2458 else if ( monName == _T("Mar") )
2459 mon = Mar;
2460 else if ( monName == _T("Apr") )
2461 mon = Apr;
2462 else if ( monName == _T("May") )
2463 mon = May;
2464 else if ( monName == _T("Jun") )
2465 mon = Jun;
2466 else if ( monName == _T("Jul") )
2467 mon = Jul;
2468 else if ( monName == _T("Aug") )
2469 mon = Aug;
2470 else if ( monName == _T("Sep") )
2471 mon = Sep;
2472 else if ( monName == _T("Oct") )
2473 mon = Oct;
2474 else if ( monName == _T("Nov") )
2475 mon = Nov;
2476 else if ( monName == _T("Dec") )
2477 mon = Dec;
2478 else
2479 {
2480 wxLogDebug(_T("Invalid RFC 822 month name '%s'"), monName.c_str());
2481
2482 return (wxChar *)NULL;
2483 }
2484
2485 p += 3;
2486
2487 if ( *p++ != _T(' ') )
2488 {
2489 return (wxChar *)NULL;
2490 }
2491
2492 // next is the year
2493 if ( !wxIsdigit(*p) )
2494 {
2495 // no year?
2496 return (wxChar *)NULL;
2497 }
2498
2499 int year = *p++ - _T('0');
2500
2501 if ( !wxIsdigit(*p) )
2502 {
2503 // should have at least 2 digits in the year
2504 return (wxChar *)NULL;
2505 }
2506
2507 year *= 10;
2508 year += *p++ - _T('0');
2509
2510 // is it a 2 digit year (as per original RFC 822) or a 4 digit one?
2511 if ( wxIsdigit(*p) )
2512 {
2513 year *= 10;
2514 year += *p++ - _T('0');
2515
2516 if ( !wxIsdigit(*p) )
2517 {
2518 // no 3 digit years please
2519 return (wxChar *)NULL;
2520 }
2521
2522 year *= 10;
2523 year += *p++ - _T('0');
2524 }
2525
2526 if ( *p++ != _T(' ') )
2527 {
2528 return (wxChar *)NULL;
2529 }
2530
2531 // time is in the format hh:mm:ss and seconds are optional
2532 if ( !wxIsdigit(*p) )
2533 {
2534 return (wxChar *)NULL;
2535 }
2536
2537 wxDateTime_t hour = *p++ - _T('0');
2538
2539 if ( !wxIsdigit(*p) )
2540 {
2541 return (wxChar *)NULL;
2542 }
2543
2544 hour *= 10;
2545 hour += *p++ - _T('0');
2546
2547 if ( *p++ != _T(':') )
2548 {
2549 return (wxChar *)NULL;
2550 }
2551
2552 if ( !wxIsdigit(*p) )
2553 {
2554 return (wxChar *)NULL;
2555 }
2556
2557 wxDateTime_t min = *p++ - _T('0');
2558
2559 if ( !wxIsdigit(*p) )
2560 {
2561 return (wxChar *)NULL;
2562 }
2563
2564 min *= 10;
2565 min += *p++ - _T('0');
2566
2567 wxDateTime_t sec = 0;
2568 if ( *p++ == _T(':') )
2569 {
2570 if ( !wxIsdigit(*p) )
2571 {
2572 return (wxChar *)NULL;
2573 }
2574
2575 sec = *p++ - _T('0');
2576
2577 if ( !wxIsdigit(*p) )
2578 {
2579 return (wxChar *)NULL;
2580 }
2581
2582 sec *= 10;
2583 sec += *p++ - _T('0');
2584 }
2585
2586 if ( *p++ != _T(' ') )
2587 {
2588 return (wxChar *)NULL;
2589 }
2590
2591 // and now the interesting part: the timezone
2592 int offset;
2593 if ( *p == _T('-') || *p == _T('+') )
2594 {
2595 // the explicit offset given: it has the form of hhmm
2596 bool plus = *p++ == _T('+');
2597
2598 if ( !wxIsdigit(*p) || !wxIsdigit(*(p + 1)) )
2599 {
2600 return (wxChar *)NULL;
2601 }
2602
2603 // hours
2604 offset = 60*(10*(*p - _T('0')) + (*(p + 1) - _T('0')));
2605
2606 p += 2;
2607
2608 if ( !wxIsdigit(*p) || !wxIsdigit(*(p + 1)) )
2609 {
2610 return (wxChar *)NULL;
2611 }
2612
2613 // minutes
2614 offset += 10*(*p - _T('0')) + (*(p + 1) - _T('0'));
2615
2616 if ( !plus )
2617 {
2618 offset = -offset;
2619 }
2620
2621 p += 2;
2622 }
2623 else
2624 {
2625 // the symbolic timezone given: may be either military timezone or one
2626 // of standard abbreviations
2627 if ( !*(p + 1) )
2628 {
2629 // military: Z = UTC, J unused, A = -1, ..., Y = +12
2630 static const int offsets[26] =
2631 {
2632 //A B C D E F G H I J K L M
2633 -1, -2, -3, -4, -5, -6, -7, -8, -9, 0, -10, -11, -12,
2634 //N O P R Q S T U V W Z Y Z
2635 +1, +2, +3, +4, +5, +6, +7, +8, +9, +10, +11, +12, 0
2636 };
2637
2638 if ( *p < _T('A') || *p > _T('Z') || *p == _T('J') )
2639 {
2640 wxLogDebug(_T("Invalid militaty timezone '%c'"), *p);
2641
2642 return (wxChar *)NULL;
2643 }
2644
2645 offset = offsets[*p++ - _T('A')];
2646 }
2647 else
2648 {
2649 // abbreviation
2650 wxString tz = p;
2651 if ( tz == _T("UT") || tz == _T("UTC") || tz == _T("GMT") )
2652 offset = 0;
2653 else if ( tz == _T("AST") )
2654 offset = AST - GMT0;
2655 else if ( tz == _T("ADT") )
2656 offset = ADT - GMT0;
2657 else if ( tz == _T("EST") )
2658 offset = EST - GMT0;
2659 else if ( tz == _T("EDT") )
2660 offset = EDT - GMT0;
2661 else if ( tz == _T("CST") )
2662 offset = CST - GMT0;
2663 else if ( tz == _T("CDT") )
2664 offset = CDT - GMT0;
2665 else if ( tz == _T("MST") )
2666 offset = MST - GMT0;
2667 else if ( tz == _T("MDT") )
2668 offset = MDT - GMT0;
2669 else if ( tz == _T("PST") )
2670 offset = PST - GMT0;
2671 else if ( tz == _T("PDT") )
2672 offset = PDT - GMT0;
2673 else
2674 {
2675 wxLogDebug(_T("Unknown RFC 822 timezone '%s'"), p);
2676
2677 return (wxChar *)NULL;
2678 }
2679
2680 p += tz.length();
2681 }
2682
2683 // make it minutes
2684 offset *= 60;
2685 }
2686
2687 // the spec was correct
2688 Set(day, mon, year, hour, min, sec);
2689 MakeTimezone((wxDateTime_t)(60*offset));
2690
2691 return p;
2692 }
2693
2694 const wxChar *wxDateTime::ParseFormat(const wxChar *date,
2695 const wxChar *format,
2696 const wxDateTime& dateDef)
2697 {
2698 wxCHECK_MSG( date && format, (wxChar *)NULL,
2699 _T("NULL pointer in wxDateTime::ParseFormat()") );
2700
2701 wxString str;
2702 unsigned long num;
2703
2704 // what fields have we found?
2705 bool haveWDay = false,
2706 haveYDay = false,
2707 haveDay = false,
2708 haveMon = false,
2709 haveYear = false,
2710 haveHour = false,
2711 haveMin = false,
2712 haveSec = false;
2713
2714 bool hourIsIn12hFormat = false, // or in 24h one?
2715 isPM = false; // AM by default
2716
2717 // and the value of the items we have (init them to get rid of warnings)
2718 wxDateTime_t sec = 0,
2719 min = 0,
2720 hour = 0;
2721 WeekDay wday = Inv_WeekDay;
2722 wxDateTime_t yday = 0,
2723 mday = 0;
2724 wxDateTime::Month mon = Inv_Month;
2725 int year = 0;
2726
2727 const wxChar *input = date;
2728 for ( const wxChar *fmt = format; *fmt; fmt++ )
2729 {
2730 if ( *fmt != _T('%') )
2731 {
2732 if ( wxIsspace(*fmt) )
2733 {
2734 // a white space in the format string matches 0 or more white
2735 // spaces in the input
2736 while ( wxIsspace(*input) )
2737 {
2738 input++;
2739 }
2740 }
2741 else // !space
2742 {
2743 // any other character (not whitespace, not '%') must be
2744 // matched by itself in the input
2745 if ( *input++ != *fmt )
2746 {
2747 // no match
2748 return (wxChar *)NULL;
2749 }
2750 }
2751
2752 // done with this format char
2753 continue;
2754 }
2755
2756 // start of a format specification
2757
2758 // parse the optional width
2759 size_t width = 0;
2760 while ( wxIsdigit(*++fmt) )
2761 {
2762 width *= 10;
2763 width += *fmt - _T('0');
2764 }
2765
2766 // the default widths for the various fields
2767 if ( !width )
2768 {
2769 switch ( *fmt )
2770 {
2771 case _T('Y'): // year has 4 digits
2772 width = 4;
2773 break;
2774
2775 case _T('j'): // day of year has 3 digits
2776 case _T('l'): // milliseconds have 3 digits
2777 width = 3;
2778 break;
2779
2780 case _T('w'): // week day as number has only one
2781 width = 1;
2782 break;
2783
2784 default:
2785 // default for all other fields
2786 width = 2;
2787 }
2788 }
2789
2790 // then the format itself
2791 switch ( *fmt )
2792 {
2793 case _T('a'): // a weekday name
2794 case _T('A'):
2795 {
2796 int flag = *fmt == _T('a') ? Name_Abbr : Name_Full;
2797 wday = GetWeekDayFromName(GetAlphaToken(input), flag);
2798 if ( wday == Inv_WeekDay )
2799 {
2800 // no match
2801 return (wxChar *)NULL;
2802 }
2803 }
2804 haveWDay = true;
2805 break;
2806
2807 case _T('b'): // a month name
2808 case _T('B'):
2809 {
2810 int flag = *fmt == _T('b') ? Name_Abbr : Name_Full;
2811 mon = GetMonthFromName(GetAlphaToken(input), flag);
2812 if ( mon == Inv_Month )
2813 {
2814 // no match
2815 return (wxChar *)NULL;
2816 }
2817 }
2818 haveMon = true;
2819 break;
2820
2821 case _T('c'): // locale default date and time representation
2822 {
2823 wxDateTime dt;
2824
2825 // this is the format which corresponds to ctime() output
2826 // and strptime("%c") should parse it, so try it first
2827 static const wxChar *fmtCtime = _T("%a %b %d %H:%M:%S %Y");
2828
2829 const wxChar *result = dt.ParseFormat(input, fmtCtime);
2830 if ( !result )
2831 {
2832 result = dt.ParseFormat(input, _T("%x %X"));
2833 }
2834
2835 if ( !result )
2836 {
2837 result = dt.ParseFormat(input, _T("%X %x"));
2838 }
2839
2840 if ( !result )
2841 {
2842 // we've tried everything and still no match
2843 return (wxChar *)NULL;
2844 }
2845
2846 Tm tm = dt.GetTm();
2847
2848 haveDay = haveMon = haveYear =
2849 haveHour = haveMin = haveSec = true;
2850
2851 hour = tm.hour;
2852 min = tm.min;
2853 sec = tm.sec;
2854
2855 year = tm.year;
2856 mon = tm.mon;
2857 mday = tm.mday;
2858
2859 input = result;
2860 }
2861 break;
2862
2863 case _T('d'): // day of a month (01-31)
2864 if ( !GetNumericToken(width, input, &num) ||
2865 (num > 31) || (num < 1) )
2866 {
2867 // no match
2868 return (wxChar *)NULL;
2869 }
2870
2871 // we can't check whether the day range is correct yet, will
2872 // do it later - assume ok for now
2873 haveDay = true;
2874 mday = (wxDateTime_t)num;
2875 break;
2876
2877 case _T('H'): // hour in 24h format (00-23)
2878 if ( !GetNumericToken(width, input, &num) || (num > 23) )
2879 {
2880 // no match
2881 return (wxChar *)NULL;
2882 }
2883
2884 haveHour = true;
2885 hour = (wxDateTime_t)num;
2886 break;
2887
2888 case _T('I'): // hour in 12h format (01-12)
2889 if ( !GetNumericToken(width, input, &num) || !num || (num > 12) )
2890 {
2891 // no match
2892 return (wxChar *)NULL;
2893 }
2894
2895 haveHour = true;
2896 hourIsIn12hFormat = true;
2897 hour = (wxDateTime_t)(num % 12); // 12 should be 0
2898 break;
2899
2900 case _T('j'): // day of the year
2901 if ( !GetNumericToken(width, input, &num) || !num || (num > 366) )
2902 {
2903 // no match
2904 return (wxChar *)NULL;
2905 }
2906
2907 haveYDay = true;
2908 yday = (wxDateTime_t)num;
2909 break;
2910
2911 case _T('m'): // month as a number (01-12)
2912 if ( !GetNumericToken(width, input, &num) || !num || (num > 12) )
2913 {
2914 // no match
2915 return (wxChar *)NULL;
2916 }
2917
2918 haveMon = true;
2919 mon = (Month)(num - 1);
2920 break;
2921
2922 case _T('M'): // minute as a decimal number (00-59)
2923 if ( !GetNumericToken(width, input, &num) || (num > 59) )
2924 {
2925 // no match
2926 return (wxChar *)NULL;
2927 }
2928
2929 haveMin = true;
2930 min = (wxDateTime_t)num;
2931 break;
2932
2933 case _T('p'): // AM or PM string
2934 {
2935 wxString am, pm, token = GetAlphaToken(input);
2936
2937 GetAmPmStrings(&am, &pm);
2938 if (am.IsEmpty() && pm.IsEmpty())
2939 return (wxChar *)NULL; // no am/pm strings defined
2940 if ( token.CmpNoCase(pm) == 0 )
2941 {
2942 isPM = true;
2943 }
2944 else if ( token.CmpNoCase(am) != 0 )
2945 {
2946 // no match
2947 return (wxChar *)NULL;
2948 }
2949 }
2950 break;
2951
2952 case _T('r'): // time as %I:%M:%S %p
2953 {
2954 wxDateTime dt;
2955 input = dt.ParseFormat(input, _T("%I:%M:%S %p"));
2956 if ( !input )
2957 {
2958 // no match
2959 return (wxChar *)NULL;
2960 }
2961
2962 haveHour = haveMin = haveSec = true;
2963
2964 Tm tm = dt.GetTm();
2965 hour = tm.hour;
2966 min = tm.min;
2967 sec = tm.sec;
2968 }
2969 break;
2970
2971 case _T('R'): // time as %H:%M
2972 {
2973 wxDateTime dt;
2974 input = dt.ParseFormat(input, _T("%H:%M"));
2975 if ( !input )
2976 {
2977 // no match
2978 return (wxChar *)NULL;
2979 }
2980
2981 haveHour = haveMin = true;
2982
2983 Tm tm = dt.GetTm();
2984 hour = tm.hour;
2985 min = tm.min;
2986 }
2987
2988 case _T('S'): // second as a decimal number (00-61)
2989 if ( !GetNumericToken(width, input, &num) || (num > 61) )
2990 {
2991 // no match
2992 return (wxChar *)NULL;
2993 }
2994
2995 haveSec = true;
2996 sec = (wxDateTime_t)num;
2997 break;
2998
2999 case _T('T'): // time as %H:%M:%S
3000 {
3001 wxDateTime dt;
3002 input = dt.ParseFormat(input, _T("%H:%M:%S"));
3003 if ( !input )
3004 {
3005 // no match
3006 return (wxChar *)NULL;
3007 }
3008
3009 haveHour = haveMin = haveSec = true;
3010
3011 Tm tm = dt.GetTm();
3012 hour = tm.hour;
3013 min = tm.min;
3014 sec = tm.sec;
3015 }
3016 break;
3017
3018 case _T('w'): // weekday as a number (0-6), Sunday = 0
3019 if ( !GetNumericToken(width, input, &num) || (wday > 6) )
3020 {
3021 // no match
3022 return (wxChar *)NULL;
3023 }
3024
3025 haveWDay = true;
3026 wday = (WeekDay)num;
3027 break;
3028
3029 case _T('x'): // locale default date representation
3030 #ifdef HAVE_STRPTIME
3031 // try using strptime() -- it may fail even if the input is
3032 // correct but the date is out of range, so we will fall back
3033 // to our generic code anyhow
3034 {
3035 struct tm tm;
3036
3037 const wxChar *result = CallStrptime(input, "%x", &tm);
3038 if ( result )
3039 {
3040 input = result;
3041
3042 haveDay = haveMon = haveYear = true;
3043
3044 year = 1900 + tm.tm_year;
3045 mon = (Month)tm.tm_mon;
3046 mday = tm.tm_mday;
3047
3048 break;
3049 }
3050 }
3051 #endif // HAVE_STRPTIME
3052
3053 // TODO query the LOCALE_IDATE setting under Win32
3054 {
3055 wxDateTime dt;
3056
3057 wxString fmtDate, fmtDateAlt;
3058 if ( IsWestEuropeanCountry(GetCountry()) ||
3059 GetCountry() == Russia )
3060 {
3061 fmtDate = _T("%d/%m/%y");
3062 fmtDateAlt = _T("%m/%d/%y");
3063 }
3064 else // assume USA
3065 {
3066 fmtDate = _T("%m/%d/%y");
3067 fmtDateAlt = _T("%d/%m/%y");
3068 }
3069
3070 const wxChar *result = dt.ParseFormat(input, fmtDate);
3071
3072 if ( !result )
3073 {
3074 // ok, be nice and try another one
3075 result = dt.ParseFormat(input, fmtDateAlt);
3076 }
3077
3078 if ( !result )
3079 {
3080 // bad luck
3081 return (wxChar *)NULL;
3082 }
3083
3084 Tm tm = dt.GetTm();
3085
3086 haveDay = haveMon = haveYear = true;
3087
3088 year = tm.year;
3089 mon = tm.mon;
3090 mday = tm.mday;
3091
3092 input = result;
3093 }
3094
3095 break;
3096
3097 case _T('X'): // locale default time representation
3098 #ifdef HAVE_STRPTIME
3099 {
3100 // use strptime() to do it for us (FIXME !Unicode friendly)
3101 struct tm tm;
3102 input = CallStrptime(input, "%X", &tm);
3103 if ( !input )
3104 {
3105 return (wxChar *)NULL;
3106 }
3107
3108 haveHour = haveMin = haveSec = true;
3109
3110 hour = tm.tm_hour;
3111 min = tm.tm_min;
3112 sec = tm.tm_sec;
3113 }
3114 #else // !HAVE_STRPTIME
3115 // TODO under Win32 we can query the LOCALE_ITIME system
3116 // setting which says whether the default time format is
3117 // 24 or 12 hour
3118 {
3119 // try to parse what follows as "%H:%M:%S" and, if this
3120 // fails, as "%I:%M:%S %p" - this should catch the most
3121 // common cases
3122 wxDateTime dt;
3123
3124 const wxChar *result = dt.ParseFormat(input, _T("%T"));
3125 if ( !result )
3126 {
3127 result = dt.ParseFormat(input, _T("%r"));
3128 }
3129
3130 if ( !result )
3131 {
3132 // no match
3133 return (wxChar *)NULL;
3134 }
3135
3136 haveHour = haveMin = haveSec = true;
3137
3138 Tm tm = dt.GetTm();
3139 hour = tm.hour;
3140 min = tm.min;
3141 sec = tm.sec;
3142
3143 input = result;
3144 }
3145 #endif // HAVE_STRPTIME/!HAVE_STRPTIME
3146 break;
3147
3148 case _T('y'): // year without century (00-99)
3149 if ( !GetNumericToken(width, input, &num) || (num > 99) )
3150 {
3151 // no match
3152 return (wxChar *)NULL;
3153 }
3154
3155 haveYear = true;
3156
3157 // TODO should have an option for roll over date instead of
3158 // hard coding it here
3159 year = (num > 30 ? 1900 : 2000) + (wxDateTime_t)num;
3160 break;
3161
3162 case _T('Y'): // year with century
3163 if ( !GetNumericToken(width, input, &num) )
3164 {
3165 // no match
3166 return (wxChar *)NULL;
3167 }
3168
3169 haveYear = true;
3170 year = (wxDateTime_t)num;
3171 break;
3172
3173 case _T('Z'): // timezone name
3174 wxFAIL_MSG(_T("TODO"));
3175 break;
3176
3177 case _T('%'): // a percent sign
3178 if ( *input++ != _T('%') )
3179 {
3180 // no match
3181 return (wxChar *)NULL;
3182 }
3183 break;
3184
3185 case 0: // the end of string
3186 wxFAIL_MSG(_T("unexpected format end"));
3187
3188 // fall through
3189
3190 default: // not a known format spec
3191 return (wxChar *)NULL;
3192 }
3193 }
3194
3195 // format matched, try to construct a date from what we have now
3196 Tm tmDef;
3197 if ( dateDef.IsValid() )
3198 {
3199 // take this date as default
3200 tmDef = dateDef.GetTm();
3201 }
3202 else if ( IsValid() )
3203 {
3204 // if this date is valid, don't change it
3205 tmDef = GetTm();
3206 }
3207 else
3208 {
3209 // no default and this date is invalid - fall back to Today()
3210 tmDef = Today().GetTm();
3211 }
3212
3213 Tm tm = tmDef;
3214
3215 // set the date
3216 if ( haveYear )
3217 {
3218 tm.year = year;
3219 }
3220
3221 // TODO we don't check here that the values are consistent, if both year
3222 // day and month/day were found, we just ignore the year day and we
3223 // also always ignore the week day
3224 if ( haveMon && haveDay )
3225 {
3226 if ( mday > GetNumOfDaysInMonth(tm.year, mon) )
3227 {
3228 wxLogDebug(_T("bad month day in wxDateTime::ParseFormat"));
3229
3230 return (wxChar *)NULL;
3231 }
3232
3233 tm.mon = mon;
3234 tm.mday = mday;
3235 }
3236 else if ( haveYDay )
3237 {
3238 if ( yday > GetNumberOfDays(tm.year) )
3239 {
3240 wxLogDebug(_T("bad year day in wxDateTime::ParseFormat"));
3241
3242 return (wxChar *)NULL;
3243 }
3244
3245 Tm tm2 = wxDateTime(1, Jan, tm.year).SetToYearDay(yday).GetTm();
3246
3247 tm.mon = tm2.mon;
3248 tm.mday = tm2.mday;
3249 }
3250
3251 // deal with AM/PM
3252 if ( haveHour && hourIsIn12hFormat && isPM )
3253 {
3254 // translate to 24hour format
3255 hour += 12;
3256 }
3257 //else: either already in 24h format or no translation needed
3258
3259 // set the time
3260 if ( haveHour )
3261 {
3262 tm.hour = hour;
3263 }
3264
3265 if ( haveMin )
3266 {
3267 tm.min = min;
3268 }
3269
3270 if ( haveSec )
3271 {
3272 tm.sec = sec;
3273 }
3274
3275 Set(tm);
3276
3277 // finally check that the week day is consistent -- if we had it
3278 if ( haveWDay && GetWeekDay() != wday )
3279 {
3280 wxLogDebug(_T("inconsistsnet week day in wxDateTime::ParseFormat()"));
3281
3282 return NULL;
3283 }
3284
3285 return input;
3286 }
3287
3288 const wxChar *wxDateTime::ParseDateTime(const wxChar *date)
3289 {
3290 wxCHECK_MSG( date, (wxChar *)NULL, _T("NULL pointer in wxDateTime::Parse") );
3291
3292 // Set to current day and hour, so strings like '14:00' becomes today at
3293 // 14, not some other random date
3294 wxDateTime dtDate = wxDateTime::Today();
3295 wxDateTime dtTime = wxDateTime::Today();
3296
3297 const wxChar* pchTime;
3298
3299 // Try to parse the beginning of the string as a date
3300 const wxChar* pchDate = dtDate.ParseDate(date);
3301
3302 // We got a date in the beginning, see if there is a time specified after the date
3303 if ( pchDate )
3304 {
3305 // Skip spaces, as the ParseTime() function fails on spaces
3306 while ( wxIsspace(*pchDate) )
3307 pchDate++;
3308
3309 pchTime = dtTime.ParseTime(pchDate);
3310 }
3311 else // no date in the beginning
3312 {
3313 // check and see if we have a time followed by a date
3314 pchTime = dtTime.ParseTime(date);
3315 if ( pchTime )
3316 {
3317 while ( wxIsspace(*pchTime) )
3318 pchTime++;
3319
3320 pchDate = dtDate.ParseDate(pchTime);
3321 }
3322 }
3323
3324 // If we have a date specified, set our own data to the same date
3325 if ( !pchDate || !pchTime )
3326 return NULL;
3327
3328 Set(dtDate.GetDay(), dtDate.GetMonth(), dtDate.GetYear(),
3329 dtTime.GetHour(), dtTime.GetMinute(), dtTime.GetSecond(),
3330 dtTime.GetMillisecond());
3331
3332 // Return endpoint of scan
3333 return pchDate > pchTime ? pchDate : pchTime;
3334 }
3335
3336 const wxChar *wxDateTime::ParseDate(const wxChar *date)
3337 {
3338 // this is a simplified version of ParseDateTime() which understands only
3339 // "today" (for wxDate compatibility) and digits only otherwise (and not
3340 // all esoteric constructions ParseDateTime() knows about)
3341
3342 wxCHECK_MSG( date, (wxChar *)NULL, _T("NULL pointer in wxDateTime::Parse") );
3343
3344 const wxChar *p = date;
3345 while ( wxIsspace(*p) )
3346 p++;
3347
3348 // some special cases
3349 static struct
3350 {
3351 const wxChar *str;
3352 int dayDiffFromToday;
3353 } literalDates[] =
3354 {
3355 { wxTRANSLATE("today"), 0 },
3356 { wxTRANSLATE("yesterday"), -1 },
3357 { wxTRANSLATE("tomorrow"), 1 },
3358 };
3359
3360 for ( size_t n = 0; n < WXSIZEOF(literalDates); n++ )
3361 {
3362 wxString date = wxGetTranslation(literalDates[n].str);
3363 size_t len = date.length();
3364 if ( wxStrlen(p) >= len && (wxString(p, len).CmpNoCase(date) == 0) )
3365 {
3366 // nothing can follow this, so stop here
3367 p += len;
3368
3369 int dayDiffFromToday = literalDates[n].dayDiffFromToday;
3370 *this = Today();
3371 if ( dayDiffFromToday )
3372 {
3373 *this += wxDateSpan::Days(dayDiffFromToday);
3374 }
3375
3376 return p;
3377 }
3378 }
3379
3380 // We try to guess what we have here: for each new (numeric) token, we
3381 // determine if it can be a month, day or a year. Of course, there is an
3382 // ambiguity as some numbers may be days as well as months, so we also
3383 // have the ability to back track.
3384
3385 // what do we have?
3386 bool haveDay = false, // the months day?
3387 haveWDay = false, // the day of week?
3388 haveMon = false, // the month?
3389 haveYear = false; // the year?
3390
3391 // and the value of the items we have (init them to get rid of warnings)
3392 WeekDay wday = Inv_WeekDay;
3393 wxDateTime_t day = 0;
3394 wxDateTime::Month mon = Inv_Month;
3395 int year = 0;
3396
3397 // tokenize the string
3398 size_t nPosCur = 0;
3399 static const wxChar *dateDelimiters = _T(".,/-\t\r\n ");
3400 wxStringTokenizer tok(p, dateDelimiters);
3401 while ( tok.HasMoreTokens() )
3402 {
3403 wxString token = tok.GetNextToken();
3404 if ( !token )
3405 continue;
3406
3407 // is it a number?
3408 unsigned long val;
3409 if ( token.ToULong(&val) )
3410 {
3411 // guess what this number is
3412
3413 bool isDay = false,
3414 isMonth = false,
3415 isYear = false;
3416
3417 if ( !haveMon && val > 0 && val <= 12 )
3418 {
3419 // assume it is month
3420 isMonth = true;
3421 }
3422 else // not the month
3423 {
3424 if ( haveDay )
3425 {
3426 // this can only be the year
3427 isYear = true;
3428 }
3429 else // may be either day or year
3430 {
3431 wxDateTime_t maxDays = haveMon
3432 ? GetNumOfDaysInMonth(haveYear ? year : Inv_Year, mon)
3433 : 31;
3434
3435 // can it be day?
3436 if ( (val == 0) || (val > (unsigned long)maxDays) )
3437 {
3438 // no
3439 isYear = true;
3440 }
3441 else // yes, suppose it's the day
3442 {
3443 isDay = true;
3444 }
3445 }
3446 }
3447
3448 if ( isYear )
3449 {
3450 if ( haveYear )
3451 break;
3452
3453 haveYear = true;
3454
3455 year = (wxDateTime_t)val;
3456 }
3457 else if ( isDay )
3458 {
3459 if ( haveDay )
3460 break;
3461
3462 haveDay = true;
3463
3464 day = (wxDateTime_t)val;
3465 }
3466 else if ( isMonth )
3467 {
3468 haveMon = true;
3469
3470 mon = (Month)(val - 1);
3471 }
3472 }
3473 else // not a number
3474 {
3475 // be careful not to overwrite the current mon value
3476 Month mon2 = GetMonthFromName(token, Name_Full | Name_Abbr);
3477 if ( mon2 != Inv_Month )
3478 {
3479 // it's a month
3480 if ( haveMon )
3481 {
3482 // but we already have a month - maybe we guessed wrong?
3483 if ( !haveDay )
3484 {
3485 // no need to check in month range as always < 12, but
3486 // the days are counted from 1 unlike the months
3487 day = (wxDateTime_t)mon + 1;
3488 haveDay = true;
3489 }
3490 else
3491 {
3492 // could possible be the year (doesn't the year come
3493 // before the month in the japanese format?) (FIXME)
3494 break;
3495 }
3496 }
3497
3498 mon = mon2;
3499
3500 haveMon = true;
3501 }
3502 else // not a valid month name
3503 {
3504 wday = GetWeekDayFromName(token, Name_Full | Name_Abbr);
3505 if ( wday != Inv_WeekDay )
3506 {
3507 // a week day
3508 if ( haveWDay )
3509 {
3510 break;
3511 }
3512
3513 haveWDay = true;
3514 }
3515 else // not a valid weekday name
3516 {
3517 // try the ordinals
3518 static const wxChar *ordinals[] =
3519 {
3520 wxTRANSLATE("first"),
3521 wxTRANSLATE("second"),
3522 wxTRANSLATE("third"),
3523 wxTRANSLATE("fourth"),
3524 wxTRANSLATE("fifth"),
3525 wxTRANSLATE("sixth"),
3526 wxTRANSLATE("seventh"),
3527 wxTRANSLATE("eighth"),
3528 wxTRANSLATE("ninth"),
3529 wxTRANSLATE("tenth"),
3530 wxTRANSLATE("eleventh"),
3531 wxTRANSLATE("twelfth"),
3532 wxTRANSLATE("thirteenth"),
3533 wxTRANSLATE("fourteenth"),
3534 wxTRANSLATE("fifteenth"),
3535 wxTRANSLATE("sixteenth"),
3536 wxTRANSLATE("seventeenth"),
3537 wxTRANSLATE("eighteenth"),
3538 wxTRANSLATE("nineteenth"),
3539 wxTRANSLATE("twentieth"),
3540 // that's enough - otherwise we'd have problems with
3541 // composite (or not) ordinals
3542 };
3543
3544 size_t n;
3545 for ( n = 0; n < WXSIZEOF(ordinals); n++ )
3546 {
3547 if ( token.CmpNoCase(ordinals[n]) == 0 )
3548 {
3549 break;
3550 }
3551 }
3552
3553 if ( n == WXSIZEOF(ordinals) )
3554 {
3555 // stop here - something unknown
3556 break;
3557 }
3558
3559 // it's a day
3560 if ( haveDay )
3561 {
3562 // don't try anything here (as in case of numeric day
3563 // above) - the symbolic day spec should always
3564 // precede the month/year
3565 break;
3566 }
3567
3568 haveDay = true;
3569
3570 day = (wxDateTime_t)(n + 1);
3571 }
3572 }
3573 }
3574
3575 nPosCur = tok.GetPosition();
3576 }
3577
3578 // either no more tokens or the scan was stopped by something we couldn't
3579 // parse - in any case, see if we can construct a date from what we have
3580 if ( !haveDay && !haveWDay )
3581 {
3582 wxLogDebug(_T("ParseDate: no day, no weekday hence no date."));
3583
3584 return (wxChar *)NULL;
3585 }
3586
3587 if ( haveWDay && (haveMon || haveYear || haveDay) &&
3588 !(haveDay && haveMon && haveYear) )
3589 {
3590 // without adjectives (which we don't support here) the week day only
3591 // makes sense completely separately or with the full date
3592 // specification (what would "Wed 1999" mean?)
3593 return (wxChar *)NULL;
3594 }
3595
3596 if ( !haveWDay && haveYear && !(haveDay && haveMon) )
3597 {
3598 // may be we have month and day instead of day and year?
3599 if ( haveDay && !haveMon )
3600 {
3601 if ( day <= 12 )
3602 {
3603 // exchange day and month
3604 mon = (wxDateTime::Month)(day - 1);
3605
3606 // we're in the current year then
3607 if ( (year > 0) && (year <= (int)GetNumOfDaysInMonth(Inv_Year, mon)) )
3608 {
3609 day = year;
3610
3611 haveMon = true;
3612 haveYear = false;
3613 }
3614 //else: no, can't exchange, leave haveMon == false
3615 }
3616 }
3617
3618 if ( !haveMon )
3619 {
3620 // if we give the year, month and day must be given too
3621 wxLogDebug(_T("ParseDate: day and month should be specified if year is."));
3622
3623 return (wxChar *)NULL;
3624 }
3625 }
3626
3627 if ( !haveMon )
3628 {
3629 mon = GetCurrentMonth();
3630 }
3631
3632 if ( !haveYear )
3633 {
3634 year = GetCurrentYear();
3635 }
3636
3637 if ( haveDay )
3638 {
3639 Set(day, mon, year);
3640
3641 if ( haveWDay )
3642 {
3643 // check that it is really the same
3644 if ( GetWeekDay() != wday )
3645 {
3646 // inconsistency detected
3647 wxLogDebug(_T("ParseDate: inconsistent day/weekday."));
3648
3649 return (wxChar *)NULL;
3650 }
3651 }
3652 }
3653 else // haveWDay
3654 {
3655 *this = Today();
3656
3657 SetToWeekDayInSameWeek(wday);
3658 }
3659
3660 // return the pointer to the first unparsed char
3661 p += nPosCur;
3662 if ( nPosCur && wxStrchr(dateDelimiters, *(p - 1)) )
3663 {
3664 // if we couldn't parse the token after the delimiter, put back the
3665 // delimiter as well
3666 p--;
3667 }
3668
3669 return p;
3670 }
3671
3672 const wxChar *wxDateTime::ParseTime(const wxChar *time)
3673 {
3674 wxCHECK_MSG( time, (wxChar *)NULL, _T("NULL pointer in wxDateTime::Parse") );
3675
3676 // first try some extra things
3677 static const struct
3678 {
3679 const wxChar *name;
3680 wxDateTime_t hour;
3681 } stdTimes[] =
3682 {
3683 { wxTRANSLATE("noon"), 12 },
3684 { wxTRANSLATE("midnight"), 00 },
3685 // anything else?
3686 };
3687
3688 for ( size_t n = 0; n < WXSIZEOF(stdTimes); n++ )
3689 {
3690 wxString timeString = wxGetTranslation(stdTimes[n].name);
3691 size_t len = timeString.length();
3692 if ( timeString.CmpNoCase(wxString(time, len)) == 0 )
3693 {
3694 // casts required by DigitalMars
3695 Set(stdTimes[n].hour, wxDateTime_t(0), wxDateTime_t(0));
3696
3697 return time + len;
3698 }
3699 }
3700
3701 // try all time formats we may think about in the order from longest to
3702 // shortest
3703
3704 // 12hour with AM/PM?
3705 const wxChar *result = ParseFormat(time, _T("%I:%M:%S %p"));
3706
3707 if ( !result )
3708 {
3709 // normally, it's the same, but why not try it?
3710 result = ParseFormat(time, _T("%H:%M:%S"));
3711 }
3712
3713 if ( !result )
3714 {
3715 // 12hour with AM/PM but without seconds?
3716 result = ParseFormat(time, _T("%I:%M %p"));
3717 }
3718
3719 if ( !result )
3720 {
3721 // without seconds?
3722 result = ParseFormat(time, _T("%H:%M"));
3723 }
3724
3725 if ( !result )
3726 {
3727 // just the hour and AM/PM?
3728 result = ParseFormat(time, _T("%I %p"));
3729 }
3730
3731 if ( !result )
3732 {
3733 // just the hour?
3734 result = ParseFormat(time, _T("%H"));
3735 }
3736
3737 if ( !result )
3738 {
3739 // parse the standard format: normally it is one of the formats above
3740 // but it may be set to something completely different by the user
3741 result = ParseFormat(time, _T("%X"));
3742 }
3743
3744 // TODO: parse timezones
3745
3746 return result;
3747 }
3748
3749 // ----------------------------------------------------------------------------
3750 // Workdays and holidays support
3751 // ----------------------------------------------------------------------------
3752
3753 bool wxDateTime::IsWorkDay(Country WXUNUSED(country)) const
3754 {
3755 return !wxDateTimeHolidayAuthority::IsHoliday(*this);
3756 }
3757
3758 // ============================================================================
3759 // wxTimeSpan
3760 // ============================================================================
3761
3762 // this enum is only used in wxTimeSpan::Format() below but we can't declare
3763 // it locally to the method as it provokes an internal compiler error in egcs
3764 // 2.91.60 when building with -O2
3765 enum TimeSpanPart
3766 {
3767 Part_Week,
3768 Part_Day,
3769 Part_Hour,
3770 Part_Min,
3771 Part_Sec,
3772 Part_MSec
3773 };
3774
3775 // not all strftime(3) format specifiers make sense here because, for example,
3776 // a time span doesn't have a year nor a timezone
3777 //
3778 // Here are the ones which are supported (all of them are supported by strftime
3779 // as well):
3780 // %H hour in 24 hour format
3781 // %M minute (00 - 59)
3782 // %S second (00 - 59)
3783 // %% percent sign
3784 //
3785 // Also, for MFC CTimeSpan compatibility, we support
3786 // %D number of days
3787 //
3788 // And, to be better than MFC :-), we also have
3789 // %E number of wEeks
3790 // %l milliseconds (000 - 999)
3791 wxString wxTimeSpan::Format(const wxChar *format) const
3792 {
3793 wxCHECK_MSG( format, _T(""), _T("NULL format in wxTimeSpan::Format") );
3794
3795 wxString str;
3796 str.Alloc(wxStrlen(format));
3797
3798 // Suppose we have wxTimeSpan ts(1 /* hour */, 2 /* min */, 3 /* sec */)
3799 //
3800 // Then, of course, ts.Format("%H:%M:%S") must return "01:02:03", but the
3801 // question is what should ts.Format("%S") do? The code here returns "3273"
3802 // in this case (i.e. the total number of seconds, not just seconds % 60)
3803 // because, for me, this call means "give me entire time interval in
3804 // seconds" and not "give me the seconds part of the time interval"
3805 //
3806 // If we agree that it should behave like this, it is clear that the
3807 // interpretation of each format specifier depends on the presence of the
3808 // other format specs in the string: if there was "%H" before "%M", we
3809 // should use GetMinutes() % 60, otherwise just GetMinutes() &c
3810
3811 // we remember the most important unit found so far
3812 TimeSpanPart partBiggest = Part_MSec;
3813
3814 for ( const wxChar *pch = format; *pch; pch++ )
3815 {
3816 wxChar ch = *pch;
3817
3818 if ( ch == _T('%') )
3819 {
3820 // the start of the format specification of the printf() below
3821 wxString fmtPrefix = _T('%');
3822
3823 // the number
3824 long n;
3825
3826 ch = *++pch; // get the format spec char
3827 switch ( ch )
3828 {
3829 default:
3830 wxFAIL_MSG( _T("invalid format character") );
3831 // fall through
3832
3833 case _T('%'):
3834 str += ch;
3835
3836 // skip the part below switch
3837 continue;
3838
3839 case _T('D'):
3840 n = GetDays();
3841 if ( partBiggest < Part_Day )
3842 {
3843 n %= DAYS_PER_WEEK;
3844 }
3845 else
3846 {
3847 partBiggest = Part_Day;
3848 }
3849 break;
3850
3851 case _T('E'):
3852 partBiggest = Part_Week;
3853 n = GetWeeks();
3854 break;
3855
3856 case _T('H'):
3857 n = GetHours();
3858 if ( partBiggest < Part_Hour )
3859 {
3860 n %= HOURS_PER_DAY;
3861 }
3862 else
3863 {
3864 partBiggest = Part_Hour;
3865 }
3866
3867 fmtPrefix += _T("02");
3868 break;
3869
3870 case _T('l'):
3871 n = GetMilliseconds().ToLong();
3872 if ( partBiggest < Part_MSec )
3873 {
3874 n %= 1000;
3875 }
3876 //else: no need to reset partBiggest to Part_MSec, it is
3877 // the least significant one anyhow
3878
3879 fmtPrefix += _T("03");
3880 break;
3881
3882 case _T('M'):
3883 n = GetMinutes();
3884 if ( partBiggest < Part_Min )
3885 {
3886 n %= MIN_PER_HOUR;
3887 }
3888 else
3889 {
3890 partBiggest = Part_Min;
3891 }
3892
3893 fmtPrefix += _T("02");
3894 break;
3895
3896 case _T('S'):
3897 n = GetSeconds().ToLong();
3898 if ( partBiggest < Part_Sec )
3899 {
3900 n %= SEC_PER_MIN;
3901 }
3902 else
3903 {
3904 partBiggest = Part_Sec;
3905 }
3906
3907 fmtPrefix += _T("02");
3908 break;
3909 }
3910
3911 str += wxString::Format(fmtPrefix + _T("ld"), n);
3912 }
3913 else
3914 {
3915 // normal character, just copy
3916 str += ch;
3917 }
3918 }
3919
3920 return str;
3921 }
3922
3923 // ============================================================================
3924 // wxDateTimeHolidayAuthority and related classes
3925 // ============================================================================
3926
3927 #include "wx/arrimpl.cpp"
3928
3929 WX_DEFINE_OBJARRAY(wxDateTimeArray);
3930
3931 static int wxCMPFUNC_CONV
3932 wxDateTimeCompareFunc(wxDateTime **first, wxDateTime **second)
3933 {
3934 wxDateTime dt1 = **first,
3935 dt2 = **second;
3936
3937 return dt1 == dt2 ? 0 : dt1 < dt2 ? -1 : +1;
3938 }
3939
3940 // ----------------------------------------------------------------------------
3941 // wxDateTimeHolidayAuthority
3942 // ----------------------------------------------------------------------------
3943
3944 wxHolidayAuthoritiesArray wxDateTimeHolidayAuthority::ms_authorities;
3945
3946 /* static */
3947 bool wxDateTimeHolidayAuthority::IsHoliday(const wxDateTime& dt)
3948 {
3949 size_t count = ms_authorities.size();
3950 for ( size_t n = 0; n < count; n++ )
3951 {
3952 if ( ms_authorities[n]->DoIsHoliday(dt) )
3953 {
3954 return true;
3955 }
3956 }
3957
3958 return false;
3959 }
3960
3961 /* static */
3962 size_t
3963 wxDateTimeHolidayAuthority::GetHolidaysInRange(const wxDateTime& dtStart,
3964 const wxDateTime& dtEnd,
3965 wxDateTimeArray& holidays)
3966 {
3967 wxDateTimeArray hol;
3968
3969 holidays.Clear();
3970
3971 size_t count = ms_authorities.size();
3972 for ( size_t nAuth = 0; nAuth < count; nAuth++ )
3973 {
3974 ms_authorities[nAuth]->DoGetHolidaysInRange(dtStart, dtEnd, hol);
3975
3976 WX_APPEND_ARRAY(holidays, hol);
3977 }
3978
3979 holidays.Sort(wxDateTimeCompareFunc);
3980
3981 return holidays.size();
3982 }
3983
3984 /* static */
3985 void wxDateTimeHolidayAuthority::ClearAllAuthorities()
3986 {
3987 WX_CLEAR_ARRAY(ms_authorities);
3988 }
3989
3990 /* static */
3991 void wxDateTimeHolidayAuthority::AddAuthority(wxDateTimeHolidayAuthority *auth)
3992 {
3993 ms_authorities.push_back(auth);
3994 }
3995
3996 wxDateTimeHolidayAuthority::~wxDateTimeHolidayAuthority()
3997 {
3998 // nothing to do here
3999 }
4000
4001 // ----------------------------------------------------------------------------
4002 // wxDateTimeWorkDays
4003 // ----------------------------------------------------------------------------
4004
4005 bool wxDateTimeWorkDays::DoIsHoliday(const wxDateTime& dt) const
4006 {
4007 wxDateTime::WeekDay wd = dt.GetWeekDay();
4008
4009 return (wd == wxDateTime::Sun) || (wd == wxDateTime::Sat);
4010 }
4011
4012 size_t wxDateTimeWorkDays::DoGetHolidaysInRange(const wxDateTime& dtStart,
4013 const wxDateTime& dtEnd,
4014 wxDateTimeArray& holidays) const
4015 {
4016 if ( dtStart > dtEnd )
4017 {
4018 wxFAIL_MSG( _T("invalid date range in GetHolidaysInRange") );
4019
4020 return 0u;
4021 }
4022
4023 holidays.Empty();
4024
4025 // instead of checking all days, start with the first Sat after dtStart and
4026 // end with the last Sun before dtEnd
4027 wxDateTime dtSatFirst = dtStart.GetNextWeekDay(wxDateTime::Sat),
4028 dtSatLast = dtEnd.GetPrevWeekDay(wxDateTime::Sat),
4029 dtSunFirst = dtStart.GetNextWeekDay(wxDateTime::Sun),
4030 dtSunLast = dtEnd.GetPrevWeekDay(wxDateTime::Sun),
4031 dt;
4032
4033 for ( dt = dtSatFirst; dt <= dtSatLast; dt += wxDateSpan::Week() )
4034 {
4035 holidays.Add(dt);
4036 }
4037
4038 for ( dt = dtSunFirst; dt <= dtSunLast; dt += wxDateSpan::Week() )
4039 {
4040 holidays.Add(dt);
4041 }
4042
4043 return holidays.GetCount();
4044 }
4045
4046 #endif // wxUSE_DATETIME