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 /////////////////////////////////////////////////////////////////////////////// 
  19 // TODO: for $DEITY sake, someone please fix the #ifdef __WXWINCE__ everywhere, 
  20 //       the proper way to do it is to implement (subset of) wxStrftime() for 
  21 //       CE instead of this horror!! 
  24  * Implementation notes: 
  26  * 1. the time is stored as a 64bit integer containing the signed number of 
  27  *    milliseconds since Jan 1. 1970 (the Unix Epoch) - so it is always 
  30  * 2. the range is thus something about 580 million years, but due to current 
  31  *    algorithms limitations, only dates from Nov 24, 4714BC are handled 
  33  * 3. standard ANSI C functions are used to do time calculations whenever 
  34  *    possible, i.e. when the date is in the range Jan 1, 1970 to 2038 
  36  * 4. otherwise, the calculations are done by converting the date to/from JDN 
  37  *    first (the range limitation mentioned above comes from here: the 
  38  *    algorithm used by Scott E. Lee's code only works for positive JDNs, more 
  41  * 5. the object constructed for the given DD-MM-YYYY HH:MM:SS corresponds to 
  42  *    this moment in local time and may be converted to the object 
  43  *    corresponding to the same date/time in another time zone by using 
  46  * 6. the conversions to the current (or any other) timezone are done when the 
  47  *    internal time representation is converted to the broken-down one in 
  51 // ============================================================================ 
  53 // ============================================================================ 
  55 // ---------------------------------------------------------------------------- 
  57 // ---------------------------------------------------------------------------- 
  59 // For compilers that support precompilation, includes "wx.h". 
  60 #include "wx/wxprec.h" 
  66 #if !defined(wxUSE_DATETIME) || wxUSE_DATETIME 
  69     #include "wx/string.h" 
  74 #include "wx/thread.h" 
  75 #include "wx/tokenzr.h" 
  76 #include "wx/module.h" 
  81     #include "wx/msw/wrapwin.h" 
  88 #include "wx/datetime.h" 
  89 #include "wx/stopwatch.h"           // for wxGetLocalTimeMillis() 
  91 const long wxDateTime::TIME_T_FACTOR 
= 1000l; 
  93 #if wxUSE_EXTENDED_RTTI 
  95 template<> void wxStringReadValue(const wxString 
&s 
, wxDateTime 
&data 
) 
  97     data
.ParseFormat(s
,wxT("%Y-%m-%d %H:%M:%S")) ; 
 100 template<> void wxStringWriteValue(wxString 
&s 
, const wxDateTime 
&data 
) 
 102     s 
= data
.Format(wxT("%Y-%m-%d %H:%M:%S")) ; 
 105 wxCUSTOM_TYPE_INFO(wxDateTime
, wxToStringConverter
<wxDateTime
> , wxFromStringConverter
<wxDateTime
>) 
 110 // ---------------------------------------------------------------------------- 
 111 // conditional compilation 
 112 // ---------------------------------------------------------------------------- 
 114 #if defined(HAVE_STRPTIME) && defined(__GLIBC__) && \ 
 115         ((__GLIBC__ == 2) && (__GLIBC_MINOR__ == 0)) 
 116     // glibc 2.0.7 strptime() is broken - the following snippet causes it to 
 117     // crash (instead of just failing): 
 119     //      strncpy(buf, "Tue Dec 21 20:25:40 1999", 128); 
 120     //      strptime(buf, "%x", &tm); 
 124 #endif // broken strptime() 
 126 #if defined(HAVE_STRPTIME) && defined(__DARWIN__) && defined(_MSL_USING_MW_C_HEADERS) && _MSL_USING_MW_C_HEADERS 
 127     // configure detects strptime as linkable because it's in the OS X 
 128     // System library but MSL headers don't declare it. 
 130 //    char *strptime(const char *, const char *, struct tm *); 
 131     // However, we DON'T want to just provide it here because we would 
 132     // crash and/or overwrite data when strptime from OS X tries 
 133     // to fill in MW's struct tm which is two fields shorter (no TZ stuff) 
 134     // So for now let's just say we don't have strptime 
 138 #if defined(__MWERKS__) && wxUSE_UNICODE 
 142 #if !defined(WX_TIMEZONE) && !defined(WX_GMTOFF_IN_TM) 
 143     #if defined(__WXPALMOS__) 
 144         #define WX_GMTOFF_IN_TM 
 145     #elif defined(__BORLANDC__) || defined(__MINGW32__) || defined(__VISAGECPP__) 
 146         #define WX_TIMEZONE _timezone 
 147     #elif defined(__MWERKS__) 
 148         long wxmw_timezone 
= 28800; 
 149         #define WX_TIMEZONE wxmw_timezone 
 150     #elif defined(__DJGPP__) || defined(__WINE__) 
 151         #include <sys/timeb.h> 
 153         static long wxGetTimeZone() 
 155             static long timezone 
= MAXLONG
; // invalid timezone 
 156             if (timezone 
== MAXLONG
) 
 160                 timezone 
= tb
.timezone
; 
 164         #define WX_TIMEZONE wxGetTimeZone() 
 165     #elif defined(__DARWIN__) 
 166         #define WX_GMTOFF_IN_TM 
 167     #else // unknown platform - try timezone 
 168         #define WX_TIMEZONE timezone 
 170 #endif // !WX_TIMEZONE && !WX_GMTOFF_IN_TM 
 172 #if (!defined(HAVE_LOCALTIME_R) || !defined(HAVE_GMTIME_R)) && wxUSE_THREADS && !defined(__WINDOWS__) 
 173 static wxMutex timeLock
; 
 176 #ifndef HAVE_LOCALTIME_R 
 177 struct tm 
*wxLocaltime_r(const time_t* ticks
, struct tm
* temp
) 
 179 #if wxUSE_THREADS && !defined(__WINDOWS__) 
 180   // No need to waste time with a mutex on windows since it's using 
 181   // thread local storage for localtime anyway. 
 182   wxMutexLocker 
locker(timeLock
); 
 184   memcpy(temp
, localtime(ticks
), sizeof(struct tm
)); 
 189 #ifndef HAVE_GMTIME_R 
 190 struct tm 
*wxGmtime_r(const time_t* ticks
, struct tm
* temp
) 
 192 #if wxUSE_THREADS && !defined(__WINDOWS__) 
 193   // No need to waste time with a mutex on windows since it's 
 194   // using thread local storage for gmtime anyway. 
 195   wxMutexLocker 
locker(timeLock
); 
 197   memcpy(temp
, gmtime(ticks
), sizeof(struct tm
)); 
 202 // ---------------------------------------------------------------------------- 
 204 // ---------------------------------------------------------------------------- 
 206 // debugging helper: just a convenient replacement of wxCHECK() 
 207 #define wxDATETIME_CHECK(expr, msg) \ 
 208     wxCHECK2_MSG(expr, *this = wxInvalidDateTime; return *this, msg) 
 210 // ---------------------------------------------------------------------------- 
 212 // ---------------------------------------------------------------------------- 
 214 class wxDateTimeHolidaysModule 
: public wxModule
 
 217     virtual bool OnInit() 
 219         wxDateTimeHolidayAuthority::AddAuthority(new wxDateTimeWorkDays
); 
 224     virtual void OnExit() 
 226         wxDateTimeHolidayAuthority::ClearAllAuthorities(); 
 227         wxDateTimeHolidayAuthority::ms_authorities
.clear(); 
 231     DECLARE_DYNAMIC_CLASS(wxDateTimeHolidaysModule
) 
 234 IMPLEMENT_DYNAMIC_CLASS(wxDateTimeHolidaysModule
, wxModule
) 
 236 // ---------------------------------------------------------------------------- 
 238 // ---------------------------------------------------------------------------- 
 241 static const int MONTHS_IN_YEAR 
= 12; 
 243 static const int SEC_PER_MIN 
= 60; 
 245 static const int MIN_PER_HOUR 
= 60; 
 247 static const int HOURS_PER_DAY 
= 24; 
 249 static const long SECONDS_PER_DAY 
= 86400l; 
 251 static const int DAYS_PER_WEEK 
= 7; 
 253 static const long MILLISECONDS_PER_DAY 
= 86400000l; 
 255 // this is the integral part of JDN of the midnight of Jan 1, 1970 
 256 // (i.e. JDN(Jan 1, 1970) = 2440587.5) 
 257 static const long EPOCH_JDN 
= 2440587l; 
 259 // used only in asserts 
 261 // the date of JDN -0.5 (as we don't work with fractional parts, this is the 
 262 // reference date for us) is Nov 24, 4714BC 
 263 static const int JDN_0_YEAR 
= -4713; 
 264 static const int JDN_0_MONTH 
= wxDateTime::Nov
; 
 265 static const int JDN_0_DAY 
= 24; 
 266 #endif // __WXDEBUG__ 
 268 // the constants used for JDN calculations 
 269 static const long JDN_OFFSET         
= 32046l; 
 270 static const long DAYS_PER_5_MONTHS  
= 153l; 
 271 static const long DAYS_PER_4_YEARS   
= 1461l; 
 272 static const long DAYS_PER_400_YEARS 
= 146097l; 
 274 // this array contains the cumulated number of days in all previous months for 
 275 // normal and leap years 
 276 static const wxDateTime::wxDateTime_t gs_cumulatedDays
[2][MONTHS_IN_YEAR
] = 
 278     { 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334 }, 
 279     { 0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335 } 
 282 // ---------------------------------------------------------------------------- 
 284 // ---------------------------------------------------------------------------- 
 286 const wxChar 
* wxDefaultDateTimeFormat 
= wxT("%c"); 
 287 const wxChar 
* wxDefaultTimeSpanFormat 
= wxT("%H:%M:%S"); 
 289 // in the fine tradition of ANSI C we use our equivalent of (time_t)-1 to 
 290 // indicate an invalid wxDateTime object 
 291 const wxDateTime wxDefaultDateTime
; 
 293 wxDateTime::Country 
wxDateTime::ms_country 
= wxDateTime::Country_Unknown
; 
 295 // ---------------------------------------------------------------------------- 
 297 // ---------------------------------------------------------------------------- 
 299 // debugger helper: shows what the date really is 
 301 extern const wxChar 
*wxDumpDate(const wxDateTime
* dt
) 
 303     static wxChar buf
[128]; 
 305     wxStrcpy(buf
, dt
->Format(_T("%Y-%m-%d (%a) %H:%M:%S"))); 
 311 // get the number of days in the given month of the given year 
 313 wxDateTime::wxDateTime_t 
GetNumOfDaysInMonth(int year
, wxDateTime::Month month
) 
 315     // the number of days in month in Julian/Gregorian calendar: the first line 
 316     // is for normal years, the second one is for the leap ones 
 317     static wxDateTime::wxDateTime_t daysInMonth
[2][MONTHS_IN_YEAR
] = 
 319         { 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 }, 
 320         { 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 } 
 323     return daysInMonth
[wxDateTime::IsLeapYear(year
)][month
]; 
 326 // returns the time zone in the C sense, i.e. the difference UTC - local 
 328 static int GetTimeZone() 
 330     // set to true when the timezone is set 
 331     static bool s_timezoneSet 
= false; 
 332     static long gmtoffset 
= LONG_MAX
; // invalid timezone 
 334     // ensure that the timezone variable is set by calling wxLocaltime_r 
 335     if ( !s_timezoneSet 
) 
 337         // just call wxLocaltime_r() instead of figuring out whether this 
 338         // system supports tzset(), _tzset() or something else 
 342         wxLocaltime_r(&t
, &tm
); 
 343         s_timezoneSet 
= true; 
 345 #ifdef WX_GMTOFF_IN_TM 
 346         // note that GMT offset is the opposite of time zone and so to return 
 347         // consistent results in both WX_GMTOFF_IN_TM and !WX_GMTOFF_IN_TM 
 348         // cases we have to negate it 
 349         gmtoffset 
= -tm
.tm_gmtoff
; 
 350 #else // !WX_GMTOFF_IN_TM 
 351         gmtoffset 
= WX_TIMEZONE
; 
 352 #endif // WX_GMTOFF_IN_TM/!WX_GMTOFF_IN_TM 
 355     return (int)gmtoffset
; 
 358 // return the integral part of the JDN for the midnight of the given date (to 
 359 // get the real JDN you need to add 0.5, this is, in fact, JDN of the 
 360 // noon of the previous day) 
 361 static long GetTruncatedJDN(wxDateTime::wxDateTime_t day
, 
 362                             wxDateTime::Month mon
, 
 365     // CREDIT: code below is by Scott E. Lee (but bugs are mine) 
 367     // check the date validity 
 369       (year 
> JDN_0_YEAR
) || 
 370       ((year 
== JDN_0_YEAR
) && (mon 
> JDN_0_MONTH
)) || 
 371       ((year 
== JDN_0_YEAR
) && (mon 
== JDN_0_MONTH
) && (day 
>= JDN_0_DAY
)), 
 372       _T("date out of range - can't convert to JDN") 
 375     // make the year positive to avoid problems with negative numbers division 
 378     // months are counted from March here 
 380     if ( mon 
>= wxDateTime::Mar 
) 
 390     // now we can simply add all the contributions together 
 391     return ((year 
/ 100) * DAYS_PER_400_YEARS
) / 4 
 392             + ((year 
% 100) * DAYS_PER_4_YEARS
) / 4 
 393             + (month 
* DAYS_PER_5_MONTHS 
+ 2) / 5 
 399 // this function is a wrapper around strftime(3) adding error checking 
 400 static wxString 
CallStrftime(const wxChar 
*format
, const tm
* tm
) 
 403     // Create temp wxString here to work around mingw/cygwin bug 1046059 
 404     // http://sourceforge.net/tracker/?func=detail&atid=102435&aid=1046059&group_id=2435 
 407     if ( !wxStrftime(buf
, WXSIZEOF(buf
), format
, tm
) ) 
 409         // buffer is too small? 
 410         wxFAIL_MSG(_T("strftime() failed")); 
 420 #if wxUSE_UNIX && !defined(HAVE_STRPTIME_DECL) 
 421     // configure detected that we had strptime() but not its declaration, 
 422     // provide it ourselves 
 423     extern "C" char *strptime(const char *, const char *, struct tm 
*); 
 426 // Unicode-friendly strptime() wrapper 
 427 static const wxChar 
* 
 428 CallStrptime(const wxChar 
*input
, const char *fmt
, tm 
*tm
) 
 430     // the problem here is that strptime() returns pointer into the string we 
 431     // passed to it while we're really interested in the pointer into the 
 432     // original, Unicode, string so we try to transform the pointer back 
 434     wxCharBuffer 
inputMB(wxConvertWX2MB(input
)); 
 436     const char * const inputMB 
= input
; 
 437 #endif // Unicode/Ascii 
 439     const char *result 
= strptime(inputMB
, fmt
, tm
); 
 444     // FIXME: this is wrong in presence of surrogates &c 
 445     return input 
+ (result 
- inputMB
.data()); 
 448 #endif // Unicode/Ascii 
 451 #endif // HAVE_STRPTIME 
 453 // if year and/or month have invalid values, replace them with the current ones 
 454 static void ReplaceDefaultYearMonthWithCurrent(int *year
, 
 455                                                wxDateTime::Month 
*month
) 
 457     struct tm 
*tmNow 
= NULL
; 
 460     if ( *year 
== wxDateTime::Inv_Year 
) 
 462         tmNow 
= wxDateTime::GetTmNow(&tmstruct
); 
 464         *year 
= 1900 + tmNow
->tm_year
; 
 467     if ( *month 
== wxDateTime::Inv_Month 
) 
 470             tmNow 
= wxDateTime::GetTmNow(&tmstruct
); 
 472         *month 
= (wxDateTime::Month
)tmNow
->tm_mon
; 
 476 // fll the struct tm with default values 
 477 static void InitTm(struct tm
& tm
) 
 479     // struct tm may have etxra fields (undocumented and with unportable 
 480     // names) which, nevertheless, must be set to 0 
 481     memset(&tm
, 0, sizeof(struct tm
)); 
 483     tm
.tm_mday 
= 1;   // mday 0 is invalid 
 484     tm
.tm_year 
= 76;  // any valid year 
 485     tm
.tm_isdst 
= -1; // auto determine 
 491 // return the month if the string is a month name or Inv_Month otherwise 
 492 static wxDateTime::Month 
GetMonthFromName(const wxString
& name
, int flags
) 
 494     wxDateTime::Month mon
; 
 495     for ( mon 
= wxDateTime::Jan
; mon 
< wxDateTime::Inv_Month
; wxNextMonth(mon
) ) 
 497         // case-insensitive comparison either one of or with both abbreviated 
 499         if ( flags 
& wxDateTime::Name_Full 
) 
 501             if ( name
.CmpNoCase(wxDateTime:: 
 502                         GetMonthName(mon
, wxDateTime::Name_Full
)) == 0 ) 
 508         if ( flags 
& wxDateTime::Name_Abbr 
) 
 510             if ( name
.CmpNoCase(wxDateTime:: 
 511                         GetMonthName(mon
, wxDateTime::Name_Abbr
)) == 0 ) 
 521 // return the weekday if the string is a weekday name or Inv_WeekDay otherwise 
 522 static wxDateTime::WeekDay 
GetWeekDayFromName(const wxString
& name
, int flags
) 
 524     wxDateTime::WeekDay wd
; 
 525     for ( wd 
= wxDateTime::Sun
; wd 
< wxDateTime::Inv_WeekDay
; wxNextWDay(wd
) ) 
 527         // case-insensitive comparison either one of or with both abbreviated 
 529         if ( flags 
& wxDateTime::Name_Full 
) 
 531             if ( name
.CmpNoCase(wxDateTime:: 
 532                         GetWeekDayName(wd
, wxDateTime::Name_Full
)) == 0 ) 
 538         if ( flags 
& wxDateTime::Name_Abbr 
) 
 540             if ( name
.CmpNoCase(wxDateTime:: 
 541                         GetWeekDayName(wd
, wxDateTime::Name_Abbr
)) == 0 ) 
 552 struct tm 
*wxDateTime::GetTmNow(struct tm 
*tmstruct
) 
 554     time_t t 
= GetTimeNow(); 
 555     return wxLocaltime_r(&t
, tmstruct
); 
 558 // scans all digits (but no more than len) and returns the resulting number 
 559 static bool GetNumericToken(size_t len
, const wxChar
*& p
, unsigned long *number
) 
 563     while ( wxIsdigit(*p
) ) 
 567         if ( len 
&& ++n 
> len 
) 
 571     return !s
.empty() && s
.ToULong(number
); 
 574 // scans all alphabetic characters and returns the resulting string 
 575 static wxString 
GetAlphaToken(const wxChar
*& p
) 
 578     while ( wxIsalpha(*p
) ) 
 586 // ============================================================================ 
 587 // implementation of wxDateTime 
 588 // ============================================================================ 
 590 // ---------------------------------------------------------------------------- 
 592 // ---------------------------------------------------------------------------- 
 596     year 
= (wxDateTime_t
)wxDateTime::Inv_Year
; 
 597     mon 
= wxDateTime::Inv_Month
; 
 599     hour 
= min 
= sec 
= msec 
= 0; 
 600     wday 
= wxDateTime::Inv_WeekDay
; 
 603 wxDateTime::Tm::Tm(const struct tm
& tm
, const TimeZone
& tz
) 
 607     sec 
= (wxDateTime::wxDateTime_t
)tm
.tm_sec
; 
 608     min 
= (wxDateTime::wxDateTime_t
)tm
.tm_min
; 
 609     hour 
= (wxDateTime::wxDateTime_t
)tm
.tm_hour
; 
 610     mday 
= (wxDateTime::wxDateTime_t
)tm
.tm_mday
; 
 611     mon 
= (wxDateTime::Month
)tm
.tm_mon
; 
 612     year 
= 1900 + tm
.tm_year
; 
 613     wday 
= (wxDateTime::wxDateTime_t
)tm
.tm_wday
; 
 614     yday 
= (wxDateTime::wxDateTime_t
)tm
.tm_yday
; 
 617 bool wxDateTime::Tm::IsValid() const 
 619     // we allow for the leap seconds, although we don't use them (yet) 
 620     return (year 
!= wxDateTime::Inv_Year
) && (mon 
!= wxDateTime::Inv_Month
) && 
 621            (mday 
<= GetNumOfDaysInMonth(year
, mon
)) && 
 622            (hour 
< 24) && (min 
< 60) && (sec 
< 62) && (msec 
< 1000); 
 625 void wxDateTime::Tm::ComputeWeekDay() 
 627     // compute the week day from day/month/year: we use the dumbest algorithm 
 628     // possible: just compute our JDN and then use the (simple to derive) 
 629     // formula: weekday = (JDN + 1.5) % 7 
 630     wday 
= (wxDateTime::wxDateTime_t
)((wxDateTime::WeekDay
)(GetTruncatedJDN(mday
, mon
, year
) + 2) % 7); 
 633 void wxDateTime::Tm::AddMonths(int monDiff
) 
 635     // normalize the months field 
 636     while ( monDiff 
< -mon 
) 
 640         monDiff 
+= MONTHS_IN_YEAR
; 
 643     while ( monDiff 
+ mon 
>= MONTHS_IN_YEAR 
) 
 647         monDiff 
-= MONTHS_IN_YEAR
; 
 650     mon 
= (wxDateTime::Month
)(mon 
+ monDiff
); 
 652     wxASSERT_MSG( mon 
>= 0 && mon 
< MONTHS_IN_YEAR
, _T("logic error") ); 
 654     // NB: we don't check here that the resulting date is valid, this function 
 655     //     is private and the caller must check it if needed 
 658 void wxDateTime::Tm::AddDays(int dayDiff
) 
 660     // normalize the days field 
 661     while ( dayDiff 
+ mday 
< 1 ) 
 665         dayDiff 
+= GetNumOfDaysInMonth(year
, mon
); 
 668     mday 
= (wxDateTime::wxDateTime_t
)( mday 
+ dayDiff 
); 
 669     while ( mday 
> GetNumOfDaysInMonth(year
, mon
) ) 
 671         mday 
-= GetNumOfDaysInMonth(year
, mon
); 
 676     wxASSERT_MSG( mday 
> 0 && mday 
<= GetNumOfDaysInMonth(year
, mon
), 
 680 // ---------------------------------------------------------------------------- 
 682 // ---------------------------------------------------------------------------- 
 684 wxDateTime::TimeZone::TimeZone(wxDateTime::TZ tz
) 
 688         case wxDateTime::Local
: 
 689             // get the offset from C RTL: it returns the difference GMT-local 
 690             // while we want to have the offset _from_ GMT, hence the '-' 
 691             m_offset 
= -GetTimeZone(); 
 694         case wxDateTime::GMT_12
: 
 695         case wxDateTime::GMT_11
: 
 696         case wxDateTime::GMT_10
: 
 697         case wxDateTime::GMT_9
: 
 698         case wxDateTime::GMT_8
: 
 699         case wxDateTime::GMT_7
: 
 700         case wxDateTime::GMT_6
: 
 701         case wxDateTime::GMT_5
: 
 702         case wxDateTime::GMT_4
: 
 703         case wxDateTime::GMT_3
: 
 704         case wxDateTime::GMT_2
: 
 705         case wxDateTime::GMT_1
: 
 706             m_offset 
= -3600*(wxDateTime::GMT0 
- tz
); 
 709         case wxDateTime::GMT0
: 
 710         case wxDateTime::GMT1
: 
 711         case wxDateTime::GMT2
: 
 712         case wxDateTime::GMT3
: 
 713         case wxDateTime::GMT4
: 
 714         case wxDateTime::GMT5
: 
 715         case wxDateTime::GMT6
: 
 716         case wxDateTime::GMT7
: 
 717         case wxDateTime::GMT8
: 
 718         case wxDateTime::GMT9
: 
 719         case wxDateTime::GMT10
: 
 720         case wxDateTime::GMT11
: 
 721         case wxDateTime::GMT12
: 
 722             m_offset 
= 3600*(tz 
- wxDateTime::GMT0
); 
 725         case wxDateTime::A_CST
: 
 726             // Central Standard Time in use in Australia = UTC + 9.5 
 727             m_offset 
= 60l*(9*MIN_PER_HOUR 
+ MIN_PER_HOUR
/2); 
 731             wxFAIL_MSG( _T("unknown time zone") ); 
 735 // ---------------------------------------------------------------------------- 
 737 // ---------------------------------------------------------------------------- 
 740 bool wxDateTime::IsLeapYear(int year
, wxDateTime::Calendar cal
) 
 742     if ( year 
== Inv_Year 
) 
 743         year 
= GetCurrentYear(); 
 745     if ( cal 
== Gregorian 
) 
 747         // in Gregorian calendar leap years are those divisible by 4 except 
 748         // those divisible by 100 unless they're also divisible by 400 
 749         // (in some countries, like Russia and Greece, additional corrections 
 750         // exist, but they won't manifest themselves until 2700) 
 751         return (year 
% 4 == 0) && ((year 
% 100 != 0) || (year 
% 400 == 0)); 
 753     else if ( cal 
== Julian 
) 
 755         // in Julian calendar the rule is simpler 
 756         return year 
% 4 == 0; 
 760         wxFAIL_MSG(_T("unknown calendar")); 
 767 int wxDateTime::GetCentury(int year
) 
 769     return year 
> 0 ? year 
/ 100 : year 
/ 100 - 1; 
 773 int wxDateTime::ConvertYearToBC(int year
) 
 776     return year 
> 0 ? year 
: year 
- 1; 
 780 int wxDateTime::GetCurrentYear(wxDateTime::Calendar cal
) 
 785             return Now().GetYear(); 
 788             wxFAIL_MSG(_T("TODO")); 
 792             wxFAIL_MSG(_T("unsupported calendar")); 
 800 wxDateTime::Month 
wxDateTime::GetCurrentMonth(wxDateTime::Calendar cal
) 
 805             return Now().GetMonth(); 
 808             wxFAIL_MSG(_T("TODO")); 
 812             wxFAIL_MSG(_T("unsupported calendar")); 
 820 wxDateTime::wxDateTime_t 
wxDateTime::GetNumberOfDays(int year
, Calendar cal
) 
 822     if ( year 
== Inv_Year 
) 
 824         // take the current year if none given 
 825         year 
= GetCurrentYear(); 
 832             return IsLeapYear(year
) ? 366 : 365; 
 835             wxFAIL_MSG(_T("unsupported calendar")); 
 843 wxDateTime::wxDateTime_t 
wxDateTime::GetNumberOfDays(wxDateTime::Month month
, 
 845                                                      wxDateTime::Calendar cal
) 
 847     wxCHECK_MSG( month 
< MONTHS_IN_YEAR
, 0, _T("invalid month") ); 
 849     if ( cal 
== Gregorian 
|| cal 
== Julian 
) 
 851         if ( year 
== Inv_Year 
) 
 853             // take the current year if none given 
 854             year 
= GetCurrentYear(); 
 857         return GetNumOfDaysInMonth(year
, month
); 
 861         wxFAIL_MSG(_T("unsupported calendar")); 
 868 wxString 
wxDateTime::GetMonthName(wxDateTime::Month month
, 
 869                                   wxDateTime::NameFlags flags
) 
 871     wxCHECK_MSG( month 
!= Inv_Month
, wxEmptyString
, _T("invalid month") ); 
 873     // notice that we must set all the fields to avoid confusing libc (GNU one 
 874     // gets confused to a crash if we don't do this) 
 879     return CallStrftime(flags 
== Name_Abbr 
? _T("%b") : _T("%B"), &tm
); 
 885             ret 
= (flags 
== Name_Abbr 
? wxT("Jan"): wxT("January")); 
 888             ret 
= (flags 
== Name_Abbr 
? wxT("Feb"): wxT("Febuary")); 
 891             ret 
= (flags 
== Name_Abbr 
? wxT("Mar"): wxT("March")); 
 894             ret 
= (flags 
== Name_Abbr 
? wxT("Apr"): wxT("April")); 
 897             ret 
= (flags 
== Name_Abbr 
? wxT("May"): wxT("May")); 
 900             ret 
= (flags 
== Name_Abbr 
? wxT("Jun"): wxT("June")); 
 903             ret 
= (flags 
== Name_Abbr 
? wxT("Jul"): wxT("July")); 
 906             ret 
= (flags 
== Name_Abbr 
? wxT("Aug"): wxT("August")); 
 909             ret 
= (flags 
== Name_Abbr 
? wxT("Sep"): wxT("September")); 
 912             ret 
= (flags 
== Name_Abbr 
? wxT("Oct"): wxT("October")); 
 915             ret 
= (flags 
== Name_Abbr 
? wxT("Nov"): wxT("November")); 
 918             ret 
= (flags 
== Name_Abbr 
? wxT("Dec"): wxT("December")); 
 926 wxString 
wxDateTime::GetWeekDayName(wxDateTime::WeekDay wday
, 
 927                                     wxDateTime::NameFlags flags
) 
 929     wxCHECK_MSG( wday 
!= Inv_WeekDay
, wxEmptyString
, _T("invalid weekday") ); 
 931     // take some arbitrary Sunday (but notice that the day should be such that 
 932     // after adding wday to it below we still have a valid date, e.g. don't 
 940     // and offset it by the number of days needed to get the correct wday 
 943     // call mktime() to normalize it... 
 946     // ... and call strftime() 
 947     return CallStrftime(flags 
== Name_Abbr 
? _T("%a") : _T("%A"), &tm
); 
 953             ret 
= (flags 
== Name_Abbr 
? wxT("Sun") : wxT("Sunday")); 
 956             ret 
= (flags 
== Name_Abbr 
? wxT("Mon") : wxT("Monday")); 
 959             ret 
= (flags 
== Name_Abbr 
? wxT("Tue") : wxT("Tuesday")); 
 962             ret 
= (flags 
== Name_Abbr 
? wxT("Wed") : wxT("Wednesday")); 
 965             ret 
= (flags 
== Name_Abbr 
? wxT("Thu") : wxT("Thursday")); 
 968             ret 
= (flags 
== Name_Abbr 
? wxT("Fri") : wxT("Friday")); 
 971             ret 
= (flags 
== Name_Abbr 
? wxT("Sat") : wxT("Saturday")); 
 980 void wxDateTime::GetAmPmStrings(wxString 
*am
, wxString 
*pm
) 
 985     // @Note: Do not call 'CallStrftime' here! CallStrftime checks the return code 
 986     // and causes an assertion failed if the buffer is to small (which is good) - OR - 
 987     // if strftime does not return anything because the format string is invalid - OR - 
 988     // if there are no 'am' / 'pm' tokens defined for the current locale (which is not good). 
 989     // wxDateTime::ParseTime will try several different formats to parse the time. 
 990     // As a result, GetAmPmStrings might get called, even if the current locale 
 991     // does not define any 'am' / 'pm' tokens. In this case, wxStrftime would 
 992     // assert, even though it is a perfectly legal use. 
 995         if (wxStrftime(buffer
, sizeof(buffer
)/sizeof(wxChar
), _T("%p"), &tm
) > 0) 
 996             *am 
= wxString(buffer
); 
1003         if (wxStrftime(buffer
, sizeof(buffer
)/sizeof(wxChar
), _T("%p"), &tm
) > 0) 
1004             *pm 
= wxString(buffer
); 
1010 // ---------------------------------------------------------------------------- 
1011 // Country stuff: date calculations depend on the country (DST, work days, 
1012 // ...), so we need to know which rules to follow. 
1013 // ---------------------------------------------------------------------------- 
1016 wxDateTime::Country 
wxDateTime::GetCountry() 
1018     // TODO use LOCALE_ICOUNTRY setting under Win32 
1020     if ( ms_country 
== Country_Unknown 
) 
1022         // try to guess from the time zone name 
1023         time_t t 
= time(NULL
); 
1025         struct tm 
*tm 
= wxLocaltime_r(&t
, &tmstruct
); 
1027         wxString tz 
= CallStrftime(_T("%Z"), tm
); 
1028         if ( tz 
== _T("WET") || tz 
== _T("WEST") ) 
1032         else if ( tz 
== _T("CET") || tz 
== _T("CEST") ) 
1034             ms_country 
= Country_EEC
; 
1036         else if ( tz 
== _T("MSK") || tz 
== _T("MSD") ) 
1038             ms_country 
= Russia
; 
1040         else if ( tz 
== _T("AST") || tz 
== _T("ADT") || 
1041                   tz 
== _T("EST") || tz 
== _T("EDT") || 
1042                   tz 
== _T("CST") || tz 
== _T("CDT") || 
1043                   tz 
== _T("MST") || tz 
== _T("MDT") || 
1044                   tz 
== _T("PST") || tz 
== _T("PDT") ) 
1050             // well, choose a default one 
1062 void wxDateTime::SetCountry(wxDateTime::Country country
) 
1064     ms_country 
= country
; 
1068 bool wxDateTime::IsWestEuropeanCountry(Country country
) 
1070     if ( country 
== Country_Default 
) 
1072         country 
= GetCountry(); 
1075     return (Country_WesternEurope_Start 
<= country
) && 
1076            (country 
<= Country_WesternEurope_End
); 
1079 // ---------------------------------------------------------------------------- 
1080 // DST calculations: we use 3 different rules for the West European countries, 
1081 // USA and for the rest of the world. This is undoubtedly false for many 
1082 // countries, but I lack the necessary info (and the time to gather it), 
1083 // please add the other rules here! 
1084 // ---------------------------------------------------------------------------- 
1087 bool wxDateTime::IsDSTApplicable(int year
, Country country
) 
1089     if ( year 
== Inv_Year 
) 
1091         // take the current year if none given 
1092         year 
= GetCurrentYear(); 
1095     if ( country 
== Country_Default 
) 
1097         country 
= GetCountry(); 
1104             // DST was first observed in the US and UK during WWI, reused 
1105             // during WWII and used again since 1966 
1106             return year 
>= 1966 || 
1107                    (year 
>= 1942 && year 
<= 1945) || 
1108                    (year 
== 1918 || year 
== 1919); 
1111             // assume that it started after WWII 
1117 wxDateTime 
wxDateTime::GetBeginDST(int year
, Country country
) 
1119     if ( year 
== Inv_Year 
) 
1121         // take the current year if none given 
1122         year 
= GetCurrentYear(); 
1125     if ( country 
== Country_Default 
) 
1127         country 
= GetCountry(); 
1130     if ( !IsDSTApplicable(year
, country
) ) 
1132         return wxInvalidDateTime
; 
1137     if ( IsWestEuropeanCountry(country
) || (country 
== Russia
) ) 
1139         // DST begins at 1 a.m. GMT on the last Sunday of March 
1140         if ( !dt
.SetToLastWeekDay(Sun
, Mar
, year
) ) 
1143             wxFAIL_MSG( _T("no last Sunday in March?") ); 
1146         dt 
+= wxTimeSpan::Hours(1); 
1148         // disable DST tests because it could result in an infinite recursion! 
1151     else switch ( country 
) 
1158                     // don't know for sure - assume it was in effect all year 
1163                     dt
.Set(1, Jan
, year
); 
1167                     // DST was installed Feb 2, 1942 by the Congress 
1168                     dt
.Set(2, Feb
, year
); 
1171                     // Oil embargo changed the DST period in the US 
1173                     dt
.Set(6, Jan
, 1974); 
1177                     dt
.Set(23, Feb
, 1975); 
1181                     // before 1986, DST begun on the last Sunday of April, but 
1182                     // in 1986 Reagan changed it to begin at 2 a.m. of the 
1183                     // first Sunday in April 
1186                         if ( !dt
.SetToLastWeekDay(Sun
, Apr
, year
) ) 
1189                             wxFAIL_MSG( _T("no first Sunday in April?") ); 
1194                         if ( !dt
.SetToWeekDay(Sun
, 1, Apr
, year
) ) 
1197                             wxFAIL_MSG( _T("no first Sunday in April?") ); 
1201                     dt 
+= wxTimeSpan::Hours(2); 
1203                     // TODO what about timezone?? 
1209             // assume Mar 30 as the start of the DST for the rest of the world 
1210             // - totally bogus, of course 
1211             dt
.Set(30, Mar
, year
); 
1218 wxDateTime 
wxDateTime::GetEndDST(int year
, Country country
) 
1220     if ( year 
== Inv_Year 
) 
1222         // take the current year if none given 
1223         year 
= GetCurrentYear(); 
1226     if ( country 
== Country_Default 
) 
1228         country 
= GetCountry(); 
1231     if ( !IsDSTApplicable(year
, country
) ) 
1233         return wxInvalidDateTime
; 
1238     if ( IsWestEuropeanCountry(country
) || (country 
== Russia
) ) 
1240         // DST ends at 1 a.m. GMT on the last Sunday of October 
1241         if ( !dt
.SetToLastWeekDay(Sun
, Oct
, year
) ) 
1243             // weirder and weirder... 
1244             wxFAIL_MSG( _T("no last Sunday in October?") ); 
1247         dt 
+= wxTimeSpan::Hours(1); 
1249         // disable DST tests because it could result in an infinite recursion! 
1252     else switch ( country 
) 
1259                     // don't know for sure - assume it was in effect all year 
1263                     dt
.Set(31, Dec
, year
); 
1267                     // the time was reset after the end of the WWII 
1268                     dt
.Set(30, Sep
, year
); 
1272                     // DST ends at 2 a.m. on the last Sunday of October 
1273                     if ( !dt
.SetToLastWeekDay(Sun
, Oct
, year
) ) 
1275                         // weirder and weirder... 
1276                         wxFAIL_MSG( _T("no last Sunday in October?") ); 
1279                     dt 
+= wxTimeSpan::Hours(2); 
1281                     // TODO what about timezone?? 
1286             // assume October 26th as the end of the DST - totally bogus too 
1287             dt
.Set(26, Oct
, year
); 
1293 // ---------------------------------------------------------------------------- 
1294 // constructors and assignment operators 
1295 // ---------------------------------------------------------------------------- 
1297 // return the current time with ms precision 
1298 /* static */ wxDateTime 
wxDateTime::UNow() 
1300     return wxDateTime(wxGetLocalTimeMillis()); 
1303 // the values in the tm structure contain the local time 
1304 wxDateTime
& wxDateTime::Set(const struct tm
& tm
) 
1307     time_t timet 
= mktime(&tm2
); 
1309     if ( timet 
== (time_t)-1 ) 
1311         // mktime() rather unintuitively fails for Jan 1, 1970 if the hour is 
1312         // less than timezone - try to make it work for this case 
1313         if ( tm2
.tm_year 
== 70 && tm2
.tm_mon 
== 0 && tm2
.tm_mday 
== 1 ) 
1315             return Set((time_t)( 
1317                        tm2
.tm_hour 
* MIN_PER_HOUR 
* SEC_PER_MIN 
+ 
1318                        tm2
.tm_min 
* SEC_PER_MIN 
+ 
1322         wxFAIL_MSG( _T("mktime() failed") ); 
1324         *this = wxInvalidDateTime
; 
1334 wxDateTime
& wxDateTime::Set(wxDateTime_t hour
, 
1335                             wxDateTime_t minute
, 
1336                             wxDateTime_t second
, 
1337                             wxDateTime_t millisec
) 
1339     // we allow seconds to be 61 to account for the leap seconds, even if we 
1340     // don't use them really 
1341     wxDATETIME_CHECK( hour 
< 24 && 
1345                       _T("Invalid time in wxDateTime::Set()") ); 
1347     // get the current date from system 
1349     struct tm 
*tm 
= GetTmNow(&tmstruct
); 
1351     wxDATETIME_CHECK( tm
, _T("wxLocaltime_r() failed") ); 
1353     // make a copy so it isn't clobbered by the call to mktime() below 
1358     tm1
.tm_min 
= minute
; 
1359     tm1
.tm_sec 
= second
; 
1361     // and the DST in case it changes on this date 
1364     if ( tm2
.tm_isdst 
!= tm1
.tm_isdst 
) 
1365         tm1
.tm_isdst 
= tm2
.tm_isdst
; 
1369     // and finally adjust milliseconds 
1370     return SetMillisecond(millisec
); 
1373 wxDateTime
& wxDateTime::Set(wxDateTime_t day
, 
1377                             wxDateTime_t minute
, 
1378                             wxDateTime_t second
, 
1379                             wxDateTime_t millisec
) 
1381     wxDATETIME_CHECK( hour 
< 24 && 
1385                       _T("Invalid time in wxDateTime::Set()") ); 
1387     ReplaceDefaultYearMonthWithCurrent(&year
, &month
); 
1389     wxDATETIME_CHECK( (0 < day
) && (day 
<= GetNumberOfDays(month
, year
)), 
1390                       _T("Invalid date in wxDateTime::Set()") ); 
1392     // the range of time_t type (inclusive) 
1393     static const int yearMinInRange 
= 1970; 
1394     static const int yearMaxInRange 
= 2037; 
1396     // test only the year instead of testing for the exact end of the Unix 
1397     // time_t range - it doesn't bring anything to do more precise checks 
1398     if ( year 
>= yearMinInRange 
&& year 
<= yearMaxInRange 
) 
1400         // use the standard library version if the date is in range - this is 
1401         // probably more efficient than our code 
1403         tm
.tm_year 
= year 
- 1900; 
1409         tm
.tm_isdst 
= -1;       // mktime() will guess it 
1413         // and finally adjust milliseconds 
1415             SetMillisecond(millisec
); 
1421         // do time calculations ourselves: we want to calculate the number of 
1422         // milliseconds between the given date and the epoch 
1424         // get the JDN for the midnight of this day 
1425         m_time 
= GetTruncatedJDN(day
, month
, year
); 
1426         m_time 
-= EPOCH_JDN
; 
1427         m_time 
*= SECONDS_PER_DAY 
* TIME_T_FACTOR
; 
1429         // JDN corresponds to GMT, we take localtime 
1430         Add(wxTimeSpan(hour
, minute
, second 
+ GetTimeZone(), millisec
)); 
1436 wxDateTime
& wxDateTime::Set(double jdn
) 
1438     // so that m_time will be 0 for the midnight of Jan 1, 1970 which is jdn 
1440     jdn 
-= EPOCH_JDN 
+ 0.5; 
1442     m_time
.Assign(jdn
*MILLISECONDS_PER_DAY
); 
1444     // JDNs always are in UTC, so we don't need any adjustments for time zone 
1449 wxDateTime
& wxDateTime::ResetTime() 
1453     if ( tm
.hour 
|| tm
.min 
|| tm
.sec 
|| tm
.msec 
) 
1466 // ---------------------------------------------------------------------------- 
1467 // DOS Date and Time Format functions 
1468 // ---------------------------------------------------------------------------- 
1469 // the dos date and time value is an unsigned 32 bit value in the format: 
1470 // YYYYYYYMMMMDDDDDhhhhhmmmmmmsssss 
1472 // Y = year offset from 1980 (0-127) 
1474 // D = day of month (1-31) 
1476 // m = minute (0-59) 
1477 // s = bisecond (0-29) each bisecond indicates two seconds 
1478 // ---------------------------------------------------------------------------- 
1480 wxDateTime
& wxDateTime::SetFromDOS(unsigned long ddt
) 
1485     long year 
= ddt 
& 0xFE000000; 
1490     long month 
= ddt 
& 0x1E00000; 
1495     long day 
= ddt 
& 0x1F0000; 
1499     long hour 
= ddt 
& 0xF800; 
1503     long minute 
= ddt 
& 0x7E0; 
1507     long second 
= ddt 
& 0x1F; 
1508     tm
.tm_sec 
= second 
* 2; 
1510     return Set(mktime(&tm
)); 
1513 unsigned long wxDateTime::GetAsDOS() const 
1516     time_t ticks 
= GetTicks(); 
1518     struct tm 
*tm 
= wxLocaltime_r(&ticks
, &tmstruct
); 
1520     long year 
= tm
->tm_year
; 
1524     long month 
= tm
->tm_mon
; 
1528     long day 
= tm
->tm_mday
; 
1531     long hour 
= tm
->tm_hour
; 
1534     long minute 
= tm
->tm_min
; 
1537     long second 
= tm
->tm_sec
; 
1540     ddt 
= year 
| month 
| day 
| hour 
| minute 
| second
; 
1544 // ---------------------------------------------------------------------------- 
1545 // time_t <-> broken down time conversions 
1546 // ---------------------------------------------------------------------------- 
1548 wxDateTime::Tm 
wxDateTime::GetTm(const TimeZone
& tz
) const 
1550     wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") ); 
1552     time_t time 
= GetTicks(); 
1553     if ( time 
!= (time_t)-1 ) 
1555         // use C RTL functions 
1558         if ( tz
.GetOffset() == -GetTimeZone() ) 
1560             // we are working with local time 
1561             tm 
= wxLocaltime_r(&time
, &tmstruct
); 
1563             // should never happen 
1564             wxCHECK_MSG( tm
, Tm(), _T("wxLocaltime_r() failed") ); 
1568             time 
+= (time_t)tz
.GetOffset(); 
1569 #if defined(__VMS__) || defined(__WATCOMC__) // time is unsigned so avoid warning 
1570             int time2 
= (int) time
; 
1576                 tm 
= wxGmtime_r(&time
, &tmstruct
); 
1578                 // should never happen 
1579                 wxCHECK_MSG( tm
, Tm(), _T("wxGmtime_r() failed") ); 
1583                 tm 
= (struct tm 
*)NULL
; 
1589             // adjust the milliseconds 
1591             long timeOnly 
= (m_time 
% MILLISECONDS_PER_DAY
).ToLong(); 
1592             tm2
.msec 
= (wxDateTime_t
)(timeOnly 
% 1000); 
1595         //else: use generic code below 
1598     // remember the time and do the calculations with the date only - this 
1599     // eliminates rounding errors of the floating point arithmetics 
1601     wxLongLong timeMidnight 
= m_time 
+ tz
.GetOffset() * 1000; 
1603     long timeOnly 
= (timeMidnight 
% MILLISECONDS_PER_DAY
).ToLong(); 
1605     // we want to always have positive time and timeMidnight to be really 
1606     // the midnight before it 
1609         timeOnly 
= MILLISECONDS_PER_DAY 
+ timeOnly
; 
1612     timeMidnight 
-= timeOnly
; 
1614     // calculate the Gregorian date from JDN for the midnight of our date: 
1615     // this will yield day, month (in 1..12 range) and year 
1617     // actually, this is the JDN for the noon of the previous day 
1618     long jdn 
= (timeMidnight 
/ MILLISECONDS_PER_DAY
).ToLong() + EPOCH_JDN
; 
1620     // CREDIT: code below is by Scott E. Lee (but bugs are mine) 
1622     wxASSERT_MSG( jdn 
> -2, _T("JDN out of range") ); 
1624     // calculate the century 
1625     long temp 
= (jdn 
+ JDN_OFFSET
) * 4 - 1; 
1626     long century 
= temp 
/ DAYS_PER_400_YEARS
; 
1628     // then the year and day of year (1 <= dayOfYear <= 366) 
1629     temp 
= ((temp 
% DAYS_PER_400_YEARS
) / 4) * 4 + 3; 
1630     long year 
= (century 
* 100) + (temp 
/ DAYS_PER_4_YEARS
); 
1631     long dayOfYear 
= (temp 
% DAYS_PER_4_YEARS
) / 4 + 1; 
1633     // and finally the month and day of the month 
1634     temp 
= dayOfYear 
* 5 - 3; 
1635     long month 
= temp 
/ DAYS_PER_5_MONTHS
; 
1636     long day 
= (temp 
% DAYS_PER_5_MONTHS
) / 5 + 1; 
1638     // month is counted from March - convert to normal 
1649     // year is offset by 4800 
1652     // check that the algorithm gave us something reasonable 
1653     wxASSERT_MSG( (0 < month
) && (month 
<= 12), _T("invalid month") ); 
1654     wxASSERT_MSG( (1 <= day
) && (day 
< 32), _T("invalid day") ); 
1656     // construct Tm from these values 
1658     tm
.year 
= (int)year
; 
1659     tm
.mon 
= (Month
)(month 
- 1); // algorithm yields 1 for January, not 0 
1660     tm
.mday 
= (wxDateTime_t
)day
; 
1661     tm
.msec 
= (wxDateTime_t
)(timeOnly 
% 1000); 
1662     timeOnly 
-= tm
.msec
; 
1663     timeOnly 
/= 1000;               // now we have time in seconds 
1665     tm
.sec 
= (wxDateTime_t
)(timeOnly 
% SEC_PER_MIN
); 
1667     timeOnly 
/= SEC_PER_MIN
;        // now we have time in minutes 
1669     tm
.min 
= (wxDateTime_t
)(timeOnly 
% MIN_PER_HOUR
); 
1672     tm
.hour 
= (wxDateTime_t
)(timeOnly 
/ MIN_PER_HOUR
); 
1677 wxDateTime
& wxDateTime::SetYear(int year
) 
1679     wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") ); 
1688 wxDateTime
& wxDateTime::SetMonth(Month month
) 
1690     wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") ); 
1699 wxDateTime
& wxDateTime::SetDay(wxDateTime_t mday
) 
1701     wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") ); 
1710 wxDateTime
& wxDateTime::SetHour(wxDateTime_t hour
) 
1712     wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") ); 
1721 wxDateTime
& wxDateTime::SetMinute(wxDateTime_t min
) 
1723     wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") ); 
1732 wxDateTime
& wxDateTime::SetSecond(wxDateTime_t sec
) 
1734     wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") ); 
1743 wxDateTime
& wxDateTime::SetMillisecond(wxDateTime_t millisecond
) 
1745     wxASSERT_MSG( IsValid(), _T("invalid wxDateTime") ); 
1747     // we don't need to use GetTm() for this one 
1748     m_time 
-= m_time 
% 1000l; 
1749     m_time 
+= millisecond
; 
1754 // ---------------------------------------------------------------------------- 
1755 // wxDateTime arithmetics 
1756 // ---------------------------------------------------------------------------- 
1758 wxDateTime
& wxDateTime::Add(const wxDateSpan
& diff
) 
1762     tm
.year 
+= diff
.GetYears(); 
1763     tm
.AddMonths(diff
.GetMonths()); 
1765     // check that the resulting date is valid 
1766     if ( tm
.mday 
> GetNumOfDaysInMonth(tm
.year
, tm
.mon
) ) 
1768         // We suppose that when adding one month to Jan 31 we want to get Feb 
1769         // 28 (or 29), i.e. adding a month to the last day of the month should 
1770         // give the last day of the next month which is quite logical. 
1772         // Unfortunately, there is no logic way to understand what should 
1773         // Jan 30 + 1 month be - Feb 28 too or Feb 27 (assuming non leap year)? 
1774         // We make it Feb 28 (last day too), but it is highly questionable. 
1775         tm
.mday 
= GetNumOfDaysInMonth(tm
.year
, tm
.mon
); 
1778     tm
.AddDays(diff
.GetTotalDays()); 
1782     wxASSERT_MSG( IsSameTime(tm
), 
1783                   _T("Add(wxDateSpan) shouldn't modify time") ); 
1788 // ---------------------------------------------------------------------------- 
1789 // Weekday and monthday stuff 
1790 // ---------------------------------------------------------------------------- 
1792 // convert Sun, Mon, ..., Sat into 6, 0, ..., 5 
1793 static inline int ConvertWeekDayToMondayBase(int wd
) 
1795     return wd 
== wxDateTime::Sun 
? 6 : wd 
- 1; 
1800 wxDateTime::SetToWeekOfYear(int year
, wxDateTime_t numWeek
, WeekDay wd
) 
1802     wxASSERT_MSG( numWeek 
> 0, 
1803                   _T("invalid week number: weeks are counted from 1") ); 
1805     // Jan 4 always lies in the 1st week of the year 
1806     wxDateTime 
dt(4, Jan
, year
); 
1807     dt
.SetToWeekDayInSameWeek(wd
); 
1808     dt 
+= wxDateSpan::Weeks(numWeek 
- 1); 
1813 #if WXWIN_COMPATIBILITY_2_6 
1814 // use a separate function to avoid warnings about using deprecated 
1815 // SetToTheWeek in GetWeek below 
1817 SetToTheWeek(int year
, 
1818              wxDateTime::wxDateTime_t numWeek
, 
1819              wxDateTime::WeekDay weekday
, 
1820              wxDateTime::WeekFlags flags
) 
1822     // Jan 4 always lies in the 1st week of the year 
1823     wxDateTime 
dt(4, wxDateTime::Jan
, year
); 
1824     dt
.SetToWeekDayInSameWeek(weekday
, flags
); 
1825     dt 
+= wxDateSpan::Weeks(numWeek 
- 1); 
1830 bool wxDateTime::SetToTheWeek(wxDateTime_t numWeek
, 
1834     int year 
= GetYear(); 
1835     *this = ::SetToTheWeek(year
, numWeek
, weekday
, flags
); 
1836     if ( GetYear() != year 
) 
1838         // oops... numWeek was too big 
1845 wxDateTime 
wxDateTime::GetWeek(wxDateTime_t numWeek
, 
1847                                WeekFlags flags
) const 
1849     return ::SetToTheWeek(GetYear(), numWeek
, weekday
, flags
); 
1851 #endif // WXWIN_COMPATIBILITY_2_6 
1853 wxDateTime
& wxDateTime::SetToLastMonthDay(Month month
, 
1856     // take the current month/year if none specified 
1857     if ( year 
== Inv_Year 
) 
1859     if ( month 
== Inv_Month 
) 
1862     return Set(GetNumOfDaysInMonth(year
, month
), month
, year
); 
1865 wxDateTime
& wxDateTime::SetToWeekDayInSameWeek(WeekDay weekday
, WeekFlags flags
) 
1867     wxDATETIME_CHECK( weekday 
!= Inv_WeekDay
, _T("invalid weekday") ); 
1869     int wdayDst 
= weekday
, 
1870         wdayThis 
= GetWeekDay(); 
1871     if ( wdayDst 
== wdayThis 
) 
1877     if ( flags 
== Default_First 
) 
1879         flags 
= GetCountry() == USA 
? Sunday_First 
: Monday_First
; 
1882     // the logic below based on comparing weekday and wdayThis works if Sun (0) 
1883     // is the first day in the week, but breaks down for Monday_First case so 
1884     // we adjust the week days in this case 
1885     if ( flags 
== Monday_First 
) 
1887         if ( wdayThis 
== Sun 
) 
1889         if ( wdayDst 
== Sun 
) 
1892     //else: Sunday_First, nothing to do 
1894     // go forward or back in time to the day we want 
1895     if ( wdayDst 
< wdayThis 
) 
1897         return Subtract(wxDateSpan::Days(wdayThis 
- wdayDst
)); 
1899     else // weekday > wdayThis 
1901         return Add(wxDateSpan::Days(wdayDst 
- wdayThis
)); 
1905 wxDateTime
& wxDateTime::SetToNextWeekDay(WeekDay weekday
) 
1907     wxDATETIME_CHECK( weekday 
!= Inv_WeekDay
, _T("invalid weekday") ); 
1910     WeekDay wdayThis 
= GetWeekDay(); 
1911     if ( weekday 
== wdayThis 
) 
1916     else if ( weekday 
< wdayThis 
) 
1918         // need to advance a week 
1919         diff 
= 7 - (wdayThis 
- weekday
); 
1921     else // weekday > wdayThis 
1923         diff 
= weekday 
- wdayThis
; 
1926     return Add(wxDateSpan::Days(diff
)); 
1929 wxDateTime
& wxDateTime::SetToPrevWeekDay(WeekDay weekday
) 
1931     wxDATETIME_CHECK( weekday 
!= Inv_WeekDay
, _T("invalid weekday") ); 
1934     WeekDay wdayThis 
= GetWeekDay(); 
1935     if ( weekday 
== wdayThis 
) 
1940     else if ( weekday 
> wdayThis 
) 
1942         // need to go to previous week 
1943         diff 
= 7 - (weekday 
- wdayThis
); 
1945     else // weekday < wdayThis 
1947         diff 
= wdayThis 
- weekday
; 
1950     return Subtract(wxDateSpan::Days(diff
)); 
1953 bool wxDateTime::SetToWeekDay(WeekDay weekday
, 
1958     wxCHECK_MSG( weekday 
!= Inv_WeekDay
, false, _T("invalid weekday") ); 
1960     // we don't check explicitly that -5 <= n <= 5 because we will return false 
1961     // anyhow in such case - but may be should still give an assert for it? 
1963     // take the current month/year if none specified 
1964     ReplaceDefaultYearMonthWithCurrent(&year
, &month
); 
1968     // TODO this probably could be optimised somehow... 
1972         // get the first day of the month 
1973         dt
.Set(1, month
, year
); 
1976         WeekDay wdayFirst 
= dt
.GetWeekDay(); 
1978         // go to the first weekday of the month 
1979         int diff 
= weekday 
- wdayFirst
; 
1983         // add advance n-1 weeks more 
1986         dt 
+= wxDateSpan::Days(diff
); 
1988     else // count from the end of the month 
1990         // get the last day of the month 
1991         dt
.SetToLastMonthDay(month
, year
); 
1994         WeekDay wdayLast 
= dt
.GetWeekDay(); 
1996         // go to the last weekday of the month 
1997         int diff 
= wdayLast 
- weekday
; 
2001         // and rewind n-1 weeks from there 
2004         dt 
-= wxDateSpan::Days(diff
); 
2007     // check that it is still in the same month 
2008     if ( dt
.GetMonth() == month 
) 
2016         // no such day in this month 
2022 wxDateTime::wxDateTime_t 
GetDayOfYearFromTm(const wxDateTime::Tm
& tm
) 
2024     return (wxDateTime::wxDateTime_t
)(gs_cumulatedDays
[wxDateTime::IsLeapYear(tm
.year
)][tm
.mon
] + tm
.mday
); 
2027 wxDateTime::wxDateTime_t 
wxDateTime::GetDayOfYear(const TimeZone
& tz
) const 
2029     return GetDayOfYearFromTm(GetTm(tz
)); 
2032 wxDateTime::wxDateTime_t
 
2033 wxDateTime::GetWeekOfYear(wxDateTime::WeekFlags flags
, const TimeZone
& tz
) const 
2035     if ( flags 
== Default_First 
) 
2037         flags 
= GetCountry() == USA 
? Sunday_First 
: Monday_First
; 
2041     wxDateTime_t nDayInYear 
= GetDayOfYearFromTm(tm
); 
2043     int wdTarget 
= GetWeekDay(tz
); 
2044     int wdYearStart 
= wxDateTime(1, Jan
, GetYear()).GetWeekDay(); 
2046     if ( flags 
== Sunday_First 
) 
2048         // FIXME: First week is not calculated correctly. 
2049         week 
= (nDayInYear 
- wdTarget 
+ 7) / 7; 
2050         if ( wdYearStart 
== Wed 
|| wdYearStart 
== Thu 
) 
2053     else // week starts with monday 
2055         // adjust the weekdays to non-US style. 
2056         wdYearStart 
= ConvertWeekDayToMondayBase(wdYearStart
); 
2057         wdTarget 
= ConvertWeekDayToMondayBase(wdTarget
); 
2059         // quoting from http://www.cl.cam.ac.uk/~mgk25/iso-time.html: 
2061         //      Week 01 of a year is per definition the first week that has the 
2062         //      Thursday in this year, which is equivalent to the week that 
2063         //      contains the fourth day of January. In other words, the first 
2064         //      week of a new year is the week that has the majority of its 
2065         //      days in the new year. Week 01 might also contain days from the 
2066         //      previous year and the week before week 01 of a year is the last 
2067         //      week (52 or 53) of the previous year even if it contains days 
2068         //      from the new year. A week starts with Monday (day 1) and ends 
2069         //      with Sunday (day 7). 
2072         // if Jan 1 is Thursday or less, it is in the first week of this year 
2073         if ( wdYearStart 
< 4 ) 
2075             // count the number of entire weeks between Jan 1 and this date 
2076             week 
= (nDayInYear 
+ wdYearStart 
+ 6 - wdTarget
)/7; 
2078             // be careful to check for overflow in the next year 
2079             if ( week 
== 53 && tm
.mday 
- wdTarget 
> 28 ) 
2082         else // Jan 1 is in the last week of the previous year 
2084             // check if we happen to be at the last week of previous year: 
2085             if ( tm
.mon 
== Jan 
&& tm
.mday 
< 8 - wdYearStart 
) 
2086                 week 
= wxDateTime(31, Dec
, GetYear()-1).GetWeekOfYear(); 
2088                 week 
= (nDayInYear 
+ wdYearStart 
- 1 - wdTarget
)/7; 
2092     return (wxDateTime::wxDateTime_t
)week
; 
2095 wxDateTime::wxDateTime_t 
wxDateTime::GetWeekOfMonth(wxDateTime::WeekFlags flags
, 
2096                                                     const TimeZone
& tz
) const 
2099     wxDateTime dtMonthStart 
= wxDateTime(1, tm
.mon
, tm
.year
); 
2100     int nWeek 
= GetWeekOfYear(flags
) - dtMonthStart
.GetWeekOfYear(flags
) + 1; 
2103         // this may happen for January when Jan, 1 is the last week of the 
2105         nWeek 
+= IsLeapYear(tm
.year 
- 1) ? 53 : 52; 
2108     return (wxDateTime::wxDateTime_t
)nWeek
; 
2111 wxDateTime
& wxDateTime::SetToYearDay(wxDateTime::wxDateTime_t yday
) 
2113     int year 
= GetYear(); 
2114     wxDATETIME_CHECK( (0 < yday
) && (yday 
<= GetNumberOfDays(year
)), 
2115                       _T("invalid year day") ); 
2117     bool isLeap 
= IsLeapYear(year
); 
2118     for ( Month mon 
= Jan
; mon 
< Inv_Month
; wxNextMonth(mon
) ) 
2120         // for Dec, we can't compare with gs_cumulatedDays[mon + 1], but we 
2121         // don't need it neither - because of the CHECK above we know that 
2122         // yday lies in December then 
2123         if ( (mon 
== Dec
) || (yday 
<= gs_cumulatedDays
[isLeap
][mon 
+ 1]) ) 
2125             Set((wxDateTime::wxDateTime_t
)(yday 
- gs_cumulatedDays
[isLeap
][mon
]), mon
, year
); 
2134 // ---------------------------------------------------------------------------- 
2135 // Julian day number conversion and related stuff 
2136 // ---------------------------------------------------------------------------- 
2138 double wxDateTime::GetJulianDayNumber() const 
2140     return m_time
.ToDouble() / MILLISECONDS_PER_DAY 
+ EPOCH_JDN 
+ 0.5; 
2143 double wxDateTime::GetRataDie() const 
2145     // March 1 of the year 0 is Rata Die day -306 and JDN 1721119.5 
2146     return GetJulianDayNumber() - 1721119.5 - 306; 
2149 // ---------------------------------------------------------------------------- 
2150 // timezone and DST stuff 
2151 // ---------------------------------------------------------------------------- 
2153 int wxDateTime::IsDST(wxDateTime::Country country
) const 
2155     wxCHECK_MSG( country 
== Country_Default
, -1, 
2156                  _T("country support not implemented") ); 
2158     // use the C RTL for the dates in the standard range 
2159     time_t timet 
= GetTicks(); 
2160     if ( timet 
!= (time_t)-1 ) 
2163         tm 
*tm 
= wxLocaltime_r(&timet
, &tmstruct
); 
2165         wxCHECK_MSG( tm
, -1, _T("wxLocaltime_r() failed") ); 
2167         return tm
->tm_isdst
; 
2171         int year 
= GetYear(); 
2173         if ( !IsDSTApplicable(year
, country
) ) 
2175             // no DST time in this year in this country 
2179         return IsBetween(GetBeginDST(year
, country
), GetEndDST(year
, country
)); 
2183 wxDateTime
& wxDateTime::MakeTimezone(const TimeZone
& tz
, bool noDST
) 
2185     long secDiff 
= GetTimeZone() + tz
.GetOffset(); 
2187     // we need to know whether DST is or not in effect for this date unless 
2188     // the test disabled by the caller 
2189     if ( !noDST 
&& (IsDST() == 1) ) 
2191         // FIXME we assume that the DST is always shifted by 1 hour 
2195     return Add(wxTimeSpan::Seconds(secDiff
)); 
2198 wxDateTime
& wxDateTime::MakeFromTimezone(const TimeZone
& tz
, bool noDST
) 
2200     long secDiff 
= GetTimeZone() + tz
.GetOffset(); 
2202     // we need to know whether DST is or not in effect for this date unless 
2203     // the test disabled by the caller 
2204     if ( !noDST 
&& (IsDST() == 1) ) 
2206         // FIXME we assume that the DST is always shifted by 1 hour 
2210     return Subtract(wxTimeSpan::Seconds(secDiff
)); 
2213 // ---------------------------------------------------------------------------- 
2214 // wxDateTime to/from text representations 
2215 // ---------------------------------------------------------------------------- 
2217 wxString 
wxDateTime::Format(const wxChar 
*format
, const TimeZone
& tz
) const 
2219     wxCHECK_MSG( format
, wxEmptyString
, _T("NULL format in wxDateTime::Format") ); 
2221     // we have to use our own implementation if the date is out of range of 
2222     // strftime() or if we use non standard specificators 
2223     time_t time 
= GetTicks(); 
2224     if ( (time 
!= (time_t)-1) && !wxStrstr(format
, _T("%l")) ) 
2229         if ( tz
.GetOffset() == -GetTimeZone() ) 
2231             // we are working with local time 
2232             tm 
= wxLocaltime_r(&time
, &tmstruct
); 
2234             // should never happen 
2235             wxCHECK_MSG( tm
, wxEmptyString
, _T("wxLocaltime_r() failed") ); 
2239             time 
+= (int)tz
.GetOffset(); 
2241 #if defined(__VMS__) || defined(__WATCOMC__) // time is unsigned so avoid warning 
2242             int time2 
= (int) time
; 
2248                 tm 
= wxGmtime_r(&time
, &tmstruct
); 
2250                 // should never happen 
2251                 wxCHECK_MSG( tm
, wxEmptyString
, _T("wxGmtime_r() failed") ); 
2255                 tm 
= (struct tm 
*)NULL
; 
2259     //Windows CE doesn't support strftime or wcsftime, so we use the generic implementation 
2262             return CallStrftime(format
, tm
); 
2265         //else: use generic code below 
2268     // we only parse ANSI C format specifications here, no POSIX 2 
2269     // complications, no GNU extensions but we do add support for a "%l" format 
2270     // specifier allowing to get the number of milliseconds 
2273     // used for calls to strftime() when we only deal with time 
2274     struct tm tmTimeOnly
; 
2275     tmTimeOnly
.tm_hour 
= tm
.hour
; 
2276     tmTimeOnly
.tm_min 
= tm
.min
; 
2277     tmTimeOnly
.tm_sec 
= tm
.sec
; 
2278     tmTimeOnly
.tm_wday 
= 0; 
2279     tmTimeOnly
.tm_yday 
= 0; 
2280     tmTimeOnly
.tm_mday 
= 1;         // any date will do 
2281     tmTimeOnly
.tm_mon 
= 0; 
2282     tmTimeOnly
.tm_year 
= 76; 
2283     tmTimeOnly
.tm_isdst 
= 0;        // no DST, we adjust for tz ourselves 
2285     wxString tmp
, res
, fmt
; 
2286     for ( const wxChar 
*p 
= format
; *p
; p
++ ) 
2288         if ( *p 
!= _T('%') ) 
2296         // set the default format 
2299             case _T('Y'):               // year has 4 digits 
2303             case _T('j'):               // day of year has 3 digits 
2304             case _T('l'):               // milliseconds have 3 digits 
2308             case _T('w'):               // week day as number has only one 
2313                 // it's either another valid format specifier in which case 
2314                 // the format is "%02d" (for all the rest) or we have the 
2315                 // field width preceding the format in which case it will 
2316                 // override the default format anyhow 
2320         bool restart 
= true; 
2325             // start of the format specification 
2328                 case _T('a'):       // a weekday name 
2330                     // second parameter should be true for abbreviated names 
2331                     res 
+= GetWeekDayName(tm
.GetWeekDay(), 
2332                                           *p 
== _T('a') ? Name_Abbr 
: Name_Full
); 
2335                 case _T('b'):       // a month name 
2337                     res 
+= GetMonthName(tm
.mon
, 
2338                                         *p 
== _T('b') ? Name_Abbr 
: Name_Full
); 
2341                 case _T('c'):       // locale default date and time  representation 
2342                 case _T('x'):       // locale default date representation 
2345                     // the problem: there is no way to know what do these format 
2346                     // specifications correspond to for the current locale. 
2348                     // the solution: use a hack and still use strftime(): first 
2349                     // find the YEAR which is a year in the strftime() range (1970 
2350                     // - 2038) whose Jan 1 falls on the same week day as the Jan 1 
2351                     // of the real year. Then make a copy of the format and 
2352                     // replace all occurrences of YEAR in it with some unique 
2353                     // string not appearing anywhere else in it, then use 
2354                     // strftime() to format the date in year YEAR and then replace 
2355                     // YEAR back by the real year and the unique replacement 
2356                     // string back with YEAR. Notice that "all occurrences of YEAR" 
2357                     // means all occurrences of 4 digit as well as 2 digit form! 
2359                     // the bugs: we assume that neither of %c nor %x contains any 
2360                     // fields which may change between the YEAR and real year. For 
2361                     // example, the week number (%U, %W) and the day number (%j) 
2362                     // will change if one of these years is leap and the other one 
2365                         // find the YEAR: normally, for any year X, Jan 1 or the 
2366                         // year X + 28 is the same weekday as Jan 1 of X (because 
2367                         // the weekday advances by 1 for each normal X and by 2 
2368                         // for each leap X, hence by 5 every 4 years or by 35 
2369                         // which is 0 mod 7 every 28 years) but this rule breaks 
2370                         // down if there are years between X and Y which are 
2371                         // divisible by 4 but not leap (i.e. divisible by 100 but 
2372                         // not 400), hence the correction. 
2374                         int yearReal 
= GetYear(tz
); 
2375                         int mod28 
= yearReal 
% 28; 
2377                         // be careful to not go too far - we risk to leave the 
2382                             year 
= 1988 + mod28
;      // 1988 == 0 (mod 28) 
2386                             year 
= 1970 + mod28 
- 10; // 1970 == 10 (mod 28) 
2389                         int nCentury 
= year 
/ 100, 
2390                             nCenturyReal 
= yearReal 
/ 100; 
2392                         // need to adjust for the years divisble by 400 which are 
2393                         // not leap but are counted like leap ones if we just take 
2394                         // the number of centuries in between for nLostWeekDays 
2395                         int nLostWeekDays 
= (nCentury 
- nCenturyReal
) - 
2396                                             (nCentury 
/ 4 - nCenturyReal 
/ 4); 
2398                         // we have to gain back the "lost" weekdays: note that the 
2399                         // effect of this loop is to not do anything to 
2400                         // nLostWeekDays (which we won't use any more), but to 
2401                         // (indirectly) set the year correctly 
2402                         while ( (nLostWeekDays 
% 7) != 0 ) 
2404                             nLostWeekDays 
+= year
++ % 4 ? 1 : 2; 
2407                         // Keep year below 2000 so the 2digit year number 
2408                         // can never match the month or day of the month 
2409                         if (year
>=2000) year
-=28; 
2410                         // at any rate, we couldn't go further than 1988 + 9 + 28! 
2411                         wxASSERT_MSG( year 
< 2030, 
2412                                       _T("logic error in wxDateTime::Format") ); 
2414                         wxString strYear
, strYear2
; 
2415                         strYear
.Printf(_T("%d"), year
); 
2416                         strYear2
.Printf(_T("%d"), year 
% 100); 
2418                         // find four strings not occurring in format (this is surely 
2419                         // not the optimal way of doing it... improvements welcome!) 
2420                         wxString fmt2 
= format
; 
2421                         wxString replacement
,replacement2
,replacement3
,replacement4
; 
2422                         for (int rnr
=1; rnr
<5 ; rnr
++) 
2424                             wxString r 
= (wxChar
)-rnr
; 
2425                             while ( fmt2
.Find(r
) != wxNOT_FOUND 
) 
2432                                 case 1: replacement
=r
; break; 
2433                                 case 2: replacement2
=r
; break; 
2434                                 case 3: replacement3
=r
; break; 
2435                                 case 4: replacement4
=r
; break; 
2438                         // replace all occurrences of year with it 
2439                         bool wasReplaced 
= fmt2
.Replace(strYear
, replacement
) > 0; 
2440                         // evaluation order ensures we always attempt the replacement. 
2441                         wasReplaced 
= (fmt2
.Replace(strYear2
, replacement2
) > 0) | wasReplaced 
; 
2443                         // use strftime() to format the same date but in supported 
2446                         // NB: we assume that strftime() doesn't check for the 
2447                         //     date validity and will happily format the date 
2448                         //     corresponding to Feb 29 of a non leap year (which 
2449                         //     may happen if yearReal was leap and year is not) 
2450                         struct tm tmAdjusted
; 
2452                         tmAdjusted
.tm_hour 
= tm
.hour
; 
2453                         tmAdjusted
.tm_min 
= tm
.min
; 
2454                         tmAdjusted
.tm_sec 
= tm
.sec
; 
2455                         tmAdjusted
.tm_wday 
= tm
.GetWeekDay(); 
2456                         tmAdjusted
.tm_yday 
= GetDayOfYear(); 
2457                         tmAdjusted
.tm_mday 
= tm
.mday
; 
2458                         tmAdjusted
.tm_mon 
= tm
.mon
; 
2459                         tmAdjusted
.tm_year 
= year 
- 1900; 
2460                         tmAdjusted
.tm_isdst 
= 0; // no DST, already adjusted 
2461                         wxString str 
= CallStrftime(*p 
== _T('c') ? _T("%c") 
2465                         // now replace the occurrence of 1999 with the real year 
2466                         // we do this in two stages to stop the 2 digit year 
2467                         // matching any substring of the 4 digit year. 
2468                         // Any day,month hours and minutes components should be safe due 
2469                         // to ensuring the range of the years. 
2470                         wxString strYearReal
, strYearReal2
; 
2471                         strYearReal
.Printf(_T("%04d"), yearReal
); 
2472                         strYearReal2
.Printf(_T("%02d"), yearReal 
% 100); 
2473                         str
.Replace(strYear
, replacement3
); 
2474                         str
.Replace(strYear2
,replacement4
); 
2475                         str
.Replace(replacement3
, strYearReal
); 
2476                         str
.Replace(replacement4
, strYearReal2
); 
2478                         // and replace back all occurrences of replacement string 
2481                             str
.Replace(replacement2
, strYear2
); 
2482                             str
.Replace(replacement
, strYear
); 
2488                     //Use "%m/%d/%y %H:%M:%S" format instead 
2489                     res 
+= wxString::Format(wxT("%02d/%02d/%04d %02d:%02d:%02d"), 
2490                             tm
.mon
+1,tm
.mday
, tm
.year
, tm
.hour
, tm
.min
, tm
.sec
); 
2494                 case _T('d'):       // day of a month (01-31) 
2495                     res 
+= wxString::Format(fmt
, tm
.mday
); 
2498                 case _T('H'):       // hour in 24h format (00-23) 
2499                     res 
+= wxString::Format(fmt
, tm
.hour
); 
2502                 case _T('I'):       // hour in 12h format (01-12) 
2504                         // 24h -> 12h, 0h -> 12h too 
2505                         int hour12 
= tm
.hour 
> 12 ? tm
.hour 
- 12 
2506                                                   : tm
.hour 
? tm
.hour 
: 12; 
2507                         res 
+= wxString::Format(fmt
, hour12
); 
2511                 case _T('j'):       // day of the year 
2512                     res 
+= wxString::Format(fmt
, GetDayOfYear(tz
)); 
2515                 case _T('l'):       // milliseconds (NOT STANDARD) 
2516                     res 
+= wxString::Format(fmt
, GetMillisecond(tz
)); 
2519                 case _T('m'):       // month as a number (01-12) 
2520                     res 
+= wxString::Format(fmt
, tm
.mon 
+ 1); 
2523                 case _T('M'):       // minute as a decimal number (00-59) 
2524                     res 
+= wxString::Format(fmt
, tm
.min
); 
2527                 case _T('p'):       // AM or PM string 
2529                     res 
+= CallStrftime(_T("%p"), &tmTimeOnly
); 
2531                     res 
+= (tmTimeOnly
.tm_hour 
> 12) ? wxT("pm") : wxT("am"); 
2535                 case _T('S'):       // second as a decimal number (00-61) 
2536                     res 
+= wxString::Format(fmt
, tm
.sec
); 
2539                 case _T('U'):       // week number in the year (Sunday 1st week day) 
2540                     res 
+= wxString::Format(fmt
, GetWeekOfYear(Sunday_First
, tz
)); 
2543                 case _T('W'):       // week number in the year (Monday 1st week day) 
2544                     res 
+= wxString::Format(fmt
, GetWeekOfYear(Monday_First
, tz
)); 
2547                 case _T('w'):       // weekday as a number (0-6), Sunday = 0 
2548                     res 
+= wxString::Format(fmt
, tm
.GetWeekDay()); 
2551                 // case _T('x'): -- handled with "%c" 
2553                 case _T('X'):       // locale default time representation 
2554                     // just use strftime() to format the time for us 
2556                     res 
+= CallStrftime(_T("%X"), &tmTimeOnly
); 
2558                     res 
+= wxString::Format(wxT("%02d:%02d:%02d"),tm
.hour
, tm
.min
, tm
.sec
); 
2562                 case _T('y'):       // year without century (00-99) 
2563                     res 
+= wxString::Format(fmt
, tm
.year 
% 100); 
2566                 case _T('Y'):       // year with century 
2567                     res 
+= wxString::Format(fmt
, tm
.year
); 
2570                 case _T('Z'):       // timezone name 
2572                     res 
+= CallStrftime(_T("%Z"), &tmTimeOnly
); 
2577                     // is it the format width? 
2579                     while ( *p 
== _T('-') || *p 
== _T('+') || 
2580                             *p 
== _T(' ') || wxIsdigit(*p
) ) 
2587                         // we've only got the flags and width so far in fmt 
2588                         fmt
.Prepend(_T('%')); 
2589                         fmt
.Append(_T('d')); 
2596                     // no, it wasn't the width 
2597                     wxFAIL_MSG(_T("unknown format specificator")); 
2599                     // fall through and just copy it nevertheless 
2601                 case _T('%'):       // a percent sign 
2605                 case 0:             // the end of string 
2606                     wxFAIL_MSG(_T("missing format at the end of string")); 
2608                     // just put the '%' which was the last char in format 
2618 // this function parses a string in (strict) RFC 822 format: see the section 5 
2619 // of the RFC for the detailed description, but briefly it's something of the 
2620 // form "Sat, 18 Dec 1999 00:48:30 +0100" 
2622 // this function is "strict" by design - it must reject anything except true 
2623 // RFC822 time specs. 
2625 // TODO a great candidate for using reg exps 
2626 const wxChar 
*wxDateTime::ParseRfc822Date(const wxChar
* date
) 
2628     wxCHECK_MSG( date
, (wxChar 
*)NULL
, _T("NULL pointer in wxDateTime::Parse") ); 
2630     const wxChar 
*p 
= date
; 
2631     const wxChar 
*comma 
= wxStrchr(p
, _T(',')); 
2634         // the part before comma is the weekday 
2636         // skip it for now - we don't use but might check that it really 
2637         // corresponds to the specfied date 
2640         if ( *p 
!= _T(' ') ) 
2642             wxLogDebug(_T("no space after weekday in RFC822 time spec")); 
2644             return (wxChar 
*)NULL
; 
2650     // the following 1 or 2 digits are the day number 
2651     if ( !wxIsdigit(*p
) ) 
2653         wxLogDebug(_T("day number expected in RFC822 time spec, none found")); 
2655         return (wxChar 
*)NULL
; 
2658     wxDateTime_t day 
= (wxDateTime_t
)(*p
++ - _T('0')); 
2659     if ( wxIsdigit(*p
) ) 
2662         day 
= (wxDateTime_t
)(day 
+ (*p
++ - _T('0'))); 
2665     if ( *p
++ != _T(' ') ) 
2667         return (wxChar 
*)NULL
; 
2670     // the following 3 letters specify the month 
2671     wxString 
monName(p
, 3); 
2673     if ( monName 
== _T("Jan") ) 
2675     else if ( monName 
== _T("Feb") ) 
2677     else if ( monName 
== _T("Mar") ) 
2679     else if ( monName 
== _T("Apr") ) 
2681     else if ( monName 
== _T("May") ) 
2683     else if ( monName 
== _T("Jun") ) 
2685     else if ( monName 
== _T("Jul") ) 
2687     else if ( monName 
== _T("Aug") ) 
2689     else if ( monName 
== _T("Sep") ) 
2691     else if ( monName 
== _T("Oct") ) 
2693     else if ( monName 
== _T("Nov") ) 
2695     else if ( monName 
== _T("Dec") ) 
2699         wxLogDebug(_T("Invalid RFC 822 month name '%s'"), monName
.c_str()); 
2701         return (wxChar 
*)NULL
; 
2706     if ( *p
++ != _T(' ') ) 
2708         return (wxChar 
*)NULL
; 
2712     if ( !wxIsdigit(*p
) ) 
2715         return (wxChar 
*)NULL
; 
2718     int year 
= *p
++ - _T('0'); 
2720     if ( !wxIsdigit(*p
) ) 
2722         // should have at least 2 digits in the year 
2723         return (wxChar 
*)NULL
; 
2727     year 
+= *p
++ - _T('0'); 
2729     // is it a 2 digit year (as per original RFC 822) or a 4 digit one? 
2730     if ( wxIsdigit(*p
) ) 
2733         year 
+= *p
++ - _T('0'); 
2735         if ( !wxIsdigit(*p
) ) 
2737             // no 3 digit years please 
2738             return (wxChar 
*)NULL
; 
2742         year 
+= *p
++ - _T('0'); 
2745     if ( *p
++ != _T(' ') ) 
2747         return (wxChar 
*)NULL
; 
2750     // time is in the format hh:mm:ss and seconds are optional 
2751     if ( !wxIsdigit(*p
) ) 
2753         return (wxChar 
*)NULL
; 
2756     wxDateTime_t hour 
= (wxDateTime_t
)(*p
++ - _T('0')); 
2758     if ( !wxIsdigit(*p
) ) 
2760         return (wxChar 
*)NULL
; 
2764     hour 
= (wxDateTime_t
)(hour 
+ (*p
++ - _T('0'))); 
2766     if ( *p
++ != _T(':') ) 
2768         return (wxChar 
*)NULL
; 
2771     if ( !wxIsdigit(*p
) ) 
2773         return (wxChar 
*)NULL
; 
2776     wxDateTime_t min 
= (wxDateTime_t
)(*p
++ - _T('0')); 
2778     if ( !wxIsdigit(*p
) ) 
2780         return (wxChar 
*)NULL
; 
2784     min 
= (wxDateTime_t
)(min 
+ *p
++ - _T('0')); 
2786     wxDateTime_t sec 
= 0; 
2787     if ( *p
++ == _T(':') ) 
2789         if ( !wxIsdigit(*p
) ) 
2791             return (wxChar 
*)NULL
; 
2794         sec 
= (wxDateTime_t
)(*p
++ - _T('0')); 
2796         if ( !wxIsdigit(*p
) ) 
2798             return (wxChar 
*)NULL
; 
2802         sec 
= (wxDateTime_t
)(sec 
+ *p
++ - _T('0')); 
2805     if ( *p
++ != _T(' ') ) 
2807         return (wxChar 
*)NULL
; 
2810     // and now the interesting part: the timezone 
2811     int offset 
wxDUMMY_INITIALIZE(0); 
2812     if ( *p 
== _T('-') || *p 
== _T('+') ) 
2814         // the explicit offset given: it has the form of hhmm 
2815         bool plus 
= *p
++ == _T('+'); 
2817         if ( !wxIsdigit(*p
) || !wxIsdigit(*(p 
+ 1)) ) 
2819             return (wxChar 
*)NULL
; 
2823         offset 
= MIN_PER_HOUR
*(10*(*p 
- _T('0')) + (*(p 
+ 1) - _T('0'))); 
2827         if ( !wxIsdigit(*p
) || !wxIsdigit(*(p 
+ 1)) ) 
2829             return (wxChar 
*)NULL
; 
2833         offset 
+= 10*(*p 
- _T('0')) + (*(p 
+ 1) - _T('0')); 
2844         // the symbolic timezone given: may be either military timezone or one 
2845         // of standard abbreviations 
2848             // military: Z = UTC, J unused, A = -1, ..., Y = +12 
2849             static const int offsets
[26] = 
2851                 //A  B   C   D   E   F   G   H   I    J    K    L    M 
2852                 -1, -2, -3, -4, -5, -6, -7, -8, -9,   0, -10, -11, -12, 
2853                 //N  O   P   R   Q   S   T   U   V    W    Z    Y    Z 
2854                 +1, +2, +3, +4, +5, +6, +7, +8, +9, +10, +11, +12, 0 
2857             if ( *p 
< _T('A') || *p 
> _T('Z') || *p 
== _T('J') ) 
2859                 wxLogDebug(_T("Invalid militaty timezone '%c'"), *p
); 
2861                 return (wxChar 
*)NULL
; 
2864             offset 
= offsets
[*p
++ - _T('A')]; 
2870             if ( tz 
== _T("UT") || tz 
== _T("UTC") || tz 
== _T("GMT") ) 
2872             else if ( tz 
== _T("AST") ) 
2873                 offset 
= AST 
- GMT0
; 
2874             else if ( tz 
== _T("ADT") ) 
2875                 offset 
= ADT 
- GMT0
; 
2876             else if ( tz 
== _T("EST") ) 
2877                 offset 
= EST 
- GMT0
; 
2878             else if ( tz 
== _T("EDT") ) 
2879                 offset 
= EDT 
- GMT0
; 
2880             else if ( tz 
== _T("CST") ) 
2881                 offset 
= CST 
- GMT0
; 
2882             else if ( tz 
== _T("CDT") ) 
2883                 offset 
= CDT 
- GMT0
; 
2884             else if ( tz 
== _T("MST") ) 
2885                 offset 
= MST 
- GMT0
; 
2886             else if ( tz 
== _T("MDT") ) 
2887                 offset 
= MDT 
- GMT0
; 
2888             else if ( tz 
== _T("PST") ) 
2889                 offset 
= PST 
- GMT0
; 
2890             else if ( tz 
== _T("PDT") ) 
2891                 offset 
= PDT 
- GMT0
; 
2894                 wxLogDebug(_T("Unknown RFC 822 timezone '%s'"), p
); 
2896                 return (wxChar 
*)NULL
; 
2903         offset 
*= MIN_PER_HOUR
; 
2906     // the spec was correct, construct the date from the values we found 
2907     Set(day
, mon
, year
, hour
, min
, sec
); 
2908     MakeFromTimezone(TimeZone((wxDateTime_t
)(offset
*SEC_PER_MIN
))); 
2915 // returns the string containing strftime() format used for short dates in the 
2916 // current locale or an empty string 
2917 static wxString 
GetLocaleDateFormat() 
2921     // there is no setlocale() under Windows CE, so just always query the 
2924     if ( strcmp(setlocale(LC_ALL
, NULL
), "C") != 0 ) 
2927         // The locale was programatically set to non-C. We assume that this was 
2928         // done using wxLocale, in which case thread's current locale is also 
2929         // set to correct LCID value and we can use GetLocaleInfo to determine 
2930         // the correct formatting string: 
2932         LCID lcid 
= LOCALE_USER_DEFAULT
; 
2934         LCID lcid 
= GetThreadLocale(); 
2936         // according to MSDN 80 chars is max allowed for short date format 
2938         if ( ::GetLocaleInfo(lcid
, LOCALE_SSHORTDATE
, fmt
, WXSIZEOF(fmt
)) ) 
2940             wxChar chLast 
= _T('\0'); 
2941             size_t lastCount 
= 0; 
2942             for ( const wxChar 
*p 
= fmt
; /* NUL handled inside */; p
++ ) 
2952                     // these characters come in groups, start counting them 
2962                         // first deal with any special characters we have had 
2968                                     switch ( lastCount 
) 
2972                                             // these two are the same as we 
2973                                             // don't distinguish between 1 and 
2974                                             // 2 digits for days 
2987                                             wxFAIL_MSG( _T("too many 'd's") ); 
2992                                     switch ( lastCount 
) 
2996                                             // as for 'd' and 'dd' above 
3009                                             wxFAIL_MSG( _T("too many 'M's") ); 
3014                                     switch ( lastCount 
) 
3026                                             wxFAIL_MSG( _T("wrong number of 'y's") ); 
3031                                     // strftime() doesn't have era string, 
3032                                     // ignore this format 
3033                                     wxASSERT_MSG( lastCount 
<= 2, 
3034                                                   _T("too many 'g's") ); 
3038                                     wxFAIL_MSG( _T("unreachable") ); 
3045                         // not a special character so must be just a separator, 
3047                         if ( *p 
!= _T('\0') ) 
3049                             if ( *p 
== _T('%') ) 
3051                                 // this one needs to be escaped 
3059                 if ( *p 
== _T('\0') ) 
3063         //else: GetLocaleInfo() failed, leave fmtDate value unchanged and 
3064         //      try our luck with the default formats 
3066     //else: default C locale, default formats should work 
3071 #endif // __WINDOWS__ 
3073 const wxChar 
*wxDateTime::ParseFormat(const wxChar 
*date
, 
3074                                       const wxChar 
*format
, 
3075                                       const wxDateTime
& dateDef
) 
3077     wxCHECK_MSG( date 
&& format
, (wxChar 
*)NULL
, 
3078                  _T("NULL pointer in wxDateTime::ParseFormat()") ); 
3083     // what fields have we found? 
3084     bool haveWDay 
= false, 
3093     bool hourIsIn12hFormat 
= false, // or in 24h one? 
3094          isPM 
= false;              // AM by default 
3096     // and the value of the items we have (init them to get rid of warnings) 
3097     wxDateTime_t sec 
= 0, 
3100     WeekDay wday 
= Inv_WeekDay
; 
3101     wxDateTime_t yday 
= 0, 
3103     wxDateTime::Month mon 
= Inv_Month
; 
3106     const wxChar 
*input 
= date
; 
3107     for ( const wxChar 
*fmt 
= format
; *fmt
; fmt
++ ) 
3109         if ( *fmt 
!= _T('%') ) 
3111             if ( wxIsspace(*fmt
) ) 
3113                 // a white space in the format string matches 0 or more white 
3114                 // spaces in the input 
3115                 while ( wxIsspace(*input
) ) 
3122                 // any other character (not whitespace, not '%') must be 
3123                 // matched by itself in the input 
3124                 if ( *input
++ != *fmt 
) 
3127                     return (wxChar 
*)NULL
; 
3131             // done with this format char 
3135         // start of a format specification 
3137         // parse the optional width 
3139         while ( wxIsdigit(*++fmt
) ) 
3142             width 
+= *fmt 
- _T('0'); 
3145         // the default widths for the various fields 
3150                 case _T('Y'):               // year has 4 digits 
3154                 case _T('j'):               // day of year has 3 digits 
3155                 case _T('l'):               // milliseconds have 3 digits 
3159                 case _T('w'):               // week day as number has only one 
3164                     // default for all other fields 
3169         // then the format itself 
3172             case _T('a'):       // a weekday name 
3175                     int flag 
= *fmt 
== _T('a') ? Name_Abbr 
: Name_Full
; 
3176                     wday 
= GetWeekDayFromName(GetAlphaToken(input
), flag
); 
3177                     if ( wday 
== Inv_WeekDay 
) 
3180                         return (wxChar 
*)NULL
; 
3186             case _T('b'):       // a month name 
3189                     int flag 
= *fmt 
== _T('b') ? Name_Abbr 
: Name_Full
; 
3190                     mon 
= GetMonthFromName(GetAlphaToken(input
), flag
); 
3191                     if ( mon 
== Inv_Month 
) 
3194                         return (wxChar 
*)NULL
; 
3200             case _T('c'):       // locale default date and time  representation 
3204                     // this is the format which corresponds to ctime() output 
3205                     // and strptime("%c") should parse it, so try it first 
3206                     static const wxChar 
*fmtCtime 
= _T("%a %b %d %H:%M:%S %Y"); 
3208                     const wxChar 
*result 
= dt
.ParseFormat(input
, fmtCtime
); 
3211                         result 
= dt
.ParseFormat(input
, _T("%x %X")); 
3216                         result 
= dt
.ParseFormat(input
, _T("%X %x")); 
3221                         // we've tried everything and still no match 
3222                         return (wxChar 
*)NULL
; 
3227                     haveDay 
= haveMon 
= haveYear 
= 
3228                     haveHour 
= haveMin 
= haveSec 
= true; 
3242             case _T('d'):       // day of a month (01-31) 
3243                 if ( !GetNumericToken(width
, input
, &num
) || 
3244                         (num 
> 31) || (num 
< 1) ) 
3247                     return (wxChar 
*)NULL
; 
3250                 // we can't check whether the day range is correct yet, will 
3251                 // do it later - assume ok for now 
3253                 mday 
= (wxDateTime_t
)num
; 
3256             case _T('H'):       // hour in 24h format (00-23) 
3257                 if ( !GetNumericToken(width
, input
, &num
) || (num 
> 23) ) 
3260                     return (wxChar 
*)NULL
; 
3264                 hour 
= (wxDateTime_t
)num
; 
3267             case _T('I'):       // hour in 12h format (01-12) 
3268                 if ( !GetNumericToken(width
, input
, &num
) || !num 
|| (num 
> 12) ) 
3271                     return (wxChar 
*)NULL
; 
3275                 hourIsIn12hFormat 
= true; 
3276                 hour 
= (wxDateTime_t
)(num 
% 12);        // 12 should be 0 
3279             case _T('j'):       // day of the year 
3280                 if ( !GetNumericToken(width
, input
, &num
) || !num 
|| (num 
> 366) ) 
3283                     return (wxChar 
*)NULL
; 
3287                 yday 
= (wxDateTime_t
)num
; 
3290             case _T('m'):       // month as a number (01-12) 
3291                 if ( !GetNumericToken(width
, input
, &num
) || !num 
|| (num 
> 12) ) 
3294                     return (wxChar 
*)NULL
; 
3298                 mon 
= (Month
)(num 
- 1); 
3301             case _T('M'):       // minute as a decimal number (00-59) 
3302                 if ( !GetNumericToken(width
, input
, &num
) || (num 
> 59) ) 
3305                     return (wxChar 
*)NULL
; 
3309                 min 
= (wxDateTime_t
)num
; 
3312             case _T('p'):       // AM or PM string 
3314                     wxString am
, pm
, token 
= GetAlphaToken(input
); 
3316                     GetAmPmStrings(&am
, &pm
); 
3317                     if (am
.empty() && pm
.empty()) 
3318                         return (wxChar 
*)NULL
;  // no am/pm strings defined 
3319                     if ( token
.CmpNoCase(pm
) == 0 ) 
3323                     else if ( token
.CmpNoCase(am
) != 0 ) 
3326                         return (wxChar 
*)NULL
; 
3331             case _T('r'):       // time as %I:%M:%S %p 
3334                     input 
= dt
.ParseFormat(input
, _T("%I:%M:%S %p")); 
3338                         return (wxChar 
*)NULL
; 
3341                     haveHour 
= haveMin 
= haveSec 
= true; 
3350             case _T('R'):       // time as %H:%M 
3353                     input 
= dt
.ParseFormat(input
, _T("%H:%M")); 
3357                         return (wxChar 
*)NULL
; 
3360                     haveHour 
= haveMin 
= true; 
3368             case _T('S'):       // second as a decimal number (00-61) 
3369                 if ( !GetNumericToken(width
, input
, &num
) || (num 
> 61) ) 
3372                     return (wxChar 
*)NULL
; 
3376                 sec 
= (wxDateTime_t
)num
; 
3379             case _T('T'):       // time as %H:%M:%S 
3382                     input 
= dt
.ParseFormat(input
, _T("%H:%M:%S")); 
3386                         return (wxChar 
*)NULL
; 
3389                     haveHour 
= haveMin 
= haveSec 
= true; 
3398             case _T('w'):       // weekday as a number (0-6), Sunday = 0 
3399                 if ( !GetNumericToken(width
, input
, &num
) || (wday 
> 6) ) 
3402                     return (wxChar 
*)NULL
; 
3406                 wday 
= (WeekDay
)num
; 
3409             case _T('x'):       // locale default date representation 
3410 #ifdef HAVE_STRPTIME 
3411                 // try using strptime() -- it may fail even if the input is 
3412                 // correct but the date is out of range, so we will fall back 
3413                 // to our generic code anyhow 
3417                     const wxChar 
*result 
= CallStrptime(input
, "%x", &tm
); 
3422                         haveDay 
= haveMon 
= haveYear 
= true; 
3424                         year 
= 1900 + tm
.tm_year
; 
3425                         mon 
= (Month
)tm
.tm_mon
; 
3431 #endif // HAVE_STRPTIME 
3439                     // The above doesn't work for all locales, try to query 
3440                     // Windows for the right way of formatting the date: 
3441                     fmtDate 
= GetLocaleDateFormat(); 
3442                     if ( fmtDate
.empty() ) 
3445                         if ( IsWestEuropeanCountry(GetCountry()) || 
3446                              GetCountry() == Russia 
) 
3448                             fmtDate 
= _T("%d/%m/%y"); 
3449                             fmtDateAlt 
= _T("%m/%d/%y"); 
3453                             fmtDate 
= _T("%m/%d/%y"); 
3454                             fmtDateAlt 
= _T("%d/%m/%y"); 
3458                     const wxChar 
*result 
= dt
.ParseFormat(input
, fmtDate
); 
3460                     if ( !result 
&& !fmtDateAlt
.empty() ) 
3462                         // ok, be nice and try another one 
3463                         result 
= dt
.ParseFormat(input
, fmtDateAlt
); 
3469                         return (wxChar 
*)NULL
; 
3474                     haveDay 
= haveMon 
= haveYear 
= true; 
3485             case _T('X'):       // locale default time representation 
3486 #ifdef HAVE_STRPTIME 
3488                     // use strptime() to do it for us (FIXME !Unicode friendly) 
3490                     input 
= CallStrptime(input
, "%X", &tm
); 
3493                         return (wxChar 
*)NULL
; 
3496                     haveHour 
= haveMin 
= haveSec 
= true; 
3502 #else // !HAVE_STRPTIME 
3503                 // TODO under Win32 we can query the LOCALE_ITIME system 
3504                 //      setting which says whether the default time format is 
3507                     // try to parse what follows as "%H:%M:%S" and, if this 
3508                     // fails, as "%I:%M:%S %p" - this should catch the most 
3512                     const wxChar 
*result 
= dt
.ParseFormat(input
, _T("%T")); 
3515                         result 
= dt
.ParseFormat(input
, _T("%r")); 
3521                         return (wxChar 
*)NULL
; 
3524                     haveHour 
= haveMin 
= haveSec 
= true; 
3533 #endif // HAVE_STRPTIME/!HAVE_STRPTIME 
3536             case _T('y'):       // year without century (00-99) 
3537                 if ( !GetNumericToken(width
, input
, &num
) || (num 
> 99) ) 
3540                     return (wxChar 
*)NULL
; 
3545                 // TODO should have an option for roll over date instead of 
3546                 //      hard coding it here 
3547                 year 
= (num 
> 30 ? 1900 : 2000) + (wxDateTime_t
)num
; 
3550             case _T('Y'):       // year with century 
3551                 if ( !GetNumericToken(width
, input
, &num
) ) 
3554                     return (wxChar 
*)NULL
; 
3558                 year 
= (wxDateTime_t
)num
; 
3561             case _T('Z'):       // timezone name 
3562                 wxFAIL_MSG(_T("TODO")); 
3565             case _T('%'):       // a percent sign 
3566                 if ( *input
++ != _T('%') ) 
3569                     return (wxChar 
*)NULL
; 
3573             case 0:             // the end of string 
3574                 wxFAIL_MSG(_T("unexpected format end")); 
3578             default:            // not a known format spec 
3579                 return (wxChar 
*)NULL
; 
3583     // format matched, try to construct a date from what we have now 
3585     if ( dateDef
.IsValid() ) 
3587         // take this date as default 
3588         tmDef 
= dateDef
.GetTm(); 
3590     else if ( IsValid() ) 
3592         // if this date is valid, don't change it 
3597         // no default and this date is invalid - fall back to Today() 
3598         tmDef 
= Today().GetTm(); 
3609     // TODO we don't check here that the values are consistent, if both year 
3610     //      day and month/day were found, we just ignore the year day and we 
3611     //      also always ignore the week day 
3612     if ( haveMon 
&& haveDay 
) 
3614         if ( mday 
> GetNumOfDaysInMonth(tm
.year
, mon
) ) 
3616             wxLogDebug(_T("bad month day in wxDateTime::ParseFormat")); 
3618             return (wxChar 
*)NULL
; 
3624     else if ( haveYDay 
) 
3626         if ( yday 
> GetNumberOfDays(tm
.year
) ) 
3628             wxLogDebug(_T("bad year day in wxDateTime::ParseFormat")); 
3630             return (wxChar 
*)NULL
; 
3633         Tm tm2 
= wxDateTime(1, Jan
, tm
.year
).SetToYearDay(yday
).GetTm(); 
3640     if ( haveHour 
&& hourIsIn12hFormat 
&& isPM 
) 
3642         // translate to 24hour format 
3645     //else: either already in 24h format or no translation needed 
3665     // finally check that the week day is consistent -- if we had it 
3666     if ( haveWDay 
&& GetWeekDay() != wday 
) 
3668         wxLogDebug(_T("inconsistsnet week day in wxDateTime::ParseFormat()")); 
3676 const wxChar 
*wxDateTime::ParseDateTime(const wxChar 
*date
) 
3678     wxCHECK_MSG( date
, (wxChar 
*)NULL
, _T("NULL pointer in wxDateTime::Parse") ); 
3680     // Set to current day and hour, so strings like '14:00' becomes today at 
3681     // 14, not some other random date 
3682     wxDateTime dtDate 
= wxDateTime::Today(); 
3683     wxDateTime dtTime 
= wxDateTime::Today(); 
3685     const wxChar
* pchTime
; 
3687     // Try to parse the beginning of the string as a date 
3688     const wxChar
* pchDate 
= dtDate
.ParseDate(date
); 
3690     // We got a date in the beginning, see if there is a time specified after the date 
3693         // Skip spaces, as the ParseTime() function fails on spaces 
3694         while ( wxIsspace(*pchDate
) ) 
3697         pchTime 
= dtTime
.ParseTime(pchDate
); 
3699     else // no date in the beginning 
3701         // check and see if we have a time followed by a date 
3702         pchTime 
= dtTime
.ParseTime(date
); 
3705             while ( wxIsspace(*pchTime
) ) 
3708             pchDate 
= dtDate
.ParseDate(pchTime
); 
3712     // If we have a date specified, set our own data to the same date 
3713     if ( !pchDate 
|| !pchTime 
) 
3716     Set(dtDate
.GetDay(), dtDate
.GetMonth(), dtDate
.GetYear(), 
3717         dtTime
.GetHour(), dtTime
.GetMinute(), dtTime
.GetSecond(), 
3718         dtTime
.GetMillisecond()); 
3720     // Return endpoint of scan 
3721     return pchDate 
> pchTime 
? pchDate 
: pchTime
; 
3724 const wxChar 
*wxDateTime::ParseDate(const wxChar 
*date
) 
3726     // this is a simplified version of ParseDateTime() which understands only 
3727     // "today" (for wxDate compatibility) and digits only otherwise (and not 
3728     // all esoteric constructions ParseDateTime() knows about) 
3730     wxCHECK_MSG( date
, (wxChar 
*)NULL
, _T("NULL pointer in wxDateTime::Parse") ); 
3732     const wxChar 
*p 
= date
; 
3733     while ( wxIsspace(*p
) ) 
3736     // some special cases 
3740         int dayDiffFromToday
; 
3743         { wxTRANSLATE("today"),             0 }, 
3744         { wxTRANSLATE("yesterday"),        -1 }, 
3745         { wxTRANSLATE("tomorrow"),          1 }, 
3748     for ( size_t n 
= 0; n 
< WXSIZEOF(literalDates
); n
++ ) 
3750         const wxString dateStr 
= wxGetTranslation(literalDates
[n
].str
); 
3751         size_t len 
= dateStr
.length(); 
3752         if ( wxStrlen(p
) >= len 
) 
3754             wxString 
str(p
, len
); 
3755             if ( str
.CmpNoCase(dateStr
) == 0 ) 
3757                 // nothing can follow this, so stop here 
3760                 int dayDiffFromToday 
= literalDates
[n
].dayDiffFromToday
; 
3762                 if ( dayDiffFromToday 
) 
3764                     *this += wxDateSpan::Days(dayDiffFromToday
); 
3772     // We try to guess what we have here: for each new (numeric) token, we 
3773     // determine if it can be a month, day or a year. Of course, there is an 
3774     // ambiguity as some numbers may be days as well as months, so we also 
3775     // have the ability to back track. 
3778     bool haveDay 
= false,       // the months day? 
3779          haveWDay 
= false,      // the day of week? 
3780          haveMon 
= false,       // the month? 
3781          haveYear 
= false;      // the year? 
3783     // and the value of the items we have (init them to get rid of warnings) 
3784     WeekDay wday 
= Inv_WeekDay
; 
3785     wxDateTime_t day 
= 0; 
3786     wxDateTime::Month mon 
= Inv_Month
; 
3789     // tokenize the string 
3791     static const wxChar 
*dateDelimiters 
= _T(".,/-\t\r\n "); 
3792     wxStringTokenizer 
tok(p
, dateDelimiters
); 
3793     while ( tok
.HasMoreTokens() ) 
3795         wxString token 
= tok
.GetNextToken(); 
3801         if ( token
.ToULong(&val
) ) 
3803             // guess what this number is 
3809             if ( !haveMon 
&& val 
> 0 && val 
<= 12 ) 
3811                 // assume it is month 
3814             else // not the month 
3818                     // this can only be the year 
3821                 else // may be either day or year 
3823                     // use a leap year if we don't have the year yet to allow 
3824                     // dates like 2/29/1976 which would be rejected otherwise 
3825                     wxDateTime_t max_days 
= (wxDateTime_t
)( 
3827                         ? GetNumOfDaysInMonth(haveYear 
? year 
: 1976, mon
) 
3832                     if ( (val 
== 0) || (val 
> (unsigned long)max_days
) ) 
3837                     else // yes, suppose it's the day 
3851                 year 
= (wxDateTime_t
)val
; 
3860                 day 
= (wxDateTime_t
)val
; 
3866                 mon 
= (Month
)(val 
- 1); 
3869         else // not a number 
3871             // be careful not to overwrite the current mon value 
3872             Month mon2 
= GetMonthFromName(token
, Name_Full 
| Name_Abbr
); 
3873             if ( mon2 
!= Inv_Month 
) 
3878                     // but we already have a month - maybe we guessed wrong? 
3881                         // no need to check in month range as always < 12, but 
3882                         // the days are counted from 1 unlike the months 
3883                         day 
= (wxDateTime_t
)(mon 
+ 1); 
3888                         // could possible be the year (doesn't the year come 
3889                         // before the month in the japanese format?) (FIXME) 
3898             else // not a valid month name 
3900                 wday 
= GetWeekDayFromName(token
, Name_Full 
| Name_Abbr
); 
3901                 if ( wday 
!= Inv_WeekDay 
) 
3911                 else // not a valid weekday name 
3914                     static const wxChar 
*ordinals
[] = 
3916                         wxTRANSLATE("first"), 
3917                         wxTRANSLATE("second"), 
3918                         wxTRANSLATE("third"), 
3919                         wxTRANSLATE("fourth"), 
3920                         wxTRANSLATE("fifth"), 
3921                         wxTRANSLATE("sixth"), 
3922                         wxTRANSLATE("seventh"), 
3923                         wxTRANSLATE("eighth"), 
3924                         wxTRANSLATE("ninth"), 
3925                         wxTRANSLATE("tenth"), 
3926                         wxTRANSLATE("eleventh"), 
3927                         wxTRANSLATE("twelfth"), 
3928                         wxTRANSLATE("thirteenth"), 
3929                         wxTRANSLATE("fourteenth"), 
3930                         wxTRANSLATE("fifteenth"), 
3931                         wxTRANSLATE("sixteenth"), 
3932                         wxTRANSLATE("seventeenth"), 
3933                         wxTRANSLATE("eighteenth"), 
3934                         wxTRANSLATE("nineteenth"), 
3935                         wxTRANSLATE("twentieth"), 
3936                         // that's enough - otherwise we'd have problems with 
3937                         // composite (or not) ordinals 
3941                     for ( n 
= 0; n 
< WXSIZEOF(ordinals
); n
++ ) 
3943                         if ( token
.CmpNoCase(ordinals
[n
]) == 0 ) 
3949                     if ( n 
== WXSIZEOF(ordinals
) ) 
3951                         // stop here - something unknown 
3958                         // don't try anything here (as in case of numeric day 
3959                         // above) - the symbolic day spec should always 
3960                         // precede the month/year 
3966                     day 
= (wxDateTime_t
)(n 
+ 1); 
3971         nPosCur 
= tok
.GetPosition(); 
3974     // either no more tokens or the scan was stopped by something we couldn't 
3975     // parse - in any case, see if we can construct a date from what we have 
3976     if ( !haveDay 
&& !haveWDay 
) 
3978         wxLogDebug(_T("ParseDate: no day, no weekday hence no date.")); 
3983     if ( haveWDay 
&& (haveMon 
|| haveYear 
|| haveDay
) && 
3984          !(haveDay 
&& haveMon 
&& haveYear
) ) 
3986         // without adjectives (which we don't support here) the week day only 
3987         // makes sense completely separately or with the full date 
3988         // specification (what would "Wed 1999" mean?) 
3992     if ( !haveWDay 
&& haveYear 
&& !(haveDay 
&& haveMon
) ) 
3994         // may be we have month and day instead of day and year? 
3995         if ( haveDay 
&& !haveMon 
) 
3999                 // exchange day and month 
4000                 mon 
= (wxDateTime::Month
)(day 
- 1); 
4002                 // we're in the current year then 
4003                 if ( (year 
> 0) && (year 
<= (int)GetNumOfDaysInMonth(Inv_Year
, mon
)) ) 
4005                     day 
= (wxDateTime_t
)year
; 
4010                 //else: no, can't exchange, leave haveMon == false 
4016             // if we give the year, month and day must be given too 
4017             wxLogDebug(_T("ParseDate: day and month should be specified if year is.")); 
4025         mon 
= GetCurrentMonth(); 
4030         year 
= GetCurrentYear(); 
4035         // normally we check the day above but the check is optimistic in case 
4036         // we find the day before its month/year so we have to redo it now 
4037         if ( day 
> GetNumOfDaysInMonth(year
, mon
) ) 
4040         Set(day
, mon
, year
); 
4044             // check that it is really the same 
4045             if ( GetWeekDay() != wday 
) 
4047                 // inconsistency detected 
4048                 wxLogDebug(_T("ParseDate: inconsistent day/weekday.")); 
4050                 return (wxChar 
*)NULL
; 
4058         SetToWeekDayInSameWeek(wday
); 
4061     // return the pointer to the first unparsed char 
4063     if ( nPosCur 
&& wxStrchr(dateDelimiters
, *(p 
- 1)) ) 
4065         // if we couldn't parse the token after the delimiter, put back the 
4066         // delimiter as well 
4073 const wxChar 
*wxDateTime::ParseTime(const wxChar 
*time
) 
4075     wxCHECK_MSG( time
, (wxChar 
*)NULL
, _T("NULL pointer in wxDateTime::Parse") ); 
4077     // first try some extra things 
4084         { wxTRANSLATE("noon"),      12 }, 
4085         { wxTRANSLATE("midnight"),  00 }, 
4089     for ( size_t n 
= 0; n 
< WXSIZEOF(stdTimes
); n
++ ) 
4091         wxString timeString 
= wxGetTranslation(stdTimes
[n
].name
); 
4092         size_t len 
= timeString
.length(); 
4093         if ( timeString
.CmpNoCase(wxString(time
, len
)) == 0 ) 
4095             // casts required by DigitalMars 
4096             Set(stdTimes
[n
].hour
, wxDateTime_t(0), wxDateTime_t(0)); 
4102     // try all time formats we may think about in the order from longest to 
4105     // 12hour with AM/PM? 
4106     const wxChar 
*result 
= ParseFormat(time
, _T("%I:%M:%S %p")); 
4110         // normally, it's the same, but why not try it? 
4111         result 
= ParseFormat(time
, _T("%H:%M:%S")); 
4116         // 12hour with AM/PM but without seconds? 
4117         result 
= ParseFormat(time
, _T("%I:%M %p")); 
4123         result 
= ParseFormat(time
, _T("%H:%M")); 
4128         // just the hour and AM/PM? 
4129         result 
= ParseFormat(time
, _T("%I %p")); 
4135         result 
= ParseFormat(time
, _T("%H")); 
4140         // parse the standard format: normally it is one of the formats above 
4141         // but it may be set to something completely different by the user 
4142         result 
= ParseFormat(time
, _T("%X")); 
4145     // TODO: parse timezones 
4150 // ---------------------------------------------------------------------------- 
4151 // Workdays and holidays support 
4152 // ---------------------------------------------------------------------------- 
4154 bool wxDateTime::IsWorkDay(Country 
WXUNUSED(country
)) const 
4156     return !wxDateTimeHolidayAuthority::IsHoliday(*this); 
4159 // ============================================================================ 
4161 // ============================================================================ 
4163 wxDateSpan WXDLLIMPEXP_BASE 
operator*(int n
, const wxDateSpan
& ds
) 
4166     return ds1
.Multiply(n
); 
4169 // ============================================================================ 
4171 // ============================================================================ 
4173 wxTimeSpan WXDLLIMPEXP_BASE 
operator*(int n
, const wxTimeSpan
& ts
) 
4175     return wxTimeSpan(ts
).Multiply(n
); 
4178 // this enum is only used in wxTimeSpan::Format() below but we can't declare 
4179 // it locally to the method as it provokes an internal compiler error in egcs 
4180 // 2.91.60 when building with -O2 
4191 // not all strftime(3) format specifiers make sense here because, for example, 
4192 // a time span doesn't have a year nor a timezone 
4194 // Here are the ones which are supported (all of them are supported by strftime 
4196 //  %H          hour in 24 hour format 
4197 //  %M          minute (00 - 59) 
4198 //  %S          second (00 - 59) 
4201 // Also, for MFC CTimeSpan compatibility, we support 
4202 //  %D          number of days 
4204 // And, to be better than MFC :-), we also have 
4205 //  %E          number of wEeks 
4206 //  %l          milliseconds (000 - 999) 
4207 wxString 
wxTimeSpan::Format(const wxChar 
*format
) const 
4209     wxCHECK_MSG( format
, wxEmptyString
, _T("NULL format in wxTimeSpan::Format") ); 
4212     str
.Alloc(wxStrlen(format
)); 
4214     // Suppose we have wxTimeSpan ts(1 /* hour */, 2 /* min */, 3 /* sec */) 
4216     // Then, of course, ts.Format("%H:%M:%S") must return "01:02:03", but the 
4217     // question is what should ts.Format("%S") do? The code here returns "3273" 
4218     // in this case (i.e. the total number of seconds, not just seconds % 60) 
4219     // because, for me, this call means "give me entire time interval in 
4220     // seconds" and not "give me the seconds part of the time interval" 
4222     // If we agree that it should behave like this, it is clear that the 
4223     // interpretation of each format specifier depends on the presence of the 
4224     // other format specs in the string: if there was "%H" before "%M", we 
4225     // should use GetMinutes() % 60, otherwise just GetMinutes() &c 
4227     // we remember the most important unit found so far 
4228     TimeSpanPart partBiggest 
= Part_MSec
; 
4230     for ( const wxChar 
*pch 
= format
; *pch
; pch
++ ) 
4234         if ( ch 
== _T('%') ) 
4236             // the start of the format specification of the printf() below 
4237             wxString fmtPrefix 
= _T('%'); 
4242             ch 
= *++pch
;    // get the format spec char 
4246                     wxFAIL_MSG( _T("invalid format character") ); 
4252                     // skip the part below switch 
4257                     if ( partBiggest 
< Part_Day 
) 
4263                         partBiggest 
= Part_Day
; 
4268                     partBiggest 
= Part_Week
; 
4274                     if ( partBiggest 
< Part_Hour 
) 
4280                         partBiggest 
= Part_Hour
; 
4283                     fmtPrefix 
+= _T("02"); 
4287                     n 
= GetMilliseconds().ToLong(); 
4288                     if ( partBiggest 
< Part_MSec 
) 
4292                     //else: no need to reset partBiggest to Part_MSec, it is 
4293                     //      the least significant one anyhow 
4295                     fmtPrefix 
+= _T("03"); 
4300                     if ( partBiggest 
< Part_Min 
) 
4306                         partBiggest 
= Part_Min
; 
4309                     fmtPrefix 
+= _T("02"); 
4313                     n 
= GetSeconds().ToLong(); 
4314                     if ( partBiggest 
< Part_Sec 
) 
4320                         partBiggest 
= Part_Sec
; 
4323                     fmtPrefix 
+= _T("02"); 
4327             str 
+= wxString::Format(fmtPrefix 
+ _T("ld"), n
); 
4331             // normal character, just copy 
4339 // ============================================================================ 
4340 // wxDateTimeHolidayAuthority and related classes 
4341 // ============================================================================ 
4343 #include "wx/arrimpl.cpp" 
4345 WX_DEFINE_OBJARRAY(wxDateTimeArray
) 
4347 static int wxCMPFUNC_CONV
 
4348 wxDateTimeCompareFunc(wxDateTime 
**first
, wxDateTime 
**second
) 
4350     wxDateTime dt1 
= **first
, 
4353     return dt1 
== dt2 
? 0 : dt1 
< dt2 
? -1 : +1; 
4356 // ---------------------------------------------------------------------------- 
4357 // wxDateTimeHolidayAuthority 
4358 // ---------------------------------------------------------------------------- 
4360 wxHolidayAuthoritiesArray 
wxDateTimeHolidayAuthority::ms_authorities
; 
4363 bool wxDateTimeHolidayAuthority::IsHoliday(const wxDateTime
& dt
) 
4365     size_t count 
= ms_authorities
.size(); 
4366     for ( size_t n 
= 0; n 
< count
; n
++ ) 
4368         if ( ms_authorities
[n
]->DoIsHoliday(dt
) ) 
4379 wxDateTimeHolidayAuthority::GetHolidaysInRange(const wxDateTime
& dtStart
, 
4380                                                const wxDateTime
& dtEnd
, 
4381                                                wxDateTimeArray
& holidays
) 
4383     wxDateTimeArray hol
; 
4387     const size_t countAuth 
= ms_authorities
.size(); 
4388     for ( size_t nAuth 
= 0; nAuth 
< countAuth
; nAuth
++ ) 
4390         ms_authorities
[nAuth
]->DoGetHolidaysInRange(dtStart
, dtEnd
, hol
); 
4392         WX_APPEND_ARRAY(holidays
, hol
); 
4395     holidays
.Sort(wxDateTimeCompareFunc
); 
4397     return holidays
.size(); 
4401 void wxDateTimeHolidayAuthority::ClearAllAuthorities() 
4403     WX_CLEAR_ARRAY(ms_authorities
); 
4407 void wxDateTimeHolidayAuthority::AddAuthority(wxDateTimeHolidayAuthority 
*auth
) 
4409     ms_authorities
.push_back(auth
); 
4412 wxDateTimeHolidayAuthority::~wxDateTimeHolidayAuthority() 
4414     // required here for Darwin 
4417 // ---------------------------------------------------------------------------- 
4418 // wxDateTimeWorkDays 
4419 // ---------------------------------------------------------------------------- 
4421 bool wxDateTimeWorkDays::DoIsHoliday(const wxDateTime
& dt
) const 
4423     wxDateTime::WeekDay wd 
= dt
.GetWeekDay(); 
4425     return (wd 
== wxDateTime::Sun
) || (wd 
== wxDateTime::Sat
); 
4428 size_t wxDateTimeWorkDays::DoGetHolidaysInRange(const wxDateTime
& dtStart
, 
4429                                                 const wxDateTime
& dtEnd
, 
4430                                                 wxDateTimeArray
& holidays
) const 
4432     if ( dtStart 
> dtEnd 
) 
4434         wxFAIL_MSG( _T("invalid date range in GetHolidaysInRange") ); 
4441     // instead of checking all days, start with the first Sat after dtStart and 
4442     // end with the last Sun before dtEnd 
4443     wxDateTime dtSatFirst 
= dtStart
.GetNextWeekDay(wxDateTime::Sat
), 
4444                dtSatLast 
= dtEnd
.GetPrevWeekDay(wxDateTime::Sat
), 
4445                dtSunFirst 
= dtStart
.GetNextWeekDay(wxDateTime::Sun
), 
4446                dtSunLast 
= dtEnd
.GetPrevWeekDay(wxDateTime::Sun
), 
4449     for ( dt 
= dtSatFirst
; dt 
<= dtSatLast
; dt 
+= wxDateSpan::Week() ) 
4454     for ( dt 
= dtSunFirst
; dt 
<= dtSunLast
; dt 
+= wxDateSpan::Week() ) 
4459     return holidays
.GetCount(); 
4462 // ============================================================================ 
4463 // other helper functions 
4464 // ============================================================================ 
4466 // ---------------------------------------------------------------------------- 
4467 // iteration helpers: can be used to write a for loop over enum variable like 
4469 //  for ( m = wxDateTime::Jan; m < wxDateTime::Inv_Month; wxNextMonth(m) ) 
4470 // ---------------------------------------------------------------------------- 
4472 WXDLLIMPEXP_BASE 
void wxNextMonth(wxDateTime::Month
& m
) 
4474     wxASSERT_MSG( m 
< wxDateTime::Inv_Month
, _T("invalid month") ); 
4476     // no wrapping or the for loop above would never end! 
4477     m 
= (wxDateTime::Month
)(m 
+ 1); 
4480 WXDLLIMPEXP_BASE 
void wxPrevMonth(wxDateTime::Month
& m
) 
4482     wxASSERT_MSG( m 
< wxDateTime::Inv_Month
, _T("invalid month") ); 
4484     m 
= m 
== wxDateTime::Jan 
? wxDateTime::Inv_Month
 
4485                              : (wxDateTime::Month
)(m 
- 1); 
4488 WXDLLIMPEXP_BASE 
void wxNextWDay(wxDateTime::WeekDay
& wd
) 
4490     wxASSERT_MSG( wd 
< wxDateTime::Inv_WeekDay
, _T("invalid week day") ); 
4492     // no wrapping or the for loop above would never end! 
4493     wd 
= (wxDateTime::WeekDay
)(wd 
+ 1); 
4496 WXDLLIMPEXP_BASE 
void wxPrevWDay(wxDateTime::WeekDay
& wd
) 
4498     wxASSERT_MSG( wd 
< wxDateTime::Inv_WeekDay
, _T("invalid week day") ); 
4500     wd 
= wd 
== wxDateTime::Sun 
? wxDateTime::Inv_WeekDay
 
4501                                : (wxDateTime::WeekDay
)(wd 
- 1); 
4504 #endif // wxUSE_DATETIME