565 lines
15 KiB
C++
565 lines
15 KiB
C++
/////////////////////////////////////////////////////////////////////////////
|
|
// Name: src/generic/datectlg.cpp
|
|
// Purpose: generic wxDatePickerCtrlGeneric implementation
|
|
// Author: Andreas Pflug
|
|
// Modified by:
|
|
// Created: 2005-01-19
|
|
// Copyright: (c) 2005 Andreas Pflug <pgadmin@pse-consulting.de>
|
|
// Licence: wxWindows licence
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
// ============================================================================
|
|
// declarations
|
|
// ============================================================================
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// headers
|
|
// ----------------------------------------------------------------------------
|
|
|
|
#include "wx/wxprec.h"
|
|
|
|
|
|
#if wxUSE_DATEPICKCTRL
|
|
|
|
#ifndef WX_PRECOMP
|
|
#include "wx/dialog.h"
|
|
#include "wx/dcmemory.h"
|
|
#include "wx/intl.h"
|
|
#include "wx/msgdlg.h"
|
|
#include "wx/panel.h"
|
|
#include "wx/textctrl.h"
|
|
#endif
|
|
|
|
#include "wx/calctrl.h"
|
|
#include "wx/combo.h"
|
|
|
|
#include "wx/datectrl.h"
|
|
#include "wx/generic/datectrl.h"
|
|
#include "wx/uilocale.h"
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// constants
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// global variables
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// local classes
|
|
// ----------------------------------------------------------------------------
|
|
|
|
namespace
|
|
{
|
|
|
|
#if wxUSE_VALIDATORS
|
|
|
|
class DateValidator : public wxValidator
|
|
{
|
|
public:
|
|
explicit DateValidator(const wxString& format) : m_format(format) {}
|
|
DateValidator(const DateValidator& val)
|
|
: wxValidator(val),
|
|
m_format(val.m_format)
|
|
{
|
|
}
|
|
|
|
virtual wxObject *Clone() const wxOVERRIDE
|
|
{
|
|
return new DateValidator(*this);
|
|
}
|
|
|
|
virtual bool Validate(wxWindow *parent) wxOVERRIDE
|
|
{
|
|
// We can only be used with wxComboCtrl, so just a static_cast<> would
|
|
// be safe, but use checked cast to notice any problems in debug build.
|
|
const wxString
|
|
s = wxStaticCast(m_validatorWindow, wxComboCtrl)->GetValue();
|
|
if ( s.empty() )
|
|
{
|
|
// There is no need to tell the user that an empty string is
|
|
// invalid, this shouldn't be a surprise for them.
|
|
return true;
|
|
}
|
|
|
|
wxDateTime dt;
|
|
if ( !dt.ParseFormat(s, m_format) )
|
|
{
|
|
wxMessageBox
|
|
(
|
|
wxString::Format
|
|
(
|
|
_("\"%s\" is not in the expected date format, "
|
|
"please enter it as e.g. \"%s\"."),
|
|
s, wxDateTime::Today().Format(m_format)
|
|
),
|
|
_("Invalid date"),
|
|
wxOK | wxICON_WARNING,
|
|
parent
|
|
);
|
|
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// We don't transfer any data, this validator is used only for validation.
|
|
virtual bool TransferFromWindow() wxOVERRIDE { return true; }
|
|
virtual bool TransferToWindow() wxOVERRIDE { return true; }
|
|
|
|
private:
|
|
const wxString m_format;
|
|
};
|
|
|
|
#endif // wxUSE_VALIDATORS
|
|
|
|
} // anonymous namespace
|
|
|
|
class wxCalendarComboPopup : public wxCalendarCtrl,
|
|
public wxComboPopup
|
|
{
|
|
public:
|
|
|
|
wxCalendarComboPopup() : wxCalendarCtrl(),
|
|
wxComboPopup()
|
|
{
|
|
}
|
|
|
|
virtual void Init() wxOVERRIDE
|
|
{
|
|
}
|
|
|
|
// NB: Don't create lazily since it didn't work that way before
|
|
// wxComboCtrl was used, and changing behaviour would almost
|
|
// certainly introduce new bugs.
|
|
virtual bool Create(wxWindow* parent) wxOVERRIDE
|
|
{
|
|
if ( !wxCalendarCtrl::Create(parent, wxID_ANY, wxDefaultDateTime,
|
|
wxPoint(0, 0), wxDefaultSize,
|
|
wxCAL_SEQUENTIAL_MONTH_SELECTION
|
|
| wxCAL_SHOW_HOLIDAYS | wxBORDER_SUNKEN) )
|
|
return false;
|
|
|
|
SetFormat(GetLocaleDateFormat());
|
|
|
|
m_useSize = wxCalendarCtrl::GetBestSize();
|
|
|
|
wxWindow* tx = m_combo->GetTextCtrl();
|
|
if ( !tx )
|
|
tx = m_combo;
|
|
|
|
tx->Bind(wxEVT_KILL_FOCUS, &wxCalendarComboPopup::OnKillTextFocus, this);
|
|
|
|
return true;
|
|
}
|
|
|
|
virtual wxSize GetAdjustedSize(int WXUNUSED(minWidth),
|
|
int WXUNUSED(prefHeight),
|
|
int WXUNUSED(maxHeight)) wxOVERRIDE
|
|
{
|
|
return m_useSize;
|
|
}
|
|
|
|
virtual wxWindow *GetControl() wxOVERRIDE { return this; }
|
|
|
|
void SetDateValue(const wxDateTime& date)
|
|
{
|
|
if ( date.IsValid() )
|
|
{
|
|
m_combo->SetText(date.Format(m_format));
|
|
SetDate(date);
|
|
}
|
|
else // invalid date
|
|
{
|
|
wxASSERT_MSG( HasDPFlag(wxDP_ALLOWNONE),
|
|
wxT("this control must have a valid date") );
|
|
|
|
m_combo->SetText(wxEmptyString);
|
|
}
|
|
}
|
|
|
|
bool IsTextEmpty() const
|
|
{
|
|
return m_combo->GetTextCtrl()->IsEmpty();
|
|
}
|
|
|
|
// This is public because it is used by wxDatePickerCtrlGeneric itself to
|
|
// change the date when the text control field changes. The reason it's
|
|
// done there and not in this class itself is mostly historic.
|
|
void ChangeDateAndNotifyIfValid()
|
|
{
|
|
wxDateTime dt;
|
|
if ( !ParseDateTime(m_combo->GetValue(), &dt) )
|
|
{
|
|
// The user must be in the process of updating the date, don't do
|
|
// anything -- we'll take care of ensuring it's valid on focus loss
|
|
// later.
|
|
return;
|
|
}
|
|
|
|
if ( dt == GetDate() )
|
|
{
|
|
// No need to send event if the date hasn't changed.
|
|
return;
|
|
}
|
|
|
|
// We change the date immediately, as it's more consistent with the
|
|
// native MSW version and avoids another event on focus loss.
|
|
SetDate(dt);
|
|
|
|
SendDateEvent(dt);
|
|
}
|
|
|
|
private:
|
|
bool ParseDateTime(const wxString& s, wxDateTime* pDt) const
|
|
{
|
|
pDt->ParseFormat(s, m_format);
|
|
if ( !pDt->IsValid() )
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
void SendDateEvent(const wxDateTime& dt)
|
|
{
|
|
// Sends both wxCalendarEvent and wxDateEvent
|
|
wxWindow* datePicker = m_combo->GetParent();
|
|
|
|
wxCalendarEvent cev(datePicker, dt, wxEVT_CALENDAR_SEL_CHANGED);
|
|
datePicker->GetEventHandler()->ProcessEvent(cev);
|
|
|
|
wxDateEvent event(datePicker, dt, wxEVT_DATE_CHANGED);
|
|
datePicker->GetEventHandler()->ProcessEvent(event);
|
|
}
|
|
|
|
void OnCalKey(wxKeyEvent & ev)
|
|
{
|
|
if (ev.GetKeyCode() == WXK_ESCAPE && !ev.HasModifiers())
|
|
Dismiss();
|
|
else
|
|
ev.Skip();
|
|
}
|
|
|
|
void OnSelChange(wxCalendarEvent &ev)
|
|
{
|
|
m_combo->SetText(GetDate().Format(m_format));
|
|
|
|
if ( ev.GetEventType() == wxEVT_CALENDAR_DOUBLECLICKED )
|
|
{
|
|
Dismiss();
|
|
}
|
|
|
|
SendDateEvent(GetDate());
|
|
}
|
|
|
|
void OnKillTextFocus(wxFocusEvent &ev)
|
|
{
|
|
ev.Skip();
|
|
|
|
const wxDateTime& dtOld = GetDate();
|
|
|
|
wxDateTime dt;
|
|
wxString value = m_combo->GetValue();
|
|
if ( !ParseDateTime(value, &dt) )
|
|
{
|
|
if ( !HasDPFlag(wxDP_ALLOWNONE) )
|
|
dt = dtOld;
|
|
}
|
|
|
|
if ( dt.IsValid() )
|
|
{
|
|
// Set it at wxCalendarCtrl level.
|
|
SetDate(dt);
|
|
|
|
// And show it in the text field.
|
|
m_combo->SetText(GetStringValue());
|
|
|
|
// And if the date has really changed, send an event about it.
|
|
if ( dt != dtOld )
|
|
SendDateEvent(dt);
|
|
}
|
|
else // Invalid date currently entered.
|
|
{
|
|
if ( HasDPFlag(wxDP_ALLOWNONE) )
|
|
{
|
|
// Clear the text part to indicate that the date is invalid.
|
|
// Would it be better to indicate this in some more clear way,
|
|
// e.g. maybe by using "[none]" or something like this?
|
|
m_combo->SetText(wxString());
|
|
}
|
|
else
|
|
{
|
|
// Restore the original value, as we can't have invalid value
|
|
// in this control.
|
|
m_combo->SetText(GetStringValue());
|
|
}
|
|
}
|
|
}
|
|
|
|
bool HasDPFlag(int flag) const
|
|
{
|
|
return m_combo->GetParent()->HasFlag(flag);
|
|
}
|
|
|
|
// Return the format to be used for the dates shown by the control. This
|
|
// functions honours wxDP_SHOWCENTURY flag.
|
|
wxString GetLocaleDateFormat() const
|
|
{
|
|
#if wxUSE_INTL
|
|
wxString fmt = wxUILocale::GetCurrent().GetInfo(wxLOCALE_SHORT_DATE_FMT);
|
|
if ( HasDPFlag(wxDP_SHOWCENTURY) )
|
|
fmt.Replace("%y", "%Y");
|
|
|
|
return fmt;
|
|
#else // !wxUSE_INTL
|
|
return wxS("%x");
|
|
#endif // wxUSE_INTL/!wxUSE_INTL
|
|
}
|
|
|
|
bool SetFormat(const wxString& fmt)
|
|
{
|
|
m_format = fmt;
|
|
|
|
if ( m_combo )
|
|
{
|
|
#if wxUSE_VALIDATORS
|
|
m_combo->SetValidator(DateValidator(m_format));
|
|
#endif // wxUSE_VALIDATORS
|
|
|
|
if ( GetDate().IsValid() )
|
|
m_combo->SetText(GetDate().Format(m_format));
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
virtual void SetStringValue(const wxString& s) wxOVERRIDE
|
|
{
|
|
wxDateTime dt;
|
|
if ( ParseDateTime(s, &dt) )
|
|
SetDate(dt);
|
|
//else: keep the old value
|
|
}
|
|
|
|
virtual wxString GetStringValue() const wxOVERRIDE
|
|
{
|
|
return GetStringValueFor(GetDate());
|
|
}
|
|
|
|
private:
|
|
// returns either the given date representation using the current format or
|
|
// an empty string if it's invalid
|
|
wxString GetStringValueFor(const wxDateTime& dt) const
|
|
{
|
|
wxString val;
|
|
if ( dt.IsValid() )
|
|
val = dt.Format(m_format);
|
|
|
|
return val;
|
|
}
|
|
|
|
wxSize m_useSize;
|
|
wxString m_format;
|
|
|
|
wxDECLARE_EVENT_TABLE();
|
|
};
|
|
|
|
|
|
wxBEGIN_EVENT_TABLE(wxCalendarComboPopup, wxCalendarCtrl)
|
|
EVT_KEY_DOWN(wxCalendarComboPopup::OnCalKey)
|
|
EVT_CALENDAR_SEL_CHANGED(wxID_ANY, wxCalendarComboPopup::OnSelChange)
|
|
EVT_CALENDAR_PAGE_CHANGED(wxID_ANY, wxCalendarComboPopup::OnSelChange)
|
|
EVT_CALENDAR(wxID_ANY, wxCalendarComboPopup::OnSelChange)
|
|
wxEND_EVENT_TABLE()
|
|
|
|
|
|
// ============================================================================
|
|
// wxDatePickerCtrlGeneric implementation
|
|
// ============================================================================
|
|
|
|
wxBEGIN_EVENT_TABLE(wxDatePickerCtrlGeneric, wxDatePickerCtrlBase)
|
|
EVT_TEXT(wxID_ANY, wxDatePickerCtrlGeneric::OnText)
|
|
EVT_SIZE(wxDatePickerCtrlGeneric::OnSize)
|
|
wxEND_EVENT_TABLE()
|
|
|
|
#ifndef wxHAS_NATIVE_DATEPICKCTRL
|
|
wxIMPLEMENT_DYNAMIC_CLASS(wxDatePickerCtrl, wxControl);
|
|
#endif
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// creation
|
|
// ----------------------------------------------------------------------------
|
|
|
|
bool wxDatePickerCtrlGeneric::Create(wxWindow *parent,
|
|
wxWindowID id,
|
|
const wxDateTime& date,
|
|
const wxPoint& pos,
|
|
const wxSize& size,
|
|
long style,
|
|
const wxValidator& validator,
|
|
const wxString& name)
|
|
{
|
|
wxASSERT_MSG( !(style & wxDP_SPIN),
|
|
wxT("wxDP_SPIN style not supported, use wxDP_DEFAULT") );
|
|
|
|
if ( !wxControl::Create(parent, id, pos, size,
|
|
style | wxCLIP_CHILDREN | wxWANTS_CHARS | wxBORDER_NONE,
|
|
validator, name) )
|
|
{
|
|
return false;
|
|
}
|
|
|
|
InheritAttributes();
|
|
|
|
m_combo = new wxComboCtrl(this, -1, wxEmptyString,
|
|
wxDefaultPosition, wxDefaultSize);
|
|
|
|
m_combo->SetCtrlMainWnd(this);
|
|
|
|
m_popup = new wxCalendarComboPopup();
|
|
|
|
#if defined(__WXMSW__)
|
|
// without this keyboard navigation in month control doesn't work
|
|
m_combo->UseAltPopupWindow();
|
|
#endif
|
|
m_combo->SetPopupControl(m_popup);
|
|
|
|
m_popup->SetDateValue(date.IsValid() ? date : wxDateTime::Today());
|
|
|
|
SetInitialSize(size);
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
void wxDatePickerCtrlGeneric::Init()
|
|
{
|
|
m_combo = NULL;
|
|
m_popup = NULL;
|
|
}
|
|
|
|
wxDatePickerCtrlGeneric::~wxDatePickerCtrlGeneric()
|
|
{
|
|
}
|
|
|
|
bool wxDatePickerCtrlGeneric::Destroy()
|
|
{
|
|
if ( m_combo )
|
|
m_combo->Destroy();
|
|
|
|
m_combo = NULL;
|
|
m_popup = NULL;
|
|
|
|
return wxControl::Destroy();
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// overridden base class methods
|
|
// ----------------------------------------------------------------------------
|
|
|
|
wxSize wxDatePickerCtrlGeneric::DoGetBestSize() const
|
|
{
|
|
wxSize size = m_combo->GetButtonSize();
|
|
|
|
wxTextCtrl* const text = m_combo->GetTextCtrl();
|
|
int w;
|
|
text->GetTextExtent(text->GetValue(), &w, NULL);
|
|
size.x += text->GetSizeFromTextSize(w + 1).x;
|
|
|
|
return size;
|
|
}
|
|
|
|
wxWindowList wxDatePickerCtrlGeneric::GetCompositeWindowParts() const
|
|
{
|
|
wxWindowList parts;
|
|
parts.push_back(m_combo);
|
|
parts.push_back(m_popup);
|
|
return parts;
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// wxDatePickerCtrlGeneric API
|
|
// ----------------------------------------------------------------------------
|
|
|
|
bool
|
|
wxDatePickerCtrlGeneric::SetDateRange(const wxDateTime& lowerdate,
|
|
const wxDateTime& upperdate)
|
|
{
|
|
if ( !m_popup->SetDateRange(lowerdate, upperdate) )
|
|
return false;
|
|
|
|
// If the limits were, check that our current value lies between them and
|
|
// adjust it if it doesn't.
|
|
const wxDateTime old = GetValue();
|
|
if ( old.IsValid() )
|
|
{
|
|
if ( lowerdate.IsValid() && old < lowerdate )
|
|
SetValue(lowerdate);
|
|
else if ( upperdate.IsValid() && old > upperdate )
|
|
SetValue(upperdate);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
wxDateTime wxDatePickerCtrlGeneric::GetValue() const
|
|
{
|
|
if ( HasFlag(wxDP_ALLOWNONE) && m_popup->IsTextEmpty() )
|
|
return wxInvalidDateTime;
|
|
return m_popup->GetDate();
|
|
}
|
|
|
|
|
|
void wxDatePickerCtrlGeneric::SetValue(const wxDateTime& date)
|
|
{
|
|
m_popup->SetDateValue(date);
|
|
}
|
|
|
|
|
|
bool wxDatePickerCtrlGeneric::GetRange(wxDateTime *dt1, wxDateTime *dt2) const
|
|
{
|
|
return m_popup->GetDateRange(dt1, dt2);
|
|
}
|
|
|
|
|
|
void
|
|
wxDatePickerCtrlGeneric::SetRange(const wxDateTime &dt1, const wxDateTime &dt2)
|
|
{
|
|
SetDateRange(dt1, dt2);
|
|
}
|
|
|
|
wxCalendarCtrl *wxDatePickerCtrlGeneric::GetCalendar() const
|
|
{
|
|
return m_popup;
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// event handlers
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
|
void wxDatePickerCtrlGeneric::OnSize(wxSizeEvent& event)
|
|
{
|
|
if ( m_combo )
|
|
m_combo->SetSize(GetClientSize());
|
|
|
|
event.Skip();
|
|
}
|
|
|
|
|
|
void wxDatePickerCtrlGeneric::OnText(wxCommandEvent &ev)
|
|
{
|
|
ev.SetEventObject(this);
|
|
ev.SetId(GetId());
|
|
GetParent()->GetEventHandler()->ProcessEvent(ev);
|
|
|
|
if ( m_popup )
|
|
m_popup->ChangeDateAndNotifyIfValid();
|
|
}
|
|
|
|
#endif // wxUSE_DATEPICKCTRL
|