You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tdepim/kalarm/lib/timeperiod.cpp

385 lines
11 KiB

/*
* timeperiod.h - time period data entry widget
* Program: kalarm
* Copyright © 2003,2004,2007,2008 by David Jarvie <djarvie@kde.org>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
#include "kalarm.h"
#include <tqwidgetstack.h>
#include <tqwhatsthis.h>
#include <klocale.h>
#include <kdialog.h>
#include "combobox.h"
#include "spinbox.h"
#include "timespinbox.h"
#include "timeperiod.moc"
// Collect these widget labels together to ensure consistent wording and
// translations across different modules.
TQString TimePeriod::i18n_minutes() { return i18n("minutes"); }
TQString TimePeriod::i18n_Minutes() { return i18n("Minutes"); }
TQString TimePeriod::i18n_hours_mins() { return i18n("hours/minutes"); }
TQString TimePeriod::i18n_Hours_Mins() { return i18n("Hours/Minutes"); }
TQString TimePeriod::i18n_days() { return i18n("days"); }
TQString TimePeriod::i18n_Days() { return i18n("Days"); }
TQString TimePeriod::i18n_weeks() { return i18n("weeks"); }
TQString TimePeriod::i18n_Weeks() { return i18n("Weeks"); }
static const int maxMinutes = 1000*60-1; // absolute maximum value for hours:minutes = 999H59M
/*=============================================================================
= Class TimePeriod
= Contains a time unit combo box, plus a time spinbox, to select a time period.
=============================================================================*/
TimePeriod::TimePeriod(bool allowHourMinute, TQWidget* parent, const char* name)
: TQHBox(parent, name),
mMaxDays(9999),
mNoHourMinute(!allowHourMinute),
mReadOnly(false)
{
setSpacing(KDialog::spacingHint());
mSpinStack = new TQWidgetStack(this);
mSpinBox = new SpinBox(mSpinStack);
mSpinBox->setLineStep(1);
mSpinBox->setLineShiftStep(10);
mSpinBox->setRange(1, mMaxDays);
connect(mSpinBox, TQT_SIGNAL(valueChanged(int)), TQT_SLOT(slotDaysChanged(int)));
mSpinStack->addWidget(mSpinBox, 0);
mTimeSpinBox = new TimeSpinBox(0, 99999, mSpinStack);
mTimeSpinBox->setRange(1, maxMinutes); // max 999H59M
connect(mTimeSpinBox, TQT_SIGNAL(valueChanged(int)), TQT_SLOT(slotTimeChanged(int)));
mSpinStack->addWidget(mTimeSpinBox, 1);
mSpinStack->setFixedSize(mSpinBox->sizeHint().expandedTo(mTimeSpinBox->sizeHint()));
mHourMinuteRaised = mNoHourMinute;
showHourMin(!mNoHourMinute);
mUnitsCombo = new ComboBox(false, this);
if (mNoHourMinute)
mDateOnlyOffset = 2;
else
{
mDateOnlyOffset = 0;
mUnitsCombo->insertItem(i18n_minutes());
mUnitsCombo->insertItem(i18n_hours_mins());
}
mUnitsCombo->insertItem(i18n_days());
mUnitsCombo->insertItem(i18n_weeks());
mMaxUnitShown = WEEKS;
mUnitsCombo->setFixedSize(mUnitsCombo->sizeHint());
connect(mUnitsCombo, TQT_SIGNAL(activated(int)), TQT_SLOT(slotUnitsSelected(int)));
setFocusProxy(mUnitsCombo);
setTabOrder(mUnitsCombo, mSpinStack);
}
void TimePeriod::setReadOnly(bool ro)
{
if (ro != mReadOnly)
{
mReadOnly = ro;
mSpinBox->setReadOnly(ro);
mTimeSpinBox->setReadOnly(ro);
mUnitsCombo->setReadOnly(ro);
}
}
/******************************************************************************
* Set whether the editor text is to be selected whenever spin buttons are
* clicked. Default is to select them.
*/
void TimePeriod::setSelectOnStep(bool sel)
{
mSpinBox->setSelectOnStep(sel);
mTimeSpinBox->setSelectOnStep(sel);
}
/******************************************************************************
* Set the input focus on the count field.
*/
void TimePeriod::setFocusOnCount()
{
mSpinStack->setFocus();
}
/******************************************************************************
* Set the maximum values for the hours:minutes and days/weeks spinboxes.
* If 'hourmin' = 0, the hours:minutes maximum is left unchanged.
*/
void TimePeriod::setMaximum(int hourmin, int days)
{
int oldmins = minutes();
if (hourmin > 0)
{
if (hourmin > maxMinutes)
hourmin = maxMinutes;
mTimeSpinBox->setRange(1, hourmin);
}
mMaxDays = (days >= 0) ? days : 0;
adjustDayWeekShown();
setUnitRange();
int mins = minutes();
if (mins != oldmins)
emit valueChanged(mins);
}
/******************************************************************************
* Get the specified number of minutes.
* Reply = 0 if error.
*/
int TimePeriod::minutes() const
{
int factor = 0;
switch (mUnitsCombo->currentItem() + mDateOnlyOffset)
{
case HOURS_MINUTES:
return mTimeSpinBox->value();
case MINUTES: factor = 1; break;
case DAYS: factor = 24*60; break;
case WEEKS: factor = 7*24*60; break;
}
return mSpinBox->value() * factor;
}
/******************************************************************************
* Initialise the controls with a specified time period.
* The time unit combo-box is initialised to 'defaultUnits', but if 'dateOnly'
* is true, it will never be initialised to minutes or hours/minutes.
*/
void TimePeriod::setMinutes(int mins, bool dateOnly, TimePeriod::Units defaultUnits)
{
int oldmins = minutes();
if (!dateOnly && mNoHourMinute)
dateOnly = true;
int item;
if (mins)
{
int count = mins;
if (mins % (24*60))
item = (defaultUnits == MINUTES && count <= mSpinBox->maxValue()) ? MINUTES : HOURS_MINUTES;
else if (mins % (7*24*60))
{
item = DAYS;
count = mins / (24*60);
}
else
{
item = WEEKS;
count = mins / (7*24*60);
}
if (item < mDateOnlyOffset)
item = mDateOnlyOffset;
else if (item > mMaxUnitShown)
item = mMaxUnitShown;
mUnitsCombo->setCurrentItem(item - mDateOnlyOffset);
if (item == HOURS_MINUTES)
mTimeSpinBox->setValue(count);
else
mSpinBox->setValue(count);
item = setDateOnly(mins, dateOnly, false);
}
else
{
item = defaultUnits;
if (item < mDateOnlyOffset)
item = mDateOnlyOffset;
else if (item > mMaxUnitShown)
item = mMaxUnitShown;
mUnitsCombo->setCurrentItem(item - mDateOnlyOffset);
if (dateOnly && !mDateOnlyOffset || !dateOnly && mDateOnlyOffset)
item = setDateOnly(mins, dateOnly, false);
}
showHourMin(item == HOURS_MINUTES && !mNoHourMinute);
int newmins = minutes();
if (newmins != oldmins)
emit valueChanged(newmins);
}
/******************************************************************************
* Enable/disable hours/minutes units (if hours/minutes were permitted in the
* constructor).
*/
TimePeriod::Units TimePeriod::setDateOnly(int mins, bool dateOnly, bool signal)
{
int oldmins = 0;
if (signal)
oldmins = minutes();
int index = mUnitsCombo->currentItem();
Units units = static_cast<Units>(index + mDateOnlyOffset);
if (!mNoHourMinute)
{
if (!dateOnly && mDateOnlyOffset)
{
// Change from date-only to allow hours/minutes
mUnitsCombo->insertItem(i18n_minutes(), 0);
mUnitsCombo->insertItem(i18n_hours_mins(), 1);
mDateOnlyOffset = 0;
adjustDayWeekShown();
mUnitsCombo->setCurrentItem(index += 2);
}
else if (dateOnly && !mDateOnlyOffset)
{
// Change from allowing hours/minutes to date-only
mUnitsCombo->removeItem(0);
mUnitsCombo->removeItem(0);
mDateOnlyOffset = 2;
if (index > 2)
index -= 2;
else
index = 0;
adjustDayWeekShown();
mUnitsCombo->setCurrentItem(index);
if (units == HOURS_MINUTES || units == MINUTES)
{
// Set units to days and round up the warning period
units = DAYS;
mUnitsCombo->setCurrentItem(DAYS - mDateOnlyOffset);
mSpinBox->setValue((mins + 1439) / 1440);
}
showHourMin(false);
}
}
if (signal)
{
int newmins = minutes();
if (newmins != oldmins)
emit valueChanged(newmins);
}
return units;
}
/******************************************************************************
* Adjust the days/weeks units shown to suit the maximum days limit.
*/
void TimePeriod::adjustDayWeekShown()
{
Units newMaxUnitShown = (mMaxDays >= 7) ? WEEKS : (mMaxDays || mDateOnlyOffset) ? DAYS : HOURS_MINUTES;
if (newMaxUnitShown > mMaxUnitShown)
{
if (mMaxUnitShown < DAYS)
mUnitsCombo->insertItem(i18n_days());
if (newMaxUnitShown == WEEKS)
mUnitsCombo->insertItem(i18n_weeks());
}
else if (newMaxUnitShown < mMaxUnitShown)
{
if (mMaxUnitShown == WEEKS)
mUnitsCombo->removeItem(WEEKS - mDateOnlyOffset);
if (newMaxUnitShown < DAYS)
mUnitsCombo->removeItem(DAYS - mDateOnlyOffset);
}
mMaxUnitShown = newMaxUnitShown;
}
/******************************************************************************
* Set the maximum value which may be entered into the day/week count field,
* depending on the current unit selection.
*/
void TimePeriod::setUnitRange()
{
int maxval;
switch (static_cast<Units>(mUnitsCombo->currentItem() + mDateOnlyOffset))
{
case WEEKS:
maxval = mMaxDays / 7;
if (maxval)
break;
mUnitsCombo->setCurrentItem(DAYS - mDateOnlyOffset);
// fall through to DAYS
case DAYS:
maxval = mMaxDays ? mMaxDays : 1;
break;
case MINUTES:
maxval = mTimeSpinBox->maxValue();
break;
case HOURS_MINUTES:
default:
return;
}
mSpinBox->setRange(1, maxval);
}
/******************************************************************************
* Called when a new item is made current in the time units combo box.
*/
void TimePeriod::slotUnitsSelected(int index)
{
setUnitRange();
showHourMin(index + mDateOnlyOffset == HOURS_MINUTES);
emit valueChanged(minutes());
}
/******************************************************************************
* Called when the value of the days/weeks spin box changes.
*/
void TimePeriod::slotDaysChanged(int)
{
if (!mHourMinuteRaised)
emit valueChanged(minutes());
}
/******************************************************************************
* Called when the value of the time spin box changes.
*/
void TimePeriod::slotTimeChanged(int value)
{
if (mHourMinuteRaised)
emit valueChanged(value);
}
/******************************************************************************
* Set the currently displayed count widget.
*/
void TimePeriod::showHourMin(bool hourMinute)
{
if (hourMinute != mHourMinuteRaised)
{
mHourMinuteRaised = hourMinute;
if (hourMinute)
{
mSpinStack->raiseWidget(mTimeSpinBox);
mSpinStack->setFocusProxy(mTimeSpinBox);
}
else
{
mSpinStack->raiseWidget(mSpinBox);
mSpinStack->setFocusProxy(mSpinBox);
}
}
}
/******************************************************************************
* Set separate WhatsThis texts for the count spinboxes and the units combobox.
* If the hours:minutes text is omitted, both spinboxes are set to the same
* WhatsThis text.
*/
void TimePeriod::setWhatsThis(const TQString& units, const TQString& dayWeek, const TQString& hourMin)
{
TQWhatsThis::add(mUnitsCombo, units);
TQWhatsThis::add(mSpinBox, dayWeek);
TQWhatsThis::add(mTimeSpinBox, (hourMin.isNull() ? dayWeek : hourMin));
}