1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: generic/calctrl.cpp
3 // Purpose: implementation fo the generic wxCalendarCtrl
4 // Author: Vadim Zeitlin
8 // Copyright: (c) 1999 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9 // Licence: wxWindows license
10 ///////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
21 #pragma implementation "calctrl.h"
24 // For compilers that support precompilation, includes "wx.h".
25 #include "wx/wxprec.h"
32 #include "wx/dcclient.h"
33 #include "wx/settings.h"
37 #include "wx/calctrl.h"
41 // ----------------------------------------------------------------------------
43 // ----------------------------------------------------------------------------
45 class wxMonthComboBox
: public wxComboBox
48 wxMonthComboBox(wxCalendarCtrl
*cal
);
50 void OnMonthChange(wxCommandEvent
& event
) { m_cal
->OnMonthChange(event
); }
53 wxCalendarCtrl
*m_cal
;
58 class wxYearSpinCtrl
: public wxSpinCtrl
61 wxYearSpinCtrl(wxCalendarCtrl
*cal
);
63 void OnYearChange(wxSpinEvent
& event
) { m_cal
->OnYearChange(event
); }
66 wxCalendarCtrl
*m_cal
;
71 // ----------------------------------------------------------------------------
73 // ----------------------------------------------------------------------------
75 BEGIN_EVENT_TABLE(wxCalendarCtrl
, wxControl
)
76 EVT_PAINT(wxCalendarCtrl::OnPaint
)
78 EVT_CHAR(wxCalendarCtrl::OnChar
)
80 EVT_LEFT_DOWN(wxCalendarCtrl::OnClick
)
81 EVT_LEFT_DCLICK(wxCalendarCtrl::OnDClick
)
84 BEGIN_EVENT_TABLE(wxMonthComboBox
, wxComboBox
)
85 EVT_COMBOBOX(-1, wxMonthComboBox::OnMonthChange
)
88 BEGIN_EVENT_TABLE(wxYearSpinCtrl
, wxSpinCtrl
)
89 EVT_SPINCTRL(-1, wxYearSpinCtrl::OnYearChange
)
92 IMPLEMENT_DYNAMIC_CLASS(wxCalendarCtrl
, wxControl
)
94 // ============================================================================
96 // ============================================================================
98 // ----------------------------------------------------------------------------
99 // wxMonthComboBox and wxYearSpinCtrl
100 // ----------------------------------------------------------------------------
102 wxMonthComboBox::wxMonthComboBox(wxCalendarCtrl
*cal
)
103 : wxComboBox(cal
->GetParent(), -1,
113 for ( m
= wxDateTime::Jan
; m
< wxDateTime::Inv_Month
; wxNextMonth(m
) )
115 Append(wxDateTime::GetMonthName(m
));
118 SetSelection(m_cal
->GetDate().GetMonth());
121 wxYearSpinCtrl::wxYearSpinCtrl(wxCalendarCtrl
*cal
)
122 : wxSpinCtrl(cal
->GetParent(), -1,
123 cal
->GetDate().Format(_T("%Y")),
127 -4300, 10000, cal
->GetDate().GetYear())
132 // ----------------------------------------------------------------------------
134 // ----------------------------------------------------------------------------
136 void wxCalendarCtrl::Init()
144 wxDateTime::WeekDay wd
;
145 for ( wd
= wxDateTime::Sun
; wd
< wxDateTime::Inv_WeekDay
; wxNextWDay(wd
) )
147 m_weekdays
[wd
] = wxDateTime::GetWeekDayName(wd
, wxDateTime::Name_Abbr
);
151 bool wxCalendarCtrl::Create(wxWindow
* WXUNUSED(parent
),
152 wxWindowID
WXUNUSED(id
),
153 const wxDateTime
& date
,
154 const wxPoint
& WXUNUSED(pos
),
157 const wxString
& WXUNUSED(name
))
159 SetWindowStyle(style
| (wxRAISED_BORDER
| wxWANTS_CHARS
));
161 m_date
= date
.IsValid() ? date
: wxDateTime::Today();
163 m_comboMonth
= new wxMonthComboBox(this);
164 m_spinYear
= new wxYearSpinCtrl(this);
167 if ( size
.x
== -1 || size
.y
== -1 )
169 sizeReal
= DoGetBestSize();
182 SetBackgroundColour(*wxWHITE
);
183 SetFont(*wxSWISS_FONT
);
188 wxCalendarCtrl::~wxCalendarCtrl()
191 m_comboMonth
->PopEventHandler();
192 m_spinYear
->PopEventHandler();
196 // ----------------------------------------------------------------------------
197 // forward wxWin functions to subcontrols
198 // ----------------------------------------------------------------------------
200 bool wxCalendarCtrl::Show(bool show
)
202 if ( !wxControl::Show(show
) )
207 m_comboMonth
->Show(show
);
208 m_spinYear
->Show(show
);
213 bool wxCalendarCtrl::Enable(bool enable
)
215 if ( !wxControl::Enable(enable
) )
220 m_comboMonth
->Enable(enable
);
221 m_spinYear
->Enable(enable
);
226 // ----------------------------------------------------------------------------
228 // ----------------------------------------------------------------------------
230 void wxCalendarCtrl::SetDate(const wxDateTime
& date
)
232 if ( m_date
.GetMonth() == date
.GetMonth() &&
233 m_date
.GetYear() == date
.GetYear() )
235 // just change the day
243 // update the controls
244 m_comboMonth
->SetSelection(m_date
.GetMonth());
245 m_spinYear
->SetValue(m_date
.Format(_T("%Y")));
247 // update the calendar
252 void wxCalendarCtrl::ChangeDay(const wxDateTime
& date
)
254 if ( m_date
!= date
)
256 // we need to refresh the row containing the old date and the one
257 // containing the new one
258 wxDateTime dateOld
= m_date
;
261 RefreshDate(dateOld
);
263 // if the date is in the same row, it was already drawn correctly
264 if ( GetWeek(m_date
) != GetWeek(dateOld
) )
271 void wxCalendarCtrl::SetDateAndNotify(const wxDateTime
& date
)
273 wxDateTime::Tm tm1
= m_date
.GetTm(),
277 if ( tm1
.year
!= tm2
.year
)
278 type
= wxEVT_CALENDAR_YEAR_CHANGED
;
279 else if ( tm1
.mon
!= tm2
.mon
)
280 type
= wxEVT_CALENDAR_MONTH_CHANGED
;
281 else if ( tm1
.mday
!= tm2
.mday
)
282 type
= wxEVT_CALENDAR_DAY_CHANGED
;
291 // ----------------------------------------------------------------------------
293 // ----------------------------------------------------------------------------
295 wxDateTime
wxCalendarCtrl::GetStartDate() const
297 wxDateTime::Tm tm
= m_date
.GetTm();
299 wxDateTime date
= wxDateTime(1, tm
.mon
, tm
.year
);
302 date
.SetToPrevWeekDay(GetWindowStyle() & wxCAL_MONDAY_FIRST
303 ? wxDateTime::Mon
: wxDateTime::Sun
);
305 // be sure to do it or it might gain 1 hour if DST changed in between
311 bool wxCalendarCtrl::IsDateShown(const wxDateTime
& date
) const
313 return date
.GetMonth() == m_date
.GetMonth();
316 size_t wxCalendarCtrl::GetWeek(const wxDateTime
& date
) const
318 return date
.GetWeekOfMonth(GetWindowStyle() & wxCAL_MONDAY_FIRST
319 ? wxDateTime::Monday_First
320 : wxDateTime::Sunday_First
);
323 // ----------------------------------------------------------------------------
325 // ----------------------------------------------------------------------------
327 // this is a composite control and it must arrange its parts each time its
328 // size or position changes: the combobox and spinctrl are along the top of
329 // the available area and the calendar takes up therest of the space
331 // the constants used for the layout
332 #define VERT_MARGIN 5 // distance between combo and calendar
333 #define HORZ_MARGIN 15 // spin
335 wxSize
wxCalendarCtrl::DoGetBestSize() const
337 // calc the size of the calendar
338 ((wxCalendarCtrl
*)this)->RecalcGeometry(); // const_cast
340 wxCoord width
= 7*m_widthCol
,
341 height
= 7*m_heightRow
;
343 wxSize sizeCombo
= m_comboMonth
->GetBestSize(),
344 sizeSpin
= m_spinYear
->GetBestSize();
346 height
+= VERT_MARGIN
+ wxMax(sizeCombo
.y
, sizeSpin
.y
);
348 return wxSize(width
, height
);
351 void wxCalendarCtrl::DoSetSize(int x
, int y
,
352 int width
, int height
,
355 wxControl::DoSetSize(x
, y
, width
, height
, sizeFlags
);
358 void wxCalendarCtrl::DoMoveWindow(int x
, int y
, int width
, int height
)
360 wxSize sizeCombo
= m_comboMonth
->GetSize();
361 m_comboMonth
->Move(x
, y
);
363 int xDiff
= sizeCombo
.x
+ HORZ_MARGIN
;
364 m_spinYear
->SetSize(x
+ xDiff
, y
, width
- xDiff
, sizeCombo
.y
);
366 wxSize sizeSpin
= m_spinYear
->GetSize();
367 int yDiff
= wxMax(sizeSpin
.y
, sizeCombo
.y
) + VERT_MARGIN
;
369 wxControl::DoMoveWindow(x
, y
+ yDiff
, width
, height
- yDiff
);
372 void wxCalendarCtrl::DoGetPosition(int *x
, int *y
) const
374 wxControl::DoGetPosition(x
, y
);
376 // our real top corner is not in this position
379 *y
-= m_comboMonth
->GetSize().y
+ VERT_MARGIN
;
383 void wxCalendarCtrl::DoGetSize(int *width
, int *height
) const
385 wxControl::DoGetSize(width
, height
);
387 // our real height is bigger
390 *height
+= m_comboMonth
->GetSize().y
+ VERT_MARGIN
;
394 void wxCalendarCtrl::RecalcGeometry()
396 if ( m_widthCol
!= 0 )
403 // determine the column width (we assume that the weekday names are always
404 // wider (in any language) than the numbers)
406 wxDateTime::WeekDay wd
;
407 for ( wd
= wxDateTime::Sun
; wd
< wxDateTime::Inv_WeekDay
; wxNextWDay(wd
) )
410 dc
.GetTextExtent(m_weekdays
[wd
], &width
, &m_heightRow
);
411 if ( width
> m_widthCol
)
417 // leave some margins
422 // ----------------------------------------------------------------------------
424 // ----------------------------------------------------------------------------
426 void wxCalendarCtrl::OnPaint(wxPaintEvent
& WXUNUSED(event
))
435 printf("--- starting to paint, selection: %s, week %u\n",
436 m_date
.Format("%a %d-%m-%Y %H:%M:%S").c_str(),
440 // first draw the week days
441 if ( IsExposed(0, 0, 7*m_widthCol
, m_heightRow
) )
444 puts("painting the header");
447 dc
.SetTextForeground(*wxBLUE
);
448 dc
.SetBrush(wxBrush(*wxLIGHT_GREY
, wxSOLID
));
449 dc
.SetBackgroundMode(wxTRANSPARENT
);
450 dc
.SetPen(*wxLIGHT_GREY_PEN
);
451 dc
.DrawRectangle(0, 0, 7*m_widthCol
, m_heightRow
);
453 bool startOnMonday
= (GetWindowStyle() & wxCAL_MONDAY_FIRST
) != 0;
454 for ( size_t wd
= 0; wd
< 7; wd
++ )
458 n
= wd
== 6 ? 0 : wd
+ 1;
462 dc
.DrawText(m_weekdays
[n
], wd
*m_widthCol
+ 1, 0);
466 // then the calendar itself
467 dc
.SetTextForeground(*wxBLACK
);
468 //dc.SetFont(*wxNORMAL_FONT);
470 wxCoord y
= m_heightRow
;
472 wxDateTime date
= GetStartDate();
474 printf("starting calendar from %s\n",
475 date
.Format("%a %d-%m-%Y %H:%M:%S").c_str());
478 dc
.SetBackgroundMode(wxSOLID
);
479 for ( size_t nWeek
= 1; nWeek
<= 6; nWeek
++, y
+= m_heightRow
)
481 // if the update region doesn't intersect this row, don't paint it
482 if ( !IsExposed(0, y
, 7*m_widthCol
, m_heightRow
- 1) )
484 date
+= wxDateSpan::Week();
490 printf("painting week %d at y = %d\n", nWeek
, y
);
493 for ( size_t wd
= 0; wd
< 7; wd
++ )
495 if ( IsDateShown(date
) )
497 // don't use wxDate::Format() which prepends 0s
498 wxString day
= wxString::Format(_T("%u"), date
.GetDay());
500 dc
.GetTextExtent(day
, &width
, (wxCoord
*)NULL
);
502 bool isSel
= m_date
== date
;
505 dc
.SetTextForeground(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_HIGHLIGHTTEXT
));
506 dc
.SetTextBackground(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_HIGHLIGHT
));
509 dc
.DrawText(day
, wd
*m_widthCol
+ (m_widthCol
- width
) / 2, y
);
513 dc
.SetTextForeground(m_foregroundColour
);
514 dc
.SetTextBackground(m_backgroundColour
);
517 //else: just don't draw it
519 date
+= wxDateSpan::Day();
523 puts("+++ finished painting");
527 void wxCalendarCtrl::RefreshDate(const wxDateTime
& date
)
533 // always refresh the whole row at once because our OnPaint() will draw
534 // the whole row anyhow - and this allows the small optimisation in
535 // OnClick() below to work
537 rect
.y
= m_heightRow
* GetWeek(date
);
538 rect
.width
= 7*m_widthCol
;
539 rect
.height
= m_heightRow
;
542 printf("*** refreshing week %d at (%d, %d)-(%d, %d)\n",
545 rect
.x
+ rect
.width
, rect
.y
+ rect
.height
);
548 Refresh(TRUE
, &rect
);
551 // ----------------------------------------------------------------------------
553 // ----------------------------------------------------------------------------
555 void wxCalendarCtrl::OnDClick(wxMouseEvent
& event
)
557 if ( HitTest(event
.GetPosition()) != wxCAL_HITTEST_DAY
)
563 GenerateEvent(wxEVT_CALENDAR_DOUBLECLICKED
, FALSE
);
567 void wxCalendarCtrl::OnClick(wxMouseEvent
& event
)
570 wxDateTime::WeekDay wday
;
571 switch ( HitTest(event
.GetPosition(), &date
, &wday
) )
573 case wxCAL_HITTEST_DAY
:
576 GenerateEvent(wxEVT_CALENDAR_DAY_CHANGED
);
579 case wxCAL_HITTEST_HEADER
:
581 wxCalendarEvent
event(this, wxEVT_CALENDAR_WEEKDAY_CLICKED
);
583 (void)GetEventHandler()->ProcessEvent(event
);
588 wxFAIL_MSG(_T("unknown hittest code"));
591 case wxCAL_HITTEST_NOWHERE
:
597 wxCalendarHitTestResult
wxCalendarCtrl::HitTest(const wxPoint
& pos
,
599 wxDateTime::WeekDay
*wd
)
603 int wday
= pos
.x
/ m_widthCol
;
606 if ( y
< m_heightRow
)
610 if ( GetWindowStyle() & wxCAL_MONDAY_FIRST
)
612 wday
= wday
== 6 ? 0 : wday
+ 1;
615 *wd
= (wxDateTime::WeekDay
)wday
;
618 return wxCAL_HITTEST_HEADER
;
621 int week
= (y
- m_heightRow
) / m_heightRow
;
622 if ( week
>= 6 || wday
>= 7 )
624 return wxCAL_HITTEST_NOWHERE
;
627 wxDateTime dt
= GetStartDate() + wxDateSpan::Days(7*week
+ wday
);
629 if ( IsDateShown(dt
) )
634 return wxCAL_HITTEST_DAY
;
638 return wxCAL_HITTEST_NOWHERE
;
642 // ----------------------------------------------------------------------------
643 // subcontrols events handling
644 // ----------------------------------------------------------------------------
646 void wxCalendarCtrl::OnMonthChange(wxCommandEvent
& event
)
648 wxDateTime::Tm tm
= m_date
.GetTm();
650 wxDateTime::Month mon
= (wxDateTime::Month
)event
.GetInt();
651 if ( tm
.mday
> wxDateTime::GetNumberOfDays(mon
, tm
.year
) )
653 tm
.mday
= wxDateTime::GetNumberOfDays(mon
, tm
.year
);
656 SetDate(wxDateTime(tm
.mday
, mon
, tm
.year
));
658 GenerateEvent(wxEVT_CALENDAR_MONTH_CHANGED
);
661 void wxCalendarCtrl::OnYearChange(wxSpinEvent
& event
)
663 wxDateTime::Tm tm
= m_date
.GetTm();
665 int year
= (int)event
.GetInt();
666 if ( tm
.mday
> wxDateTime::GetNumberOfDays(tm
.mon
, year
) )
668 tm
.mday
= wxDateTime::GetNumberOfDays(tm
.mon
, year
);
671 SetDate(wxDateTime(tm
.mday
, tm
.mon
, year
));
673 GenerateEvent(wxEVT_CALENDAR_YEAR_CHANGED
);
676 // ----------------------------------------------------------------------------
677 // keyboard interface
678 // ----------------------------------------------------------------------------
680 void wxCalendarCtrl::OnChar(wxKeyEvent
& event
)
682 switch ( event
.KeyCode() )
686 SetDateAndNotify(m_date
+ wxDateSpan::Year());
691 SetDateAndNotify(m_date
- wxDateSpan::Year());
695 SetDateAndNotify(m_date
- wxDateSpan::Month());
699 SetDateAndNotify(m_date
+ wxDateSpan::Month());
703 if ( event
.ControlDown() )
704 SetDateAndNotify(wxDateTime(m_date
).SetToNextWeekDay(
705 GetWindowStyle() & wxCAL_MONDAY_FIRST
706 ? wxDateTime::Sun
: wxDateTime::Sat
));
708 SetDateAndNotify(m_date
+ wxDateSpan::Day());
712 if ( event
.ControlDown() )
713 SetDateAndNotify(wxDateTime(m_date
).SetToPrevWeekDay(
714 GetWindowStyle() & wxCAL_MONDAY_FIRST
715 ? wxDateTime::Mon
: wxDateTime::Sun
));
717 SetDateAndNotify(m_date
- wxDateSpan::Day());
721 SetDateAndNotify(m_date
- wxDateSpan::Week());
725 SetDateAndNotify(m_date
+ wxDateSpan::Week());
729 if ( event
.ControlDown() )
730 SetDateAndNotify(wxDateTime::Today());
732 SetDateAndNotify(wxDateTime(1, m_date
.GetMonth(), m_date
.GetYear()));
736 SetDateAndNotify(wxDateTime(m_date
).SetToLastMonthDay());
744 // ----------------------------------------------------------------------------
746 // ----------------------------------------------------------------------------
748 void wxCalendarCtrl::GenerateEvent(wxEventType type
, bool selChanged
)
750 // we're called for a change in some particular date field but we always
751 // also generate a generic "changed" event
752 wxCalendarEvent
event(this, type
);
753 (void)GetEventHandler()->ProcessEvent(event
);
757 wxCalendarEvent
event2(this, wxEVT_CALENDAR_SEL_CHANGED
);
759 (void)GetEventHandler()->ProcessEvent(event2
);
763 void wxCalendarEvent::Init()
765 m_wday
= wxDateTime::Inv_WeekDay
;
768 wxCalendarEvent::wxCalendarEvent(wxCalendarCtrl
*cal
, wxEventType type
)
769 : wxCommandEvent(type
, cal
->GetId())
771 m_date
= cal
->GetDate();