mirror of
https://git.mirrors.martin98.com/https://github.com/Ultimaker/Cura
synced 2025-04-29 07:14:24 +08:00
Merge pull request #9012 from Ultimaker/CURA-7868_Introduce_file_provider_plugin_type
CURA-7868: Introduce the file provider plugin type
This commit is contained in:
commit
d16c5f1991
@ -81,7 +81,8 @@ class Account(QObject):
|
|||||||
CLIENT_ID="um----------------------------ultimaker_cura",
|
CLIENT_ID="um----------------------------ultimaker_cura",
|
||||||
CLIENT_SCOPES="account.user.read drive.backup.read drive.backup.write packages.download "
|
CLIENT_SCOPES="account.user.read drive.backup.read drive.backup.write packages.download "
|
||||||
"packages.rating.read packages.rating.write connect.cluster.read connect.cluster.write "
|
"packages.rating.read packages.rating.write connect.cluster.read connect.cluster.write "
|
||||||
"cura.printjob.read cura.printjob.write cura.mesh.read cura.mesh.write",
|
"library.project.read library.project.write cura.printjob.read cura.printjob.write "
|
||||||
|
"cura.mesh.read cura.mesh.write",
|
||||||
AUTH_DATA_PREFERENCE_KEY="general/ultimaker_auth_data",
|
AUTH_DATA_PREFERENCE_KEY="general/ultimaker_auth_data",
|
||||||
AUTH_SUCCESS_REDIRECT="{}/app/auth-success".format(self._oauth_root),
|
AUTH_SUCCESS_REDIRECT="{}/app/auth-success".format(self._oauth_root),
|
||||||
AUTH_FAILED_REDIRECT="{}/app/auth-error".format(self._oauth_root)
|
AUTH_FAILED_REDIRECT="{}/app/auth-error".format(self._oauth_root)
|
||||||
|
@ -30,6 +30,7 @@ from UM.Operations.SetTransformOperation import SetTransformOperation
|
|||||||
from UM.Platform import Platform
|
from UM.Platform import Platform
|
||||||
from UM.PluginError import PluginNotFoundError
|
from UM.PluginError import PluginNotFoundError
|
||||||
from UM.Preferences import Preferences
|
from UM.Preferences import Preferences
|
||||||
|
from UM.Qt.Bindings.FileProviderModel import FileProviderModel
|
||||||
from UM.Qt.QtApplication import QtApplication # The class we're inheriting from.
|
from UM.Qt.QtApplication import QtApplication # The class we're inheriting from.
|
||||||
from UM.Resources import Resources
|
from UM.Resources import Resources
|
||||||
from UM.Scene.Camera import Camera
|
from UM.Scene.Camera import Camera
|
||||||
@ -822,6 +823,9 @@ class CuraApplication(QtApplication):
|
|||||||
self._add_printer_pages_model_without_cancel.initialize(cancellable = False)
|
self._add_printer_pages_model_without_cancel.initialize(cancellable = False)
|
||||||
self._whats_new_pages_model.initialize()
|
self._whats_new_pages_model.initialize()
|
||||||
|
|
||||||
|
# Initialize the FileProviderModel
|
||||||
|
self._file_provider_model.initialize(self._onFileProviderEnabledChanged)
|
||||||
|
|
||||||
# Detect in which mode to run and execute that mode
|
# Detect in which mode to run and execute that mode
|
||||||
if self._is_headless:
|
if self._is_headless:
|
||||||
self.runWithoutGUI()
|
self.runWithoutGUI()
|
||||||
@ -1051,6 +1055,13 @@ class CuraApplication(QtApplication):
|
|||||||
self._simple_mode_settings_manager = SimpleModeSettingsManager()
|
self._simple_mode_settings_manager = SimpleModeSettingsManager()
|
||||||
return self._simple_mode_settings_manager
|
return self._simple_mode_settings_manager
|
||||||
|
|
||||||
|
@pyqtSlot(result = QObject)
|
||||||
|
def getFileProviderModel(self) -> FileProviderModel:
|
||||||
|
return self._file_provider_model
|
||||||
|
|
||||||
|
def _onFileProviderEnabledChanged(self):
|
||||||
|
self._file_provider_model.update()
|
||||||
|
|
||||||
def event(self, event):
|
def event(self, event):
|
||||||
"""Handle Qt events"""
|
"""Handle Qt events"""
|
||||||
|
|
||||||
|
@ -416,9 +416,13 @@ Item
|
|||||||
Action
|
Action
|
||||||
{
|
{
|
||||||
id: openAction;
|
id: openAction;
|
||||||
|
property var fileProviderModel: CuraApplication.getFileProviderModel()
|
||||||
|
|
||||||
text: catalog.i18nc("@action:inmenu menubar:file","&Open File(s)...");
|
text: catalog.i18nc("@action:inmenu menubar:file","&Open File(s)...");
|
||||||
iconName: "document-open";
|
iconName: "document-open";
|
||||||
shortcut: StandardKey.Open;
|
// Unassign the shortcut when there are more than one file providers, since then the file provider's shortcut is
|
||||||
|
// enabled instead, and Ctrl+O is assigned to the local file provider
|
||||||
|
shortcut: fileProviderModel.count == 1 ? StandardKey.Open : "";
|
||||||
}
|
}
|
||||||
|
|
||||||
Action
|
Action
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
// Copyright (c) 2020 Ultimaker B.V.
|
// Copyright (c) 2021 Ultimaker B.V.
|
||||||
// Cura is released under the terms of the LGPLv3 or higher.
|
// Cura is released under the terms of the LGPLv3 or higher.
|
||||||
|
|
||||||
import QtQuick 2.7
|
import QtQuick 2.7
|
||||||
@ -149,15 +149,6 @@ UM.MainWindow
|
|||||||
id: backgroundItem
|
id: backgroundItem
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
|
|
||||||
signal hasMesh(string name) //this signal sends the filebase name so it can be used for the JobSpecs.qml
|
|
||||||
function getMeshName(path)
|
|
||||||
{
|
|
||||||
//takes the path the complete path of the meshname and returns only the filebase
|
|
||||||
var fileName = path.slice(path.lastIndexOf("/") + 1)
|
|
||||||
var fileBase = fileName.slice(0, fileName.indexOf("."))
|
|
||||||
return fileBase
|
|
||||||
}
|
|
||||||
|
|
||||||
//DeleteSelection on the keypress backspace event
|
//DeleteSelection on the keypress backspace event
|
||||||
Keys.onPressed:
|
Keys.onPressed:
|
||||||
{
|
{
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
// Copyright (c) 2015 Ultimaker B.V.
|
// Copyright (c) 2021 Ultimaker B.V.
|
||||||
// Cura is released under the terms of the LGPLv3 or higher.
|
// Cura is released under the terms of the LGPLv3 or higher.
|
||||||
|
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
@ -32,30 +32,24 @@ UM.Dialog
|
|||||||
|
|
||||||
// load the entire project
|
// load the entire project
|
||||||
function loadProjectFile() {
|
function loadProjectFile() {
|
||||||
|
|
||||||
// update preference
|
// update preference
|
||||||
if (rememberChoiceCheckBox.checked) {
|
if (rememberChoiceCheckBox.checked) {
|
||||||
UM.Preferences.setValue("cura/choice_on_open_project", "open_as_project")
|
UM.Preferences.setValue("cura/choice_on_open_project", "open_as_project")
|
||||||
}
|
}
|
||||||
|
|
||||||
UM.WorkspaceFileHandler.readLocalFile(base.fileUrl)
|
UM.WorkspaceFileHandler.readLocalFile(base.fileUrl)
|
||||||
var meshName = backgroundItem.getMeshName(base.fileUrl.toString())
|
|
||||||
backgroundItem.hasMesh(decodeURIComponent(meshName))
|
|
||||||
|
|
||||||
base.hide()
|
base.hide()
|
||||||
}
|
}
|
||||||
|
|
||||||
// load the project file as separated models
|
// load the project file as separated models
|
||||||
function loadModelFiles() {
|
function loadModelFiles() {
|
||||||
|
|
||||||
// update preference
|
// update preference
|
||||||
if (rememberChoiceCheckBox.checked) {
|
if (rememberChoiceCheckBox.checked) {
|
||||||
UM.Preferences.setValue("cura/choice_on_open_project", "open_as_model")
|
UM.Preferences.setValue("cura/choice_on_open_project", "open_as_model")
|
||||||
}
|
}
|
||||||
|
|
||||||
CuraApplication.readLocalFile(base.fileUrl, "open_as_model")
|
CuraApplication.readLocalFile(base.fileUrl, "open_as_model")
|
||||||
var meshName = backgroundItem.getMeshName(base.fileUrl.toString())
|
|
||||||
backgroundItem.hasMesh(decodeURIComponent(meshName))
|
|
||||||
|
|
||||||
base.hide()
|
base.hide()
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
// Copyright (c) 2017 Ultimaker B.V.
|
// Copyright (c) 2021 Ultimaker B.V.
|
||||||
// Cura is released under the terms of the LGPLv3 or higher.
|
// Cura is released under the terms of the LGPLv3 or higher.
|
||||||
|
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
@ -33,9 +33,6 @@ UM.Dialog
|
|||||||
function loadProjectFile(projectFile)
|
function loadProjectFile(projectFile)
|
||||||
{
|
{
|
||||||
UM.WorkspaceFileHandler.readLocalFile(projectFile);
|
UM.WorkspaceFileHandler.readLocalFile(projectFile);
|
||||||
|
|
||||||
var meshName = backgroundItem.getMeshName(projectFile.toString());
|
|
||||||
backgroundItem.hasMesh(decodeURIComponent(meshName));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function loadModelFiles(fileUrls)
|
function loadModelFiles(fileUrls)
|
||||||
@ -44,9 +41,6 @@ UM.Dialog
|
|||||||
{
|
{
|
||||||
CuraApplication.readLocalFile(fileUrls[i], "open_as_model");
|
CuraApplication.readLocalFile(fileUrls[i], "open_as_model");
|
||||||
}
|
}
|
||||||
|
|
||||||
var meshName = backgroundItem.getMeshName(fileUrls[0].toString());
|
|
||||||
backgroundItem.hasMesh(decodeURIComponent(meshName));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Column
|
Column
|
||||||
@ -108,5 +102,11 @@ UM.Dialog
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
UM.I18nCatalog
|
||||||
|
{
|
||||||
|
id: catalog
|
||||||
|
name: "cura"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -11,6 +11,7 @@ Menu
|
|||||||
{
|
{
|
||||||
id: base
|
id: base
|
||||||
title: catalog.i18nc("@title:menu menubar:toplevel", "&File")
|
title: catalog.i18nc("@title:menu menubar:toplevel", "&File")
|
||||||
|
property var fileProviderModel: CuraApplication.getFileProviderModel()
|
||||||
|
|
||||||
MenuItem
|
MenuItem
|
||||||
{
|
{
|
||||||
@ -22,6 +23,13 @@ Menu
|
|||||||
{
|
{
|
||||||
id: openMenu
|
id: openMenu
|
||||||
action: Cura.Actions.open
|
action: Cura.Actions.open
|
||||||
|
visible: (base.fileProviderModel.count == 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
OpenFilesMenu
|
||||||
|
{
|
||||||
|
id: openFilesMenu
|
||||||
|
visible: (base.fileProviderModel.count > 1)
|
||||||
}
|
}
|
||||||
|
|
||||||
RecentFilesMenu { }
|
RecentFilesMenu { }
|
||||||
|
46
resources/qml/Menus/OpenFilesMenu.qml
Normal file
46
resources/qml/Menus/OpenFilesMenu.qml
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
// Copyright (c) 2020 Ultimaker B.V.
|
||||||
|
// Cura is released under the terms of the LGPLv3 or higher.
|
||||||
|
|
||||||
|
import QtQuick 2.2
|
||||||
|
import QtQuick.Controls 1.1
|
||||||
|
|
||||||
|
import UM 1.6 as UM
|
||||||
|
import Cura 1.0 as Cura
|
||||||
|
|
||||||
|
import "../Dialogs"
|
||||||
|
|
||||||
|
Menu
|
||||||
|
{
|
||||||
|
id: openFilesMenu
|
||||||
|
title: catalog.i18nc("@title:menu menubar:file", "Open File(s)...")
|
||||||
|
iconName: "document-open-recent";
|
||||||
|
|
||||||
|
Instantiator
|
||||||
|
{
|
||||||
|
id: fileProviders
|
||||||
|
model: CuraApplication.getFileProviderModel()
|
||||||
|
MenuItem
|
||||||
|
{
|
||||||
|
text:
|
||||||
|
{
|
||||||
|
return model.displayText;
|
||||||
|
}
|
||||||
|
onTriggered:
|
||||||
|
{
|
||||||
|
if (model.index == 0) // The 0th element is the "From Disk" option, which should activate the open local file dialog
|
||||||
|
{
|
||||||
|
Cura.Actions.open.trigger()
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
CuraApplication.getFileProviderModel().trigger(model.name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Unassign the shortcuts when the submenu is invisible (i.e. when there is only one file provider) to avoid ambiguous shortcuts.
|
||||||
|
// When there is a signle file provider, the openAction is assigned with the Ctrl+O shortcut instead.
|
||||||
|
shortcut: openFilesMenu.visible ? model.shortcut : ""
|
||||||
|
}
|
||||||
|
onObjectAdded: openFilesMenu.insertItem(index, object)
|
||||||
|
onObjectRemoved: openFilesMenu.removeItem(object)
|
||||||
|
}
|
||||||
|
}
|
@ -1,4 +1,4 @@
|
|||||||
// Copyright (c) 2016 Ultimaker B.V.
|
// Copyright (c) 2021 Ultimaker B.V.
|
||||||
// Cura is released under the terms of the LGPLv3 or higher.
|
// Cura is released under the terms of the LGPLv3 or higher.
|
||||||
|
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
@ -27,13 +27,7 @@ Menu
|
|||||||
var path = decodeURIComponent(modelData.toString())
|
var path = decodeURIComponent(modelData.toString())
|
||||||
return (index + 1) + ". " + path.slice(path.lastIndexOf("/") + 1);
|
return (index + 1) + ". " + path.slice(path.lastIndexOf("/") + 1);
|
||||||
}
|
}
|
||||||
onTriggered:
|
onTriggered: CuraApplication.readLocalFile(modelData)
|
||||||
{
|
|
||||||
CuraApplication.readLocalFile(modelData);
|
|
||||||
|
|
||||||
var meshName = backgroundItem.getMeshName(modelData.toString())
|
|
||||||
backgroundItem.hasMesh(decodeURIComponent(meshName))
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
onObjectAdded: menu.insertItem(index, object)
|
onObjectAdded: menu.insertItem(index, object)
|
||||||
onObjectRemoved: menu.removeItem(object)
|
onObjectRemoved: menu.removeItem(object)
|
||||||
|
68
resources/qml/TableView.qml
Normal file
68
resources/qml/TableView.qml
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
// Copyright (C) 2021 Ultimaker B.V.
|
||||||
|
// Cura is released under the terms of the LGPLv3 or higher.
|
||||||
|
|
||||||
|
import QtQuick 2.10
|
||||||
|
import QtQuick.Controls 1.4 as OldControls // TableView doesn't exist in the QtQuick Controls 2.x in 5.10, so use the old one
|
||||||
|
import QtQuick.Controls 2.3
|
||||||
|
import QtQuick.Controls.Styles 1.4
|
||||||
|
|
||||||
|
import UM 1.2 as UM
|
||||||
|
|
||||||
|
|
||||||
|
OldControls.TableView
|
||||||
|
{
|
||||||
|
itemDelegate: Item
|
||||||
|
{
|
||||||
|
height: tableCellLabel.implicitHeight
|
||||||
|
|
||||||
|
Label
|
||||||
|
{
|
||||||
|
id: tableCellLabel
|
||||||
|
color: UM.Theme.getColor("text")
|
||||||
|
elide: Text.ElideRight
|
||||||
|
text: styleData.value
|
||||||
|
anchors.fill: parent
|
||||||
|
anchors.leftMargin: 10 * screenScaleFactor
|
||||||
|
verticalAlignment: Text.AlignVCenter
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
rowDelegate: Rectangle
|
||||||
|
{
|
||||||
|
color: styleData.selected ? UM.Theme.getColor("secondary") : UM.Theme.getColor("main_background")
|
||||||
|
height: UM.Theme.getSize("table_row").height
|
||||||
|
}
|
||||||
|
|
||||||
|
// Use the old styling technique since it's the only way to make the scrollbars themed in the TableView
|
||||||
|
style: TableViewStyle
|
||||||
|
{
|
||||||
|
backgroundColor: UM.Theme.getColor("main_background")
|
||||||
|
|
||||||
|
handle: Rectangle
|
||||||
|
{
|
||||||
|
// Both implicit width and height have to be set, since the handle is used by both the horizontal and the vertical scrollbars
|
||||||
|
implicitWidth: UM.Theme.getSize("scrollbar").width
|
||||||
|
implicitHeight: UM.Theme.getSize("scrollbar").width
|
||||||
|
radius: width / 2
|
||||||
|
color: UM.Theme.getColor(styleData.pressed ? "scrollbar_handle_down" : (styleData.hovered ? "scrollbar_handle_hover" : "scrollbar_handle"))
|
||||||
|
}
|
||||||
|
|
||||||
|
scrollBarBackground: Rectangle
|
||||||
|
{
|
||||||
|
// Both implicit width and height have to be set, since the handle is used by both the horizontal and the vertical scrollbars
|
||||||
|
implicitWidth: UM.Theme.getSize("scrollbar").width
|
||||||
|
implicitHeight: UM.Theme.getSize("scrollbar").width
|
||||||
|
color: UM.Theme.getColor("main_background")
|
||||||
|
}
|
||||||
|
|
||||||
|
// The little rectangle between the vertical and horizontal scrollbars
|
||||||
|
corner: Rectangle
|
||||||
|
{
|
||||||
|
color: UM.Theme.getColor("main_background")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Override the control arrows
|
||||||
|
incrementControl: Item { }
|
||||||
|
decrementControl: Item { }
|
||||||
|
}
|
||||||
|
}
|
@ -630,6 +630,8 @@
|
|||||||
"monitor_external_link_icon": [1.16, 1.16],
|
"monitor_external_link_icon": [1.16, 1.16],
|
||||||
"monitor_column": [18.0, 1.0],
|
"monitor_column": [18.0, 1.0],
|
||||||
"monitor_progress_bar": [16.5, 1.0],
|
"monitor_progress_bar": [16.5, 1.0],
|
||||||
"monitor_margin": [1.5, 1.5]
|
"monitor_margin": [1.5, 1.5],
|
||||||
|
|
||||||
|
"table_row": [2.0, 2.0]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user