StackView.qml 41 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996
  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 StackView
  44. \inherits Item
  45. \ingroup views
  46. \ingroup controls
  47. \inqmlmodule QtQuick.Controls
  48. \since 5.1
  49. \brief Provides a stack-based navigation model.
  50. \image stackview.png
  51. StackView implements a stack-based navigation model, which can be used
  52. with a set of interlinked information pages. Items are pushed onto the stack
  53. as the user navigates deeper into the material, and popped off again when he
  54. chooses to go back.
  55. The \l{Qt Quick Controls - Touch Gallery}{touch gallery} example is a good
  56. starting point to understand how StackView works. The following snippet
  57. from the example shows how it can be used:
  58. \qml
  59. StackView {
  60. id: stack
  61. initialItem: view
  62. Component {
  63. id: view
  64. MouseArea {
  65. Text {
  66. text: stack.depth
  67. anchors.centerIn: parent
  68. }
  69. onClicked: stack.push(view)
  70. }
  71. }
  72. }
  73. \endqml
  74. \section1 Using StackView in an Application
  75. Using StackView in an application is typically a simple matter of adding
  76. the StackView as a child of a Window. The stack is usually anchored to the
  77. edges of the window, except at the top or bottom where it might be anchored
  78. to a status bar, or some other similar UI component. The stack can then be
  79. used by invoking its navigation methods. The first item to show in the StackView
  80. is the one that was assigned to \l initialItem.
  81. \note Items pushed onto the stack view have \l{Supported Attached Properties}{Stack attached properties}.
  82. \section1 Basic Navigation
  83. There are three primary navigation operations in StackView: push(), pop(), and
  84. replace (replace by specifying argument \c replace to push()).
  85. These correspond to classic stack operations where "push" adds an item to the
  86. top of a stack, "pop" removes the top item from the stack, and "replace" is like a
  87. pop followed by a push, in that it replaces the topmost item on the stack with
  88. a new item (but the applied transtition might be different). The topmost item
  89. in the stack corresponds to the one that is \l{StackView::currentItem} {currently}
  90. visible on the screen. That means that "push" is the logical equivalent of navigating
  91. forward or deeper into the application, "pop" is the equivalent of navigating back,
  92. and "replace" is the equivalent of replacing the current item.
  93. Sometimes it is necessary to go back more than a single step in the stack, for
  94. example, to return to a "main" item or some kind of section item in the application.
  95. For this use case, it is possible to specify an item as a parameter for pop().
  96. This is called an "unwind" operation as the stack gets unwound to the specified item.
  97. If the item is not found, then the stack unwinds until there is only a single item in
  98. the stack, which then becomes the current item. To explicitly unwind to the bottom
  99. of the stack, it is recommended to use \l{pop()} {pop(null)}, though technically any
  100. non-existent item will do.
  101. Given the stack [A, B, C]:
  102. \list
  103. \li \l{push()}{push(D)} => [A, B, C, D] - "push" transition animation between C and D
  104. \li pop() => [A, B] - "pop" transition animation between C and B
  105. \li \l{push()}{push(D, replace)} => [A, B, D] - "replace" transition between C and D
  106. \li \l{pop()}{pop(A)} => [A] - "pop" transition between C and A
  107. \endlist
  108. \note When the stack is empty, a push() will not perform a
  109. transition animation because there is nothing to transition from (typically during
  110. application start-up). A pop() on a stack with depth 1 or 0 is a no-operation.
  111. If all items need to be removed from the stack, a separate function clear() is
  112. available.
  113. Calling push() returns the item that was pushed onto the stack.
  114. Calling pop() returns the item that was popped off the stack. When pop() is
  115. called in an unwind operation, the top-most item (the first item that was
  116. popped, which will also be the one transitioning out) is returned.
  117. \section1 Deep Linking
  118. \e{Deep linking} means launching an application into a particular state. For example,
  119. a newspaper application could be launched into showing a particular article,
  120. bypassing the front item (and possibly a section item) that would normally have
  121. to be navigated through to get to the article concerned. In terms of StackView, deep
  122. linking means the ability to modify the state of the stack, so much so that it is
  123. possible to push a set of items to the top of the stack, or to completely reset
  124. the stack to a given state.
  125. The API for deep linking in StackView is the same as for basic navigation. Pushing
  126. an array instead of a single item, will involve that all the items in that array will
  127. be pushed onto the stack. The transition animation, however, will be conducted as
  128. if only the last item in the array was pushed onto the stack. The normal semantics
  129. of push() apply for deep linking, meaning that push() adds whatever is pushed onto
  130. the stack. Note also that only the last item of the array will be loaded.
  131. The rest will be lazy-loaded as needed when entering the screen upon subsequent
  132. calls to pop (or when requesting the item by using \a get).
  133. This gives us the following result, given the stack [A, B, C]:
  134. \list
  135. \li \l{push()}{push([D, E, F])} => [A, B, C, D, E, F] - "push" transition animation between C and F
  136. \li \l{push()}{push([D, E, F], replace)} => [A, B, D, E, F] - "replace" transition animation between C and F
  137. \li clear(); \l{push()}{push([D, E, F])} => [D, E, F] - no transition animation (since the stack was empty)
  138. \endlist
  139. \section1 Pushing items
  140. An item pushed onto the StackView can be either an Item, a URL, a string
  141. containing a URL, or a Component. To push it, assign it to a property "item"
  142. inside a property list, and pass it as an argument to \l{StackView::push}{push}:
  143. \code
  144. stackView.push({item: yourItem})
  145. \endcode
  146. The list can contain several properties that control how the item should be pushed:
  147. \list
  148. \li \c item: this property is required, and holds the item to be pushed.
  149. \li \c properties: a list of QML properties to be assigned to the item upon push. These
  150. properties will be copied into the item at load time, or when the item will become
  151. the current item (normally upon push).
  152. \li \c immediate: set this property to \c true to skip transition effects. When pushing
  153. an array, this property only needs to be set on the first element to make the
  154. whole operation immediate.
  155. \li \c replace: set this property to replace the current item on the stack. When pushing
  156. an array, you only need to set this property on the first element to replace
  157. as many elements on the stack as inside the array.
  158. \li \c destroyOnPop: set this boolean to \c true if StackView needs to destroy the item when
  159. it is popped off the stack. By default (if \a destroyOnPop is not specified), StackView
  160. will destroy items pushed as components or URLs. Items not destroyed will be re-parented
  161. back to the original parents they had before being pushed onto the stack and hidden.
  162. If you need to set this property, do it with care, so that items are not leaked.
  163. \endlist
  164. If the only argument needed is "item", the following short-hand notation can be applied:
  165. \code
  166. stackView.push(yourItem)
  167. \endcode
  168. You can push several items in one go by using an array of property lists. This is
  169. more efficient than pushing items one by one, as StackView can then load only the
  170. last item in the list. The rest will be loaded as they are about to become
  171. the current item (which happens when the stack is popped). The following example shows how
  172. to push an array of items:
  173. \code
  174. stackView.push([{item: yourItem1}, {item: yourItem2}])
  175. \endcode
  176. If an inline item is pushed, the item is temporarily re-parented into the StackView. When the item
  177. is later popped off, it gets re-parented back to its original owner again.
  178. If, however, an item is pushed as a component or a URL, the actual item will be created as an
  179. item from that component. This happens automatically when the item is about to become the current
  180. item in the stack. Ownership of the item will then normally be taken by the StackView, which will
  181. automatically destroy the item when it is later popped off. The component that declared the item, by
  182. contrast, remains in the ownership of the application and is not destroyed by the stack.
  183. This can be overridden by explicitly setting \c{destroyOnPop} in the list of arguments given to push.
  184. If the \c properties to be pushed are specified, they will be copied into the item at loading time
  185. (in case of a component or URL), or when the item becomes the current item (in case of an inline
  186. item). The following example shows how this can be done:
  187. \code
  188. stackView.push({item: someItem, properties: {fgcolor: "red", bgcolor: "blue"}})
  189. \endcode
  190. \note If an item is declared inside another item, and that parent gets destroyed,
  191. (even if a component was used), that child item will also be destroyed.
  192. This follows normal Qt parent-child destruction rules, but sometimes comes as a surprise
  193. for developers.
  194. \section1 Lifecycle
  195. An item's lifecycle in the StackView can have the following transitions:
  196. \list 1
  197. \li instantiation
  198. \li inactive
  199. \li activating
  200. \li active
  201. \li deactivating
  202. \li inactive
  203. \li destruction
  204. \endlist
  205. It can move any number of times between inactive and active. When an item is activated,
  206. it's visible on the screen and is considered to be the current item. An item
  207. in a StackView that is not visible is not activated, even if the item is currently the
  208. top-most item in the stack. When the stack becomes visible, the item that is top-most gets
  209. activated. Likewise if the stack is then hidden, the topmost item would be deactivated.
  210. Popping the item off the top of the stack at this point would not result in further
  211. deactivation since the item is not active.
  212. There is an attached \l{Stack::status}{Stack.status} property that tracks the lifecycle. This
  213. property is an enumeration with the following values: \c Stack.Inactive, \c Stack.Activating,
  214. \c Stack.Active and \c Stack.Deactivating. Combined with the normal \c Component.onComplete and
  215. \c Component.onDestruction signals, the entire lifecycle is thus:
  216. \list
  217. \li Created: Component.onCompleted()
  218. \li Activating: Stack.onStatusChanged (Stack.status is Stack.Activating)
  219. \li Acivated: Stack.onStatusChanged (Stack.status is Stack.Active)
  220. \li Deactivating: Stack.onStatusChanged (Stack.status is Stack.Deactivating)
  221. \li Deactivated: Stack.onStatusChanged (Stack.status is Stack.Inactive)
  222. \li Destruction: Component.onDestruction()
  223. \endlist
  224. \section1 Finding items
  225. Sometimes it is necessary to search for an item, for example, in order to unwind the stack to
  226. an item to which the application does not have a reference. This is facilitated using a
  227. function find() in StackView. The find() function takes a callback function as its
  228. only argument. The callback gets invoked for each item in the stack (starting at the top).
  229. If the callback returns true, then it signals that a match has been found and the find()
  230. function returns that item. If the callback fails to return true (no match is found),
  231. then find() returns \c null.
  232. The code below searches for an item in the stack that has a name "order_id" and then unwinds to
  233. that item. Note that since find() returns \c {null} if no item is found, and since pop unwinds to
  234. the bottom of the stack if null is given as the target item, the code works well even in
  235. case no matching item is found.
  236. \code
  237. stackView.pop(stackView.find(function(item) {
  238. return item.name == "order_id";
  239. }));
  240. \endcode
  241. You can also get to an item in the stack using \l {get()}{get(index)}. You should use
  242. this function if your item depends on another item in the stack, as the function will
  243. ensure that the item at the given index gets loaded before it is returned.
  244. \code
  245. previousItem = stackView.get(myItem.Stack.index - 1));
  246. \endcode
  247. \section1 Transitions
  248. A transition is performed whenever a item is pushed or popped, and consists of
  249. two items: enterItem and exitItem. The StackView itself will never move items
  250. around, but instead delegates the job to an external animation set provided
  251. by the style or the application developer. How items should visually enter and leave the stack
  252. (and the geometry they should end up with) is therefore completely controlled from the outside.
  253. When the transition starts, the StackView will search for a transition that
  254. matches the operation executed. There are three transitions to choose
  255. from: \l {StackViewDelegate::}{pushTransition}, \l {StackViewDelegate::}{popTransition},
  256. and \l {StackViewDelegate::}{replaceTransition}. Each implements how
  257. \c enterItem should animate in, and \c exitItem out. The transitions are
  258. collected inside a StackViewDelegate object assigned to
  259. \l {StackView::delegate}{delegate}. By default, popTransition and
  260. replaceTransition will be the same as pushTransition, unless you set them
  261. to something else.
  262. A simple fade transition could be implemented as:
  263. \qml
  264. StackView {
  265. delegate: StackViewDelegate {
  266. function transitionFinished(properties)
  267. {
  268. properties.exitItem.opacity = 1
  269. }
  270. pushTransition: StackViewTransition {
  271. PropertyAnimation {
  272. target: enterItem
  273. property: "opacity"
  274. from: 0
  275. to: 1
  276. }
  277. PropertyAnimation {
  278. target: exitItem
  279. property: "opacity"
  280. from: 1
  281. to: 0
  282. }
  283. }
  284. }
  285. }
  286. \endqml
  287. PushTransition needs to inherit from StackViewTransition, which is a ParallelAnimation that
  288. contains the properties \c enterItem and \c exitItem. These items should be assigned to the
  289. \c target property of animations within the transition. Since the same items instance can
  290. be pushed several times to a StackView, you should always override
  291. \l {StackViewDelegate::transitionFinished()}{StackViewDelegate.transitionFinished()}.
  292. Implement this function to reset any properties animated on the exitItem so that later
  293. transitions can expect the items to be in a default state.
  294. A more complex example could look like the following. Here, the items are lying on the side before
  295. being rotated to an upright position:
  296. \qml
  297. StackView {
  298. delegate: StackViewDelegate {
  299. function transitionFinished(properties)
  300. {
  301. properties.exitItem.x = 0
  302. properties.exitItem.rotation = 0
  303. }
  304. pushTransition: StackViewTransition {
  305. SequentialAnimation {
  306. ScriptAction {
  307. script: enterItem.rotation = 90
  308. }
  309. PropertyAnimation {
  310. target: enterItem
  311. property: "x"
  312. from: enterItem.width
  313. to: 0
  314. }
  315. PropertyAnimation {
  316. target: enterItem
  317. property: "rotation"
  318. from: 90
  319. to: 0
  320. }
  321. }
  322. PropertyAnimation {
  323. target: exitItem
  324. property: "x"
  325. from: 0
  326. to: -exitItem.width
  327. }
  328. }
  329. }
  330. }
  331. \endqml
  332. \section2 Advanced usage
  333. When the StackView needs a new transition, it first calls
  334. \l {StackViewDelegate::getTransition()}{StackViewDelegate.getTransition()}.
  335. The base implementation of this function just looks for a property named \c properties.name inside
  336. itself (root), which is how it finds \c {property Component pushTransition} in the examples above.
  337. \code
  338. function getTransition(properties)
  339. {
  340. return root[properties.name]
  341. }
  342. \endcode
  343. You can override this function for your delegate if you need extra logic to decide which
  344. transition to return. You could for example introspect the items, and return different animations
  345. depending on the their internal state. StackView will expect you to return a Component that
  346. contains a StackViewTransition, or a StackViewTransition directly. The former is easier, as StackView will
  347. then create the transition and later destroy it when it's done, while avoiding any side effects
  348. caused by the transition being alive long after it has run. Returning a StackViewTransition directly
  349. can be useful if you need to write some sort of transition caching for performance reasons.
  350. As an optimization, you can also return \c null to signal that you just want to show/hide the items
  351. immediately without creating or running any transitions. You can also override this function if
  352. you need to alter the items in any way before the transition starts.
  353. \c properties contains the properties that will be assigned to the StackViewTransition before
  354. it runs. In fact, you can add more properties to this object during the call
  355. if you need to initialize additional properties of your custom StackViewTransition when the returned
  356. component is instantiated.
  357. The following example shows how you can decide which animation to use at runtime:
  358. \qml
  359. StackViewDelegate {
  360. function getTransition(properties)
  361. {
  362. return (properties.enterItem.Stack.index % 2) ? horizontalTransition : verticalTransition
  363. }
  364. function transitionFinished(properties)
  365. {
  366. properties.exitItem.x = 0
  367. properties.exitItem.y = 0
  368. }
  369. property Component horizontalTransition: StackViewTransition {
  370. PropertyAnimation {
  371. target: enterItem
  372. property: "x"
  373. from: target.width
  374. to: 0
  375. duration: 300
  376. }
  377. PropertyAnimation {
  378. target: exitItem
  379. property: "x"
  380. from: 0
  381. to: target.width
  382. duration: 300
  383. }
  384. }
  385. property Component verticalTransition: StackViewTransition {
  386. PropertyAnimation {
  387. target: enterItem
  388. property: "y"
  389. from: target.height
  390. to: 0
  391. duration: 300
  392. }
  393. PropertyAnimation {
  394. target: exitItem
  395. property: "y"
  396. from: 0
  397. to: target.height
  398. duration: 300
  399. }
  400. }
  401. }
  402. \endqml
  403. \section1 Supported Attached Properties
  404. Items in a StackView support these attached properties:
  405. \list
  406. \li \l{Stack::index}{Stack.index} - Contains the index of the item inside the StackView
  407. \li \l{Stack::view}{Stack.view} - Contains the StackView the item is in
  408. \li \l{Stack::status}{Stack.status} - Contains the status of the item
  409. \endlist
  410. */
  411. FocusScope {
  412. id: root
  413. /*! \qmlproperty int StackView::depth
  414. \readonly
  415. The number of items currently pushed onto the stack.
  416. */
  417. readonly property alias depth: root.__depth
  418. /*! \qmlproperty Item StackView::currentItem
  419. \readonly
  420. The currently top-most item in the stack.
  421. */
  422. readonly property alias currentItem: root.__currentItem
  423. /*! The first item that should be shown when the StackView is created.
  424. \a initialItem can take same value as the first argument to \l{StackView::push()}
  425. {StackView.push()}. Note that this is just a convenience for writing
  426. \c{Component.onCompleted: stackView.push(myInitialItem)}
  427. Examples:
  428. \list
  429. \li initialItem: Qt.resolvedUrl("MyItem.qml")
  430. \li initialItem: myItem
  431. \li initialItem: {"item" : Qt.resolvedUrl("MyRectangle.qml"), "properties" : {"color" : "red"}}
  432. \endlist
  433. \sa push
  434. */
  435. property var initialItem: null
  436. /*! \readonly
  437. \a busy is \c true if a transition is running, and \c false otherwise. */
  438. readonly property bool busy: __currentTransition !== null
  439. /*! The transitions to use when pushing or popping items.
  440. For better understanding on how to apply custom transitions, read \l{Transitions}.
  441. \sa {Transitions} */
  442. property StackViewDelegate delegate: StackViewSlideDelegate {}
  443. /*! \qmlmethod Item StackView::push(Item item)
  444. Pushes an item onto the stack.
  445. The function can also take a property list as argument - \c {Item StackView::push(jsobject dict)}, which
  446. should contain one or more of the following properties:
  447. \list
  448. \li \c item: this property is required, and holds the item you want to push.
  449. \li \c properties: a list of QML properties that should be assigned
  450. to the item upon push. These properties will be copied into the item when it is
  451. loaded (in case of a component or URL), or when it becomes the current item for the
  452. first time (normally upon push).
  453. \li \c immediate: set this property to \c true to skip transition effects. When pushing
  454. an array, you only need to set this property on the first element to make the
  455. whole operation immediate.
  456. \li \c replace: set this property to replace the current item on the stack. When pushing
  457. an array, you only need to set this property on the first element to replace
  458. as many elements on the stack as inside the array.
  459. \li \c destroyOnPop: set this property to specify if the item needs to be destroyed
  460. when its popped off the stack. By default (if \a destroyOnPop is not specified),
  461. StackView will destroy items pushed as components or URLs. Items
  462. not destroyed will be re-parented to the original parents they had before being
  463. pushed onto the stack, and hidden. If you need to set this property, do it with
  464. care, so that items are not leaked.
  465. \endlist
  466. You can also push an array of items (property lists) if you need to push several items
  467. in one go. A transition will then only occur between the current item and the last
  468. item in the list. Loading the other items will be deferred until needed.
  469. Examples:
  470. \list
  471. \li stackView.push({item:anItem})
  472. \li stackView.push({item:aURL, immediate: true, replace: true})
  473. \li stackView.push({item:aRectangle, properties:{color:"red"}})
  474. \li stackView.push({item:aComponent, properties:{color:"red"}})
  475. \li stackView.push({item:aComponent.createObject(), destroyOnPop:true})
  476. \li stackView.push([{item:anitem, immediate:true}, {item:aURL}])
  477. \endlist
  478. \note If the only argument needed is "item", you can apply the following short-
  479. hand notation: \c{stackView.push(anItem)}.
  480. Returns the item that became current.
  481. \sa initialItem
  482. \sa {Pushing items}
  483. */
  484. function push(item) {
  485. // Note: we support two different APIs in this function; The old meego API, and
  486. // the new "property list" API. Hence the reason for hiding the fact that you
  487. // can pass more arguments than shown in the signature:
  488. if (__recursionGuard(true))
  489. return
  490. var properties = arguments[1]
  491. var immediate = arguments[2]
  492. var replace = arguments[3]
  493. var arrayPushed = (item instanceof Array)
  494. var firstItem = arrayPushed ? item[0] : item
  495. immediate = (immediate || JSArray.stackView.length === 0)
  496. if (firstItem && firstItem.item && firstItem.hasOwnProperty("x") === false) {
  497. // Property list API used:
  498. immediate = immediate || firstItem.immediate
  499. replace = replace || firstItem.replace
  500. }
  501. // Create, and push, a new javascript object, called "element", onto the stack.
  502. // This element contains all the information necessary to construct the item, and
  503. // will, after loaded, also contain the loaded item:
  504. if (arrayPushed) {
  505. if (item.length === 0)
  506. return
  507. var outElement = replace ? JSArray.pop() : JSArray.current()
  508. for (var i=0; i<item.length; ++i)
  509. JSArray.push({itemComponent:item[i], loaded: false, index: __depth, properties: properties});
  510. } else {
  511. outElement = replace ? JSArray.pop() : JSArray.current()
  512. JSArray.push({itemComponent:item, loaded: false, index: __depth, properties: properties})
  513. }
  514. var currentElement = JSArray.current()
  515. var transition = {
  516. inElement: currentElement,
  517. outElement: outElement,
  518. immediate: immediate,
  519. replace: replace,
  520. push: true
  521. }
  522. __performTransition(transition)
  523. __recursionGuard(false)
  524. return __currentItem
  525. }
  526. /*! \qmlmethod Item StackView::pop(Item item = undefined)
  527. Pops one or more items off the stack.
  528. The function can also take a property list as argument - \c {Item StackView::pop(jsobject dict)},
  529. which can contain one or more of the following properties:
  530. \list
  531. \li \c item: if specified, all items down to (but not including) \a item will be
  532. popped off. If \a item is \c null, all items down to (but not including) the
  533. first item will be popped. If not specified, only the current item will be
  534. popped.
  535. \li \c immediate: set this property to \c true to skip transition effects.
  536. \endlist
  537. Examples:
  538. \list
  539. \li stackView.pop()
  540. \li stackView.pop({item:someItem, immediate: true})
  541. \li stackView.pop({immediate: true})
  542. \li stackView.pop(null)
  543. \endlist
  544. \note If the only argument needed is "item", you can apply the following short-
  545. hand notation: \c{stackView.pop(anItem)}.
  546. Returns the item that was popped off
  547. \sa clear()
  548. */
  549. function pop(item) {
  550. if (__depth <= 1)
  551. return null
  552. if (item && item.hasOwnProperty("x") === false) {
  553. // Property list API used:
  554. var immediate = (item.immediate === true)
  555. item = item.item
  556. } else {
  557. immediate = (arguments[1] === true)
  558. }
  559. if (item === __currentItem)
  560. return
  561. if (__recursionGuard(true))
  562. return
  563. var outElement = JSArray.pop()
  564. var inElement = JSArray.current()
  565. if (__depth > 1 && item !== undefined && item !== inElement.item) {
  566. // Pop from the top until we find 'item', and return the corresponding
  567. // element. Skip all non-loaded items (except the first), since no one
  568. // has any references to such items anyway:
  569. while (__depth > 1 && !JSArray.current().loaded)
  570. JSArray.pop()
  571. inElement = JSArray.current()
  572. while (__depth > 1 && item !== inElement.item) {
  573. JSArray.pop()
  574. __cleanup(inElement)
  575. while (__depth > 1 && !JSArray.current().loaded)
  576. JSArray.pop()
  577. inElement = JSArray.current()
  578. }
  579. }
  580. var transition = {
  581. inElement: inElement,
  582. outElement: outElement,
  583. immediate: immediate,
  584. replace: false,
  585. push: false
  586. }
  587. __performTransition(transition)
  588. __recursionGuard(false)
  589. return outElement.item;
  590. }
  591. /*! \qmlmethod void StackView::clear()
  592. Remove all items from the stack. No animations will be applied. */
  593. function clear() {
  594. if (__recursionGuard(true))
  595. return
  596. if (__currentTransition)
  597. __currentTransition.animation.complete()
  598. __currentItem = null
  599. var count = __depth
  600. for (var i=0; i<count; ++i) {
  601. var element = JSArray.pop()
  602. if (element.item)
  603. __cleanup(element);
  604. }
  605. __recursionGuard(false)
  606. }
  607. /*! \qmlmethod Item StackView::find(function, bool onlySearchLoadedItems = false)
  608. Search for a specific item inside the stack. \a func will
  609. be called for each item in the stack (with the item as argument)
  610. until the function returns true. Return value will be the item found. For
  611. example:
  612. find(function(item, index) { return item.isTheOne })
  613. Set \a onlySearchLoadedItems to \c true to not load items that are
  614. not loaded into memory */
  615. function find(func, onlySearchLoadedItems) {
  616. for (var i=__depth-1; i>=0; --i) {
  617. var element = JSArray.stackView[i];
  618. if (onlySearchLoadedItems !== true)
  619. __loadElement(element)
  620. else if (!element.item)
  621. continue
  622. if (func(element.item))
  623. return element.item
  624. }
  625. return null;
  626. }
  627. /*! \qmlmethod Item StackView::get(int index, bool dontLoad = false)
  628. Returns the item at position \a index in
  629. the stack. If \a dontLoad is true, the
  630. item will not be forced to load (and \c null
  631. will be returned if not yet loaded) */
  632. function get(index, dontLoad)
  633. {
  634. if (index < 0 || index >= JSArray.stackView.length)
  635. return null
  636. var element = JSArray.stackView[index]
  637. if (dontLoad !== true) {
  638. __loadElement(element)
  639. return element.item
  640. } else if (element.item) {
  641. return element.item
  642. } else {
  643. return null
  644. }
  645. }
  646. /*! \qmlmethod void StackView::completeTransition()
  647. Immediately completes any ongoing transition.
  648. /sa Animation.complete
  649. */
  650. function completeTransition()
  651. {
  652. if (__recursionGuard(true))
  653. return
  654. if (__currentTransition)
  655. __currentTransition.animation.complete()
  656. __recursionGuard(false)
  657. }
  658. /********* DEPRECATED API *********/
  659. /*! \internal
  660. \deprecated Use Push() instead */
  661. function replace(item, properties, immediate) {
  662. push(item, properties, immediate, true)
  663. }
  664. /********* PRIVATE API *********/
  665. /*! \internal The currently top-most item on the stack. */
  666. property Item __currentItem: null
  667. /*! \internal The number of items currently pushed onto the stack. */
  668. property int __depth: 0
  669. /*! \internal Stores the transition info while a transition is ongoing */
  670. property var __currentTransition: null
  671. /*! \internal Stops the user from pushing items while preparing a transition */
  672. property bool __guard: false
  673. Component.onCompleted: {
  674. if (initialItem)
  675. push(initialItem)
  676. }
  677. Component.onDestruction: {
  678. if (__currentTransition)
  679. __currentTransition.animation.complete()
  680. __currentItem = null
  681. }
  682. /*! \internal */
  683. function __recursionGuard(use)
  684. {
  685. if (use && __guard) {
  686. console.warn("Warning: StackView: You cannot push/pop recursively!")
  687. console.trace()
  688. return true
  689. }
  690. __guard = use
  691. }
  692. /*! \internal */
  693. function __loadElement(element)
  694. {
  695. if (element.loaded) {
  696. if (!element.item) {
  697. element.item = invalidItemReplacement.createObject(root)
  698. element.item.text = "\nError: The item has been deleted outside StackView!"
  699. }
  700. return
  701. }
  702. if (!element.itemComponent) {
  703. element.item = invalidItemReplacement.createObject(root)
  704. element.item.text = "\nError: Invalid item (item was 'null'). "
  705. + "This might indicate that the item was deleted outside StackView!"
  706. return
  707. }
  708. var comp = __resolveComponent(element.itemComponent, element)
  709. // Assign properties to item:
  710. if (!element.properties)
  711. element.properties = {}
  712. if (comp.hasOwnProperty("createObject")) {
  713. if (comp.status === Component.Error) {
  714. element.item = invalidItemReplacement.createObject(root)
  715. element.item.text = "\nError: Could not load: " + comp.errorString()
  716. } else {
  717. element.item = comp.createObject(root, element.properties)
  718. // Destroy items we create unless the user specified something else:
  719. if (!element.hasOwnProperty("destroyOnPop"))
  720. element.destroyOnPop = true
  721. }
  722. } else {
  723. // comp is already an Item, so just re-parent it into the StackView:
  724. element.item = comp
  725. element.originalParent = parent
  726. element.item.parent = root
  727. for (var prop in element.properties) {
  728. if (element.item.hasOwnProperty(prop))
  729. element.item[prop] = element.properties[prop];
  730. }
  731. // Do not destroy items we didn't create, unless the user specified something else:
  732. if (!element.hasOwnProperty("destroyOnPop"))
  733. element.destroyOnPop = false
  734. }
  735. element.item.Stack.__index = element.index
  736. element.item.Stack.__view = root
  737. // Let item fill all available space by default:
  738. element.item.width = Qt.binding(function() { return root.width })
  739. element.item.height = Qt.binding(function() { return root.height })
  740. element.loaded = true
  741. }
  742. /*! \internal */
  743. function __resolveComponent(unknownObjectType, element)
  744. {
  745. // We need this extra resolve function since we don't really
  746. // know what kind of object the user pushed. So we try to
  747. // figure it out by inspecting the object:
  748. if (unknownObjectType.hasOwnProperty("createObject")) {
  749. return unknownObjectType
  750. } else if (typeof unknownObjectType == "string") {
  751. return Qt.createComponent(unknownObjectType)
  752. } else if (unknownObjectType.hasOwnProperty("x")) {
  753. return unknownObjectType
  754. } else if (unknownObjectType.hasOwnProperty("item")) {
  755. // INVARIANT: user pushed a JS-object
  756. element.properties = unknownObjectType.properties
  757. if (!unknownObjectType.item)
  758. unknownObjectType.item = invalidItemReplacement
  759. if (unknownObjectType.hasOwnProperty("destroyOnPop"))
  760. element.destroyOnPop = unknownObjectType.destroyOnPop
  761. return __resolveComponent(unknownObjectType.item, element)
  762. } else {
  763. // We cannot determine the type, so assume its a URL:
  764. return Qt.createComponent(unknownObjectType)
  765. }
  766. }
  767. /*! \internal */
  768. function __cleanup(element) {
  769. // INVARIANT: element has been removed from JSArray. Destroy its
  770. // item, or re-parent it back to the parent it had before it was pushed:
  771. var item = element.item
  772. if (element.destroyOnPop) {
  773. item.destroy()
  774. } else {
  775. // Mark the item as no longer part of the StackView. It
  776. // might reenter on pop if pushed several times:
  777. item.visible = false
  778. __setStatus(item, Stack.Inactive)
  779. item.Stack.__view = null
  780. item.Stack.__index = -1
  781. if (element.originalParent)
  782. item.parent = element.originalParent
  783. }
  784. }
  785. /*! \internal */
  786. function __setStatus(item, status) {
  787. item.Stack.__status = status
  788. }
  789. /*! \internal */
  790. function __performTransition(transition)
  791. {
  792. // Animate item in "outElement" out, and item in "inElement" in. Set a guard to protect
  793. // the user from pushing new items on signals that will fire while preparing for the transition
  794. // (e.g Stack.onCompleted, Stack.onStatusChanged, Stack.onIndexChanged etc). Otherwise, we will enter
  795. // this function several times, which causes the items to be updated half-way.
  796. if (__currentTransition)
  797. __currentTransition.animation.complete()
  798. __loadElement(transition.inElement)
  799. transition.name = transition.replace ? "replaceTransition" : (transition.push ? "pushTransition" : "popTransition")
  800. var enterItem = transition.inElement.item
  801. transition.enterItem = enterItem
  802. // Since an item can be pushed several times, we need to update its properties:
  803. enterItem.parent = root
  804. enterItem.Stack.__view = root
  805. enterItem.Stack.__index = transition.inElement.index
  806. __currentItem = enterItem
  807. if (!transition.outElement) {
  808. // A transition consists of two items, but we got just one. So just show the item:
  809. enterItem.visible = true
  810. __setStatus(enterItem, Stack.Activating)
  811. __setStatus(enterItem, Stack.Active)
  812. return
  813. }
  814. var exitItem = transition.outElement.item
  815. transition.exitItem = exitItem
  816. if (enterItem === exitItem)
  817. return
  818. if (root.delegate) {
  819. transition.properties = {
  820. "name":transition.name,
  821. "enterItem":transition.enterItem,
  822. "exitItem":transition.exitItem,
  823. "immediate":transition.immediate }
  824. var anim = root.delegate.getTransition(transition.properties)
  825. if (anim.createObject) {
  826. anim = anim.createObject(null, transition.properties)
  827. anim.runningChanged.connect(function(){ if (anim.running === false) anim.destroy() })
  828. }
  829. transition.animation = anim
  830. }
  831. if (!transition.animation) {
  832. console.warn("Warning: StackView: no", transition.name, "found!")
  833. return
  834. }
  835. if (enterItem.anchors.fill || exitItem.anchors.fill)
  836. console.warn("Warning: StackView: cannot transition an item that is anchored!")
  837. __currentTransition = transition
  838. __setStatus(exitItem, Stack.Deactivating)
  839. enterItem.visible = true
  840. __setStatus(enterItem, Stack.Activating)
  841. transition.animation.runningChanged.connect(animationFinished)
  842. transition.animation.start()
  843. // NB! For empty animations, "animationFinished" is already
  844. // executed at this point, leaving __animation === null:
  845. if (transition.immediate === true && transition.animation)
  846. transition.animation.complete()
  847. }
  848. /*! \internal */
  849. function animationFinished()
  850. {
  851. if (!__currentTransition || __currentTransition.animation.running)
  852. return
  853. __currentTransition.animation.runningChanged.disconnect(animationFinished)
  854. __currentTransition.exitItem.visible = false
  855. __setStatus(__currentTransition.exitItem, Stack.Inactive);
  856. __setStatus(__currentTransition.enterItem, Stack.Active);
  857. __currentTransition.properties.animation = __currentTransition.animation
  858. root.delegate.transitionFinished(__currentTransition.properties)
  859. if (!__currentTransition.push || __currentTransition.replace)
  860. __cleanup(__currentTransition.outElement)
  861. __currentTransition = null
  862. }
  863. /*! \internal */
  864. property Component invalidItemReplacement: Component {
  865. Text {
  866. width: parent.width
  867. height: parent.height
  868. wrapMode: Text.WrapAtWordBoundaryOrAnywhere
  869. }
  870. }
  871. }