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"
35 #include "wx/combobox.h"
36 #include "wx/stattext.h"
37 #include "wx/textctrl.h"
40 #if wxUSE_CALENDARCTRL
42 #include "wx/spinctrl.h"
44 #include "wx/calctrl.h"
48 // ----------------------------------------------------------------------------
50 // ----------------------------------------------------------------------------
52 class wxMonthComboBox
: public wxComboBox
55 wxMonthComboBox(wxCalendarCtrl
*cal
);
57 void OnMonthChange(wxCommandEvent
& event
) { m_cal
->OnMonthChange(event
); }
60 wxCalendarCtrl
*m_cal
;
63 DECLARE_NO_COPY_CLASS(wxMonthComboBox
)
66 class wxYearSpinCtrl
: public wxSpinCtrl
69 wxYearSpinCtrl(wxCalendarCtrl
*cal
);
71 void OnYearTextChange(wxCommandEvent
& event
) { m_cal
->OnYearChange(event
); }
72 void OnYearChange(wxSpinEvent
& event
) { m_cal
->OnYearChange(event
); }
75 wxCalendarCtrl
*m_cal
;
78 DECLARE_NO_COPY_CLASS(wxYearSpinCtrl
)
81 // ----------------------------------------------------------------------------
83 // ----------------------------------------------------------------------------
85 BEGIN_EVENT_TABLE(wxCalendarCtrl
, wxControl
)
86 EVT_PAINT(wxCalendarCtrl::OnPaint
)
88 EVT_CHAR(wxCalendarCtrl::OnChar
)
90 EVT_LEFT_DOWN(wxCalendarCtrl::OnClick
)
91 EVT_LEFT_DCLICK(wxCalendarCtrl::OnDClick
)
94 BEGIN_EVENT_TABLE(wxMonthComboBox
, wxComboBox
)
95 EVT_COMBOBOX(-1, wxMonthComboBox::OnMonthChange
)
98 BEGIN_EVENT_TABLE(wxYearSpinCtrl
, wxSpinCtrl
)
99 EVT_TEXT(-1, wxYearSpinCtrl::OnYearTextChange
)
100 EVT_SPINCTRL(-1, wxYearSpinCtrl::OnYearChange
)
103 IMPLEMENT_DYNAMIC_CLASS(wxCalendarCtrl
, wxControl
)
104 IMPLEMENT_DYNAMIC_CLASS(wxCalendarEvent
, wxCommandEvent
)
106 // ----------------------------------------------------------------------------
108 // ----------------------------------------------------------------------------
110 DEFINE_EVENT_TYPE(wxEVT_CALENDAR_SEL_CHANGED
)
111 DEFINE_EVENT_TYPE(wxEVT_CALENDAR_DAY_CHANGED
)
112 DEFINE_EVENT_TYPE(wxEVT_CALENDAR_MONTH_CHANGED
)
113 DEFINE_EVENT_TYPE(wxEVT_CALENDAR_YEAR_CHANGED
)
114 DEFINE_EVENT_TYPE(wxEVT_CALENDAR_DOUBLECLICKED
)
115 DEFINE_EVENT_TYPE(wxEVT_CALENDAR_WEEKDAY_CLICKED
)
117 // ============================================================================
119 // ============================================================================
121 // ----------------------------------------------------------------------------
122 // wxMonthComboBox and wxYearSpinCtrl
123 // ----------------------------------------------------------------------------
125 wxMonthComboBox::wxMonthComboBox(wxCalendarCtrl
*cal
)
126 : wxComboBox(cal
->GetParent(), -1,
131 wxCB_READONLY
| wxCLIP_SIBLINGS
)
136 for ( m
= wxDateTime::Jan
; m
< wxDateTime::Inv_Month
; wxNextMonth(m
) )
138 Append(wxDateTime::GetMonthName(m
));
141 SetSelection(m_cal
->GetDate().GetMonth());
142 SetSize(-1, -1, -1, -1, wxSIZE_AUTO_WIDTH
|wxSIZE_AUTO_HEIGHT
);
145 wxYearSpinCtrl::wxYearSpinCtrl(wxCalendarCtrl
*cal
)
146 : wxSpinCtrl(cal
->GetParent(), -1,
147 cal
->GetDate().Format(_T("%Y")),
150 wxSP_ARROW_KEYS
| wxCLIP_SIBLINGS
,
151 -4300, 10000, cal
->GetDate().GetYear())
157 // ----------------------------------------------------------------------------
159 // ----------------------------------------------------------------------------
161 wxCalendarCtrl::wxCalendarCtrl(wxWindow
*parent
,
163 const wxDateTime
& date
,
167 const wxString
& name
)
171 (void)Create(parent
, id
, date
, pos
, size
, style
, name
);
174 void wxCalendarCtrl::Init()
179 m_userChangedYear
= FALSE
;
184 wxDateTime::WeekDay wd
;
185 for ( wd
= wxDateTime::Sun
; wd
< wxDateTime::Inv_WeekDay
; wxNextWDay(wd
) )
187 m_weekdays
[wd
] = wxDateTime::GetWeekDayName(wd
, wxDateTime::Name_Abbr
);
190 for ( size_t n
= 0; n
< WXSIZEOF(m_attrs
); n
++ )
195 m_colHighlightFg
= wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHTTEXT
);
196 m_colHighlightBg
= wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHT
);
198 m_colHolidayFg
= *wxRED
;
199 // don't set m_colHolidayBg - by default, same as our bg colour
201 m_colHeaderFg
= *wxBLUE
;
202 m_colHeaderBg
= *wxLIGHT_GREY
;
205 bool wxCalendarCtrl::Create(wxWindow
*parent
,
207 const wxDateTime
& date
,
211 const wxString
& name
)
213 if ( !wxControl::Create(parent
, id
, pos
, size
,
214 style
| wxCLIP_CHILDREN
| wxWANTS_CHARS
,
215 wxDefaultValidator
, name
) )
220 // needed to get the arrow keys normally used for the dialog navigation
221 SetWindowStyle(style
| wxWANTS_CHARS
);
223 m_date
= date
.IsValid() ? date
: wxDateTime::Today();
225 m_lowdate
= wxDefaultDateTime
;
226 m_highdate
= wxDefaultDateTime
;
228 if ( !HasFlag(wxCAL_SEQUENTIAL_MONTH_SELECTION
) )
230 m_spinYear
= new wxYearSpinCtrl(this);
231 m_staticYear
= new wxStaticText(GetParent(), -1, m_date
.Format(_T("%Y")),
232 wxDefaultPosition
, wxDefaultSize
,
235 m_comboMonth
= new wxMonthComboBox(this);
236 m_staticMonth
= new wxStaticText(GetParent(), -1, m_date
.Format(_T("%B")),
237 wxDefaultPosition
, wxDefaultSize
,
241 ShowCurrentControls();
244 if ( size
.x
== -1 || size
.y
== -1 )
246 sizeReal
= DoGetBestSize();
257 // we need to set the position as well because the main control position
258 // is not the same as the one specified in pos if we have the controls
260 SetSize(pos
.x
, pos
.y
, sizeReal
.x
, sizeReal
.y
);
262 SetBackgroundColour(*wxWHITE
);
263 SetFont(*wxSWISS_FONT
);
270 wxCalendarCtrl::~wxCalendarCtrl()
272 for ( size_t n
= 0; n
< WXSIZEOF(m_attrs
); n
++ )
278 // ----------------------------------------------------------------------------
279 // forward wxWin functions to subcontrols
280 // ----------------------------------------------------------------------------
282 bool wxCalendarCtrl::Destroy()
285 m_staticYear
->Destroy();
287 m_spinYear
->Destroy();
289 m_comboMonth
->Destroy();
291 m_staticMonth
->Destroy();
296 m_staticMonth
= NULL
;
298 return wxControl::Destroy();
301 bool wxCalendarCtrl::Show(bool show
)
303 if ( !wxControl::Show(show
) )
308 if ( !(GetWindowStyle() & wxCAL_SEQUENTIAL_MONTH_SELECTION
) )
310 if ( GetMonthControl() )
312 GetMonthControl()->Show(show
);
313 GetYearControl()->Show(show
);
320 bool wxCalendarCtrl::Enable(bool enable
)
322 if ( !wxControl::Enable(enable
) )
327 if ( !(GetWindowStyle() & wxCAL_SEQUENTIAL_MONTH_SELECTION
) )
329 GetMonthControl()->Enable(enable
);
330 GetYearControl()->Enable(enable
);
336 // ----------------------------------------------------------------------------
337 // enable/disable month/year controls
338 // ----------------------------------------------------------------------------
340 void wxCalendarCtrl::ShowCurrentControls()
342 if ( !HasFlag(wxCAL_SEQUENTIAL_MONTH_SELECTION
) )
344 if ( AllowMonthChange() )
346 m_comboMonth
->Show();
347 m_staticMonth
->Hide();
349 if ( AllowYearChange() )
352 m_staticYear
->Hide();
360 m_comboMonth
->Hide();
361 m_staticMonth
->Show();
364 // year change not allowed here
366 m_staticYear
->Show();
370 wxControl
*wxCalendarCtrl::GetMonthControl() const
372 return AllowMonthChange() ? (wxControl
*)m_comboMonth
: (wxControl
*)m_staticMonth
;
375 wxControl
*wxCalendarCtrl::GetYearControl() const
377 return AllowYearChange() ? (wxControl
*)m_spinYear
: (wxControl
*)m_staticYear
;
380 void wxCalendarCtrl::EnableYearChange(bool enable
)
382 if ( enable
!= AllowYearChange() )
384 long style
= GetWindowStyle();
386 style
&= ~wxCAL_NO_YEAR_CHANGE
;
388 style
|= wxCAL_NO_YEAR_CHANGE
;
389 SetWindowStyle(style
);
391 ShowCurrentControls();
392 if ( GetWindowStyle() & wxCAL_SEQUENTIAL_MONTH_SELECTION
)
399 void wxCalendarCtrl::EnableMonthChange(bool enable
)
401 if ( enable
!= AllowMonthChange() )
403 long style
= GetWindowStyle();
405 style
&= ~wxCAL_NO_MONTH_CHANGE
;
407 style
|= wxCAL_NO_MONTH_CHANGE
;
408 SetWindowStyle(style
);
410 ShowCurrentControls();
411 if ( GetWindowStyle() & wxCAL_SEQUENTIAL_MONTH_SELECTION
)
418 // ----------------------------------------------------------------------------
420 // ----------------------------------------------------------------------------
422 bool wxCalendarCtrl::SetDate(const wxDateTime
& date
)
426 bool sameMonth
= m_date
.GetMonth() == date
.GetMonth(),
427 sameYear
= m_date
.GetYear() == date
.GetYear();
429 if ( IsDateInRange(date
) )
431 if ( sameMonth
&& sameYear
)
433 // just change the day
438 if ( AllowMonthChange() && (AllowYearChange() || sameYear
) )
443 if ( !(GetWindowStyle() & wxCAL_SEQUENTIAL_MONTH_SELECTION
) )
445 // update the controls
446 m_comboMonth
->SetSelection(m_date
.GetMonth());
448 if ( AllowYearChange() )
450 if ( !m_userChangedYear
)
451 m_spinYear
->SetValue(m_date
.Format(_T("%Y")));
452 else // don't overwrite what the user typed in
453 m_userChangedYear
= FALSE
;
457 // as the month changed, holidays did too
460 // update the calendar
474 void wxCalendarCtrl::ChangeDay(const wxDateTime
& date
)
476 if ( m_date
!= date
)
478 // we need to refresh the row containing the old date and the one
479 // containing the new one
480 wxDateTime dateOld
= m_date
;
483 RefreshDate(dateOld
);
485 // if the date is in the same row, it was already drawn correctly
486 if ( GetWeek(m_date
) != GetWeek(dateOld
) )
493 void wxCalendarCtrl::SetDateAndNotify(const wxDateTime
& date
)
495 wxDateTime::Tm tm1
= m_date
.GetTm(),
499 if ( tm1
.year
!= tm2
.year
)
500 type
= wxEVT_CALENDAR_YEAR_CHANGED
;
501 else if ( tm1
.mon
!= tm2
.mon
)
502 type
= wxEVT_CALENDAR_MONTH_CHANGED
;
503 else if ( tm1
.mday
!= tm2
.mday
)
504 type
= wxEVT_CALENDAR_DAY_CHANGED
;
510 GenerateEvents(type
, wxEVT_CALENDAR_SEL_CHANGED
);
514 // ----------------------------------------------------------------------------
516 // ----------------------------------------------------------------------------
518 bool wxCalendarCtrl::SetLowerDateLimit(const wxDateTime
& date
/* = wxDefaultDateTime */)
522 if ( !(date
.IsValid()) || ( ( m_highdate
.IsValid() ) ? ( date
<= m_highdate
) : TRUE
) )
534 bool wxCalendarCtrl::SetUpperDateLimit(const wxDateTime
& date
/* = wxDefaultDateTime */)
538 if ( !(date
.IsValid()) || ( ( m_lowdate
.IsValid() ) ? ( date
>= m_lowdate
) : TRUE
) )
550 bool wxCalendarCtrl::SetDateRange(const wxDateTime
& lowerdate
/* = wxDefaultDateTime */, const wxDateTime
& upperdate
/* = wxDefaultDateTime */)
555 ( !( lowerdate
.IsValid() ) || ( ( upperdate
.IsValid() ) ? ( lowerdate
<= upperdate
) : TRUE
) ) &&
556 ( !( upperdate
.IsValid() ) || ( ( lowerdate
.IsValid() ) ? ( upperdate
>= lowerdate
) : TRUE
) ) )
558 m_lowdate
= lowerdate
;
559 m_highdate
= upperdate
;
569 // ----------------------------------------------------------------------------
571 // ----------------------------------------------------------------------------
573 wxDateTime
wxCalendarCtrl::GetStartDate() const
575 wxDateTime::Tm tm
= m_date
.GetTm();
577 wxDateTime date
= wxDateTime(1, tm
.mon
, tm
.year
);
580 date
.SetToPrevWeekDay(GetWindowStyle() & wxCAL_MONDAY_FIRST
581 ? wxDateTime::Mon
: wxDateTime::Sun
);
583 if ( GetWindowStyle() & wxCAL_SHOW_SURROUNDING_WEEKS
)
585 // We want to offset the calendar if we start on the first..
586 if ( date
.GetDay() == 1 )
588 date
-= wxDateSpan::Week();
595 bool wxCalendarCtrl::IsDateShown(const wxDateTime
& date
) const
597 if ( !(GetWindowStyle() & wxCAL_SHOW_SURROUNDING_WEEKS
) )
599 return date
.GetMonth() == m_date
.GetMonth();
607 bool wxCalendarCtrl::IsDateInRange(const wxDateTime
& date
) const
610 // Check if the given date is in the range specified
611 retval
= ( ( ( m_lowdate
.IsValid() ) ? ( date
>= m_lowdate
) : TRUE
)
612 && ( ( m_highdate
.IsValid() ) ? ( date
<= m_highdate
) : TRUE
) );
616 bool wxCalendarCtrl::ChangeYear(wxDateTime
* target
) const
620 if ( !(IsDateInRange(*target
)) )
622 if ( target
->GetYear() < m_date
.GetYear() )
624 if ( target
->GetYear() >= GetLowerDateLimit().GetYear() )
626 *target
= GetLowerDateLimit();
636 if ( target
->GetYear() <= GetUpperDateLimit().GetYear() )
638 *target
= GetUpperDateLimit();
655 bool wxCalendarCtrl::ChangeMonth(wxDateTime
* target
) const
659 if ( !(IsDateInRange(*target
)) )
663 if ( target
->GetMonth() < m_date
.GetMonth() )
665 *target
= GetLowerDateLimit();
669 *target
= GetUpperDateLimit();
676 size_t wxCalendarCtrl::GetWeek(const wxDateTime
& date
) const
678 size_t retval
= date
.GetWeekOfMonth(GetWindowStyle() & wxCAL_MONDAY_FIRST
679 ? wxDateTime::Monday_First
680 : wxDateTime::Sunday_First
);
682 if ( (GetWindowStyle() & wxCAL_SHOW_SURROUNDING_WEEKS
) )
684 // we need to offset an extra week if we "start" on the 1st of the month
685 wxDateTime::Tm tm
= date
.GetTm();
687 wxDateTime datetest
= wxDateTime(1, tm
.mon
, tm
.year
);
690 datetest
.SetToPrevWeekDay(GetWindowStyle() & wxCAL_MONDAY_FIRST
691 ? wxDateTime::Mon
: wxDateTime::Sun
);
693 if ( datetest
.GetDay() == 1 )
702 // ----------------------------------------------------------------------------
704 // ----------------------------------------------------------------------------
706 // this is a composite control and it must arrange its parts each time its
707 // size or position changes: the combobox and spinctrl are along the top of
708 // the available area and the calendar takes up therest of the space
710 // the static controls are supposed to be always smaller than combo/spin so we
711 // always use the latter for size calculations and position the static to take
714 // the constants used for the layout
715 #define VERT_MARGIN 5 // distance between combo and calendar
717 #define HORZ_MARGIN 5 // spin
719 #define HORZ_MARGIN 15 // spin
721 wxSize
wxCalendarCtrl::DoGetBestSize() const
723 // calc the size of the calendar
724 ((wxCalendarCtrl
*)this)->RecalcGeometry(); // const_cast
726 wxCoord width
= 7*m_widthCol
,
727 height
= 7*m_heightRow
+ m_rowOffset
+ VERT_MARGIN
;
729 if ( !HasFlag(wxCAL_SEQUENTIAL_MONTH_SELECTION
) )
731 // the combobox doesn't report its height correctly (it returns the
732 // height including the drop down list) so don't use it
733 height
+= m_spinYear
->GetBestSize().y
;
736 if ( !HasFlag(wxBORDER_NONE
) )
738 // the border would clip the last line otherwise
743 return wxSize(width
, height
);
746 void wxCalendarCtrl::DoSetSize(int x
, int y
,
747 int width
, int height
,
750 wxControl::DoSetSize(x
, y
, width
, height
, sizeFlags
);
753 void wxCalendarCtrl::DoMoveWindow(int x
, int y
, int width
, int height
)
757 if ( !HasFlag(wxCAL_SEQUENTIAL_MONTH_SELECTION
) )
759 wxSize sizeCombo
= m_comboMonth
->GetSize();
760 wxSize sizeStatic
= m_staticMonth
->GetSize();
761 wxSize sizeSpin
= m_spinYear
->GetSize();
763 int dy
= (sizeCombo
.y
- sizeStatic
.y
) / 2;
765 if (sizeCombo
.x
+ HORZ_MARGIN
- sizeSpin
.x
> width
)
767 m_comboMonth
->SetSize(x
, y
, width
- HORZ_MARGIN
- sizeSpin
.x
, sizeCombo
.y
);
771 m_comboMonth
->Move(x
, y
);
773 m_staticMonth
->Move(x
, y
+ dy
);
774 m_spinYear
->Move(x
+ width
- sizeSpin
.x
, y
);
775 m_staticYear
->Move(x
+ width
- sizeSpin
.x
, y
+ dy
);
777 yDiff
= wxMax(sizeSpin
.y
, sizeCombo
.y
) + VERT_MARGIN
;
779 else // no controls on the top
784 wxControl::DoMoveWindow(x
, y
+ yDiff
, width
, height
- yDiff
);
787 void wxCalendarCtrl::DoGetPosition(int *x
, int *y
) const
789 wxControl::DoGetPosition(x
, y
);
791 if ( !(GetWindowStyle() & wxCAL_SEQUENTIAL_MONTH_SELECTION
) )
793 // our real top corner is not in this position
796 *y
-= GetMonthControl()->GetSize().y
+ VERT_MARGIN
;
801 void wxCalendarCtrl::DoGetSize(int *width
, int *height
) const
803 wxControl::DoGetSize(width
, height
);
805 if ( !(GetWindowStyle() & wxCAL_SEQUENTIAL_MONTH_SELECTION
) )
807 // our real height is bigger
808 if ( height
&& GetMonthControl())
810 *height
+= GetMonthControl()->GetSize().y
+ VERT_MARGIN
;
815 void wxCalendarCtrl::RecalcGeometry()
817 if ( m_widthCol
!= 0 )
824 // determine the column width (we assume that the weekday names are always
825 // wider (in any language) than the numbers)
827 wxDateTime::WeekDay wd
;
828 for ( wd
= wxDateTime::Sun
; wd
< wxDateTime::Inv_WeekDay
; wxNextWDay(wd
) )
831 dc
.GetTextExtent(m_weekdays
[wd
], &width
, &m_heightRow
);
832 if ( width
> m_widthCol
)
838 // leave some margins
842 m_rowOffset
= (GetWindowStyle() & wxCAL_SEQUENTIAL_MONTH_SELECTION
) ? m_heightRow
: 0; // conditional in relation to style
845 // ----------------------------------------------------------------------------
847 // ----------------------------------------------------------------------------
849 void wxCalendarCtrl::OnPaint(wxPaintEvent
& WXUNUSED(event
))
858 wxLogDebug("--- starting to paint, selection: %s, week %u\n",
859 m_date
.Format("%a %d-%m-%Y %H:%M:%S").c_str(),
865 if ( HasFlag(wxCAL_SEQUENTIAL_MONTH_SELECTION
) )
867 // draw the sequential month-selector
869 dc
.SetBackgroundMode(wxTRANSPARENT
);
870 dc
.SetTextForeground(*wxBLACK
);
871 dc
.SetBrush(wxBrush(m_colHeaderBg
, wxSOLID
));
872 dc
.SetPen(wxPen(m_colHeaderBg
, 1, wxSOLID
));
873 dc
.DrawRectangle(0, y
, 7*m_widthCol
, m_heightRow
);
875 // Get extent of month-name + year
876 wxCoord monthw
, monthh
;
877 wxString headertext
= m_date
.Format(wxT("%B %Y"));
878 dc
.GetTextExtent(headertext
, &monthw
, &monthh
);
880 // draw month-name centered above weekdays
881 wxCoord monthx
= ((m_widthCol
* 7) - monthw
) / 2;
882 wxCoord monthy
= ((m_heightRow
- monthh
) / 2) + y
;
883 dc
.DrawText(headertext
, monthx
, monthy
);
885 // calculate the "month-arrows"
886 wxPoint leftarrow
[3];
887 wxPoint rightarrow
[3];
889 int arrowheight
= monthh
/ 2;
891 leftarrow
[0] = wxPoint(0, arrowheight
/ 2);
892 leftarrow
[1] = wxPoint(arrowheight
/ 2, 0);
893 leftarrow
[2] = wxPoint(arrowheight
/ 2, arrowheight
- 1);
895 rightarrow
[0] = wxPoint(0, 0);
896 rightarrow
[1] = wxPoint(arrowheight
/ 2, arrowheight
/ 2);
897 rightarrow
[2] = wxPoint(0, arrowheight
- 1);
899 // draw the "month-arrows"
901 wxCoord arrowy
= (m_heightRow
- arrowheight
) / 2;
902 wxCoord larrowx
= (m_widthCol
- (arrowheight
/ 2)) / 2;
903 wxCoord rarrowx
= ((m_widthCol
- (arrowheight
/ 2)) / 2) + m_widthCol
*6;
904 m_leftArrowRect
= wxRect(0, 0, 0, 0);
905 m_rightArrowRect
= wxRect(0, 0, 0, 0);
907 if ( AllowMonthChange() )
909 wxDateTime ldpm
= wxDateTime(1,m_date
.GetMonth(), m_date
.GetYear()) - wxDateSpan::Day(); // last day prev month
910 // Check if range permits change
911 if ( IsDateInRange(ldpm
) && ( ( ldpm
.GetYear() == m_date
.GetYear() ) ? TRUE
: AllowYearChange() ) )
913 m_leftArrowRect
= wxRect(larrowx
- 3, arrowy
- 3, (arrowheight
/ 2) + 8, (arrowheight
+ 6));
914 dc
.SetBrush(wxBrush(*wxBLACK
, wxSOLID
));
915 dc
.SetPen(wxPen(*wxBLACK
, 1, wxSOLID
));
916 dc
.DrawPolygon(3, leftarrow
, larrowx
, arrowy
, wxWINDING_RULE
);
917 dc
.SetBrush(*wxTRANSPARENT_BRUSH
);
918 dc
.DrawRectangle(m_leftArrowRect
);
920 wxDateTime fdnm
= wxDateTime(1,m_date
.GetMonth(), m_date
.GetYear()) + wxDateSpan::Month(); // first day next month
921 if ( IsDateInRange(fdnm
) && ( ( fdnm
.GetYear() == m_date
.GetYear() ) ? TRUE
: AllowYearChange() ) )
923 m_rightArrowRect
= wxRect(rarrowx
- 4, arrowy
- 3, (arrowheight
/ 2) + 8, (arrowheight
+ 6));
924 dc
.SetBrush(wxBrush(*wxBLACK
, wxSOLID
));
925 dc
.SetPen(wxPen(*wxBLACK
, 1, wxSOLID
));
926 dc
.DrawPolygon(3, rightarrow
, rarrowx
, arrowy
, wxWINDING_RULE
);
927 dc
.SetBrush(*wxTRANSPARENT_BRUSH
);
928 dc
.DrawRectangle(m_rightArrowRect
);
935 // first draw the week days
936 if ( IsExposed(0, y
, 7*m_widthCol
, m_heightRow
) )
939 wxLogDebug("painting the header");
942 dc
.SetBackgroundMode(wxTRANSPARENT
);
943 dc
.SetTextForeground(m_colHeaderFg
);
944 dc
.SetBrush(wxBrush(m_colHeaderBg
, wxSOLID
));
945 dc
.SetPen(wxPen(m_colHeaderBg
, 1, wxSOLID
));
946 dc
.DrawRectangle(0, y
, GetClientSize().x
, m_heightRow
);
948 bool startOnMonday
= (GetWindowStyle() & wxCAL_MONDAY_FIRST
) != 0;
949 for ( size_t wd
= 0; wd
< 7; wd
++ )
953 n
= wd
== 6 ? 0 : wd
+ 1;
957 dc
.GetTextExtent(m_weekdays
[n
], &dayw
, &dayh
);
958 dc
.DrawText(m_weekdays
[n
], (wd
*m_widthCol
) + ((m_widthCol
- dayw
) / 2), y
); // center the day-name
962 // then the calendar itself
963 dc
.SetTextForeground(*wxBLACK
);
964 //dc.SetFont(*wxNORMAL_FONT);
967 wxDateTime date
= GetStartDate();
970 wxLogDebug("starting calendar from %s\n",
971 date
.Format("%a %d-%m-%Y %H:%M:%S").c_str());
974 dc
.SetBackgroundMode(wxSOLID
);
975 for ( size_t nWeek
= 1; nWeek
<= 6; nWeek
++, y
+= m_heightRow
)
977 // if the update region doesn't intersect this row, don't paint it
978 if ( !IsExposed(0, y
, 7*m_widthCol
, m_heightRow
- 1) )
980 date
+= wxDateSpan::Week();
986 wxLogDebug("painting week %d at y = %d\n", nWeek
, y
);
989 for ( size_t wd
= 0; wd
< 7; wd
++ )
991 if ( IsDateShown(date
) )
993 // don't use wxDate::Format() which prepends 0s
994 unsigned int day
= date
.GetDay();
995 wxString dayStr
= wxString::Format(_T("%u"), day
);
997 dc
.GetTextExtent(dayStr
, &width
, (wxCoord
*)NULL
);
999 bool changedColours
= FALSE
,
1000 changedFont
= FALSE
;
1003 wxCalendarDateAttr
*attr
= NULL
;
1005 if ( date
.GetMonth() != m_date
.GetMonth() || !IsDateInRange(date
) )
1007 // surrounding week or out-of-range
1009 dc
.SetTextForeground(*wxLIGHT_GREY
);
1010 changedColours
= TRUE
;
1014 isSel
= date
.IsSameDate(m_date
);
1015 attr
= m_attrs
[day
- 1];
1019 dc
.SetTextForeground(m_colHighlightFg
);
1020 dc
.SetTextBackground(m_colHighlightBg
);
1022 changedColours
= TRUE
;
1026 wxColour colFg
, colBg
;
1028 if ( attr
->IsHoliday() )
1030 colFg
= m_colHolidayFg
;
1031 colBg
= m_colHolidayBg
;
1035 colFg
= attr
->GetTextColour();
1036 colBg
= attr
->GetBackgroundColour();
1041 dc
.SetTextForeground(colFg
);
1042 changedColours
= TRUE
;
1047 dc
.SetTextBackground(colBg
);
1048 changedColours
= TRUE
;
1051 if ( attr
->HasFont() )
1053 dc
.SetFont(attr
->GetFont());
1059 wxCoord x
= wd
*m_widthCol
+ (m_widthCol
- width
) / 2;
1060 dc
.DrawText(dayStr
, x
, y
+ 1);
1062 if ( !isSel
&& attr
&& attr
->HasBorder() )
1065 if ( attr
->HasBorderColour() )
1067 colBorder
= attr
->GetBorderColour();
1071 colBorder
= m_foregroundColour
;
1074 wxPen
pen(colBorder
, 1, wxSOLID
);
1076 dc
.SetBrush(*wxTRANSPARENT_BRUSH
);
1078 switch ( attr
->GetBorder() )
1080 case wxCAL_BORDER_SQUARE
:
1081 dc
.DrawRectangle(x
- 2, y
,
1082 width
+ 4, m_heightRow
);
1085 case wxCAL_BORDER_ROUND
:
1086 dc
.DrawEllipse(x
- 2, y
,
1087 width
+ 4, m_heightRow
);
1091 wxFAIL_MSG(_T("unknown border type"));
1095 if ( changedColours
)
1097 dc
.SetTextForeground(m_foregroundColour
);
1098 dc
.SetTextBackground(m_backgroundColour
);
1106 //else: just don't draw it
1108 date
+= wxDateSpan::Day();
1112 // Greying out out-of-range background
1113 bool showSurrounding
= (GetWindowStyle() & wxCAL_SHOW_SURROUNDING_WEEKS
) != 0;
1115 date
= ( showSurrounding
) ? GetStartDate() : wxDateTime(1, m_date
.GetMonth(), m_date
.GetYear());
1116 if ( !IsDateInRange(date
) )
1118 wxDateTime firstOOR
= GetLowerDateLimit() - wxDateSpan::Day(); // first out-of-range
1120 wxBrush oorbrush
= *wxLIGHT_GREY_BRUSH
;
1121 oorbrush
.SetStyle(wxFDIAGONAL_HATCH
);
1123 HighlightRange(&dc
, date
, firstOOR
, wxTRANSPARENT_PEN
, &oorbrush
);
1126 date
= ( showSurrounding
) ? GetStartDate() + wxDateSpan::Weeks(6) - wxDateSpan::Day() : wxDateTime().SetToLastMonthDay(m_date
.GetMonth(), m_date
.GetYear());
1127 if ( !IsDateInRange(date
) )
1129 wxDateTime firstOOR
= GetUpperDateLimit() + wxDateSpan::Day(); // first out-of-range
1131 wxBrush oorbrush
= *wxLIGHT_GREY_BRUSH
;
1132 oorbrush
.SetStyle(wxFDIAGONAL_HATCH
);
1134 HighlightRange(&dc
, firstOOR
, date
, wxTRANSPARENT_PEN
, &oorbrush
);
1138 wxLogDebug("+++ finished painting");
1142 void wxCalendarCtrl::RefreshDate(const wxDateTime
& date
)
1148 // always refresh the whole row at once because our OnPaint() will draw
1149 // the whole row anyhow - and this allows the small optimisation in
1150 // OnClick() below to work
1153 rect
.y
= (m_heightRow
* GetWeek(date
)) + m_rowOffset
;
1155 rect
.width
= 7*m_widthCol
;
1156 rect
.height
= m_heightRow
;
1159 // VZ: for some reason, the selected date seems to occupy more space under
1160 // MSW - this is probably some bug in the font size calculations, but I
1161 // don't know where exactly. This fix is ugly and leads to more
1162 // refreshes than really needed, but without it the selected days
1163 // leaves even more ugly underscores on screen.
1168 wxLogDebug("*** refreshing week %d at (%d, %d)-(%d, %d)\n",
1171 rect
.x
+ rect
.width
, rect
.y
+ rect
.height
);
1174 Refresh(TRUE
, &rect
);
1177 void wxCalendarCtrl::HighlightRange(wxPaintDC
* pDC
, const wxDateTime
& fromdate
, const wxDateTime
& todate
, wxPen
* pPen
, wxBrush
* pBrush
)
1179 // Highlights the given range using pen and brush
1180 // Does nothing if todate < fromdate
1184 wxLogDebug("+++ HighlightRange: (%s) - (%s) +++", fromdate
.Format("%d %m %Y"), todate
.Format("%d %m %Y"));
1187 if ( todate
>= fromdate
)
1194 // implicit: both dates must be currently shown - checked by GetDateCoord
1195 if ( GetDateCoord(fromdate
, &fd
, &fw
) && GetDateCoord(todate
, &td
, &tw
) )
1198 wxLogDebug("Highlight range: (%i, %i) - (%i, %i)", fd
, fw
, td
, tw
);
1200 if ( ( (tw
- fw
) == 1 ) && ( td
< fd
) )
1202 // special case: interval 7 days or less not in same week
1203 // split in two seperate intervals
1204 wxDateTime tfd
= fromdate
+ wxDateSpan::Days(7-fd
);
1205 wxDateTime ftd
= tfd
+ wxDateSpan::Day();
1207 wxLogDebug("Highlight: Seperate segments");
1210 HighlightRange(pDC
, fromdate
, tfd
, pPen
, pBrush
);
1211 HighlightRange(pDC
, ftd
, todate
, pPen
, pBrush
);
1216 wxPoint corners
[8]; // potentially 8 corners in polygon
1220 // simple case: same week
1222 corners
[0] = wxPoint((fd
- 1) * m_widthCol
, (fw
* m_heightRow
) + m_rowOffset
);
1223 corners
[1] = wxPoint((fd
- 1) * m_widthCol
, ((fw
+ 1 ) * m_heightRow
) + m_rowOffset
);
1224 corners
[2] = wxPoint(td
* m_widthCol
, ((tw
+ 1) * m_heightRow
) + m_rowOffset
);
1225 corners
[3] = wxPoint(td
* m_widthCol
, (tw
* m_heightRow
) + m_rowOffset
);
1230 // "complex" polygon
1231 corners
[cidx
] = wxPoint((fd
- 1) * m_widthCol
, (fw
* m_heightRow
) + m_rowOffset
); cidx
++;
1235 corners
[cidx
] = wxPoint((fd
- 1) * m_widthCol
, ((fw
+ 1) * m_heightRow
) + m_rowOffset
); cidx
++;
1236 corners
[cidx
] = wxPoint(0, ((fw
+ 1) * m_heightRow
) + m_rowOffset
); cidx
++;
1239 corners
[cidx
] = wxPoint(0, ((tw
+ 1) * m_heightRow
) + m_rowOffset
); cidx
++;
1240 corners
[cidx
] = wxPoint(td
* m_widthCol
, ((tw
+ 1) * m_heightRow
) + m_rowOffset
); cidx
++;
1244 corners
[cidx
] = wxPoint(td
* m_widthCol
, (tw
* m_heightRow
) + m_rowOffset
); cidx
++;
1245 corners
[cidx
] = wxPoint(7 * m_widthCol
, (tw
* m_heightRow
) + m_rowOffset
); cidx
++;
1248 corners
[cidx
] = wxPoint(7 * m_widthCol
, (fw
* m_heightRow
) + m_rowOffset
); cidx
++;
1254 pDC
->SetBrush(*pBrush
);
1256 pDC
->DrawPolygon(numpoints
, corners
);
1262 wxLogDebug("--- HighlightRange ---");
1266 bool wxCalendarCtrl::GetDateCoord(const wxDateTime
& date
, int *day
, int *week
) const
1271 wxLogDebug("+++ GetDateCoord: (%s) +++", date
.Format("%d %m %Y"));
1274 if ( IsDateShown(date
) )
1276 bool startOnMonday
= ( GetWindowStyle() & wxCAL_MONDAY_FIRST
) != 0;
1279 *day
= date
.GetWeekDay();
1281 if ( *day
== 0 ) // sunday
1283 *day
= ( startOnMonday
) ? 7 : 1;
1287 day
+= ( startOnMonday
) ? 0 : 1;
1290 int targetmonth
= date
.GetMonth() + (12 * date
.GetYear());
1291 int thismonth
= m_date
.GetMonth() + (12 * m_date
.GetYear());
1294 if ( targetmonth
== thismonth
)
1296 *week
= GetWeek(date
);
1300 if ( targetmonth
< thismonth
)
1302 *week
= 1; // trivial
1304 else // targetmonth > thismonth
1310 // get the datecoord of the last day in the month currently shown
1312 wxLogDebug(" +++ LDOM +++");
1314 GetDateCoord(ldcm
.SetToLastMonthDay(m_date
.GetMonth(), m_date
.GetYear()), &lastday
, &lastweek
);
1316 wxLogDebug(" --- LDOM ---");
1319 wxTimeSpan span
= date
- ldcm
;
1321 int daysfromlast
= span
.GetDays();
1323 wxLogDebug("daysfromlast: %i", daysfromlast
);
1325 if ( daysfromlast
+ lastday
> 7 ) // past week boundary
1327 int wholeweeks
= (daysfromlast
/ 7);
1328 *week
= wholeweeks
+ lastweek
;
1329 if ( (daysfromlast
- (7 * wholeweeks
) + lastday
) > 7 )
1349 wxLogDebug("--- GetDateCoord: (%s) = (%i, %i) ---", date
.Format("%d %m %Y"), *day
, *week
);
1355 // ----------------------------------------------------------------------------
1357 // ----------------------------------------------------------------------------
1359 void wxCalendarCtrl::OnDClick(wxMouseEvent
& event
)
1361 if ( HitTest(event
.GetPosition()) != wxCAL_HITTEST_DAY
)
1367 GenerateEvent(wxEVT_CALENDAR_DOUBLECLICKED
);
1371 void wxCalendarCtrl::OnClick(wxMouseEvent
& event
)
1374 wxDateTime::WeekDay wday
;
1375 switch ( HitTest(event
.GetPosition(), &date
, &wday
) )
1377 case wxCAL_HITTEST_DAY
:
1378 if ( IsDateInRange(date
) )
1382 GenerateEvents(wxEVT_CALENDAR_DAY_CHANGED
,
1383 wxEVT_CALENDAR_SEL_CHANGED
);
1387 case wxCAL_HITTEST_HEADER
:
1389 wxCalendarEvent
event(this, wxEVT_CALENDAR_WEEKDAY_CLICKED
);
1390 event
.m_wday
= wday
;
1391 (void)GetEventHandler()->ProcessEvent(event
);
1395 case wxCAL_HITTEST_DECMONTH
:
1396 case wxCAL_HITTEST_INCMONTH
:
1397 case wxCAL_HITTEST_SURROUNDING_WEEK
:
1398 SetDateAndNotify(date
); // we probably only want to refresh the control. No notification.. (maybe as an option?)
1402 wxFAIL_MSG(_T("unknown hittest code"));
1405 case wxCAL_HITTEST_NOWHERE
:
1411 wxCalendarHitTestResult
wxCalendarCtrl::HitTest(const wxPoint
& pos
,
1413 wxDateTime::WeekDay
*wd
)
1419 ///////////////////////////////////////////////////////////////////////////////////////////////////////
1420 if ( (GetWindowStyle() & wxCAL_SEQUENTIAL_MONTH_SELECTION
) )
1424 // we need to find out if the hit is on left arrow, on month or on right arrow
1426 if ( wxRegion(m_leftArrowRect
).Contains(pos
) == wxInRegion
)
1430 if ( IsDateInRange(m_date
- wxDateSpan::Month()) )
1432 *date
= m_date
- wxDateSpan::Month();
1436 *date
= GetLowerDateLimit();
1440 return wxCAL_HITTEST_DECMONTH
;
1443 if ( wxRegion(m_rightArrowRect
).Contains(pos
) == wxInRegion
)
1447 if ( IsDateInRange(m_date
+ wxDateSpan::Month()) )
1449 *date
= m_date
+ wxDateSpan::Month();
1453 *date
= GetUpperDateLimit();
1457 return wxCAL_HITTEST_INCMONTH
;
1462 ///////////////////////////////////////////////////////////////////////////////////////////////////////
1464 int wday
= pos
.x
/ m_widthCol
;
1465 // if ( y < m_heightRow )
1466 if ( y
< (m_heightRow
+ m_rowOffset
) )
1468 if ( y
> m_rowOffset
)
1472 if ( GetWindowStyle() & wxCAL_MONDAY_FIRST
)
1474 wday
= wday
== 6 ? 0 : wday
+ 1;
1477 *wd
= (wxDateTime::WeekDay
)wday
;
1480 return wxCAL_HITTEST_HEADER
;
1484 return wxCAL_HITTEST_NOWHERE
;
1488 // int week = (y - m_heightRow) / m_heightRow;
1489 int week
= (y
- (m_heightRow
+ m_rowOffset
)) / m_heightRow
;
1490 if ( week
>= 6 || wday
>= 7 )
1492 return wxCAL_HITTEST_NOWHERE
;
1495 wxDateTime dt
= GetStartDate() + wxDateSpan::Days(7*week
+ wday
);
1497 if ( IsDateShown(dt
) )
1502 if ( dt
.GetMonth() == m_date
.GetMonth() )
1505 return wxCAL_HITTEST_DAY
;
1509 return wxCAL_HITTEST_SURROUNDING_WEEK
;
1514 return wxCAL_HITTEST_NOWHERE
;
1518 // ----------------------------------------------------------------------------
1519 // subcontrols events handling
1520 // ----------------------------------------------------------------------------
1522 void wxCalendarCtrl::OnMonthChange(wxCommandEvent
& event
)
1524 wxDateTime::Tm tm
= m_date
.GetTm();
1526 wxDateTime::Month mon
= (wxDateTime::Month
)event
.GetInt();
1527 if ( tm
.mday
> wxDateTime::GetNumberOfDays(mon
, tm
.year
) )
1529 tm
.mday
= wxDateTime::GetNumberOfDays(mon
, tm
.year
);
1532 wxDateTime target
= wxDateTime(tm
.mday
, mon
, tm
.year
);
1534 ChangeMonth(&target
);
1535 SetDateAndNotify(target
);
1538 void wxCalendarCtrl::OnYearChange(wxCommandEvent
& event
)
1540 int year
= (int)event
.GetInt();
1541 if ( year
== INT_MIN
)
1543 // invalid year in the spin control, ignore it
1547 // set the flag for SetDate(): otherwise it would overwrite the year
1548 // typed in by the user
1549 m_userChangedYear
= TRUE
;
1551 wxDateTime::Tm tm
= m_date
.GetTm();
1553 if ( tm
.mday
> wxDateTime::GetNumberOfDays(tm
.mon
, year
) )
1555 tm
.mday
= wxDateTime::GetNumberOfDays(tm
.mon
, year
);
1558 wxDateTime target
= wxDateTime(tm
.mday
, tm
.mon
, year
);
1560 if ( ChangeYear(&target
) )
1562 SetDateAndNotify(target
);
1566 // In this case we don't want to change the date. That would put us
1567 // inside the same year but a strange number of months forward/back..
1568 m_spinYear
->SetValue(target
.GetYear());
1572 // ----------------------------------------------------------------------------
1573 // keyboard interface
1574 // ----------------------------------------------------------------------------
1576 void wxCalendarCtrl::OnChar(wxKeyEvent
& event
)
1579 switch ( event
.GetKeyCode() )
1583 target
= m_date
+ wxDateSpan::Year();
1584 if ( ChangeYear(&target
) )
1586 SetDateAndNotify(target
);
1592 target
= m_date
- wxDateSpan::Year();
1593 if ( ChangeYear(&target
) )
1595 SetDateAndNotify(target
);
1600 target
= m_date
- wxDateSpan::Month();
1601 ChangeMonth(&target
);
1602 SetDateAndNotify(target
); // always
1606 target
= m_date
+ wxDateSpan::Month();
1607 ChangeMonth(&target
);
1608 SetDateAndNotify(target
); // always
1612 if ( event
.ControlDown() )
1614 target
= wxDateTime(m_date
).SetToNextWeekDay(
1615 GetWindowStyle() & wxCAL_MONDAY_FIRST
1616 ? wxDateTime::Sun
: wxDateTime::Sat
);
1617 if ( !IsDateInRange(target
) )
1619 target
= GetUpperDateLimit();
1621 SetDateAndNotify(target
);
1624 SetDateAndNotify(m_date
+ wxDateSpan::Day());
1628 if ( event
.ControlDown() )
1630 target
= wxDateTime(m_date
).SetToPrevWeekDay(
1631 GetWindowStyle() & wxCAL_MONDAY_FIRST
1632 ? wxDateTime::Mon
: wxDateTime::Sun
);
1633 if ( !IsDateInRange(target
) )
1635 target
= GetLowerDateLimit();
1637 SetDateAndNotify(target
);
1640 SetDateAndNotify(m_date
- wxDateSpan::Day());
1644 SetDateAndNotify(m_date
- wxDateSpan::Week());
1648 SetDateAndNotify(m_date
+ wxDateSpan::Week());
1652 if ( event
.ControlDown() )
1653 SetDateAndNotify(wxDateTime::Today());
1655 SetDateAndNotify(wxDateTime(1, m_date
.GetMonth(), m_date
.GetYear()));
1659 SetDateAndNotify(wxDateTime(m_date
).SetToLastMonthDay());
1663 GenerateEvent(wxEVT_CALENDAR_DOUBLECLICKED
);
1671 // ----------------------------------------------------------------------------
1672 // holidays handling
1673 // ----------------------------------------------------------------------------
1675 void wxCalendarCtrl::EnableHolidayDisplay(bool display
)
1677 long style
= GetWindowStyle();
1679 style
|= wxCAL_SHOW_HOLIDAYS
;
1681 style
&= ~wxCAL_SHOW_HOLIDAYS
;
1683 SetWindowStyle(style
);
1688 ResetHolidayAttrs();
1693 void wxCalendarCtrl::SetHolidayAttrs()
1695 if ( GetWindowStyle() & wxCAL_SHOW_HOLIDAYS
)
1697 ResetHolidayAttrs();
1699 wxDateTime::Tm tm
= m_date
.GetTm();
1700 wxDateTime
dtStart(1, tm
.mon
, tm
.year
),
1701 dtEnd
= dtStart
.GetLastMonthDay();
1703 wxDateTimeArray hol
;
1704 wxDateTimeHolidayAuthority::GetHolidaysInRange(dtStart
, dtEnd
, hol
);
1706 size_t count
= hol
.GetCount();
1707 for ( size_t n
= 0; n
< count
; n
++ )
1709 SetHoliday(hol
[n
].GetDay());
1714 void wxCalendarCtrl::SetHoliday(size_t day
)
1716 wxCHECK_RET( day
> 0 && day
< 32, _T("invalid day in SetHoliday") );
1718 wxCalendarDateAttr
*attr
= GetAttr(day
);
1721 attr
= new wxCalendarDateAttr
;
1724 attr
->SetHoliday(TRUE
);
1726 // can't use SetAttr() because it would delete this pointer
1727 m_attrs
[day
- 1] = attr
;
1730 void wxCalendarCtrl::ResetHolidayAttrs()
1732 for ( size_t day
= 0; day
< 31; day
++ )
1736 m_attrs
[day
]->SetHoliday(FALSE
);
1741 // ----------------------------------------------------------------------------
1743 // ----------------------------------------------------------------------------
1745 void wxCalendarEvent::Init()
1747 m_wday
= wxDateTime::Inv_WeekDay
;
1750 wxCalendarEvent::wxCalendarEvent(wxCalendarCtrl
*cal
, wxEventType type
)
1751 : wxCommandEvent(type
, cal
->GetId())
1753 m_date
= cal
->GetDate();
1754 SetEventObject(cal
);
1757 #endif // wxUSE_CALENDARCTRL