123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663 |
- /*=========================================================================
- Library: CTK
-
- Copyright (c) 2010 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.commontk.org/LICENSE
- 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 <QMouseEvent>
- #include <QKeyEvent>
- #include <QStyleOptionSlider>
- #include <QApplication>
- #include <QStylePainter>
- #include <QStyle>
- // CTK includes
- #include "ctkRangeSlider.h"
- class ctkRangeSliderPrivate:public ctkPrivate<ctkRangeSlider>
- {
- public:
- ctkRangeSliderPrivate();
- void init();
-
- // Description:
- // Copied verbatim from QSliderPrivate class (see QSlider.cpp)
- int pixelPosToRangeValue(int pos) const;
- int pixelPosFromRangeValue(int val) const;
- // Description:
- // Draw the bottom and top sliders.
- void drawMinimumSlider( QStylePainter* painter ) const;
- void drawMaximumSlider( QStylePainter* painter ) const;
-
- // Description:
- // End points of the range on the Model
- int m_MaximumValue;
- int m_MinimumValue;
- // Description:
- // End points of the range on the GUI. This is synced with the model.
- int m_MaximumPosition;
- int m_MinimumPosition;
- // Description:
- // Controls selected ?
- QStyle::SubControl m_MinimumSliderSelected;
- QStyle::SubControl m_MaximumSliderSelected;
- // Description:
- // See QSliderPrivate::clickOffset.
- // Overrides this ivar
- int m_SubclassClickOffset;
-
- // Description:
- // See QSliderPrivate::position
- // Overrides this ivar.
- int m_SubclassPosition;
- int m_SubclassWidth;
- // Description:
- // Boolean indicates the selected handle
- // True for the minimum range handle, false for the maximum range handle
- enum Handle {
- NoHandle = 0x0,
- MinimumHandle = 0x1,
- MaximumHandle = 0x2
- };
- Q_DECLARE_FLAGS(Handles, Handle);
- ctkRangeSliderPrivate::Handles m_SelectedHandles;
- };
- // --------------------------------------------------------------------------
- ctkRangeSliderPrivate::ctkRangeSliderPrivate()
- {
- this->m_MinimumValue = 0;
- this->m_MaximumValue = 100;
- this->m_MinimumPosition = 0;
- this->m_MaximumPosition = 100;
- this->m_MinimumSliderSelected = QStyle::SC_None;
- this->m_MaximumSliderSelected = QStyle::SC_None;
- this->m_SubclassClickOffset = 0;
- this->m_SubclassPosition = 0;
- this->m_SubclassWidth = 0;
- this->m_SelectedHandles = 0;
- }
- // --------------------------------------------------------------------------
- void ctkRangeSliderPrivate::init()
- {
- CTK_P(ctkRangeSlider);
- this->m_MinimumValue = p->minimum();
- this->m_MaximumValue = p->maximum();
- this->m_MinimumPosition = p->minimum();
- this->m_MaximumPosition = p->maximum();
- p->connect(p, SIGNAL(rangeChanged(int, int)), p, SLOT(onRangeChanged(int, int)));
- }
- // --------------------------------------------------------------------------
- // Copied verbatim from QSliderPrivate::pixelPosToRangeValue. See QSlider.cpp
- //
- int ctkRangeSliderPrivate::pixelPosToRangeValue( int pos ) const
- {
- CTK_P(const ctkRangeSlider);
- QStyleOptionSlider option;
- p->initStyleOption( &option );
- QRect gr = p->style()->subControlRect( QStyle::CC_Slider,
- &option,
- QStyle::SC_SliderGroove,
- p );
- QRect sr = p->style()->subControlRect( QStyle::CC_Slider,
- &option,
- QStyle::SC_SliderHandle,
- p );
- int sliderMin, sliderMax, sliderLength;
- if (option.orientation == Qt::Horizontal)
- {
- sliderLength = sr.width();
- sliderMin = gr.x();
- sliderMax = gr.right() - sliderLength + 1;
- }
- else
- {
- sliderLength = sr.height();
- sliderMin = gr.y();
- sliderMax = gr.bottom() - sliderLength + 1;
- }
- return QStyle::sliderValueFromPosition( p->minimum(),
- p->maximum(),
- pos - sliderMin,
- sliderMax - sliderMin,
- option.upsideDown );
- }
- //---------------------------------------------------------------------------
- int ctkRangeSliderPrivate::pixelPosFromRangeValue( int val ) const
- {
- CTK_P(const ctkRangeSlider);
- QStyleOptionSlider option;
- p->initStyleOption( &option );
- QRect gr = p->style()->subControlRect( QStyle::CC_Slider,
- &option,
- QStyle::SC_SliderGroove,
- p );
- QRect sr = p->style()->subControlRect( QStyle::CC_Slider,
- &option,
- QStyle::SC_SliderHandle,
- p );
- int sliderMin, sliderMax, sliderLength;
- if (option.orientation == Qt::Horizontal)
- {
- sliderLength = sr.width();
- sliderMin = gr.x();
- sliderMax = gr.right() - sliderLength + 1;
- }
- else
- {
- sliderLength = sr.height();
- sliderMin = gr.y();
- sliderMax = gr.bottom() - sliderLength + 1;
- }
- return QStyle::sliderPositionFromValue( p->minimum(),
- p->maximum(),
- val,
- sliderMax - sliderMin,
- option.upsideDown ) + sliderMin;
- }
- //---------------------------------------------------------------------------
- // Draw slider at the bottom end of the range
- void ctkRangeSliderPrivate::drawMinimumSlider( QStylePainter* painter ) const
- {
- CTK_P(const ctkRangeSlider);
- QStyleOptionSlider option;
- p->initStyleOption( &option );
- option.subControls = QStyle::SC_SliderHandle;
- option.sliderValue = m_MinimumValue;
- option.sliderPosition = m_MinimumPosition;
- if (this->m_SelectedHandles & MinimumHandle)
- {
- option.activeSubControls = QStyle::SC_SliderHandle;
- option.state |= QStyle::State_Sunken;
- }
- painter->drawComplexControl(QStyle::CC_Slider, option);
- }
- //---------------------------------------------------------------------------
- // Draw slider at the top end of the range
- void ctkRangeSliderPrivate::drawMaximumSlider( QStylePainter* painter ) const
- {
- CTK_P(const ctkRangeSlider);
- QStyleOptionSlider option;
- p->Superclass::initStyleOption( &option );
- option.subControls = QStyle::SC_SliderHandle;
- option.sliderValue = m_MaximumValue;
- option.sliderPosition = m_MaximumPosition;
- if (this->m_SelectedHandles & MaximumHandle)
- {
- option.activeSubControls = QStyle::SC_SliderHandle;
- option.state |= QStyle::State_Sunken;
- }
- painter->drawComplexControl(QStyle::CC_Slider, option);
- }
- // --------------------------------------------------------------------------
- ctkRangeSlider::ctkRangeSlider(QWidget* parent)
- : QSlider(parent)
- {
- CTK_INIT_PRIVATE(ctkRangeSlider);
- ctk_d()->init();
- }
- // --------------------------------------------------------------------------
- ctkRangeSlider::ctkRangeSlider( Qt::Orientation o,
- QWidget* parentObject )
- :QSlider(o, parentObject)
- {
- CTK_INIT_PRIVATE(ctkRangeSlider);
- ctk_d()->init();
- }
- // --------------------------------------------------------------------------
- ctkRangeSlider::~ctkRangeSlider()
- {
- }
- // --------------------------------------------------------------------------
- int ctkRangeSlider::minimumValue() const
- {
- CTK_D(const ctkRangeSlider);
- return d->m_MinimumValue;
- }
- // --------------------------------------------------------------------------
- void ctkRangeSlider::setMinimumValue( int min )
- {
- CTK_D(ctkRangeSlider);
- this->setValues( min, qMax(d->m_MaximumValue,min) );
- }
- // --------------------------------------------------------------------------
- int ctkRangeSlider::maximumValue() const
- {
- CTK_D(const ctkRangeSlider);
- return d->m_MaximumValue;
- }
- // --------------------------------------------------------------------------
- void ctkRangeSlider::setMaximumValue( int max )
- {
- CTK_D(ctkRangeSlider);
- this->setValues( qMin(d->m_MinimumValue, max), max );
- }
- // --------------------------------------------------------------------------
- void ctkRangeSlider::setValues(int l, int u)
- {
- CTK_D(ctkRangeSlider);
- const int minimumValue =
- qBound(this->minimum(), qMin(l,u), this->maximum());
- const int maximumValue =
- qBound(this->minimum(), qMax(l,u), this->maximum());
- bool emitMinValChanged = (minimumValue != d->m_MinimumValue);
- bool emitMaxValChanged = (maximumValue != d->m_MaximumValue);
-
- d->m_MinimumValue = minimumValue;
- d->m_MaximumValue = maximumValue;
-
- bool emitMinPosChanged =
- (minimumValue != d->m_MinimumPosition);
- bool emitMaxPosChanged =
- (maximumValue != d->m_MaximumPosition);
- d->m_MinimumPosition = minimumValue;
- d->m_MaximumPosition = maximumValue;
-
- if (isSliderDown())
- {
- if (emitMinPosChanged || emitMaxPosChanged)
- {
- emit positionsChanged(minimumValue, maximumValue);
- }
- if (emitMinPosChanged)
- {
- emit minimumPositionChanged(minimumValue);
- }
- if (emitMaxPosChanged)
- {
- emit maximumPositionChanged(maximumValue);
- }
- }
- if (emitMinValChanged || emitMaxValChanged)
- {
- emit valuesChanged(d->m_MinimumValue,
- d->m_MaximumValue);
- }
- if (emitMinValChanged)
- {
- emit minimumValueChanged(minimumValue);
- }
- if (emitMaxValChanged)
- {
- emit maximumValueChanged(maximumValue);
- }
- if (emitMinPosChanged || emitMaxPosChanged ||
- emitMinValChanged || emitMaxValChanged)
- {
- this->update();
- }
- }
- // --------------------------------------------------------------------------
- int ctkRangeSlider::minimumPosition() const
- {
- CTK_D(const ctkRangeSlider);
- return d->m_MinimumPosition;
- }
- // --------------------------------------------------------------------------
- int ctkRangeSlider::maximumPosition() const
- {
- CTK_D(const ctkRangeSlider);
- return d->m_MaximumPosition;
- }
- // --------------------------------------------------------------------------
- void ctkRangeSlider::setMinimumPosition(int l)
- {
- CTK_D(const ctkRangeSlider);
- this->setPositions(l, qMax(l, d->m_MaximumPosition));
- }
- // --------------------------------------------------------------------------
- void ctkRangeSlider::setMaximumPosition(int u)
- {
- CTK_D(const ctkRangeSlider);
- this->setPositions(qMin(d->m_MinimumPosition, u), u);
- }
- // --------------------------------------------------------------------------
- void ctkRangeSlider::setPositions(int min, int max)
- {
- CTK_D(ctkRangeSlider);
- const int minPosition =
- qBound(this->minimum(), qMin(min, max), this->maximum());
- const int maxPosition =
- qBound(this->minimum(), qMax(min, max), this->maximum());
- bool emitMinPosChanged = (minPosition != d->m_MinimumPosition);
- bool emitMaxPosChanged = (maxPosition != d->m_MaximumPosition);
-
- if (!emitMinPosChanged && !emitMaxPosChanged)
- {
- return;
- }
- d->m_MinimumPosition = minPosition;
- d->m_MaximumPosition = maxPosition;
- if (!this->hasTracking())
- {
- this->update();
- }
- if (isSliderDown())
- {
- if (emitMinPosChanged)
- {
- emit minimumPositionChanged(d->m_MinimumPosition);
- }
- if (emitMaxPosChanged)
- {
- emit maximumPositionChanged(d->m_MaximumPosition);
- }
- if (emitMinPosChanged || emitMaxPosChanged)
- {
- emit positionsChanged(d->m_MinimumPosition, d->m_MaximumPosition);
- }
- }
- if (this->hasTracking())
- {
- this->triggerAction(SliderMove);
- this->setValues(d->m_MinimumPosition, d->m_MaximumPosition);
- }
- }
- // --------------------------------------------------------------------------
- void ctkRangeSlider::onRangeChanged(int minimum, int maximum)
- {
- Q_UNUSED(minimum);
- Q_UNUSED(maximum);
- CTK_D(ctkRangeSlider);
- this->setValues(d->m_MinimumValue, d->m_MaximumValue);
- }
- // --------------------------------------------------------------------------
- // Render
- void ctkRangeSlider::paintEvent( QPaintEvent* )
- {
- CTK_D(ctkRangeSlider);
- QStyleOptionSlider option;
- this->initStyleOption(&option);
- QStylePainter painter(this);
- option.subControls = QStyle::SC_SliderGroove;
- painter.drawComplexControl(QStyle::CC_Slider, option);
- option.sliderPosition = d->m_MinimumPosition;
- const QRect lr = style()->subControlRect( QStyle::CC_Slider,
- &option,
- QStyle::SC_SliderHandle,
- this);
- option.sliderPosition = d->m_MaximumPosition;
- const QRect ur = style()->subControlRect( QStyle::CC_Slider,
- &option,
- QStyle::SC_SliderHandle,
- this);
- QRect sr = style()->subControlRect( QStyle::CC_Slider,
- &option,
- QStyle::SC_SliderGroove,
- this);
- QRect rangeBox = QRect(
- QPoint( qMin( lr.center().x(), ur.center().x() ), sr.center().y() - 2),
- QPoint(qMax( lr.center().x(), ur.center().x() ), sr.center().y() + 1));
- // -----------------------------
- // Render the range
- //
- QRect groove = this->style()->subControlRect( QStyle::CC_Slider,
- &option,
- QStyle::SC_SliderGroove,
- this );
- groove.adjust(0, 0, -1, 0);
- // Create default colors based on the transfer function.
- //
- QColor highlight = this->palette().color(QPalette::Normal, QPalette::Highlight);
- QLinearGradient gradient( groove.center().x(), groove.top(),
- groove.center().x(), groove.bottom());
- // TODO: Set this based on the supplied transfer function
- QColor l = Qt::darkGray;
- QColor u = Qt::black;
- gradient.setColorAt(0, highlight.darker(120));
- gradient.setColorAt(1, highlight.lighter(160));
- painter.setPen(QPen(highlight.darker(150), 0));
- painter.setBrush(gradient);
- painter.drawRect( rangeBox.intersected(groove) );
- // -----------------------------------
- // Render the sliders
- //
- if (d->m_SelectedHandles & ctkRangeSliderPrivate::MinimumHandle)
- {
- d->drawMaximumSlider( &painter );
- d->drawMinimumSlider( &painter );
- }
- else
- {
- d->drawMinimumSlider( &painter );
- d->drawMaximumSlider( &painter );
- }
- }
- // --------------------------------------------------------------------------
- // Standard Qt UI events
- void ctkRangeSlider::mousePressEvent(QMouseEvent* mouseEvent)
- {
- CTK_D(ctkRangeSlider);
- if (minimum() == maximum() || (mouseEvent->buttons() ^ mouseEvent->button()))
- {
- mouseEvent->ignore();
- return;
- }
-
- QStyleOptionSlider option;
- this->initStyleOption( &option );
- // Check if the first handle is pressed
-
- option.sliderPosition = d->m_MinimumPosition;
- option.sliderValue = d->m_MinimumValue;
- QStyle::SubControl control;
- control = this->style()->hitTestComplexControl( QStyle::CC_Slider,
- &option,
- mouseEvent->pos(),
- this);
- const QRect lr = this->style()->subControlRect( QStyle::CC_Slider,
- &option,
- QStyle::SC_SliderHandle,
- this);
- if (control == QStyle::SC_SliderHandle)
- {
- d->m_SubclassPosition = d->m_MinimumPosition;
- // save the position of the mouse inside the handle for later
- d->m_SubclassClickOffset = mouseEvent->x() - lr.left();
- this->setSliderDown(true);
- if (d->m_SelectedHandles != ctkRangeSliderPrivate::MinimumHandle)
- {
- d->m_SelectedHandles = ctkRangeSliderPrivate::MinimumHandle;
- this->update(lr);
- }
- // Accept the mouseEvent
- mouseEvent->accept();
- return;
- }
- // The user didn't press on the minimum handle,
- // Check if the other handle is pressed
- option.sliderPosition = d->m_MaximumPosition;
- option.sliderValue = d->m_MaximumValue;
- control = this->style()->hitTestComplexControl( QStyle::CC_Slider,
- &option,
- mouseEvent->pos(),
- this);
- const QRect ur = this->style()->subControlRect( QStyle::CC_Slider,
- &option,
- QStyle::SC_SliderHandle,
- this);
- if (control == QStyle::SC_SliderHandle)
- {
- d->m_SubclassPosition = d->m_MaximumPosition;
- // save the position of the mouse inside the handle for later
- d->m_SubclassClickOffset = mouseEvent->x() - ur.left();
- this->setSliderDown(true);
- if (d->m_SelectedHandles != ctkRangeSliderPrivate::MaximumHandle)
- {
- d->m_SelectedHandles = ctkRangeSliderPrivate::MaximumHandle;
- this->update(ur);
- }
- // Accept the mouseEvent
- mouseEvent->accept();
- return;
- }
- // if we are here, no handles have been pressed
- // Check if we pressed on the groove between the 2 handles
-
- control = this->style()->hitTestComplexControl( QStyle::CC_Slider,
- &option,
- mouseEvent->pos(),
- this);
- QRect sr = style()->subControlRect( QStyle::CC_Slider,
- &option,
- QStyle::SC_SliderGroove,
- this);
- if (control == QStyle::SC_SliderGroove &&
- mouseEvent->x() > lr.center().x() &&
- mouseEvent->x() < ur.center().x())
- {
- // warning lost of precision it might be fatal
- d->m_SubclassPosition = (d->m_MinimumPosition + d->m_MaximumPosition) / 2.;
- d->m_SubclassClickOffset = mouseEvent->x() - d->pixelPosFromRangeValue(d->m_SubclassPosition);
- d->m_SubclassWidth = (d->m_MaximumPosition - d->m_MinimumPosition) / 2;
- qMax(d->m_SubclassPosition - d->m_MinimumPosition, d->m_MaximumPosition - d->m_SubclassPosition);
- this->setSliderDown(true);
- if (!(d->m_SelectedHandles & QFlags<ctkRangeSliderPrivate::Handle>(
- ctkRangeSliderPrivate::MinimumHandle)) ||
- !(d->m_SelectedHandles & QFlags<ctkRangeSliderPrivate::Handle>(ctkRangeSliderPrivate::MaximumHandle)))
- {
- d->m_SelectedHandles =
- QFlags<ctkRangeSliderPrivate::Handle>(ctkRangeSliderPrivate::MinimumHandle) |
- QFlags<ctkRangeSliderPrivate::Handle>(ctkRangeSliderPrivate::MaximumHandle);
- this->update(lr.united(ur).united(sr));
- }
- mouseEvent->accept();
- return;
- }
- mouseEvent->ignore();
- }
- // --------------------------------------------------------------------------
- // Standard Qt UI events
- void ctkRangeSlider::mouseMoveEvent(QMouseEvent* mouseEvent)
- {
- CTK_D(ctkRangeSlider);
- if (!d->m_SelectedHandles)
- {
- mouseEvent->ignore();
- return;
- }
- QStyleOptionSlider option;
- this->initStyleOption(&option);
- const int m = style()->pixelMetric( QStyle::PM_MaximumDragDistance, &option, this );
- int newPosition = d->pixelPosToRangeValue(
- mouseEvent->x() - d->m_SubclassClickOffset);
- if (m >= 0)
- {
- const QRect r = rect().adjusted(-m, -m, m, m);
- if (!r.contains(mouseEvent->pos()))
- {
- newPosition = d->m_SubclassPosition;
- }
- }
- if (d->m_SelectedHandles == ctkRangeSliderPrivate::MinimumHandle)
- {
- this->setMinimumPosition(qMin(newPosition,d->m_MaximumPosition));
- }
- else if (d->m_SelectedHandles == ctkRangeSliderPrivate::MaximumHandle)
- {
- this->setMaximumPosition(qMax(d->m_MinimumPosition, newPosition));
- }
- else if (d->m_SelectedHandles & ctkRangeSliderPrivate::MinimumHandle &&
- d->m_SelectedHandles & ctkRangeSliderPrivate::MaximumHandle)
- {
- this->setPositions(newPosition - d->m_SubclassWidth, newPosition + d->m_SubclassWidth );
- }
- mouseEvent->accept();
- }
- // --------------------------------------------------------------------------
- // Standard Qt UI mouseEvents
- void ctkRangeSlider::mouseReleaseEvent(QMouseEvent* mouseEvent)
- {
- CTK_D(ctkRangeSlider);
- this->QSlider::mouseReleaseEvent(mouseEvent);
- setSliderDown(false);
- d->m_SelectedHandles = 0;
- this->update();
- }
|