Slider.qml 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333
  1. /****************************************************************************
  2. **
  3. ** Copyright (C) 2016 The Qt Company Ltd.
  4. ** Contact: https://www.qt.io/licensing/
  5. **
  6. ** This file is part of the Qt Quick Controls module of the Qt Toolkit.
  7. **
  8. ** $QT_BEGIN_LICENSE:LGPL$
  9. ** Commercial License Usage
  10. ** Licensees holding valid commercial Qt licenses may use this file in
  11. ** accordance with the commercial license agreement provided with the
  12. ** Software or, alternatively, in accordance with the terms contained in
  13. ** a written agreement between you and The Qt Company. For licensing terms
  14. ** and conditions see https://www.qt.io/terms-conditions. For further
  15. ** information use the contact form at https://www.qt.io/contact-us.
  16. **
  17. ** GNU Lesser General Public License Usage
  18. ** Alternatively, this file may be used under the terms of the GNU Lesser
  19. ** General Public License version 3 as published by the Free Software
  20. ** Foundation and appearing in the file LICENSE.LGPL3 included in the
  21. ** packaging of this file. Please review the following information to
  22. ** ensure the GNU Lesser General Public License version 3 requirements
  23. ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html.
  24. **
  25. ** GNU General Public License Usage
  26. ** Alternatively, this file may be used under the terms of the GNU
  27. ** General Public License version 2.0 or (at your option) the GNU General
  28. ** Public license version 3 or any later version approved by the KDE Free
  29. ** Qt Foundation. The licenses are as published by the Free Software
  30. ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3
  31. ** included in the packaging of this file. Please review the following
  32. ** information to ensure the GNU General Public License requirements will
  33. ** be met: https://www.gnu.org/licenses/gpl-2.0.html and
  34. ** https://www.gnu.org/licenses/gpl-3.0.html.
  35. **
  36. ** $QT_END_LICENSE$
  37. **
  38. ****************************************************************************/
  39. import QtQuick 2.2
  40. import QtQuick.Controls 1.2
  41. import QtQuick.Controls.Private 1.0
  42. /*!
  43. \qmltype Slider
  44. \inqmlmodule QtQuick.Controls
  45. \since 5.1
  46. \ingroup controls
  47. \brief Provides a vertical or horizontal slider control.
  48. \image slider.png
  49. The slider is the classic control for providing a bounded value. It lets
  50. the user move a slider handle along a horizontal or vertical groove
  51. and translates the handle's position into a value within the legal range.
  52. \code
  53. Slider {
  54. value: 0.5
  55. }
  56. \endcode
  57. The slider value is by default in the range [0, 1]. If integer values are
  58. needed, you can set the \l stepSize.
  59. You can create a custom appearance for a Slider by
  60. assigning a \l {SliderStyle}.
  61. */
  62. Control {
  63. id: slider
  64. /*!
  65. \qmlproperty enumeration Slider::orientation
  66. This property holds the layout orientation of the slider.
  67. The default value is \c Qt.Horizontal.
  68. */
  69. property int orientation: Qt.Horizontal
  70. /*!
  71. \qmlproperty real Slider::minimumValue
  72. This property holds the minimum value of the slider.
  73. The default value is \c{0.0}.
  74. */
  75. property alias minimumValue: range.minimumValue
  76. /*!
  77. \qmlproperty real Slider::maximumValue
  78. This property holds the maximum value of the slider.
  79. The default value is \c{1.0}.
  80. */
  81. property alias maximumValue: range.maximumValue
  82. /*!
  83. \qmlproperty bool Slider::updateValueWhileDragging
  84. This property indicates whether the current \l value should be updated while
  85. the user is moving the slider handle, or only when the button has been released.
  86. This property could for instance be modified if changing the slider value would turn
  87. out to be too time consuming.
  88. The default value is \c true.
  89. */
  90. property bool updateValueWhileDragging: true
  91. /*!
  92. \qmlproperty bool Slider::pressed
  93. This property indicates whether the slider handle is being pressed.
  94. */
  95. readonly property alias pressed: mouseArea.pressed
  96. /*!
  97. \qmlproperty bool Slider::hovered
  98. This property indicates whether the slider handle is being hovered.
  99. */
  100. readonly property alias hovered: mouseArea.handleHovered
  101. /*!
  102. \qmlproperty real Slider::stepSize
  103. This property indicates the slider step size.
  104. A value of 0 indicates that the value of the slider operates in a
  105. continuous range between \l minimumValue and \l maximumValue.
  106. Any non 0 value indicates a discrete stepSize. The following example
  107. will generate a slider with integer values in the range [0-5].
  108. \qml
  109. Slider {
  110. maximumValue: 5.0
  111. stepSize: 1.0
  112. }
  113. \endqml
  114. The default value is \c{0.0}.
  115. */
  116. property alias stepSize: range.stepSize
  117. /*!
  118. \qmlproperty real Slider::value
  119. This property holds the current value of the slider.
  120. The default value is \c{0.0}.
  121. */
  122. property alias value: range.value
  123. /*!
  124. \qmlproperty bool Slider::activeFocusOnPress
  125. This property indicates whether the slider should receive active focus when
  126. pressed.
  127. */
  128. property bool activeFocusOnPress: false
  129. /*!
  130. \qmlproperty bool Slider::tickmarksEnabled
  131. This property indicates whether the slider should display tickmarks
  132. at step intervals. Tick mark spacing is calculated based on the
  133. \l stepSize property.
  134. The default value is \c false.
  135. \note This property may be ignored on some platforms when using the native style (e.g. Android).
  136. */
  137. property bool tickmarksEnabled: false
  138. /*! \internal */
  139. property bool __horizontal: orientation === Qt.Horizontal
  140. /*! \internal
  141. The extra arguments positionAtMinimum and positionAtMaximum are there to force
  142. re-evaluation of the handle position when the constraints change (QTBUG-41255),
  143. and the same for range.minimumValue (QTBUG-51765).
  144. */
  145. property real __handlePos: range.valueForPosition(__horizontal ? fakeHandle.x : fakeHandle.y,
  146. range.positionAtMinimum, range.positionAtMaximum, range.minimumValue)
  147. activeFocusOnTab: true
  148. Accessible.role: Accessible.Slider
  149. /*! \internal */
  150. function accessibleIncreaseAction() {
  151. range.increaseSingleStep()
  152. }
  153. /*! \internal */
  154. function accessibleDecreaseAction() {
  155. range.decreaseSingleStep()
  156. }
  157. style: Settings.styleComponent(Settings.style, "SliderStyle.qml", slider)
  158. Keys.onRightPressed: if (__horizontal) range.increaseSingleStep()
  159. Keys.onLeftPressed: if (__horizontal) range.decreaseSingleStep()
  160. Keys.onUpPressed: if (!__horizontal) range.increaseSingleStep()
  161. Keys.onDownPressed: if (!__horizontal) range.decreaseSingleStep()
  162. RangeModel {
  163. id: range
  164. minimumValue: 0.0
  165. maximumValue: 1.0
  166. value: 0
  167. stepSize: 0.0
  168. inverted: __horizontal ? false : true
  169. positionAtMinimum: 0
  170. positionAtMaximum: __horizontal ? slider.width - fakeHandle.width : slider.height - fakeHandle.height
  171. }
  172. Item {
  173. id: fakeHandle
  174. anchors.verticalCenter: __horizontal ? parent.verticalCenter : undefined
  175. anchors.horizontalCenter: !__horizontal ? parent.horizontalCenter : undefined
  176. width: __panel.handleWidth
  177. height: __panel.handleHeight
  178. function updatePos() {
  179. if (updateValueWhileDragging && !mouseArea.drag.active)
  180. range.position = __horizontal ? x : y
  181. }
  182. onXChanged: updatePos();
  183. onYChanged: updatePos();
  184. }
  185. MouseArea {
  186. id: mouseArea
  187. anchors.fill: parent
  188. hoverEnabled: Settings.hoverEnabled
  189. property int clickOffset: 0
  190. property real pressX: 0
  191. property real pressY: 0
  192. property bool handleHovered: false
  193. function clamp ( val ) {
  194. return Math.max(range.positionAtMinimum, Math.min(range.positionAtMaximum, val))
  195. }
  196. function updateHandlePosition(mouse, force) {
  197. var pos, overThreshold
  198. if (__horizontal) {
  199. pos = clamp (mouse.x + clickOffset - fakeHandle.width/2)
  200. overThreshold = Math.abs(mouse.x - pressX) >= Settings.dragThreshold
  201. if (overThreshold)
  202. preventStealing = true
  203. if (overThreshold || force)
  204. fakeHandle.x = pos
  205. } else if (!__horizontal) {
  206. pos = clamp (mouse.y + clickOffset- fakeHandle.height/2)
  207. overThreshold = Math.abs(mouse.y - pressY) >= Settings.dragThreshold
  208. if (overThreshold)
  209. preventStealing = true
  210. if (overThreshold || force)
  211. fakeHandle.y = pos
  212. }
  213. }
  214. onPositionChanged: {
  215. if (pressed)
  216. updateHandlePosition(mouse, preventStealing)
  217. var point = mouseArea.mapToItem(fakeHandle, mouse.x, mouse.y)
  218. handleHovered = fakeHandle.contains(Qt.point(point.x, point.y))
  219. }
  220. onPressed: {
  221. if (slider.activeFocusOnPress)
  222. slider.forceActiveFocus();
  223. if (handleHovered) {
  224. var point = mouseArea.mapToItem(fakeHandle, mouse.x, mouse.y)
  225. clickOffset = __horizontal ? fakeHandle.width/2 - point.x : fakeHandle.height/2 - point.y
  226. }
  227. pressX = mouse.x
  228. pressY = mouse.y
  229. updateHandlePosition(mouse, !Settings.hasTouchScreen)
  230. }
  231. onReleased: {
  232. updateHandlePosition(mouse, Settings.hasTouchScreen)
  233. // If we don't update while dragging, this is the only
  234. // moment that the range is updated.
  235. if (!slider.updateValueWhileDragging)
  236. range.position = __horizontal ? fakeHandle.x : fakeHandle.y;
  237. clickOffset = 0
  238. preventStealing = false
  239. }
  240. onExited: handleHovered = false
  241. }
  242. // During the drag, we simply ignore the position set from the range, this
  243. // means that setting a value while dragging will not "interrupt" the
  244. // dragging activity.
  245. Binding {
  246. when: !mouseArea.drag.active
  247. target: fakeHandle
  248. property: __horizontal ? "x" : "y"
  249. value: range.position
  250. }
  251. WheelArea {
  252. id: wheelarea
  253. anchors.fill: parent
  254. horizontalMinimumValue: slider.minimumValue
  255. horizontalMaximumValue: slider.maximumValue
  256. verticalMinimumValue: slider.minimumValue
  257. verticalMaximumValue: slider.maximumValue
  258. property real step: (slider.maximumValue - slider.minimumValue)/(range.positionAtMaximum - range.positionAtMinimum)
  259. onVerticalWheelMoved: {
  260. if (verticalDelta !== 0) {
  261. var delta = Math.abs(verticalDelta)*step > stepSize ? verticalDelta*step : verticalDelta/Math.abs(verticalDelta)*stepSize
  262. value -= delta * (inverted ? 1 : -1)
  263. }
  264. }
  265. onHorizontalWheelMoved: {
  266. if (horizontalDelta !== 0) {
  267. var delta = Math.abs(horizontalDelta)*step > stepSize ? horizontalDelta*step : horizontalDelta/Math.abs(horizontalDelta)*stepSize
  268. value += delta * (inverted ? 1 : -1)
  269. }
  270. }
  271. }
  272. }