// Copyright (c) 2021 Ultimaker B.V. // Cura is released under the terms of the LGPLv3 or higher. import QtQuick 2.15 import QtQuick.Controls 2.15 import UM 1.4 as UM ScrollView { id: packages clip: true ScrollBar.horizontal.policy: ScrollBar.AlwaysOff property alias model: packagesListview.model property string pageTitle Component.onCompleted: model.updatePackages() Component.onDestruction: model.abortUpdating() ListView { id: packagesListview width: parent.width spacing: UM.Theme.getSize("default_margin").height section.property: "package.sectionTitle" section.delegate: Rectangle { width: packagesListview.width height: sectionHeaderText.height + UM.Theme.getSize("default_margin").height color: UM.Theme.getColor("detail_background") required property string section Label { id: sectionHeaderText anchors.verticalCenter: parent.verticalCenter anchors.left: parent.left text: parent.section font: UM.Theme.getFont("large") color: UM.Theme.getColor("text") } } delegate: Rectangle { width: packagesListview.width height: UM.Theme.getSize("card").height color: UM.Theme.getColor("main_background") radius: UM.Theme.getSize("default_radius").width Label { anchors.verticalCenter: parent.verticalCenter anchors.left: parent.left anchors.leftMargin: Math.round((parent.height - height) / 2) text: model.package.displayName font: UM.Theme.getFont("medium_bold") color: UM.Theme.getColor("text") } } //Wrapper item to add spacing between content and footer. footer: Item { width: parent.width height: model.hasFooter || packages.model.errorMessage != "" ? UM.Theme.getSize("card").height + packagesListview.spacing : 0 visible: model.hasFooter || packages.model.errorMessage != "" Button { id: loadMoreButton width: parent.width height: UM.Theme.getSize("card").height anchors.bottom: parent.bottom enabled: packages.model.hasMore && !packages.model.isLoading || packages.model.errorMessage != "" onClicked: packages.model.updatePackages() //Load next page in plug-in list. background: Rectangle { anchors.fill: parent radius: UM.Theme.getSize("default_radius").width color: UM.Theme.getColor("main_background") } Row { anchors.centerIn: parent spacing: UM.Theme.getSize("thin_margin").width states: [ State { name: "Error" when: packages.model.errorMessage != "" PropertyChanges { target: errorIcon visible: true } PropertyChanges { target: loadMoreIcon visible: false } PropertyChanges { target: loadMoreLabel text: catalog.i18nc("@button", "Failed to load packages:") + " " + packages.model.errorMessage + "\n" + catalog.i18nc("@button", "Retry?") } }, State { name: "Loading" when: packages.model.isLoading PropertyChanges { target: loadMoreIcon source: UM.Theme.getIcon("ArrowDoubleCircleRight") color: UM.Theme.getColor("action_button_disabled_text") } PropertyChanges { target: loadMoreLabel text: catalog.i18nc("@button", "Loading") color: UM.Theme.getColor("action_button_disabled_text") } }, State { name: "LastPage" when: !packages.model.hasMore PropertyChanges { target: loadMoreIcon visible: false } PropertyChanges { target: loadMoreLabel text: catalog.i18nc("@button", "No more results to load") color: UM.Theme.getColor("action_button_disabled_text") } } ] Item { width: (errorIcon.visible || loadMoreIcon.visible) ? UM.Theme.getSize("small_button_icon").width : 0 height: UM.Theme.getSize("small_button_icon").height anchors.verticalCenter: loadMoreLabel.verticalCenter UM.StatusIcon { id: errorIcon anchors.fill: parent status: UM.StatusIcon.Status.ERROR visible: false } UM.RecolorImage { id: loadMoreIcon anchors.fill: parent source: UM.Theme.getIcon("ArrowDown") color: UM.Theme.getColor("secondary_button_text") RotationAnimator { target: loadMoreIcon from: 0 to: 360 duration: 1000 loops: Animation.Infinite running: packages.model.isLoading alwaysRunToEnd: true } } } Label { id: loadMoreLabel text: catalog.i18nc("@button", "Load more") font: UM.Theme.getFont("medium_bold") color: UM.Theme.getColor("secondary_button_text") } } } } } }