diff options
Diffstat (limited to 'kdeui/kdatepicker.cpp')
-rw-r--r-- | kdeui/kdatepicker.cpp | 549 |
1 files changed, 549 insertions, 0 deletions
diff --git a/kdeui/kdatepicker.cpp b/kdeui/kdatepicker.cpp new file mode 100644 index 000000000..ae9f97cb7 --- /dev/null +++ b/kdeui/kdatepicker.cpp @@ -0,0 +1,549 @@ +/* -*- C++ -*- + This file is part of the KDE libraries + Copyright (C) 1997 Tim D. Gilman (tdgilman@best.org) + (C) 1998-2001 Mirko Boehm (mirko@kde.org) + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Library General Public + License as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + This library is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + Library General Public License for more details. + + You should have received a copy of the GNU Library General Public License + along with this library; see the file COPYING.LIB. If not, write to + the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, + Boston, MA 02110-1301, USA. +*/ + +#include <qlayout.h> +#include <qframe.h> +#include <qpainter.h> +#include <qdialog.h> +#include <qstyle.h> +#include <qtoolbutton.h> +#include <qcombobox.h> +#include <qtooltip.h> +#include <qfont.h> +#include <qvalidator.h> +#include <qpopupmenu.h> +#include <qtimer.h> + +#include "kdatepicker.h" +#include <kglobal.h> +#include <kapplication.h> +#include <kdialog.h> +#include <klocale.h> +#include <kiconloader.h> +#include <ktoolbar.h> +#include <klineedit.h> +#include <kdebug.h> +#include <knotifyclient.h> +#include <kcalendarsystem.h> + +#include "kdatetbl.h" +#include "kdatepicker.moc" + +// Week numbers are defined by ISO 8601 +// See http://www.merlyn.demon.co.uk/weekinfo.htm for details + +class KDatePicker::KDatePickerPrivate +{ +public: + KDatePickerPrivate() : closeButton(0L), selectWeek(0L), todayButton(0), navigationLayout(0) {} + + void fillWeeksCombo(const QDate &date); + + QToolButton *closeButton; + QComboBox *selectWeek; + QToolButton *todayButton; + QBoxLayout *navigationLayout; +}; + +void KDatePicker::fillWeeksCombo(const QDate &date) +{ + // every year can have a different number of weeks + const KCalendarSystem * calendar = KGlobal::locale()->calendar(); + + // it could be that we had 53,1..52 and now 1..53 which is the same number but different + // so always fill with new values + + d->selectWeek->clear(); + + // We show all week numbers for all weeks between first day of year to last day of year + // This of course can be a list like 53,1,2..52 + + QDate day; + int year = calendar->year(date); + calendar->setYMD(day, year, 1, 1); + int lastMonth = calendar->monthsInYear(day); + QDate lastDay, firstDayOfLastMonth; + calendar->setYMD(firstDayOfLastMonth, year, lastMonth, 1); + calendar->setYMD(lastDay, year, lastMonth, calendar->daysInMonth(firstDayOfLastMonth)); + + for (; day <= lastDay ; day = calendar->addDays(day, 7 /*calendar->daysOfWeek()*/) ) + { + QString week = i18n("Week %1").arg(calendar->weekNumber(day, &year)); + if ( year != calendar->year(day) ) week += "*"; // show that this is a week from a different year + d->selectWeek->insertItem(week); + + // make sure that the week of the lastDay is always inserted: in Chinese calendar + // system, this is not always the case + if(day < lastDay && day.daysTo(lastDay) < 7 && calendar->weekNumber(day) != calendar->weekNumber(lastDay)) + day = lastDay.addDays(-7); + } +} + +KDatePicker::KDatePicker(QWidget *parent, QDate dt, const char *name) + : QFrame(parent,name) +{ + init( dt ); +} + +KDatePicker::KDatePicker(QWidget *parent, QDate dt, const char *name, WFlags f) + : QFrame(parent,name, f) +{ + init( dt ); +} + +KDatePicker::KDatePicker( QWidget *parent, const char *name ) + : QFrame(parent,name) +{ + init( QDate::currentDate() ); +} + +void KDatePicker::init( const QDate &dt ) +{ + d = new KDatePickerPrivate(); + + QBoxLayout * topLayout = new QVBoxLayout(this); + + d->navigationLayout = new QHBoxLayout(topLayout); + d->navigationLayout->addStretch(); + yearBackward = new QToolButton(this); + yearBackward->setAutoRaise(true); + d->navigationLayout->addWidget(yearBackward); + monthBackward = new QToolButton(this); + monthBackward ->setAutoRaise(true); + d->navigationLayout->addWidget(monthBackward); + d->navigationLayout->addSpacing(KDialog::spacingHint()); + + selectMonth = new QToolButton(this); + selectMonth ->setAutoRaise(true); + d->navigationLayout->addWidget(selectMonth); + selectYear = new QToolButton(this); + selectYear->setToggleButton(true); + selectYear->setAutoRaise(true); + d->navigationLayout->addWidget(selectYear); + d->navigationLayout->addSpacing(KDialog::spacingHint()); + + monthForward = new QToolButton(this); + monthForward ->setAutoRaise(true); + d->navigationLayout->addWidget(monthForward); + yearForward = new QToolButton(this); + yearForward ->setAutoRaise(true); + d->navigationLayout->addWidget(yearForward); + d->navigationLayout->addStretch(); + + line = new KLineEdit(this); + val = new KDateValidator(this); + table = new KDateTable(this); + fontsize = KGlobalSettings::generalFont().pointSize(); + if (fontsize == -1) + fontsize = QFontInfo(KGlobalSettings::generalFont()).pointSize(); + + fontsize++; // Make a little bigger + + d->selectWeek = new QComboBox(false, this); // read only week selection + d->todayButton = new QToolButton(this); + d->todayButton->setIconSet(SmallIconSet("today")); + + QToolTip::add(yearForward, i18n("Next year")); + QToolTip::add(yearBackward, i18n("Previous year")); + QToolTip::add(monthForward, i18n("Next month")); + QToolTip::add(monthBackward, i18n("Previous month")); + QToolTip::add(d->selectWeek, i18n("Select a week")); + QToolTip::add(selectMonth, i18n("Select a month")); + QToolTip::add(selectYear, i18n("Select a year")); + QToolTip::add(d->todayButton, i18n("Select the current day")); + + // ----- + setFontSize(fontsize); + line->setValidator(val); + line->installEventFilter( this ); + if ( QApplication::reverseLayout() ) + { + yearForward->setIconSet(BarIconSet(QString::fromLatin1("2leftarrow"))); + yearBackward->setIconSet(BarIconSet(QString::fromLatin1("2rightarrow"))); + monthForward->setIconSet(BarIconSet(QString::fromLatin1("1leftarrow"))); + monthBackward->setIconSet(BarIconSet(QString::fromLatin1("1rightarrow"))); + } + else + { + yearForward->setIconSet(BarIconSet(QString::fromLatin1("2rightarrow"))); + yearBackward->setIconSet(BarIconSet(QString::fromLatin1("2leftarrow"))); + monthForward->setIconSet(BarIconSet(QString::fromLatin1("1rightarrow"))); + monthBackward->setIconSet(BarIconSet(QString::fromLatin1("1leftarrow"))); + } + connect(table, SIGNAL(dateChanged(QDate)), SLOT(dateChangedSlot(QDate))); + connect(table, SIGNAL(tableClicked()), SLOT(tableClickedSlot())); + connect(monthForward, SIGNAL(clicked()), SLOT(monthForwardClicked())); + connect(monthBackward, SIGNAL(clicked()), SLOT(monthBackwardClicked())); + connect(yearForward, SIGNAL(clicked()), SLOT(yearForwardClicked())); + connect(yearBackward, SIGNAL(clicked()), SLOT(yearBackwardClicked())); + connect(d->selectWeek, SIGNAL(activated(int)), SLOT(weekSelected(int))); + connect(d->todayButton, SIGNAL(clicked()), SLOT(todayButtonClicked())); + connect(selectMonth, SIGNAL(clicked()), SLOT(selectMonthClicked())); + connect(selectYear, SIGNAL(toggled(bool)), SLOT(selectYearClicked())); + connect(line, SIGNAL(returnPressed()), SLOT(lineEnterPressed())); + table->setFocus(); + + + topLayout->addWidget(table); + + QBoxLayout * bottomLayout = new QHBoxLayout(topLayout); + bottomLayout->addWidget(d->todayButton); + bottomLayout->addWidget(line); + bottomLayout->addWidget(d->selectWeek); + + table->setDate(dt); + dateChangedSlot(dt); // needed because table emits changed only when newDate != oldDate +} + +KDatePicker::~KDatePicker() +{ + delete d; +} + +bool +KDatePicker::eventFilter(QObject *o, QEvent *e ) +{ + if ( e->type() == QEvent::KeyPress ) { + QKeyEvent *k = (QKeyEvent *)e; + + if ( (k->key() == Qt::Key_Prior) || + (k->key() == Qt::Key_Next) || + (k->key() == Qt::Key_Up) || + (k->key() == Qt::Key_Down) ) + { + QApplication::sendEvent( table, e ); + table->setFocus(); + return true; // eat event + } + } + return QFrame::eventFilter( o, e ); +} + +void +KDatePicker::resizeEvent(QResizeEvent* e) +{ + QWidget::resizeEvent(e); +} + +void +KDatePicker::dateChangedSlot(QDate date) +{ + kdDebug(298) << "KDatePicker::dateChangedSlot: date changed (" << date.year() << "/" << date.month() << "/" << date.day() << ")." << endl; + + const KCalendarSystem * calendar = KGlobal::locale()->calendar(); + + line->setText(KGlobal::locale()->formatDate(date, true)); + selectMonth->setText(calendar->monthName(date, false)); + fillWeeksCombo(date); + + // calculate the item num in the week combo box; normalize selected day so as if 1.1. is the first day of the week + QDate firstDay; + calendar->setYMD(firstDay, calendar->year(date), 1, 1); + d->selectWeek->setCurrentItem((calendar->dayOfYear(date) + calendar->dayOfWeek(firstDay) - 2) / 7/*calendar->daysInWeek()*/); + selectYear->setText(calendar->yearString(date, false)); + + emit(dateChanged(date)); +} + +void +KDatePicker::tableClickedSlot() +{ + kdDebug(298) << "KDatePicker::tableClickedSlot: table clicked." << endl; + emit(dateSelected(table->getDate())); + emit(tableClicked()); +} + +const QDate& +KDatePicker::getDate() const +{ + return table->getDate(); +} + +const QDate & +KDatePicker::date() const +{ + return table->getDate(); +} + +bool +KDatePicker::setDate(const QDate& date) +{ + if(date.isValid()) + { + table->setDate(date); // this also emits dateChanged() which then calls our dateChangedSlot() + return true; + } + else + { + kdDebug(298) << "KDatePicker::setDate: refusing to set invalid date." << endl; + return false; + } +} + +void +KDatePicker::monthForwardClicked() +{ + QDate temp; + temp = KGlobal::locale()->calendar()->addMonths( table->getDate(), 1 ); + + setDate( temp ); +} + +void +KDatePicker::monthBackwardClicked() +{ + QDate temp; + temp = KGlobal::locale()->calendar()->addMonths( table->getDate(), -1 ); + + setDate( temp ); +} + +void +KDatePicker::yearForwardClicked() +{ + QDate temp; + temp = KGlobal::locale()->calendar()->addYears( table->getDate(), 1 ); + + setDate( temp ); +} + +void +KDatePicker::yearBackwardClicked() +{ + QDate temp; + temp = KGlobal::locale()->calendar()->addYears( table->getDate(), -1 ); + + setDate( temp ); +} + +void KDatePicker::selectWeekClicked() {} // ### in 3.2 obsolete; kept for binary compatibility + +void +KDatePicker::weekSelected(int week) +{ + const KCalendarSystem * calendar = KGlobal::locale()->calendar(); + + QDate date = table->getDate(); + int year = calendar->year(date); + + calendar->setYMD(date, year, 1, 1); // first day of selected year + + // calculate the first day in the selected week (day 1 is first day of week) + date = calendar->addDays(date, week * 7/*calendar->daysOfWeek()*/ -calendar->dayOfWeek(date) + 1); + + setDate(date); +} + +void +KDatePicker::selectMonthClicked() +{ + // every year can have different month names (in some calendar systems) + const KCalendarSystem * calendar = KGlobal::locale()->calendar(); + QDate date = table->getDate(); + int i, month, months = calendar->monthsInYear(date); + + QPopupMenu popup(selectMonth); + + for (i = 1; i <= months; i++) + popup.insertItem(calendar->monthName(i, calendar->year(date)), i); + + popup.setActiveItem(calendar->month(date) - 1); + + if ( (month = popup.exec(selectMonth->mapToGlobal(QPoint(0, 0)), calendar->month(date) - 1)) == -1 ) return; // canceled + + int day = calendar->day(date); + // ----- construct a valid date in this month: + calendar->setYMD(date, calendar->year(date), month, 1); + date = date.addDays(QMIN(day, calendar->daysInMonth(date)) - 1); + // ----- set this month + setDate(date); +} + +void +KDatePicker::selectYearClicked() +{ + const KCalendarSystem * calendar = KGlobal::locale()->calendar(); + + if (selectYear->state() == QButton::Off) + { + return; + } + + int year; + KPopupFrame* popup = new KPopupFrame(this); + KDateInternalYearSelector* picker = new KDateInternalYearSelector(popup); + // ----- + picker->resize(picker->sizeHint()); + picker->setYear( table->getDate().year() ); + picker->selectAll(); + popup->setMainWidget(picker); + connect(picker, SIGNAL(closeMe(int)), popup, SLOT(close(int))); + picker->setFocus(); + if(popup->exec(selectYear->mapToGlobal(QPoint(0, selectMonth->height())))) + { + QDate date; + int day; + // ----- + year=picker->getYear(); + date=table->getDate(); + day=calendar->day(date); + // ----- construct a valid date in this month: + //date.setYMD(year, date.month(), 1); + //date.setYMD(year, date.month(), QMIN(day, date.daysInMonth())); + calendar->setYMD(date, year, calendar->month(date), + QMIN(day, calendar->daysInMonth(date))); + // ----- set this month + setDate(date); + } else { + KNotifyClient::beep(); + } + + delete popup; + QTimer::singleShot(0, this, SLOT(ensureSelectYearIsUp())); +} + +void +KDatePicker::ensureSelectYearIsUp() +{ + if (!selectYear->isDown()) + { + selectYear->setOn( false ); + } +} + +void +KDatePicker::setEnabled(bool enable) +{ + QWidget *widgets[]= { + yearForward, yearBackward, monthForward, monthBackward, + selectMonth, selectYear, + line, table, d->selectWeek, d->todayButton }; + const int Size=sizeof(widgets)/sizeof(widgets[0]); + int count; + // ----- + for(count=0; count<Size; ++count) + { + widgets[count]->setEnabled(enable); + } +} + +void +KDatePicker::lineEnterPressed() +{ + QDate temp; + // ----- + if(val->date(line->text(), temp)==QValidator::Acceptable) + { + kdDebug(298) << "KDatePicker::lineEnterPressed: valid date entered." << endl; + emit(dateEntered(temp)); + setDate(temp); + } else { + KNotifyClient::beep(); + kdDebug(298) << "KDatePicker::lineEnterPressed: invalid date entered." << endl; + } +} + +void +KDatePicker::todayButtonClicked() +{ + setDate(QDate::currentDate()); +} + +QSize +KDatePicker::sizeHint() const +{ + return QWidget::sizeHint(); +} + +void +KDatePicker::setFontSize(int s) +{ + QWidget *buttons[]= { + // yearBackward, + // monthBackward, + selectMonth, + selectYear, + // monthForward, + // yearForward + }; + const int NoOfButtons=sizeof(buttons)/sizeof(buttons[0]); + int count; + QFont font; + QRect r; + // ----- + fontsize=s; + for(count=0; count<NoOfButtons; ++count) + { + font=buttons[count]->font(); + font.setPointSize(s); + buttons[count]->setFont(font); + } + QFontMetrics metrics(selectMonth->fontMetrics()); + + for (int i = 1; ; ++i) + { + QString str = KGlobal::locale()->calendar()->monthName(i, + KGlobal::locale()->calendar()->year(table->getDate()), false); + if (str.isNull()) break; + r=metrics.boundingRect(str); + maxMonthRect.setWidth(QMAX(r.width(), maxMonthRect.width())); + maxMonthRect.setHeight(QMAX(r.height(), maxMonthRect.height())); + } + + QSize metricBound = style().sizeFromContents(QStyle::CT_ToolButton, + selectMonth, + maxMonthRect); + selectMonth->setMinimumSize(metricBound); + + table->setFontSize(s); +} + +void +KDatePicker::setCloseButton( bool enable ) +{ + if ( enable == (d->closeButton != 0L) ) + return; + + if ( enable ) { + d->closeButton = new QToolButton( this ); + d->closeButton->setAutoRaise(true); + d->navigationLayout->addSpacing(KDialog::spacingHint()); + d->navigationLayout->addWidget(d->closeButton); + QToolTip::add(d->closeButton, i18n("Close")); + d->closeButton->setPixmap( SmallIcon("remove") ); + connect( d->closeButton, SIGNAL( clicked() ), + topLevelWidget(), SLOT( close() ) ); + } + else { + delete d->closeButton; + d->closeButton = 0L; + } + + updateGeometry(); +} + +bool KDatePicker::hasCloseButton() const +{ + return (d->closeButton); +} + +void KDatePicker::virtual_hook( int /*id*/, void* /*data*/ ) +{ /*BASE::virtual_hook( id, data );*/ } + |