//Copyright (C) 2022 Ultimaker B.V. //Cura is released under the terms of the LGPLv3 or higher. import Qt.labs.qmlmodels 1.0 import QtQuick 2.10 import QtQuick.Window 2.2 import QtQuick.Controls 2.3 import UM 1.5 as UM import Cura 1.6 as Cura import DigitalFactory 1.0 as DF Item { id: base width: parent.width height: parent.height property var fileModel: manager.digitalFactoryFileModel signal savePressed() signal selectDifferentProjectPressed() anchors { fill: parent margins: UM.Theme.getSize("default_margin").width } ProjectSummaryCard { id: projectSummaryCard anchors.top: parent.top property var selectedItem: manager.digitalFactoryProjectModel.getItem(manager.selectedProjectIndex) imageSource: selectedItem.thumbnailUrl || "../images/placeholder.svg" projectNameText: selectedItem.displayName || "" projectUsernameText: selectedItem.username || "" projectLastUpdatedText: "Last updated: " + selectedItem.lastUpdated cardMouseAreaEnabled: false } Label { id: fileNameLabel anchors.top: projectSummaryCard.bottom anchors.topMargin: UM.Theme.getSize("default_margin").height text: "Cura project name" font: UM.Theme.getFont("medium") color: UM.Theme.getColor("text") } Cura.TextField { id: dfFilenameTextfield width: parent.width anchors.left: parent.left anchors.top: fileNameLabel.bottom anchors.topMargin: UM.Theme.getSize("thin_margin").height validator: RegExpValidator { regExp: /^[\w\-\. ()]{0,255}$/ } text: PrintInformation.jobName font: fontMetrics.font height: fontMetrics.height + 2 * UM.Theme.getSize("thin_margin").height placeholderText: "Enter the name of the file." onAccepted: { if (saveButton.enabled) {saveButton.clicked()}} } FontMetrics { id: fontMetrics font: UM.Theme.getFont("medium") } Rectangle { id: projectFilesContent width: parent.width anchors.top: dfFilenameTextfield.bottom anchors.topMargin: UM.Theme.getSize("wide_margin").height anchors.bottom: selectDifferentProjectButton.top anchors.bottomMargin: UM.Theme.getSize("default_margin").width color: UM.Theme.getColor("main_background") border.width: UM.Theme.getSize("default_lining").width border.color: UM.Theme.getColor("lining") //We can't use Cura's TableView here, since in Cura >= 5.0 this uses QtQuick.TableView, while in Cura < 5.0 this uses QtControls1.TableView. //So we have to define our own. Once support for 4.13 and earlier is dropped, we can switch to Cura.TableView. Table { id: filesTableView anchors.fill: parent anchors.margins: parent.border.width allowSelection: false columnHeaders: ["Name", "Uploaded by", "Uploaded at"] model: TableModel { TableModelColumn { display: "fileName" } TableModelColumn { display: "username" } TableModelColumn { display: "uploadedAt" } rows: manager.digitalFactoryFileModel.items } } Label { id: emptyProjectLabel anchors.horizontalCenter: parent.horizontalCenter anchors.verticalCenter: parent.verticalCenter text: "Select a project to view its files." font: UM.Theme.getFont("default") color: UM.Theme.getColor("setting_category_text") Connections { target: manager function onSelectedProjectIndexChanged() { emptyProjectLabel.visible = (manager.newProjectIndex == -1) } } } Label { id: noFilesInProjectLabel anchors.horizontalCenter: parent.horizontalCenter anchors.verticalCenter: parent.verticalCenter visible: (manager.digitalFactoryFileModel.count == 0 && !emptyProjectLabel.visible && !retrievingFilesBusyIndicator.visible) text: "No supported files in this project." font: UM.Theme.getFont("default") color: UM.Theme.getColor("setting_category_text") } BusyIndicator { // Shows up while Cura is waiting to receive the files of a project from the digital factory library id: retrievingFilesBusyIndicator anchors { verticalCenter: parent.verticalCenter horizontalCenter: parent.horizontalCenter } width: parent.width / 4 height: width visible: manager.retrievingFilesStatus == DF.RetrievalStatus.InProgress running: visible palette.dark: UM.Theme.getColor("text") } Connections { target: manager.digitalFactoryFileModel function onItemsChanged() { // Make sure no files are selected when the file model changes filesTableView.currentRow = -1; } } } Cura.SecondaryButton { id: selectDifferentProjectButton anchors.bottom: parent.bottom anchors.left: parent.left text: "Change Library project" onClicked: { manager.selectedProjectIndex = -1 } busy: false } Cura.PrimaryButton { id: saveButton anchors.bottom: parent.bottom anchors.right: parent.right text: "Save" enabled: (asProjectCheckbox.checked || asSlicedCheckbox.checked) && dfFilenameTextfield.text.length >= 1 && dfFilenameTextfield.state !== 'invalid' onClicked: { let saveAsFormats = []; if (asProjectCheckbox.checked) { saveAsFormats.push("3mf"); } if (asSlicedCheckbox.checked) { saveAsFormats.push("ufp"); } manager.saveFileToSelectedProject(dfFilenameTextfield.text, saveAsFormats); } busy: false } Row { id: saveAsFormatRow anchors.verticalCenter: saveButton.verticalCenter anchors.right: saveButton.left anchors.rightMargin: UM.Theme.getSize("thin_margin").height width: childrenRect.width spacing: UM.Theme.getSize("default_margin").width UM.CheckBox { id: asProjectCheckbox height: UM.Theme.getSize("checkbox").height anchors.verticalCenter: parent.verticalCenter checked: true text: "Save Cura project" font: UM.Theme.getFont("medium") } UM.CheckBox { id: asSlicedCheckbox height: UM.Theme.getSize("checkbox").height anchors.verticalCenter: parent.verticalCenter enabled: UM.Backend.state == UM.Backend.Done checked: UM.Backend.state == UM.Backend.Done text: "Save print file" font: UM.Theme.getFont("medium") } } Component.onCompleted: { saveButton.clicked.connect(base.savePressed) selectDifferentProjectButton.clicked.connect(base.selectDifferentProjectPressed) } }