TabView.qml 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329
  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 TabView
  44. \inqmlmodule QtQuick.Controls
  45. \since 5.1
  46. \ingroup views
  47. \ingroup controls
  48. \brief A control that allows the user to select one of multiple stacked items.
  49. \image tabview.png
  50. TabView provides tab-based navigation model for your application.
  51. For example, the following snippet uses tabs to present rectangles of
  52. different color on each tab page:
  53. \qml
  54. TabView {
  55. Tab {
  56. title: "Red"
  57. Rectangle { color: "red" }
  58. }
  59. Tab {
  60. title: "Blue"
  61. Rectangle { color: "blue" }
  62. }
  63. Tab {
  64. title: "Green"
  65. Rectangle { color: "green" }
  66. }
  67. }
  68. \endqml
  69. \note You can create a custom appearance for a TabView by
  70. assigning a \l {TabViewStyle}.
  71. \l Tab represents the content of a tab in a TabView.
  72. */
  73. FocusScope {
  74. id: root
  75. implicitWidth: 240
  76. implicitHeight: 150
  77. /*! The current tab index */
  78. property int currentIndex: 0
  79. /*! The current tab count */
  80. readonly property int count: __tabs.count
  81. /*! The visibility of the tab frame around contents */
  82. property bool frameVisible: true
  83. /*! The visibility of the tab bar */
  84. property bool tabsVisible: true
  85. /*!
  86. \qmlproperty enumeration TabView::tabPosition
  87. \list
  88. \li Qt.TopEdge (default)
  89. \li Qt.BottomEdge
  90. \endlist
  91. */
  92. property int tabPosition: Qt.TopEdge
  93. /*!
  94. \qmlproperty Item TabView::contentItem
  95. \since QtQuick.Controls 1.3
  96. This property holds the content item of the tab view.
  97. Tabs declared as children of a TabView are automatically parented to the TabView's contentItem.
  98. */
  99. readonly property alias contentItem: stack
  100. /*! \internal */
  101. default property alias data: stack.data
  102. /*!
  103. \qmlmethod Tab TabView::addTab(string title, Component component)
  104. Adds a new tab with the given \a title and an optional \a component.
  105. Returns the newly added tab.
  106. */
  107. function addTab(title, component) {
  108. return insertTab(__tabs.count, title, component)
  109. }
  110. /*!
  111. \qmlmethod Tab TabView::insertTab(int index, string title, Component component)
  112. Inserts a new tab at \a index, with the given \a title and
  113. an optional \a component.
  114. Returns the newly added tab.
  115. */
  116. function insertTab(index, title, component) {
  117. var tab = tabcomp.createObject()
  118. tab.sourceComponent = component
  119. tab.title = title
  120. // insert at appropriate index first, then set the parent to
  121. // avoid onChildrenChanged appending it to the end of the list
  122. __tabs.insert(index, {tab: tab})
  123. tab.__inserted = true
  124. tab.parent = stack
  125. __didInsertIndex(index)
  126. __setOpacities()
  127. return tab
  128. }
  129. /*! \qmlmethod void TabView::removeTab(int index)
  130. Removes and destroys a tab at the given \a index. */
  131. function removeTab(index) {
  132. var tab = __tabs.get(index).tab
  133. __willRemoveIndex(index)
  134. __tabs.remove(index, 1)
  135. tab.destroy()
  136. __setOpacities()
  137. }
  138. /*! \qmlmethod void TabView::moveTab(int from, int to)
  139. Moves a tab \a from index \a to another. */
  140. function moveTab(from, to) {
  141. __tabs.move(from, to, 1)
  142. if (currentIndex == from) {
  143. currentIndex = to
  144. } else {
  145. var start = Math.min(from, to)
  146. var end = Math.max(from, to)
  147. if (currentIndex >= start && currentIndex <= end) {
  148. if (from < to)
  149. --currentIndex
  150. else
  151. ++currentIndex
  152. }
  153. }
  154. }
  155. /*! \qmlmethod Tab TabView::getTab(int index)
  156. Returns the \l Tab item at \a index. */
  157. function getTab(index) {
  158. var data = __tabs.get(index)
  159. return data && data.tab
  160. }
  161. /*! \internal */
  162. property ListModel __tabs: ListModel { }
  163. /*! \internal */
  164. property Component style: Settings.styleComponent(Settings.style, "TabViewStyle.qml", root)
  165. /*! \internal */
  166. property var __styleItem: loader.item
  167. onCurrentIndexChanged: __setOpacities()
  168. /*! \internal */
  169. function __willRemoveIndex(index) {
  170. // Make sure currentIndex will points to the same tab after the removal.
  171. // Also activate the next index if the current index is being removed,
  172. // except when it's both the current and last index.
  173. if (count > 1 && (currentIndex > index || currentIndex == count -1))
  174. --currentIndex
  175. }
  176. function __didInsertIndex(index) {
  177. // Make sure currentIndex points to the same tab as before the insertion.
  178. if (count > 1 && currentIndex >= index)
  179. currentIndex++
  180. }
  181. function __setOpacities() {
  182. for (var i = 0; i < __tabs.count; ++i) {
  183. var child = __tabs.get(i).tab
  184. child.visible = (i == currentIndex ? true : false)
  185. }
  186. }
  187. activeFocusOnTab: false
  188. Component {
  189. id: tabcomp
  190. Tab {}
  191. }
  192. TabBar {
  193. id: tabbarItem
  194. objectName: "tabbar"
  195. tabView: root
  196. style: loader.item
  197. anchors.top: parent.top
  198. anchors.left: root.left
  199. anchors.right: root.right
  200. }
  201. Loader {
  202. id: loader
  203. z: tabbarItem.z - 1
  204. sourceComponent: style
  205. property var __control: root
  206. }
  207. Loader {
  208. id: frameLoader
  209. z: tabbarItem.z - 1
  210. anchors.fill: parent
  211. anchors.topMargin: tabPosition === Qt.TopEdge && tabbarItem && tabsVisible ? Math.max(0, tabbarItem.height - baseOverlap) : 0
  212. anchors.bottomMargin: tabPosition === Qt.BottomEdge && tabbarItem && tabsVisible ? Math.max(0, tabbarItem.height -baseOverlap) : 0
  213. sourceComponent: frameVisible && loader.item ? loader.item.frame : null
  214. property int baseOverlap: __styleItem ? __styleItem.frameOverlap : 0
  215. Item {
  216. id: stack
  217. anchors.fill: parent
  218. anchors.margins: (frameVisible ? frameWidth : 0)
  219. anchors.topMargin: anchors.margins + (style =="mac" ? 6 : 0)
  220. anchors.bottomMargin: anchors.margins
  221. property int frameWidth
  222. property string style
  223. property bool completed: false
  224. Component.onCompleted: {
  225. addTabs(stack.children)
  226. completed = true
  227. }
  228. onChildrenChanged: {
  229. if (completed)
  230. stack.addTabs(stack.children)
  231. }
  232. function addTabs(tabs) {
  233. var tabAdded = false
  234. for (var i = 0 ; i < tabs.length ; ++i) {
  235. var tab = tabs[i]
  236. if (!tab.__inserted && tab.Accessible.role === Accessible.LayeredPane) {
  237. tab.__inserted = true
  238. // reparent tabs created dynamically by createObject(tabView)
  239. tab.parent = stack
  240. // a dynamically added tab should also get automatically removed when destructed
  241. if (completed)
  242. tab.Component.onDestruction.connect(stack.onDynamicTabDestroyed.bind(tab))
  243. __tabs.append({tab: tab})
  244. tabAdded = true
  245. }
  246. }
  247. if (tabAdded)
  248. __setOpacities()
  249. }
  250. function onDynamicTabDestroyed() {
  251. for (var i = 0; i < __tabs.count; ++i) {
  252. if (__tabs.get(i).tab === this) {
  253. __willRemoveIndex(i)
  254. __tabs.remove(i, 1)
  255. __setOpacities()
  256. break
  257. }
  258. }
  259. }
  260. }
  261. onLoaded: { item.z = -1 }
  262. }
  263. onChildrenChanged: stack.addTabs(root.children)
  264. states: [
  265. State {
  266. name: "Bottom"
  267. when: tabPosition === Qt.BottomEdge && tabbarItem != undefined
  268. PropertyChanges {
  269. target: tabbarItem
  270. anchors.topMargin: -frameLoader.baseOverlap
  271. }
  272. AnchorChanges {
  273. target: tabbarItem
  274. anchors.top: frameLoader.bottom
  275. }
  276. }
  277. ]
  278. }