1 /////////////////////////////////////////////////////////////////////////////
2 // Name: generic/datectlg.cpp
3 // Purpose: generic wxDatePickerCtrl implementation
4 // Author: Andreas Pflug
8 // Copyright: (c) 2005 Andreas Pflug <pgadmin@pse-consulting.de>
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 #include "wx/wxprec.h"
27 #include "wx/bmpbuttn.h"
28 #include "wx/dialog.h"
29 #include "wx/dcmemory.h"
31 #include "wx/textctrl.h"
32 #include "wx/valtext.h"
35 #define _WX_DEFINE_DATE_EVENTS_
36 #include "wx/dateevt.h"
38 #include "wx/datectrl.h"
39 #include "wx/calctrl.h"
40 #include "wx/popupwin.h"
42 // ----------------------------------------------------------------------------
44 // ----------------------------------------------------------------------------
54 #ifndef DEFAULT_ITEM_WIDTH
55 #define DEFAULT_ITEM_WIDTH 100
58 // ============================================================================
59 // wxDatePickerCtrl implementation
60 // ============================================================================
62 BEGIN_EVENT_TABLE(wxDatePickerCtrl
, wxDatePickerCtrlBase
)
63 EVT_BUTTON(CTRLID_BTN
, wxDatePickerCtrl::OnClick
)
64 EVT_TEXT(CTRLID_TXT
, wxDatePickerCtrl::OnText
)
65 EVT_CHILD_FOCUS(wxDatePickerCtrl::OnChildSetFocus
)
68 IMPLEMENT_DYNAMIC_CLASS(wxDatePickerCtrl
, wxDatePickerCtrlBase
)
70 // ----------------------------------------------------------------------------
72 // ----------------------------------------------------------------------------
74 wxDatePickerCtrl::wxDatePickerCtrl(wxWindow
*parent
,
76 const wxDateTime
& date
,
83 Create(parent
, id
, date
, pos
, size
, style
, name
);
87 bool wxDatePickerCtrl::Create(wxWindow
*parent
,
89 const wxDateTime
& date
,
95 wxASSERT_MSG( !(style
& wxDP_SPIN
),
96 _T("wxDP_SPIN style not supported, use wxDP_DEFAULT") );
98 if ( !wxControl::Create(parent
, id
, pos
, size
,
99 style
| wxCLIP_CHILDREN
| wxWANTS_CHARS
,
100 wxDefaultValidator
, name
) )
106 SetWindowStyle(style
| wxWANTS_CHARS
);
113 dc
.SelectObject(bmp
);
114 dc
.SetBrush(wxBrush(GetBackgroundColour()));
115 dc
.SetPen(wxPen(GetBackgroundColour()));
116 dc
.DrawRectangle(0,0, 8,4);
118 dc
.SetBrush(wxBrush(GetForegroundColour()));
119 dc
.SetPen(wxPen(GetForegroundColour()));
120 wxPoint pt
[3] = { wxPoint(0,0), wxPoint(6,0), wxPoint(3,3) };
121 dc
.DrawPolygon(3, pt
);
122 dc
.SelectObject(wxNullBitmap
);
125 m_txt
=new wxTextCtrl(this, CTRLID_TXT
);
126 m_txt
->Connect(wxID_ANY
, wxID_ANY
, wxEVT_KEY_DOWN
,
127 (wxObjectEventFunction
)&wxDatePickerCtrl::OnEditKey
,
129 m_txt
->Connect(wxID_ANY
, wxID_ANY
, wxEVT_KILL_FOCUS
,
130 (wxObjectEventFunction
)&wxDatePickerCtrl::OnKillFocus
,
133 m_btn
= new wxBitmapButton(this, CTRLID_BTN
, bmp
);
135 m_popup
= new wxPopupWindow(this);
136 m_popup
->SetFont(GetFont());
138 wxPanel
*panel
=new wxPanel(m_popup
, CTRLID_PAN
,
139 wxPoint(0, 0), wxDefaultSize
,
141 m_cal
= new wxCalendarCtrl(panel
, CTRLID_CAL
, wxDefaultDateTime
,
142 wxPoint(0,0), wxDefaultSize
,
143 wxCAL_SHOW_HOLIDAYS
| wxSUNKEN_BORDER
);
144 m_cal
->Connect(CTRLID_CAL
, CTRLID_CAL
, wxEVT_CALENDAR_SEL_CHANGED
,
145 (wxObjectEventFunction
)&wxDatePickerCtrl::OnSelChange
,
147 m_cal
->Connect(wxID_ANY
, wxID_ANY
, wxEVT_KEY_DOWN
,
148 (wxObjectEventFunction
)&wxDatePickerCtrl::OnCalKey
,
150 m_cal
->Connect(CTRLID_CAL
, CTRLID_CAL
, wxEVT_CALENDAR_DOUBLECLICKED
,
151 (wxObjectEventFunction
)&wxDatePickerCtrl::OnSelChange
,
153 m_cal
->Connect(CTRLID_CAL
, CTRLID_CAL
, wxEVT_CALENDAR_DAY_CHANGED
,
154 (wxObjectEventFunction
)&wxDatePickerCtrl::OnSelChange
,
156 m_cal
->Connect(CTRLID_CAL
, CTRLID_CAL
, wxEVT_CALENDAR_MONTH_CHANGED
,
157 (wxObjectEventFunction
)&wxDatePickerCtrl::OnSelChange
,
159 m_cal
->Connect(CTRLID_CAL
, CTRLID_CAL
, wxEVT_CALENDAR_YEAR_CHANGED
,
160 (wxObjectEventFunction
)&wxDatePickerCtrl::OnSelChange
,
163 wxWindow
*yearControl
= m_cal
->GetYearControl();
165 Connect(wxID_ANY
, wxID_ANY
, wxEVT_SET_FOCUS
,
166 (wxObjectEventFunction
)&wxDatePickerCtrl::OnSetFocus
);
168 wxClientDC
dc(yearControl
);
170 wxCoord width
, dummy
;
171 dc
.GetTextExtent(wxT("2000"), &width
, &dummy
);
172 width
+= ConvertDialogToPixels(wxSize(20,0)).x
;
174 wxSize calSize
= m_cal
->GetBestSize();
175 wxSize yearSize
= yearControl
->GetSize();
178 wxPoint yearPosition
= yearControl
->GetPosition();
180 SetFormat(wxT("%x"));
183 m_txt
->SetValue(date
.Format(m_format
));
192 width
= yearPosition
.x
+ yearSize
.x
+2+CALBORDER
/2;
193 if (width
< calSize
.x
-4)
196 int calPos
= (width
-calSize
.x
)/2;
202 m_cal
->SetSize(calPos
, 0, calSize
.x
, calSize
.y
);
203 yearControl
->SetSize(width
-yearSize
.x
-CALBORDER
/2, yearPosition
.y
,
204 yearSize
.x
, yearSize
.y
);
205 m_cal
->GetMonthControl()->Move(0, 0);
209 panel
->SetClientSize(width
+CALBORDER
/2, calSize
.y
-2+CALBORDER
);
210 m_popup
->SetClientSize(panel
->GetSize());
217 void wxDatePickerCtrl::Init()
225 m_ignoreDrop
= false;
229 bool wxDatePickerCtrl::Destroy()
245 return wxControl::Destroy();
248 // ----------------------------------------------------------------------------
249 // overridden base class methods
250 // ----------------------------------------------------------------------------
252 void wxDatePickerCtrl::DoMoveWindow(int x
, int y
, int w
, int h
)
254 wxControl::DoMoveWindow(x
, y
, w
, h
);
255 wxSize bs
=m_btn
->GetBestSize();
256 int eh
=m_txt
->GetBestSize().y
;
258 m_txt
->SetSize(0, 0, w
-bs
.x
-1, h
> eh
? eh
: h
);
259 m_btn
->SetSize(w
- bs
.x
, 0, bs
.x
, h
> bs
.y
? bs
.y
: h
);
265 wxSize
wxDatePickerCtrl::DoGetBestSize() const
267 int bh
=m_btn
->GetBestSize().y
;
268 int eh
=m_txt
->GetBestSize().y
;
269 return wxSize(DEFAULT_ITEM_WIDTH
, bh
> eh
? bh
: eh
);
273 bool wxDatePickerCtrl::Show(bool show
)
275 if ( !wxControl::Show(show
) )
293 bool wxDatePickerCtrl::Enable(bool enable
)
295 if ( !wxControl::Enable(enable
) )
306 m_btn
->Enable(enable
);
310 // ----------------------------------------------------------------------------
311 // wxDatePickerCtrl API
312 // ----------------------------------------------------------------------------
315 wxDatePickerCtrl::SetDateRange(const wxDateTime
& lowerdate
,
316 const wxDateTime
& upperdate
)
318 return m_cal
->SetDateRange(lowerdate
, upperdate
);
321 bool wxDatePickerCtrl::SetFormat(const wxChar
*fmt
)
323 wxString currentText
;
324 wxDateTime currentDate
;
327 currentText
= m_txt
->GetValue();
328 if (!currentText
.IsEmpty())
329 currentDate
.ParseFormat(currentText
, m_format
);
332 dt
.ParseFormat(wxT("2003-10-13"), wxT("%Y-%m-%d"));
333 wxString str
=dt
.Format(fmt
);
334 wxChar
*p
=(wxChar
*)str
.c_str();
336 m_format
=wxEmptyString
;
341 if (n
== dt
.GetDay())
343 m_format
.Append(wxT("%d"));
346 else if (n
== (int)dt
.GetMonth()+1)
348 m_format
.Append(wxT("%m"));
351 else if (n
== dt
.GetYear())
353 m_format
.Append(wxT("%Y"));
356 else if (n
== (dt
.GetYear() % 100))
358 m_format
.Append(wxT("%y"));
362 m_format
.Append(*p
++);
367 wxStringList valList
;
369 for (c
='0'; c
<= '9'; c
++)
370 valList
.Add(wxString(c
, 1));
371 wxChar
*p
=(wxChar
*)m_format
.c_str();
377 valList
.Add(wxString(*p
++, 1));
379 wxTextValidator
tv(wxFILTER_INCLUDE_CHAR_LIST
);
380 tv
.SetIncludeList(valList
);
382 m_txt
->SetValidator(tv
);
384 if (!currentText
.IsEmpty())
385 m_txt
->SetValue(currentDate
.Format(m_format
));
391 wxDateTime
wxDatePickerCtrl::GetValue() const
394 wxString txt
=m_txt
->GetValue();
397 dt
.ParseFormat(txt
, m_format
);
403 void wxDatePickerCtrl::SetValue(const wxDateTime
& date
)
408 m_txt
->SetValue(date
.Format(m_format
));
410 m_txt
->SetValue(wxEmptyString
);
415 bool wxDatePickerCtrl::GetRange(wxDateTime
*dt1
, wxDateTime
*dt2
) const
418 *dt1
= m_cal
->GetLowerDateLimit();
420 *dt2
= m_cal
->GetUpperDateLimit();
425 void wxDatePickerCtrl::SetRange(const wxDateTime
&dt1
, const wxDateTime
&dt2
)
427 m_cal
->SetDateRange(dt1
, dt2
);
430 // ----------------------------------------------------------------------------
432 // ----------------------------------------------------------------------------
434 void wxDatePickerCtrl::DropDown(bool down
)
441 if (!m_txt
->GetValue().IsEmpty())
442 dt
.ParseFormat(m_txt
->GetValue(), m_format
);
447 m_cal
->SetDate(wxDateTime::Today());
449 wxPoint pos
=GetParent()->ClientToScreen(GetPosition());
450 m_popup
->Move(pos
.x
, pos
.y
+ GetSize().y
);
464 void wxDatePickerCtrl::OnChildSetFocus(wxChildFocusEvent
&ev
)
467 m_ignoreDrop
= false;
469 wxWindow
*w
=(wxWindow
*)ev
.GetEventObject();
480 if (ev
.GetEventObject() == m_btn
)
486 void wxDatePickerCtrl::OnClick(wxCommandEvent
& event
)
490 m_ignoreDrop
= false;
501 void wxDatePickerCtrl::OnSetFocus(wxFocusEvent
&ev
)
506 m_txt
->SetSelection(0, 100);
511 void wxDatePickerCtrl::OnKillFocus(wxFocusEvent
&ev
)
516 dt
.ParseFormat(m_txt
->GetValue(), m_format
);
518 m_txt
->SetValue(wxEmptyString
);
520 m_txt
->SetValue(dt
.Format(m_format
));
524 void wxDatePickerCtrl::OnSelChange(wxCalendarEvent
&ev
)
528 m_txt
->SetValue(m_cal
->GetDate().Format(m_format
));
529 if (ev
.GetEventType() == wxEVT_CALENDAR_DOUBLECLICKED
)
535 ev
.SetEventObject(this);
537 GetParent()->ProcessEvent(ev
);
541 void wxDatePickerCtrl::OnText(wxCommandEvent
&ev
)
543 ev
.SetEventObject(this);
545 GetParent()->ProcessEvent(ev
);
547 // We'll create an additional event if the date is valid.
548 // If the date isn't valid, the user's probable in the middle of typing
549 wxString txt
=m_txt
->GetValue();
553 dt
.ParseFormat(txt
, m_format
);
558 wxCalendarEvent
cev(m_cal
, wxEVT_CALENDAR_SEL_CHANGED
);
559 cev
.SetEventObject(this);
563 GetParent()->ProcessEvent(cev
);
567 void wxDatePickerCtrl::OnEditKey(wxKeyEvent
& ev
)
569 if (ev
.GetKeyCode() == WXK_DOWN
&& !ev
.HasModifiers())
576 void wxDatePickerCtrl::OnCalKey(wxKeyEvent
& ev
)
578 if (ev
.GetKeyCode() == WXK_ESCAPE
&& !ev
.HasModifiers())