ScrollView.qml 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359
  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. import QtQuick.Controls.Styles 1.1
  43. /*!
  44. \qmltype ScrollView
  45. \inqmlmodule QtQuick.Controls
  46. \since 5.1
  47. \ingroup views
  48. \ingroup controls
  49. \brief Provides a scrolling view within another Item.
  50. \image scrollview.png
  51. A ScrollView can be used either to replace a \l Flickable or decorate an
  52. existing \l Flickable. Depending on the platform, it will add scroll bars and
  53. a content frame.
  54. Only one Item can be a direct child of the ScrollView and the child is implicitly anchored
  55. to fill the scroll view.
  56. Example:
  57. \code
  58. ScrollView {
  59. Image { source: "largeImage.png" }
  60. }
  61. \endcode
  62. In the previous example the Image item will implicitly get scroll behavior as if it was
  63. used within a \l Flickable. The width and height of the child item will be used to
  64. define the size of the content area.
  65. Example:
  66. \code
  67. ScrollView {
  68. ListView {
  69. ...
  70. }
  71. }
  72. \endcode
  73. In this case the content size of the ScrollView will simply mirror that of its contained
  74. \l flickableItem.
  75. You can create a custom appearance for a ScrollView by
  76. assigning a \l {ScrollViewStyle}.
  77. */
  78. FocusScope {
  79. id: root
  80. implicitWidth: 240
  81. implicitHeight: 150
  82. /*!
  83. This property tells the ScrollView if it should render
  84. a frame around its content.
  85. The default value is \c false.
  86. */
  87. property bool frameVisible: false
  88. /*! \qmlproperty enumeration ScrollView::horizontalScrollBarPolicy
  89. \since QtQuick.Controls 1.3
  90. This property holds the policy for showing the horizontal scrollbar.
  91. It can be any of the following values:
  92. \list
  93. \li Qt.ScrollBarAsNeeded
  94. \li Qt.ScrollBarAlwaysOff
  95. \li Qt.ScrollBarAlwaysOn
  96. \endlist
  97. The default policy is \c Qt.ScrollBarAsNeeded.
  98. */
  99. property alias horizontalScrollBarPolicy: scroller.horizontalScrollBarPolicy
  100. /*! \qmlproperty enumeration ScrollView::verticalScrollBarPolicy
  101. \since QtQuick.Controls 1.3
  102. This property holds the policy for showing the vertical scrollbar.
  103. It can be any of the following values:
  104. \list
  105. \li Qt.ScrollBarAsNeeded
  106. \li Qt.ScrollBarAlwaysOff
  107. \li Qt.ScrollBarAlwaysOn
  108. \endlist
  109. The default policy is \c Qt.ScrollBarAsNeeded.
  110. */
  111. property alias verticalScrollBarPolicy: scroller.verticalScrollBarPolicy
  112. /*!
  113. This property controls if there should be a highlight
  114. around the frame when the ScrollView has input focus.
  115. The default value is \c false.
  116. \note This property is only applicable on some platforms, such
  117. as Mac OS.
  118. */
  119. property bool highlightOnFocus: false
  120. /*!
  121. \qmlproperty Item ScrollView::viewport
  122. The viewport determines the current "window" on the contentItem.
  123. In other words, it clips it and the size of the viewport tells you
  124. how much of the content area is visible.
  125. */
  126. property alias viewport: viewportItem
  127. /*!
  128. \qmlproperty Item ScrollView::flickableItem
  129. The flickableItem of the ScrollView. If the contentItem provided
  130. to the ScrollView is a Flickable, it will be the \l contentItem.
  131. */
  132. readonly property alias flickableItem: internal.flickableItem
  133. /*!
  134. The contentItem of the ScrollView. This is set by the user.
  135. Note that the definition of contentItem is somewhat different to that
  136. of a Flickable, where the contentItem is implicitly created.
  137. */
  138. default property Item contentItem
  139. /*! \internal */
  140. property alias __scroller: scroller
  141. /*! \internal */
  142. property alias __verticalScrollbarOffset: scroller.verticalScrollbarOffset
  143. /*! \internal */
  144. property alias __wheelAreaScrollSpeed: wheelArea.scrollSpeed
  145. /*! \internal */
  146. property int __scrollBarTopMargin: 0
  147. /*! \internal */
  148. property int __viewTopMargin: 0
  149. /*! \internal */
  150. property alias __horizontalScrollBar: scroller.horizontalScrollBar
  151. /*! \internal */
  152. property alias __verticalScrollBar: scroller.verticalScrollBar
  153. /*! \qmlproperty Component ScrollView::style
  154. The style Component for this control.
  155. \sa {Qt Quick Controls Styles QML Types}
  156. */
  157. property Component style: Settings.styleComponent(Settings.style, "ScrollViewStyle.qml", root)
  158. /*! \internal */
  159. property Style __style: styleLoader.item
  160. activeFocusOnTab: true
  161. onContentItemChanged: {
  162. if (contentItem.hasOwnProperty("contentY") && // Check if flickable
  163. contentItem.hasOwnProperty("contentHeight")) {
  164. internal.flickableItem = contentItem // "Use content if it is a flickable
  165. internal.flickableItem.parent = viewportItem
  166. } else {
  167. internal.flickableItem = flickableComponent.createObject(viewportItem)
  168. contentItem.parent = internal.flickableItem.contentItem
  169. }
  170. internal.flickableItem.anchors.fill = viewportItem
  171. if (!Settings.hasTouchScreen)
  172. internal.flickableItem.interactive = false
  173. }
  174. children: Item {
  175. id: internal
  176. property Flickable flickableItem
  177. Loader {
  178. id: styleLoader
  179. sourceComponent: style
  180. onStatusChanged: {
  181. if (status === Loader.Error)
  182. console.error("Failed to load Style for", root)
  183. }
  184. property alias __control: root
  185. }
  186. Binding {
  187. target: flickableItem
  188. property: "contentHeight"
  189. when: contentItem !== flickableItem
  190. value: contentItem ? contentItem.height : 0
  191. }
  192. Binding {
  193. target: flickableItem
  194. when: contentItem !== flickableItem
  195. property: "contentWidth"
  196. value: contentItem ? contentItem.width : 0
  197. }
  198. Connections {
  199. target: flickableItem
  200. onContentYChanged: {
  201. scroller.blockUpdates = true
  202. scroller.verticalScrollBar.value = flickableItem.contentY - flickableItem.originY
  203. scroller.blockUpdates = false
  204. }
  205. onContentXChanged: {
  206. scroller.blockUpdates = true
  207. scroller.horizontalScrollBar.value = flickableItem.contentX - flickableItem.originX
  208. scroller.blockUpdates = false
  209. }
  210. }
  211. anchors.fill: parent
  212. Component {
  213. id: flickableComponent
  214. Flickable {}
  215. }
  216. WheelArea {
  217. id: wheelArea
  218. parent: flickableItem
  219. z: -1
  220. // ### Note this is needed due to broken mousewheel behavior in Flickable.
  221. anchors.fill: parent
  222. property int acceleration: 40
  223. property int flickThreshold: Settings.dragThreshold
  224. property real speedThreshold: 3
  225. property real ignored: 0.001 // ## flick() does not work with 0 yVelocity
  226. property int maxFlick: 400
  227. property bool horizontalRecursionGuard: false
  228. property bool verticalRecursionGuard: false
  229. horizontalMinimumValue: 0
  230. horizontalMaximumValue: flickableItem ? flickableItem.contentWidth - viewport.width : 0
  231. verticalMinimumValue: 0
  232. verticalMaximumValue: flickableItem ? flickableItem.contentHeight - viewport.height + __viewTopMargin : 0
  233. // The default scroll speed for typical angle-based mouse wheels. The value
  234. // comes originally from QTextEdit, which sets 20px steps by default, as well as
  235. // QQuickWheelArea.
  236. // TODO: centralize somewhere, QPlatformTheme?
  237. scrollSpeed: 20 * (__style && __style.__wheelScrollLines || 1)
  238. Connections {
  239. target: flickableItem
  240. onContentYChanged: {
  241. wheelArea.verticalRecursionGuard = true
  242. wheelArea.verticalValue = flickableItem.contentY - flickableItem.originY
  243. wheelArea.verticalRecursionGuard = false
  244. }
  245. onContentXChanged: {
  246. wheelArea.horizontalRecursionGuard = true
  247. wheelArea.horizontalValue = flickableItem.contentX - flickableItem.originX
  248. wheelArea.horizontalRecursionGuard = false
  249. }
  250. }
  251. onVerticalValueChanged: {
  252. if (!verticalRecursionGuard) {
  253. var effectiveContentY = flickableItem.contentY - flickableItem.originY
  254. if (effectiveContentY < flickThreshold && verticalDelta > speedThreshold) {
  255. flickableItem.flick(ignored, Math.min(maxFlick, acceleration * verticalDelta))
  256. } else if (effectiveContentY > flickableItem.contentHeight - flickThreshold - viewport.height
  257. && verticalDelta < -speedThreshold) {
  258. flickableItem.flick(ignored, Math.max(-maxFlick, acceleration * verticalDelta))
  259. } else {
  260. flickableItem.contentY = verticalValue + flickableItem.originY
  261. }
  262. }
  263. }
  264. onHorizontalValueChanged: {
  265. if (!horizontalRecursionGuard)
  266. flickableItem.contentX = horizontalValue + flickableItem.originX
  267. }
  268. }
  269. ScrollViewHelper {
  270. id: scroller
  271. anchors.fill: parent
  272. active: wheelArea.active
  273. property bool outerFrame: !frameVisible || !(__style ? __style.__externalScrollBars : 0)
  274. property int scrollBarSpacing: outerFrame ? 0 : (__style ? __style.__scrollBarSpacing : 0)
  275. property int verticalScrollbarOffset: verticalScrollBar.visible && !verticalScrollBar.isTransient ?
  276. verticalScrollBar.width + scrollBarSpacing : 0
  277. property int horizontalScrollbarOffset: horizontalScrollBar.visible && !horizontalScrollBar.isTransient ?
  278. horizontalScrollBar.height + scrollBarSpacing : 0
  279. Loader {
  280. id: frameLoader
  281. sourceComponent: __style ? __style.frame : null
  282. anchors.fill: parent
  283. anchors.rightMargin: scroller.outerFrame ? 0 : scroller.verticalScrollbarOffset
  284. anchors.bottomMargin: scroller.outerFrame ? 0 : scroller.horizontalScrollbarOffset
  285. }
  286. Item {
  287. id: viewportItem
  288. anchors.fill: frameLoader
  289. anchors.topMargin: frameVisible ? __style.padding.top : 0
  290. anchors.leftMargin: frameVisible ? __style.padding.left : 0
  291. anchors.rightMargin: (frameVisible ? __style.padding.right : 0) + (scroller.outerFrame ? scroller.verticalScrollbarOffset : 0)
  292. anchors.bottomMargin: (frameVisible ? __style.padding.bottom : 0) + (scroller.outerFrame ? scroller.horizontalScrollbarOffset : 0)
  293. clip: true
  294. }
  295. }
  296. FocusFrame { visible: highlightOnFocus && root.activeFocus }
  297. }
  298. }