aboutsummaryrefslogtreecommitdiffstats
path: root/examples/quick/localstorage/localstorage.qml
diff options
context:
space:
mode:
authorOliver Eftevaag <oliver.eftevaag@qt.io>2021-09-01 15:40:28 +0200
committerOliver Eftevaag <oliver.eftevaag@qt.io>2021-09-03 10:39:22 +0200
commitb0e567e7e3927945c904e9a613af4b9064d70bc1 (patch)
treedd705bae2d74a62e0c1c309b2ade233e8d145989 /examples/quick/localstorage/localstorage.qml
parent5e6777be1d793740c184b4f49536b12fd83735b1 (diff)
Fix localstorage example to use a QML module
Removing the directory 'localstorage' that is inside the top-level 'localstorage' directory, and moving the content one level up in the file hierarchy. Pick-to: 6.2 Change-Id: I97cf1ff92be9860d9a92bce66c88af927618e2b0 Reviewed-by: Ulf Hermann <ulf.hermann@qt.io>
Diffstat (limited to 'examples/quick/localstorage/localstorage.qml')
-rw-r--r--examples/quick/localstorage/localstorage.qml273
1 files changed, 273 insertions, 0 deletions
diff --git a/examples/quick/localstorage/localstorage.qml b/examples/quick/localstorage/localstorage.qml
new file mode 100644
index 0000000000..15710fc55c
--- /dev/null
+++ b/examples/quick/localstorage/localstorage.qml
@@ -0,0 +1,273 @@
+/****************************************************************************
+**
+** Copyright (C) 2021 The Qt Company Ltd.
+** Contact: https://www.qt.io/licensing/
+**
+** This file is part of the documentation of the Qt Toolkit.
+**
+** $QT_BEGIN_LICENSE:BSD$
+** Commercial License Usage
+** Licensees holding valid commercial Qt licenses may use this file in
+** accordance with the commercial license agreement provided with the
+** Software or, alternatively, in accordance with the terms contained in
+** a written agreement between you and The Qt Company. For licensing terms
+** and conditions see https://www.qt.io/terms-conditions. For further
+** information use the contact form at https://www.qt.io/contact-us.
+**
+** BSD License Usage
+** Alternatively, you may use this file under the terms of the BSD license
+** as follows:
+**
+** "Redistribution and use in source and binary forms, with or without
+** modification, are permitted provided that the following conditions are
+** met:
+** * Redistributions of source code must retain the above copyright
+** notice, this list of conditions and the following disclaimer.
+** * Redistributions in binary form must reproduce the above copyright
+** notice, this list of conditions and the following disclaimer in
+** the documentation and/or other materials provided with the
+** distribution.
+** * Neither the name of The Qt Company Ltd nor the names of its
+** contributors may be used to endorse or promote products derived
+** from this software without specific prior written permission.
+**
+**
+** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE."
+**
+** $QT_END_LICENSE$
+**
+****************************************************************************/
+
+import QtQuick
+import QtQuick.Controls
+import QtQuick.Layouts
+import QtQuick.LocalStorage
+import "Database.js" as JS
+
+Window {
+ id: window
+ visible: true
+ width: Screen.width / 2
+ height: Screen.height / 1.8
+ color: "#161616"
+
+ property bool creatingNewEntry: false
+ property bool editingEntry: false
+
+ Rectangle {
+ anchors.fill: parent
+
+ ColumnLayout {
+ anchors.fill: parent
+ anchors.margins: 10
+
+ Header {
+ id: input
+ Layout.fillWidth: true
+ listView: listView
+ enabled: window.creatingNewEntry || window.editingEntry
+ }
+
+ RowLayout {
+ Button {
+ text: qsTr("New")
+ onClicked: {
+ input.initrec_new()
+ window.creatingNewEntry = true
+ listView.model.setProperty(listView.currentIndex, "id", 0)
+ }
+ }
+ Button {
+ id: saveButton
+ enabled: (window.creatingNewEntry || window.editingEntry) && listView.currentIndex != -1
+ text: qsTr("Save")
+ onClicked: {
+ var insertedRow = false;
+ if (listView.model.get(listView.currentIndex).id < 1) {
+ //insert mode
+ if (input.insertrec()) {
+ // Successfully inserted a row.
+ input.setlistview()
+ insertedRow = true
+ } else {
+ // Failed to insert a row; display an error message.
+ statustext.displayWarning(qsTr("Failed to insert row"))
+ }
+ } else {
+ // edit mode
+ input.setlistview()
+ JS.dbUpdate(listView.model.get(listView.currentIndex).date,
+ listView.model.get(listView.currentIndex).trip_desc,
+ listView.model.get(listView.currentIndex).distance,
+ listView.model.get(listView.currentIndex).id)
+ }
+
+ if (insertedRow) {
+ input.initrec()
+ window.creatingNewEntry = false
+ window.editingEntry = false
+ listView.forceLayout()
+ }
+ }
+ }
+ Button {
+ id: editButton
+ text: qsTr("Edit")
+ enabled: !window.creatingNewEntry && !window.editingEntry && listView.currentIndex != -1
+ onClicked: {
+ input.editrec(listView.model.get(listView.currentIndex).date,
+ listView.model.get(listView.currentIndex).trip_desc,
+ listView.model.get(listView.currentIndex).distance,
+ listView.model.get(listView.currentIndex).id)
+
+ window.editingEntry = true
+ }
+ }
+ Button {
+ id: deleteButton
+ text: qsTr("Delete")
+ enabled: !window.creatingNewEntry && listView.currentIndex != -1
+ onClicked: {
+ JS.dbDeleteRow(listView.model.get(listView.currentIndex).id)
+ listView.model.remove(listView.currentIndex, 1)
+ if (listView.count == 0) {
+ // ListView doesn't automatically set its currentIndex to -1
+ // when the count becomes 0.
+ listView.currentIndex = -1
+ }
+ }
+ }
+ Button {
+ id: cancelButton
+ text: qsTr("Cancel")
+ enabled: (window.creatingNewEntry || window.editingEntry) && listView.currentIndex != -1
+ onClicked: {
+ if (listView.model.get(listView.currentIndex).id === 0) {
+ // This entry had an id of 0, which means it was being created and hadn't
+ // been saved to the database yet, so we can safely remove it from the model.
+ listView.model.remove(listView.currentIndex, 1)
+ }
+ listView.forceLayout()
+ window.creatingNewEntry = false
+ window.editingEntry = false
+ input.initrec()
+ }
+ }
+ Button {
+ text: qsTr("Exit")
+ onClicked: Qt.quit()
+ }
+ }
+ Item {
+ Layout.fillWidth: true
+ height: 5
+ }
+ Label {
+ Layout.alignment: Qt.AlignCenter
+ text: qsTr("Saved activities")
+ font.pointSize: 15
+ }
+ Component {
+ id: highlightBar
+ Rectangle {
+ width: listView.currentItem !== null ? listView.currentItem.width : implicitWidth
+ height: listView.currentItem !== null ? listView.currentItem.height : implicitHeight
+ color: "lightgreen"
+ }
+ }
+ ListView {
+ id: listView
+ Layout.fillWidth: true
+ Layout.fillHeight: true
+ model: MyModel {}
+ delegate: MyDelegate {
+ width: listView.width
+ onClicked: ()=> listView.currentIndex = index
+ }
+ // Don't allow changing the currentIndex while the user is creating/editing values.
+ enabled: !window.creatingNewEntry && !window.editingEntry
+
+ highlight: highlightBar
+ highlightFollowsCurrentItem: true
+ focus: true
+ clip: true
+
+ header: Component {
+ RowLayout {
+ property var headerTitles: [qsTr("Date"), qsTr("Description"), qsTr("Distance")]
+ width: ListView.view.width
+ Repeater {
+ model: headerTitles
+ delegate: Label {
+ id: headerTitleDelegate
+ Layout.fillWidth: true
+ Layout.fillHeight: true
+ Layout.preferredWidth: 1
+ text: modelData
+ font.pointSize: 15
+ font.bold: true
+ font.underline: true
+ padding: 12
+ horizontalAlignment: Label.AlignHCenter
+ }
+ }
+ }
+ }
+ }
+ Label {
+ id: statustext
+ color: "red"
+ font.bold: true
+ font.pointSize: 20
+ opacity: 0.0
+ visible: opacity !== 0 // properly cull item if effectively invisible
+ Layout.alignment: Layout.Center
+
+ function displayWarning(text) {
+ statustext.text = text
+ statusAnim.restart()
+ }
+
+ Connections {
+ target: input
+ function onStatusMessage(msg) { statustext.displayWarning(msg); }
+ }
+
+ SequentialAnimation {
+ id: statusAnim
+
+ OpacityAnimator {
+ target: statustext
+ from: 0.0
+ to: 1.0
+ duration: 50
+ }
+
+ PauseAnimation {
+ duration: 2000
+ }
+
+ OpacityAnimator {
+ target: statustext
+ from: 1.0
+ to: 0.0
+ duration: 50
+ }
+ }
+ }
+ }
+ }
+ Component.onCompleted: {
+ JS.dbInit()
+ }
+}