2. several wxDateTime bugs corrected, a couple of missing functions added 3. GetBestSize() corrections for several wxGTK controls 4. wxStaticLine doesn't get focus any more under MSW 5. added DoMoveWindow() to wxMotif git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@5142 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
569 lines
16 KiB
C++
569 lines
16 KiB
C++
///////////////////////////////////////////////////////////////////////////////
|
|
// Name: generic/calctrl.cpp
|
|
// Purpose: implementation fo the generic wxCalendarCtrl
|
|
// Author: Vadim Zeitlin
|
|
// Modified by:
|
|
// Created: 29.12.99
|
|
// RCS-ID: $Id$
|
|
// Copyright: (c) 1999 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
|
|
// Licence: wxWindows license
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// ============================================================================
|
|
// declarations
|
|
// ============================================================================
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// headers
|
|
// ----------------------------------------------------------------------------
|
|
|
|
#ifdef __GNUG__
|
|
#pragma implementation "calctrl.h"
|
|
#endif
|
|
|
|
// For compilers that support precompilation, includes "wx.h".
|
|
#include "wx/wxprec.h"
|
|
|
|
#ifdef __BORLANDC__
|
|
#pragma hdrstop
|
|
#endif
|
|
|
|
#ifndef WX_PRECOMP
|
|
#include "wx/dcclient.h"
|
|
#include "wx/settings.h"
|
|
#include "wx/brush.h"
|
|
#endif //WX_PRECOMP
|
|
|
|
#include "wx/calctrl.h"
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// wxWin macros
|
|
// ----------------------------------------------------------------------------
|
|
|
|
BEGIN_EVENT_TABLE(wxCalendarCtrl, wxControl)
|
|
EVT_PAINT(wxCalendarCtrl::OnPaint)
|
|
|
|
EVT_CHAR(wxCalendarCtrl::OnChar)
|
|
|
|
EVT_LEFT_DOWN(wxCalendarCtrl::OnClick)
|
|
|
|
EVT_COMBOBOX(-1, wxCalendarCtrl::OnMonthChange)
|
|
EVT_SPINCTRL(-1, wxCalendarCtrl::OnYearChange)
|
|
END_EVENT_TABLE()
|
|
|
|
IMPLEMENT_DYNAMIC_CLASS(wxCalendarCtrl, wxControl)
|
|
|
|
// ============================================================================
|
|
// implementation
|
|
// ============================================================================
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// wxCalendarCtrl
|
|
// ----------------------------------------------------------------------------
|
|
|
|
void wxCalendarCtrl::Init()
|
|
{
|
|
m_comboMonth = NULL;
|
|
m_spinYear = NULL;
|
|
|
|
m_widthCol =
|
|
m_heightRow = 0;
|
|
|
|
wxDateTime::WeekDay wd;
|
|
for ( wd = wxDateTime::Sun; wd < wxDateTime::Inv_WeekDay; wxNextWDay(wd) )
|
|
{
|
|
m_weekdays[wd] = wxDateTime::GetWeekDayName(wd, wxDateTime::Name_Abbr);
|
|
}
|
|
}
|
|
|
|
bool wxCalendarCtrl::Create(wxWindow *parent,
|
|
wxWindowID id,
|
|
const wxDateTime& date,
|
|
const wxPoint& pos,
|
|
const wxSize& size,
|
|
long style,
|
|
const wxString& name)
|
|
{
|
|
m_date = date.IsValid() ? date : wxDateTime::Today();
|
|
|
|
wxString monthNames[12];
|
|
wxDateTime::Month m;
|
|
for ( m = wxDateTime::Jan; m < wxDateTime::Inv_Month; wxNextMonth(m) )
|
|
{
|
|
monthNames[m] = wxDateTime::GetMonthName(m);
|
|
}
|
|
|
|
m_comboMonth = new wxComboBox(parent, -1,
|
|
monthNames[m_date.GetMonth()],
|
|
wxDefaultPosition,
|
|
wxDefaultSize,
|
|
WXSIZEOF(monthNames), monthNames,
|
|
wxCB_READONLY);
|
|
|
|
m_spinYear = new wxSpinCtrl(parent, -1,
|
|
m_date.Format(_T("%Y")),
|
|
wxDefaultPosition,
|
|
wxDefaultSize,
|
|
wxSP_ARROW_KEYS,
|
|
-4300, 10000, m_date.GetYear());
|
|
|
|
// we want to process the events from these controls
|
|
m_comboMonth->PushEventHandler(this);
|
|
m_spinYear->PushEventHandler(this);
|
|
|
|
wxSize sizeReal;
|
|
if ( size.x == -1 || size.y == -1 )
|
|
{
|
|
sizeReal = DoGetBestSize();
|
|
if ( size.x != -1 )
|
|
sizeReal.x = size.x;
|
|
if ( size.y != -1 )
|
|
sizeReal.y = size.y;
|
|
}
|
|
else
|
|
{
|
|
sizeReal = size;
|
|
}
|
|
|
|
SetSize(sizeReal);
|
|
|
|
SetBackgroundColour(*wxWHITE);
|
|
SetFont(*wxSWISS_FONT);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// changing date
|
|
// ----------------------------------------------------------------------------
|
|
|
|
void wxCalendarCtrl::SetDate(const wxDateTime& date)
|
|
{
|
|
if ( m_date.GetMonth() == date.GetMonth() &&
|
|
m_date.GetYear() == date.GetYear() )
|
|
{
|
|
// just change the day
|
|
ChangeDay(date);
|
|
}
|
|
else
|
|
{
|
|
// change everything
|
|
m_date = date;
|
|
|
|
// update the controls
|
|
m_comboMonth->SetSelection(m_date.GetMonth());
|
|
m_spinYear->SetValue(m_date.Format(_T("%Y")));
|
|
|
|
// update the calendar
|
|
Refresh();
|
|
}
|
|
}
|
|
|
|
void wxCalendarCtrl::ChangeDay(const wxDateTime& date)
|
|
{
|
|
if ( m_date != date )
|
|
{
|
|
// we need to refresh the row containing the old date and the one
|
|
// containing the new one
|
|
wxDateTime dateOld = m_date;
|
|
m_date = date;
|
|
|
|
RefreshDate(dateOld);
|
|
|
|
// if the date is in the same row, it was already drawn correctly
|
|
if ( GetWeek(m_date) != GetWeek(dateOld) )
|
|
{
|
|
RefreshDate(m_date);
|
|
}
|
|
}
|
|
}
|
|
|
|
void wxCalendarCtrl::SetDateAndNotify(const wxDateTime& date)
|
|
{
|
|
wxDateTime::Tm tm1 = m_date.GetTm(),
|
|
tm2 = date.GetTm();
|
|
|
|
wxEventType type;
|
|
if ( tm1.year != tm2.year )
|
|
type = wxEVT_CALENDAR_YEAR_CHANGED;
|
|
else if ( tm1.mon != tm2.mon )
|
|
type = wxEVT_CALENDAR_MONTH_CHANGED;
|
|
else if ( tm1.mday != tm2.mday )
|
|
type = wxEVT_CALENDAR_DAY_CHANGED;
|
|
else
|
|
return;
|
|
|
|
SetDate(date);
|
|
|
|
GenerateEvent(type);
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// date helpers
|
|
// ----------------------------------------------------------------------------
|
|
|
|
wxDateTime wxCalendarCtrl::GetStartDate() const
|
|
{
|
|
wxDateTime::Tm tm = m_date.GetTm();
|
|
|
|
wxDateTime date = wxDateTime(1, tm.mon, tm.year);
|
|
if ( date.GetWeekDay() != wxDateTime::Sun )
|
|
{
|
|
date.SetToPrevWeekDay(wxDateTime::Sun);
|
|
|
|
// be sure to do it or it might gain 1 hour if DST changed in between
|
|
date.ResetTime();
|
|
}
|
|
//else: we already have it
|
|
|
|
return date;
|
|
}
|
|
|
|
bool wxCalendarCtrl::IsDateShown(const wxDateTime& date) const
|
|
{
|
|
return date.GetMonth() == m_date.GetMonth();
|
|
}
|
|
|
|
size_t wxCalendarCtrl::GetWeek(const wxDateTime& date) const
|
|
{
|
|
return date.GetWeekOfMonth(wxDateTime::Sunday_First);
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// size management
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// this is a composite control and it must arrange its parts each time its
|
|
// size or position changes: the combobox and spinctrl are along the top of
|
|
// the available area and the calendar takes up therest of the space
|
|
|
|
// the constants used for the layout
|
|
#define VERT_MARGIN 5 // distance between combo and calendar
|
|
#define HORZ_MARGIN 15 // spin
|
|
|
|
wxSize wxCalendarCtrl::DoGetBestSize() const
|
|
{
|
|
// calc the size of the calendar
|
|
((wxCalendarCtrl *)this)->RecalcGeometry(); // const_cast
|
|
|
|
wxCoord width = 7*m_widthCol,
|
|
height = 7*m_heightRow;
|
|
|
|
wxSize sizeCombo = m_comboMonth->GetBestSize(),
|
|
sizeSpin = m_spinYear->GetBestSize();
|
|
|
|
height += VERT_MARGIN + wxMax(sizeCombo.y, sizeSpin.y);
|
|
|
|
return wxSize(width, height);
|
|
}
|
|
|
|
void wxCalendarCtrl::DoSetSize(int x, int y,
|
|
int width, int height,
|
|
int sizeFlags)
|
|
{
|
|
wxControl::DoSetSize(x, y, width, height, sizeFlags);
|
|
}
|
|
|
|
void wxCalendarCtrl::DoMoveWindow(int x, int y, int width, int height)
|
|
{
|
|
wxSize sizeCombo = m_comboMonth->GetSize();
|
|
m_comboMonth->Move(x, y);
|
|
|
|
int xDiff = sizeCombo.x + HORZ_MARGIN;
|
|
m_spinYear->SetSize(x + xDiff, y, width - xDiff, -1);
|
|
|
|
wxSize sizeSpin = m_spinYear->GetSize();
|
|
int yDiff = wxMax(sizeSpin.y, sizeCombo.y) + VERT_MARGIN;
|
|
|
|
wxControl::DoMoveWindow(x, y + yDiff, width, height - yDiff);
|
|
}
|
|
|
|
void wxCalendarCtrl::RecalcGeometry()
|
|
{
|
|
if ( m_widthCol != 0 )
|
|
return;
|
|
|
|
wxClientDC dc(this);
|
|
|
|
dc.SetFont(m_font);
|
|
|
|
// determine the column width (we assume that the weekday names are always
|
|
// wider (in any language) than the numbers)
|
|
m_widthCol = 0;
|
|
wxDateTime::WeekDay wd;
|
|
for ( wd = wxDateTime::Sun; wd < wxDateTime::Inv_WeekDay; wxNextWDay(wd) )
|
|
{
|
|
wxCoord width;
|
|
dc.GetTextExtent(m_weekdays[wd], &width, &m_heightRow);
|
|
if ( width > m_widthCol )
|
|
{
|
|
m_widthCol = width;
|
|
}
|
|
}
|
|
|
|
// leave some margins
|
|
m_widthCol += 2;
|
|
m_heightRow += 2;
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// drawing
|
|
// ----------------------------------------------------------------------------
|
|
|
|
void wxCalendarCtrl::OnPaint(wxPaintEvent& event)
|
|
{
|
|
wxPaintDC dc(this);
|
|
|
|
wxDateTime::WeekDay wd;
|
|
|
|
dc.SetFont(m_font);
|
|
|
|
RecalcGeometry();
|
|
|
|
printf("--- starting to paint, selection: %s, week %u\n",
|
|
m_date.Format("%a %d-%m-%Y %H:%M:%S").c_str(),
|
|
GetWeek(m_date));
|
|
|
|
// first draw the week days
|
|
if ( IsExposed(0, 0, 7*m_widthCol, m_heightRow) )
|
|
{
|
|
puts("painting the header");
|
|
|
|
dc.SetTextForeground(*wxBLUE);
|
|
dc.SetBrush(wxBrush(*wxLIGHT_GREY, wxSOLID));
|
|
dc.SetBackgroundMode(wxTRANSPARENT);
|
|
dc.SetPen(*wxLIGHT_GREY_PEN);
|
|
dc.DrawRectangle(0, 0, 7*m_widthCol, m_heightRow);
|
|
for ( wd = wxDateTime::Sun; wd < wxDateTime::Inv_WeekDay; wxNextWDay(wd) )
|
|
{
|
|
dc.DrawText(m_weekdays[wd], wd*m_widthCol + 1, 0);
|
|
}
|
|
}
|
|
|
|
// then the calendar itself
|
|
dc.SetTextForeground(*wxBLACK);
|
|
//dc.SetFont(*wxNORMAL_FONT);
|
|
|
|
wxCoord y = m_heightRow;
|
|
|
|
wxDateTime date = GetStartDate();
|
|
printf("starting calendar from %s\n",
|
|
date.Format("%a %d-%m-%Y %H:%M:%S").c_str());
|
|
|
|
dc.SetBackgroundMode(wxSOLID);
|
|
for ( size_t nWeek = 1; nWeek <= 6; nWeek++, y += m_heightRow )
|
|
{
|
|
// if the update region doesn't intersect this row, don't paint it
|
|
if ( !IsExposed(0, y, 7*m_widthCol, y + m_heightRow - 1) )
|
|
{
|
|
date += wxDateSpan::Week();
|
|
|
|
continue;
|
|
}
|
|
|
|
printf("painting week %d at y = %d\n", nWeek, y);
|
|
|
|
for ( wd = wxDateTime::Sun; wd < wxDateTime::Inv_WeekDay; wxNextWDay(wd) )
|
|
{
|
|
if ( IsDateShown(date) )
|
|
{
|
|
wxString day = wxString::Format(_T("%u"), date.GetDay());
|
|
wxCoord width;
|
|
dc.GetTextExtent(day, &width, (wxCoord *)NULL);
|
|
|
|
bool isSel = m_date == date;
|
|
if ( isSel )
|
|
{
|
|
dc.SetTextForeground(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_HIGHLIGHTTEXT));
|
|
dc.SetTextBackground(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_HIGHLIGHT));
|
|
}
|
|
|
|
dc.DrawText(day, wd*m_widthCol + (m_widthCol - width) / 2, y);
|
|
|
|
if ( isSel )
|
|
{
|
|
dc.SetTextForeground(m_foregroundColour);
|
|
dc.SetTextBackground(m_backgroundColour);
|
|
}
|
|
}
|
|
//else: just don't draw it
|
|
|
|
date += wxDateSpan::Day();
|
|
}
|
|
}
|
|
|
|
puts("+++ finished painting");
|
|
}
|
|
|
|
void wxCalendarCtrl::RefreshDate(const wxDateTime& date)
|
|
{
|
|
RecalcGeometry();
|
|
|
|
wxRect rect;
|
|
|
|
// always refresh the whole row at once because our OnPaint() will draw
|
|
// the whole row anyhow - and this allows the small optimisation in
|
|
// OnClick() below to work
|
|
rect.x = 0;
|
|
rect.y = m_heightRow * GetWeek(date);
|
|
rect.width = 7*m_widthCol;
|
|
rect.height = m_heightRow;
|
|
|
|
printf("*** refreshing week %d at (%d, %d)-(%d, %d)\n",
|
|
GetWeek(date),
|
|
rect.x, rect.y,
|
|
rect.x + rect.width, rect.y + rect.height);
|
|
|
|
Refresh(TRUE, &rect);
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// mouse handling
|
|
// ----------------------------------------------------------------------------
|
|
|
|
void wxCalendarCtrl::OnClick(wxMouseEvent& event)
|
|
{
|
|
RecalcGeometry();
|
|
|
|
wxDateTime date;
|
|
if ( !HitTest(event.GetPosition(), &date) )
|
|
{
|
|
event.Skip();
|
|
}
|
|
else
|
|
{
|
|
ChangeDay(date);
|
|
|
|
GenerateEvent(wxEVT_CALENDAR_DAY_CHANGED);
|
|
}
|
|
}
|
|
|
|
bool wxCalendarCtrl::HitTest(const wxPoint& pos, wxDateTime *date)
|
|
{
|
|
RecalcGeometry();
|
|
|
|
wxCoord y = pos.y;
|
|
if ( y < m_heightRow )
|
|
return FALSE;
|
|
|
|
y -= m_heightRow;
|
|
int week = y / m_heightRow,
|
|
wday = pos.x / m_widthCol;
|
|
|
|
if ( week >= 6 || wday >= 7 )
|
|
return FALSE;
|
|
|
|
wxCHECK_MSG( date, FALSE, _T("bad pointer in wxCalendarCtrl::HitTest") );
|
|
|
|
*date = GetStartDate();
|
|
*date += wxDateSpan::Days(7*week + wday);
|
|
|
|
return IsDateShown(*date);
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// subcontrols events handling
|
|
// ----------------------------------------------------------------------------
|
|
|
|
void wxCalendarCtrl::OnMonthChange(wxCommandEvent& event)
|
|
{
|
|
wxDateTime::Tm tm = m_date.GetTm();
|
|
|
|
wxDateTime::Month mon = (wxDateTime::Month)event.GetInt();
|
|
if ( tm.mday > wxDateTime::GetNumberOfDays(mon, tm.year) )
|
|
{
|
|
tm.mday = wxDateTime::GetNumberOfDays(mon, tm.year);
|
|
}
|
|
|
|
SetDate(wxDateTime(tm.mday, mon, tm.year));
|
|
|
|
GenerateEvent(wxEVT_CALENDAR_MONTH_CHANGED);
|
|
}
|
|
|
|
void wxCalendarCtrl::OnYearChange(wxSpinEvent& event)
|
|
{
|
|
wxDateTime::Tm tm = m_date.GetTm();
|
|
|
|
int year = event.GetInt();
|
|
if ( tm.mday > wxDateTime::GetNumberOfDays(tm.mon, year) )
|
|
{
|
|
tm.mday = wxDateTime::GetNumberOfDays(tm.mon, year);
|
|
}
|
|
|
|
SetDate(wxDateTime(tm.mday, tm.mon, year));
|
|
|
|
GenerateEvent(wxEVT_CALENDAR_YEAR_CHANGED);
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// keyboard interface
|
|
// ----------------------------------------------------------------------------
|
|
|
|
void wxCalendarCtrl::OnChar(wxKeyEvent& event)
|
|
{
|
|
switch ( event.KeyCode() )
|
|
{
|
|
case _T('+'):
|
|
case WXK_ADD:
|
|
SetDateAndNotify(m_date + wxDateSpan::Year());
|
|
break;
|
|
|
|
case _T('-'):
|
|
case WXK_SUBTRACT:
|
|
SetDateAndNotify(m_date - wxDateSpan::Year());
|
|
break;
|
|
|
|
case WXK_PAGEDOWN:
|
|
SetDateAndNotify(m_date + wxDateSpan::Year());
|
|
break;
|
|
|
|
case WXK_PAGEUP:
|
|
SetDateAndNotify(m_date - wxDateSpan::Year());
|
|
break;
|
|
|
|
case WXK_RIGHT:
|
|
SetDateAndNotify(m_date + wxDateSpan::Day());
|
|
break;
|
|
|
|
case WXK_LEFT:
|
|
SetDateAndNotify(m_date - wxDateSpan::Day());
|
|
break;
|
|
|
|
case WXK_UP:
|
|
SetDateAndNotify(m_date - wxDateSpan::Week());
|
|
break;
|
|
|
|
case WXK_DOWN:
|
|
SetDateAndNotify(m_date + wxDateSpan::Week());
|
|
break;
|
|
|
|
case WXK_HOME:
|
|
SetDateAndNotify(wxDateTime::Today());
|
|
break;
|
|
|
|
default:
|
|
event.Skip();
|
|
}
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// wxCalendarEvent
|
|
// ----------------------------------------------------------------------------
|
|
|
|
void wxCalendarCtrl::GenerateEvent(wxEventType type)
|
|
{
|
|
// we're called for a change in some particular date field but we always
|
|
// also generate a generic "changed" event
|
|
wxCalendarEvent event(this, type);
|
|
wxCalendarEvent event2(this, wxEVT_CALENDAR_SEL_CHANGED);
|
|
|
|
(void)GetEventHandler()->ProcessEvent(event);
|
|
(void)GetEventHandler()->ProcessEvent(event2);
|
|
}
|
|
|
|
wxCalendarEvent::wxCalendarEvent(wxCalendarCtrl *cal, wxEventType type)
|
|
: wxCommandEvent(type, cal->GetId())
|
|
{
|
|
m_date = cal->GetDate();
|
|
}
|