added wxDateTime::SetFrom/GetAsMSWSysTime() (#9161)
[wxWidgets.git] / src / msw / datectrl.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/msw/datectrl.cpp
3 // Purpose: wxDatePickerCtrl implementation
4 // Author: Vadim Zeitlin
5 // Modified by:
6 // Created: 2005-01-09
7 // RCS-ID: $Id$
8 // Copyright: (c) 2005 Vadim Zeitlin <vadim@wxwindows.org>
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
11
12 // ============================================================================
13 // declarations
14 // ============================================================================
15
16 // ----------------------------------------------------------------------------
17 // headers
18 // ----------------------------------------------------------------------------
19
20 #include "wx/wxprec.h"
21
22 #ifdef __BORLANDC__
23 #pragma hdrstop
24 #endif
25
26 #if wxUSE_DATEPICKCTRL
27
28 #ifndef WX_PRECOMP
29 #include "wx/msw/wrapwin.h"
30 #include "wx/msw/wrapcctl.h" // include <commctrl.h> "properly"
31 #include "wx/app.h"
32 #include "wx/intl.h"
33 #include "wx/dcclient.h"
34 #include "wx/msw/private.h"
35 #endif
36
37 #include "wx/datectrl.h"
38
39 #include "wx/msw/private/datecontrols.h"
40
41 #include "wx/dateevt.h"
42
43 // apparently some versions of mingw define these macros erroneously
44 #ifndef DateTime_GetSystemtime
45 #define DateTime_GetSystemtime DateTime_GetSystemTime
46 #endif
47
48 #ifndef DateTime_SetSystemtime
49 #define DateTime_SetSystemtime DateTime_SetSystemTime
50 #endif
51
52 IMPLEMENT_DYNAMIC_CLASS(wxDatePickerCtrl, wxControl)
53
54 // ============================================================================
55 // implementation
56 // ============================================================================
57
58 // ----------------------------------------------------------------------------
59 // wxDatePickerCtrl creation
60 // ----------------------------------------------------------------------------
61
62 bool
63 wxDatePickerCtrl::Create(wxWindow *parent,
64 wxWindowID id,
65 const wxDateTime& dt,
66 const wxPoint& pos,
67 const wxSize& size,
68 long style,
69 const wxValidator& validator,
70 const wxString& name)
71 {
72 if ( !wxMSWDateControls::CheckInitialization() )
73 return false;
74
75 // use wxDP_SPIN if wxDP_DEFAULT (0) was given as style
76 if ( !(style & wxDP_DROPDOWN) )
77 style |= wxDP_SPIN;
78
79 // initialize the base class
80 if ( !CreateControl(parent, id, pos, size, style, validator, name) )
81 return false;
82
83 // create the native control
84 if ( !MSWCreateControl(DATETIMEPICK_CLASS, wxEmptyString, pos, size) )
85 return false;
86
87 if ( dt.IsValid() || (style & wxDP_ALLOWNONE) )
88 SetValue(dt);
89 else
90 SetValue(wxDateTime::Today());
91
92 return true;
93 }
94
95 WXDWORD wxDatePickerCtrl::MSWGetStyle(long style, WXDWORD *exstyle) const
96 {
97 WXDWORD styleMSW = wxDatePickerCtrlBase::MSWGetStyle(style, exstyle);
98
99 // although MSDN doesn't mention it, DTS_UPDOWN doesn't work with
100 // comctl32.dll 4.72
101 if ( wxApp::GetComCtl32Version() > 472 && (style & wxDP_SPIN) )
102 styleMSW |= DTS_UPDOWN;
103 //else: drop down by default
104
105 #ifdef DTS_SHORTDATECENTURYFORMAT
106 if ( style & wxDP_SHOWCENTURY )
107 styleMSW |= DTS_SHORTDATECENTURYFORMAT;
108 else
109 #endif // DTS_SHORTDATECENTURYFORMAT
110 styleMSW |= DTS_SHORTDATEFORMAT;
111
112 if ( style & wxDP_ALLOWNONE )
113 styleMSW |= DTS_SHOWNONE;
114
115 return styleMSW;
116 }
117
118 // TODO: handle WM_WININICHANGE
119
120 // ----------------------------------------------------------------------------
121 // wxDatePickerCtrl geometry
122 // ----------------------------------------------------------------------------
123
124 wxSize wxDatePickerCtrl::DoGetBestSize() const
125 {
126 wxClientDC dc(wx_const_cast(wxDatePickerCtrl *, this));
127 dc.SetFont(GetFont());
128
129 // we can't use FormatDate() here as the CRT doesn't always use the same
130 // format as the date picker control
131 wxString s;
132 for ( int len = 100; ; len *= 2 )
133 {
134 if ( ::GetDateFormat
135 (
136 LOCALE_USER_DEFAULT, // the control should use the same
137 DATE_SHORTDATE, // the format used by the control
138 NULL, // use current date (we don't care)
139 NULL, // no custom format
140 wxStringBuffer(s, len), // output buffer
141 len // and its length
142 ) )
143 {
144 // success
145 break;
146 }
147
148 const DWORD rc = ::GetLastError();
149 if ( rc != ERROR_INSUFFICIENT_BUFFER )
150 {
151 wxLogApiError(_T("GetDateFormat"), rc);
152
153 // fall back on wxDateTime, what else to do?
154 s = wxDateTime::Today().FormatDate();
155 break;
156 }
157 }
158
159 // the control adds a lot of extra space around separators
160 s.Replace(_T(","), _T(" , "));
161
162 int x, y;
163 dc.GetTextExtent(s, &x, &y);
164
165 wxSize best(x + 40 /* margin + arrows */, EDIT_HEIGHT_FROM_CHAR_HEIGHT(y));
166 CacheBestSize(best);
167 return best;
168 }
169
170 // ----------------------------------------------------------------------------
171 // wxDatePickerCtrl operations
172 // ----------------------------------------------------------------------------
173
174 void wxDatePickerCtrl::SetValue(const wxDateTime& dt)
175 {
176 wxCHECK_RET( dt.IsValid() || HasFlag(wxDP_ALLOWNONE),
177 _T("this control requires a valid date") );
178
179 SYSTEMTIME st;
180 if ( dt.IsValid() )
181 dt.GetAsMSWSysTime(&st);
182 if ( !DateTime_SetSystemtime(GetHwnd(),
183 dt.IsValid() ? GDT_VALID : GDT_NONE,
184 &st) )
185 {
186 wxLogDebug(_T("DateTime_SetSystemtime() failed"));
187 }
188
189 // we need to keep only the date part, times don't make sense for this
190 // control (in particular, comparisons with other dates would fail)
191 m_date = dt;
192 if ( m_date.IsValid() )
193 m_date.ResetTime();
194 }
195
196 wxDateTime wxDatePickerCtrl::GetValue() const
197 {
198 #ifdef __WXDEBUG__
199 wxDateTime dt;
200 SYSTEMTIME st;
201 if ( DateTime_GetSystemtime(GetHwnd(), &st) == GDT_VALID )
202 {
203 dt.SetFromMSWSysTime(st);
204 }
205
206 wxASSERT_MSG( m_date.IsValid() == dt.IsValid() &&
207 (!dt.IsValid() || dt == m_date),
208 _T("bug in wxDatePickerCtrl: m_date not in sync") );
209 #endif // __WXDEBUG__
210
211 return m_date;
212 }
213
214 void wxDatePickerCtrl::SetRange(const wxDateTime& dt1, const wxDateTime& dt2)
215 {
216 SYSTEMTIME st[2];
217
218 DWORD flags = 0;
219 if ( dt1.IsValid() )
220 {
221 dt1.GetAsMSWSysTime(st + 0);
222 flags |= GDTR_MIN;
223 }
224
225 if ( dt2.IsValid() )
226 {
227 dt2.GetAsMSWSysTime(st + 1);
228 flags |= GDTR_MAX;
229 }
230
231 if ( !DateTime_SetRange(GetHwnd(), flags, st) )
232 {
233 wxLogDebug(_T("DateTime_SetRange() failed"));
234 }
235 }
236
237 bool wxDatePickerCtrl::GetRange(wxDateTime *dt1, wxDateTime *dt2) const
238 {
239 SYSTEMTIME st[2];
240
241 DWORD flags = DateTime_GetRange(GetHwnd(), st);
242 if ( dt1 )
243 {
244 if ( flags & GDTR_MIN )
245 dt1->SetFromMSWSysTime(st[0]);
246 else
247 *dt1 = wxDefaultDateTime;
248 }
249
250 if ( dt2 )
251 {
252 if ( flags & GDTR_MAX )
253 dt2->SetFromMSWSysTime(st[1]);
254 else
255 *dt2 = wxDefaultDateTime;
256 }
257
258 return flags != 0;
259 }
260
261 // ----------------------------------------------------------------------------
262 // wxDatePickerCtrl events
263 // ----------------------------------------------------------------------------
264
265 bool
266 wxDatePickerCtrl::MSWOnNotify(int idCtrl, WXLPARAM lParam, WXLPARAM *result)
267 {
268 NMHDR* hdr = (NMHDR *)lParam;
269 switch ( hdr->code )
270 {
271 case DTN_DATETIMECHANGE:
272 {
273 NMDATETIMECHANGE *dtch = (NMDATETIMECHANGE *)hdr;
274 wxDateTime dt;
275 if ( dtch->dwFlags == GDT_VALID )
276 dt.SetFromMSWSysTime(dtch->st);
277
278 // filter out duplicate DTN_DATETIMECHANGE events which the native
279 // control sends us when using wxDP_DROPDOWN style
280 if ( (m_date.IsValid() != dt.IsValid()) ||
281 (m_date.IsValid() && dt != m_date) )
282 {
283 m_date = dt;
284 wxDateEvent event(this, dt, wxEVT_DATE_CHANGED);
285 if ( HandleWindowEvent(event) )
286 {
287 *result = 0;
288 return true;
289 }
290 }
291 //else: both the old and new values are invalid, nothing changed
292 }
293 }
294
295 return wxDatePickerCtrlBase::MSWOnNotify(idCtrl, lParam, result);
296 }
297
298 #endif // wxUSE_DATEPICKCTRL