qtdeclarative/examples/quick/shared/LauncherList.qml

210 lines
5.9 KiB
QML
Raw Normal View History

// Copyright (C) 2017 Crimson AS <info@crimson.no>
// Copyright (C) 2017 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
import QtQml
import QtQml.Models
import QtQuick
Rectangle {
id: root
property int activePageCount: 0
//model is a list of {"name":"somename", "url":"file:///some/url/mainfile.qml"}
//function used to add to model A) to enforce scheme B) to allow Qt.resolveUrl in url assignments
color: "#eee"
function addExample(name, desc, url) {
myModel.append({"name":name, "description":desc, "url":url})
}
function showExample(url) {
pageComponent.createObject(pageContainer, { exampleUrl: url }).show()
}
// The container rectangle here is used to give a nice "feel" when
// transitioning into an example.
Rectangle {
anchors.fill: parent
color: "black"
ListView {
id: launcherList
clip: true
delegate: SimpleLauncherDelegate{
required property url url
onClicked: root.showExample(url)
}
model: ListModel {id:myModel}
anchors.fill: parent
enabled: opacity == 1.0
}
}
Item {
id: pageContainer
anchors.fill: parent
}
Component {
id: pageComponent
Rectangle {
id: page
clip: true
property url exampleUrl
width: parent.width
height: parent.height - bar.height
color: "white"
TapHandler {
//Eats mouse events
}
Loader{
focus: true
source: parent.exampleUrl
anchors.fill: parent
}
function show() {
showAnim.start()
}
function exit() {
exitAnim.start()
}
ParallelAnimation {
id: showAnim
ScriptAction {
script: root.activePageCount++
}
NumberAnimation {
target: launcherList
property: "opacity"
from: 1.0
to: 0.0
duration: 500
}
NumberAnimation {
target: launcherList
property: "scale"
from: 1.0
to: 0.0
duration: 500
}
NumberAnimation {
target: page
property: "x"
from: -page.width
to: 0
duration: 300
}
}
SequentialAnimation {
id: exitAnim
ScriptAction {
script: root.activePageCount--
}
ParallelAnimation {
NumberAnimation {
target: launcherList
property: "opacity"
from: 0.0
to: 1.0
duration: 300
}
NumberAnimation {
target: launcherList
property: "scale"
from: 0.0
to: 1.0
duration: 300
}
NumberAnimation {
target: page
property: "x"
from: 0
to: -page.width
duration: 300
}
}
ScriptAction {
script: page.destroy()
}
}
}
}
Rectangle {
id: bar
visible: height > 0
anchors.bottom: parent.bottom
width: parent.width
height: root.activePageCount > 0 ? 40 : 0
Behavior on height {
NumberAnimation {
duration: 300
}
}
Rectangle {
height: 1
color: "#ccc"
anchors.top: parent.top
anchors.left: parent.left
anchors.right: parent.right
}
Rectangle {
height: 1
color: "#fff"
anchors.top: parent.top
anchors.topMargin: 1
anchors.left: parent.left
anchors.right: parent.right
}
gradient: Gradient {
GradientStop { position: 0 ; color: "#eee" }
GradientStop { position: 1 ; color: "#ccc" }
}
Image {
id: back
source: "images/back.png"
anchors.verticalCenter: parent.verticalCenter
anchors.verticalCenterOffset: 1
anchors.left: parent.left
anchors.leftMargin: 6
width: 38
height: 31
fillMode: Image.Pad
horizontalAlignment: Image.AlignHCenter
verticalAlignment: Image.AlignVCenter
TapHandler {
id: tapHandler
enabled: root.activePageCount > 0
Make TapHandler longPressed/tapped exclusive and reliable; fix example The back button in the examples' LauncherList.qml has been flaky. As described in the docs, a TapHandler used to implement a Button should have `gesturePolicy: TapHandler.ReleaseWithinBounds` to get the common behavior that you can drag out of the button to cancel the click, and you can also drag back into the button to change your mind and let it click after all. But when trying to test this behavior, another problem became evident: if you spend a longer time than longPressThreshold for the whole gesture, then at the time of release you could see the debug output "long press threshold exceeded" and the tapped signal was not emitted. Our intention was that if you are dragging around, the TapHandler is not eligible to emit the longPressed signal; it follows that it should not become ineligible to emit tapped, either (tapped can be emitted if other constraints are satisfied). The intention of the ReleaseWithinBounds policy is that it doesn't matter how much you drag, as long as the point is within the bounds of the parent at the time of release. So we begin keeping track of whether we have actually emitted the longPressed signal, rather than merely looking at the time difference. This changed behavior in tst_qquickdeliveryagent::passiveGrabberOrder: 1 second is more than enough time for long press with the default longPressThreshold, and now the tapped signals are no longer emitted after longPressed. So we just wait for pressed state rather than waiting so long. qWaits in tests are best avoided anyway (although I think the intention in 152e12dc22cc0fd07cf90bcd35ae0e05b8b46fa0 might have been to wait long enough to ensure that nothing undesired would occur, rather than waiting for something specific to occur). Task-number: QTBUG-65012 Task-number: QTBUG-105810 Change-Id: If6a86d955e19810cb06de659f5e39b50a72fa762 Reviewed-by: Richard Moe Gustavsen <richard.gustavsen@qt.io> (cherry picked from commit 1b166c87d06ef53de9ccc76218dfae7b0359c5e0) Reviewed-by: Shawn Rutledge <shawn.rutledge@qt.io> (cherry picked from commit 6e3c25f94426ec449d4ce48f92e4df46fb7c60c2) Reviewed-by: Qt Cherry-pick Bot <cherrypick_bot@qt-project.org>
2023-11-14 17:35:56 +00:00
gesturePolicy: TapHandler.ReleaseWithinBounds
longPressThreshold: 0
onTapped: {
pageContainer.children[pageContainer.children.length - 1].exit()
}
}
Rectangle {
anchors.fill: parent
opacity: tapHandler.pressed ? 1 : 0
Behavior on opacity { NumberAnimation{ duration: 100 }}
gradient: Gradient {
GradientStop { position: 0 ; color: "#22000000" }
GradientStop { position: 0.2 ; color: "#11000000" }
}
border.color: "darkgray"
antialiasing: true
radius: 4
}
}
}
}