1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/datetime.cpp
3 // Purpose: implementation of time/date related classes
4 // Author: Vadim Zeitlin
8 // Copyright: (c) 1999 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9 // parts of code taken from sndcal library by Scott E. Lee:
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.
16 // Licence: wxWindows licence
17 ///////////////////////////////////////////////////////////////////////////////
20 * Implementation notes:
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
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
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
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
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
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
47 // ============================================================================
49 // ============================================================================
51 // ----------------------------------------------------------------------------
53 // ----------------------------------------------------------------------------
55 // For compilers that support precompilation, includes "wx.h".
56 #include "wx/wxprec.h"
62 #if !defined(wxUSE_DATETIME) || wxUSE_DATETIME
66 #include "wx/msw/wrapwin.h"
68 #include "wx/string.h"
71 #include "wx/stopwatch.h" // for wxGetLocalTimeMillis()
72 #include "wx/module.h"
76 #include "wx/thread.h"
77 #include "wx/tokenzr.h"
88 #include "wx/datetime.h"
90 const long wxDateTime::TIME_T_FACTOR
= 1000l;
92 #if wxUSE_EXTENDED_RTTI
94 template<> void wxStringReadValue(const wxString
&s
, wxDateTime
&data
)
96 data
.ParseFormat(s
,"%Y-%m-%d %H:%M:%S", NULL
);
99 template<> void wxStringWriteValue(wxString
&s
, const wxDateTime
&data
)
101 s
= data
.Format("%Y-%m-%d %H:%M:%S");
104 wxCUSTOM_TYPE_INFO(wxDateTime
, wxToStringConverter
<wxDateTime
> , wxFromStringConverter
<wxDateTime
>)
106 #endif // wxUSE_EXTENDED_RTTI
109 // ----------------------------------------------------------------------------
110 // conditional compilation
111 // ----------------------------------------------------------------------------
113 #if defined(HAVE_STRPTIME) && defined(__GLIBC__) && \
114 ((__GLIBC__ == 2) && (__GLIBC_MINOR__ == 0))
115 // glibc 2.0.7 strptime() is broken - the following snippet causes it to
116 // crash (instead of just failing):
118 // strncpy(buf, "Tue Dec 21 20:25:40 1999", 128);
119 // strptime(buf, "%x", &tm);
123 #endif // broken strptime()
125 #if defined(HAVE_STRPTIME) && defined(__DARWIN__) && defined(_MSL_USING_MW_C_HEADERS) && _MSL_USING_MW_C_HEADERS
126 // configure detects strptime as linkable because it's in the OS X
127 // System library but MSL headers don't declare it.
129 // char *strptime(const char *, const char *, struct tm *);
130 // However, we DON'T want to just provide it here because we would
131 // crash and/or overwrite data when strptime from OS X tries
132 // to fill in MW's struct tm which is two fields shorter (no TZ stuff)
133 // So for now let's just say we don't have strptime
137 #if defined(__MWERKS__) && wxUSE_UNICODE
141 #if !defined(WX_TIMEZONE) && !defined(WX_GMTOFF_IN_TM)
142 #if defined(__WXPALMOS__)
143 #define WX_GMTOFF_IN_TM
144 #elif defined(__BORLANDC__) || defined(__MINGW32__) || defined(__VISAGECPP__)
145 #define WX_TIMEZONE _timezone
146 #elif defined(__MWERKS__)
147 long wxmw_timezone
= 28800;
148 #define WX_TIMEZONE wxmw_timezone
149 #elif defined(__DJGPP__) || defined(__WINE__)
150 #include <sys/timeb.h>
152 static long wxGetTimeZone()
154 static long timezone
= MAXLONG
; // invalid timezone
155 if (timezone
== MAXLONG
)
159 timezone
= tb
.timezone
;
163 #define WX_TIMEZONE wxGetTimeZone()
164 #elif defined(__DARWIN__)
165 #define WX_GMTOFF_IN_TM
166 #elif defined(__WXWINCE__) && defined(__VISUALC8__)
167 // _timezone is not present in dynamic run-time library
169 // Solution (1): use the function equivalent of _timezone
170 static long wxGetTimeZone()
172 static long s_Timezone
= MAXLONG
; // invalid timezone
173 if (s_Timezone
== MAXLONG
)
177 s_Timezone
= (long) t
;
181 #define WX_TIMEZONE wxGetTimeZone()
183 // Solution (2): using GetTimeZoneInformation
184 static long wxGetTimeZone()
186 static long timezone
= MAXLONG
; // invalid timezone
187 if (timezone
== MAXLONG
)
189 TIME_ZONE_INFORMATION tzi
;
190 ::GetTimeZoneInformation(&tzi
);
195 #define WX_TIMEZONE wxGetTimeZone()
197 // Old method using _timezone: this symbol doesn't exist in the dynamic run-time library (i.e. using /MD)
198 #define WX_TIMEZONE _timezone
200 #else // unknown platform - try timezone
201 #define WX_TIMEZONE timezone
203 #endif // !WX_TIMEZONE && !WX_GMTOFF_IN_TM
205 // everyone has strftime except Win CE unless VC8 is used
206 #if !defined(__WXWINCE__) || defined(__VISUALC8__)
207 #define HAVE_STRFTIME
210 // NB: VC8 safe time functions could/should be used for wxMSW as well probably
211 #if defined(__WXWINCE__) && defined(__VISUALC8__)
213 struct tm
*wxLocaltime_r(const time_t *t
, struct tm
* tm
)
216 return _localtime64_s(tm
, &t64
) == 0 ? tm
: NULL
;
219 struct tm
*wxGmtime_r(const time_t* t
, struct tm
* tm
)
222 return _gmtime64_s(tm
, &t64
) == 0 ? tm
: NULL
;
225 #else // !wxWinCE with VC8
227 #if (!defined(HAVE_LOCALTIME_R) || !defined(HAVE_GMTIME_R)) && wxUSE_THREADS && !defined(__WINDOWS__)
228 static wxMutex timeLock
;
231 #ifndef HAVE_LOCALTIME_R
232 struct tm
*wxLocaltime_r(const time_t* ticks
, struct tm
* temp
)
234 #if wxUSE_THREADS && !defined(__WINDOWS__)
235 // No need to waste time with a mutex on windows since it's using
236 // thread local storage for localtime anyway.
237 wxMutexLocker
locker(timeLock
);
240 // Borland CRT crashes when passed 0 ticks for some reason, see SF bug 1704438
246 const tm
* const t
= localtime(ticks
);
250 memcpy(temp
, t
, sizeof(struct tm
));
253 #endif // !HAVE_LOCALTIME_R
255 #ifndef HAVE_GMTIME_R
256 struct tm
*wxGmtime_r(const time_t* ticks
, struct tm
* temp
)
258 #if wxUSE_THREADS && !defined(__WINDOWS__)
259 // No need to waste time with a mutex on windows since it's
260 // using thread local storage for gmtime anyway.
261 wxMutexLocker
locker(timeLock
);
269 const tm
* const t
= gmtime(ticks
);
273 memcpy(temp
, gmtime(ticks
), sizeof(struct tm
));
276 #endif // !HAVE_GMTIME_R
278 #endif // wxWinCE with VC8/other platforms
280 // ----------------------------------------------------------------------------
282 // ----------------------------------------------------------------------------
284 // debugging helper: just a convenient replacement of wxCHECK()
285 #define wxDATETIME_CHECK(expr, msg) \
286 wxCHECK2_MSG(expr, *this = wxInvalidDateTime; return *this, msg)
288 // ----------------------------------------------------------------------------
290 // ----------------------------------------------------------------------------
292 class wxDateTimeHolidaysModule
: public wxModule
295 virtual bool OnInit()
297 wxDateTimeHolidayAuthority::AddAuthority(new wxDateTimeWorkDays
);
302 virtual void OnExit()
304 wxDateTimeHolidayAuthority::ClearAllAuthorities();
305 wxDateTimeHolidayAuthority::ms_authorities
.clear();
309 DECLARE_DYNAMIC_CLASS(wxDateTimeHolidaysModule
)
312 IMPLEMENT_DYNAMIC_CLASS(wxDateTimeHolidaysModule
, wxModule
)
314 // ----------------------------------------------------------------------------
316 // ----------------------------------------------------------------------------
319 static const int MONTHS_IN_YEAR
= 12;
321 static const int SEC_PER_MIN
= 60;
323 static const int MIN_PER_HOUR
= 60;
325 static const int HOURS_PER_DAY
= 24;
327 static const long SECONDS_PER_DAY
= 86400l;
329 static const int DAYS_PER_WEEK
= 7;
331 static const long MILLISECONDS_PER_DAY
= 86400000l;
333 // this is the integral part of JDN of the midnight of Jan 1, 1970
334 // (i.e. JDN(Jan 1, 1970) = 2440587.5)
335 static const long EPOCH_JDN
= 2440587l;
337 // used only in asserts
339 // the date of JDN -0.5 (as we don't work with fractional parts, this is the
340 // reference date for us) is Nov 24, 4714BC
341 static const int JDN_0_YEAR
= -4713;
342 static const int JDN_0_MONTH
= wxDateTime::Nov
;
343 static const int JDN_0_DAY
= 24;
344 #endif // __WXDEBUG__
346 // the constants used for JDN calculations
347 static const long JDN_OFFSET
= 32046l;
348 static const long DAYS_PER_5_MONTHS
= 153l;
349 static const long DAYS_PER_4_YEARS
= 1461l;
350 static const long DAYS_PER_400_YEARS
= 146097l;
352 // this array contains the cumulated number of days in all previous months for
353 // normal and leap years
354 static const wxDateTime::wxDateTime_t gs_cumulatedDays
[2][MONTHS_IN_YEAR
] =
356 { 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334 },
357 { 0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335 }
360 // ----------------------------------------------------------------------------
362 // ----------------------------------------------------------------------------
364 const char *wxDefaultDateTimeFormat
= "%c";
365 const char *wxDefaultTimeSpanFormat
= "%H:%M:%S";
367 // in the fine tradition of ANSI C we use our equivalent of (time_t)-1 to
368 // indicate an invalid wxDateTime object
369 const wxDateTime wxDefaultDateTime
;
371 wxDateTime::Country
wxDateTime::ms_country
= wxDateTime::Country_Unknown
;
373 // ----------------------------------------------------------------------------
375 // ----------------------------------------------------------------------------
377 // debugger helper: shows what the date really is
379 extern const char *wxDumpDate(const wxDateTime
* dt
)
381 static char buf
[128];
383 wxString
fmt(dt
->Format("%Y-%m-%d (%a) %H:%M:%S"));
384 wxStrncpy(buf
, fmt
+ " (" + dt
->GetValue().ToString() + " ticks)",
391 // get the number of days in the given month of the given year
393 wxDateTime::wxDateTime_t
GetNumOfDaysInMonth(int year
, wxDateTime::Month month
)
395 // the number of days in month in Julian/Gregorian calendar: the first line
396 // is for normal years, the second one is for the leap ones
397 static wxDateTime::wxDateTime_t daysInMonth
[2][MONTHS_IN_YEAR
] =
399 { 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 },
400 { 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 }
403 return daysInMonth
[wxDateTime::IsLeapYear(year
)][month
];
406 // returns the time zone in the C sense, i.e. the difference UTC - local
408 static int GetTimeZone()
410 // set to true when the timezone is set
411 static bool s_timezoneSet
= false;
412 static long gmtoffset
= LONG_MAX
; // invalid timezone
414 // ensure that the timezone variable is set by calling wxLocaltime_r
415 if ( !s_timezoneSet
)
417 // just call wxLocaltime_r() instead of figuring out whether this
418 // system supports tzset(), _tzset() or something else
422 wxLocaltime_r(&t
, &tm
);
423 s_timezoneSet
= true;
425 #ifdef WX_GMTOFF_IN_TM
426 // note that GMT offset is the opposite of time zone and so to return
427 // consistent results in both WX_GMTOFF_IN_TM and !WX_GMTOFF_IN_TM
428 // cases we have to negate it
429 gmtoffset
= -tm
.tm_gmtoff
;
430 #else // !WX_GMTOFF_IN_TM
431 gmtoffset
= WX_TIMEZONE
;
432 #endif // WX_GMTOFF_IN_TM/!WX_GMTOFF_IN_TM
435 return (int)gmtoffset
;
438 // return the integral part of the JDN for the midnight of the given date (to
439 // get the real JDN you need to add 0.5, this is, in fact, JDN of the
440 // noon of the previous day)
441 static long GetTruncatedJDN(wxDateTime::wxDateTime_t day
,
442 wxDateTime::Month mon
,
445 // CREDIT: code below is by Scott E. Lee (but bugs are mine)
447 // check the date validity
449 (year
> JDN_0_YEAR
) ||
450 ((year
== JDN_0_YEAR
) && (mon
> JDN_0_MONTH
)) ||
451 ((year
== JDN_0_YEAR
) && (mon
== JDN_0_MONTH
) && (day
>= JDN_0_DAY
)),
452 _T("date out of range - can't convert to JDN")
455 // make the year positive to avoid problems with negative numbers division
458 // months are counted from March here
460 if ( mon
>= wxDateTime::Mar
)
470 // now we can simply add all the contributions together
471 return ((year
/ 100) * DAYS_PER_400_YEARS
) / 4
472 + ((year
% 100) * DAYS_PER_4_YEARS
) / 4
473 + (month
* DAYS_PER_5_MONTHS
+ 2) / 5
480 // this function is a wrapper around strftime(3) adding error checking
481 static wxString
CallStrftime(const wxString
& format
, const tm
* tm
)
484 // Create temp wxString here to work around mingw/cygwin bug 1046059
485 // http://sourceforge.net/tracker/?func=detail&atid=102435&aid=1046059&group_id=2435
488 if ( !wxStrftime(buf
, WXSIZEOF(buf
), format
, tm
) )
490 // if the format is valid, buffer must be too small?
491 wxFAIL_MSG(_T("strftime() failed"));
500 #endif // HAVE_STRFTIME
504 #if wxUSE_UNIX && !defined(HAVE_STRPTIME_DECL)
505 // configure detected that we had strptime() but not its declaration,
506 // provide it ourselves
507 extern "C" char *strptime(const char *, const char *, struct tm
*);
510 // Unicode-friendly strptime() wrapper
511 static const wxStringCharType
*
512 CallStrptime(const wxStringCharType
*input
, const char *fmt
, tm
*tm
)
514 // the problem here is that strptime() returns pointer into the string we
515 // passed to it while we're really interested in the pointer into the
516 // original, Unicode, string so we try to transform the pointer back
517 #if wxUSE_UNICODE_WCHAR
518 wxCharBuffer
inputMB(wxConvertWX2MB(input
));
520 const char * const inputMB
= input
;
521 #endif // Unicode/Ascii
523 const char *result
= strptime(inputMB
, fmt
, tm
);
527 #if wxUSE_UNICODE_WCHAR
528 // FIXME: this is wrong in presence of surrogates &c
529 return input
+ (result
- inputMB
.data());
532 #endif // Unicode/Ascii
535 #endif // HAVE_STRPTIME
537 // if year and/or month have invalid values, replace them with the current ones
538 static void ReplaceDefaultYearMonthWithCurrent(int *year
,
539 wxDateTime::Month
*month
)
541 struct tm
*tmNow
= NULL
;
544 if ( *year
== wxDateTime::Inv_Year
)
546 tmNow
= wxDateTime::GetTmNow(&tmstruct
);
548 *year
= 1900 + tmNow
->tm_year
;
551 if ( *month
== wxDateTime::Inv_Month
)
554 tmNow
= wxDateTime::GetTmNow(&tmstruct
);
556 *month
= (wxDateTime::Month
)tmNow
->tm_mon
;
560 // fll the struct tm with default values
561 static void InitTm(struct tm
& tm
)
563 // struct tm may have etxra fields (undocumented and with unportable
564 // names) which, nevertheless, must be set to 0
565 memset(&tm
, 0, sizeof(struct tm
));
567 tm
.tm_mday
= 1; // mday 0 is invalid
568 tm
.tm_year
= 76; // any valid year
569 tm
.tm_isdst
= -1; // auto determine
575 // return the month if the string is a month name or Inv_Month otherwise
576 static wxDateTime::Month
GetMonthFromName(const wxString
& name
, int flags
)
578 wxDateTime::Month mon
;
579 for ( mon
= wxDateTime::Jan
; mon
< wxDateTime::Inv_Month
; wxNextMonth(mon
) )
581 // case-insensitive comparison either one of or with both abbreviated
583 if ( flags
& wxDateTime::Name_Full
)
585 if ( name
.CmpNoCase(wxDateTime::
586 GetMonthName(mon
, wxDateTime::Name_Full
)) == 0 )
592 if ( flags
& wxDateTime::Name_Abbr
)
594 if ( name
.CmpNoCase(wxDateTime::
595 GetMonthName(mon
, wxDateTime::Name_Abbr
)) == 0 )
605 // return the weekday if the string is a weekday name or Inv_WeekDay otherwise
606 static wxDateTime::WeekDay
GetWeekDayFromName(const wxString
& name
, int flags
)
608 wxDateTime::WeekDay wd
;
609 for ( wd
= wxDateTime::Sun
; wd
< wxDateTime::Inv_WeekDay
; wxNextWDay(wd
) )
611 // case-insensitive comparison either one of or with both abbreviated
613 if ( flags
& wxDateTime::Name_Full
)
615 if ( name
.CmpNoCase(wxDateTime::
616 GetWeekDayName(wd
, wxDateTime::Name_Full
)) == 0 )
622 if ( flags
& wxDateTime::Name_Abbr
)
624 if ( name
.CmpNoCase(wxDateTime::
625 GetWeekDayName(wd
, wxDateTime::Name_Abbr
)) == 0 )
636 struct tm
*wxDateTime::GetTmNow(struct tm
*tmstruct
)
638 time_t t
= GetTimeNow();
639 return wxLocaltime_r(&t
, tmstruct
);
642 // scans all digits (but no more than len) and returns the resulting number
643 static bool GetNumericToken(size_t len
,
644 const wxStringCharType
*& p
,
645 unsigned long *number
)
649 while ( wxIsdigit(*p
) )
653 if ( len
&& ++n
> len
)
657 return !s
.empty() && s
.ToULong(number
);
660 // scans all alphabetic characters and returns the resulting string
661 static wxString
GetAlphaToken(const wxStringCharType
*& p
)
664 while ( wxIsalpha(*p
) )
672 // ============================================================================
673 // implementation of wxDateTime
674 // ============================================================================
676 // ----------------------------------------------------------------------------
678 // ----------------------------------------------------------------------------
682 year
= (wxDateTime_t
)wxDateTime::Inv_Year
;
683 mon
= wxDateTime::Inv_Month
;
685 hour
= min
= sec
= msec
= 0;
686 wday
= wxDateTime::Inv_WeekDay
;
689 wxDateTime::Tm::Tm(const struct tm
& tm
, const TimeZone
& tz
)
693 sec
= (wxDateTime::wxDateTime_t
)tm
.tm_sec
;
694 min
= (wxDateTime::wxDateTime_t
)tm
.tm_min
;
695 hour
= (wxDateTime::wxDateTime_t
)tm
.tm_hour
;
696 mday
= (wxDateTime::wxDateTime_t
)tm
.tm_mday
;
697 mon
= (wxDateTime::Month
)tm
.tm_mon
;
698 year
= 1900 + tm
.tm_year
;
699 wday
= (wxDateTime::wxDateTime_t
)tm
.tm_wday
;
700 yday
= (wxDateTime::wxDateTime_t
)tm
.tm_yday
;
703 bool wxDateTime::Tm::IsValid() const
705 // we allow for the leap seconds, although we don't use them (yet)
706 return (year
!= wxDateTime::Inv_Year
) && (mon
!= wxDateTime::Inv_Month
) &&
707 (mday
<= GetNumOfDaysInMonth(year
, mon
)) &&
708 (hour
< 24) && (min
< 60) && (sec
< 62) && (msec
< 1000);
711 void wxDateTime::Tm::ComputeWeekDay()
713 // compute the week day from day/month/year: we use the dumbest algorithm
714 // possible: just compute our JDN and then use the (simple to derive)
715 // formula: weekday = (JDN + 1.5) % 7
716 wday
= (wxDateTime::wxDateTime_t
)((GetTruncatedJDN(mday
, mon
, year
) + 2) % 7);
719 void wxDateTime::Tm::AddMonths(int monDiff
)
721 // normalize the months field
722 while ( monDiff
< -mon
)
726 monDiff
+= MONTHS_IN_YEAR
;
729 while ( monDiff
+ mon
>= MONTHS_IN_YEAR
)
733 monDiff
-= MONTHS_IN_YEAR
;
736 mon
= (wxDateTime::Month
)(mon
+ monDiff
);
738 wxASSERT_MSG( mon
>= 0 && mon
< MONTHS_IN_YEAR
, _T("logic error") );
740 // NB: we don't check here that the resulting date is valid, this function
741 // is private and the caller must check it if needed
744 void wxDateTime::Tm::AddDays(int dayDiff
)
746 // normalize the days field
747 while ( dayDiff
+ mday
< 1 )
751 dayDiff
+= GetNumOfDaysInMonth(year
, mon
);
754 mday
= (wxDateTime::wxDateTime_t
)( mday
+ dayDiff
);
755 while ( mday
> GetNumOfDaysInMonth(year
, mon
) )
757 mday
-= GetNumOfDaysInMonth(year
, mon
);
762 wxASSERT_MSG( mday
> 0 && mday
<= GetNumOfDaysInMonth(year
, mon
),
766 // ----------------------------------------------------------------------------
768 // ----------------------------------------------------------------------------
770 wxDateTime::TimeZone::TimeZone(wxDateTime::TZ tz
)
774 case wxDateTime::Local
:
775 // get the offset from C RTL: it returns the difference GMT-local
776 // while we want to have the offset _from_ GMT, hence the '-'
777 m_offset
= -GetTimeZone();
780 case wxDateTime::GMT_12
:
781 case wxDateTime::GMT_11
:
782 case wxDateTime::GMT_10
:
783 case wxDateTime::GMT_9
:
784 case wxDateTime::GMT_8
:
785 case wxDateTime::GMT_7
:
786 case wxDateTime::GMT_6
:
787 case wxDateTime::GMT_5
:
788 case wxDateTime::GMT_4
:
789 case wxDateTime::GMT_3
:
790 case wxDateTime::GMT_2
:
791 case wxDateTime::GMT_1
:
792 m_offset
= -3600*(wxDateTime::GMT0
- tz
);
795 case wxDateTime::GMT0
:
796 case wxDateTime::GMT1
:
797 case wxDateTime::GMT2
:
798 case wxDateTime::GMT3
:
799 case wxDateTime::GMT4
:
800 case wxDateTime::GMT5
:
801 case wxDateTime::GMT6
:
802 case wxDateTime::GMT7
:
803 case wxDateTime::GMT8
:
804 case wxDateTime::GMT9
:
805 case wxDateTime::GMT10
:
806 case wxDateTime::GMT11
:
807 case wxDateTime::GMT12
:
808 case wxDateTime::GMT13
:
809 m_offset
= 3600*(tz
- wxDateTime::GMT0
);
812 case wxDateTime::A_CST
:
813 // Central Standard Time in use in Australia = UTC + 9.5
814 m_offset
= 60l*(9*MIN_PER_HOUR
+ MIN_PER_HOUR
/2);
818 wxFAIL_MSG( _T("unknown time zone") );
822 // ----------------------------------------------------------------------------
824 // ----------------------------------------------------------------------------
827 bool wxDateTime::IsLeapYear(int year
, wxDateTime::Calendar cal
)
829 if ( year
== Inv_Year
)
830 year
= GetCurrentYear();
832 if ( cal
== Gregorian
)
834 // in Gregorian calendar leap years are those divisible by 4 except
835 // those divisible by 100 unless they're also divisible by 400
836 // (in some countries, like Russia and Greece, additional corrections
837 // exist, but they won't manifest themselves until 2700)
838 return (year
% 4 == 0) && ((year
% 100 != 0) || (year
% 400 == 0));
840 else if ( cal
== Julian
)
842 // in Julian calendar the rule is simpler
843 return year
% 4 == 0;
847 wxFAIL_MSG(_T("unknown calendar"));
854 int wxDateTime::GetCentury(int year
)
856 return year
> 0 ? year
/ 100 : year
/ 100 - 1;
860 int wxDateTime::ConvertYearToBC(int year
)
863 return year
> 0 ? year
: year
- 1;
867 int wxDateTime::GetCurrentYear(wxDateTime::Calendar cal
)
872 return Now().GetYear();
875 wxFAIL_MSG(_T("TODO"));
879 wxFAIL_MSG(_T("unsupported calendar"));
887 wxDateTime::Month
wxDateTime::GetCurrentMonth(wxDateTime::Calendar cal
)
892 return Now().GetMonth();
895 wxFAIL_MSG(_T("TODO"));
899 wxFAIL_MSG(_T("unsupported calendar"));
907 wxDateTime::wxDateTime_t
wxDateTime::GetNumberOfDays(int year
, Calendar cal
)
909 if ( year
== Inv_Year
)
911 // take the current year if none given
912 year
= GetCurrentYear();
919 return IsLeapYear(year
) ? 366 : 365;
922 wxFAIL_MSG(_T("unsupported calendar"));
930 wxDateTime::wxDateTime_t
wxDateTime::GetNumberOfDays(wxDateTime::Month month
,
932 wxDateTime::Calendar cal
)
934 wxCHECK_MSG( month
< MONTHS_IN_YEAR
, 0, _T("invalid month") );
936 if ( cal
== Gregorian
|| cal
== Julian
)
938 if ( year
== Inv_Year
)
940 // take the current year if none given
941 year
= GetCurrentYear();
944 return GetNumOfDaysInMonth(year
, month
);
948 wxFAIL_MSG(_T("unsupported calendar"));
955 wxString
wxDateTime::GetMonthName(wxDateTime::Month month
,
956 wxDateTime::NameFlags flags
)
958 wxCHECK_MSG( month
!= Inv_Month
, wxEmptyString
, _T("invalid month") );
960 // notice that we must set all the fields to avoid confusing libc (GNU one
961 // gets confused to a crash if we don't do this)
966 return CallStrftime(flags
== Name_Abbr
? _T("%b") : _T("%B"), &tm
);
967 #else // !HAVE_STRFTIME
972 ret
= (flags
== Name_Abbr
? wxT("Jan"): wxT("January"));
975 ret
= (flags
== Name_Abbr
? wxT("Feb"): wxT("Febuary"));
978 ret
= (flags
== Name_Abbr
? wxT("Mar"): wxT("March"));
981 ret
= (flags
== Name_Abbr
? wxT("Apr"): wxT("April"));
984 ret
= (flags
== Name_Abbr
? wxT("May"): wxT("May"));
987 ret
= (flags
== Name_Abbr
? wxT("Jun"): wxT("June"));
990 ret
= (flags
== Name_Abbr
? wxT("Jul"): wxT("July"));
993 ret
= (flags
== Name_Abbr
? wxT("Aug"): wxT("August"));
996 ret
= (flags
== Name_Abbr
? wxT("Sep"): wxT("September"));
999 ret
= (flags
== Name_Abbr
? wxT("Oct"): wxT("October"));
1002 ret
= (flags
== Name_Abbr
? wxT("Nov"): wxT("November"));
1005 ret
= (flags
== Name_Abbr
? wxT("Dec"): wxT("December"));
1009 #endif // HAVE_STRFTIME/!HAVE_STRFTIME
1013 wxString
wxDateTime::GetWeekDayName(wxDateTime::WeekDay wday
,
1014 wxDateTime::NameFlags flags
)
1016 wxCHECK_MSG( wday
!= Inv_WeekDay
, wxEmptyString
, _T("invalid weekday") );
1017 #ifdef HAVE_STRFTIME
1018 // take some arbitrary Sunday (but notice that the day should be such that
1019 // after adding wday to it below we still have a valid date, e.g. don't
1027 // and offset it by the number of days needed to get the correct wday
1030 // call mktime() to normalize it...
1033 // ... and call strftime()
1034 return CallStrftime(flags
== Name_Abbr
? _T("%a") : _T("%A"), &tm
);
1035 #else // !HAVE_STRFTIME
1040 ret
= (flags
== Name_Abbr
? wxT("Sun") : wxT("Sunday"));
1043 ret
= (flags
== Name_Abbr
? wxT("Mon") : wxT("Monday"));
1046 ret
= (flags
== Name_Abbr
? wxT("Tue") : wxT("Tuesday"));
1049 ret
= (flags
== Name_Abbr
? wxT("Wed") : wxT("Wednesday"));
1052 ret
= (flags
== Name_Abbr
? wxT("Thu") : wxT("Thursday"));
1055 ret
= (flags
== Name_Abbr
? wxT("Fri") : wxT("Friday"));
1058 ret
= (flags
== Name_Abbr
? wxT("Sat") : wxT("Saturday"));
1062 #endif // HAVE_STRFTIME/!HAVE_STRFTIME
1066 void wxDateTime::GetAmPmStrings(wxString
*am
, wxString
*pm
)
1071 // @Note: Do not call 'CallStrftime' here! CallStrftime checks the return code
1072 // and causes an assertion failed if the buffer is to small (which is good) - OR -
1073 // if strftime does not return anything because the format string is invalid - OR -
1074 // if there are no 'am' / 'pm' tokens defined for the current locale (which is not good).
1075 // wxDateTime::ParseTime will try several different formats to parse the time.
1076 // As a result, GetAmPmStrings might get called, even if the current locale
1077 // does not define any 'am' / 'pm' tokens. In this case, wxStrftime would
1078 // assert, even though it is a perfectly legal use.
1081 if (wxStrftime(buffer
, sizeof(buffer
)/sizeof(wxChar
), _T("%p"), &tm
) > 0)
1082 *am
= wxString(buffer
);
1089 if (wxStrftime(buffer
, sizeof(buffer
)/sizeof(wxChar
), _T("%p"), &tm
) > 0)
1090 *pm
= wxString(buffer
);
1096 // ----------------------------------------------------------------------------
1097 // Country stuff: date calculations depend on the country (DST, work days,
1098 // ...), so we need to know which rules to follow.
1099 // ----------------------------------------------------------------------------
1102 wxDateTime::Country
wxDateTime::GetCountry()
1104 // TODO use LOCALE_ICOUNTRY setting under Win32
1106 if ( ms_country
== Country_Unknown
)
1108 // try to guess from the time zone name
1109 time_t t
= time(NULL
);
1111 struct tm
*tm
= wxLocaltime_r(&t
, &tmstruct
);
1113 wxString tz
= CallStrftime(_T("%Z"), tm
);
1114 if ( tz
== _T("WET") || tz
== _T("WEST") )
1118 else if ( tz
== _T("CET") || tz
== _T("CEST") )
1120 ms_country
= Country_EEC
;
1122 else if ( tz
== _T("MSK") || tz
== _T("MSD") )
1124 ms_country
= Russia
;
1126 else if ( tz
== _T("AST") || tz
== _T("ADT") ||
1127 tz
== _T("EST") || tz
== _T("EDT") ||
1128 tz
== _T("CST") || tz
== _T("CDT") ||
1129 tz
== _T("MST") || tz
== _T("MDT") ||
1130 tz
== _T("PST") || tz
== _T("PDT") )
1136 // well, choose a default one
1140 #else // __WXWINCE__
1142 #endif // !__WXWINCE__/__WXWINCE__
1148 void wxDateTime::SetCountry(wxDateTime::Country country
)
1150 ms_country
= country
;
1154 bool wxDateTime::IsWestEuropeanCountry(Country country
)
1156 if ( country
== Country_Default
)
1158 country
= GetCountry();
1161 return (Country_WesternEurope_Start
<= country
) &&
1162 (country
<= Country_WesternEurope_End
);
1165 // ----------------------------------------------------------------------------
1166 // DST calculations: we use 3 different rules for the West European countries,
1167 // USA and for the rest of the world. This is undoubtedly false for many
1168 // countries, but I lack the necessary info (and the time to gather it),
1169 // please add the other rules here!
1170 // ----------------------------------------------------------------------------
1173 bool wxDateTime::IsDSTApplicable(int year
, Country country
)
1175 if ( year
== Inv_Year
)
1177 // take the current year if none given
1178 year
= GetCurrentYear();
1181 if ( country
== Country_Default
)
1183 country
= GetCountry();
1190 // DST was first observed in the US and UK during WWI, reused
1191 // during WWII and used again since 1966
1192 return year
>= 1966 ||
1193 (year
>= 1942 && year
<= 1945) ||
1194 (year
== 1918 || year
== 1919);
1197 // assume that it started after WWII
1203 wxDateTime
wxDateTime::GetBeginDST(int year
, Country country
)
1205 if ( year
== Inv_Year
)
1207 // take the current year if none given
1208 year
= GetCurrentYear();
1211 if ( country
== Country_Default
)
1213 country
= GetCountry();
1216 if ( !IsDSTApplicable(year
, country
) )
1218 return wxInvalidDateTime
;
1223 if ( IsWestEuropeanCountry(country
) || (country
== Russia
) )
1225 // DST begins at 1 a.m. GMT on the last Sunday of March
1226 if ( !dt
.SetToLastWeekDay(Sun
, Mar
, year
) )
1229 wxFAIL_MSG( _T("no last Sunday in March?") );
1232 dt
+= wxTimeSpan::Hours(1);
1234 else switch ( country
)
1241 // don't know for sure - assume it was in effect all year
1246 dt
.Set(1, Jan
, year
);
1250 // DST was installed Feb 2, 1942 by the Congress
1251 dt
.Set(2, Feb
, year
);
1254 // Oil embargo changed the DST period in the US
1256 dt
.Set(6, Jan
, 1974);
1260 dt
.Set(23, Feb
, 1975);
1264 // before 1986, DST begun on the last Sunday of April, but
1265 // in 1986 Reagan changed it to begin at 2 a.m. of the
1266 // first Sunday in April
1269 if ( !dt
.SetToLastWeekDay(Sun
, Apr
, year
) )
1272 wxFAIL_MSG( _T("no first Sunday in April?") );
1277 if ( !dt
.SetToWeekDay(Sun
, 1, Apr
, year
) )
1280 wxFAIL_MSG( _T("no first Sunday in April?") );
1284 dt
+= wxTimeSpan::Hours(2);
1286 // TODO what about timezone??
1292 // assume Mar 30 as the start of the DST for the rest of the world
1293 // - totally bogus, of course
1294 dt
.Set(30, Mar
, year
);
1301 wxDateTime
wxDateTime::GetEndDST(int year
, Country country
)
1303 if ( year
== Inv_Year
)
1305 // take the current year if none given
1306 year
= GetCurrentYear();
1309 if ( country
== Country_Default
)
1311 country
= GetCountry();
1314 if ( !IsDSTApplicable(year
, country
) )
1316 return wxInvalidDateTime
;
1321 if ( IsWestEuropeanCountry(country
) || (country
== Russia
) )
1323 // DST ends at 1 a.m. GMT on the last Sunday of October
1324 if ( !dt
.SetToLastWeekDay(Sun
, Oct
, year
) )
1326 // weirder and weirder...
1327 wxFAIL_MSG( _T("no last Sunday in October?") );
1330 dt
+= wxTimeSpan::Hours(1);
1332 else switch ( country
)
1339 // don't know for sure - assume it was in effect all year
1343 dt
.Set(31, Dec
, year
);
1347 // the time was reset after the end of the WWII
1348 dt
.Set(30, Sep
, year
);
1352 // DST ends at 2 a.m. on the last Sunday of October
1353 if ( !dt
.SetToLastWeekDay(Sun
, Oct
, year
) )
1355 // weirder and weirder...
1356 wxFAIL_MSG( _T("no last Sunday in October?") );
1359 dt
+= wxTimeSpan::Hours(2);
1361 // TODO what about timezone??
1366 // assume October 26th as the end of the DST - totally bogus too
1367 dt
.Set(26, Oct
, year
);
1373 // ----------------------------------------------------------------------------
1374 // constructors and assignment operators
1375 // ----------------------------------------------------------------------------
1377 // return the current time with ms precision
1378 /* static */ wxDateTime
wxDateTime::UNow()
1380 return wxDateTime(wxGetLocalTimeMillis());
1383 // the values in the tm structure contain the local time
1384 wxDateTime
& wxDateTime::Set(const struct tm
& tm
)
1387 time_t timet
= mktime(&tm2
);
1389 if ( timet
== (time_t)-1 )
1391 // mktime() rather unintuitively fails for Jan 1, 1970 if the hour is
1392 // less than timezone - try to make it work for this case
1393 if ( tm2
.tm_year
== 70 && tm2
.tm_mon
== 0 && tm2
.tm_mday
== 1 )
1395 return Set((time_t)(
1397 tm2
.tm_hour
* MIN_PER_HOUR
* SEC_PER_MIN
+
1398 tm2
.tm_min
* SEC_PER_MIN
+
1402 wxFAIL_MSG( _T("mktime() failed") );
1404 *this = wxInvalidDateTime
;
1414 wxDateTime
& wxDateTime::Set(wxDateTime_t hour
,
1415 wxDateTime_t minute
,
1416 wxDateTime_t second
,
1417 wxDateTime_t millisec
)
1419 // we allow seconds to be 61 to account for the leap seconds, even if we
1420 // don't use them really
1421 wxDATETIME_CHECK( hour
< 24 &&
1425 _T("Invalid time in wxDateTime::Set()") );
1427 // get the current date from system
1429 struct tm
*tm
= GetTmNow(&tmstruct
);
1431 wxDATETIME_CHECK( tm
, _T("wxLocaltime_r() failed") );
1433 // make a copy so it isn't clobbered by the call to mktime() below
1438 tm1
.tm_min
= minute
;
1439 tm1
.tm_sec
= second
;
1441 // and the DST in case it changes on this date
1444 if ( tm2
.tm_isdst
!= tm1
.tm_isdst
)
1445 tm1
.tm_isdst
= tm2
.tm_isdst
;
1449 // and finally adjust milliseconds
1450 return SetMillisecond(millisec
);
1453 wxDateTime
& wxDateTime::Set(wxDateTime_t day
,
1457 wxDateTime_t minute
,
1458 wxDateTime_t second
,
1459 wxDateTime_t millisec
)
1461 wxDATETIME_CHECK( hour
< 24 &&
1465 _T("Invalid time in wxDateTime::Set()") );
1467 ReplaceDefaultYearMonthWithCurrent(&year
, &month
);
1469 wxDATETIME_CHECK( (0 < day
) && (day
<= GetNumberOfDays(month
, year
)),
1470 _T("Invalid date in wxDateTime::Set()") );
1472 // the range of time_t type (inclusive)
1473 static const int yearMinInRange
= 1970;
1474 static const int yearMaxInRange
= 2037;
1476 // test only the year instead of testing for the exact end of the Unix
1477 // time_t range - it doesn't bring anything to do more precise checks
1478 if ( year
>= yearMinInRange
&& year
<= yearMaxInRange
)
1480 // use the standard library version if the date is in range - this is
1481 // probably more efficient than our code
1483 tm
.tm_year
= year
- 1900;
1489 tm
.tm_isdst
= -1; // mktime() will guess it
1493 // and finally adjust milliseconds
1495 SetMillisecond(millisec
);
1501 // do time calculations ourselves: we want to calculate the number of
1502 // milliseconds between the given date and the epoch
1504 // get the JDN for the midnight of this day
1505 m_time
= GetTruncatedJDN(day
, month
, year
);
1506 m_time
-= EPOCH_JDN
;
1507 m_time
*= SECONDS_PER_DAY
* TIME_T_FACTOR
;
1509 // JDN corresponds to GMT, we take localtime
1510 Add(wxTimeSpan(hour
, minute
, second
+ GetTimeZone(), millisec
));
1516 wxDateTime
& wxDateTime::Set(double jdn
)
1518 // so that m_time will be 0 for the midnight of Jan 1, 1970 which is jdn
1520 jdn
-= EPOCH_JDN
+ 0.5;
1522 m_time
.Assign(jdn
*MILLISECONDS_PER_DAY
);
1524 // JDNs always are in UTC, so we don't need any adjustments for time zone
1529 wxDateTime
& wxDateTime::ResetTime()
1533 if ( tm
.hour
|| tm
.min
|| tm
.sec
|| tm
.msec
)
1546 wxDateTime
wxDateTime::GetDateOnly() const
1553 return wxDateTime(tm
);
1556 // ----------------------------------------------------------------------------
1557 // DOS Date and Time Format functions
1558 // ----------------------------------------------------------------------------
1559 // the dos date and time value is an unsigned 32 bit value in the format:
1560 // YYYYYYYMMMMDDDDDhhhhhmmmmmmsssss
1562 // Y = year offset from 1980 (0-127)
1564 // D = day of month (1-31)
1566 // m = minute (0-59)
1567 // s = bisecond (0-29) each bisecond indicates two seconds
1568 // ----------------------------------------------------------------------------
1570 wxDateTime
& wxDateTime::SetFromDOS(unsigned long ddt
)
1575 long year
= ddt
& 0xFE000000;
1580 long month
= ddt
& 0x1E00000;
1585 long day
= ddt
& 0x1F0000;
1589 long hour
= ddt
& 0xF800;
1593 long minute
= ddt
& 0x7E0;
1597 long second
= ddt
& 0x1F;
1598 tm
.tm_sec
= second
* 2;
1600 return Set(mktime(&tm
));
1603 unsigned long wxDateTime::GetAsDOS() const
1606 time_t ticks
= GetTicks();
1608 struct tm
*tm
= wxLocaltime_r(&ticks
, &tmstruct
);
1609 wxCHECK_MSG( tm
, ULONG_MAX
, _T("time can't be represented in DOS format") );
1611 long year
= tm
->tm_year
;
1615 long month
= tm
->tm_mon
;
1619 long day
= tm
->tm_mday
;
1622 long hour
= tm
->tm_hour
;
1625 long minute
= tm
->tm_min
;
1628 long second
= tm
->tm_sec
;
1631 ddt
= year
| month
| day
| hour
| minute
| second
;
1635 // ----------------------------------------------------------------------------
1636 // time_t <-> broken down time conversions
1637 // ----------------------------------------------------------------------------
1639 wxDateTime::Tm
wxDateTime::GetTm(const TimeZone
& tz
) const
1641 wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") );
1643 time_t time
= GetTicks();
1644 if ( time
!= (time_t)-1 )
1646 // use C RTL functions
1649 if ( tz
.GetOffset() == -GetTimeZone() )
1651 // we are working with local time
1652 tm
= wxLocaltime_r(&time
, &tmstruct
);
1654 // should never happen
1655 wxCHECK_MSG( tm
, Tm(), _T("wxLocaltime_r() failed") );
1659 time
+= (time_t)tz
.GetOffset();
1660 #if defined(__VMS__) || defined(__WATCOMC__) // time is unsigned so avoid warning
1661 int time2
= (int) time
;
1667 tm
= wxGmtime_r(&time
, &tmstruct
);
1669 // should never happen
1670 wxCHECK_MSG( tm
, Tm(), _T("wxGmtime_r() failed") );
1674 tm
= (struct tm
*)NULL
;
1680 // adjust the milliseconds
1682 long timeOnly
= (m_time
% MILLISECONDS_PER_DAY
).ToLong();
1683 tm2
.msec
= (wxDateTime_t
)(timeOnly
% 1000);
1686 //else: use generic code below
1689 // remember the time and do the calculations with the date only - this
1690 // eliminates rounding errors of the floating point arithmetics
1692 wxLongLong timeMidnight
= m_time
+ tz
.GetOffset() * 1000;
1694 long timeOnly
= (timeMidnight
% MILLISECONDS_PER_DAY
).ToLong();
1696 // we want to always have positive time and timeMidnight to be really
1697 // the midnight before it
1700 timeOnly
= MILLISECONDS_PER_DAY
+ timeOnly
;
1703 timeMidnight
-= timeOnly
;
1705 // calculate the Gregorian date from JDN for the midnight of our date:
1706 // this will yield day, month (in 1..12 range) and year
1708 // actually, this is the JDN for the noon of the previous day
1709 long jdn
= (timeMidnight
/ MILLISECONDS_PER_DAY
).ToLong() + EPOCH_JDN
;
1711 // CREDIT: code below is by Scott E. Lee (but bugs are mine)
1713 wxASSERT_MSG( jdn
> -2, _T("JDN out of range") );
1715 // calculate the century
1716 long temp
= (jdn
+ JDN_OFFSET
) * 4 - 1;
1717 long century
= temp
/ DAYS_PER_400_YEARS
;
1719 // then the year and day of year (1 <= dayOfYear <= 366)
1720 temp
= ((temp
% DAYS_PER_400_YEARS
) / 4) * 4 + 3;
1721 long year
= (century
* 100) + (temp
/ DAYS_PER_4_YEARS
);
1722 long dayOfYear
= (temp
% DAYS_PER_4_YEARS
) / 4 + 1;
1724 // and finally the month and day of the month
1725 temp
= dayOfYear
* 5 - 3;
1726 long month
= temp
/ DAYS_PER_5_MONTHS
;
1727 long day
= (temp
% DAYS_PER_5_MONTHS
) / 5 + 1;
1729 // month is counted from March - convert to normal
1740 // year is offset by 4800
1743 // check that the algorithm gave us something reasonable
1744 wxASSERT_MSG( (0 < month
) && (month
<= 12), _T("invalid month") );
1745 wxASSERT_MSG( (1 <= day
) && (day
< 32), _T("invalid day") );
1747 // construct Tm from these values
1749 tm
.year
= (int)year
;
1750 tm
.mon
= (Month
)(month
- 1); // algorithm yields 1 for January, not 0
1751 tm
.mday
= (wxDateTime_t
)day
;
1752 tm
.msec
= (wxDateTime_t
)(timeOnly
% 1000);
1753 timeOnly
-= tm
.msec
;
1754 timeOnly
/= 1000; // now we have time in seconds
1756 tm
.sec
= (wxDateTime_t
)(timeOnly
% SEC_PER_MIN
);
1758 timeOnly
/= SEC_PER_MIN
; // now we have time in minutes
1760 tm
.min
= (wxDateTime_t
)(timeOnly
% MIN_PER_HOUR
);
1763 tm
.hour
= (wxDateTime_t
)(timeOnly
/ MIN_PER_HOUR
);
1768 wxDateTime
& wxDateTime::SetYear(int year
)
1770 wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") );
1779 wxDateTime
& wxDateTime::SetMonth(Month month
)
1781 wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") );
1790 wxDateTime
& wxDateTime::SetDay(wxDateTime_t mday
)
1792 wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") );
1801 wxDateTime
& wxDateTime::SetHour(wxDateTime_t hour
)
1803 wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") );
1812 wxDateTime
& wxDateTime::SetMinute(wxDateTime_t min
)
1814 wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") );
1823 wxDateTime
& wxDateTime::SetSecond(wxDateTime_t sec
)
1825 wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") );
1834 wxDateTime
& wxDateTime::SetMillisecond(wxDateTime_t millisecond
)
1836 wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") );
1838 // we don't need to use GetTm() for this one
1839 m_time
-= m_time
% 1000l;
1840 m_time
+= millisecond
;
1845 // ----------------------------------------------------------------------------
1846 // wxDateTime arithmetics
1847 // ----------------------------------------------------------------------------
1849 wxDateTime
& wxDateTime::Add(const wxDateSpan
& diff
)
1853 tm
.year
+= diff
.GetYears();
1854 tm
.AddMonths(diff
.GetMonths());
1856 // check that the resulting date is valid
1857 if ( tm
.mday
> GetNumOfDaysInMonth(tm
.year
, tm
.mon
) )
1859 // We suppose that when adding one month to Jan 31 we want to get Feb
1860 // 28 (or 29), i.e. adding a month to the last day of the month should
1861 // give the last day of the next month which is quite logical.
1863 // Unfortunately, there is no logic way to understand what should
1864 // Jan 30 + 1 month be - Feb 28 too or Feb 27 (assuming non leap year)?
1865 // We make it Feb 28 (last day too), but it is highly questionable.
1866 tm
.mday
= GetNumOfDaysInMonth(tm
.year
, tm
.mon
);
1869 tm
.AddDays(diff
.GetTotalDays());
1873 wxASSERT_MSG( IsSameTime(tm
),
1874 _T("Add(wxDateSpan) shouldn't modify time") );
1879 // ----------------------------------------------------------------------------
1880 // Weekday and monthday stuff
1881 // ----------------------------------------------------------------------------
1883 // convert Sun, Mon, ..., Sat into 6, 0, ..., 5
1884 static inline int ConvertWeekDayToMondayBase(int wd
)
1886 return wd
== wxDateTime::Sun
? 6 : wd
- 1;
1891 wxDateTime::SetToWeekOfYear(int year
, wxDateTime_t numWeek
, WeekDay wd
)
1893 wxASSERT_MSG( numWeek
> 0,
1894 _T("invalid week number: weeks are counted from 1") );
1896 // Jan 4 always lies in the 1st week of the year
1897 wxDateTime
dt(4, Jan
, year
);
1898 dt
.SetToWeekDayInSameWeek(wd
);
1899 dt
+= wxDateSpan::Weeks(numWeek
- 1);
1904 #if WXWIN_COMPATIBILITY_2_6
1905 // use a separate function to avoid warnings about using deprecated
1906 // SetToTheWeek in GetWeek below
1908 SetToTheWeek(int year
,
1909 wxDateTime::wxDateTime_t numWeek
,
1910 wxDateTime::WeekDay weekday
,
1911 wxDateTime::WeekFlags flags
)
1913 // Jan 4 always lies in the 1st week of the year
1914 wxDateTime
dt(4, wxDateTime::Jan
, year
);
1915 dt
.SetToWeekDayInSameWeek(weekday
, flags
);
1916 dt
+= wxDateSpan::Weeks(numWeek
- 1);
1921 bool wxDateTime::SetToTheWeek(wxDateTime_t numWeek
,
1925 int year
= GetYear();
1926 *this = ::SetToTheWeek(year
, numWeek
, weekday
, flags
);
1927 if ( GetYear() != year
)
1929 // oops... numWeek was too big
1936 wxDateTime
wxDateTime::GetWeek(wxDateTime_t numWeek
,
1938 WeekFlags flags
) const
1940 return ::SetToTheWeek(GetYear(), numWeek
, weekday
, flags
);
1942 #endif // WXWIN_COMPATIBILITY_2_6
1944 wxDateTime
& wxDateTime::SetToLastMonthDay(Month month
,
1947 // take the current month/year if none specified
1948 if ( year
== Inv_Year
)
1950 if ( month
== Inv_Month
)
1953 return Set(GetNumOfDaysInMonth(year
, month
), month
, year
);
1956 wxDateTime
& wxDateTime::SetToWeekDayInSameWeek(WeekDay weekday
, WeekFlags flags
)
1958 wxDATETIME_CHECK( weekday
!= Inv_WeekDay
, _T("invalid weekday") );
1960 int wdayDst
= weekday
,
1961 wdayThis
= GetWeekDay();
1962 if ( wdayDst
== wdayThis
)
1968 if ( flags
== Default_First
)
1970 flags
= GetCountry() == USA
? Sunday_First
: Monday_First
;
1973 // the logic below based on comparing weekday and wdayThis works if Sun (0)
1974 // is the first day in the week, but breaks down for Monday_First case so
1975 // we adjust the week days in this case
1976 if ( flags
== Monday_First
)
1978 if ( wdayThis
== Sun
)
1980 if ( wdayDst
== Sun
)
1983 //else: Sunday_First, nothing to do
1985 // go forward or back in time to the day we want
1986 if ( wdayDst
< wdayThis
)
1988 return Subtract(wxDateSpan::Days(wdayThis
- wdayDst
));
1990 else // weekday > wdayThis
1992 return Add(wxDateSpan::Days(wdayDst
- wdayThis
));
1996 wxDateTime
& wxDateTime::SetToNextWeekDay(WeekDay weekday
)
1998 wxDATETIME_CHECK( weekday
!= Inv_WeekDay
, _T("invalid weekday") );
2001 WeekDay wdayThis
= GetWeekDay();
2002 if ( weekday
== wdayThis
)
2007 else if ( weekday
< wdayThis
)
2009 // need to advance a week
2010 diff
= 7 - (wdayThis
- weekday
);
2012 else // weekday > wdayThis
2014 diff
= weekday
- wdayThis
;
2017 return Add(wxDateSpan::Days(diff
));
2020 wxDateTime
& wxDateTime::SetToPrevWeekDay(WeekDay weekday
)
2022 wxDATETIME_CHECK( weekday
!= Inv_WeekDay
, _T("invalid weekday") );
2025 WeekDay wdayThis
= GetWeekDay();
2026 if ( weekday
== wdayThis
)
2031 else if ( weekday
> wdayThis
)
2033 // need to go to previous week
2034 diff
= 7 - (weekday
- wdayThis
);
2036 else // weekday < wdayThis
2038 diff
= wdayThis
- weekday
;
2041 return Subtract(wxDateSpan::Days(diff
));
2044 bool wxDateTime::SetToWeekDay(WeekDay weekday
,
2049 wxCHECK_MSG( weekday
!= Inv_WeekDay
, false, _T("invalid weekday") );
2051 // we don't check explicitly that -5 <= n <= 5 because we will return false
2052 // anyhow in such case - but may be should still give an assert for it?
2054 // take the current month/year if none specified
2055 ReplaceDefaultYearMonthWithCurrent(&year
, &month
);
2059 // TODO this probably could be optimised somehow...
2063 // get the first day of the month
2064 dt
.Set(1, month
, year
);
2067 WeekDay wdayFirst
= dt
.GetWeekDay();
2069 // go to the first weekday of the month
2070 int diff
= weekday
- wdayFirst
;
2074 // add advance n-1 weeks more
2077 dt
+= wxDateSpan::Days(diff
);
2079 else // count from the end of the month
2081 // get the last day of the month
2082 dt
.SetToLastMonthDay(month
, year
);
2085 WeekDay wdayLast
= dt
.GetWeekDay();
2087 // go to the last weekday of the month
2088 int diff
= wdayLast
- weekday
;
2092 // and rewind n-1 weeks from there
2095 dt
-= wxDateSpan::Days(diff
);
2098 // check that it is still in the same month
2099 if ( dt
.GetMonth() == month
)
2107 // no such day in this month
2113 wxDateTime::wxDateTime_t
GetDayOfYearFromTm(const wxDateTime::Tm
& tm
)
2115 return (wxDateTime::wxDateTime_t
)(gs_cumulatedDays
[wxDateTime::IsLeapYear(tm
.year
)][tm
.mon
] + tm
.mday
);
2118 wxDateTime::wxDateTime_t
wxDateTime::GetDayOfYear(const TimeZone
& tz
) const
2120 return GetDayOfYearFromTm(GetTm(tz
));
2123 wxDateTime::wxDateTime_t
2124 wxDateTime::GetWeekOfYear(wxDateTime::WeekFlags flags
, const TimeZone
& tz
) const
2126 if ( flags
== Default_First
)
2128 flags
= GetCountry() == USA
? Sunday_First
: Monday_First
;
2132 wxDateTime_t nDayInYear
= GetDayOfYearFromTm(tm
);
2134 int wdTarget
= GetWeekDay(tz
);
2135 int wdYearStart
= wxDateTime(1, Jan
, GetYear()).GetWeekDay();
2137 if ( flags
== Sunday_First
)
2139 // FIXME: First week is not calculated correctly.
2140 week
= (nDayInYear
- wdTarget
+ 7) / 7;
2141 if ( wdYearStart
== Wed
|| wdYearStart
== Thu
)
2144 else // week starts with monday
2146 // adjust the weekdays to non-US style.
2147 wdYearStart
= ConvertWeekDayToMondayBase(wdYearStart
);
2148 wdTarget
= ConvertWeekDayToMondayBase(wdTarget
);
2150 // quoting from http://www.cl.cam.ac.uk/~mgk25/iso-time.html:
2152 // Week 01 of a year is per definition the first week that has the
2153 // Thursday in this year, which is equivalent to the week that
2154 // contains the fourth day of January. In other words, the first
2155 // week of a new year is the week that has the majority of its
2156 // days in the new year. Week 01 might also contain days from the
2157 // previous year and the week before week 01 of a year is the last
2158 // week (52 or 53) of the previous year even if it contains days
2159 // from the new year. A week starts with Monday (day 1) and ends
2160 // with Sunday (day 7).
2163 // if Jan 1 is Thursday or less, it is in the first week of this year
2164 if ( wdYearStart
< 4 )
2166 // count the number of entire weeks between Jan 1 and this date
2167 week
= (nDayInYear
+ wdYearStart
+ 6 - wdTarget
)/7;
2169 // be careful to check for overflow in the next year
2170 if ( week
== 53 && tm
.mday
- wdTarget
> 28 )
2173 else // Jan 1 is in the last week of the previous year
2175 // check if we happen to be at the last week of previous year:
2176 if ( tm
.mon
== Jan
&& tm
.mday
< 8 - wdYearStart
)
2177 week
= wxDateTime(31, Dec
, GetYear()-1).GetWeekOfYear();
2179 week
= (nDayInYear
+ wdYearStart
- 1 - wdTarget
)/7;
2183 return (wxDateTime::wxDateTime_t
)week
;
2186 wxDateTime::wxDateTime_t
wxDateTime::GetWeekOfMonth(wxDateTime::WeekFlags flags
,
2187 const TimeZone
& tz
) const
2190 wxDateTime dtMonthStart
= wxDateTime(1, tm
.mon
, tm
.year
);
2191 int nWeek
= GetWeekOfYear(flags
) - dtMonthStart
.GetWeekOfYear(flags
) + 1;
2194 // this may happen for January when Jan, 1 is the last week of the
2196 nWeek
+= IsLeapYear(tm
.year
- 1) ? 53 : 52;
2199 return (wxDateTime::wxDateTime_t
)nWeek
;
2202 wxDateTime
& wxDateTime::SetToYearDay(wxDateTime::wxDateTime_t yday
)
2204 int year
= GetYear();
2205 wxDATETIME_CHECK( (0 < yday
) && (yday
<= GetNumberOfDays(year
)),
2206 _T("invalid year day") );
2208 bool isLeap
= IsLeapYear(year
);
2209 for ( Month mon
= Jan
; mon
< Inv_Month
; wxNextMonth(mon
) )
2211 // for Dec, we can't compare with gs_cumulatedDays[mon + 1], but we
2212 // don't need it neither - because of the CHECK above we know that
2213 // yday lies in December then
2214 if ( (mon
== Dec
) || (yday
<= gs_cumulatedDays
[isLeap
][mon
+ 1]) )
2216 Set((wxDateTime::wxDateTime_t
)(yday
- gs_cumulatedDays
[isLeap
][mon
]), mon
, year
);
2225 // ----------------------------------------------------------------------------
2226 // Julian day number conversion and related stuff
2227 // ----------------------------------------------------------------------------
2229 double wxDateTime::GetJulianDayNumber() const
2231 return m_time
.ToDouble() / MILLISECONDS_PER_DAY
+ EPOCH_JDN
+ 0.5;
2234 double wxDateTime::GetRataDie() const
2236 // March 1 of the year 0 is Rata Die day -306 and JDN 1721119.5
2237 return GetJulianDayNumber() - 1721119.5 - 306;
2240 // ----------------------------------------------------------------------------
2241 // timezone and DST stuff
2242 // ----------------------------------------------------------------------------
2244 int wxDateTime::IsDST(wxDateTime::Country country
) const
2246 wxCHECK_MSG( country
== Country_Default
, -1,
2247 _T("country support not implemented") );
2249 // use the C RTL for the dates in the standard range
2250 time_t timet
= GetTicks();
2251 if ( timet
!= (time_t)-1 )
2254 tm
*tm
= wxLocaltime_r(&timet
, &tmstruct
);
2256 wxCHECK_MSG( tm
, -1, _T("wxLocaltime_r() failed") );
2258 return tm
->tm_isdst
;
2262 int year
= GetYear();
2264 if ( !IsDSTApplicable(year
, country
) )
2266 // no DST time in this year in this country
2270 return IsBetween(GetBeginDST(year
, country
), GetEndDST(year
, country
));
2274 wxDateTime
& wxDateTime::MakeTimezone(const TimeZone
& tz
, bool noDST
)
2276 long secDiff
= GetTimeZone() + tz
.GetOffset();
2278 // we need to know whether DST is or not in effect for this date unless
2279 // the test disabled by the caller
2280 if ( !noDST
&& (IsDST() == 1) )
2282 // FIXME we assume that the DST is always shifted by 1 hour
2286 return Add(wxTimeSpan::Seconds(secDiff
));
2289 wxDateTime
& wxDateTime::MakeFromTimezone(const TimeZone
& tz
, bool noDST
)
2291 long secDiff
= GetTimeZone() + tz
.GetOffset();
2293 // we need to know whether DST is or not in effect for this date unless
2294 // the test disabled by the caller
2295 if ( !noDST
&& (IsDST() == 1) )
2297 // FIXME we assume that the DST is always shifted by 1 hour
2301 return Subtract(wxTimeSpan::Seconds(secDiff
));
2304 // ----------------------------------------------------------------------------
2305 // wxDateTime to/from text representations
2306 // ----------------------------------------------------------------------------
2308 wxString
wxDateTime::Format(const wxString
& format
, const TimeZone
& tz
) const
2310 wxCHECK_MSG( !format
.empty(), wxEmptyString
,
2311 _T("NULL format in wxDateTime::Format") );
2313 // we have to use our own implementation if the date is out of range of
2314 // strftime() or if we use non standard specificators
2315 #ifdef HAVE_STRFTIME
2316 time_t time
= GetTicks();
2318 if ( (time
!= (time_t)-1) && !wxStrstr(format
, _T("%l")) )
2323 if ( tz
.GetOffset() == -GetTimeZone() )
2325 // we are working with local time
2326 tm
= wxLocaltime_r(&time
, &tmstruct
);
2328 // should never happen
2329 wxCHECK_MSG( tm
, wxEmptyString
, _T("wxLocaltime_r() failed") );
2333 time
+= (int)tz
.GetOffset();
2335 #if defined(__VMS__) || defined(__WATCOMC__) // time is unsigned so avoid warning
2336 int time2
= (int) time
;
2342 tm
= wxGmtime_r(&time
, &tmstruct
);
2344 // should never happen
2345 wxCHECK_MSG( tm
, wxEmptyString
, _T("wxGmtime_r() failed") );
2349 tm
= (struct tm
*)NULL
;
2355 return CallStrftime(format
, tm
);
2358 //else: use generic code below
2359 #endif // HAVE_STRFTIME
2361 // we only parse ANSI C format specifications here, no POSIX 2
2362 // complications, no GNU extensions but we do add support for a "%l" format
2363 // specifier allowing to get the number of milliseconds
2366 // used for calls to strftime() when we only deal with time
2367 struct tm tmTimeOnly
;
2368 tmTimeOnly
.tm_hour
= tm
.hour
;
2369 tmTimeOnly
.tm_min
= tm
.min
;
2370 tmTimeOnly
.tm_sec
= tm
.sec
;
2371 tmTimeOnly
.tm_wday
= 0;
2372 tmTimeOnly
.tm_yday
= 0;
2373 tmTimeOnly
.tm_mday
= 1; // any date will do
2374 tmTimeOnly
.tm_mon
= 0;
2375 tmTimeOnly
.tm_year
= 76;
2376 tmTimeOnly
.tm_isdst
= 0; // no DST, we adjust for tz ourselves
2378 wxString tmp
, res
, fmt
;
2379 for ( wxString::const_iterator p
= format
.begin(); p
!= format
.end(); ++p
)
2381 if ( *p
!= _T('%') )
2389 // set the default format
2390 switch ( (*++p
).GetValue() )
2392 case _T('Y'): // year has 4 digits
2396 case _T('j'): // day of year has 3 digits
2397 case _T('l'): // milliseconds have 3 digits
2401 case _T('w'): // week day as number has only one
2406 // it's either another valid format specifier in which case
2407 // the format is "%02d" (for all the rest) or we have the
2408 // field width preceding the format in which case it will
2409 // override the default format anyhow
2413 bool restart
= true;
2418 // start of the format specification
2419 switch ( (*p
).GetValue() )
2421 case _T('a'): // a weekday name
2423 // second parameter should be true for abbreviated names
2424 res
+= GetWeekDayName(tm
.GetWeekDay(),
2425 *p
== _T('a') ? Name_Abbr
: Name_Full
);
2428 case _T('b'): // a month name
2430 res
+= GetMonthName(tm
.mon
,
2431 *p
== _T('b') ? Name_Abbr
: Name_Full
);
2434 case _T('c'): // locale default date and time representation
2435 case _T('x'): // locale default date representation
2436 #ifdef HAVE_STRFTIME
2438 // the problem: there is no way to know what do these format
2439 // specifications correspond to for the current locale.
2441 // the solution: use a hack and still use strftime(): first
2442 // find the YEAR which is a year in the strftime() range (1970
2443 // - 2038) whose Jan 1 falls on the same week day as the Jan 1
2444 // of the real year. Then make a copy of the format and
2445 // replace all occurrences of YEAR in it with some unique
2446 // string not appearing anywhere else in it, then use
2447 // strftime() to format the date in year YEAR and then replace
2448 // YEAR back by the real year and the unique replacement
2449 // string back with YEAR. Notice that "all occurrences of YEAR"
2450 // means all occurrences of 4 digit as well as 2 digit form!
2452 // the bugs: we assume that neither of %c nor %x contains any
2453 // fields which may change between the YEAR and real year. For
2454 // example, the week number (%U, %W) and the day number (%j)
2455 // will change if one of these years is leap and the other one
2458 // find the YEAR: normally, for any year X, Jan 1 of the
2459 // year X + 28 is the same weekday as Jan 1 of X (because
2460 // the weekday advances by 1 for each normal X and by 2
2461 // for each leap X, hence by 5 every 4 years or by 35
2462 // which is 0 mod 7 every 28 years) but this rule breaks
2463 // down if there are years between X and Y which are
2464 // divisible by 4 but not leap (i.e. divisible by 100 but
2465 // not 400), hence the correction.
2467 int yearReal
= GetYear(tz
);
2468 int mod28
= yearReal
% 28;
2470 // be careful to not go too far - we risk to leave the
2475 year
= 1988 + mod28
; // 1988 == 0 (mod 28)
2479 year
= 1970 + mod28
- 10; // 1970 == 10 (mod 28)
2482 int nCentury
= year
/ 100,
2483 nCenturyReal
= yearReal
/ 100;
2485 // need to adjust for the years divisble by 400 which are
2486 // not leap but are counted like leap ones if we just take
2487 // the number of centuries in between for nLostWeekDays
2488 int nLostWeekDays
= (nCentury
- nCenturyReal
) -
2489 (nCentury
/ 4 - nCenturyReal
/ 4);
2491 // we have to gain back the "lost" weekdays: note that the
2492 // effect of this loop is to not do anything to
2493 // nLostWeekDays (which we won't use any more), but to
2494 // (indirectly) set the year correctly
2495 while ( (nLostWeekDays
% 7) != 0 )
2497 nLostWeekDays
+= year
++ % 4 ? 1 : 2;
2500 // finally move the year below 2000 so that the 2-digit
2501 // year number can never match the month or day of the
2502 // month when we do the replacements below
2506 wxASSERT_MSG( year
>= 1970 && year
< 2000,
2507 _T("logic error in wxDateTime::Format") );
2510 // use strftime() to format the same date but in supported
2513 // NB: we assume that strftime() doesn't check for the
2514 // date validity and will happily format the date
2515 // corresponding to Feb 29 of a non leap year (which
2516 // may happen if yearReal was leap and year is not)
2517 struct tm tmAdjusted
;
2519 tmAdjusted
.tm_hour
= tm
.hour
;
2520 tmAdjusted
.tm_min
= tm
.min
;
2521 tmAdjusted
.tm_sec
= tm
.sec
;
2522 tmAdjusted
.tm_wday
= tm
.GetWeekDay();
2523 tmAdjusted
.tm_yday
= GetDayOfYear();
2524 tmAdjusted
.tm_mday
= tm
.mday
;
2525 tmAdjusted
.tm_mon
= tm
.mon
;
2526 tmAdjusted
.tm_year
= year
- 1900;
2527 tmAdjusted
.tm_isdst
= 0; // no DST, already adjusted
2528 wxString str
= CallStrftime(*p
== _T('c') ? _T("%c")
2532 // now replace the replacement year with the real year:
2533 // notice that we have to replace the 4 digit year with
2534 // a unique string not appearing in strftime() output
2535 // first to prevent the 2 digit year from matching any
2536 // substring of the 4 digit year (but any day, month,
2537 // hours or minutes components should be safe because
2538 // they are never in 70-99 range)
2539 wxString
replacement("|");
2540 while ( str
.find(replacement
) != wxString::npos
)
2543 str
.Replace(wxString::Format("%d", year
),
2545 str
.Replace(wxString::Format("%d", year
% 100),
2546 wxString::Format("%d", yearReal
% 100));
2547 str
.Replace(replacement
,
2548 wxString::Format("%d", yearReal
));
2552 #else // !HAVE_STRFTIME
2553 // Use "%m/%d/%y %H:%M:%S" format instead
2554 res
+= wxString::Format(wxT("%02d/%02d/%04d %02d:%02d:%02d"),
2555 tm
.mon
+1,tm
.mday
, tm
.year
, tm
.hour
, tm
.min
, tm
.sec
);
2556 #endif // HAVE_STRFTIME/!HAVE_STRFTIME
2559 case _T('d'): // day of a month (01-31)
2560 res
+= wxString::Format(fmt
, tm
.mday
);
2563 case _T('H'): // hour in 24h format (00-23)
2564 res
+= wxString::Format(fmt
, tm
.hour
);
2567 case _T('I'): // hour in 12h format (01-12)
2569 // 24h -> 12h, 0h -> 12h too
2570 int hour12
= tm
.hour
> 12 ? tm
.hour
- 12
2571 : tm
.hour
? tm
.hour
: 12;
2572 res
+= wxString::Format(fmt
, hour12
);
2576 case _T('j'): // day of the year
2577 res
+= wxString::Format(fmt
, GetDayOfYear(tz
));
2580 case _T('l'): // milliseconds (NOT STANDARD)
2581 res
+= wxString::Format(fmt
, GetMillisecond(tz
));
2584 case _T('m'): // month as a number (01-12)
2585 res
+= wxString::Format(fmt
, tm
.mon
+ 1);
2588 case _T('M'): // minute as a decimal number (00-59)
2589 res
+= wxString::Format(fmt
, tm
.min
);
2592 case _T('p'): // AM or PM string
2593 #ifdef HAVE_STRFTIME
2594 res
+= CallStrftime(_T("%p"), &tmTimeOnly
);
2595 #else // !HAVE_STRFTIME
2596 res
+= (tmTimeOnly
.tm_hour
> 12) ? wxT("pm") : wxT("am");
2597 #endif // HAVE_STRFTIME/!HAVE_STRFTIME
2600 case _T('S'): // second as a decimal number (00-61)
2601 res
+= wxString::Format(fmt
, tm
.sec
);
2604 case _T('U'): // week number in the year (Sunday 1st week day)
2605 res
+= wxString::Format(fmt
, GetWeekOfYear(Sunday_First
, tz
));
2608 case _T('W'): // week number in the year (Monday 1st week day)
2609 res
+= wxString::Format(fmt
, GetWeekOfYear(Monday_First
, tz
));
2612 case _T('w'): // weekday as a number (0-6), Sunday = 0
2613 res
+= wxString::Format(fmt
, tm
.GetWeekDay());
2616 // case _T('x'): -- handled with "%c"
2618 case _T('X'): // locale default time representation
2619 // just use strftime() to format the time for us
2620 #ifdef HAVE_STRFTIME
2621 res
+= CallStrftime(_T("%X"), &tmTimeOnly
);
2622 #else // !HAVE_STRFTIME
2623 res
+= wxString::Format(wxT("%02d:%02d:%02d"),tm
.hour
, tm
.min
, tm
.sec
);
2624 #endif // HAVE_STRFTIME/!HAVE_STRFTIME
2627 case _T('y'): // year without century (00-99)
2628 res
+= wxString::Format(fmt
, tm
.year
% 100);
2631 case _T('Y'): // year with century
2632 res
+= wxString::Format(fmt
, tm
.year
);
2635 case _T('Z'): // timezone name
2636 #ifdef HAVE_STRFTIME
2637 res
+= CallStrftime(_T("%Z"), &tmTimeOnly
);
2642 // is it the format width?
2644 while ( *p
== _T('-') || *p
== _T('+') ||
2645 *p
== _T(' ') || wxIsdigit(*p
) )
2652 // we've only got the flags and width so far in fmt
2653 fmt
.Prepend(_T('%'));
2654 fmt
.Append(_T('d'));
2661 // no, it wasn't the width
2662 wxFAIL_MSG(_T("unknown format specificator"));
2664 // fall through and just copy it nevertheless
2666 case _T('%'): // a percent sign
2670 case 0: // the end of string
2671 wxFAIL_MSG(_T("missing format at the end of string"));
2673 // just put the '%' which was the last char in format
2683 // this function parses a string in (strict) RFC 822 format: see the section 5
2684 // of the RFC for the detailed description, but briefly it's something of the
2685 // form "Sat, 18 Dec 1999 00:48:30 +0100"
2687 // this function is "strict" by design - it must reject anything except true
2688 // RFC822 time specs.
2690 // TODO a great candidate for using reg exps
2692 wxDateTime::ParseRfc822Date(const wxString
& date
, wxString::const_iterator
*end
)
2694 // TODO: rewrite using iterators instead of wxChar pointers
2695 const wxStringCharType
*p
= date
.wx_str();
2696 const wxStringCharType
*comma
= wxStrchr(p
, wxS(','));
2699 // the part before comma is the weekday
2701 // skip it for now - we don't use but might check that it really
2702 // corresponds to the specfied date
2705 if ( *p
!= _T(' ') )
2707 wxLogDebug(_T("no space after weekday in RFC822 time spec"));
2715 // the following 1 or 2 digits are the day number
2716 if ( !wxIsdigit(*p
) )
2718 wxLogDebug(_T("day number expected in RFC822 time spec, none found"));
2723 wxDateTime_t day
= (wxDateTime_t
)(*p
++ - _T('0'));
2724 if ( wxIsdigit(*p
) )
2727 day
= (wxDateTime_t
)(day
+ (*p
++ - _T('0')));
2730 if ( *p
++ != _T(' ') )
2735 // the following 3 letters specify the month
2736 wxString
monName(p
, 3);
2738 if ( monName
== _T("Jan") )
2740 else if ( monName
== _T("Feb") )
2742 else if ( monName
== _T("Mar") )
2744 else if ( monName
== _T("Apr") )
2746 else if ( monName
== _T("May") )
2748 else if ( monName
== _T("Jun") )
2750 else if ( monName
== _T("Jul") )
2752 else if ( monName
== _T("Aug") )
2754 else if ( monName
== _T("Sep") )
2756 else if ( monName
== _T("Oct") )
2758 else if ( monName
== _T("Nov") )
2760 else if ( monName
== _T("Dec") )
2764 wxLogDebug(_T("Invalid RFC 822 month name '%s'"), monName
.c_str());
2771 if ( *p
++ != _T(' ') )
2777 if ( !wxIsdigit(*p
) )
2783 int year
= *p
++ - _T('0');
2785 if ( !wxIsdigit(*p
) )
2787 // should have at least 2 digits in the year
2792 year
+= *p
++ - _T('0');
2794 // is it a 2 digit year (as per original RFC 822) or a 4 digit one?
2795 if ( wxIsdigit(*p
) )
2798 year
+= *p
++ - _T('0');
2800 if ( !wxIsdigit(*p
) )
2802 // no 3 digit years please
2807 year
+= *p
++ - _T('0');
2810 if ( *p
++ != _T(' ') )
2815 // time is in the format hh:mm:ss and seconds are optional
2816 if ( !wxIsdigit(*p
) )
2821 wxDateTime_t hour
= (wxDateTime_t
)(*p
++ - _T('0'));
2823 if ( !wxIsdigit(*p
) )
2829 hour
= (wxDateTime_t
)(hour
+ (*p
++ - _T('0')));
2831 if ( *p
++ != _T(':') )
2836 if ( !wxIsdigit(*p
) )
2841 wxDateTime_t min
= (wxDateTime_t
)(*p
++ - _T('0'));
2843 if ( !wxIsdigit(*p
) )
2849 min
= (wxDateTime_t
)(min
+ *p
++ - _T('0'));
2851 wxDateTime_t sec
= 0;
2852 if ( *p
== _T(':') )
2855 if ( !wxIsdigit(*p
) )
2860 sec
= (wxDateTime_t
)(*p
++ - _T('0'));
2862 if ( !wxIsdigit(*p
) )
2868 sec
= (wxDateTime_t
)(sec
+ *p
++ - _T('0'));
2871 if ( *p
++ != _T(' ') )
2876 // and now the interesting part: the timezone
2877 int offset
wxDUMMY_INITIALIZE(0);
2878 if ( *p
== _T('-') || *p
== _T('+') )
2880 // the explicit offset given: it has the form of hhmm
2881 bool plus
= *p
++ == _T('+');
2883 if ( !wxIsdigit(*p
) || !wxIsdigit(*(p
+ 1)) )
2889 offset
= MIN_PER_HOUR
*(10*(*p
- _T('0')) + (*(p
+ 1) - _T('0')));
2893 if ( !wxIsdigit(*p
) || !wxIsdigit(*(p
+ 1)) )
2899 offset
+= 10*(*p
- _T('0')) + (*(p
+ 1) - _T('0'));
2910 // the symbolic timezone given: may be either military timezone or one
2911 // of standard abbreviations
2914 // military: Z = UTC, J unused, A = -1, ..., Y = +12
2915 static const int offsets
[26] =
2917 //A B C D E F G H I J K L M
2918 -1, -2, -3, -4, -5, -6, -7, -8, -9, 0, -10, -11, -12,
2919 //N O P R Q S T U V W Z Y Z
2920 +1, +2, +3, +4, +5, +6, +7, +8, +9, +10, +11, +12, 0
2923 if ( *p
< _T('A') || *p
> _T('Z') || *p
== _T('J') )
2925 wxLogDebug(_T("Invalid militaty timezone '%c'"), *p
);
2930 offset
= offsets
[*p
++ - _T('A')];
2936 if ( tz
== _T("UT") || tz
== _T("UTC") || tz
== _T("GMT") )
2938 else if ( tz
== _T("AST") )
2939 offset
= AST
- GMT0
;
2940 else if ( tz
== _T("ADT") )
2941 offset
= ADT
- GMT0
;
2942 else if ( tz
== _T("EST") )
2943 offset
= EST
- GMT0
;
2944 else if ( tz
== _T("EDT") )
2945 offset
= EDT
- GMT0
;
2946 else if ( tz
== _T("CST") )
2947 offset
= CST
- GMT0
;
2948 else if ( tz
== _T("CDT") )
2949 offset
= CDT
- GMT0
;
2950 else if ( tz
== _T("MST") )
2951 offset
= MST
- GMT0
;
2952 else if ( tz
== _T("MDT") )
2953 offset
= MDT
- GMT0
;
2954 else if ( tz
== _T("PST") )
2955 offset
= PST
- GMT0
;
2956 else if ( tz
== _T("PDT") )
2957 offset
= PDT
- GMT0
;
2960 wxLogDebug(_T("Unknown RFC 822 timezone '%s'"), p
);
2969 offset
*= MIN_PER_HOUR
;
2972 // the spec was correct, construct the date from the values we found
2973 Set(day
, mon
, year
, hour
, min
, sec
);
2974 MakeFromTimezone(TimeZone::Make(offset
*SEC_PER_MIN
));
2976 const size_t endpos
= p
- date
.wx_str();
2978 *end
= date
.begin() + endpos
;
2980 return date
.c_str() + endpos
;
2985 // returns the string containing strftime() format used for short dates in the
2986 // current locale or an empty string
2987 static wxString
GetLocaleDateFormat()
2991 // there is no setlocale() under Windows CE, so just always query the
2994 if ( strcmp(setlocale(LC_ALL
, NULL
), "C") != 0 )
2997 // The locale was programatically set to non-C. We assume that this was
2998 // done using wxLocale, in which case thread's current locale is also
2999 // set to correct LCID value and we can use GetLocaleInfo to determine
3000 // the correct formatting string:
3002 LCID lcid
= LOCALE_USER_DEFAULT
;
3004 LCID lcid
= GetThreadLocale();
3006 // according to MSDN 80 chars is max allowed for short date format
3008 if ( ::GetLocaleInfo(lcid
, LOCALE_SSHORTDATE
, fmt
, WXSIZEOF(fmt
)) )
3010 wxChar chLast
= _T('\0');
3011 size_t lastCount
= 0;
3012 for ( const wxChar
*p
= fmt
; /* NUL handled inside */; p
++ )
3022 // these characters come in groups, start counting them
3032 // first deal with any special characters we have had
3038 switch ( lastCount
)
3042 // these two are the same as we
3043 // don't distinguish between 1 and
3044 // 2 digits for days
3057 wxFAIL_MSG( _T("too many 'd's") );
3062 switch ( lastCount
)
3066 // as for 'd' and 'dd' above
3079 wxFAIL_MSG( _T("too many 'M's") );
3084 switch ( lastCount
)
3096 wxFAIL_MSG( _T("wrong number of 'y's") );
3101 // strftime() doesn't have era string,
3102 // ignore this format
3103 wxASSERT_MSG( lastCount
<= 2,
3104 _T("too many 'g's") );
3108 wxFAIL_MSG( _T("unreachable") );
3115 // not a special character so must be just a separator,
3117 if ( *p
!= _T('\0') )
3119 if ( *p
== _T('%') )
3121 // this one needs to be escaped
3129 if ( *p
== _T('\0') )
3133 //else: GetLocaleInfo() failed, leave fmtDate value unchanged and
3134 // try our luck with the default formats
3136 //else: default C locale, default formats should work
3141 #endif // __WINDOWS__
3144 wxDateTime::ParseFormat(const wxString
& date
,
3145 const wxString
& format
,
3146 const wxDateTime
& dateDef
,
3147 wxString::const_iterator
*end
)
3149 wxCHECK_MSG( !format
.empty(), NULL
, "format can't be empty" );
3154 // what fields have we found?
3155 bool haveWDay
= false,
3164 bool hourIsIn12hFormat
= false, // or in 24h one?
3165 isPM
= false; // AM by default
3167 // and the value of the items we have (init them to get rid of warnings)
3168 wxDateTime_t sec
= 0,
3171 WeekDay wday
= Inv_WeekDay
;
3172 wxDateTime_t yday
= 0,
3174 wxDateTime::Month mon
= Inv_Month
;
3177 const wxStringCharType
*input
= date
.wx_str();
3178 for ( wxString::const_iterator fmt
= format
.begin(); fmt
!= format
.end(); ++fmt
)
3180 if ( *fmt
!= _T('%') )
3182 if ( wxIsspace(*fmt
) )
3184 // a white space in the format string matches 0 or more white
3185 // spaces in the input
3186 while ( wxIsspace(*input
) )
3193 // any other character (not whitespace, not '%') must be
3194 // matched by itself in the input
3195 if ( *input
++ != *fmt
)
3202 // done with this format char
3206 // start of a format specification
3208 // parse the optional width
3210 while ( wxIsdigit(*++fmt
) )
3213 width
+= *fmt
- _T('0');
3216 // the default widths for the various fields
3219 switch ( (*fmt
).GetValue() )
3221 case _T('Y'): // year has 4 digits
3225 case _T('j'): // day of year has 3 digits
3226 case _T('l'): // milliseconds have 3 digits
3230 case _T('w'): // week day as number has only one
3235 // default for all other fields
3240 // then the format itself
3241 switch ( (*fmt
).GetValue() )
3243 case _T('a'): // a weekday name
3246 int flag
= *fmt
== _T('a') ? Name_Abbr
: Name_Full
;
3247 wday
= GetWeekDayFromName(GetAlphaToken(input
), flag
);
3248 if ( wday
== Inv_WeekDay
)
3257 case _T('b'): // a month name
3260 int flag
= *fmt
== _T('b') ? Name_Abbr
: Name_Full
;
3261 mon
= GetMonthFromName(GetAlphaToken(input
), flag
);
3262 if ( mon
== Inv_Month
)
3271 case _T('c'): // locale default date and time representation
3275 const wxString
inc(input
);
3277 // try the format which corresponds to ctime() output first
3278 wxString::const_iterator endc
;
3279 if ( !dt
.ParseFormat(inc
, wxS("%a %b %d %H:%M:%S %Y"), &endc
) &&
3280 !dt
.ParseFormat(inc
, wxS("%x %X"), &endc
) &&
3281 !dt
.ParseFormat(inc
, wxS("%X %x"), &endc
) )
3283 // we've tried everything and still no match
3289 haveDay
= haveMon
= haveYear
=
3290 haveHour
= haveMin
= haveSec
= true;
3300 input
+= endc
- inc
.begin();
3304 case _T('d'): // day of a month (01-31)
3305 if ( !GetNumericToken(width
, input
, &num
) ||
3306 (num
> 31) || (num
< 1) )
3312 // we can't check whether the day range is correct yet, will
3313 // do it later - assume ok for now
3315 mday
= (wxDateTime_t
)num
;
3318 case _T('H'): // hour in 24h format (00-23)
3319 if ( !GetNumericToken(width
, input
, &num
) || (num
> 23) )
3326 hour
= (wxDateTime_t
)num
;
3329 case _T('I'): // hour in 12h format (01-12)
3330 if ( !GetNumericToken(width
, input
, &num
) || !num
|| (num
> 12) )
3337 hourIsIn12hFormat
= true;
3338 hour
= (wxDateTime_t
)(num
% 12); // 12 should be 0
3341 case _T('j'): // day of the year
3342 if ( !GetNumericToken(width
, input
, &num
) || !num
|| (num
> 366) )
3349 yday
= (wxDateTime_t
)num
;
3352 case _T('m'): // month as a number (01-12)
3353 if ( !GetNumericToken(width
, input
, &num
) || !num
|| (num
> 12) )
3360 mon
= (Month
)(num
- 1);
3363 case _T('M'): // minute as a decimal number (00-59)
3364 if ( !GetNumericToken(width
, input
, &num
) || (num
> 59) )
3371 min
= (wxDateTime_t
)num
;
3374 case _T('p'): // AM or PM string
3376 wxString am
, pm
, token
= GetAlphaToken(input
);
3378 GetAmPmStrings(&am
, &pm
);
3379 if (am
.empty() && pm
.empty())
3380 return NULL
; // no am/pm strings defined
3381 if ( token
.CmpNoCase(pm
) == 0 )
3385 else if ( token
.CmpNoCase(am
) != 0 )
3393 case _T('r'): // time as %I:%M:%S %p
3396 input
= dt
.ParseFormat(input
, wxS("%I:%M:%S %p"));
3403 haveHour
= haveMin
= haveSec
= true;
3412 case _T('R'): // time as %H:%M
3415 input
= dt
.ParseFormat(input
, wxS("%H:%M"));
3422 haveHour
= haveMin
= true;
3430 case _T('S'): // second as a decimal number (00-61)
3431 if ( !GetNumericToken(width
, input
, &num
) || (num
> 61) )
3438 sec
= (wxDateTime_t
)num
;
3441 case _T('T'): // time as %H:%M:%S
3444 input
= dt
.ParseFormat(input
, _T("%H:%M:%S"));
3451 haveHour
= haveMin
= haveSec
= true;
3460 case _T('w'): // weekday as a number (0-6), Sunday = 0
3461 if ( !GetNumericToken(width
, input
, &num
) || (wday
> 6) )
3468 wday
= (WeekDay
)num
;
3471 case _T('x'): // locale default date representation
3472 #ifdef HAVE_STRPTIME
3473 // try using strptime() -- it may fail even if the input is
3474 // correct but the date is out of range, so we will fall back
3475 // to our generic code anyhow
3479 const wxStringCharType
*
3480 result
= CallStrptime(input
, "%x", &tm
);
3485 haveDay
= haveMon
= haveYear
= true;
3487 year
= 1900 + tm
.tm_year
;
3488 mon
= (Month
)tm
.tm_mon
;
3494 #endif // HAVE_STRPTIME
3502 // The above doesn't work for all locales, try to query
3503 // Windows for the right way of formatting the date:
3504 fmtDate
= GetLocaleDateFormat();
3505 if ( fmtDate
.empty() )
3506 #endif // __WINDOWS__
3508 if ( IsWestEuropeanCountry(GetCountry()) ||
3509 GetCountry() == Russia
)
3511 fmtDate
= _T("%d/%m/%y");
3512 fmtDateAlt
= _T("%m/%d/%y");
3516 fmtDate
= _T("%m/%d/%y");
3517 fmtDateAlt
= _T("%d/%m/%y");
3521 const wxString
indate(input
);
3522 wxString::const_iterator endDate
;
3523 if ( !dt
.ParseFormat(indate
, fmtDate
, &endDate
) )
3525 // try another one if we have it
3526 if ( fmtDateAlt
.empty() ||
3527 !dt
.ParseFormat(indate
, fmtDateAlt
, &endDate
) )
3543 input
+= endDate
- indate
.begin();
3548 case _T('X'): // locale default time representation
3549 #ifdef HAVE_STRPTIME
3551 // use strptime() to do it for us (FIXME !Unicode friendly)
3553 input
= CallStrptime(input
, "%X", &tm
);
3559 haveHour
= haveMin
= haveSec
= true;
3565 #else // !HAVE_STRPTIME
3566 // TODO under Win32 we can query the LOCALE_ITIME system
3567 // setting which says whether the default time format is
3570 // try to parse what follows as "%H:%M:%S" and, if this
3571 // fails, as "%I:%M:%S %p" - this should catch the most
3575 const wxStringCharType
*
3576 result
= dt
.ParseFormat(input
, wxS("%T"));
3579 result
= dt
.ParseFormat(input
, wxS("%r"));
3599 #endif // HAVE_STRPTIME/!HAVE_STRPTIME
3602 case _T('y'): // year without century (00-99)
3603 if ( !GetNumericToken(width
, input
, &num
) || (num
> 99) )
3611 // TODO should have an option for roll over date instead of
3612 // hard coding it here
3613 year
= (num
> 30 ? 1900 : 2000) + (wxDateTime_t
)num
;
3616 case _T('Y'): // year with century
3617 if ( !GetNumericToken(width
, input
, &num
) )
3624 year
= (wxDateTime_t
)num
;
3627 case _T('Z'): // timezone name
3628 wxFAIL_MSG(_T("TODO"));
3631 case _T('%'): // a percent sign
3632 if ( *input
++ != _T('%') )
3639 case 0: // the end of string
3640 wxFAIL_MSG(_T("unexpected format end"));
3644 default: // not a known format spec
3649 // format matched, try to construct a date from what we have now
3651 if ( dateDef
.IsValid() )
3653 // take this date as default
3654 tmDef
= dateDef
.GetTm();
3656 else if ( IsValid() )
3658 // if this date is valid, don't change it
3663 // no default and this date is invalid - fall back to Today()
3664 tmDef
= Today().GetTm();
3675 // TODO we don't check here that the values are consistent, if both year
3676 // day and month/day were found, we just ignore the year day and we
3677 // also always ignore the week day
3678 if ( haveMon
&& haveDay
)
3680 if ( mday
> GetNumOfDaysInMonth(tm
.year
, mon
) )
3682 wxLogDebug(_T("bad month day in wxDateTime::ParseFormat"));
3690 else if ( haveYDay
)
3692 if ( yday
> GetNumberOfDays(tm
.year
) )
3694 wxLogDebug(_T("bad year day in wxDateTime::ParseFormat"));
3699 Tm tm2
= wxDateTime(1, Jan
, tm
.year
).SetToYearDay(yday
).GetTm();
3706 if ( haveHour
&& hourIsIn12hFormat
&& isPM
)
3708 // translate to 24hour format
3711 //else: either already in 24h format or no translation needed
3731 // finally check that the week day is consistent -- if we had it
3732 if ( haveWDay
&& GetWeekDay() != wday
)
3734 wxLogDebug(_T("inconsistsnet week day in wxDateTime::ParseFormat()"));
3739 const size_t endpos
= input
- date
.wx_str();
3741 *end
= date
.begin() + endpos
;
3743 return date
.c_str() + endpos
;
3747 wxDateTime::ParseDateTime(const wxString
& date
, wxString::const_iterator
*end
)
3749 // Set to current day and hour, so strings like '14:00' becomes today at
3750 // 14, not some other random date
3751 wxDateTime dtDate
= wxDateTime::Today();
3752 wxDateTime dtTime
= wxDateTime::Today();
3754 wxString::const_iterator
3759 // If we got a date in the beginning, see if there is a time specified
3761 if ( dtDate
.ParseDate(date
, &endDate
) )
3763 // Skip spaces, as the ParseTime() function fails on spaces
3764 while ( endDate
!= date
.end() && wxIsspace(*endDate
) )
3767 const wxString
timestr(endDate
, date
.end());
3768 if ( !dtTime
.ParseTime(timestr
, &endTime
) )
3771 endBoth
= endDate
+ (endTime
- timestr
.begin());
3773 else // no date in the beginning
3775 // check if we have a time followed by a date
3776 if ( !dtTime
.ParseTime(date
, &endTime
) )
3779 while ( endTime
!= date
.end() && wxIsspace(*endTime
) )
3782 const wxString
datestr(endTime
, date
.end());
3783 if ( !dtDate
.ParseDate(datestr
, &endDate
) )
3786 endBoth
= endTime
+ (endDate
- datestr
.begin());
3789 Set(dtDate
.GetDay(), dtDate
.GetMonth(), dtDate
.GetYear(),
3790 dtTime
.GetHour(), dtTime
.GetMinute(), dtTime
.GetSecond(),
3791 dtTime
.GetMillisecond());
3793 // Return endpoint of scan
3797 return date
.c_str() + (endBoth
- date
.begin());
3801 wxDateTime::ParseDate(const wxString
& date
, wxString::const_iterator
*end
)
3803 // this is a simplified version of ParseDateTime() which understands only
3804 // "today" (for wxDate compatibility) and digits only otherwise (and not
3805 // all esoteric constructions ParseDateTime() knows about)
3807 const wxStringCharType
*p
= date
.wx_str();
3808 while ( wxIsspace(*p
) )
3811 // some special cases
3815 int dayDiffFromToday
;
3818 { wxTRANSLATE("today"), 0 },
3819 { wxTRANSLATE("yesterday"), -1 },
3820 { wxTRANSLATE("tomorrow"), 1 },
3823 for ( size_t n
= 0; n
< WXSIZEOF(literalDates
); n
++ )
3825 const wxString dateStr
= wxGetTranslation(literalDates
[n
].str
);
3826 size_t len
= dateStr
.length();
3827 if ( wxStrlen(p
) >= len
)
3829 wxString
str(p
, len
);
3830 if ( str
.CmpNoCase(dateStr
) == 0 )
3832 // nothing can follow this, so stop here
3835 int dayDiffFromToday
= literalDates
[n
].dayDiffFromToday
;
3837 if ( dayDiffFromToday
)
3839 *this += wxDateSpan::Days(dayDiffFromToday
);
3842 const size_t endpos
= p
- date
.wx_str();
3845 *end
= date
.begin() + endpos
;
3846 return date
.c_str() + endpos
;
3851 // We try to guess what we have here: for each new (numeric) token, we
3852 // determine if it can be a month, day or a year. Of course, there is an
3853 // ambiguity as some numbers may be days as well as months, so we also
3854 // have the ability to back track.
3857 bool haveDay
= false, // the months day?
3858 haveWDay
= false, // the day of week?
3859 haveMon
= false, // the month?
3860 haveYear
= false; // the year?
3862 // and the value of the items we have (init them to get rid of warnings)
3863 WeekDay wday
= Inv_WeekDay
;
3864 wxDateTime_t day
= 0;
3865 wxDateTime::Month mon
= Inv_Month
;
3868 // tokenize the string
3870 static const wxStringCharType
*dateDelimiters
= wxS(".,/-\t\r\n ");
3871 wxStringTokenizer
tok(p
, dateDelimiters
);
3872 while ( tok
.HasMoreTokens() )
3874 wxString token
= tok
.GetNextToken();
3880 if ( token
.ToULong(&val
) )
3882 // guess what this number is
3888 if ( !haveMon
&& val
> 0 && val
<= 12 )
3890 // assume it is month
3893 else // not the month
3897 // this can only be the year
3900 else // may be either day or year
3902 // use a leap year if we don't have the year yet to allow
3903 // dates like 2/29/1976 which would be rejected otherwise
3904 wxDateTime_t max_days
= (wxDateTime_t
)(
3906 ? GetNumOfDaysInMonth(haveYear
? year
: 1976, mon
)
3911 if ( (val
== 0) || (val
> (unsigned long)max_days
) )
3916 else // yes, suppose it's the day
3930 year
= (wxDateTime_t
)val
;
3939 day
= (wxDateTime_t
)val
;
3945 mon
= (Month
)(val
- 1);
3948 else // not a number
3950 // be careful not to overwrite the current mon value
3951 Month mon2
= GetMonthFromName(token
, Name_Full
| Name_Abbr
);
3952 if ( mon2
!= Inv_Month
)
3957 // but we already have a month - maybe we guessed wrong?
3960 // no need to check in month range as always < 12, but
3961 // the days are counted from 1 unlike the months
3962 day
= (wxDateTime_t
)(mon
+ 1);
3967 // could possible be the year (doesn't the year come
3968 // before the month in the japanese format?) (FIXME)
3977 else // not a valid month name
3979 WeekDay wday2
= GetWeekDayFromName(token
, Name_Full
| Name_Abbr
);
3980 if ( wday2
!= Inv_WeekDay
)
3992 else // not a valid weekday name
3995 static const char *ordinals
[] =
3997 wxTRANSLATE("first"),
3998 wxTRANSLATE("second"),
3999 wxTRANSLATE("third"),
4000 wxTRANSLATE("fourth"),
4001 wxTRANSLATE("fifth"),
4002 wxTRANSLATE("sixth"),
4003 wxTRANSLATE("seventh"),
4004 wxTRANSLATE("eighth"),
4005 wxTRANSLATE("ninth"),
4006 wxTRANSLATE("tenth"),
4007 wxTRANSLATE("eleventh"),
4008 wxTRANSLATE("twelfth"),
4009 wxTRANSLATE("thirteenth"),
4010 wxTRANSLATE("fourteenth"),
4011 wxTRANSLATE("fifteenth"),
4012 wxTRANSLATE("sixteenth"),
4013 wxTRANSLATE("seventeenth"),
4014 wxTRANSLATE("eighteenth"),
4015 wxTRANSLATE("nineteenth"),
4016 wxTRANSLATE("twentieth"),
4017 // that's enough - otherwise we'd have problems with
4018 // composite (or not) ordinals
4022 for ( n
= 0; n
< WXSIZEOF(ordinals
); n
++ )
4024 if ( token
.CmpNoCase(ordinals
[n
]) == 0 )
4030 if ( n
== WXSIZEOF(ordinals
) )
4032 // stop here - something unknown
4039 // don't try anything here (as in case of numeric day
4040 // above) - the symbolic day spec should always
4041 // precede the month/year
4047 day
= (wxDateTime_t
)(n
+ 1);
4052 nPosCur
= tok
.GetPosition();
4055 // either no more tokens or the scan was stopped by something we couldn't
4056 // parse - in any case, see if we can construct a date from what we have
4057 if ( !haveDay
&& !haveWDay
)
4059 wxLogDebug(_T("ParseDate: no day, no weekday hence no date."));
4064 if ( haveWDay
&& (haveMon
|| haveYear
|| haveDay
) &&
4065 !(haveDay
&& haveMon
&& haveYear
) )
4067 // without adjectives (which we don't support here) the week day only
4068 // makes sense completely separately or with the full date
4069 // specification (what would "Wed 1999" mean?)
4073 if ( !haveWDay
&& haveYear
&& !(haveDay
&& haveMon
) )
4075 // may be we have month and day instead of day and year?
4076 if ( haveDay
&& !haveMon
)
4080 // exchange day and month
4081 mon
= (wxDateTime::Month
)(day
- 1);
4083 // we're in the current year then
4084 if ( (year
> 0) && (year
<= (int)GetNumOfDaysInMonth(Inv_Year
, mon
)) )
4086 day
= (wxDateTime_t
)year
;
4091 //else: no, can't exchange, leave haveMon == false
4097 // if we give the year, month and day must be given too
4098 wxLogDebug(_T("ParseDate: day and month should be specified if year is."));
4106 mon
= GetCurrentMonth();
4111 year
= GetCurrentYear();
4116 // normally we check the day above but the check is optimistic in case
4117 // we find the day before its month/year so we have to redo it now
4118 if ( day
> GetNumOfDaysInMonth(year
, mon
) )
4121 Set(day
, mon
, year
);
4125 // check that it is really the same
4126 if ( GetWeekDay() != wday
)
4128 // inconsistency detected
4129 wxLogDebug(_T("ParseDate: inconsistent day/weekday."));
4139 SetToWeekDayInSameWeek(wday
);
4142 // return the pointer to the first unparsed char
4144 if ( nPosCur
&& wxStrchr(dateDelimiters
, *(p
- 1)) )
4146 // if we couldn't parse the token after the delimiter, put back the
4147 // delimiter as well
4151 const size_t endpos
= p
- date
.wx_str();
4153 *end
= date
.begin() + endpos
;
4155 return date
.c_str() + endpos
;
4159 wxDateTime::ParseTime(const wxString
& time
, wxString::const_iterator
*end
)
4161 // first try some extra things
4168 { wxTRANSLATE("noon"), 12 },
4169 { wxTRANSLATE("midnight"), 00 },
4173 for ( size_t n
= 0; n
< WXSIZEOF(stdTimes
); n
++ )
4175 wxString timeString
= wxGetTranslation(stdTimes
[n
].name
);
4176 size_t len
= timeString
.length();
4177 if ( timeString
.CmpNoCase(wxString(time
, len
)) == 0 )
4179 // casts required by DigitalMars
4180 Set(stdTimes
[n
].hour
, wxDateTime_t(0), wxDateTime_t(0));
4183 *end
= time
.begin() + len
;
4185 return time
.c_str() + len
;
4189 // try all time formats we may think about in the order from longest to
4191 static const char *timeFormats
[] =
4193 "%I:%M:%S %p", // 12hour with AM/PM
4194 "%H:%M:%S", // could be the same or 24 hour one so try it too
4195 "%I:%M %p", // 12hour with AM/PM but without seconds
4196 "%H:%M:%S", // and a possibly 24 hour version without seconds
4197 "%X", // possibly something from above or maybe something
4198 // completely different -- try it last
4200 // TODO: parse timezones
4203 for ( size_t nFmt
= 0; nFmt
< WXSIZEOF(timeFormats
); nFmt
++ )
4205 const char *result
= ParseFormat(time
, timeFormats
[nFmt
], end
);
4213 // ----------------------------------------------------------------------------
4214 // Workdays and holidays support
4215 // ----------------------------------------------------------------------------
4217 bool wxDateTime::IsWorkDay(Country
WXUNUSED(country
)) const
4219 return !wxDateTimeHolidayAuthority::IsHoliday(*this);
4222 // ============================================================================
4224 // ============================================================================
4226 wxDateSpan WXDLLIMPEXP_BASE
operator*(int n
, const wxDateSpan
& ds
)
4229 return ds1
.Multiply(n
);
4232 // ============================================================================
4234 // ============================================================================
4236 wxTimeSpan WXDLLIMPEXP_BASE
operator*(int n
, const wxTimeSpan
& ts
)
4238 return wxTimeSpan(ts
).Multiply(n
);
4241 // this enum is only used in wxTimeSpan::Format() below but we can't declare
4242 // it locally to the method as it provokes an internal compiler error in egcs
4243 // 2.91.60 when building with -O2
4254 // not all strftime(3) format specifiers make sense here because, for example,
4255 // a time span doesn't have a year nor a timezone
4257 // Here are the ones which are supported (all of them are supported by strftime
4259 // %H hour in 24 hour format
4260 // %M minute (00 - 59)
4261 // %S second (00 - 59)
4264 // Also, for MFC CTimeSpan compatibility, we support
4265 // %D number of days
4267 // And, to be better than MFC :-), we also have
4268 // %E number of wEeks
4269 // %l milliseconds (000 - 999)
4270 wxString
wxTimeSpan::Format(const wxString
& format
) const
4272 wxCHECK_MSG( !format
.empty(), wxEmptyString
,
4273 _T("NULL format in wxTimeSpan::Format") );
4276 str
.Alloc(format
.length());
4278 // Suppose we have wxTimeSpan ts(1 /* hour */, 2 /* min */, 3 /* sec */)
4280 // Then, of course, ts.Format("%H:%M:%S") must return "01:02:03", but the
4281 // question is what should ts.Format("%S") do? The code here returns "3273"
4282 // in this case (i.e. the total number of seconds, not just seconds % 60)
4283 // because, for me, this call means "give me entire time interval in
4284 // seconds" and not "give me the seconds part of the time interval"
4286 // If we agree that it should behave like this, it is clear that the
4287 // interpretation of each format specifier depends on the presence of the
4288 // other format specs in the string: if there was "%H" before "%M", we
4289 // should use GetMinutes() % 60, otherwise just GetMinutes() &c
4291 // we remember the most important unit found so far
4292 TimeSpanPart partBiggest
= Part_MSec
;
4294 for ( wxString::const_iterator pch
= format
.begin(); pch
!= format
.end(); ++pch
)
4298 if ( ch
== _T('%') )
4300 // the start of the format specification of the printf() below
4301 wxString
fmtPrefix(_T('%'));
4306 // the number of digits for the format string, 0 if unused
4307 unsigned digits
= 0;
4309 ch
= *++pch
; // get the format spec char
4313 wxFAIL_MSG( _T("invalid format character") );
4319 // skip the part below switch
4324 if ( partBiggest
< Part_Day
)
4330 partBiggest
= Part_Day
;
4335 partBiggest
= Part_Week
;
4341 if ( partBiggest
< Part_Hour
)
4345 // the sign has already been taken into account
4346 // when outputting the biggest part
4354 partBiggest
= Part_Hour
;
4361 n
= GetMilliseconds().ToLong();
4362 if ( partBiggest
< Part_MSec
)
4369 //else: no need to reset partBiggest to Part_MSec, it is
4370 // the least significant one anyhow
4377 if ( partBiggest
< Part_Min
)
4386 partBiggest
= Part_Min
;
4393 n
= GetSeconds().ToLong();
4394 if ( partBiggest
< Part_Sec
)
4403 partBiggest
= Part_Sec
;
4412 // negative numbers need one extra position for '-' display
4416 fmtPrefix
<< _T("0") << digits
;
4419 str
+= wxString::Format(fmtPrefix
+ _T("ld"), n
);
4423 // normal character, just copy
4431 // ============================================================================
4432 // wxDateTimeHolidayAuthority and related classes
4433 // ============================================================================
4435 #include "wx/arrimpl.cpp"
4437 WX_DEFINE_OBJARRAY(wxDateTimeArray
)
4439 static int wxCMPFUNC_CONV
4440 wxDateTimeCompareFunc(wxDateTime
**first
, wxDateTime
**second
)
4442 wxDateTime dt1
= **first
,
4445 return dt1
== dt2
? 0 : dt1
< dt2
? -1 : +1;
4448 // ----------------------------------------------------------------------------
4449 // wxDateTimeHolidayAuthority
4450 // ----------------------------------------------------------------------------
4452 wxHolidayAuthoritiesArray
wxDateTimeHolidayAuthority::ms_authorities
;
4455 bool wxDateTimeHolidayAuthority::IsHoliday(const wxDateTime
& dt
)
4457 size_t count
= ms_authorities
.size();
4458 for ( size_t n
= 0; n
< count
; n
++ )
4460 if ( ms_authorities
[n
]->DoIsHoliday(dt
) )
4471 wxDateTimeHolidayAuthority::GetHolidaysInRange(const wxDateTime
& dtStart
,
4472 const wxDateTime
& dtEnd
,
4473 wxDateTimeArray
& holidays
)
4475 wxDateTimeArray hol
;
4479 const size_t countAuth
= ms_authorities
.size();
4480 for ( size_t nAuth
= 0; nAuth
< countAuth
; nAuth
++ )
4482 ms_authorities
[nAuth
]->DoGetHolidaysInRange(dtStart
, dtEnd
, hol
);
4484 WX_APPEND_ARRAY(holidays
, hol
);
4487 holidays
.Sort(wxDateTimeCompareFunc
);
4489 return holidays
.size();
4493 void wxDateTimeHolidayAuthority::ClearAllAuthorities()
4495 WX_CLEAR_ARRAY(ms_authorities
);
4499 void wxDateTimeHolidayAuthority::AddAuthority(wxDateTimeHolidayAuthority
*auth
)
4501 ms_authorities
.push_back(auth
);
4504 wxDateTimeHolidayAuthority::~wxDateTimeHolidayAuthority()
4506 // required here for Darwin
4509 // ----------------------------------------------------------------------------
4510 // wxDateTimeWorkDays
4511 // ----------------------------------------------------------------------------
4513 bool wxDateTimeWorkDays::DoIsHoliday(const wxDateTime
& dt
) const
4515 wxDateTime::WeekDay wd
= dt
.GetWeekDay();
4517 return (wd
== wxDateTime::Sun
) || (wd
== wxDateTime::Sat
);
4520 size_t wxDateTimeWorkDays::DoGetHolidaysInRange(const wxDateTime
& dtStart
,
4521 const wxDateTime
& dtEnd
,
4522 wxDateTimeArray
& holidays
) const
4524 if ( dtStart
> dtEnd
)
4526 wxFAIL_MSG( _T("invalid date range in GetHolidaysInRange") );
4533 // instead of checking all days, start with the first Sat after dtStart and
4534 // end with the last Sun before dtEnd
4535 wxDateTime dtSatFirst
= dtStart
.GetNextWeekDay(wxDateTime::Sat
),
4536 dtSatLast
= dtEnd
.GetPrevWeekDay(wxDateTime::Sat
),
4537 dtSunFirst
= dtStart
.GetNextWeekDay(wxDateTime::Sun
),
4538 dtSunLast
= dtEnd
.GetPrevWeekDay(wxDateTime::Sun
),
4541 for ( dt
= dtSatFirst
; dt
<= dtSatLast
; dt
+= wxDateSpan::Week() )
4546 for ( dt
= dtSunFirst
; dt
<= dtSunLast
; dt
+= wxDateSpan::Week() )
4551 return holidays
.GetCount();
4554 // ============================================================================
4555 // other helper functions
4556 // ============================================================================
4558 // ----------------------------------------------------------------------------
4559 // iteration helpers: can be used to write a for loop over enum variable like
4561 // for ( m = wxDateTime::Jan; m < wxDateTime::Inv_Month; wxNextMonth(m) )
4562 // ----------------------------------------------------------------------------
4564 WXDLLIMPEXP_BASE
void wxNextMonth(wxDateTime::Month
& m
)
4566 wxASSERT_MSG( m
< wxDateTime::Inv_Month
, _T("invalid month") );
4568 // no wrapping or the for loop above would never end!
4569 m
= (wxDateTime::Month
)(m
+ 1);
4572 WXDLLIMPEXP_BASE
void wxPrevMonth(wxDateTime::Month
& m
)
4574 wxASSERT_MSG( m
< wxDateTime::Inv_Month
, _T("invalid month") );
4576 m
= m
== wxDateTime::Jan
? wxDateTime::Inv_Month
4577 : (wxDateTime::Month
)(m
- 1);
4580 WXDLLIMPEXP_BASE
void wxNextWDay(wxDateTime::WeekDay
& wd
)
4582 wxASSERT_MSG( wd
< wxDateTime::Inv_WeekDay
, _T("invalid week day") );
4584 // no wrapping or the for loop above would never end!
4585 wd
= (wxDateTime::WeekDay
)(wd
+ 1);
4588 WXDLLIMPEXP_BASE
void wxPrevWDay(wxDateTime::WeekDay
& wd
)
4590 wxASSERT_MSG( wd
< wxDateTime::Inv_WeekDay
, _T("invalid week day") );
4592 wd
= wd
== wxDateTime::Sun
? wxDateTime::Inv_WeekDay
4593 : (wxDateTime::WeekDay
)(wd
- 1);
4598 wxDateTime
& wxDateTime::SetFromMSWSysTime(const SYSTEMTIME
& st
)
4601 wx_static_cast(wxDateTime::Month
, wxDateTime::Jan
+ st
.wMonth
- 1),
4606 void wxDateTime::GetAsMSWSysTime(SYSTEMTIME
* st
) const
4608 const wxDateTime::Tm
tm(GetTm());
4610 st
->wYear
= (WXWORD
)tm
.year
;
4611 st
->wMonth
= (WXWORD
)(tm
.mon
- wxDateTime::Jan
+ 1);
4618 st
->wMilliseconds
= 0;
4622 #endif // wxUSE_DATETIME