| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674 | /*=========================================================================  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 rangevoid ctkRangeSliderPrivate::drawMinimumSlider( QStylePainter* painter ) const{  CTK_P(const ctkRangeSlider);  QStyleOptionSlider option;  p->initMinimumSliderStyleOption( &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 rangevoid ctkRangeSliderPrivate::drawMaximumSlider( QStylePainter* painter ) const{  CTK_P(const ctkRangeSlider);  QStyleOptionSlider option;  p->initMaximumSliderStyleOption( &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);}// --------------------------------------------------------------------------// Rendervoid ctkRangeSlider::paintEvent( QPaintEvent* ){  CTK_D(ctkRangeSlider);  QStyleOptionSlider option;  this->initStyleOption(&option);  QStylePainter painter(this);  option.subControls = QStyle::SC_SliderGroove;  option.sliderPosition = this->minimum(); // don't highlight the 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 eventsvoid 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 eventsvoid 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 mouseEventsvoid ctkRangeSlider::mouseReleaseEvent(QMouseEvent* mouseEvent){  CTK_D(ctkRangeSlider);  this->QSlider::mouseReleaseEvent(mouseEvent);  setSliderDown(false);  d->m_SelectedHandles = 0;  this->update();}// --------------------------------------------------------------------------void ctkRangeSlider::initMinimumSliderStyleOption(QStyleOptionSlider* option) const{  this->initStyleOption(option);}// --------------------------------------------------------------------------void ctkRangeSlider::initMaximumSliderStyleOption(QStyleOptionSlider* option) const{  this->initStyleOption(option);}
 |