kde-playground/kdepim/kalarm/lib/spinbox.cpp
2015-04-14 21:49:29 +00:00

540 lines
16 KiB
C++

/*
* spinbox.cpp - spin box with read-only option and shift-click step value
* Program: kalarm
* Copyright © 2002,2004-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 <kdebug.h>
#include "spinbox.h"
#include <QLineEdit>
#include <QObject>
#include <QApplication>
#include <QStyle>
#include <QStyleOptionSpinBox>
#include <QPainter>
#include <QMouseEvent>
#include <QKeyEvent>
#include <QEvent>
SpinBox::SpinBox(QWidget* parent)
: QSpinBox(parent),
mMinValue(QSpinBox::minimum()),
mMaxValue(QSpinBox::maximum())
{
setRange(0, 99999);
init();
}
SpinBox::SpinBox(int minValue, int maxValue, QWidget* parent)
: QSpinBox(parent),
mMinValue(minValue),
mMaxValue(maxValue)
{
setRange(minValue, maxValue);
init();
}
void SpinBox::init()
{
int step = QSpinBox::singleStep();
mLineStep = step;
mLineShiftStep = step;
mCurrentButton = NO_BUTTON;
mShiftMouse = false;
mShiftMinBound = false;
mShiftMaxBound = false;
mSelectOnStep = true;
mUpDownOnly = false;
mReadOnly = false;
mSuppressSignals = false;
mEdited = false;
lineEdit()->installEventFilter(this); // handle shift-up/down arrow presses
// Detect when the text field is edited
connect(lineEdit(), SIGNAL(textChanged(QString)), SLOT(textEdited()));
connect(this, SIGNAL(valueChanged(int)), SLOT(valueChange()));
}
void SpinBox::setReadOnly(bool ro)
{
if ((int)ro != (int)mReadOnly)
{
mReadOnly = ro;
lineEdit()->setReadOnly(ro);
if (ro)
setShiftStepping(false, mCurrentButton);
}
}
int SpinBox::bound(int val) const
{
return (val < mMinValue) ? mMinValue : (val > mMaxValue) ? mMaxValue : val;
}
void SpinBox::setMinimum(int val)
{
mMinValue = val;
QSpinBox::setMinimum(val);
mShiftMinBound = false;
}
void SpinBox::setMaximum(int val)
{
mMaxValue = val;
QSpinBox::setMaximum(val);
mShiftMaxBound = false;
}
void SpinBox::setSingleStep(int step)
{
mLineStep = step;
if (!mShiftMouse)
QSpinBox::setSingleStep(step);
}
void SpinBox::setSingleShiftStep(int step)
{
mLineShiftStep = step;
if (mShiftMouse)
QSpinBox::setSingleStep(step);
}
void SpinBox::stepBy(int steps)
{
int increment = steps * QSpinBox::singleStep();
addValue(increment);
emit stepped(increment);
}
/******************************************************************************
* Adds a positive or negative increment to the current value, wrapping as appropriate.
* If 'current' is true, any temporary 'shift' values for the range are used instead
* of the real maximum and minimum values.
*/
void SpinBox::addValue(int change, bool current)
{
int newval = value() + change;
int maxval = current ? QSpinBox::maximum() : mMaxValue;
int minval = current ? QSpinBox::minimum() : mMinValue;
if (wrapping())
{
int range = maxval - minval + 1;
if (newval > maxval)
newval = minval + (newval - maxval - 1) % range;
else if (newval < minval)
newval = maxval - (minval - 1 - newval) % range;
}
else
{
if (newval > maxval)
newval = maxval;
else if (newval < minval)
newval = minval;
}
setValue(newval);
}
void SpinBox::valueChange()
{
if (!mSuppressSignals)
{
int val = value();
if (mShiftMinBound && val >= mMinValue)
{
// Reinstate the minimum bound now that the value has returned to the normal range.
QSpinBox::setMinimum(mMinValue);
mShiftMinBound = false;
}
if (mShiftMaxBound && val <= mMaxValue)
{
// Reinstate the maximum bound now that the value has returned to the normal range.
QSpinBox::setMaximum(mMaxValue);
mShiftMaxBound = false;
}
if (!mSelectOnStep && hasFocus())
lineEdit()->deselect(); // prevent selection of the spin box text
}
}
/******************************************************************************
* Called whenever the line edit text is changed.
*/
void SpinBox::textEdited()
{
mEdited = true;
}
/******************************************************************************
* Receives events destined for the spin widget or for the edit field.
*/
bool SpinBox::eventFilter(QObject* obj, QEvent* e)
{
if (obj == lineEdit())
{
int step = 0;
switch (e->type())
{
case QEvent::KeyPress:
{
// Up and down arrow keys step the value
QKeyEvent* ke = (QKeyEvent*)e;
int key = ke->key();
if (key == Qt::Key_Up)
step = 1;
else if (key == Qt::Key_Down)
step = -1;
break;
}
case QEvent::Wheel:
{
QWheelEvent* we = (QWheelEvent*)e;
step = (we->delta() > 0) ? 1 : -1;
break;
}
default:
break;
}
if (step)
{
if (mReadOnly)
return true; // discard up/down arrow keys
QInputEvent* ie = (QInputEvent*)e;
if ((ie->modifiers() & (Qt::ShiftModifier | Qt::AltModifier)) == Qt::ShiftModifier)
{
// Shift stepping
int val = value();
if (step > 0)
step = mLineShiftStep - val % mLineShiftStep;
else
step = - ((val + mLineShiftStep - 1) % mLineShiftStep + 1);
}
else
step = (step > 0) ? mLineStep : -mLineStep;
addValue(step, false);
return true;
}
}
return QSpinBox::eventFilter(obj, e);
}
void SpinBox::focusOutEvent(QFocusEvent* e)
{
if (mEdited)
{
interpretText();
mEdited = false;
}
QSpinBox::focusOutEvent(e);
}
void SpinBox::mousePressEvent(QMouseEvent* e)
{
if (!clickEvent(e))
QSpinBox::mousePressEvent(e);
}
void SpinBox::mouseDoubleClickEvent(QMouseEvent* e)
{
if (!clickEvent(e))
QSpinBox::mouseDoubleClickEvent(e);
}
bool SpinBox::clickEvent(QMouseEvent* e)
{
if (e->button() == Qt::LeftButton)
{
// It's a left button press. Set normal or shift stepping as appropriate.
if (mReadOnly)
return true; // discard the event
mCurrentButton = whichButton(e->pos());
if (mCurrentButton == NO_BUTTON)
{
e->accept();
return true;
}
bool shift = (e->modifiers() & (Qt::ShiftModifier | Qt::AltModifier)) == Qt::ShiftModifier;
if (setShiftStepping(shift, mCurrentButton))
{
e->accept();
return true; // hide the event from the spin widget
}
}
return false;
}
void SpinBox::wheelEvent(QWheelEvent* e)
{
if (mReadOnly)
return; // discard the event
bool shift = (e->modifiers() & (Qt::ShiftModifier | Qt::AltModifier)) == Qt::ShiftModifier;
if (setShiftStepping(shift, (e->delta() > 0 ? UP : DOWN)))
{
e->accept();
return; // hide the event from the spin widget
}
QSpinBox::wheelEvent(e);
}
void SpinBox::mouseReleaseEvent(QMouseEvent* e)
{
if (e->button() == Qt::LeftButton && mShiftMouse)
setShiftStepping(false, mCurrentButton); // cancel shift stepping
QSpinBox::mouseReleaseEvent(e);
}
void SpinBox::mouseMoveEvent(QMouseEvent* e)
{
if (e->buttons() & Qt::LeftButton)
{
// The left button is down. Track which spin button it's in.
if (mReadOnly)
return; // discard the event
int newButton = whichButton(e->pos());
if (newButton != mCurrentButton)
{
// The mouse has moved to a new spin button.
// Set normal or shift stepping as appropriate.
mCurrentButton = newButton;
bool shift = (e->modifiers() & (Qt::ShiftModifier | Qt::AltModifier)) == Qt::ShiftModifier;
if (setShiftStepping(shift, mCurrentButton))
{
e->accept();
return; // hide the event from the spin widget
}
}
}
QSpinBox::mouseMoveEvent(e);
}
void SpinBox::keyPressEvent(QKeyEvent* e)
{
if (!keyEvent(e))
QSpinBox::keyPressEvent(e);
}
void SpinBox::keyReleaseEvent(QKeyEvent* e)
{
if (!keyEvent(e))
QSpinBox::keyReleaseEvent(e);
}
bool SpinBox::keyEvent(QKeyEvent* e)
{
int key = e->key();
int state = e->modifiers();
if ((QApplication::mouseButtons() & Qt::LeftButton)
&& (key == Qt::Key_Shift || key == Qt::Key_Alt))
{
// The left mouse button is down, and the Shift or Alt key has changed
if (mReadOnly)
return true; // discard the event
bool shift = (state & (Qt::ShiftModifier | Qt::AltModifier)) == Qt::ShiftModifier;
if ((!shift && mShiftMouse) || (shift && !mShiftMouse))
{
// The effective shift state has changed.
// Set normal or shift stepping as appropriate.
if (setShiftStepping(shift, mCurrentButton))
{
e->accept();
return true; // hide the event from the spin widget
}
}
}
return false;
}
/******************************************************************************
* Set spin widget stepping to the normal or shift increment.
*/
bool SpinBox::setShiftStepping(bool shift, int currentButton)
{
if (currentButton == NO_BUTTON)
shift = false;
if (shift && !mShiftMouse)
{
/* The value is to be stepped to a multiple of the shift increment.
* Adjust the value so that after the spin widget steps it, it will be correct.
* Then, if the mouse button is held down, the spin widget will continue to
* step by the shift amount.
*/
int val = value();
int step = (currentButton == UP) ? mLineShiftStep : (currentButton == DOWN) ? -mLineShiftStep : 0;
int adjust = shiftStepAdjustment(val, step);
mShiftMouse = true;
if (adjust)
{
/* The value is to be stepped by other than the shift increment,
* presumably because it is being set to a multiple of the shift
* increment. Achieve this by making the adjustment here, and then
* allowing the normal step processing to complete the job by
* adding/subtracting the normal shift increment.
*/
if (!wrapping())
{
// Prevent the step from going past the spinbox's range, or
// to the minimum value if that has a special text unless it is
// already at the minimum value + 1.
int newval = val + adjust + step;
int svt = specialValueText().isEmpty() ? 0 : 1;
int minval = mMinValue + svt;
if (newval <= minval || newval >= mMaxValue)
{
// Stepping to the minimum or maximum value
if (svt && newval <= mMinValue && val == mMinValue)
newval = mMinValue;
else
newval = (newval <= minval) ? minval : mMaxValue;
QSpinBox::setValue(newval);
emit stepped(step);
return true;
}
// If the interim value will lie outside the spinbox's range,
// temporarily adjust the range to allow the value to be set.
int tempval = val + adjust;
if (tempval < mMinValue)
{
QSpinBox::setMinimum(tempval);
mShiftMinBound = true;
}
else if (tempval > mMaxValue)
{
QSpinBox::setMaximum(tempval);
mShiftMaxBound = true;
}
}
// Don't process changes since this new value will be stepped immediately
mSuppressSignals = true;
bool blocked = signalsBlocked();
blockSignals(true);
addValue(adjust, true);
blockSignals(blocked);
mSuppressSignals = false;
}
QSpinBox::setSingleStep(mLineShiftStep);
}
else if (!shift && mShiftMouse)
{
// Reinstate to normal (non-shift) stepping
QSpinBox::setSingleStep(mLineStep);
QSpinBox::setMinimum(mMinValue);
QSpinBox::setMaximum(mMaxValue);
mShiftMinBound = mShiftMaxBound = false;
mShiftMouse = false;
}
return false;
}
/******************************************************************************
* Return the initial adjustment to the value for a shift step up or down.
* The default is to step up or down to the nearest multiple of the shift
* increment, so the adjustment returned is for stepping up the decrement
* required to round down to a multiple of the shift increment <= current value,
* or for stepping down the increment required to round up to a multiple of the
* shift increment >= current value.
* This method's caller then adjusts the resultant value if necessary to cater
* for the widget's minimum/maximum value, and wrapping.
* This should really be a static method, but it needs to be virtual...
*/
int SpinBox::shiftStepAdjustment(int oldValue, int shiftStep)
{
if (oldValue == 0 || shiftStep == 0)
return 0;
if (shiftStep > 0)
{
if (oldValue >= 0)
return -(oldValue % shiftStep);
else
return (-oldValue - 1) % shiftStep + 1 - shiftStep;
}
else
{
shiftStep = -shiftStep;
if (oldValue >= 0)
return shiftStep - ((oldValue - 1) % shiftStep + 1);
else
return (-oldValue) % shiftStep;
}
}
/******************************************************************************
* Find which spin widget button a mouse event is in.
*/
int SpinBox::whichButton(const QPoint& pos)
{
QStyleOptionSpinBox option;
initStyleOption(option);
if (style()->subControlRect(QStyle::CC_SpinBox, &option, QStyle::SC_SpinBoxUp).contains(pos))
return UP;
if (style()->subControlRect(QStyle::CC_SpinBox, &option, QStyle::SC_SpinBoxDown).contains(pos))
return DOWN;
return NO_BUTTON;
}
QRect SpinBox::upRect() const
{
QStyleOptionSpinBox option;
initStyleOption(option);
return style()->subControlRect(QStyle::CC_SpinBox, &option, QStyle::SC_SpinBoxUp);
}
QRect SpinBox::downRect() const
{
QStyleOptionSpinBox option;
initStyleOption(option);
return style()->subControlRect(QStyle::CC_SpinBox, &option, QStyle::SC_SpinBoxDown);
}
QRect SpinBox::upDownRect() const
{
QStyleOptionSpinBox option;
initStyleOption(option);
return style()->subControlRect(QStyle::CC_SpinBox, &option, QStyle::SC_SpinBoxUp)
| style()->subControlRect(QStyle::CC_SpinBox, &option, QStyle::SC_SpinBoxDown);
}
void SpinBox::paintEvent(QPaintEvent* pe)
{
if (mUpDownOnly)
{
QStyleOptionSpinBox option;
initStyleOption(option);
QPainter painter(this);
style()->drawComplexControl(QStyle::CC_SpinBox, &option, &painter, this);
}
else
QSpinBox::paintEvent(pe);
}
void SpinBox::initStyleOption(QStyleOptionSpinBox& so) const
{
so.init(this);
// so.activeSubControls = ??;
so.subControls = mUpDownOnly ? (QStyle::SC_SpinBoxUp | QStyle::SC_SpinBoxDown | QStyle::SC_SpinBoxFrame)
: (QStyle::SC_SpinBoxUp | QStyle::SC_SpinBoxDown | QStyle::SC_SpinBoxFrame | QStyle::SC_SpinBoxEditField);
so.buttonSymbols = buttonSymbols();
so.frame = hasFrame();
so.stepEnabled = stepEnabled();
}
#include "moc_spinbox.cpp"
// vim: et sw=4: