1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/datetimefmt.cpp
3 // Purpose: wxDateTime formatting & parsing code
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 // ============================================================================
21 // ============================================================================
23 // ----------------------------------------------------------------------------
25 // ----------------------------------------------------------------------------
27 // For compilers that support precompilation, includes "wx.h".
28 #include "wx/wxprec.h"
34 #if !defined(wxUSE_DATETIME) || wxUSE_DATETIME
38 #include "wx/msw/wrapwin.h"
40 #include "wx/string.h"
43 #include "wx/stopwatch.h" // for wxGetLocalTimeMillis()
44 #include "wx/module.h"
48 #include "wx/thread.h"
49 #include "wx/tokenzr.h"
60 #include "wx/datetime.h"
62 // ============================================================================
63 // implementation of wxDateTime
64 // ============================================================================
66 // ----------------------------------------------------------------------------
67 // helpers shared between datetime.cpp and datetimefmt.cpp
68 // ----------------------------------------------------------------------------
70 extern void InitTm(struct tm
& tm
);
72 extern int GetTimeZone();
74 extern wxString
CallStrftime(const wxString
& format
, const tm
* tm
);
76 // ----------------------------------------------------------------------------
77 // constants (see also datetime.cpp)
78 // ----------------------------------------------------------------------------
80 static const int DAYS_PER_WEEK
= 7;
82 static const int HOURS_PER_DAY
= 24;
84 static const int SEC_PER_MIN
= 60;
86 static const int MIN_PER_HOUR
= 60;
88 // ----------------------------------------------------------------------------
90 // ----------------------------------------------------------------------------
101 // return the month if the string is a month name or Inv_Month otherwise
103 // flags can contain wxDateTime::Name_Abbr/Name_Full or both of them and lang
104 // can be either DateLang_Local (default) to interpret string as a localized
105 // month name or DateLang_English to parse it as a standard English name or
106 // their combination to interpret it in any way
108 GetMonthFromName(const wxString
& name
, int flags
, int lang
)
110 wxDateTime::Month mon
;
111 for ( mon
= wxDateTime::Jan
; mon
< wxDateTime::Inv_Month
; wxNextMonth(mon
) )
113 // case-insensitive comparison either one of or with both abbreviated
115 if ( flags
& wxDateTime::Name_Full
)
117 if ( lang
& DateLang_English
)
119 if ( name
.CmpNoCase(wxDateTime::GetEnglishMonthName(mon
,
120 wxDateTime::Name_Full
)) == 0 )
124 if ( lang
& DateLang_Local
)
126 if ( name
.CmpNoCase(wxDateTime::GetMonthName(mon
,
127 wxDateTime::Name_Full
)) == 0 )
132 if ( flags
& wxDateTime::Name_Abbr
)
134 if ( lang
& DateLang_English
)
136 if ( name
.CmpNoCase(wxDateTime::GetEnglishMonthName(mon
,
137 wxDateTime::Name_Abbr
)) == 0 )
141 if ( lang
& DateLang_Local
)
143 if ( name
.CmpNoCase(wxDateTime::GetMonthName(mon
,
144 wxDateTime::Name_Abbr
)) == 0 )
153 // return the weekday if the string is a weekday name or Inv_WeekDay otherwise
155 // flags and lang parameters have the same meaning as for GetMonthFromName()
158 GetWeekDayFromName(const wxString
& name
, int flags
, int lang
)
160 wxDateTime::WeekDay wd
;
161 for ( wd
= wxDateTime::Sun
; wd
< wxDateTime::Inv_WeekDay
; wxNextWDay(wd
) )
163 if ( flags
& wxDateTime::Name_Full
)
165 if ( lang
& DateLang_English
)
167 if ( name
.CmpNoCase(wxDateTime::GetEnglishWeekDayName(wd
,
168 wxDateTime::Name_Full
)) == 0 )
172 if ( lang
& DateLang_Local
)
174 if ( name
.CmpNoCase(wxDateTime::GetWeekDayName(wd
,
175 wxDateTime::Name_Full
)) == 0 )
180 if ( flags
& wxDateTime::Name_Abbr
)
182 if ( lang
& DateLang_English
)
184 if ( name
.CmpNoCase(wxDateTime::GetEnglishWeekDayName(wd
,
185 wxDateTime::Name_Abbr
)) == 0 )
189 if ( lang
& DateLang_Local
)
191 if ( name
.CmpNoCase(wxDateTime::GetWeekDayName(wd
,
192 wxDateTime::Name_Abbr
)) == 0 )
201 // scans all digits (but no more than len) and returns the resulting number
202 bool GetNumericToken(size_t len
,
203 wxString::const_iterator
& p
,
204 const wxString::const_iterator
& end
,
205 unsigned long *number
)
209 while ( p
!= end
&& wxIsdigit(*p
) )
213 if ( len
&& ++n
> len
)
217 return !s
.empty() && s
.ToULong(number
);
220 // scans all alphabetic characters and returns the resulting string
222 GetAlphaToken(wxString::const_iterator
& p
,
223 const wxString::const_iterator
& end
)
226 while ( p
!= end
&& wxIsalpha(*p
) )
234 // parses string starting at given iterator using the specified format and,
235 // optionally, a fall back format (and optionally another one... but it stops
238 // if unsuccessful, returns invalid wxDateTime without changing p; otherwise
239 // advance p to the end of the match and returns wxDateTime containing the
240 // results of the parsing
242 ParseFormatAt(wxString::const_iterator
& p
,
243 const wxString::const_iterator
& end
,
245 // FIXME-VC6: using wxString() instead of wxEmptyString in the
246 // line below results in error C2062: type 'class
247 // wxString (__cdecl *)(void)' unexpected
248 const wxString
& fmtAlt
= wxEmptyString
)
250 const wxString
str(p
, end
);
251 wxString::const_iterator endParse
;
253 if ( dt
.ParseFormat(str
, fmt
, &endParse
) ||
254 (!fmtAlt
.empty() && dt
.ParseFormat(str
, fmtAlt
, &endParse
)) )
256 p
+= endParse
- str
.begin();
258 //else: all formats failed
263 } // anonymous namespace
265 // ----------------------------------------------------------------------------
266 // wxDateTime to/from text representations
267 // ----------------------------------------------------------------------------
269 wxString
wxDateTime::Format(const wxString
& format
, const TimeZone
& tz
) const
271 wxCHECK_MSG( !format
.empty(), wxEmptyString
,
272 _T("NULL format in wxDateTime::Format") );
274 // we have to use our own implementation if the date is out of range of
275 // strftime() or if we use non standard specificators
277 time_t time
= GetTicks();
279 if ( (time
!= (time_t)-1) && !wxStrstr(format
, _T("%l")) )
284 if ( tz
.GetOffset() == -GetTimeZone() )
286 // we are working with local time
287 tm
= wxLocaltime_r(&time
, &tmstruct
);
289 // should never happen
290 wxCHECK_MSG( tm
, wxEmptyString
, _T("wxLocaltime_r() failed") );
294 time
+= (int)tz
.GetOffset();
296 #if defined(__VMS__) || defined(__WATCOMC__) // time is unsigned so avoid warning
297 int time2
= (int) time
;
303 tm
= wxGmtime_r(&time
, &tmstruct
);
305 // should never happen
306 wxCHECK_MSG( tm
, wxEmptyString
, _T("wxGmtime_r() failed") );
310 tm
= (struct tm
*)NULL
;
316 return CallStrftime(format
, tm
);
319 //else: use generic code below
320 #endif // HAVE_STRFTIME
322 // we only parse ANSI C format specifications here, no POSIX 2
323 // complications, no GNU extensions but we do add support for a "%l" format
324 // specifier allowing to get the number of milliseconds
327 // used for calls to strftime() when we only deal with time
328 struct tm tmTimeOnly
;
329 tmTimeOnly
.tm_hour
= tm
.hour
;
330 tmTimeOnly
.tm_min
= tm
.min
;
331 tmTimeOnly
.tm_sec
= tm
.sec
;
332 tmTimeOnly
.tm_wday
= 0;
333 tmTimeOnly
.tm_yday
= 0;
334 tmTimeOnly
.tm_mday
= 1; // any date will do
335 tmTimeOnly
.tm_mon
= 0;
336 tmTimeOnly
.tm_year
= 76;
337 tmTimeOnly
.tm_isdst
= 0; // no DST, we adjust for tz ourselves
339 wxString tmp
, res
, fmt
;
340 for ( wxString::const_iterator p
= format
.begin(); p
!= format
.end(); ++p
)
350 // set the default format
351 switch ( (*++p
).GetValue() )
353 case _T('Y'): // year has 4 digits
357 case _T('j'): // day of year has 3 digits
358 case _T('l'): // milliseconds have 3 digits
362 case _T('w'): // week day as number has only one
367 // it's either another valid format specifier in which case
368 // the format is "%02d" (for all the rest) or we have the
369 // field width preceding the format in which case it will
370 // override the default format anyhow
379 // start of the format specification
380 switch ( (*p
).GetValue() )
382 case _T('a'): // a weekday name
384 // second parameter should be true for abbreviated names
385 res
+= GetWeekDayName(tm
.GetWeekDay(),
386 *p
== _T('a') ? Name_Abbr
: Name_Full
);
389 case _T('b'): // a month name
391 res
+= GetMonthName(tm
.mon
,
392 *p
== _T('b') ? Name_Abbr
: Name_Full
);
395 case _T('c'): // locale default date and time representation
396 case _T('x'): // locale default date representation
399 // the problem: there is no way to know what do these format
400 // specifications correspond to for the current locale.
402 // the solution: use a hack and still use strftime(): first
403 // find the YEAR which is a year in the strftime() range (1970
404 // - 2038) whose Jan 1 falls on the same week day as the Jan 1
405 // of the real year. Then make a copy of the format and
406 // replace all occurrences of YEAR in it with some unique
407 // string not appearing anywhere else in it, then use
408 // strftime() to format the date in year YEAR and then replace
409 // YEAR back by the real year and the unique replacement
410 // string back with YEAR. Notice that "all occurrences of YEAR"
411 // means all occurrences of 4 digit as well as 2 digit form!
413 // the bugs: we assume that neither of %c nor %x contains any
414 // fields which may change between the YEAR and real year. For
415 // example, the week number (%U, %W) and the day number (%j)
416 // will change if one of these years is leap and the other one
419 // find the YEAR: normally, for any year X, Jan 1 of the
420 // year X + 28 is the same weekday as Jan 1 of X (because
421 // the weekday advances by 1 for each normal X and by 2
422 // for each leap X, hence by 5 every 4 years or by 35
423 // which is 0 mod 7 every 28 years) but this rule breaks
424 // down if there are years between X and Y which are
425 // divisible by 4 but not leap (i.e. divisible by 100 but
426 // not 400), hence the correction.
428 int yearReal
= GetYear(tz
);
429 int mod28
= yearReal
% 28;
431 // be careful to not go too far - we risk to leave the
436 year
= 1988 + mod28
; // 1988 == 0 (mod 28)
440 year
= 1970 + mod28
- 10; // 1970 == 10 (mod 28)
443 int nCentury
= year
/ 100,
444 nCenturyReal
= yearReal
/ 100;
446 // need to adjust for the years divisble by 400 which are
447 // not leap but are counted like leap ones if we just take
448 // the number of centuries in between for nLostWeekDays
449 int nLostWeekDays
= (nCentury
- nCenturyReal
) -
450 (nCentury
/ 4 - nCenturyReal
/ 4);
452 // we have to gain back the "lost" weekdays: note that the
453 // effect of this loop is to not do anything to
454 // nLostWeekDays (which we won't use any more), but to
455 // (indirectly) set the year correctly
456 while ( (nLostWeekDays
% 7) != 0 )
458 nLostWeekDays
+= year
++ % 4 ? 1 : 2;
461 // finally move the year below 2000 so that the 2-digit
462 // year number can never match the month or day of the
463 // month when we do the replacements below
467 wxASSERT_MSG( year
>= 1970 && year
< 2000,
468 _T("logic error in wxDateTime::Format") );
471 // use strftime() to format the same date but in supported
474 // NB: we assume that strftime() doesn't check for the
475 // date validity and will happily format the date
476 // corresponding to Feb 29 of a non leap year (which
477 // may happen if yearReal was leap and year is not)
478 struct tm tmAdjusted
;
480 tmAdjusted
.tm_hour
= tm
.hour
;
481 tmAdjusted
.tm_min
= tm
.min
;
482 tmAdjusted
.tm_sec
= tm
.sec
;
483 tmAdjusted
.tm_wday
= tm
.GetWeekDay();
484 tmAdjusted
.tm_yday
= GetDayOfYear();
485 tmAdjusted
.tm_mday
= tm
.mday
;
486 tmAdjusted
.tm_mon
= tm
.mon
;
487 tmAdjusted
.tm_year
= year
- 1900;
488 tmAdjusted
.tm_isdst
= 0; // no DST, already adjusted
489 wxString str
= CallStrftime(*p
== _T('c') ? _T("%c")
493 // now replace the replacement year with the real year:
494 // notice that we have to replace the 4 digit year with
495 // a unique string not appearing in strftime() output
496 // first to prevent the 2 digit year from matching any
497 // substring of the 4 digit year (but any day, month,
498 // hours or minutes components should be safe because
499 // they are never in 70-99 range)
500 wxString
replacement("|");
501 while ( str
.find(replacement
) != wxString::npos
)
504 str
.Replace(wxString::Format("%d", year
),
506 str
.Replace(wxString::Format("%d", year
% 100),
507 wxString::Format("%d", yearReal
% 100));
508 str
.Replace(replacement
,
509 wxString::Format("%d", yearReal
));
513 #else // !HAVE_STRFTIME
514 // Use "%m/%d/%y %H:%M:%S" format instead
515 res
+= wxString::Format(wxT("%02d/%02d/%04d %02d:%02d:%02d"),
516 tm
.mon
+1,tm
.mday
, tm
.year
, tm
.hour
, tm
.min
, tm
.sec
);
517 #endif // HAVE_STRFTIME/!HAVE_STRFTIME
520 case _T('d'): // day of a month (01-31)
521 res
+= wxString::Format(fmt
, tm
.mday
);
524 case _T('H'): // hour in 24h format (00-23)
525 res
+= wxString::Format(fmt
, tm
.hour
);
528 case _T('I'): // hour in 12h format (01-12)
530 // 24h -> 12h, 0h -> 12h too
531 int hour12
= tm
.hour
> 12 ? tm
.hour
- 12
532 : tm
.hour
? tm
.hour
: 12;
533 res
+= wxString::Format(fmt
, hour12
);
537 case _T('j'): // day of the year
538 res
+= wxString::Format(fmt
, GetDayOfYear(tz
));
541 case _T('l'): // milliseconds (NOT STANDARD)
542 res
+= wxString::Format(fmt
, GetMillisecond(tz
));
545 case _T('m'): // month as a number (01-12)
546 res
+= wxString::Format(fmt
, tm
.mon
+ 1);
549 case _T('M'): // minute as a decimal number (00-59)
550 res
+= wxString::Format(fmt
, tm
.min
);
553 case _T('p'): // AM or PM string
555 res
+= CallStrftime(_T("%p"), &tmTimeOnly
);
556 #else // !HAVE_STRFTIME
557 res
+= (tmTimeOnly
.tm_hour
> 12) ? wxT("pm") : wxT("am");
558 #endif // HAVE_STRFTIME/!HAVE_STRFTIME
561 case _T('S'): // second as a decimal number (00-61)
562 res
+= wxString::Format(fmt
, tm
.sec
);
565 case _T('U'): // week number in the year (Sunday 1st week day)
566 res
+= wxString::Format(fmt
, GetWeekOfYear(Sunday_First
, tz
));
569 case _T('W'): // week number in the year (Monday 1st week day)
570 res
+= wxString::Format(fmt
, GetWeekOfYear(Monday_First
, tz
));
573 case _T('w'): // weekday as a number (0-6), Sunday = 0
574 res
+= wxString::Format(fmt
, tm
.GetWeekDay());
577 // case _T('x'): -- handled with "%c"
579 case _T('X'): // locale default time representation
580 // just use strftime() to format the time for us
582 res
+= CallStrftime(_T("%X"), &tmTimeOnly
);
583 #else // !HAVE_STRFTIME
584 res
+= wxString::Format(wxT("%02d:%02d:%02d"),tm
.hour
, tm
.min
, tm
.sec
);
585 #endif // HAVE_STRFTIME/!HAVE_STRFTIME
588 case _T('y'): // year without century (00-99)
589 res
+= wxString::Format(fmt
, tm
.year
% 100);
592 case _T('Y'): // year with century
593 res
+= wxString::Format(fmt
, tm
.year
);
596 case _T('Z'): // timezone name
598 res
+= CallStrftime(_T("%Z"), &tmTimeOnly
);
603 // is it the format width?
605 while ( *p
== _T('-') || *p
== _T('+') ||
606 *p
== _T(' ') || wxIsdigit(*p
) )
613 // we've only got the flags and width so far in fmt
614 fmt
.Prepend(_T('%'));
622 // no, it wasn't the width
623 wxFAIL_MSG(_T("unknown format specificator"));
625 // fall through and just copy it nevertheless
627 case _T('%'): // a percent sign
631 case 0: // the end of string
632 wxFAIL_MSG(_T("missing format at the end of string"));
634 // just put the '%' which was the last char in format
644 // this function parses a string in (strict) RFC 822 format: see the section 5
645 // of the RFC for the detailed description, but briefly it's something of the
646 // form "Sat, 18 Dec 1999 00:48:30 +0100"
648 // this function is "strict" by design - it must reject anything except true
649 // RFC822 time specs.
651 wxDateTime::ParseRfc822Date(const wxString
& date
, wxString::const_iterator
*end
)
653 wxString::const_iterator p
= date
.begin();
656 static const int WDAY_LEN
= 3;
657 const wxString::const_iterator endWday
= p
+ WDAY_LEN
;
658 const wxString
wday(p
, endWday
);
659 if ( GetWeekDayFromName(wday
, Name_Abbr
, DateLang_English
) == Inv_WeekDay
)
661 //else: ignore week day for now, we could also check that it really
662 // corresponds to the specified date
666 // 2. separating comma
667 if ( *p
++ != ',' || *p
++ != ' ' )
671 if ( !wxIsdigit(*p
) )
674 wxDateTime_t day
= (wxDateTime_t
)(*p
++ - '0');
678 day
= (wxDateTime_t
)(day
+ (*p
++ - '0'));
685 static const int MONTH_LEN
= 3;
686 const wxString::const_iterator endMonth
= p
+ MONTH_LEN
;
687 const wxString
monName(p
, endMonth
);
688 Month mon
= GetMonthFromName(monName
, Name_Abbr
, DateLang_English
);
689 if ( mon
== Inv_Month
)
698 if ( !wxIsdigit(*p
) )
701 int year
= *p
++ - '0';
702 if ( !wxIsdigit(*p
) ) // should have at least 2 digits in the year
708 // is it a 2 digit year (as per original RFC 822) or a 4 digit one?
714 if ( !wxIsdigit(*p
) )
716 // no 3 digit years please
727 // 6. time in hh:mm:ss format with seconds being optional
728 if ( !wxIsdigit(*p
) )
731 wxDateTime_t hour
= (wxDateTime_t
)(*p
++ - '0');
733 if ( !wxIsdigit(*p
) )
737 hour
= (wxDateTime_t
)(hour
+ (*p
++ - '0'));
742 if ( !wxIsdigit(*p
) )
745 wxDateTime_t min
= (wxDateTime_t
)(*p
++ - '0');
747 if ( !wxIsdigit(*p
) )
751 min
+= (wxDateTime_t
)(*p
++ - '0');
753 wxDateTime_t sec
= 0;
757 if ( !wxIsdigit(*p
) )
760 sec
= (wxDateTime_t
)(*p
++ - '0');
762 if ( !wxIsdigit(*p
) )
766 sec
+= (wxDateTime_t
)(*p
++ - '0');
772 // 7. now the interesting part: the timezone
773 int offset
wxDUMMY_INITIALIZE(0);
774 if ( *p
== '-' || *p
== '+' )
776 // the explicit offset given: it has the form of hhmm
777 bool plus
= *p
++ == '+';
779 if ( !wxIsdigit(*p
) || !wxIsdigit(*(p
+ 1)) )
784 offset
= MIN_PER_HOUR
*(10*(*p
- '0') + (*(p
+ 1) - '0'));
788 if ( !wxIsdigit(*p
) || !wxIsdigit(*(p
+ 1)) )
792 offset
+= 10*(*p
- '0') + (*(p
+ 1) - '0');
801 // the symbolic timezone given: may be either military timezone or one
802 // of standard abbreviations
805 // military: Z = UTC, J unused, A = -1, ..., Y = +12
806 static const int offsets
[26] =
808 //A B C D E F G H I J K L M
809 -1, -2, -3, -4, -5, -6, -7, -8, -9, 0, -10, -11, -12,
810 //N O P R Q S T U V W Z Y Z
811 +1, +2, +3, +4, +5, +6, +7, +8, +9, +10, +11, +12, 0
814 if ( *p
< _T('A') || *p
> _T('Z') || *p
== _T('J') )
817 offset
= offsets
[*p
++ - 'A'];
822 const wxString
tz(p
, date
.end());
823 if ( tz
== _T("UT") || tz
== _T("UTC") || tz
== _T("GMT") )
825 else if ( tz
== _T("AST") )
827 else if ( tz
== _T("ADT") )
829 else if ( tz
== _T("EST") )
831 else if ( tz
== _T("EDT") )
833 else if ( tz
== _T("CST") )
835 else if ( tz
== _T("CDT") )
837 else if ( tz
== _T("MST") )
839 else if ( tz
== _T("MDT") )
841 else if ( tz
== _T("PST") )
843 else if ( tz
== _T("PDT") )
852 offset
*= MIN_PER_HOUR
;
856 // the spec was correct, construct the date from the values we found
857 Set(day
, mon
, year
, hour
, min
, sec
);
858 MakeFromTimezone(TimeZone::Make(offset
*SEC_PER_MIN
));
867 wxDateTime::ParseFormat(const wxString
& date
,
868 const wxString
& format
,
869 const wxDateTime
& dateDef
,
870 wxString::const_iterator
*endParse
)
872 wxCHECK_MSG( !format
.empty(), false, "format can't be empty" );
873 wxCHECK_MSG( endParse
, false, "end iterator pointer must be specified" );
878 // what fields have we found?
879 bool haveWDay
= false,
889 bool hourIsIn12hFormat
= false, // or in 24h one?
890 isPM
= false; // AM by default
892 // and the value of the items we have (init them to get rid of warnings)
893 wxDateTime_t msec
= 0,
897 WeekDay wday
= Inv_WeekDay
;
898 wxDateTime_t yday
= 0,
900 wxDateTime::Month mon
= Inv_Month
;
903 wxString::const_iterator input
= date
.begin();
904 const wxString::const_iterator end
= date
.end();
905 for ( wxString::const_iterator fmt
= format
.begin(); fmt
!= format
.end(); ++fmt
)
907 if ( *fmt
!= _T('%') )
909 if ( wxIsspace(*fmt
) )
911 // a white space in the format string matches 0 or more white
912 // spaces in the input
913 while ( input
!= end
&& wxIsspace(*input
) )
920 // any other character (not whitespace, not '%') must be
921 // matched by itself in the input
922 if ( input
== end
|| *input
++ != *fmt
)
929 // done with this format char
933 // start of a format specification
935 // parse the optional width
937 while ( wxIsdigit(*++fmt
) )
943 // the default widths for the various fields
946 switch ( (*fmt
).GetValue() )
948 case _T('Y'): // year has 4 digits
952 case _T('j'): // day of year has 3 digits
953 case _T('l'): // milliseconds have 3 digits
957 case _T('w'): // week day as number has only one
962 // default for all other fields
967 // then the format itself
968 switch ( (*fmt
).GetValue() )
970 case _T('a'): // a weekday name
973 wday
= GetWeekDayFromName
975 GetAlphaToken(input
, end
),
976 *fmt
== 'a' ? Name_Abbr
: Name_Full
,
979 if ( wday
== Inv_WeekDay
)
988 case _T('b'): // a month name
991 mon
= GetMonthFromName
993 GetAlphaToken(input
, end
),
994 *fmt
== 'b' ? Name_Abbr
: Name_Full
,
997 if ( mon
== Inv_Month
)
1006 case _T('c'): // locale default date and time representation
1011 fmtDateTime
= wxLocale::GetInfo(wxLOCALE_DATE_TIME_FMT
);
1012 if ( !fmtDateTime
.empty() )
1013 dt
= ParseFormatAt(input
, end
, fmtDateTime
);
1015 if ( !dt
.IsValid() )
1017 // also try the format which corresponds to ctime()
1018 // output (i.e. the "C" locale default)
1019 dt
= ParseFormatAt(input
, end
, wxS("%a %b %d %H:%M:%S %Y"));
1022 if ( !dt
.IsValid() )
1024 // and finally also the two generic date/time formats
1025 dt
= ParseFormatAt(input
, end
, wxS("%x %X"), wxS("%X %x"));
1028 if ( !dt
.IsValid() )
1031 const Tm tm
= dt
.GetTm();
1041 haveDay
= haveMon
= haveYear
=
1042 haveHour
= haveMin
= haveSec
= true;
1046 case _T('d'): // day of a month (01-31)
1047 if ( !GetNumericToken(width
, input
, end
, &num
) ||
1048 (num
> 31) || (num
< 1) )
1054 // we can't check whether the day range is correct yet, will
1055 // do it later - assume ok for now
1057 mday
= (wxDateTime_t
)num
;
1060 case _T('H'): // hour in 24h format (00-23)
1061 if ( !GetNumericToken(width
, input
, end
, &num
) || (num
> 23) )
1068 hour
= (wxDateTime_t
)num
;
1071 case _T('I'): // hour in 12h format (01-12)
1072 if ( !GetNumericToken(width
, input
, end
, &num
) ||
1073 !num
|| (num
> 12) )
1080 hourIsIn12hFormat
= true;
1081 hour
= (wxDateTime_t
)(num
% 12); // 12 should be 0
1084 case _T('j'): // day of the year
1085 if ( !GetNumericToken(width
, input
, end
, &num
) ||
1086 !num
|| (num
> 366) )
1093 yday
= (wxDateTime_t
)num
;
1096 case _T('l'): // milliseconds (0-999)
1097 if ( !GetNumericToken(width
, input
, end
, &num
) )
1101 msec
= (wxDateTime_t
)num
;
1104 case _T('m'): // month as a number (01-12)
1105 if ( !GetNumericToken(width
, input
, end
, &num
) ||
1106 !num
|| (num
> 12) )
1113 mon
= (Month
)(num
- 1);
1116 case _T('M'): // minute as a decimal number (00-59)
1117 if ( !GetNumericToken(width
, input
, end
, &num
) ||
1125 min
= (wxDateTime_t
)num
;
1128 case _T('p'): // AM or PM string
1130 wxString am
, pm
, token
= GetAlphaToken(input
, end
);
1132 // some locales have empty AM/PM tokens and thus when formatting
1133 // dates with the %p specifier nothing is generated; when trying to
1134 // parse them back, we get an empty token here... but that's not
1139 GetAmPmStrings(&am
, &pm
);
1140 if (am
.empty() && pm
.empty())
1141 return false; // no am/pm strings defined
1142 if ( token
.CmpNoCase(pm
) == 0 )
1146 else if ( token
.CmpNoCase(am
) != 0 )
1154 case _T('r'): // time as %I:%M:%S %p
1157 if ( !dt
.ParseFormat(wxString(input
, end
),
1158 wxS("%I:%M:%S %p"), &input
) )
1161 haveHour
= haveMin
= haveSec
= true;
1163 const Tm tm
= dt
.GetTm();
1170 case _T('R'): // time as %H:%M
1173 dt
= ParseFormatAt(input
, end
, wxS("%H:%M"));
1174 if ( !dt
.IsValid() )
1180 const Tm tm
= dt
.GetTm();
1186 case _T('S'): // second as a decimal number (00-61)
1187 if ( !GetNumericToken(width
, input
, end
, &num
) ||
1195 sec
= (wxDateTime_t
)num
;
1198 case _T('T'): // time as %H:%M:%S
1201 dt
= ParseFormatAt(input
, end
, wxS("%H:%M:%S"));
1202 if ( !dt
.IsValid() )
1209 const Tm tm
= dt
.GetTm();
1216 case _T('w'): // weekday as a number (0-6), Sunday = 0
1217 if ( !GetNumericToken(width
, input
, end
, &num
) ||
1225 wday
= (WeekDay
)num
;
1228 case _T('x'): // locale default date representation
1231 fmtDate
= wxLocale::GetInfo(wxLOCALE_SHORT_DATE_FMT
),
1232 fmtDateAlt
= wxLocale::GetInfo(wxLOCALE_LONG_DATE_FMT
);
1234 if ( fmtDate
.empty() )
1236 if ( IsWestEuropeanCountry(GetCountry()) ||
1237 GetCountry() == Russia
)
1239 fmtDate
= wxS("%d/%m/%Y");
1240 fmtDateAlt
= wxS("%m/%d/%Y");
1244 fmtDate
= wxS("%m/%d/%Y");
1245 fmtDateAlt
= wxS("%d/%m/%Y");
1250 dt
= ParseFormatAt(input
, end
, fmtDate
, fmtDateAlt
);
1252 if ( !dt
.IsValid() )
1254 // try with short years too
1255 fmtDate
.Replace("%Y","%y");
1256 fmtDateAlt
.Replace("%Y","%y");
1257 dt
= ParseFormatAt(input
, end
, fmtDate
, fmtDateAlt
);
1259 if ( !dt
.IsValid() )
1263 const Tm tm
= dt
.GetTm();
1276 case _T('X'): // locale default time representation
1278 wxString fmtTime
= wxLocale::GetInfo(wxLOCALE_TIME_FMT
),
1281 if ( fmtTime
.empty() )
1283 // try to parse what follows as "%H:%M:%S" and, if this
1284 // fails, as "%I:%M:%S %p" - this should catch the most
1291 dt
= ParseFormatAt(input
, end
, fmtTime
, fmtTimeAlt
);
1292 if ( !dt
.IsValid() )
1299 const Tm tm
= dt
.GetTm();
1306 case _T('y'): // year without century (00-99)
1307 if ( !GetNumericToken(width
, input
, end
, &num
) ||
1316 // TODO should have an option for roll over date instead of
1317 // hard coding it here
1318 year
= (num
> 30 ? 1900 : 2000) + (wxDateTime_t
)num
;
1321 case _T('Y'): // year with century
1322 if ( !GetNumericToken(width
, input
, end
, &num
) )
1329 year
= (wxDateTime_t
)num
;
1332 case _T('Z'): // timezone name
1333 // FIXME: currently we just ignore everything that looks like a
1335 GetAlphaToken(input
, end
);
1338 case _T('%'): // a percent sign
1339 if ( *input
++ != _T('%') )
1346 case 0: // the end of string
1347 wxFAIL_MSG(_T("unexpected format end"));
1351 default: // not a known format spec
1356 // format matched, try to construct a date from what we have now
1358 if ( dateDef
.IsValid() )
1360 // take this date as default
1361 tmDef
= dateDef
.GetTm();
1363 else if ( IsValid() )
1365 // if this date is valid, don't change it
1370 // no default and this date is invalid - fall back to Today()
1371 tmDef
= Today().GetTm();
1387 // TODO we don't check here that the values are consistent, if both year
1388 // day and month/day were found, we just ignore the year day and we
1389 // also always ignore the week day
1392 if ( mday
> GetNumberOfDays(tm
.mon
, tm
.year
) )
1397 else if ( haveYDay
)
1399 if ( yday
> GetNumberOfDays(tm
.year
) )
1402 Tm tm2
= wxDateTime(1, Jan
, tm
.year
).SetToYearDay(yday
).GetTm();
1409 if ( haveHour
&& hourIsIn12hFormat
&& isPM
)
1411 // translate to 24hour format
1414 //else: either already in 24h format or no translation needed
1437 // finally check that the week day is consistent -- if we had it
1438 if ( haveWDay
&& GetWeekDay() != wday
)
1447 wxDateTime::ParseDateTime(const wxString
& date
, wxString::const_iterator
*end
)
1449 wxCHECK_MSG( end
, false, "end iterator pointer must be specified" );
1451 // Set to current day and hour, so strings like '14:00' becomes today at
1452 // 14, not some other random date
1453 wxDateTime dtDate
= wxDateTime::Today();
1454 wxDateTime dtTime
= wxDateTime::Today();
1456 wxString::const_iterator
1461 // If we got a date in the beginning, see if there is a time specified
1463 if ( dtDate
.ParseDate(date
, &endDate
) )
1465 // Skip spaces, as the ParseTime() function fails on spaces
1466 while ( endDate
!= date
.end() && wxIsspace(*endDate
) )
1469 const wxString
timestr(endDate
, date
.end());
1470 if ( !dtTime
.ParseTime(timestr
, &endTime
) )
1473 endBoth
= endDate
+ (endTime
- timestr
.begin());
1475 else // no date in the beginning
1477 // check if we have a time followed by a date
1478 if ( !dtTime
.ParseTime(date
, &endTime
) )
1481 while ( endTime
!= date
.end() && wxIsspace(*endTime
) )
1484 const wxString
datestr(endTime
, date
.end());
1485 if ( !dtDate
.ParseDate(datestr
, &endDate
) )
1488 endBoth
= endTime
+ (endDate
- datestr
.begin());
1491 Set(dtDate
.GetDay(), dtDate
.GetMonth(), dtDate
.GetYear(),
1492 dtTime
.GetHour(), dtTime
.GetMinute(), dtTime
.GetSecond(),
1493 dtTime
.GetMillisecond());
1501 wxDateTime::ParseDate(const wxString
& date
, wxString::const_iterator
*end
)
1503 wxCHECK_MSG( end
, false, "end iterator pointer must be specified" );
1505 // this is a simplified version of ParseDateTime() which understands only
1506 // "today" (for wxDate compatibility) and digits only otherwise (and not
1507 // all esoteric constructions ParseDateTime() knows about)
1509 const wxString::const_iterator pBegin
= date
.begin();
1511 wxString::const_iterator p
= pBegin
;
1512 while ( wxIsspace(*p
) )
1515 // some special cases
1519 int dayDiffFromToday
;
1522 { wxTRANSLATE("today"), 0 },
1523 { wxTRANSLATE("yesterday"), -1 },
1524 { wxTRANSLATE("tomorrow"), 1 },
1527 const size_t lenRest
= date
.end() - p
;
1528 for ( size_t n
= 0; n
< WXSIZEOF(literalDates
); n
++ )
1530 const wxString dateStr
= wxGetTranslation(literalDates
[n
].str
);
1531 size_t len
= dateStr
.length();
1533 if ( len
> lenRest
)
1536 const wxString::const_iterator pEnd
= p
+ len
;
1537 if ( wxString(p
, pEnd
).CmpNoCase(dateStr
) == 0 )
1539 // nothing can follow this, so stop here
1543 int dayDiffFromToday
= literalDates
[n
].dayDiffFromToday
;
1545 if ( dayDiffFromToday
)
1547 *this += wxDateSpan::Days(dayDiffFromToday
);
1556 // We try to guess what we have here: for each new (numeric) token, we
1557 // determine if it can be a month, day or a year. Of course, there is an
1558 // ambiguity as some numbers may be days as well as months, so we also
1559 // have the ability to back track.
1562 bool haveDay
= false, // the months day?
1563 haveWDay
= false, // the day of week?
1564 haveMon
= false, // the month?
1565 haveYear
= false; // the year?
1567 // and the value of the items we have (init them to get rid of warnings)
1568 WeekDay wday
= Inv_WeekDay
;
1569 wxDateTime_t day
= 0;
1570 wxDateTime::Month mon
= Inv_Month
;
1573 // tokenize the string
1575 static const wxStringCharType
*dateDelimiters
= wxS(".,/-\t\r\n ");
1576 wxStringTokenizer
tok(wxString(p
, date
.end()), dateDelimiters
);
1577 while ( tok
.HasMoreTokens() )
1579 wxString token
= tok
.GetNextToken();
1585 if ( token
.ToULong(&val
) )
1587 // guess what this number is
1593 if ( !haveMon
&& val
> 0 && val
<= 12 )
1595 // assume it is month
1598 else // not the month
1602 // this can only be the year
1605 else // may be either day or year
1607 // use a leap year if we don't have the year yet to allow
1608 // dates like 2/29/1976 which would be rejected otherwise
1609 wxDateTime_t max_days
= (wxDateTime_t
)(
1611 ? GetNumberOfDays(mon
, haveYear
? year
: 1976)
1616 if ( (val
== 0) || (val
> (unsigned long)max_days
) )
1621 else // yes, suppose it's the day
1635 year
= (wxDateTime_t
)val
;
1644 day
= (wxDateTime_t
)val
;
1650 mon
= (Month
)(val
- 1);
1653 else // not a number
1655 // be careful not to overwrite the current mon value
1656 Month mon2
= GetMonthFromName
1659 Name_Full
| Name_Abbr
,
1660 DateLang_Local
| DateLang_English
1662 if ( mon2
!= Inv_Month
)
1667 // but we already have a month - maybe we guessed wrong?
1670 // no need to check in month range as always < 12, but
1671 // the days are counted from 1 unlike the months
1672 day
= (wxDateTime_t
)(mon
+ 1);
1677 // could possible be the year (doesn't the year come
1678 // before the month in the japanese format?) (FIXME)
1687 else // not a valid month name
1689 WeekDay wday2
= GetWeekDayFromName
1692 Name_Full
| Name_Abbr
,
1693 DateLang_Local
| DateLang_English
1695 if ( wday2
!= Inv_WeekDay
)
1707 else // not a valid weekday name
1710 static const char *ordinals
[] =
1712 wxTRANSLATE("first"),
1713 wxTRANSLATE("second"),
1714 wxTRANSLATE("third"),
1715 wxTRANSLATE("fourth"),
1716 wxTRANSLATE("fifth"),
1717 wxTRANSLATE("sixth"),
1718 wxTRANSLATE("seventh"),
1719 wxTRANSLATE("eighth"),
1720 wxTRANSLATE("ninth"),
1721 wxTRANSLATE("tenth"),
1722 wxTRANSLATE("eleventh"),
1723 wxTRANSLATE("twelfth"),
1724 wxTRANSLATE("thirteenth"),
1725 wxTRANSLATE("fourteenth"),
1726 wxTRANSLATE("fifteenth"),
1727 wxTRANSLATE("sixteenth"),
1728 wxTRANSLATE("seventeenth"),
1729 wxTRANSLATE("eighteenth"),
1730 wxTRANSLATE("nineteenth"),
1731 wxTRANSLATE("twentieth"),
1732 // that's enough - otherwise we'd have problems with
1733 // composite (or not) ordinals
1737 for ( n
= 0; n
< WXSIZEOF(ordinals
); n
++ )
1739 if ( token
.CmpNoCase(ordinals
[n
]) == 0 )
1745 if ( n
== WXSIZEOF(ordinals
) )
1747 // stop here - something unknown
1754 // don't try anything here (as in case of numeric day
1755 // above) - the symbolic day spec should always
1756 // precede the month/year
1762 day
= (wxDateTime_t
)(n
+ 1);
1767 nPosCur
= tok
.GetPosition();
1770 // either no more tokens or the scan was stopped by something we couldn't
1771 // parse - in any case, see if we can construct a date from what we have
1772 if ( !haveDay
&& !haveWDay
)
1775 if ( haveWDay
&& (haveMon
|| haveYear
|| haveDay
) &&
1776 !(haveDay
&& haveMon
&& haveYear
) )
1778 // without adjectives (which we don't support here) the week day only
1779 // makes sense completely separately or with the full date
1780 // specification (what would "Wed 1999" mean?)
1784 if ( !haveWDay
&& haveYear
&& !(haveDay
&& haveMon
) )
1786 // may be we have month and day instead of day and year?
1787 if ( haveDay
&& !haveMon
)
1791 // exchange day and month
1792 mon
= (wxDateTime::Month
)(day
- 1);
1794 // we're in the current year then
1795 if ( (year
> 0) && (year
<= (int)GetNumberOfDays(mon
, Inv_Year
)) )
1797 day
= (wxDateTime_t
)year
;
1802 //else: no, can't exchange, leave haveMon == false
1812 mon
= GetCurrentMonth();
1817 year
= GetCurrentYear();
1822 // normally we check the day above but the check is optimistic in case
1823 // we find the day before its month/year so we have to redo it now
1824 if ( day
> GetNumberOfDays(mon
, year
) )
1827 Set(day
, mon
, year
);
1831 // check that it is really the same
1832 if ( GetWeekDay() != wday
)
1840 SetToWeekDayInSameWeek(wday
);
1843 // return the pointer to the first unparsed char
1845 if ( nPosCur
&& wxStrchr(dateDelimiters
, *(p
- 1)) )
1847 // if we couldn't parse the token after the delimiter, put back the
1848 // delimiter as well
1858 wxDateTime::ParseTime(const wxString
& time
, wxString::const_iterator
*end
)
1860 wxCHECK_MSG( end
, false, "end iterator pointer must be specified" );
1862 // first try some extra things
1869 { wxTRANSLATE("noon"), 12 },
1870 { wxTRANSLATE("midnight"), 00 },
1874 for ( size_t n
= 0; n
< WXSIZEOF(stdTimes
); n
++ )
1876 const wxString timeString
= wxGetTranslation(stdTimes
[n
].name
);
1877 const wxString::const_iterator p
= time
.begin() + timeString
.length();
1878 if ( timeString
.CmpNoCase(wxString(time
.begin(), p
)) == 0 )
1880 // casts required by DigitalMars
1881 Set(stdTimes
[n
].hour
, wxDateTime_t(0), wxDateTime_t(0));
1890 // try all time formats we may think about in the order from longest to
1892 static const char *timeFormats
[] =
1894 "%I:%M:%S %p", // 12hour with AM/PM
1895 "%H:%M:%S", // could be the same or 24 hour one so try it too
1896 "%I:%M %p", // 12hour with AM/PM but without seconds
1897 "%H:%M:%S", // and a possibly 24 hour version without seconds
1898 "%X", // possibly something from above or maybe something
1899 // completely different -- try it last
1901 // TODO: parse timezones
1904 for ( size_t nFmt
= 0; nFmt
< WXSIZEOF(timeFormats
); nFmt
++ )
1906 if ( ParseFormat(time
, timeFormats
[nFmt
], end
) )
1913 // ----------------------------------------------------------------------------
1914 // Workdays and holidays support
1915 // ----------------------------------------------------------------------------
1917 bool wxDateTime::IsWorkDay(Country
WXUNUSED(country
)) const
1919 return !wxDateTimeHolidayAuthority::IsHoliday(*this);
1922 // ============================================================================
1924 // ============================================================================
1926 wxDateSpan WXDLLIMPEXP_BASE
operator*(int n
, const wxDateSpan
& ds
)
1929 return ds1
.Multiply(n
);
1932 // ============================================================================
1934 // ============================================================================
1936 wxTimeSpan WXDLLIMPEXP_BASE
operator*(int n
, const wxTimeSpan
& ts
)
1938 return wxTimeSpan(ts
).Multiply(n
);
1941 // this enum is only used in wxTimeSpan::Format() below but we can't declare
1942 // it locally to the method as it provokes an internal compiler error in egcs
1943 // 2.91.60 when building with -O2
1954 // not all strftime(3) format specifiers make sense here because, for example,
1955 // a time span doesn't have a year nor a timezone
1957 // Here are the ones which are supported (all of them are supported by strftime
1959 // %H hour in 24 hour format
1960 // %M minute (00 - 59)
1961 // %S second (00 - 59)
1964 // Also, for MFC CTimeSpan compatibility, we support
1965 // %D number of days
1967 // And, to be better than MFC :-), we also have
1968 // %E number of wEeks
1969 // %l milliseconds (000 - 999)
1970 wxString
wxTimeSpan::Format(const wxString
& format
) const
1972 // we deal with only positive time spans here and just add the sign in
1973 // front for the negative ones
1976 wxString
str(Negate().Format(format
));
1980 wxCHECK_MSG( !format
.empty(), wxEmptyString
,
1981 _T("NULL format in wxTimeSpan::Format") );
1984 str
.Alloc(format
.length());
1986 // Suppose we have wxTimeSpan ts(1 /* hour */, 2 /* min */, 3 /* sec */)
1988 // Then, of course, ts.Format("%H:%M:%S") must return "01:02:03", but the
1989 // question is what should ts.Format("%S") do? The code here returns "3273"
1990 // in this case (i.e. the total number of seconds, not just seconds % 60)
1991 // because, for me, this call means "give me entire time interval in
1992 // seconds" and not "give me the seconds part of the time interval"
1994 // If we agree that it should behave like this, it is clear that the
1995 // interpretation of each format specifier depends on the presence of the
1996 // other format specs in the string: if there was "%H" before "%M", we
1997 // should use GetMinutes() % 60, otherwise just GetMinutes() &c
1999 // we remember the most important unit found so far
2000 TimeSpanPart partBiggest
= Part_MSec
;
2002 for ( wxString::const_iterator pch
= format
.begin(); pch
!= format
.end(); ++pch
)
2006 if ( ch
== _T('%') )
2008 // the start of the format specification of the printf() below
2009 wxString
fmtPrefix(_T('%'));
2014 // the number of digits for the format string, 0 if unused
2015 unsigned digits
= 0;
2017 ch
= *++pch
; // get the format spec char
2021 wxFAIL_MSG( _T("invalid format character") );
2027 // skip the part below switch
2032 if ( partBiggest
< Part_Day
)
2038 partBiggest
= Part_Day
;
2043 partBiggest
= Part_Week
;
2049 if ( partBiggest
< Part_Hour
)
2055 partBiggest
= Part_Hour
;
2062 n
= GetMilliseconds().ToLong();
2063 if ( partBiggest
< Part_MSec
)
2067 //else: no need to reset partBiggest to Part_MSec, it is
2068 // the least significant one anyhow
2075 if ( partBiggest
< Part_Min
)
2081 partBiggest
= Part_Min
;
2088 n
= GetSeconds().ToLong();
2089 if ( partBiggest
< Part_Sec
)
2095 partBiggest
= Part_Sec
;
2104 fmtPrefix
<< _T("0") << digits
;
2107 str
+= wxString::Format(fmtPrefix
+ _T("ld"), n
);
2111 // normal character, just copy
2119 #endif // wxUSE_DATETIME