diff options
Diffstat (limited to 'src/widgets/qscrollbar.cpp')
-rw-r--r-- | src/widgets/qscrollbar.cpp | 1072 |
1 files changed, 1072 insertions, 0 deletions
diff --git a/src/widgets/qscrollbar.cpp b/src/widgets/qscrollbar.cpp new file mode 100644 index 0000000..9a5e8f4 --- /dev/null +++ b/src/widgets/qscrollbar.cpp @@ -0,0 +1,1072 @@ +/**************************************************************************** +** +** Implementation of QScrollBar class +** +** Created : 940427 +** +** Copyright (C) 1992-2008 Trolltech ASA. All rights reserved. +** +** This file is part of the widgets module of the Qt GUI Toolkit. +** +** This file may be used under the terms of the GNU General +** Public License versions 2.0 or 3.0 as published by the Free +** Software Foundation and appearing in the files LICENSE.GPL2 +** and LICENSE.GPL3 included in the packaging of this file. +** Alternatively you may (at your option) use any later version +** of the GNU General Public License if such license has been +** publicly approved by Trolltech ASA (or its successors, if any) +** and the KDE Free Qt Foundation. +** +** Please review the following information to ensure GNU General +** Public Licensing requirements will be met: +** http://trolltech.com/products/qt/licenses/licensing/opensource/. +** If you are unsure which license is appropriate for your use, please +** review the following information: +** http://trolltech.com/products/qt/licenses/licensing/licensingoverview +** or contact the sales department at sales@trolltech.com. +** +** This file may be used under the terms of the Q Public License as +** defined by Trolltech ASA and appearing in the file LICENSE.QPL +** included in the packaging of this file. Licensees holding valid Qt +** Commercial licenses may use this file in accordance with the Qt +** Commercial License Agreement provided with the Software. +** +** This file is provided "AS IS" with NO WARRANTY OF ANY KIND, +** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR +** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted +** herein. +** +**********************************************************************/ + +#include "qscrollbar.h" +#ifndef QT_NO_SCROLLBAR +#include "qpainter.h" +#include "qbitmap.h" +#include "qapplication.h" +#include "qtimer.h" +#include "qstyle.h" +#ifndef QT_NO_CURSOR +#include <qcursor.h> +#endif +#if defined(QT_ACCESSIBILITY_SUPPORT) +#include "qaccessible.h" +#endif +#include <limits.h> + +/*! + \class QScrollBar + \brief The QScrollBar widget provides a vertical or horizontal scroll bar. + + \ingroup basic + + A scroll bar allows the user to control a value within a + program-definable range and gives users a visible indication of + the current value of a \link QRangeControl range control \endlink. + + Scroll bars include four separate controls: + + \list + + \i The \e line-up and \e line-down controls are little buttons + which the user can use to move one "line" up or down. The meaning + of line is configurable. In editors and list boxes it means one + line of text; in an image viewer it might mean 20 pixels. + + \i The \e slider is the handle that indicates the current value of + the scroll bar, which the user can drag to change the value. This + part of the scroll bar is sometimes called the "thumb". + + \i The \e page-up/page-down control is the area on which the + slider slides (the scroll bar's background). Clicking here moves + the scroll bar towards the click. The meaning of "page" is also + configurable: in editors and list boxes it means as many lines as + there is space for in the widget. + + \endlist + + QScrollBar has very few of its own functions; it mostly relies on + QRangeControl. The most useful functions are setValue() to set the + scroll bar directly to some value; addPage(), addLine(), + subtractPage(), and subtractLine() to simulate the effects of + clicking (useful for accelerator keys); setSteps() to define the + values of pageStep() and lineStep(); and setRange() to set the + minValue() and maxValue() of the scroll bar. QScrollBar has a + convenience constructor with which you can set most of these + properties. + + Some GUI styles (for example, the Windows and Motif styles + provided with Qt), also use the pageStep() value to calculate the + size of the slider. + + In addition to the access functions from QRangeControl, QScrollBar + provides a comprehensive set of signals: + \table + \header \i Signal \i Emitted when + \row \i \l valueChanged() + \i the scroll bar's value has changed. The tracking() + determines whether this signal is emitted during user + interaction. + \row \i \l sliderPressed() + \i the user starts to drag the slider. + \row \i \l sliderMoved() + \i the user drags the slider. + \row \i \l sliderReleased() + \i the user releases the slider. + \row \i \l nextLine() + \i the scroll bar has moved one line down or right. Line is + defined in QRangeControl. + \row \i \l prevLine() + \i the scroll bar has moved one line up or left. + \row \i \l nextPage() + \i the scroll bar has moved one page down or right. + \row \i \l prevPage() + \i the scroll bar has moved one page up or left. + \endtable + + QScrollBar only provides integer ranges. Note that although + QScrollBar handles very large numbers, scroll bars on current + screens cannot usefully control ranges above about 100,000 pixels. + Beyond that, it becomes difficult for the user to control the + scroll bar using either the keyboard or the mouse. + + A scroll bar can be controlled by the keyboard, but it has a + default focusPolicy() of \c NoFocus. Use setFocusPolicy() to + enable keyboard focus. See keyPressEvent() for a list of key + bindings. + + If you need to add scroll bars to an interface, consider using the + QScrollView class, which encapsulates the common uses for scroll + bars. + + <img src=qscrbar-m.png> <img src=qscrbar-w.png> + + \sa QSlider QSpinBox QScrollView + \link guibooks.html#fowler GUI Design Handbook: Scroll Bar\endlink +*/ + + +/*! + \fn void QScrollBar::valueChanged( int value ) + + This signal is emitted when the scroll bar value has changed, with + the new scroll bar \a value as an argument. +*/ + +/*! + \fn void QScrollBar::sliderPressed() + + This signal is emitted when the user presses the slider with the + mouse. +*/ + +/*! + \fn void QScrollBar::sliderMoved( int value ) + + This signal is emitted when the slider is dragged by the user, with + the new scroll bar \a value as an argument. + + This signal is emitted even when tracking is turned off. + + \sa tracking() valueChanged() nextLine() prevLine() nextPage() + prevPage() +*/ + +/*! + \fn void QScrollBar::sliderReleased() + + This signal is emitted when the user releases the slider with the + mouse. +*/ + +/*! + \fn void QScrollBar::nextLine() + + This signal is emitted when the scroll bar scrolls one line down + or right. +*/ + +/*! + \fn void QScrollBar::prevLine() + + This signal is emitted when the scroll bar scrolls one line up or + left. +*/ + +/*! + \fn void QScrollBar::nextPage() + + This signal is emitted when the scroll bar scrolls one page down + or right. +*/ + +/*! + \fn void QScrollBar::prevPage() + + This signal is emitted when the scroll bar scrolls one page up or + left. +*/ + + + +static const int thresholdTime = 500; +static const int repeatTime = 50; + +#define HORIZONTAL (orientation() == Horizontal) +#define VERTICAL !HORIZONTAL +#define MOTIF_BORDER 2 +#define SLIDER_MIN 9 + + +/*! + Constructs a vertical scroll bar. + + The \a parent and \a name arguments are sent on to the QWidget + constructor. + + The \c minValue defaults to 0, the \c maxValue to 99, with a \c + lineStep size of 1 and a \c pageStep size of 10, and an initial + \c value of 0. +*/ + +QScrollBar::QScrollBar( QWidget *parent, const char *name ) + : QWidget( parent, name ), orient( Vertical ) +{ + init(); +} + +/*! + Constructs a scroll bar. + + The \a orientation must be \c Qt::Vertical or \c Qt::Horizontal. + + The \a parent and \a name arguments are sent on to the QWidget + constructor. + + The \c minValue defaults to 0, the \c maxValue to 99, with a \c + lineStep size of 1 and a \c pageStep size of 10, and an initial + \c value of 0. +*/ + +QScrollBar::QScrollBar( Orientation orientation, QWidget *parent, + const char *name ) + : QWidget( parent, name ), orient( orientation ) +{ + init(); +} + +/*! + Constructs a scroll bar whose value can never be smaller than \a + minValue or greater than \a maxValue, whose line step size is \a + lineStep and page step size is \a pageStep and whose value is + initially \a value (which is guaranteed to be in range using + bound()). + + If \a orientation is \c Vertical the scroll bar is vertical and if + it is \c Horizontal the scroll bar is horizontal. + + The \a parent and \a name arguments are sent on to the QWidget + constructor. +*/ + +QScrollBar::QScrollBar( int minValue, int maxValue, int lineStep, int pageStep, + int value, Orientation orientation, + QWidget *parent, const char *name ) + : QWidget( parent, name ), + QRangeControl( minValue, maxValue, lineStep, pageStep, value ), + orient( orientation ) +{ + init(); +} + +/*! + Destructor. +*/ +QScrollBar::~QScrollBar() +{ +} + +void QScrollBar::init() +{ + track = TRUE; + sliderPos = 0; + pressedControl = QStyle::SC_None; + clickedAt = FALSE; + setFocusPolicy( NoFocus ); + + repeater = 0; + + setBackgroundMode((Qt::BackgroundMode) + style().styleHint(QStyle::SH_ScrollBar_BackgroundMode)); + + QSizePolicy sp( QSizePolicy::Minimum, QSizePolicy::Fixed ); + if ( orient == Vertical ) + sp.transpose(); + setSizePolicy( sp ); + clearWState( WState_OwnSizePolicy ); +} + + +/*! + \property QScrollBar::orientation + \brief the orientation of the scroll bar + + The orientation must be \l Qt::Vertical (the default) or \l + Qt::Horizontal. +*/ + +void QScrollBar::setOrientation( Orientation orientation ) +{ + if ( orientation == orient ) + return; + if ( !testWState( WState_OwnSizePolicy ) ) { + QSizePolicy sp = sizePolicy(); + sp.transpose(); + setSizePolicy( sp ); + clearWState( WState_OwnSizePolicy ); + } + + orient = orientation; + + positionSliderFromValue(); + update(); + updateGeometry(); +} + +/*! + \property QScrollBar::tracking + \brief whether scroll bar tracking is enabled + + If tracking is enabled (the default), the scroll bar emits the + valueChanged() signal while the slider is being dragged. If + tracking is disabled, the scroll bar emits the valueChanged() + signal only when the user releases the mouse button after moving + the slider. +*/ + + +/*! + \property QScrollBar::draggingSlider + \brief whether the user has clicked the mouse on the slider and is currently dragging it +*/ + +bool QScrollBar::draggingSlider() const +{ + return pressedControl == QStyle::SC_ScrollBarSlider; +} + + +/*! + Reimplements the virtual function QWidget::setPalette(). + + Sets the background color to the mid color for Motif style scroll + bars using palette \a p. +*/ + +void QScrollBar::setPalette( const QPalette &p ) +{ + QWidget::setPalette( p ); + setBackgroundMode((Qt::BackgroundMode) + style().styleHint(QStyle::SH_ScrollBar_BackgroundMode)); +} + + +/*! \reimp */ +QSize QScrollBar::sizeHint() const +{ + constPolish(); + int sbextent = style().pixelMetric(QStyle::PM_ScrollBarExtent, this); + + if ( orient == Horizontal ) { + return QSize( 30, sbextent ); + } else { + return QSize( sbextent, 30 ); + } +} + +/*! \fn void QScrollBar::setSizePolicy( QSizePolicy::SizeType, QSizePolicy::SizeType, bool ) + \reimp +*/ + +/*! \reimp */ +void QScrollBar::setSizePolicy( QSizePolicy sp ) +{ + //## remove 4.0 + QWidget::setSizePolicy( sp ); +} + +/*! + \internal + Implements the virtual QRangeControl function. +*/ + +void QScrollBar::valueChange() +{ + int tmp = sliderPos; + positionSliderFromValue(); + if ( tmp != sliderPos && isVisible() ) + drawControls(QStyle::SC_ScrollBarAddPage | + QStyle::SC_ScrollBarSubPage | + QStyle::SC_ScrollBarSlider, + pressedControl ); + emit valueChanged(value()); +#if defined(QT_ACCESSIBILITY_SUPPORT) + QAccessible::updateAccessibility( this, 0, QAccessible::ValueChanged ); +#endif +} + +/*! + \internal + Implements the virtual QRangeControl function. +*/ + +void QScrollBar::stepChange() +{ + rangeChange(); +} + +/*! + \internal + Implements the virtual QRangeControl function. +*/ + +void QScrollBar::rangeChange() +{ + positionSliderFromValue(); + + if ( isVisible() ) + drawControls(QStyle::SC_ScrollBarAddLine | + QStyle::SC_ScrollBarSubLine | + QStyle::SC_ScrollBarAddPage | + QStyle::SC_ScrollBarSubPage | + QStyle::SC_ScrollBarFirst | + QStyle::SC_ScrollBarLast | + QStyle::SC_ScrollBarSlider, + pressedControl ); +} + + +/*! + Handles timer events for the scroll bar. +*/ + +void QScrollBar::doAutoRepeat() +{ + bool sendRepeat = clickedAt; +#if !defined( QT_NO_CURSOR ) && !defined( QT_NO_STYLE ) + if(sendRepeat && (pressedControl == QStyle::SC_ScrollBarAddPage || + pressedControl == QStyle::SC_ScrollBarSubPage) && + style().styleHint(QStyle::SH_ScrollBar_StopMouseOverSlider, this) && + style().querySubControl(QStyle::CC_ScrollBar, this, + mapFromGlobal(QCursor::pos()) ) == QStyle::SC_ScrollBarSlider) + sendRepeat = FALSE; +#endif + if ( sendRepeat ){ + if ( repeater ) + repeater->changeInterval( repeatTime ); + action( (QStyle::SubControl) pressedControl ); + QApplication::syncX(); + } else { + stopAutoRepeat(); + } +} + + +/*! + Starts the auto-repeat logic. Some time after this function is + called, the auto-repeat starts taking effect and from then on + repeats until stopAutoRepeat() is called. +*/ + +void QScrollBar::startAutoRepeat() +{ + if ( !repeater ) { + repeater = new QTimer( this, "auto-repeat timer" ); + connect( repeater, SIGNAL(timeout()), + this, SLOT(doAutoRepeat()) ); + } + repeater->start( thresholdTime, FALSE ); +} + + +/*! + Stops the auto-repeat logic. +*/ + +void QScrollBar::stopAutoRepeat() +{ + delete repeater; + repeater = 0; +} + + +/*! + \reimp +*/ +#ifndef QT_NO_WHEELEVENT +void QScrollBar::wheelEvent( QWheelEvent *e ) +{ + static float offset = 0; + static QScrollBar* offset_owner = 0; + if (offset_owner != this){ + offset_owner = this; + offset = 0; + } + if ( e->orientation() != orient && !rect().contains(e->pos()) ) + return; + e->accept(); + int step = QMIN( QApplication::wheelScrollLines()*lineStep(), + pageStep() ); + if ( ( e->state() & ControlButton ) || ( e->state() & ShiftButton ) ) + step = pageStep(); + offset += -e->delta()*step/120; + if (QABS(offset)<1) + return; + setValue( value() + int(offset) ); + offset -= int(offset); +} +#endif + +/*! + \reimp +*/ +void QScrollBar::keyPressEvent( QKeyEvent *e ) +{ + // \list + // \i Left/Right move a horizontal scrollbar by one line. + // \i Up/Down move a vertical scrollbar by one line. + // \i PageUp moves up one page. + // \i PageDown moves down one page. + // \i Home moves to the start (minValue()). + // \i End moves to the end (maxValue()). + // \endlist + + // Note that unless you call setFocusPolicy(), the default NoFocus + // will apply and the user will not be able to use the keyboard to + // interact with the scrollbar. + switch ( e->key() ) { + case Key_Left: + if ( orient == Horizontal ) + subtractLine(); + break; + case Key_Right: + if ( orient == Horizontal ) + addLine(); + break; + case Key_Up: + if ( orient == Vertical ) + subtractLine(); + break; + case Key_Down: + if ( orient == Vertical ) + addLine(); + break; + case Key_PageUp: + subtractPage(); + break; + case Key_PageDown: + addPage(); + break; + case Key_Home: + setValue( minValue() ); + break; + case Key_End: + setValue( maxValue() ); + break; + default: + e->ignore(); + break; + } +} + + +/*! + \reimp +*/ +void QScrollBar::resizeEvent( QResizeEvent * ) +{ + positionSliderFromValue(); +} + + +/*! + \reimp +*/ +void QScrollBar::paintEvent( QPaintEvent * ) +{ + QPainter p( this ); + drawControls(QStyle::SC_ScrollBarAddLine | + QStyle::SC_ScrollBarSubLine | + QStyle::SC_ScrollBarAddPage | + QStyle::SC_ScrollBarSubPage | + QStyle::SC_ScrollBarFirst | + QStyle::SC_ScrollBarLast | + QStyle::SC_ScrollBarSlider, + pressedControl, &p ); +} + +static QCOORD sliderStartPos = 0; + +/*! + \reimp + */ +void QScrollBar::contextMenuEvent( QContextMenuEvent *e ) +{ + if(clickedAt) + e->consume(); + else + e->ignore(); +} + +/*! + \reimp +*/ +void QScrollBar::mousePressEvent( QMouseEvent *e ) +{ + bool midButtonAbsPos = + style().styleHint(QStyle::SH_ScrollBar_MiddleClickAbsolutePosition, + this); + + if ( !(e->button() == LeftButton || + (midButtonAbsPos && e->button() == MidButton) ) ) + return; + + if ( maxValue() == minValue() ) // nothing to be done + return; + + if ( e->state() & MouseButtonMask ) // another button was already pressed + return; + + clickedAt = TRUE; + pressedControl = style().querySubControl(QStyle::CC_ScrollBar, this, e->pos() ); + + if ( (pressedControl == QStyle::SC_ScrollBarAddPage || + pressedControl == QStyle::SC_ScrollBarSubPage || + pressedControl == QStyle::SC_ScrollBarSlider ) && + ((midButtonAbsPos && e->button() == MidButton) || + style().styleHint(QStyle::SH_ScrollBar_LeftClickAbsolutePosition) && e->button() == LeftButton)) { + + QRect sr = style().querySubControlMetrics(QStyle::CC_ScrollBar, this, + QStyle::SC_ScrollBarSlider ), + gr = style().querySubControlMetrics(QStyle::CC_ScrollBar, this, + QStyle::SC_ScrollBarGroove ); + int sliderMin, sliderMax, sliderLength; + sliderMin = sliderMax = sliderLength = 0; + if (HORIZONTAL) { + sliderMin = gr.x(); + sliderMax = sliderMin + gr.width(); + sliderLength = sr.width(); + } else { + sliderMin = gr.y(); + sliderMax = sliderMin + gr.height(); + sliderLength = sr.height(); + } + + int newSliderPos = (HORIZONTAL ? e->pos().x() : e->pos().y()) + - sliderLength/2; + newSliderPos = QMIN( newSliderPos, sliderMax - sliderLength ); + newSliderPos = QMAX( newSliderPos, sliderMin ); + setValue( sliderPosToRangeValue(newSliderPos) ); + sliderPos = newSliderPos; + pressedControl = QStyle::SC_ScrollBarSlider; + } + + if ( pressedControl == QStyle::SC_ScrollBarSlider ) { + clickOffset = (QCOORD)( (HORIZONTAL ? e->pos().x() : e->pos().y()) + - sliderPos ); + slidePrevVal = value(); + sliderStartPos = sliderPos; + drawControls( pressedControl, pressedControl ); + emit sliderPressed(); +#if defined(QT_ACCESSIBILITY_SUPPORT) + QAccessible::updateAccessibility( this, 0, QAccessible::ScrollingStart ); +#endif + } else if ( pressedControl != QStyle::SC_None ) { + drawControls( pressedControl, pressedControl ); + action( (QStyle::SubControl) pressedControl ); + startAutoRepeat(); + } +} + + +/*! + \reimp +*/ +void QScrollBar::mouseReleaseEvent( QMouseEvent *e ) +{ + if ( !clickedAt ) + return; + + if ( e->stateAfter() & MouseButtonMask ) // some other button is still pressed + return; + + QStyle::SubControl tmp = (QStyle::SubControl) pressedControl; + clickedAt = FALSE; + stopAutoRepeat(); + mouseMoveEvent( e ); // Might have moved since last mouse move event. + pressedControl = QStyle::SC_None; + + if (tmp == QStyle::SC_ScrollBarSlider) { + directSetValue( calculateValueFromSlider() ); + emit sliderReleased(); + if ( value() != prevValue() ) { + emit valueChanged( value() ); +#if defined(QT_ACCESSIBILITY_SUPPORT) + QAccessible::updateAccessibility( this, 0, QAccessible::ValueChanged ); +#endif + } +#if defined(QT_ACCESSIBILITY_SUPPORT) + QAccessible::updateAccessibility( this, 0, QAccessible::ScrollingEnd ); +#endif + } + drawControls( tmp, pressedControl ); + if ( e->button() == MidButton ) + repaint( FALSE ); +} + + +/*! + \reimp +*/ +void QScrollBar::mouseMoveEvent( QMouseEvent *e ) +{ + if ( !isVisible() ) { + clickedAt = FALSE; + return; + } + + bool mcab = style().styleHint(QStyle::SH_ScrollBar_MiddleClickAbsolutePosition, + this); + if ( ! clickedAt || ! (e->state() & LeftButton || + ((e->state() & MidButton) && mcab))) + return; + + int newSliderPos; + if ( pressedControl == QStyle::SC_ScrollBarSlider ) { + QRect gr = style().querySubControlMetrics(QStyle::CC_ScrollBar, this, + QStyle::SC_ScrollBarGroove ), + sr = style().querySubControlMetrics(QStyle::CC_ScrollBar, this, + QStyle::SC_ScrollBarSlider ); + int sliderMin, sliderMax, sliderLength; + + if (HORIZONTAL) { + sliderLength = sr.width(); + sliderMin = gr.x(); + sliderMax = gr.right() - sliderLength + 1; + } else { + sliderLength = sr.height(); + sliderMin = gr.y(); + sliderMax = gr.bottom() - sliderLength + 1; + } + + QRect r = rect(); + int m = style().pixelMetric(QStyle::PM_MaximumDragDistance, this); + if ( m >= 0 ) { + if ( orientation() == Horizontal ) + r.setRect( r.x() - m, r.y() - 2*m, r.width() + 2*m, r.height() + 4*m ); + else + r.setRect( r.x() - 2*m, r.y() - m, r.width() + 4*m, r.height() + 2*m ); + if (! r.contains( e->pos())) + newSliderPos = sliderStartPos; + else + newSliderPos = (HORIZONTAL ? e->pos().x() : + e->pos().y()) -clickOffset; + } else + newSliderPos = (HORIZONTAL ? e->pos().x() : + e->pos().y()) -clickOffset; + + if ( newSliderPos < sliderMin ) + newSliderPos = sliderMin; + else if ( newSliderPos > sliderMax ) + newSliderPos = sliderMax; + int newVal = sliderPosToRangeValue(newSliderPos); + if ( newVal != slidePrevVal ) + emit sliderMoved( newVal ); + if ( track && newVal != value() ) { + directSetValue( newVal ); // Set directly, painting done below + emit valueChanged( value() ); +#if defined(QT_ACCESSIBILITY_SUPPORT) + QAccessible::updateAccessibility( this, 0, QAccessible::ValueChanged ); +#endif + } + slidePrevVal = newVal; + sliderPos = (QCOORD)newSliderPos; + drawControls( QStyle::SC_ScrollBarAddPage | + QStyle::SC_ScrollBarSlider | + QStyle::SC_ScrollBarSubPage, + pressedControl ); + } else if (! style().styleHint(QStyle::SH_ScrollBar_ScrollWhenPointerLeavesControl)) { + // stop scrolling when the mouse pointer leaves a control + // similar to push buttons + if ( pressedControl != (uint)style().querySubControl(QStyle::CC_ScrollBar, this, e->pos() ) ) { + drawControls( pressedControl, QStyle::SC_None ); + stopAutoRepeat(); + } else if ( !repeater ) { + drawControls( pressedControl, pressedControl ); + action( (QStyle::SubControl) pressedControl ); + startAutoRepeat(); + } + } +} + + +/*! + \fn int QScrollBar::sliderStart() const + + Returns the pixel position where the scroll bar slider starts. + + This is equivalent to sliderRect().y() for vertical scroll bars or + sliderRect().x() for horizontal scroll bars. +*/ + +/*! + Returns the scroll bar slider rectangle. + + \sa sliderStart() +*/ + +QRect QScrollBar::sliderRect() const +{ + return style().querySubControlMetrics(QStyle::CC_ScrollBar, this, + QStyle::SC_ScrollBarSlider ); +} + +void QScrollBar::positionSliderFromValue() +{ + sliderPos = (QCOORD)rangeValueToSliderPos( value() ); +} + +int QScrollBar::calculateValueFromSlider() const +{ + return sliderPosToRangeValue( sliderPos ); +} + +int QScrollBar::rangeValueToSliderPos( int v ) const +{ + QRect gr = style().querySubControlMetrics(QStyle::CC_ScrollBar, this, + QStyle::SC_ScrollBarGroove ); + QRect sr = style().querySubControlMetrics(QStyle::CC_ScrollBar, this, + QStyle::SC_ScrollBarSlider ); + int sliderMin, sliderMax, sliderLength; + + if (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 positionFromValue( v, sliderMax-sliderMin ) + sliderMin; +} + +int QScrollBar::sliderPosToRangeValue( int pos ) const +{ + QRect gr = style().querySubControlMetrics(QStyle::CC_ScrollBar, this, + QStyle::SC_ScrollBarGroove ); + QRect sr = style().querySubControlMetrics(QStyle::CC_ScrollBar, this, + QStyle::SC_ScrollBarSlider ); + int sliderMin, sliderMax, sliderLength; + + if (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 valueFromPosition( pos - sliderMin, sliderMax - sliderMin ); +} + + +void QScrollBar::action( int control ) +{ + switch( control ) { + case QStyle::SC_ScrollBarAddLine: + addLine(); + emit nextLine(); + break; + case QStyle::SC_ScrollBarSubLine: + subtractLine(); + emit prevLine(); + break; + case QStyle::SC_ScrollBarAddPage: + addPage(); + emit nextPage(); + break; + case QStyle::SC_ScrollBarSubPage: + subtractPage(); + emit prevPage(); + break; + case QStyle::SC_ScrollBarFirst: + setValue( minValue() ); +#if defined(QT_ACCESSIBILITY_SUPPORT) + QAccessible::updateAccessibility( this, 0, QAccessible::ValueChanged ); +#endif + emit valueChanged( minValue() ); + break; + case QStyle::SC_ScrollBarLast: + setValue( maxValue() ); +#if defined(QT_ACCESSIBILITY_SUPPORT) + QAccessible::updateAccessibility( this, 0, QAccessible::ValueChanged ); +#endif + emit valueChanged( maxValue() ); + break; + default: + break; + } +} + + +void QScrollBar::drawControls( uint controls, uint activeControl ) const +{ + QPainter p ( this ); + drawControls( controls, activeControl, &p ); +} + + +void QScrollBar::drawControls( uint controls, uint activeControl, + QPainter *p ) const +{ + if ( !isUpdatesEnabled() ) + return; + + QStyle::SFlags flags = QStyle::Style_Default; + if (isEnabled()) + flags |= QStyle::Style_Enabled; + if (hasFocus()) + flags |= QStyle::Style_HasFocus; + if ( orientation() == Horizontal ) + flags |= QStyle::Style_Horizontal; + + style().drawComplexControl(QStyle::CC_ScrollBar, p, this, rect(), colorGroup(), + flags, (QStyle::SubControl) controls, + (QStyle::SubControl) activeControl ); +} + +/*! + \reimp +*/ +void QScrollBar::styleChange( QStyle& old ) +{ + positionSliderFromValue(); + setBackgroundMode((Qt::BackgroundMode) + style().styleHint(QStyle::SH_ScrollBar_BackgroundMode)); + QWidget::styleChange( old ); +} + +/*! + \property QScrollBar::minValue + \brief the scroll bar's minimum value + + When setting this property, the \l QScrollBar::maxValue is + adjusted if necessary to ensure that the range remains valid. + + \sa setRange() +*/ +int QScrollBar::minValue() const +{ + return QRangeControl::minValue(); +} + +void QScrollBar::setMinValue( int minVal ) +{ + QRangeControl::setMinValue( minVal ); +} + +/*! + \property QScrollBar::maxValue + \brief the scroll bar's maximum value + + When setting this property, the \l QScrollBar::minValue is + adjusted if necessary to ensure that the range remains valid. + + \sa setRange() +*/ +int QScrollBar::maxValue() const +{ + return QRangeControl::maxValue(); +} + +void QScrollBar::setMaxValue( int maxVal ) +{ + QRangeControl::setMaxValue( maxVal ); +} + +/*! + \property QScrollBar::lineStep + \brief the line step + + When setting lineStep, the virtual stepChange() function will be + called if the new line step is different from the previous + setting. + + \sa setSteps() QRangeControl::pageStep() setRange() +*/ + +int QScrollBar::lineStep() const +{ + return QRangeControl::lineStep(); +} + +/*! + \property QScrollBar::pageStep + \brief the page step + + When setting pageStep, the virtual stepChange() function will be + called if the new page step is different from the previous + setting. + + \sa QRangeControl::setSteps() setLineStep() setRange() +*/ + +int QScrollBar::pageStep() const +{ + return QRangeControl::pageStep(); +} + +void QScrollBar::setLineStep( int i ) +{ + setSteps( i, pageStep() ); +} + +void QScrollBar::setPageStep( int i ) +{ + setSteps( lineStep(), i ); +} + +/*! + \property QScrollBar::value + \brief the scroll bar's value + + \sa QRangeControl::value() prevValue() +*/ + +int QScrollBar::value() const +{ + return QRangeControl::value(); +} + +void QScrollBar::setValue( int i ) +{ + QRangeControl::setValue( i ); +} + + +/*! + This function is called when the scrollbar is hidden. +*/ +void QScrollBar::hideEvent( QHideEvent* ) +{ + pressedControl = QStyle::SC_None; + clickedAt = FALSE; +} + + +#undef ADD_LINE_ACTIVE +#undef SUB_LINE_ACTIVE +#endif |