TreeView.qml 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401
  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.4
  40. import QtQuick.Controls 1.4
  41. import QtQuick.Controls.Private 1.0
  42. import QtQuick.Controls.Styles 1.2
  43. import QtQml.Models 2.2
  44. BasicTableView {
  45. id: root
  46. property var model: null
  47. property alias rootIndex: modelAdaptor.rootIndex
  48. readonly property var currentIndex: modelAdaptor.mapRowToModelIndex(__currentRow)
  49. property ItemSelectionModel selection: null
  50. signal activated(var index)
  51. signal clicked(var index)
  52. signal doubleClicked(var index)
  53. signal pressAndHold(var index)
  54. signal expanded(var index)
  55. signal collapsed(var index)
  56. function isExpanded(index) {
  57. if (index.valid && index.model !== model) {
  58. console.warn("TreeView.isExpanded: model and index mismatch")
  59. return false
  60. }
  61. return modelAdaptor.isExpanded(index)
  62. }
  63. function collapse(index) {
  64. if (index.valid && index.model !== model)
  65. console.warn("TreeView.collapse: model and index mismatch")
  66. else
  67. modelAdaptor.collapse(index)
  68. }
  69. function expand(index) {
  70. if (index.valid && index.model !== model)
  71. console.warn("TreeView.expand: model and index mismatch")
  72. else
  73. modelAdaptor.expand(index)
  74. }
  75. function indexAt(x, y) {
  76. var obj = root.mapToItem(__listView.contentItem, x, y)
  77. return modelAdaptor.mapRowToModelIndex(__listView.indexAt(obj.x, obj.y))
  78. }
  79. style: Settings.styleComponent(Settings.style, "TreeViewStyle.qml", root)
  80. // Internal stuff. Do not look
  81. __viewTypeName: "TreeView"
  82. __model: TreeModelAdaptor {
  83. id: modelAdaptor
  84. model: root.model
  85. onExpanded: root.expanded(index)
  86. onCollapsed: root.collapsed(index)
  87. }
  88. __itemDelegateLoader: TreeViewItemDelegateLoader {
  89. __style: root.__style
  90. __itemDelegate: root.itemDelegate
  91. __mouseArea: mouseArea
  92. __treeModel: modelAdaptor
  93. }
  94. onSelectionModeChanged: if (!!selection) selection.clear()
  95. __mouseArea: MouseArea {
  96. id: mouseArea
  97. parent: __listView
  98. width: __listView.width
  99. height: __listView.height
  100. z: -1
  101. propagateComposedEvents: true
  102. focus: true
  103. // Note: with boolean preventStealing we are keeping
  104. // the flickable from eating our mouse press events
  105. preventStealing: !Settings.hasTouchScreen
  106. property var clickedIndex: undefined
  107. property var pressedIndex: undefined
  108. property bool selectOnRelease: false
  109. property int pressedColumn: -1
  110. readonly property alias currentRow: root.__currentRow
  111. readonly property alias currentIndex: root.currentIndex
  112. // Handle vertical scrolling whem dragging mouse outside boundaries
  113. property int autoScroll: 0 // 0 -> do nothing; 1 -> increment; 2 -> decrement
  114. property bool shiftPressed: false // forward shift key state to the autoscroll timer
  115. Timer {
  116. running: mouseArea.autoScroll !== 0 && __verticalScrollBar.visible
  117. interval: 20
  118. repeat: true
  119. onTriggered: {
  120. var oldPressedIndex = mouseArea.pressedIndex
  121. var row
  122. if (mouseArea.autoScroll === 1) {
  123. __listView.incrementCurrentIndexBlocking();
  124. row = __listView.indexAt(0, __listView.height + __listView.contentY)
  125. if (row === -1)
  126. row = __listView.count - 1
  127. } else {
  128. __listView.decrementCurrentIndexBlocking();
  129. row = __listView.indexAt(0, __listView.contentY)
  130. }
  131. var index = modelAdaptor.mapRowToModelIndex(row)
  132. if (index !== oldPressedIndex) {
  133. mouseArea.pressedIndex = index
  134. var modifiers = mouseArea.shiftPressed ? Qt.ShiftModifier : Qt.NoModifier
  135. mouseArea.mouseSelect(index, modifiers, true /* drag */)
  136. }
  137. }
  138. }
  139. function mouseSelect(modelIndex, modifiers, drag) {
  140. if (!selection) {
  141. maybeWarnAboutSelectionMode()
  142. return
  143. }
  144. if (selectionMode) {
  145. selection.setCurrentIndex(modelIndex, ItemSelectionModel.NoUpdate)
  146. if (selectionMode === SelectionMode.SingleSelection) {
  147. selection.select(modelIndex, ItemSelectionModel.ClearAndSelect)
  148. } else {
  149. var selectRowRange = (drag && (selectionMode === SelectionMode.MultiSelection
  150. || (selectionMode === SelectionMode.ExtendedSelection
  151. && modifiers & Qt.ControlModifier)))
  152. || modifiers & Qt.ShiftModifier
  153. var itemSelection = !selectRowRange || clickedIndex === modelIndex ? modelIndex
  154. : modelAdaptor.selectionForRowRange(clickedIndex, modelIndex)
  155. if (selectionMode === SelectionMode.MultiSelection
  156. || selectionMode === SelectionMode.ExtendedSelection && modifiers & Qt.ControlModifier) {
  157. if (drag)
  158. selection.select(itemSelection, ItemSelectionModel.ToggleCurrent)
  159. else
  160. selection.select(modelIndex, ItemSelectionModel.Toggle)
  161. } else if (modifiers & Qt.ShiftModifier) {
  162. selection.select(itemSelection, ItemSelectionModel.SelectCurrent)
  163. } else {
  164. clickedIndex = modelIndex // Needed only when drag is true
  165. selection.select(modelIndex, ItemSelectionModel.ClearAndSelect)
  166. }
  167. }
  168. }
  169. }
  170. function keySelect(keyModifiers) {
  171. if (selectionMode) {
  172. if (!keyModifiers)
  173. clickedIndex = currentIndex
  174. if (!(keyModifiers & Qt.ControlModifier))
  175. mouseSelect(currentIndex, keyModifiers, keyModifiers & Qt.ShiftModifier)
  176. }
  177. }
  178. function selected(row) {
  179. if (selectionMode === SelectionMode.NoSelection)
  180. return false
  181. var modelIndex = null
  182. if (!!selection) {
  183. modelIndex = modelAdaptor.mapRowToModelIndex(row)
  184. if (modelIndex.valid) {
  185. if (selectionMode === SelectionMode.SingleSelection)
  186. return selection.currentIndex === modelIndex
  187. return selection.hasSelection && selection.isSelected(modelIndex)
  188. } else {
  189. return false
  190. }
  191. }
  192. return row === currentRow
  193. && (selectionMode === SelectionMode.SingleSelection
  194. || (selectionMode > SelectionMode.SingleSelection && !selection))
  195. }
  196. function branchDecorationContains(x, y) {
  197. var clickedItem = __listView.itemAt(0, y + __listView.contentY)
  198. if (!(clickedItem && clickedItem.rowItem))
  199. return false
  200. var branchDecoration = clickedItem.rowItem.branchDecoration
  201. if (!branchDecoration)
  202. return false
  203. var pos = mapToItem(branchDecoration, x, y)
  204. return branchDecoration.contains(Qt.point(pos.x, pos.y))
  205. }
  206. function maybeWarnAboutSelectionMode() {
  207. if (selectionMode > SelectionMode.SingleSelection)
  208. console.warn("TreeView: Non-single selection is not supported without an ItemSelectionModel.")
  209. }
  210. onPressed: {
  211. var pressedRow = __listView.indexAt(0, mouseY + __listView.contentY)
  212. pressedIndex = modelAdaptor.mapRowToModelIndex(pressedRow)
  213. pressedColumn = __listView.columnAt(mouseX)
  214. selectOnRelease = false
  215. __listView.forceActiveFocus()
  216. if (pressedRow === -1
  217. || Settings.hasTouchScreen
  218. || branchDecorationContains(mouse.x, mouse.y)) {
  219. return
  220. }
  221. if (selectionMode === SelectionMode.ExtendedSelection
  222. && selection.isSelected(pressedIndex)) {
  223. selectOnRelease = true
  224. return
  225. }
  226. __listView.currentIndex = pressedRow
  227. if (!clickedIndex)
  228. clickedIndex = pressedIndex
  229. mouseSelect(pressedIndex, mouse.modifiers, false)
  230. if (!mouse.modifiers)
  231. clickedIndex = pressedIndex
  232. }
  233. onReleased: {
  234. if (selectOnRelease) {
  235. var releasedRow = __listView.indexAt(0, mouseY + __listView.contentY)
  236. var releasedIndex = modelAdaptor.mapRowToModelIndex(releasedRow)
  237. if (releasedRow >= 0 && releasedIndex === pressedIndex)
  238. mouseSelect(pressedIndex, mouse.modifiers, false)
  239. }
  240. pressedIndex = undefined
  241. pressedColumn = -1
  242. autoScroll = 0
  243. selectOnRelease = false
  244. }
  245. onPositionChanged: {
  246. // NOTE: Testing for pressed is not technically needed, at least
  247. // until we decide to support tooltips or some other hover feature
  248. if (mouseY > __listView.height && pressed) {
  249. if (autoScroll === 1) return;
  250. autoScroll = 1
  251. } else if (mouseY < 0 && pressed) {
  252. if (autoScroll === 2) return;
  253. autoScroll = 2
  254. } else {
  255. autoScroll = 0
  256. }
  257. if (pressed && containsMouse) {
  258. var oldPressedIndex = pressedIndex
  259. var pressedRow = __listView.indexAt(0, mouseY + __listView.contentY)
  260. pressedIndex = modelAdaptor.mapRowToModelIndex(pressedRow)
  261. pressedColumn = __listView.columnAt(mouseX)
  262. if (pressedRow > -1 && oldPressedIndex !== pressedIndex) {
  263. __listView.currentIndex = pressedRow
  264. mouseSelect(pressedIndex, mouse.modifiers, true /* drag */)
  265. }
  266. }
  267. }
  268. onExited: {
  269. pressedIndex = undefined
  270. pressedColumn = -1
  271. selectOnRelease = false
  272. }
  273. onCanceled: {
  274. pressedIndex = undefined
  275. pressedColumn = -1
  276. autoScroll = 0
  277. selectOnRelease = false
  278. }
  279. onClicked: {
  280. var clickIndex = __listView.indexAt(0, mouseY + __listView.contentY)
  281. if (clickIndex > -1) {
  282. var modelIndex = modelAdaptor.mapRowToModelIndex(clickIndex)
  283. if (branchDecorationContains(mouse.x, mouse.y)) {
  284. if (modelAdaptor.isExpanded(modelIndex))
  285. modelAdaptor.collapse(modelIndex)
  286. else
  287. modelAdaptor.expand(modelIndex)
  288. } else if (root.__activateItemOnSingleClick) {
  289. root.activated(modelIndex)
  290. }
  291. root.clicked(modelIndex)
  292. }
  293. }
  294. onDoubleClicked: {
  295. var clickIndex = __listView.indexAt(0, mouseY + __listView.contentY)
  296. if (clickIndex > -1) {
  297. var modelIndex = modelAdaptor.mapRowToModelIndex(clickIndex)
  298. if (!root.__activateItemOnSingleClick)
  299. root.activated(modelIndex)
  300. root.doubleClicked(modelIndex)
  301. }
  302. }
  303. onPressAndHold: {
  304. var pressIndex = __listView.indexAt(0, mouseY + __listView.contentY)
  305. if (pressIndex > -1) {
  306. var modelIndex = modelAdaptor.mapRowToModelIndex(pressIndex)
  307. root.pressAndHold(modelIndex)
  308. }
  309. }
  310. Keys.forwardTo: [root]
  311. Keys.onUpPressed: {
  312. event.accepted = __listView.decrementCurrentIndexBlocking()
  313. keySelect(event.modifiers)
  314. }
  315. Keys.onDownPressed: {
  316. event.accepted = __listView.incrementCurrentIndexBlocking()
  317. keySelect(event.modifiers)
  318. }
  319. Keys.onRightPressed: {
  320. if (root.currentIndex.valid)
  321. root.expand(currentIndex)
  322. else
  323. event.accepted = false
  324. }
  325. Keys.onLeftPressed: {
  326. if (root.currentIndex.valid)
  327. root.collapse(currentIndex)
  328. else
  329. event.accepted = false
  330. }
  331. Keys.onReturnPressed: {
  332. if (root.currentIndex.valid)
  333. root.activated(currentIndex)
  334. else
  335. event.accepted = false
  336. }
  337. Keys.onPressed: {
  338. __listView.scrollIfNeeded(event.key)
  339. if (event.key === Qt.Key_A && event.modifiers & Qt.ControlModifier
  340. && !!selection && selectionMode > SelectionMode.SingleSelection) {
  341. var sel = modelAdaptor.selectionForRowRange(0, __listView.count - 1)
  342. selection.select(sel, ItemSelectionModel.SelectCurrent)
  343. } else if (event.key === Qt.Key_Shift) {
  344. shiftPressed = true
  345. }
  346. }
  347. Keys.onReleased: {
  348. if (event.key === Qt.Key_Shift)
  349. shiftPressed = false
  350. }
  351. }
  352. }