123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455 |
- /*=========================================================================
- Library: CTK
- Copyright (c) Kitware Inc.
- Licensed under the Apache License, Version 2.0 (the "License");
- you may not use this file except in compliance with the License.
- You may obtain a copy of the License at
- http://www.apache.org/licenses/LICENSE-2.0.txt
- Unless required by applicable law or agreed to in writing, software
- distributed under the License is distributed on an "AS IS" BASIS,
- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- See the License for the specific language governing permissions and
- limitations under the License.
- =========================================================================*/
- // QT includes
- #include <QDebug>
- #include <QHBoxLayout>
- #include <QHelpEvent>
- #include <QStyle>
- #include <QStyleOptionSlider>
- #include <QToolTip>
- // CTK includes
- #include "ctkDoubleSlider.h"
- // STD includes
- #include <limits>
- //-----------------------------------------------------------------------------
- class ctkSlider: public QSlider
- {
- public:
- ctkSlider(QWidget* parent);
- using QSlider::initStyleOption;
- };
- //-----------------------------------------------------------------------------
- ctkSlider::ctkSlider(QWidget* parent): QSlider(parent)
- {
- }
- //-----------------------------------------------------------------------------
- class ctkDoubleSliderPrivate
- {
- Q_DECLARE_PUBLIC(ctkDoubleSlider);
- protected:
- ctkDoubleSlider* const q_ptr;
- public:
- ctkDoubleSliderPrivate(ctkDoubleSlider& object);
- int toInt(double value)const;
- double fromInt(int value)const;
- double safeFromInt(int value)const;
- void init();
- void updateOffset(double value);
- ctkSlider* Slider;
- QString HandleToolTip;
- double Minimum;
- double Maximum;
- bool SettingRange;
- // we should have a Offset and SliderPositionOffset (and MinimumOffset?)
- double Offset;
- double SingleStep;
- double PageStep;
- double Value;
- };
- // --------------------------------------------------------------------------
- ctkDoubleSliderPrivate::ctkDoubleSliderPrivate(ctkDoubleSlider& object)
- :q_ptr(&object)
- {
- this->Slider = 0;
- this->Minimum = 0.;
- this->Maximum = 100.;
- this->SettingRange = false;
- this->Offset = 0.;
- this->SingleStep = 1.;
- this->PageStep = 10.;
- this->Value = 0.;
- }
- // --------------------------------------------------------------------------
- void ctkDoubleSliderPrivate::init()
- {
- Q_Q(ctkDoubleSlider);
- this->Slider = new ctkSlider(q);
- this->Slider->installEventFilter(q);
- QHBoxLayout* l = new QHBoxLayout(q);
- l->addWidget(this->Slider);
- l->setContentsMargins(0,0,0,0);
-
- this->Minimum = this->Slider->minimum();
- this->Maximum = this->Slider->maximum();
- // this->Slider->singleStep is always 1
- this->SingleStep = this->Slider->singleStep();
- this->PageStep = this->Slider->pageStep();
- this->Value = this->Slider->value();
- q->connect(this->Slider, SIGNAL(valueChanged(int)), q, SLOT(onValueChanged(int)));
- q->connect(this->Slider, SIGNAL(sliderMoved(int)), q, SLOT(onSliderMoved(int)));
- q->connect(this->Slider, SIGNAL(sliderPressed()), q, SIGNAL(sliderPressed()));
- q->connect(this->Slider, SIGNAL(sliderReleased()), q, SIGNAL(sliderReleased()));
- q->connect(this->Slider, SIGNAL(rangeChanged(int,int)),
- q, SLOT(onRangeChanged(int,int)));
- q->setSizePolicy(this->Slider->sizePolicy());
- q->setAttribute(Qt::WA_WState_OwnSizePolicy, false);
- }
-
- // --------------------------------------------------------------------------
- int ctkDoubleSliderPrivate::toInt(double doubleValue)const
- {
- double tmp = doubleValue / this->SingleStep;
- static const double minInt = std::numeric_limits<int>::min();
- static const double maxInt = std::numeric_limits<int>::max();
- #ifndef QT_NO_DEBUG
- if (tmp < minInt || tmp > maxInt)
- {
- qWarning("ctkDoubleSliderPrivate::toInt value out of bounds !");
- }
- #endif
- tmp = qBound(minInt, tmp, maxInt);
- int intValue = qRound(tmp);
- //qDebug() << __FUNCTION__ << doubleValue << tmp << intValue;
- return intValue;
- }
- // --------------------------------------------------------------------------
- double ctkDoubleSliderPrivate::fromInt(int intValue)const
- {
- double doubleValue = this->SingleStep * (this->Offset + intValue) ;
- //qDebug() << __FUNCTION__ << intValue << doubleValue;
- return doubleValue;
- }
- // --------------------------------------------------------------------------
- double ctkDoubleSliderPrivate::safeFromInt(int intValue)const
- {
- return qBound(this->Minimum, this->fromInt(intValue), this->Maximum);
- }
- // --------------------------------------------------------------------------
- void ctkDoubleSliderPrivate::updateOffset(double value)
- {
- this->Offset = (value / this->SingleStep) - this->toInt(value);
- }
- // --------------------------------------------------------------------------
- ctkDoubleSlider::ctkDoubleSlider(QWidget* _parent) : Superclass(_parent)
- , d_ptr(new ctkDoubleSliderPrivate(*this))
- {
- Q_D(ctkDoubleSlider);
- d->init();
- }
- // --------------------------------------------------------------------------
- ctkDoubleSlider::ctkDoubleSlider(Qt::Orientation _orientation, QWidget* _parent)
- : Superclass(_parent)
- , d_ptr(new ctkDoubleSliderPrivate(*this))
- {
- Q_D(ctkDoubleSlider);
- d->init();
- this->setOrientation(_orientation);
- }
- // --------------------------------------------------------------------------
- ctkDoubleSlider::~ctkDoubleSlider()
- {
- }
- // --------------------------------------------------------------------------
- void ctkDoubleSlider::setMinimum(double min)
- {
- Q_D(ctkDoubleSlider);
- this->setRange(min, qMax(min, d->Maximum));
- }
- // --------------------------------------------------------------------------
- void ctkDoubleSlider::setMaximum(double max)
- {
- Q_D(ctkDoubleSlider);
- this->setRange(qMin(d->Minimum, max), max);
- }
- // --------------------------------------------------------------------------
- void ctkDoubleSlider::setRange(double min, double max)
- {
- Q_D(ctkDoubleSlider);
- d->Minimum = min;
- d->Maximum = max;
-
- if (d->Minimum >= d->Value)
- {
- d->updateOffset(d->Minimum);
- }
- if (d->Maximum <= d->Value)
- {
- d->updateOffset(d->Maximum);
- }
- d->SettingRange = true;
- d->Slider->setRange(d->toInt(min), d->toInt(max));
- d->SettingRange = false;
- emit this->rangeChanged(d->Minimum, d->Maximum);
- /// In case QSlider::setRange(...) didn't notify the value
- /// has changed.
- this->setValue(d->Value);
- }
- // --------------------------------------------------------------------------
- double ctkDoubleSlider::minimum()const
- {
- Q_D(const ctkDoubleSlider);
- return d->Minimum;
- }
- // --------------------------------------------------------------------------
- double ctkDoubleSlider::maximum()const
- {
- Q_D(const ctkDoubleSlider);
- return d->Maximum;
- }
- // --------------------------------------------------------------------------
- double ctkDoubleSlider::sliderPosition()const
- {
- Q_D(const ctkDoubleSlider);
- return d->safeFromInt(d->Slider->sliderPosition());
- }
- // --------------------------------------------------------------------------
- void ctkDoubleSlider::setSliderPosition(double newSliderPosition)
- {
- Q_D(ctkDoubleSlider);
- d->Slider->setSliderPosition(d->toInt(newSliderPosition));
- }
- // --------------------------------------------------------------------------
- double ctkDoubleSlider::value()const
- {
- Q_D(const ctkDoubleSlider);
- return d->Value;
- }
- // --------------------------------------------------------------------------
- void ctkDoubleSlider::setValue(double newValue)
- {
- Q_D(ctkDoubleSlider);
- newValue = qBound(d->Minimum, newValue, d->Maximum);
- d->updateOffset(newValue);
- int newIntValue = d->toInt(newValue);
- if (newIntValue != d->Slider->value())
- {
- // d->Slider will emit a valueChanged signal that is connected to
- // ctkDoubleSlider::onValueChanged
- d->Slider->setValue(newIntValue);
- }
- else
- {
- double oldValue = d->Value;
- d->Value = newValue;
- // don't emit a valuechanged signal if the new value is quite
- // similar to the old value.
- if (qAbs(newValue - oldValue) > (d->SingleStep * 0.000000001))
- {
- emit this->valueChanged(newValue);
- }
- }
- }
- // --------------------------------------------------------------------------
- double ctkDoubleSlider::singleStep()const
- {
- Q_D(const ctkDoubleSlider);
- return d->SingleStep;
- }
- // --------------------------------------------------------------------------
- void ctkDoubleSlider::setSingleStep(double newStep)
- {
- Q_D(ctkDoubleSlider);
- d->SingleStep = newStep;
- d->updateOffset(d->Value);
- // update the new values of the QSlider
- bool oldBlockSignals = d->Slider->blockSignals(true);
- d->Slider->setRange(d->toInt(d->Minimum), d->toInt(d->Maximum));
- d->Slider->setValue(d->toInt(d->Value));
- d->Slider->setPageStep(d->toInt(d->PageStep));
- d->Slider->blockSignals(oldBlockSignals);
- Q_ASSERT(qFuzzyCompare(d->Value,d->safeFromInt(d->Slider->value())));
- }
- // --------------------------------------------------------------------------
- double ctkDoubleSlider::pageStep()const
- {
- Q_D(const ctkDoubleSlider);
- return d->PageStep;
- }
- // --------------------------------------------------------------------------
- void ctkDoubleSlider::setPageStep(double newStep)
- {
- Q_D(ctkDoubleSlider);
- d->PageStep = newStep;
- d->Slider->setPageStep(d->toInt(d->PageStep));
- }
- // --------------------------------------------------------------------------
- double ctkDoubleSlider::tickInterval()const
- {
- Q_D(const ctkDoubleSlider);
- // No need to apply Offset
- return d->SingleStep * d->Slider->tickInterval();
- }
- // --------------------------------------------------------------------------
- void ctkDoubleSlider::setTickInterval(double newTickInterval)
- {
- Q_D(ctkDoubleSlider);
- d->Slider->setTickInterval(d->toInt(newTickInterval));
- }
- // --------------------------------------------------------------------------
- bool ctkDoubleSlider::hasTracking()const
- {
- Q_D(const ctkDoubleSlider);
- return d->Slider->hasTracking();
- }
- // --------------------------------------------------------------------------
- void ctkDoubleSlider::setTracking(bool enable)
- {
- Q_D(ctkDoubleSlider);
- d->Slider->setTracking(enable);
- }
- // --------------------------------------------------------------------------
- void ctkDoubleSlider::triggerAction( QAbstractSlider::SliderAction action)
- {
- Q_D(ctkDoubleSlider);
- d->Slider->triggerAction(action);
- }
- // --------------------------------------------------------------------------
- Qt::Orientation ctkDoubleSlider::orientation()const
- {
- Q_D(const ctkDoubleSlider);
- return d->Slider->orientation();
- }
- // --------------------------------------------------------------------------
- void ctkDoubleSlider::setOrientation(Qt::Orientation newOrientation)
- {
- Q_D(ctkDoubleSlider);
- if (this->orientation() == newOrientation)
- {
- return;
- }
- if (!testAttribute(Qt::WA_WState_OwnSizePolicy))
- {
- QSizePolicy sp = this->sizePolicy();
- sp.transpose();
- this->setSizePolicy(sp);
- this->setAttribute(Qt::WA_WState_OwnSizePolicy, false);
- }
- // d->Slider will take care of calling updateGeometry
- d->Slider->setOrientation(newOrientation);
- }
- // --------------------------------------------------------------------------
- QString ctkDoubleSlider::handleToolTip()const
- {
- Q_D(const ctkDoubleSlider);
- return d->HandleToolTip;
- }
- // --------------------------------------------------------------------------
- void ctkDoubleSlider::setHandleToolTip(const QString& toolTip)
- {
- Q_D(ctkDoubleSlider);
- d->HandleToolTip = toolTip;
- }
- // --------------------------------------------------------------------------
- void ctkDoubleSlider::onValueChanged(int newValue)
- {
- Q_D(ctkDoubleSlider);
- double doubleNewValue = d->safeFromInt(newValue);
- /*
- qDebug() << "onValueChanged: " << newValue << "->"<< d->fromInt(newValue+d->Offset)
- << " old: " << d->Value << "->" << d->toInt(d->Value)
- << "offset:" << d->Offset << doubleNewValue;
- */
- if (d->Value == doubleNewValue)
- {
- return;
- }
- d->Value = doubleNewValue;
- emit this->valueChanged(d->Value);
- }
- // --------------------------------------------------------------------------
- void ctkDoubleSlider::onSliderMoved(int newPosition)
- {
- Q_D(const ctkDoubleSlider);
- emit this->sliderMoved(d->safeFromInt(newPosition));
- }
- // --------------------------------------------------------------------------
- void ctkDoubleSlider::onRangeChanged(int min, int max)
- {
- Q_D(const ctkDoubleSlider);
- if (!d->SettingRange)
- {
- this->setRange(d->fromInt(min), d->fromInt(max));
- }
- }
- // --------------------------------------------------------------------------
- bool ctkDoubleSlider::eventFilter(QObject* watched, QEvent* event)
- {
- Q_D(ctkDoubleSlider);
- if (watched == d->Slider)
- {
- switch(event->type())
- {
- case QEvent::ToolTip:
- {
- QHelpEvent* helpEvent = static_cast<QHelpEvent*>(event);
- QStyleOptionSlider opt;
- d->Slider->initStyleOption(&opt);
- QStyle::SubControl hoveredControl =
- d->Slider->style()->hitTestComplexControl(
- QStyle::CC_Slider, &opt, helpEvent->pos(), this);
- if (!d->HandleToolTip.isEmpty() &&
- hoveredControl == QStyle::SC_SliderHandle)
- {
- QToolTip::showText(helpEvent->globalPos(), d->HandleToolTip.arg(this->value()));
- event->accept();
- return true;
- }
- }
- default:
- break;
- }
- }
- return this->Superclass::eventFilter(watched, event);
- }
|