]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/generic/calctrl.cpp
moved Arrow_ constants to the base class
[wxWidgets.git] / src / generic / calctrl.cpp
... / ...
CommitLineData
1///////////////////////////////////////////////////////////////////////////////
2// Name: src/generic/calctrl.cpp
3// Purpose: implementation fo the generic wxCalendarCtrl
4// Author: Vadim Zeitlin
5// Modified by:
6// Created: 29.12.99
7// RCS-ID: $Id$
8// Copyright: (c) 1999 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9// Licence: wxWindows licence
10///////////////////////////////////////////////////////////////////////////////
11
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
20// For compilers that support precompilation, includes "wx.h".
21#include "wx/wxprec.h"
22
23#ifdef __BORLANDC__
24 #pragma hdrstop
25#endif
26
27#ifndef WX_PRECOMP
28 #include "wx/dcclient.h"
29 #include "wx/settings.h"
30 #include "wx/brush.h"
31 #include "wx/combobox.h"
32 #include "wx/listbox.h"
33 #include "wx/stattext.h"
34 #include "wx/textctrl.h"
35#endif //WX_PRECOMP
36
37#if wxUSE_CALENDARCTRL
38
39#include "wx/spinctrl.h"
40
41// if wxDatePickerCtrl code doesn't define the date event, do it here as we
42// need it as well
43#if !wxUSE_DATEPICKCTRL
44 #define _WX_DEFINE_DATE_EVENTS_
45#endif
46
47#include "wx/calctrl.h"
48
49#define DEBUG_PAINT 0
50
51// ----------------------------------------------------------------------------
52// wxWin macros
53// ----------------------------------------------------------------------------
54
55BEGIN_EVENT_TABLE(wxCalendarCtrl, wxControl)
56 EVT_PAINT(wxCalendarCtrl::OnPaint)
57
58 EVT_CHAR(wxCalendarCtrl::OnChar)
59
60 EVT_LEFT_DOWN(wxCalendarCtrl::OnClick)
61 EVT_LEFT_DCLICK(wxCalendarCtrl::OnDClick)
62
63 EVT_SYS_COLOUR_CHANGED(wxCalendarCtrl::OnSysColourChanged)
64END_EVENT_TABLE()
65
66#if wxUSE_EXTENDED_RTTI
67WX_DEFINE_FLAGS( wxCalendarCtrlStyle )
68
69wxBEGIN_FLAGS( wxCalendarCtrlStyle )
70 // new style border flags, we put them first to
71 // use them for streaming out
72 wxFLAGS_MEMBER(wxBORDER_SIMPLE)
73 wxFLAGS_MEMBER(wxBORDER_SUNKEN)
74 wxFLAGS_MEMBER(wxBORDER_DOUBLE)
75 wxFLAGS_MEMBER(wxBORDER_RAISED)
76 wxFLAGS_MEMBER(wxBORDER_STATIC)
77 wxFLAGS_MEMBER(wxBORDER_NONE)
78
79 // old style border flags
80 wxFLAGS_MEMBER(wxSIMPLE_BORDER)
81 wxFLAGS_MEMBER(wxSUNKEN_BORDER)
82 wxFLAGS_MEMBER(wxDOUBLE_BORDER)
83 wxFLAGS_MEMBER(wxRAISED_BORDER)
84 wxFLAGS_MEMBER(wxSTATIC_BORDER)
85 wxFLAGS_MEMBER(wxBORDER)
86
87 // standard window styles
88 wxFLAGS_MEMBER(wxTAB_TRAVERSAL)
89 wxFLAGS_MEMBER(wxCLIP_CHILDREN)
90 wxFLAGS_MEMBER(wxTRANSPARENT_WINDOW)
91 wxFLAGS_MEMBER(wxWANTS_CHARS)
92 wxFLAGS_MEMBER(wxFULL_REPAINT_ON_RESIZE)
93 wxFLAGS_MEMBER(wxALWAYS_SHOW_SB )
94 wxFLAGS_MEMBER(wxVSCROLL)
95 wxFLAGS_MEMBER(wxHSCROLL)
96
97 wxFLAGS_MEMBER(wxCAL_SUNDAY_FIRST)
98 wxFLAGS_MEMBER(wxCAL_MONDAY_FIRST)
99 wxFLAGS_MEMBER(wxCAL_SHOW_HOLIDAYS)
100 wxFLAGS_MEMBER(wxCAL_NO_YEAR_CHANGE)
101 wxFLAGS_MEMBER(wxCAL_NO_MONTH_CHANGE)
102 wxFLAGS_MEMBER(wxCAL_SEQUENTIAL_MONTH_SELECTION)
103 wxFLAGS_MEMBER(wxCAL_SHOW_SURROUNDING_WEEKS)
104
105wxEND_FLAGS( wxCalendarCtrlStyle )
106
107IMPLEMENT_DYNAMIC_CLASS_XTI(wxCalendarCtrl, wxControl,"wx/calctrl.h")
108
109wxBEGIN_PROPERTIES_TABLE(wxCalendarCtrl)
110 wxEVENT_RANGE_PROPERTY( Updated , wxEVT_CALENDAR_SEL_CHANGED , wxEVT_CALENDAR_WEEKDAY_CLICKED , wxCalendarEvent )
111 wxHIDE_PROPERTY( Children )
112 wxPROPERTY( Date,wxDateTime, SetDate , GetDate, , 0 /*flags*/ , wxT("Helpstring") , wxT("group"))
113 wxPROPERTY_FLAGS( WindowStyle , wxCalendarCtrlStyle , long , SetWindowStyleFlag , GetWindowStyleFlag , , 0 /*flags*/ , wxT("Helpstring") , wxT("group")) // style
114wxEND_PROPERTIES_TABLE()
115
116wxBEGIN_HANDLERS_TABLE(wxCalendarCtrl)
117wxEND_HANDLERS_TABLE()
118
119wxCONSTRUCTOR_6( wxCalendarCtrl , wxWindow* , Parent , wxWindowID , Id , wxDateTime , Date , wxPoint , Position , wxSize , Size , long , WindowStyle )
120#else
121IMPLEMENT_DYNAMIC_CLASS(wxCalendarCtrl, wxControl)
122#endif
123IMPLEMENT_DYNAMIC_CLASS(wxCalendarEvent, wxDateEvent)
124
125// ----------------------------------------------------------------------------
126// events
127// ----------------------------------------------------------------------------
128
129DEFINE_EVENT_TYPE(wxEVT_CALENDAR_SEL_CHANGED)
130DEFINE_EVENT_TYPE(wxEVT_CALENDAR_DAY_CHANGED)
131DEFINE_EVENT_TYPE(wxEVT_CALENDAR_MONTH_CHANGED)
132DEFINE_EVENT_TYPE(wxEVT_CALENDAR_YEAR_CHANGED)
133DEFINE_EVENT_TYPE(wxEVT_CALENDAR_DOUBLECLICKED)
134DEFINE_EVENT_TYPE(wxEVT_CALENDAR_WEEKDAY_CLICKED)
135
136// ============================================================================
137// implementation
138// ============================================================================
139
140// ----------------------------------------------------------------------------
141// wxCalendarCtrl
142// ----------------------------------------------------------------------------
143
144wxCalendarCtrl::wxCalendarCtrl(wxWindow *parent,
145 wxWindowID id,
146 const wxDateTime& date,
147 const wxPoint& pos,
148 const wxSize& size,
149 long style,
150 const wxString& name)
151{
152 Init();
153
154 (void)Create(parent, id, date, pos, size, style, name);
155}
156
157void wxCalendarCtrl::Init()
158{
159 m_comboMonth = NULL;
160 m_spinYear = NULL;
161 m_staticYear = NULL;
162 m_staticMonth = NULL;
163
164 m_userChangedYear = false;
165
166 m_widthCol =
167 m_heightRow = 0;
168
169 wxDateTime::WeekDay wd;
170 for ( wd = wxDateTime::Sun; wd < wxDateTime::Inv_WeekDay; wxNextWDay(wd) )
171 {
172 m_weekdays[wd] = wxDateTime::GetWeekDayName(wd, wxDateTime::Name_Abbr);
173 }
174
175 for ( size_t n = 0; n < WXSIZEOF(m_attrs); n++ )
176 {
177 m_attrs[n] = NULL;
178 }
179
180 InitColours();
181}
182
183void wxCalendarCtrl::InitColours()
184{
185 m_colHighlightFg = wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHTTEXT);
186 m_colHighlightBg = wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHT);
187 m_colBackground = wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW);
188 m_colSorrounding = wxSystemSettings::GetColour(wxSYS_COLOUR_GRAYTEXT);
189
190 m_colHolidayFg = *wxRED;
191 // don't set m_colHolidayBg - by default, same as our bg colour
192
193 m_colHeaderFg = *wxBLUE;
194 m_colHeaderBg = *wxLIGHT_GREY;
195}
196
197bool wxCalendarCtrl::Create(wxWindow *parent,
198 wxWindowID id,
199 const wxDateTime& date,
200 const wxPoint& pos,
201 const wxSize& size,
202 long style,
203 const wxString& name)
204{
205 if ( !wxControl::Create(parent, id, pos, size,
206 style | wxCLIP_CHILDREN | wxWANTS_CHARS | wxFULL_REPAINT_ON_RESIZE,
207 wxDefaultValidator, name) )
208 {
209 return false;
210 }
211
212 // needed to get the arrow keys normally used for the dialog navigation
213 SetWindowStyle(style | wxWANTS_CHARS);
214
215 m_date = date.IsValid() ? date : wxDateTime::Today();
216
217 m_lowdate = wxDefaultDateTime;
218 m_highdate = wxDefaultDateTime;
219
220 if ( !HasFlag(wxCAL_SEQUENTIAL_MONTH_SELECTION) )
221 {
222 CreateYearSpinCtrl();
223 m_staticYear = new wxStaticText(GetParent(), wxID_ANY, m_date.Format(_T("%Y")),
224 wxDefaultPosition, wxDefaultSize,
225 wxALIGN_CENTRE);
226
227 CreateMonthComboBox();
228 m_staticMonth = new wxStaticText(GetParent(), wxID_ANY, m_date.Format(_T("%B")),
229 wxDefaultPosition, wxDefaultSize,
230 wxALIGN_CENTRE);
231 }
232
233 ShowCurrentControls();
234
235 // we need to set the position as well because the main control position
236 // is not the same as the one specified in pos if we have the controls
237 // above it
238 SetBestSize(size);
239 SetPosition(pos);
240
241 // Since we don't paint the whole background make sure that the platform
242 // will use the right one.
243 SetBackgroundColour(m_colBackground);
244
245 SetHolidayAttrs();
246
247 return true;
248}
249
250wxCalendarCtrl::~wxCalendarCtrl()
251{
252 for ( size_t n = 0; n < WXSIZEOF(m_attrs); n++ )
253 {
254 delete m_attrs[n];
255 }
256}
257
258// ----------------------------------------------------------------------------
259// Create the wxComboBox and wxSpinCtrl
260// ----------------------------------------------------------------------------
261
262void wxCalendarCtrl::CreateMonthComboBox()
263{
264 m_comboMonth = new wxComboBox(GetParent(), wxID_ANY,
265 wxEmptyString,
266 wxDefaultPosition,
267 wxDefaultSize,
268 0, NULL,
269 wxCB_READONLY | wxCLIP_SIBLINGS);
270
271 wxDateTime::Month m;
272 for ( m = wxDateTime::Jan; m < wxDateTime::Inv_Month; wxNextMonth(m) )
273 {
274 m_comboMonth->Append(wxDateTime::GetMonthName(m));
275 }
276
277 m_comboMonth->SetSelection(GetDate().GetMonth());
278 m_comboMonth->SetSize(wxDefaultCoord,
279 wxDefaultCoord,
280 wxDefaultCoord,
281 wxDefaultCoord,
282 wxSIZE_AUTO_WIDTH|wxSIZE_AUTO_HEIGHT);
283
284 m_comboMonth->Connect(wxEVT_COMMAND_COMBOBOX_SELECTED,
285 wxCommandEventHandler(wxCalendarCtrl::OnMonthChange),
286 NULL, this);
287}
288
289void wxCalendarCtrl::CreateYearSpinCtrl()
290{
291 m_spinYear = new wxSpinCtrl(GetParent(), wxID_ANY,
292 GetDate().Format(_T("%Y")),
293 wxDefaultPosition,
294 wxDefaultSize,
295 wxSP_ARROW_KEYS | wxCLIP_SIBLINGS,
296 -4300, 10000, GetDate().GetYear());
297
298 m_spinYear->Connect(wxEVT_COMMAND_TEXT_UPDATED,
299 wxCommandEventHandler(wxCalendarCtrl::OnYearTextChange),
300 NULL, this);
301
302 m_spinYear->Connect(wxEVT_COMMAND_SPINCTRL_UPDATED,
303 wxCommandEventHandler(wxCalendarCtrl::OnYearChange),
304 NULL, this);
305}
306
307// ----------------------------------------------------------------------------
308// forward wxWin functions to subcontrols
309// ----------------------------------------------------------------------------
310
311bool wxCalendarCtrl::Destroy()
312{
313 if ( m_staticYear )
314 m_staticYear->Destroy();
315 if ( m_spinYear )
316 m_spinYear->Destroy();
317 if ( m_comboMonth )
318 m_comboMonth->Destroy();
319 if ( m_staticMonth )
320 m_staticMonth->Destroy();
321
322 m_staticYear = NULL;
323 m_spinYear = NULL;
324 m_comboMonth = NULL;
325 m_staticMonth = NULL;
326
327 return wxControl::Destroy();
328}
329
330bool wxCalendarCtrl::Show(bool show)
331{
332 if ( !wxControl::Show(show) )
333 {
334 return false;
335 }
336
337 if ( !(GetWindowStyle() & wxCAL_SEQUENTIAL_MONTH_SELECTION) )
338 {
339 if ( GetMonthControl() )
340 {
341 GetMonthControl()->Show(show);
342 GetYearControl()->Show(show);
343 }
344 }
345
346 return true;
347}
348
349bool wxCalendarCtrl::Enable(bool enable)
350{
351 if ( !wxControl::Enable(enable) )
352 {
353 return false;
354 }
355
356 if ( !(GetWindowStyle() & wxCAL_SEQUENTIAL_MONTH_SELECTION) )
357 {
358 GetMonthControl()->Enable(enable);
359 GetYearControl()->Enable(enable);
360 }
361
362 return true;
363}
364
365// ----------------------------------------------------------------------------
366// enable/disable month/year controls
367// ----------------------------------------------------------------------------
368
369void wxCalendarCtrl::ShowCurrentControls()
370{
371 if ( !HasFlag(wxCAL_SEQUENTIAL_MONTH_SELECTION) )
372 {
373 if ( AllowMonthChange() )
374 {
375 m_comboMonth->Show();
376 m_staticMonth->Hide();
377
378 if ( AllowYearChange() )
379 {
380 m_spinYear->Show();
381 m_staticYear->Hide();
382
383 // skip the rest
384 return;
385 }
386 }
387 else
388 {
389 m_comboMonth->Hide();
390 m_staticMonth->Show();
391 }
392
393 // year change not allowed here
394 m_spinYear->Hide();
395 m_staticYear->Show();
396 }
397}
398
399wxControl *wxCalendarCtrl::GetMonthControl() const
400{
401 return AllowMonthChange() ? (wxControl *)m_comboMonth : (wxControl *)m_staticMonth;
402}
403
404wxControl *wxCalendarCtrl::GetYearControl() const
405{
406 return AllowYearChange() ? (wxControl *)m_spinYear : (wxControl *)m_staticYear;
407}
408
409void wxCalendarCtrl::EnableYearChange(bool enable)
410{
411 if ( enable != AllowYearChange() )
412 {
413 long style = GetWindowStyle();
414 if ( enable )
415 style &= ~wxCAL_NO_YEAR_CHANGE;
416 else
417 style |= wxCAL_NO_YEAR_CHANGE;
418 SetWindowStyle(style);
419
420 ShowCurrentControls();
421 if ( GetWindowStyle() & wxCAL_SEQUENTIAL_MONTH_SELECTION )
422 {
423 Refresh();
424 }
425 }
426}
427
428void wxCalendarCtrl::EnableMonthChange(bool enable)
429{
430 if ( enable != AllowMonthChange() )
431 {
432 long style = GetWindowStyle();
433 if ( enable )
434 style &= ~wxCAL_NO_MONTH_CHANGE;
435 else
436 style |= wxCAL_NO_MONTH_CHANGE;
437 SetWindowStyle(style);
438
439 ShowCurrentControls();
440 if ( GetWindowStyle() & wxCAL_SEQUENTIAL_MONTH_SELECTION )
441 {
442 Refresh();
443 }
444 }
445}
446
447// ----------------------------------------------------------------------------
448// changing date
449// ----------------------------------------------------------------------------
450
451bool wxCalendarCtrl::SetDate(const wxDateTime& date)
452{
453 bool retval = true;
454
455 bool sameMonth = m_date.GetMonth() == date.GetMonth(),
456 sameYear = m_date.GetYear() == date.GetYear();
457
458 if ( IsDateInRange(date) )
459 {
460 if ( sameMonth && sameYear )
461 {
462 // just change the day
463 ChangeDay(date);
464 }
465 else
466 {
467 if ( AllowMonthChange() && (AllowYearChange() || sameYear) )
468 {
469 // change everything
470 m_date = date;
471
472 if ( !(GetWindowStyle() & wxCAL_SEQUENTIAL_MONTH_SELECTION) )
473 {
474 // update the controls
475 m_comboMonth->SetSelection(m_date.GetMonth());
476
477 if ( AllowYearChange() )
478 {
479 if ( !m_userChangedYear )
480 m_spinYear->SetValue(m_date.Format(_T("%Y")));
481 }
482 }
483
484 // as the month changed, holidays did too
485 SetHolidayAttrs();
486
487 // update the calendar
488 Refresh();
489 }
490 else
491 {
492 // forbidden
493 retval = false;
494 }
495 }
496 }
497
498 m_userChangedYear = false;
499
500 return retval;
501}
502
503void wxCalendarCtrl::ChangeDay(const wxDateTime& date)
504{
505 if ( m_date != date )
506 {
507 // we need to refresh the row containing the old date and the one
508 // containing the new one
509 wxDateTime dateOld = m_date;
510 m_date = date;
511
512 RefreshDate(dateOld);
513
514 // if the date is in the same row, it was already drawn correctly
515 if ( GetWeek(m_date) != GetWeek(dateOld) )
516 {
517 RefreshDate(m_date);
518 }
519 }
520}
521
522void wxCalendarCtrl::SetDateAndNotify(const wxDateTime& date)
523{
524 wxDateTime::Tm tm1 = m_date.GetTm(),
525 tm2 = date.GetTm();
526
527 wxEventType type;
528 if ( tm1.year != tm2.year )
529 type = wxEVT_CALENDAR_YEAR_CHANGED;
530 else if ( tm1.mon != tm2.mon )
531 type = wxEVT_CALENDAR_MONTH_CHANGED;
532 else if ( tm1.mday != tm2.mday )
533 type = wxEVT_CALENDAR_DAY_CHANGED;
534 else
535 return;
536
537 if ( SetDate(date) )
538 {
539 GenerateEvents(type, wxEVT_CALENDAR_SEL_CHANGED);
540 }
541}
542
543// ----------------------------------------------------------------------------
544// date range
545// ----------------------------------------------------------------------------
546
547bool wxCalendarCtrl::SetLowerDateLimit(const wxDateTime& date /* = wxDefaultDateTime */)
548{
549 bool retval = true;
550
551 if ( !(date.IsValid()) || ( ( m_highdate.IsValid() ) ? ( date <= m_highdate ) : true ) )
552 {
553 m_lowdate = date;
554 }
555 else
556 {
557 retval = false;
558 }
559
560 return retval;
561}
562
563bool wxCalendarCtrl::SetUpperDateLimit(const wxDateTime& date /* = wxDefaultDateTime */)
564{
565 bool retval = true;
566
567 if ( !(date.IsValid()) || ( ( m_lowdate.IsValid() ) ? ( date >= m_lowdate ) : true ) )
568 {
569 m_highdate = date;
570 }
571 else
572 {
573 retval = false;
574 }
575
576 return retval;
577}
578
579bool wxCalendarCtrl::SetDateRange(const wxDateTime& lowerdate /* = wxDefaultDateTime */, const wxDateTime& upperdate /* = wxDefaultDateTime */)
580{
581 bool retval = true;
582
583 if (
584 ( !( lowerdate.IsValid() ) || ( ( upperdate.IsValid() ) ? ( lowerdate <= upperdate ) : true ) ) &&
585 ( !( upperdate.IsValid() ) || ( ( lowerdate.IsValid() ) ? ( upperdate >= lowerdate ) : true ) ) )
586 {
587 m_lowdate = lowerdate;
588 m_highdate = upperdate;
589 }
590 else
591 {
592 retval = false;
593 }
594
595 return retval;
596}
597
598// ----------------------------------------------------------------------------
599// date helpers
600// ----------------------------------------------------------------------------
601
602wxDateTime wxCalendarCtrl::GetStartDate() const
603{
604 wxDateTime::Tm tm = m_date.GetTm();
605
606 wxDateTime date = wxDateTime(1, tm.mon, tm.year);
607
608 // rewind back
609 date.SetToPrevWeekDay(GetWindowStyle() & wxCAL_MONDAY_FIRST
610 ? wxDateTime::Mon : wxDateTime::Sun);
611
612 if ( GetWindowStyle() & wxCAL_SHOW_SURROUNDING_WEEKS )
613 {
614 // We want to offset the calendar if we start on the first..
615 if ( date.GetDay() == 1 )
616 {
617 date -= wxDateSpan::Week();
618 }
619 }
620
621 return date;
622}
623
624bool wxCalendarCtrl::IsDateShown(const wxDateTime& date) const
625{
626 if ( !(GetWindowStyle() & wxCAL_SHOW_SURROUNDING_WEEKS) )
627 {
628 return date.GetMonth() == m_date.GetMonth();
629 }
630 else
631 {
632 return true;
633 }
634}
635
636bool wxCalendarCtrl::IsDateInRange(const wxDateTime& date) const
637{
638 // Check if the given date is in the range specified
639 return ( ( ( m_lowdate.IsValid() ) ? ( date >= m_lowdate ) : true )
640 && ( ( m_highdate.IsValid() ) ? ( date <= m_highdate ) : true ) );
641}
642
643bool wxCalendarCtrl::ChangeYear(wxDateTime* target) const
644{
645 bool retval = false;
646
647 if ( !(IsDateInRange(*target)) )
648 {
649 if ( target->GetYear() < m_date.GetYear() )
650 {
651 if ( target->GetYear() >= GetLowerDateLimit().GetYear() )
652 {
653 *target = GetLowerDateLimit();
654 retval = true;
655 }
656 else
657 {
658 *target = m_date;
659 }
660 }
661 else
662 {
663 if ( target->GetYear() <= GetUpperDateLimit().GetYear() )
664 {
665 *target = GetUpperDateLimit();
666 retval = true;
667 }
668 else
669 {
670 *target = m_date;
671 }
672 }
673 }
674 else
675 {
676 retval = true;
677 }
678
679 return retval;
680}
681
682bool wxCalendarCtrl::ChangeMonth(wxDateTime* target) const
683{
684 bool retval = true;
685
686 if ( !(IsDateInRange(*target)) )
687 {
688 retval = false;
689
690 if ( target->GetMonth() < m_date.GetMonth() )
691 {
692 *target = GetLowerDateLimit();
693 }
694 else
695 {
696 *target = GetUpperDateLimit();
697 }
698 }
699
700 return retval;
701}
702
703size_t wxCalendarCtrl::GetWeek(const wxDateTime& date) const
704{
705 size_t retval = date.GetWeekOfMonth(GetWindowStyle() & wxCAL_MONDAY_FIRST
706 ? wxDateTime::Monday_First
707 : wxDateTime::Sunday_First);
708
709 if ( (GetWindowStyle() & wxCAL_SHOW_SURROUNDING_WEEKS) )
710 {
711 // we need to offset an extra week if we "start" on the 1st of the month
712 wxDateTime::Tm tm = date.GetTm();
713
714 wxDateTime datetest = wxDateTime(1, tm.mon, tm.year);
715
716 // rewind back
717 datetest.SetToPrevWeekDay(GetWindowStyle() & wxCAL_MONDAY_FIRST
718 ? wxDateTime::Mon : wxDateTime::Sun);
719
720 if ( datetest.GetDay() == 1 )
721 {
722 retval += 1;
723 }
724 }
725
726 return retval;
727}
728
729// ----------------------------------------------------------------------------
730// size management
731// ----------------------------------------------------------------------------
732
733// this is a composite control and it must arrange its parts each time its
734// size or position changes: the combobox and spinctrl are along the top of
735// the available area and the calendar takes up therest of the space
736
737// the static controls are supposed to be always smaller than combo/spin so we
738// always use the latter for size calculations and position the static to take
739// the same space
740
741// the constants used for the layout
742#define VERT_MARGIN 5 // distance between combo and calendar
743#ifdef __WXMAC__
744#define HORZ_MARGIN 5 // spin
745#else
746#define HORZ_MARGIN 15 // spin
747#endif
748wxSize wxCalendarCtrl::DoGetBestSize() const
749{
750 // calc the size of the calendar
751 ((wxCalendarCtrl *)this)->RecalcGeometry(); // const_cast
752
753 wxCoord width = 7*m_widthCol,
754 height = 7*m_heightRow + m_rowOffset + VERT_MARGIN;
755
756 if ( !HasFlag(wxCAL_SEQUENTIAL_MONTH_SELECTION) )
757 {
758 // the combobox doesn't report its height correctly (it returns the
759 // height including the drop down list) so don't use it
760 height += m_spinYear->GetBestSize().y;
761
762
763 wxCoord w2 = m_comboMonth->GetBestSize().x + HORZ_MARGIN + GetCharWidth()*6;
764 if (width < w2)
765 width = w2;
766 }
767
768 if ( !HasFlag(wxBORDER_NONE) )
769 {
770 // the border would clip the last line otherwise
771 height += 6;
772 width += 4;
773 }
774
775 wxSize best(width, height);
776 CacheBestSize(best);
777 return best;
778}
779
780void wxCalendarCtrl::DoSetSize(int x, int y,
781 int width, int height,
782 int sizeFlags)
783{
784 wxControl::DoSetSize(x, y, width, height, sizeFlags);
785}
786
787void wxCalendarCtrl::DoMoveWindow(int x, int y, int width, int height)
788{
789 int yDiff;
790
791 if ( !HasFlag(wxCAL_SEQUENTIAL_MONTH_SELECTION) && m_staticMonth )
792 {
793 wxSize sizeCombo = m_comboMonth->GetBestFittingSize();
794 wxSize sizeStatic = m_staticMonth->GetSize();
795 wxSize sizeSpin = m_spinYear->GetSize();
796
797 // wxMSW sometimes reports the wrong combo height,
798 // so on this platform we'll use the spin control
799 // height instead.
800#ifdef __WXMSW__
801 int maxHeight = sizeSpin.y;
802 int requiredSpinHeight = -1;
803#else
804 int maxHeight = sizeCombo.y;
805 int requiredSpinHeight = sizeCombo.y;
806#endif
807 int dy = (maxHeight - sizeStatic.y) / 2;
808 m_comboMonth->Move(x, y);
809 m_staticMonth->SetSize(x, y + dy, sizeCombo.x, -1, sizeStatic.y);
810
811 int xDiff = sizeCombo.x + HORZ_MARGIN;
812
813 m_spinYear->SetSize(x + xDiff, y, width - xDiff, requiredSpinHeight);
814 m_staticYear->SetSize(x + xDiff, y + dy, width - xDiff, sizeStatic.y);
815
816 yDiff = wxMax(sizeSpin.y, maxHeight) + VERT_MARGIN;
817 }
818 else // no controls on the top
819 {
820 yDiff = 0;
821 }
822
823 wxControl::DoMoveWindow(x, y + yDiff, width, height - yDiff);
824}
825
826void wxCalendarCtrl::DoGetPosition(int *x, int *y) const
827{
828 wxControl::DoGetPosition(x, y);
829#ifndef __WXPM__
830 if ( !HasFlag(wxCAL_SEQUENTIAL_MONTH_SELECTION) && GetMonthControl() )
831 {
832 // our real top corner is not in this position
833 if ( y )
834 {
835 *y -= GetMonthControl()->GetSize().y + VERT_MARGIN;
836 }
837 }
838#endif
839}
840
841void wxCalendarCtrl::DoGetSize(int *width, int *height) const
842{
843 wxControl::DoGetSize(width, height);
844#ifndef __WXPM__
845 if ( !HasFlag(wxCAL_SEQUENTIAL_MONTH_SELECTION) )
846 {
847 // our real height is bigger
848 if ( height && GetMonthControl())
849 {
850 *height += GetMonthControl()->GetSize().y + VERT_MARGIN;
851 }
852 }
853#endif
854}
855
856void wxCalendarCtrl::RecalcGeometry()
857{
858 wxClientDC dc(this);
859
860 dc.SetFont(GetFont());
861
862 // determine the column width (weekday names are not necessarily wider
863 // than the numbers (in some languages), so let's not assume that they are)
864 m_widthCol = 0;
865 for ( int day = 10; day <= 31; day++)
866 {
867 wxCoord width;
868 dc.GetTextExtent(wxString::Format(wxT("%d"), day), &width, &m_heightRow);
869 if ( width > m_widthCol )
870 {
871 // 1.5 times the width gives nice margins even if the weekday
872 // names are short
873 m_widthCol = width+width/2;
874 }
875 }
876 wxDateTime::WeekDay wd;
877 for ( wd = wxDateTime::Sun; wd < wxDateTime::Inv_WeekDay; wxNextWDay(wd) )
878 {
879 wxCoord width;
880 dc.GetTextExtent(m_weekdays[wd], &width, &m_heightRow);
881 if ( width > m_widthCol )
882 {
883 m_widthCol = width;
884 }
885 }
886
887 // leave some margins
888 m_widthCol += 2;
889 m_heightRow += 2;
890
891 m_rowOffset = HasFlag(wxCAL_SEQUENTIAL_MONTH_SELECTION) ? m_heightRow : 0; // conditional in relation to style
892}
893
894// ----------------------------------------------------------------------------
895// drawing
896// ----------------------------------------------------------------------------
897
898void wxCalendarCtrl::OnPaint(wxPaintEvent& WXUNUSED(event))
899{
900 wxPaintDC dc(this);
901
902 dc.SetFont(GetFont());
903
904 RecalcGeometry();
905
906#if DEBUG_PAINT
907 wxLogDebug("--- starting to paint, selection: %s, week %u\n",
908 m_date.Format("%a %d-%m-%Y %H:%M:%S").c_str(),
909 GetWeek(m_date));
910#endif
911
912 wxCoord y = 0;
913 wxCoord x0 = wxMax( (GetSize().x - m_widthCol*7) /2 , 0 );
914
915 if ( HasFlag(wxCAL_SEQUENTIAL_MONTH_SELECTION) )
916 {
917 // draw the sequential month-selector
918
919 dc.SetBackgroundMode(wxTRANSPARENT);
920 dc.SetTextForeground(*wxBLACK);
921 dc.SetBrush(wxBrush(m_colHeaderBg, wxSOLID));
922 dc.SetPen(wxPen(m_colHeaderBg, 1, wxSOLID));
923 dc.DrawRectangle(0, y, GetClientSize().x, m_heightRow);
924
925 // Get extent of month-name + year
926 wxCoord monthw, monthh;
927 wxString headertext = m_date.Format(wxT("%B %Y"));
928 dc.GetTextExtent(headertext, &monthw, &monthh);
929
930 // draw month-name centered above weekdays
931 wxCoord monthx = ((m_widthCol * 7) - monthw) / 2 + x0;
932 wxCoord monthy = ((m_heightRow - monthh) / 2) + y;
933 dc.DrawText(headertext, monthx, monthy);
934
935 // calculate the "month-arrows"
936 wxPoint leftarrow[3];
937 wxPoint rightarrow[3];
938
939 int arrowheight = monthh / 2;
940
941 leftarrow[0] = wxPoint(0, arrowheight / 2);
942 leftarrow[1] = wxPoint(arrowheight / 2, 0);
943 leftarrow[2] = wxPoint(arrowheight / 2, arrowheight - 1);
944
945 rightarrow[0] = wxPoint(0,0);
946 rightarrow[1] = wxPoint(arrowheight / 2, arrowheight / 2);
947 rightarrow[2] = wxPoint(0, arrowheight - 1);
948
949 // draw the "month-arrows"
950
951 wxCoord arrowy = (m_heightRow - arrowheight) / 2;
952 wxCoord larrowx = (m_widthCol - (arrowheight / 2)) / 2 + x0;
953 wxCoord rarrowx = ((m_widthCol - (arrowheight / 2)) / 2) + m_widthCol*6 + x0;
954 m_leftArrowRect = m_rightArrowRect = wxRect(0,0,0,0);
955
956 if ( AllowMonthChange() )
957 {
958 wxDateTime ldpm = wxDateTime(1,m_date.GetMonth(), m_date.GetYear()) - wxDateSpan::Day(); // last day prev month
959 // Check if range permits change
960 if ( IsDateInRange(ldpm) && ( ( ldpm.GetYear() == m_date.GetYear() ) ? true : AllowYearChange() ) )
961 {
962 m_leftArrowRect = wxRect(larrowx - 3, arrowy - 3, (arrowheight / 2) + 8, (arrowheight + 6));
963 dc.SetBrush(*wxBLACK_BRUSH);
964 dc.SetPen(*wxBLACK_PEN);
965 dc.DrawPolygon(3, leftarrow, larrowx , arrowy, wxWINDING_RULE);
966 dc.SetBrush(*wxTRANSPARENT_BRUSH);
967 dc.DrawRectangle(m_leftArrowRect);
968 }
969 wxDateTime fdnm = wxDateTime(1,m_date.GetMonth(), m_date.GetYear()) + wxDateSpan::Month(); // first day next month
970 if ( IsDateInRange(fdnm) && ( ( fdnm.GetYear() == m_date.GetYear() ) ? true : AllowYearChange() ) )
971 {
972 m_rightArrowRect = wxRect(rarrowx - 4, arrowy - 3, (arrowheight / 2) + 8, (arrowheight + 6));
973 dc.SetBrush(*wxBLACK_BRUSH);
974 dc.SetPen(*wxBLACK_PEN);
975 dc.DrawPolygon(3, rightarrow, rarrowx , arrowy, wxWINDING_RULE);
976 dc.SetBrush(*wxTRANSPARENT_BRUSH);
977 dc.DrawRectangle(m_rightArrowRect);
978 }
979 }
980
981 y += m_heightRow;
982 }
983
984 // first draw the week days
985 if ( IsExposed(x0, y, x0 + 7*m_widthCol, m_heightRow) )
986 {
987#if DEBUG_PAINT
988 wxLogDebug("painting the header");
989#endif
990
991 dc.SetBackgroundMode(wxTRANSPARENT);
992 dc.SetTextForeground(m_colHeaderFg);
993 dc.SetBrush(wxBrush(m_colHeaderBg, wxSOLID));
994 dc.SetPen(wxPen(m_colHeaderBg, 1, wxSOLID));
995 dc.DrawRectangle(0, y, GetClientSize().x, m_heightRow);
996
997 bool startOnMonday = (GetWindowStyle() & wxCAL_MONDAY_FIRST) != 0;
998 for ( int wd = 0; wd < 7; wd++ )
999 {
1000 size_t n;
1001 if ( startOnMonday )
1002 n = wd == 6 ? 0 : wd + 1;
1003 else
1004 n = wd;
1005 wxCoord dayw, dayh;
1006 dc.GetTextExtent(m_weekdays[n], &dayw, &dayh);
1007 dc.DrawText(m_weekdays[n], x0 + (wd*m_widthCol) + ((m_widthCol- dayw) / 2), y); // center the day-name
1008 }
1009 }
1010
1011 // then the calendar itself
1012 dc.SetTextForeground(*wxBLACK);
1013 //dc.SetFont(*wxNORMAL_FONT);
1014
1015 y += m_heightRow;
1016 wxDateTime date = GetStartDate();
1017
1018#if DEBUG_PAINT
1019 wxLogDebug("starting calendar from %s\n",
1020 date.Format("%a %d-%m-%Y %H:%M:%S").c_str());
1021#endif
1022
1023 dc.SetBackgroundMode(wxSOLID);
1024 for ( size_t nWeek = 1; nWeek <= 6; nWeek++, y += m_heightRow )
1025 {
1026 // if the update region doesn't intersect this row, don't paint it
1027 if ( !IsExposed(x0, y, x0 + 7*m_widthCol, m_heightRow - 1) )
1028 {
1029 date += wxDateSpan::Week();
1030
1031 continue;
1032 }
1033
1034#if DEBUG_PAINT
1035 wxLogDebug("painting week %d at y = %d\n", nWeek, y);
1036#endif
1037
1038 for ( int wd = 0; wd < 7; wd++ )
1039 {
1040 dc.SetTextBackground(m_colBackground);
1041 if ( IsDateShown(date) )
1042 {
1043 // don't use wxDate::Format() which prepends 0s
1044 unsigned int day = date.GetDay();
1045 wxString dayStr = wxString::Format(_T("%u"), day);
1046 wxCoord width;
1047 dc.GetTextExtent(dayStr, &width, (wxCoord *)NULL);
1048
1049 bool changedColours = false,
1050 changedFont = false;
1051
1052 bool isSel = false;
1053 wxCalendarDateAttr *attr = NULL;
1054
1055 if ( date.GetMonth() != m_date.GetMonth() || !IsDateInRange(date) )
1056 {
1057 // surrounding week or out-of-range
1058 // draw "disabled"
1059 dc.SetTextForeground(m_colSorrounding);
1060 changedColours = true;
1061 }
1062 else
1063 {
1064 isSel = date.IsSameDate(m_date);
1065 attr = m_attrs[day - 1];
1066
1067 if ( isSel )
1068 {
1069 dc.SetTextForeground(m_colHighlightFg);
1070 dc.SetTextBackground(m_colHighlightBg);
1071
1072 changedColours = true;
1073 }
1074 else if ( attr )
1075 {
1076 wxColour colFg, colBg;
1077
1078 if ( attr->IsHoliday() )
1079 {
1080 colFg = m_colHolidayFg;
1081 colBg = m_colHolidayBg;
1082 }
1083 else
1084 {
1085 colFg = attr->GetTextColour();
1086 colBg = attr->GetBackgroundColour();
1087 }
1088
1089 if ( colFg.Ok() )
1090 {
1091 dc.SetTextForeground(colFg);
1092 changedColours = true;
1093 }
1094
1095 if ( colBg.Ok() )
1096 {
1097 dc.SetTextBackground(colBg);
1098 changedColours = true;
1099 }
1100
1101 if ( attr->HasFont() )
1102 {
1103 dc.SetFont(attr->GetFont());
1104 changedFont = true;
1105 }
1106 }
1107 }
1108
1109 wxCoord x = wd*m_widthCol + (m_widthCol - width) / 2 + x0;
1110 dc.DrawText(dayStr, x, y + 1);
1111
1112 if ( !isSel && attr && attr->HasBorder() )
1113 {
1114 wxColour colBorder;
1115 if ( attr->HasBorderColour() )
1116 {
1117 colBorder = attr->GetBorderColour();
1118 }
1119 else
1120 {
1121 colBorder = GetForegroundColour();
1122 }
1123
1124 wxPen pen(colBorder, 1, wxSOLID);
1125 dc.SetPen(pen);
1126 dc.SetBrush(*wxTRANSPARENT_BRUSH);
1127
1128 switch ( attr->GetBorder() )
1129 {
1130 case wxCAL_BORDER_SQUARE:
1131 dc.DrawRectangle(x - 2, y,
1132 width + 4, m_heightRow);
1133 break;
1134
1135 case wxCAL_BORDER_ROUND:
1136 dc.DrawEllipse(x - 2, y,
1137 width + 4, m_heightRow);
1138 break;
1139
1140 default:
1141 wxFAIL_MSG(_T("unknown border type"));
1142 }
1143 }
1144
1145 if ( changedColours )
1146 {
1147 dc.SetTextForeground(GetForegroundColour());
1148 dc.SetTextBackground(GetBackgroundColour());
1149 }
1150
1151 if ( changedFont )
1152 {
1153 dc.SetFont(GetFont());
1154 }
1155 }
1156 //else: just don't draw it
1157
1158 date += wxDateSpan::Day();
1159 }
1160 }
1161
1162 // Greying out out-of-range background
1163 bool showSurrounding = (GetWindowStyle() & wxCAL_SHOW_SURROUNDING_WEEKS) != 0;
1164
1165 date = ( showSurrounding ) ? GetStartDate() : wxDateTime(1, m_date.GetMonth(), m_date.GetYear());
1166 if ( !IsDateInRange(date) )
1167 {
1168 wxDateTime firstOOR = GetLowerDateLimit() - wxDateSpan::Day(); // first out-of-range
1169
1170 wxBrush oorbrush = *wxLIGHT_GREY_BRUSH;
1171 oorbrush.SetStyle(wxFDIAGONAL_HATCH);
1172
1173 HighlightRange(&dc, date, firstOOR, wxTRANSPARENT_PEN, &oorbrush);
1174 }
1175
1176 date = ( showSurrounding ) ? GetStartDate() + wxDateSpan::Weeks(6) - wxDateSpan::Day() : wxDateTime().SetToLastMonthDay(m_date.GetMonth(), m_date.GetYear());
1177 if ( !IsDateInRange(date) )
1178 {
1179 wxDateTime firstOOR = GetUpperDateLimit() + wxDateSpan::Day(); // first out-of-range
1180
1181 wxBrush oorbrush = *wxLIGHT_GREY_BRUSH;
1182 oorbrush.SetStyle(wxFDIAGONAL_HATCH);
1183
1184 HighlightRange(&dc, firstOOR, date, wxTRANSPARENT_PEN, &oorbrush);
1185 }
1186
1187#if DEBUG_PAINT
1188 wxLogDebug("+++ finished painting");
1189#endif
1190}
1191
1192void wxCalendarCtrl::RefreshDate(const wxDateTime& date)
1193{
1194 RecalcGeometry();
1195
1196 wxRect rect;
1197
1198 // always refresh the whole row at once because our OnPaint() will draw
1199 // the whole row anyhow - and this allows the small optimisation in
1200 // OnClick() below to work
1201 rect.x = wxMax( (GetSize().x - m_widthCol*7) /2 , 0 );
1202
1203 rect.y = (m_heightRow * GetWeek(date)) + m_rowOffset;
1204
1205 rect.width = 7*m_widthCol;
1206 rect.height = m_heightRow;
1207
1208#ifdef __WXMSW__
1209 // VZ: for some reason, the selected date seems to occupy more space under
1210 // MSW - this is probably some bug in the font size calculations, but I
1211 // don't know where exactly. This fix is ugly and leads to more
1212 // refreshes than really needed, but without it the selected days
1213 // leaves even more ugly underscores on screen.
1214 rect.Inflate(0, 1);
1215#endif // MSW
1216
1217#if DEBUG_PAINT
1218 wxLogDebug("*** refreshing week %d at (%d, %d)-(%d, %d)\n",
1219 GetWeek(date),
1220 rect.x, rect.y,
1221 rect.x + rect.width, rect.y + rect.height);
1222#endif
1223
1224 Refresh(true, &rect);
1225}
1226
1227void wxCalendarCtrl::HighlightRange(wxPaintDC* pDC, const wxDateTime& fromdate, const wxDateTime& todate, const wxPen* pPen, const wxBrush* pBrush)
1228{
1229 // Highlights the given range using pen and brush
1230 // Does nothing if todate < fromdate
1231
1232
1233#if DEBUG_PAINT
1234 wxLogDebug("+++ HighlightRange: (%s) - (%s) +++", fromdate.Format("%d %m %Y"), todate.Format("%d %m %Y"));
1235#endif
1236
1237 if ( todate >= fromdate )
1238 {
1239 // do stuff
1240 // date-coordinates
1241 int fd, fw;
1242 int td, tw;
1243
1244 // implicit: both dates must be currently shown - checked by GetDateCoord
1245 if ( GetDateCoord(fromdate, &fd, &fw) && GetDateCoord(todate, &td, &tw) )
1246 {
1247#if DEBUG_PAINT
1248 wxLogDebug("Highlight range: (%i, %i) - (%i, %i)", fd, fw, td, tw);
1249#endif
1250 if ( ( (tw - fw) == 1 ) && ( td < fd ) )
1251 {
1252 // special case: interval 7 days or less not in same week
1253 // split in two separate intervals
1254 wxDateTime tfd = fromdate + wxDateSpan::Days(7-fd);
1255 wxDateTime ftd = tfd + wxDateSpan::Day();
1256#if DEBUG_PAINT
1257 wxLogDebug("Highlight: Separate segments");
1258#endif
1259 // draw separately
1260 HighlightRange(pDC, fromdate, tfd, pPen, pBrush);
1261 HighlightRange(pDC, ftd, todate, pPen, pBrush);
1262 }
1263 else
1264 {
1265 int numpoints;
1266 wxPoint corners[8]; // potentially 8 corners in polygon
1267 wxCoord x0 = wxMax( (GetSize().x - m_widthCol*7) /2 , 0 );
1268
1269 if ( fw == tw )
1270 {
1271 // simple case: same week
1272 numpoints = 4;
1273 corners[0] = wxPoint(x0 + (fd - 1) * m_widthCol, (fw * m_heightRow) + m_rowOffset);
1274 corners[1] = wxPoint(x0 + (fd - 1) * m_widthCol, ((fw + 1 ) * m_heightRow) + m_rowOffset);
1275 corners[2] = wxPoint(x0 + td * m_widthCol, ((tw + 1) * m_heightRow) + m_rowOffset);
1276 corners[3] = wxPoint(x0 + td * m_widthCol, (tw * m_heightRow) + m_rowOffset);
1277 }
1278 else
1279 {
1280 int cidx = 0;
1281 // "complex" polygon
1282 corners[cidx] = wxPoint(x0 + (fd - 1) * m_widthCol, (fw * m_heightRow) + m_rowOffset); cidx++;
1283
1284 if ( fd > 1 )
1285 {
1286 corners[cidx] = wxPoint(x0 + (fd - 1) * m_widthCol, ((fw + 1) * m_heightRow) + m_rowOffset); cidx++;
1287 corners[cidx] = wxPoint(x0, ((fw + 1) * m_heightRow) + m_rowOffset); cidx++;
1288 }
1289
1290 corners[cidx] = wxPoint(x0, ((tw + 1) * m_heightRow) + m_rowOffset); cidx++;
1291 corners[cidx] = wxPoint(x0 + td * m_widthCol, ((tw + 1) * m_heightRow) + m_rowOffset); cidx++;
1292
1293 if ( td < 7 )
1294 {
1295 corners[cidx] = wxPoint(x0 + td * m_widthCol, (tw * m_heightRow) + m_rowOffset); cidx++;
1296 corners[cidx] = wxPoint(x0 + 7 * m_widthCol, (tw * m_heightRow) + m_rowOffset); cidx++;
1297 }
1298
1299 corners[cidx] = wxPoint(x0 + 7 * m_widthCol, (fw * m_heightRow) + m_rowOffset); cidx++;
1300
1301 numpoints = cidx;
1302 }
1303
1304 // draw the polygon
1305 pDC->SetBrush(*pBrush);
1306 pDC->SetPen(*pPen);
1307 pDC->DrawPolygon(numpoints, corners);
1308 }
1309 }
1310 }
1311 // else do nothing
1312#if DEBUG_PAINT
1313 wxLogDebug("--- HighlightRange ---");
1314#endif
1315}
1316
1317bool wxCalendarCtrl::GetDateCoord(const wxDateTime& date, int *day, int *week) const
1318{
1319 bool retval = true;
1320
1321#if DEBUG_PAINT
1322 wxLogDebug("+++ GetDateCoord: (%s) +++", date.Format("%d %m %Y"));
1323#endif
1324
1325 if ( IsDateShown(date) )
1326 {
1327 bool startOnMonday = ( GetWindowStyle() & wxCAL_MONDAY_FIRST ) != 0;
1328
1329 // Find day
1330 *day = date.GetWeekDay();
1331
1332 if ( *day == 0 ) // sunday
1333 {
1334 *day = ( startOnMonday ) ? 7 : 1;
1335 }
1336 else
1337 {
1338 *day += ( startOnMonday ) ? 0 : 1;
1339 }
1340
1341 int targetmonth = date.GetMonth() + (12 * date.GetYear());
1342 int thismonth = m_date.GetMonth() + (12 * m_date.GetYear());
1343
1344 // Find week
1345 if ( targetmonth == thismonth )
1346 {
1347 *week = GetWeek(date);
1348 }
1349 else
1350 {
1351 if ( targetmonth < thismonth )
1352 {
1353 *week = 1; // trivial
1354 }
1355 else // targetmonth > thismonth
1356 {
1357 wxDateTime ldcm;
1358 int lastweek;
1359 int lastday;
1360
1361 // get the datecoord of the last day in the month currently shown
1362#if DEBUG_PAINT
1363 wxLogDebug(" +++ LDOM +++");
1364#endif
1365 GetDateCoord(ldcm.SetToLastMonthDay(m_date.GetMonth(), m_date.GetYear()), &lastday, &lastweek);
1366#if DEBUG_PAINT
1367 wxLogDebug(" --- LDOM ---");
1368#endif
1369
1370 wxTimeSpan span = date - ldcm;
1371
1372 int daysfromlast = span.GetDays();
1373#if DEBUG_PAINT
1374 wxLogDebug("daysfromlast: %i", daysfromlast);
1375#endif
1376 if ( daysfromlast + lastday > 7 ) // past week boundary
1377 {
1378 int wholeweeks = (daysfromlast / 7);
1379 *week = wholeweeks + lastweek;
1380 if ( (daysfromlast - (7 * wholeweeks) + lastday) > 7 )
1381 {
1382 *week += 1;
1383 }
1384 }
1385 else
1386 {
1387 *week = lastweek;
1388 }
1389 }
1390 }
1391 }
1392 else
1393 {
1394 *day = -1;
1395 *week = -1;
1396 retval = false;
1397 }
1398
1399#if DEBUG_PAINT
1400 wxLogDebug("--- GetDateCoord: (%s) = (%i, %i) ---", date.Format("%d %m %Y"), *day, *week);
1401#endif
1402
1403 return retval;
1404}
1405
1406// ----------------------------------------------------------------------------
1407// mouse handling
1408// ----------------------------------------------------------------------------
1409
1410void wxCalendarCtrl::OnDClick(wxMouseEvent& event)
1411{
1412 if ( HitTest(event.GetPosition()) != wxCAL_HITTEST_DAY )
1413 {
1414 event.Skip();
1415 }
1416 else
1417 {
1418 GenerateEvent(wxEVT_CALENDAR_DOUBLECLICKED);
1419 }
1420}
1421
1422void wxCalendarCtrl::OnClick(wxMouseEvent& event)
1423{
1424 wxDateTime date;
1425 wxDateTime::WeekDay wday;
1426 switch ( HitTest(event.GetPosition(), &date, &wday) )
1427 {
1428 case wxCAL_HITTEST_DAY:
1429 if ( IsDateInRange(date) )
1430 {
1431 ChangeDay(date);
1432
1433 GenerateEvents(wxEVT_CALENDAR_DAY_CHANGED,
1434 wxEVT_CALENDAR_SEL_CHANGED);
1435 }
1436 break;
1437
1438 case wxCAL_HITTEST_HEADER:
1439 {
1440 wxCalendarEvent eventWd(this, wxEVT_CALENDAR_WEEKDAY_CLICKED);
1441 eventWd.m_wday = wday;
1442 (void)GetEventHandler()->ProcessEvent(eventWd);
1443 }
1444 break;
1445
1446 case wxCAL_HITTEST_DECMONTH:
1447 case wxCAL_HITTEST_INCMONTH:
1448 case wxCAL_HITTEST_SURROUNDING_WEEK:
1449 SetDateAndNotify(date); // we probably only want to refresh the control. No notification.. (maybe as an option?)
1450 break;
1451
1452 default:
1453 wxFAIL_MSG(_T("unknown hittest code"));
1454 // fall through
1455
1456 case wxCAL_HITTEST_NOWHERE:
1457 event.Skip();
1458 break;
1459 }
1460}
1461
1462wxCalendarHitTestResult wxCalendarCtrl::HitTest(const wxPoint& pos,
1463 wxDateTime *date,
1464 wxDateTime::WeekDay *wd)
1465{
1466 RecalcGeometry();
1467 // use the correct x-pos
1468 wxCoord x0 = wxMax((GetSize().x - m_widthCol*7) /2, 0);
1469 wxPoint pos_corr = pos;
1470 pos_corr.x -= x0;
1471
1472 wxCoord y = pos_corr.y;
1473
1474///////////////////////////////////////////////////////////////////////////////////////////////////////
1475 if ( (GetWindowStyle() & wxCAL_SEQUENTIAL_MONTH_SELECTION) )
1476 {
1477 // Header: month
1478
1479 // we need to find out if the hit is on left arrow, on month or on right arrow
1480 // left arrow?
1481 if ( wxRegion(m_leftArrowRect).Contains(pos_corr) == wxInRegion )
1482 {
1483 if ( date )
1484 {
1485 if ( IsDateInRange(m_date - wxDateSpan::Month()) )
1486 {
1487 *date = m_date - wxDateSpan::Month();
1488 }
1489 else
1490 {
1491 *date = GetLowerDateLimit();
1492 }
1493 }
1494
1495 return wxCAL_HITTEST_DECMONTH;
1496 }
1497
1498 if ( wxRegion(m_rightArrowRect).Contains(pos_corr) == wxInRegion )
1499 {
1500 if ( date )
1501 {
1502 if ( IsDateInRange(m_date + wxDateSpan::Month()) )
1503 {
1504 *date = m_date + wxDateSpan::Month();
1505 }
1506 else
1507 {
1508 *date = GetUpperDateLimit();
1509 }
1510 }
1511
1512 return wxCAL_HITTEST_INCMONTH;
1513 }
1514
1515 }
1516
1517///////////////////////////////////////////////////////////////////////////////////////////////////////
1518 // Header: Days
1519
1520 int wday = pos_corr.x / m_widthCol;
1521// if ( y < m_heightRow )
1522 if ( y < (m_heightRow + m_rowOffset) )
1523 {
1524 if ( y > m_rowOffset )
1525 {
1526 if ( wd )
1527 {
1528 if ( GetWindowStyle() & wxCAL_MONDAY_FIRST )
1529 {
1530 wday = wday == 6 ? 0 : wday + 1;
1531 }
1532
1533 *wd = (wxDateTime::WeekDay)wday;
1534 }
1535
1536 return wxCAL_HITTEST_HEADER;
1537 }
1538 else
1539 {
1540 return wxCAL_HITTEST_NOWHERE;
1541 }
1542 }
1543
1544// int week = (y - m_heightRow) / m_heightRow;
1545 int week = (y - (m_heightRow + m_rowOffset)) / m_heightRow;
1546 if ( week >= 6 || wday >= 7 )
1547 {
1548 return wxCAL_HITTEST_NOWHERE;
1549 }
1550
1551 wxDateTime dt = GetStartDate() + wxDateSpan::Days(7*week + wday);
1552
1553 if ( IsDateShown(dt) )
1554 {
1555 if ( date )
1556 *date = dt;
1557
1558 if ( dt.GetMonth() == m_date.GetMonth() )
1559 {
1560
1561 return wxCAL_HITTEST_DAY;
1562 }
1563 else
1564 {
1565 return wxCAL_HITTEST_SURROUNDING_WEEK;
1566 }
1567 }
1568 else
1569 {
1570 return wxCAL_HITTEST_NOWHERE;
1571 }
1572}
1573
1574// ----------------------------------------------------------------------------
1575// subcontrols events handling
1576// ----------------------------------------------------------------------------
1577
1578void wxCalendarCtrl::OnMonthChange(wxCommandEvent& event)
1579{
1580 wxDateTime::Tm tm = m_date.GetTm();
1581
1582 wxDateTime::Month mon = (wxDateTime::Month)event.GetInt();
1583 if ( tm.mday > wxDateTime::GetNumberOfDays(mon, tm.year) )
1584 {
1585 tm.mday = wxDateTime::GetNumberOfDays(mon, tm.year);
1586 }
1587
1588 wxDateTime target = wxDateTime(tm.mday, mon, tm.year);
1589
1590 ChangeMonth(&target);
1591 SetDateAndNotify(target);
1592}
1593
1594void wxCalendarCtrl::OnYearChange(wxCommandEvent& event)
1595{
1596 int year = (int)event.GetInt();
1597 if ( year == INT_MIN )
1598 {
1599 // invalid year in the spin control, ignore it
1600 return;
1601 }
1602
1603 wxDateTime::Tm tm = m_date.GetTm();
1604
1605 if ( tm.mday > wxDateTime::GetNumberOfDays(tm.mon, year) )
1606 {
1607 tm.mday = wxDateTime::GetNumberOfDays(tm.mon, year);
1608 }
1609
1610 wxDateTime target = wxDateTime(tm.mday, tm.mon, year);
1611
1612 if ( ChangeYear(&target) )
1613 {
1614 SetDateAndNotify(target);
1615 }
1616 else
1617 {
1618 // In this case we don't want to change the date. That would put us
1619 // inside the same year but a strange number of months forward/back..
1620 m_spinYear->SetValue(target.GetYear());
1621 }
1622}
1623
1624void wxCalendarCtrl::OnYearTextChange(wxCommandEvent& event)
1625{
1626 SetUserChangedYear();
1627 OnYearChange(event);
1628}
1629
1630// Responds to colour changes, and passes event on to children.
1631void wxCalendarCtrl::OnSysColourChanged(wxSysColourChangedEvent& event)
1632{
1633 // reinit colours
1634 InitColours();
1635
1636 // Propagate the event to the children
1637 wxControl::OnSysColourChanged(event);
1638
1639 // Redraw control area
1640 SetBackgroundColour(m_colBackground);
1641 Refresh();
1642}
1643
1644// ----------------------------------------------------------------------------
1645// keyboard interface
1646// ----------------------------------------------------------------------------
1647
1648void wxCalendarCtrl::OnChar(wxKeyEvent& event)
1649{
1650 wxDateTime target;
1651 switch ( event.GetKeyCode() )
1652 {
1653 case _T('+'):
1654 case WXK_ADD:
1655 target = m_date + wxDateSpan::Year();
1656 if ( ChangeYear(&target) )
1657 {
1658 SetDateAndNotify(target);
1659 }
1660 break;
1661
1662 case _T('-'):
1663 case WXK_SUBTRACT:
1664 target = m_date - wxDateSpan::Year();
1665 if ( ChangeYear(&target) )
1666 {
1667 SetDateAndNotify(target);
1668 }
1669 break;
1670
1671 case WXK_PAGEUP:
1672 target = m_date - wxDateSpan::Month();
1673 ChangeMonth(&target);
1674 SetDateAndNotify(target); // always
1675 break;
1676
1677 case WXK_PAGEDOWN:
1678 target = m_date + wxDateSpan::Month();
1679 ChangeMonth(&target);
1680 SetDateAndNotify(target); // always
1681 break;
1682
1683 case WXK_RIGHT:
1684 if ( event.ControlDown() )
1685 {
1686 target = wxDateTime(m_date).SetToNextWeekDay(
1687 GetWindowStyle() & wxCAL_MONDAY_FIRST
1688 ? wxDateTime::Sun : wxDateTime::Sat);
1689 if ( !IsDateInRange(target) )
1690 {
1691 target = GetUpperDateLimit();
1692 }
1693 SetDateAndNotify(target);
1694 }
1695 else
1696 SetDateAndNotify(m_date + wxDateSpan::Day());
1697 break;
1698
1699 case WXK_LEFT:
1700 if ( event.ControlDown() )
1701 {
1702 target = wxDateTime(m_date).SetToPrevWeekDay(
1703 GetWindowStyle() & wxCAL_MONDAY_FIRST
1704 ? wxDateTime::Mon : wxDateTime::Sun);
1705 if ( !IsDateInRange(target) )
1706 {
1707 target = GetLowerDateLimit();
1708 }
1709 SetDateAndNotify(target);
1710 }
1711 else
1712 SetDateAndNotify(m_date - wxDateSpan::Day());
1713 break;
1714
1715 case WXK_UP:
1716 SetDateAndNotify(m_date - wxDateSpan::Week());
1717 break;
1718
1719 case WXK_DOWN:
1720 SetDateAndNotify(m_date + wxDateSpan::Week());
1721 break;
1722
1723 case WXK_HOME:
1724 if ( event.ControlDown() )
1725 SetDateAndNotify(wxDateTime::Today());
1726 else
1727 SetDateAndNotify(wxDateTime(1, m_date.GetMonth(), m_date.GetYear()));
1728 break;
1729
1730 case WXK_END:
1731 SetDateAndNotify(wxDateTime(m_date).SetToLastMonthDay());
1732 break;
1733
1734 case WXK_RETURN:
1735 GenerateEvent(wxEVT_CALENDAR_DOUBLECLICKED);
1736 break;
1737
1738 default:
1739 event.Skip();
1740 }
1741}
1742
1743// ----------------------------------------------------------------------------
1744// holidays handling
1745// ----------------------------------------------------------------------------
1746
1747void wxCalendarCtrl::EnableHolidayDisplay(bool display)
1748{
1749 long style = GetWindowStyle();
1750 if ( display )
1751 style |= wxCAL_SHOW_HOLIDAYS;
1752 else
1753 style &= ~wxCAL_SHOW_HOLIDAYS;
1754
1755 SetWindowStyle(style);
1756
1757 if ( display )
1758 SetHolidayAttrs();
1759 else
1760 ResetHolidayAttrs();
1761
1762 Refresh();
1763}
1764
1765void wxCalendarCtrl::SetHolidayAttrs()
1766{
1767 if ( GetWindowStyle() & wxCAL_SHOW_HOLIDAYS )
1768 {
1769 ResetHolidayAttrs();
1770
1771 wxDateTime::Tm tm = m_date.GetTm();
1772 wxDateTime dtStart(1, tm.mon, tm.year),
1773 dtEnd = dtStart.GetLastMonthDay();
1774
1775 wxDateTimeArray hol;
1776 wxDateTimeHolidayAuthority::GetHolidaysInRange(dtStart, dtEnd, hol);
1777
1778 size_t count = hol.GetCount();
1779 for ( size_t n = 0; n < count; n++ )
1780 {
1781 SetHoliday(hol[n].GetDay());
1782 }
1783 }
1784}
1785
1786void wxCalendarCtrl::SetHoliday(size_t day)
1787{
1788 wxCHECK_RET( day > 0 && day < 32, _T("invalid day in SetHoliday") );
1789
1790 wxCalendarDateAttr *attr = GetAttr(day);
1791 if ( !attr )
1792 {
1793 attr = new wxCalendarDateAttr;
1794 }
1795
1796 attr->SetHoliday(true);
1797
1798 // can't use SetAttr() because it would delete this pointer
1799 m_attrs[day - 1] = attr;
1800}
1801
1802void wxCalendarCtrl::ResetHolidayAttrs()
1803{
1804 for ( size_t day = 0; day < 31; day++ )
1805 {
1806 if ( m_attrs[day] )
1807 {
1808 m_attrs[day]->SetHoliday(false);
1809 }
1810 }
1811}
1812
1813
1814//static
1815wxVisualAttributes
1816wxCalendarCtrl::GetClassDefaultAttributes(wxWindowVariant variant)
1817{
1818 // Use the same color scheme as wxListBox
1819 return wxListBox::GetClassDefaultAttributes(variant);
1820}
1821
1822#endif // wxUSE_CALENDARCTRL