1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/msw/datectrl.cpp
3 // Purpose: wxDatePickerCtrl implementation
4 // Author: Vadim Zeitlin
8 // Copyright: (c) 2005 Vadim Zeitlin <vadim@wxwindows.org>
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 #include "wx/wxprec.h"
26 #if wxUSE_DATEPICKCTRL
31 #include "wx/dcclient.h"
32 #include "wx/msw/wrapwin.h"
33 #include "wx/msw/wrapcctl.h"
34 #include "wx/msw/private.h"
37 #include "wx/datectrl.h"
38 #include "wx/dynlib.h"
40 #define _WX_DEFINE_DATE_EVENTS_
41 #include "wx/dateevt.h"
43 // apparently some versions of mingw define these macros erroneously
44 #ifndef DateTime_GetSystemtime
45 #define DateTime_GetSystemtime DateTime_GetSystemTime
48 #ifndef DateTime_SetSystemtime
49 #define DateTime_SetSystemtime DateTime_SetSystemTime
52 IMPLEMENT_DYNAMIC_CLASS(wxDatePickerCtrl
, wxControl
)
54 // ============================================================================
56 // ============================================================================
58 // ----------------------------------------------------------------------------
59 // helpers for wxDateTime <-> SYSTEMTIME conversion
60 // ----------------------------------------------------------------------------
62 static inline void wxFromSystemTime(wxDateTime
*dt
, const SYSTEMTIME
& st
)
65 wx_static_cast(wxDateTime::Month
, wxDateTime::Jan
+ st
.wMonth
- 1),
70 static inline void wxToSystemTime(SYSTEMTIME
*st
, const wxDateTime
& dt
)
72 const wxDateTime::Tm
tm(dt
.GetTm());
74 st
->wYear
= (WXWORD
)tm
.year
;
75 st
->wMonth
= (WXWORD
)(tm
.mon
- wxDateTime::Jan
+ 1);
82 st
->wMilliseconds
= 0;
85 // ----------------------------------------------------------------------------
86 // wxDatePickerCtrl creation
87 // ----------------------------------------------------------------------------
90 wxDatePickerCtrl::Create(wxWindow
*parent
,
96 const wxValidator
& validator
,
99 // although we already call InitCommonControls() in app.cpp which is
100 // supposed to initialize all common controls, in comctl32.dll 4.72 (and
101 // presumably earlier versions 4.70 and 4.71, date time picker not being
102 // supported in < 4.70 anyhow) it does not do it and we have to initialize
104 static bool s_initDone
= false; // MT-ok: used from GUI thread only
107 if ( wxApp::GetComCtl32Version() < 470 )
109 wxLogError(_("This system doesn't support date picker control, please upgrade your version of comctl32.dll"));
114 #if wxUSE_DYNLIB_CLASS
115 INITCOMMONCONTROLSEX icex
;
116 icex
.dwSize
= sizeof(icex
);
117 icex
.dwICC
= ICC_DATE_CLASSES
;
119 wxDynamicLibrary
dllComCtl32(_T("comctl32.dll"), wxDL_VERBATIM
);
121 typedef BOOL (WINAPI
*ICCEx_t
)(INITCOMMONCONTROLSEX
*);
122 wxDYNLIB_FUNCTION( ICCEx_t
, InitCommonControlsEx
, dllComCtl32
);
124 if ( pfnInitCommonControlsEx
)
126 (*pfnInitCommonControlsEx
)(&icex
);
134 // use wxDP_SPIN if wxDP_DEFAULT (0) was given as style
135 if ( !(style
& wxDP_DROPDOWN
) )
138 // initialize the base class
139 if ( !CreateControl(parent
, id
, pos
, size
, style
, validator
, name
) )
142 // create the native control
143 if ( !MSWCreateControl(DATETIMEPICK_CLASS
, wxEmptyString
, pos
, size
) )
146 if ( dt
.IsValid() || (style
& wxDP_ALLOWNONE
) )
152 WXDWORD
wxDatePickerCtrl::MSWGetStyle(long style
, WXDWORD
*exstyle
) const
154 WXDWORD styleMSW
= wxDatePickerCtrlBase::MSWGetStyle(style
, exstyle
);
156 // although MSDN doesn't mention it, DTS_UPDOWN doesn't work with
158 if ( wxApp::GetComCtl32Version() > 472 && (style
& wxDP_SPIN
) )
159 styleMSW
|= DTS_UPDOWN
;
160 //else: drop down by default
162 #ifdef DTS_SHORTDATECENTURYFORMAT
163 if ( style
& wxDP_SHOWCENTURY
)
164 styleMSW
|= DTS_SHORTDATECENTURYFORMAT
;
166 #endif // DTS_SHORTDATECENTURYFORMAT
167 styleMSW
|= DTS_SHORTDATEFORMAT
;
169 if ( style
& wxDP_ALLOWNONE
)
170 styleMSW
|= DTS_SHOWNONE
;
175 // TODO: handle WM_WININICHANGE
177 // ----------------------------------------------------------------------------
178 // wxDatePickerCtrl geometry
179 // ----------------------------------------------------------------------------
181 wxSize
wxDatePickerCtrl::DoGetBestSize() const
183 wxClientDC
dc(wx_const_cast(wxDatePickerCtrl
*, this));
184 dc
.SetFont(GetFont());
186 // we can't use FormatDate() here as the CRT doesn't always use the same
187 // format as the date picker control
189 for ( int len
= 100; ; len
*= 2 )
193 LOCALE_USER_DEFAULT
, // the control should use the same
194 DATE_SHORTDATE
, // the format used by the control
195 NULL
, // use current date (we don't care)
196 NULL
, // no custom format
197 wxStringBuffer(s
, len
), // output buffer
198 len
// and its length
205 const DWORD rc
= ::GetLastError();
206 if ( rc
!= ERROR_INSUFFICIENT_BUFFER
)
208 wxLogApiError(_T("GetDateFormat"), rc
);
210 // fall back on wxDateTime, what else to do?
211 s
= wxDateTime::Today().FormatDate();
216 // the control adds a lot of extra space around separators
217 s
.Replace(_T(","), _T(" , "));
220 dc
.GetTextExtent(s
, &x
, &y
);
222 wxSize
best(x
+ 40 /* margin + arrows */, EDIT_HEIGHT_FROM_CHAR_HEIGHT(y
));
227 // ----------------------------------------------------------------------------
228 // wxDatePickerCtrl operations
229 // ----------------------------------------------------------------------------
231 void wxDatePickerCtrl::SetValue(const wxDateTime
& dt
)
233 wxCHECK_RET( dt
.IsValid() || HasFlag(wxDP_ALLOWNONE
),
234 _T("this control requires a valid date") );
238 wxToSystemTime(&st
, dt
);
239 if ( !DateTime_SetSystemtime(GetHwnd(),
240 dt
.IsValid() ? GDT_VALID
: GDT_NONE
,
243 wxLogDebug(_T("DateTime_SetSystemtime() failed"));
247 wxDateTime
wxDatePickerCtrl::GetValue() const
251 if ( DateTime_GetSystemtime(GetHwnd(), &st
) == GDT_VALID
)
253 wxFromSystemTime(&dt
, st
);
259 void wxDatePickerCtrl::SetRange(const wxDateTime
& dt1
, const wxDateTime
& dt2
)
266 wxToSystemTime(&st
[0], dt1
);
272 wxToSystemTime(&st
[1], dt2
);
276 if ( !DateTime_SetRange(GetHwnd(), flags
, st
) )
278 wxLogDebug(_T("DateTime_SetRange() failed"));
282 bool wxDatePickerCtrl::GetRange(wxDateTime
*dt1
, wxDateTime
*dt2
) const
286 DWORD flags
= DateTime_GetRange(GetHwnd(), st
);
289 if ( flags
& GDTR_MIN
)
290 wxFromSystemTime(dt1
, st
[0]);
292 *dt1
= wxDefaultDateTime
;
297 if ( flags
& GDTR_MAX
)
298 wxFromSystemTime(dt2
, st
[1]);
300 *dt2
= wxDefaultDateTime
;
306 // ----------------------------------------------------------------------------
307 // wxDatePickerCtrl events
308 // ----------------------------------------------------------------------------
311 wxDatePickerCtrl::MSWOnNotify(int idCtrl
, WXLPARAM lParam
, WXLPARAM
*result
)
313 NMHDR
* hdr
= (NMHDR
*)lParam
;
316 case DTN_DATETIMECHANGE
:
318 NMDATETIMECHANGE
*dtch
= (NMDATETIMECHANGE
*)hdr
;
320 if ( dtch
->dwFlags
== GDT_VALID
)
321 wxFromSystemTime(&dt
, dtch
->st
);
323 wxDateEvent
event(this, dt
, wxEVT_DATE_CHANGED
);
324 if ( GetEventHandler()->ProcessEvent(event
) )
332 return wxDatePickerCtrlBase::MSWOnNotify(idCtrl
, lParam
, result
);
335 #endif // wxUSE_DATEPICKCTRL