mirror of
https://github.com/Ultimaker/Cura.git
synced 2025-07-07 06:57:28 -06:00
Merge branch '4.0'
This commit is contained in:
commit
650204dae7
19 changed files with 95 additions and 66 deletions
|
@ -1,6 +1,6 @@
|
||||||
# Copyright (c) 2018 Ultimaker B.V.
|
# Copyright (c) 2018 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.
|
||||||
|
from UM.Scene.Camera import Camera
|
||||||
from cura.Scene.CuraSceneNode import CuraSceneNode
|
from cura.Scene.CuraSceneNode import CuraSceneNode
|
||||||
from cura.Settings.ExtruderManager import ExtruderManager
|
from cura.Settings.ExtruderManager import ExtruderManager
|
||||||
from UM.Application import Application #To modify the maximum zoom level.
|
from UM.Application import Application #To modify the maximum zoom level.
|
||||||
|
@ -112,8 +112,6 @@ class BuildVolume(SceneNode):
|
||||||
self._setting_change_timer.setSingleShot(True)
|
self._setting_change_timer.setSingleShot(True)
|
||||||
self._setting_change_timer.timeout.connect(self._onSettingChangeTimerFinished)
|
self._setting_change_timer.timeout.connect(self._onSettingChangeTimerFinished)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
# Must be after setting _build_volume_message, apparently that is used in getMachineManager.
|
# Must be after setting _build_volume_message, apparently that is used in getMachineManager.
|
||||||
# activeQualityChanged is always emitted after setActiveVariant, setActiveMaterial and setActiveQuality.
|
# activeQualityChanged is always emitted after setActiveVariant, setActiveMaterial and setActiveQuality.
|
||||||
# Therefore this works.
|
# Therefore this works.
|
||||||
|
@ -131,7 +129,9 @@ class BuildVolume(SceneNode):
|
||||||
|
|
||||||
def _onSceneChanged(self, source):
|
def _onSceneChanged(self, source):
|
||||||
if self._global_container_stack:
|
if self._global_container_stack:
|
||||||
self._scene_change_timer.start()
|
# Ignore anything that is not something we can slice in the first place!
|
||||||
|
if source.callDecoration("isSliceable"):
|
||||||
|
self._scene_change_timer.start()
|
||||||
|
|
||||||
def _onSceneChangeTimerFinished(self):
|
def _onSceneChangeTimerFinished(self):
|
||||||
root = self._application.getController().getScene().getRoot()
|
root = self._application.getController().getScene().getRoot()
|
||||||
|
@ -148,7 +148,7 @@ class BuildVolume(SceneNode):
|
||||||
if active_extruder_changed is not None:
|
if active_extruder_changed is not None:
|
||||||
node.callDecoration("getActiveExtruderChangedSignal").disconnect(self._updateDisallowedAreasAndRebuild)
|
node.callDecoration("getActiveExtruderChangedSignal").disconnect(self._updateDisallowedAreasAndRebuild)
|
||||||
node.decoratorsChanged.disconnect(self._updateNodeListeners)
|
node.decoratorsChanged.disconnect(self._updateNodeListeners)
|
||||||
self._updateDisallowedAreasAndRebuild() # make sure we didn't miss anything before we updated the node listeners
|
self.rebuild()
|
||||||
|
|
||||||
self._scene_objects = new_scene_objects
|
self._scene_objects = new_scene_objects
|
||||||
self._onSettingPropertyChanged("print_sequence", "value") # Create fake event, so right settings are triggered.
|
self._onSettingPropertyChanged("print_sequence", "value") # Create fake event, so right settings are triggered.
|
||||||
|
@ -667,6 +667,7 @@ class BuildVolume(SceneNode):
|
||||||
# ``_updateDisallowedAreas`` method itself shouldn't call ``rebuild``,
|
# ``_updateDisallowedAreas`` method itself shouldn't call ``rebuild``,
|
||||||
# since there may be other changes before it needs to be rebuilt, which
|
# since there may be other changes before it needs to be rebuilt, which
|
||||||
# would hit performance.
|
# would hit performance.
|
||||||
|
|
||||||
def _updateDisallowedAreasAndRebuild(self):
|
def _updateDisallowedAreasAndRebuild(self):
|
||||||
self._updateDisallowedAreas()
|
self._updateDisallowedAreas()
|
||||||
self._updateRaftThickness()
|
self._updateRaftThickness()
|
||||||
|
|
|
@ -4,6 +4,7 @@
|
||||||
from PyQt5.QtCore import QTimer, pyqtSignal, pyqtProperty
|
from PyQt5.QtCore import QTimer, pyqtSignal, pyqtProperty
|
||||||
|
|
||||||
from UM.Application import Application
|
from UM.Application import Application
|
||||||
|
from UM.Scene.Camera import Camera
|
||||||
from UM.Scene.Selection import Selection
|
from UM.Scene.Selection import Selection
|
||||||
from UM.Qt.ListModel import ListModel
|
from UM.Qt.ListModel import ListModel
|
||||||
|
|
||||||
|
@ -34,8 +35,9 @@ class MultiBuildPlateModel(ListModel):
|
||||||
self._active_build_plate = -1
|
self._active_build_plate = -1
|
||||||
|
|
||||||
def setMaxBuildPlate(self, max_build_plate):
|
def setMaxBuildPlate(self, max_build_plate):
|
||||||
self._max_build_plate = max_build_plate
|
if self._max_build_plate != max_build_plate:
|
||||||
self.maxBuildPlateChanged.emit()
|
self._max_build_plate = max_build_plate
|
||||||
|
self.maxBuildPlateChanged.emit()
|
||||||
|
|
||||||
## Return the highest build plate number
|
## Return the highest build plate number
|
||||||
@pyqtProperty(int, notify = maxBuildPlateChanged)
|
@pyqtProperty(int, notify = maxBuildPlateChanged)
|
||||||
|
@ -43,15 +45,17 @@ class MultiBuildPlateModel(ListModel):
|
||||||
return self._max_build_plate
|
return self._max_build_plate
|
||||||
|
|
||||||
def setActiveBuildPlate(self, nr):
|
def setActiveBuildPlate(self, nr):
|
||||||
self._active_build_plate = nr
|
if self._active_build_plate != nr:
|
||||||
self.activeBuildPlateChanged.emit()
|
self._active_build_plate = nr
|
||||||
|
self.activeBuildPlateChanged.emit()
|
||||||
|
|
||||||
@pyqtProperty(int, notify = activeBuildPlateChanged)
|
@pyqtProperty(int, notify = activeBuildPlateChanged)
|
||||||
def activeBuildPlate(self):
|
def activeBuildPlate(self):
|
||||||
return self._active_build_plate
|
return self._active_build_plate
|
||||||
|
|
||||||
def _updateSelectedObjectBuildPlateNumbersDelayed(self, *args):
|
def _updateSelectedObjectBuildPlateNumbersDelayed(self, *args):
|
||||||
self._update_timer.start()
|
if not isinstance(args[0], Camera):
|
||||||
|
self._update_timer.start()
|
||||||
|
|
||||||
def _updateSelectedObjectBuildPlateNumbers(self, *args):
|
def _updateSelectedObjectBuildPlateNumbers(self, *args):
|
||||||
result = set()
|
result = set()
|
||||||
|
|
|
@ -92,7 +92,7 @@ class AuthorizationHelpers:
|
||||||
})
|
})
|
||||||
except requests.exceptions.ConnectionError:
|
except requests.exceptions.ConnectionError:
|
||||||
# Connection was suddenly dropped. Nothing we can do about that.
|
# Connection was suddenly dropped. Nothing we can do about that.
|
||||||
Logger.logException("e", "Something failed while attempting to parse the JWT token")
|
Logger.log("w", "Something failed while attempting to parse the JWT token")
|
||||||
return None
|
return None
|
||||||
if token_request.status_code not in (200, 201):
|
if token_request.status_code not in (200, 201):
|
||||||
Logger.log("w", "Could not retrieve token data from auth server: %s", token_request.text)
|
Logger.log("w", "Could not retrieve token data from auth server: %s", token_request.text)
|
||||||
|
|
|
@ -5,6 +5,7 @@ from PyQt5.QtCore import QTimer
|
||||||
|
|
||||||
from UM.Application import Application
|
from UM.Application import Application
|
||||||
from UM.Qt.ListModel import ListModel
|
from UM.Qt.ListModel import ListModel
|
||||||
|
from UM.Scene.Camera import Camera
|
||||||
from UM.Scene.Iterator.DepthFirstIterator import DepthFirstIterator
|
from UM.Scene.Iterator.DepthFirstIterator import DepthFirstIterator
|
||||||
from UM.Scene.SceneNode import SceneNode
|
from UM.Scene.SceneNode import SceneNode
|
||||||
from UM.Scene.Selection import Selection
|
from UM.Scene.Selection import Selection
|
||||||
|
@ -19,19 +20,24 @@ class ObjectsModel(ListModel):
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
super().__init__()
|
super().__init__()
|
||||||
|
|
||||||
Application.getInstance().getController().getScene().sceneChanged.connect(self._updateDelayed)
|
Application.getInstance().getController().getScene().sceneChanged.connect(self._updateSceneDelayed)
|
||||||
Application.getInstance().getPreferences().preferenceChanged.connect(self._updateDelayed)
|
Application.getInstance().getPreferences().preferenceChanged.connect(self._updateDelayed)
|
||||||
|
|
||||||
self._update_timer = QTimer()
|
self._update_timer = QTimer()
|
||||||
self._update_timer.setInterval(100)
|
self._update_timer.setInterval(200)
|
||||||
self._update_timer.setSingleShot(True)
|
self._update_timer.setSingleShot(True)
|
||||||
self._update_timer.timeout.connect(self._update)
|
self._update_timer.timeout.connect(self._update)
|
||||||
|
|
||||||
self._build_plate_number = -1
|
self._build_plate_number = -1
|
||||||
|
|
||||||
def setActiveBuildPlate(self, nr):
|
def setActiveBuildPlate(self, nr):
|
||||||
self._build_plate_number = nr
|
if self._build_plate_number != nr:
|
||||||
self._update()
|
self._build_plate_number = nr
|
||||||
|
self._update()
|
||||||
|
|
||||||
|
def _updateSceneDelayed(self, source):
|
||||||
|
if not isinstance(source, Camera):
|
||||||
|
self._update_timer.start()
|
||||||
|
|
||||||
def _updateDelayed(self, *args):
|
def _updateDelayed(self, *args):
|
||||||
self._update_timer.start()
|
self._update_timer.start()
|
||||||
|
|
|
@ -17,7 +17,6 @@ from cura.Scene import ZOffsetDecorator
|
||||||
|
|
||||||
import random # used for list shuffling
|
import random # used for list shuffling
|
||||||
|
|
||||||
|
|
||||||
class PlatformPhysics:
|
class PlatformPhysics:
|
||||||
def __init__(self, controller, volume):
|
def __init__(self, controller, volume):
|
||||||
super().__init__()
|
super().__init__()
|
||||||
|
@ -40,8 +39,9 @@ class PlatformPhysics:
|
||||||
Application.getInstance().getPreferences().addPreference("physics/automatic_drop_down", True)
|
Application.getInstance().getPreferences().addPreference("physics/automatic_drop_down", True)
|
||||||
|
|
||||||
def _onSceneChanged(self, source):
|
def _onSceneChanged(self, source):
|
||||||
if not source.getMeshData():
|
if not source.callDecoration("isSliceable"):
|
||||||
return
|
return
|
||||||
|
|
||||||
self._change_timer.start()
|
self._change_timer.start()
|
||||||
|
|
||||||
def _onChangeTimerFinished(self):
|
def _onChangeTimerFinished(self):
|
||||||
|
|
|
@ -3,6 +3,7 @@ from UM.Logger import Logger
|
||||||
from PyQt5.QtCore import Qt, pyqtSlot, QObject
|
from PyQt5.QtCore import Qt, pyqtSlot, QObject
|
||||||
from PyQt5.QtWidgets import QApplication
|
from PyQt5.QtWidgets import QApplication
|
||||||
|
|
||||||
|
from UM.Scene.Camera import Camera
|
||||||
from cura.ObjectsModel import ObjectsModel
|
from cura.ObjectsModel import ObjectsModel
|
||||||
from cura.Machines.Models.MultiBuildPlateModel import MultiBuildPlateModel
|
from cura.Machines.Models.MultiBuildPlateModel import MultiBuildPlateModel
|
||||||
|
|
||||||
|
@ -33,7 +34,7 @@ class CuraSceneController(QObject):
|
||||||
source = args[0]
|
source = args[0]
|
||||||
else:
|
else:
|
||||||
source = None
|
source = None
|
||||||
if not isinstance(source, SceneNode):
|
if not isinstance(source, SceneNode) or isinstance(source, Camera):
|
||||||
return
|
return
|
||||||
max_build_plate = self._calcMaxBuildPlate()
|
max_build_plate = self._calcMaxBuildPlate()
|
||||||
changed = False
|
changed = False
|
||||||
|
|
|
@ -10,6 +10,7 @@ from time import time
|
||||||
from typing import Any, cast, Dict, List, Optional, Set, TYPE_CHECKING
|
from typing import Any, cast, Dict, List, Optional, Set, TYPE_CHECKING
|
||||||
|
|
||||||
from UM.Backend.Backend import Backend, BackendState
|
from UM.Backend.Backend import Backend, BackendState
|
||||||
|
from UM.Scene.Camera import Camera
|
||||||
from UM.Scene.SceneNode import SceneNode
|
from UM.Scene.SceneNode import SceneNode
|
||||||
from UM.Signal import Signal
|
from UM.Signal import Signal
|
||||||
from UM.Logger import Logger
|
from UM.Logger import Logger
|
||||||
|
@ -476,7 +477,7 @@ class CuraEngineBackend(QObject, Backend):
|
||||||
#
|
#
|
||||||
# \param source The scene node that was changed.
|
# \param source The scene node that was changed.
|
||||||
def _onSceneChanged(self, source: SceneNode) -> None:
|
def _onSceneChanged(self, source: SceneNode) -> None:
|
||||||
if not isinstance(source, SceneNode):
|
if not source.callDecoration("isSliceable"):
|
||||||
return
|
return
|
||||||
|
|
||||||
# This case checks if the source node is a node that contains GCode. In this case the
|
# This case checks if the source node is a node that contains GCode. In this case the
|
||||||
|
|
|
@ -9,6 +9,7 @@ from UM.Application import Application
|
||||||
from UM.Extension import Extension
|
from UM.Extension import Extension
|
||||||
from UM.Logger import Logger
|
from UM.Logger import Logger
|
||||||
from UM.Message import Message
|
from UM.Message import Message
|
||||||
|
from UM.Scene.Camera import Camera
|
||||||
from UM.i18n import i18nCatalog
|
from UM.i18n import i18nCatalog
|
||||||
from UM.PluginRegistry import PluginRegistry
|
from UM.PluginRegistry import PluginRegistry
|
||||||
from UM.Scene.Iterator.DepthFirstIterator import DepthFirstIterator
|
from UM.Scene.Iterator.DepthFirstIterator import DepthFirstIterator
|
||||||
|
@ -35,7 +36,12 @@ class ModelChecker(QObject, Extension):
|
||||||
|
|
||||||
## Pass-through to allow UM.Signal to connect with a pyqtSignal.
|
## Pass-through to allow UM.Signal to connect with a pyqtSignal.
|
||||||
def _onChanged(self, *args, **kwargs):
|
def _onChanged(self, *args, **kwargs):
|
||||||
self.onChanged.emit()
|
# Ignore camera updates.
|
||||||
|
if len(args) == 0:
|
||||||
|
self.onChanged.emit()
|
||||||
|
return
|
||||||
|
if not isinstance(args[0], Camera):
|
||||||
|
self.onChanged.emit()
|
||||||
|
|
||||||
## Called when plug-ins are initialized.
|
## Called when plug-ins are initialized.
|
||||||
#
|
#
|
||||||
|
|
|
@ -11,7 +11,7 @@ Rectangle
|
||||||
{
|
{
|
||||||
id: viewportOverlay
|
id: viewportOverlay
|
||||||
|
|
||||||
property bool isConnected: Cura.MachineManager.activeMachineHasActiveNetworkConnection || Cura.MachineManager.activeMachineHasActiveCloudConnection
|
property bool isConnected: Cura.MachineManager.activeMachineHasNetworkConnection || Cura.MachineManager.activeMachineHasCloudConnection
|
||||||
property bool isNetworkConfigurable: ["Ultimaker 3", "Ultimaker 3 Extended", "Ultimaker S5"].indexOf(Cura.MachineManager.activeMachineDefinitionName) > -1
|
property bool isNetworkConfigurable: ["Ultimaker 3", "Ultimaker 3 Extended", "Ultimaker S5"].indexOf(Cura.MachineManager.activeMachineDefinitionName) > -1
|
||||||
property bool isNetworkConfigured:
|
property bool isNetworkConfigured:
|
||||||
{
|
{
|
||||||
|
|
|
@ -7,34 +7,39 @@ import QtQuick.Controls.Styles 1.3
|
||||||
import UM 1.3 as UM
|
import UM 1.3 as UM
|
||||||
import Cura 1.0 as Cura
|
import Cura 1.0 as Cura
|
||||||
|
|
||||||
Rectangle {
|
Rectangle
|
||||||
|
{
|
||||||
id: base
|
id: base
|
||||||
|
|
||||||
property var enabled: true
|
property var enabled: true
|
||||||
|
|
||||||
property var iconSource: null;
|
property var iconSource: null
|
||||||
color: UM.Theme.getColor("monitor_icon_primary")
|
color: enabled ? UM.Theme.getColor("monitor_icon_primary") : UM.Theme.getColor("monitor_icon_disabled")
|
||||||
height: width;
|
height: width
|
||||||
radius: Math.round(0.5 * width);
|
radius: Math.round(0.5 * width)
|
||||||
width: 24 * screenScaleFactor;
|
width: 24 * screenScaleFactor
|
||||||
|
|
||||||
UM.RecolorImage {
|
UM.RecolorImage
|
||||||
id: icon;
|
{
|
||||||
anchors {
|
id: icon
|
||||||
horizontalCenter: parent.horizontalCenter;
|
anchors
|
||||||
verticalCenter: parent.verticalCenter;
|
{
|
||||||
|
horizontalCenter: parent.horizontalCenter
|
||||||
|
verticalCenter: parent.verticalCenter
|
||||||
}
|
}
|
||||||
color: UM.Theme.getColor("monitor_icon_accent");
|
color: UM.Theme.getColor("monitor_icon_accent")
|
||||||
height: width;
|
height: width
|
||||||
source: iconSource;
|
source: iconSource
|
||||||
width: Math.round(parent.width / 2);
|
width: Math.round(parent.width / 2)
|
||||||
}
|
}
|
||||||
|
|
||||||
MouseArea {
|
MouseArea
|
||||||
id: clickArea;
|
{
|
||||||
anchors.fill: parent;
|
id: clickArea
|
||||||
|
anchors.fill: parent
|
||||||
hoverEnabled: base.enabled
|
hoverEnabled: base.enabled
|
||||||
onClicked: {
|
onClicked:
|
||||||
|
{
|
||||||
if (base.enabled)
|
if (base.enabled)
|
||||||
{
|
{
|
||||||
if (OutputDevice.activeCameraUrl != "")
|
if (OutputDevice.activeCameraUrl != "")
|
||||||
|
|
|
@ -11,14 +11,14 @@ Button
|
||||||
id: base
|
id: base
|
||||||
background: Rectangle
|
background: Rectangle
|
||||||
{
|
{
|
||||||
color: UM.Theme.getColor("viewport_background") // TODO: Theme!
|
color: enabled ? UM.Theme.getColor("viewport_background") : "transparent"
|
||||||
height: base.height
|
height: base.height
|
||||||
opacity: base.down || base.hovered ? 1 : 0
|
opacity: base.down || base.hovered ? 1 : 0
|
||||||
radius: Math.round(0.5 * width)
|
radius: Math.round(0.5 * width)
|
||||||
width: base.width
|
width: base.width
|
||||||
}
|
}
|
||||||
contentItem: Label {
|
contentItem: Label {
|
||||||
color: UM.Theme.getColor("monitor_text_primary")
|
color: enabled ? UM.Theme.getColor("monitor_text_primary") : UM.Theme.getColor("monitor_text_disabled")
|
||||||
font.pixelSize: 32 * screenScaleFactor
|
font.pixelSize: 32 * screenScaleFactor
|
||||||
horizontalAlignment: Text.AlignHCenter
|
horizontalAlignment: Text.AlignHCenter
|
||||||
text: base.text
|
text: base.text
|
||||||
|
|
|
@ -14,6 +14,10 @@ import Cura 1.0 as Cura
|
||||||
*/
|
*/
|
||||||
Item
|
Item
|
||||||
{
|
{
|
||||||
|
// If the printer is a cloud printer or not. Other items base their enabled state off of this boolean. In the future
|
||||||
|
// they might not need to though.
|
||||||
|
property bool cloudConnection: Cura.MachineManager.activeMachineIsUsingCloudConnection
|
||||||
|
|
||||||
Label
|
Label
|
||||||
{
|
{
|
||||||
id: queuedLabel
|
id: queuedLabel
|
||||||
|
@ -37,6 +41,7 @@ Item
|
||||||
}
|
}
|
||||||
height: 18 * screenScaleFactor // TODO: Theme!
|
height: 18 * screenScaleFactor // TODO: Theme!
|
||||||
width: childrenRect.width
|
width: childrenRect.width
|
||||||
|
visible: !cloudConnection
|
||||||
|
|
||||||
UM.RecolorImage
|
UM.RecolorImage
|
||||||
{
|
{
|
||||||
|
@ -67,7 +72,8 @@ Item
|
||||||
MouseArea
|
MouseArea
|
||||||
{
|
{
|
||||||
anchors.fill: manageQueueLabel
|
anchors.fill: manageQueueLabel
|
||||||
hoverEnabled: true
|
enabled: !cloudConnection
|
||||||
|
hoverEnabled: !cloudConnection
|
||||||
onClicked: Cura.MachineManager.printerOutputDevices[0].openPrintJobControlPanel()
|
onClicked: Cura.MachineManager.printerOutputDevices[0].openPrintJobControlPanel()
|
||||||
onEntered:
|
onEntered:
|
||||||
{
|
{
|
||||||
|
@ -165,11 +171,11 @@ Item
|
||||||
// When printing over the cloud we don't recieve print jobs until there is one, so
|
// When printing over the cloud we don't recieve print jobs until there is one, so
|
||||||
// unless there's at least one print job we'll be stuck with skeleton loading
|
// unless there's at least one print job we'll be stuck with skeleton loading
|
||||||
// indefinitely.
|
// indefinitely.
|
||||||
if (Cura.MachineManager.activeMachineHasActiveCloudConnection)
|
if (Cura.MachineManager.activeMachineIsUsingCloudConnection || OutputDevice.receivedPrintJobs)
|
||||||
{
|
{
|
||||||
return OutputDevice.queuedPrintJobs
|
return OutputDevice.queuedPrintJobs
|
||||||
}
|
}
|
||||||
return OutputDevice.receivedPrintJobs ? OutputDevice.queuedPrintJobs : [null,null]
|
return [null, null]
|
||||||
}
|
}
|
||||||
spacing: 6 // TODO: Theme!
|
spacing: 6 // TODO: Theme!
|
||||||
}
|
}
|
||||||
|
|
|
@ -455,8 +455,8 @@ class UM3OutputDevicePlugin(OutputDevicePlugin):
|
||||||
self._start_cloud_flow_message = Message(
|
self._start_cloud_flow_message = Message(
|
||||||
text = i18n_catalog.i18nc("@info:status", "Send and monitor print jobs from anywhere using your Ultimaker account."),
|
text = i18n_catalog.i18nc("@info:status", "Send and monitor print jobs from anywhere using your Ultimaker account."),
|
||||||
lifetime = 0,
|
lifetime = 0,
|
||||||
image_source = os.path.join(os.path.dirname(os.path.abspath(__file__)), "..", "resources", "svg",
|
image_source = QUrl.fromLocalFile(os.path.join(os.path.dirname(os.path.abspath(__file__)), "..",
|
||||||
"cloud-flow-start.svg"),
|
"resources", "svg", "cloud-flow-start.svg")),
|
||||||
image_caption = i18n_catalog.i18nc("@info:status", "Connect to Ultimaker Cloud"),
|
image_caption = i18n_catalog.i18nc("@info:status", "Connect to Ultimaker Cloud"),
|
||||||
option_text = i18n_catalog.i18nc("@action", "Don't ask me again for this printer."),
|
option_text = i18n_catalog.i18nc("@action", "Don't ask me again for this printer."),
|
||||||
option_state = False
|
option_state = False
|
||||||
|
@ -477,12 +477,14 @@ class UM3OutputDevicePlugin(OutputDevicePlugin):
|
||||||
self._cloud_flow_complete_message = Message(
|
self._cloud_flow_complete_message = Message(
|
||||||
text = i18n_catalog.i18nc("@info:status", "You can now send and monitor print jobs from anywhere using your Ultimaker account."),
|
text = i18n_catalog.i18nc("@info:status", "You can now send and monitor print jobs from anywhere using your Ultimaker account."),
|
||||||
lifetime = 30,
|
lifetime = 30,
|
||||||
image_source = os.path.join(os.path.dirname(os.path.abspath(__file__)), "..", "resources", "svg",
|
image_source = QUrl.fromLocalFile(os.path.join(os.path.dirname(os.path.abspath(__file__)), "..",
|
||||||
"cloud-flow-completed.svg"),
|
"resources", "svg", "cloud-flow-completed.svg")),
|
||||||
image_caption = i18n_catalog.i18nc("@info:status", "Connected!")
|
image_caption = i18n_catalog.i18nc("@info:status", "Connected!")
|
||||||
)
|
)
|
||||||
self._cloud_flow_complete_message.addAction("", i18n_catalog.i18nc("@action", "Review your connection"), "", "", 1) # TODO: Icon
|
# Don't show the review connection link if we're not on the local network
|
||||||
self._cloud_flow_complete_message.actionTriggered.connect(self._onReviewCloudConnection)
|
if self._application.getMachineManager().activeMachineHasNetworkConnection:
|
||||||
|
self._cloud_flow_complete_message.addAction("", i18n_catalog.i18nc("@action", "Review your connection"), "", "", 1) # TODO: Icon
|
||||||
|
self._cloud_flow_complete_message.actionTriggered.connect(self._onReviewCloudConnection)
|
||||||
self._cloud_flow_complete_message.show()
|
self._cloud_flow_complete_message.show()
|
||||||
|
|
||||||
# Set the machine's cloud flow as complete so we don't ask the user again and again for cloud connected printers
|
# Set the machine's cloud flow as complete so we don't ask the user again and again for cloud connected printers
|
||||||
|
|
|
@ -72,7 +72,9 @@ class XmlMaterialProfile(InstanceContainer):
|
||||||
material_manager = CuraApplication.getInstance().getMaterialManager()
|
material_manager = CuraApplication.getInstance().getMaterialManager()
|
||||||
root_material_id = self.getMetaDataEntry("base_file") #if basefile is self.getId, this is a basefile.
|
root_material_id = self.getMetaDataEntry("base_file") #if basefile is self.getId, this is a basefile.
|
||||||
material_group = material_manager.getMaterialGroup(root_material_id)
|
material_group = material_manager.getMaterialGroup(root_material_id)
|
||||||
|
if not material_group: #If the profile is not registered in the registry but loose/temporary, it will not have a base file tree.
|
||||||
|
super().setMetaDataEntry(key, value)
|
||||||
|
return
|
||||||
# Update the root material container
|
# Update the root material container
|
||||||
root_material_container = material_group.root_material_node.getContainer()
|
root_material_container = material_group.root_material_node.getContainer()
|
||||||
if root_material_container is not None:
|
if root_material_container is not None:
|
||||||
|
@ -142,23 +144,13 @@ class XmlMaterialProfile(InstanceContainer):
|
||||||
# setting_version is derived from the "version" tag in the schema, so don't serialize it into a file
|
# setting_version is derived from the "version" tag in the schema, so don't serialize it into a file
|
||||||
if ignored_metadata_keys is None:
|
if ignored_metadata_keys is None:
|
||||||
ignored_metadata_keys = set()
|
ignored_metadata_keys = set()
|
||||||
ignored_metadata_keys |= {"setting_version"}
|
ignored_metadata_keys |= {"setting_version", "definition", "status", "variant", "type", "base_file", "approximate_diameter", "id", "container_type", "name"}
|
||||||
# remove the keys that we want to ignore in the metadata
|
# remove the keys that we want to ignore in the metadata
|
||||||
for key in ignored_metadata_keys:
|
for key in ignored_metadata_keys:
|
||||||
if key in metadata:
|
if key in metadata:
|
||||||
del metadata[key]
|
del metadata[key]
|
||||||
properties = metadata.pop("properties", {})
|
properties = metadata.pop("properties", {})
|
||||||
|
|
||||||
# Metadata properties that should not be serialized.
|
|
||||||
metadata.pop("status", "")
|
|
||||||
metadata.pop("variant", "")
|
|
||||||
metadata.pop("type", "")
|
|
||||||
metadata.pop("base_file", "")
|
|
||||||
metadata.pop("approximate_diameter", "")
|
|
||||||
metadata.pop("id", "")
|
|
||||||
metadata.pop("container_type", "")
|
|
||||||
metadata.pop("name", "")
|
|
||||||
|
|
||||||
## Begin Name Block
|
## Begin Name Block
|
||||||
builder.start("name") # type: ignore
|
builder.start("name") # type: ignore
|
||||||
|
|
||||||
|
@ -1166,6 +1158,8 @@ class XmlMaterialProfile(InstanceContainer):
|
||||||
with open(product_to_id_file, encoding = "utf-8") as f:
|
with open(product_to_id_file, encoding = "utf-8") as f:
|
||||||
product_to_id_map = json.load(f)
|
product_to_id_map = json.load(f)
|
||||||
product_to_id_map = {key: [value] for key, value in product_to_id_map.items()}
|
product_to_id_map = {key: [value] for key, value in product_to_id_map.items()}
|
||||||
|
#This also loads "Ultimaker S5" -> "ultimaker_s5" even though that is not strictly necessary with the default to change spaces into underscores.
|
||||||
|
#However it is not always loaded with that default; this mapping is also used in serialize() without that default.
|
||||||
return product_to_id_map
|
return product_to_id_map
|
||||||
|
|
||||||
## Parse the value of the "material compatible" property.
|
## Parse the value of the "material compatible" property.
|
||||||
|
|
|
@ -6,6 +6,7 @@
|
||||||
"Ultimaker 2+": "ultimaker2_plus",
|
"Ultimaker 2+": "ultimaker2_plus",
|
||||||
"Ultimaker 3": "ultimaker3",
|
"Ultimaker 3": "ultimaker3",
|
||||||
"Ultimaker 3 Extended": "ultimaker3_extended",
|
"Ultimaker 3 Extended": "ultimaker3_extended",
|
||||||
|
"Ultimaker S5": "ultimaker_s5",
|
||||||
"Ultimaker Original": "ultimaker_original",
|
"Ultimaker Original": "ultimaker_original",
|
||||||
"Ultimaker Original+": "ultimaker_original_plus",
|
"Ultimaker Original+": "ultimaker_original_plus",
|
||||||
"Ultimaker Original Dual Extrusion": "ultimaker_original_dual",
|
"Ultimaker Original Dual Extrusion": "ultimaker_original_dual",
|
||||||
|
|
|
@ -1792,7 +1792,7 @@
|
||||||
"skin_overlap":
|
"skin_overlap":
|
||||||
{
|
{
|
||||||
"label": "Skin Overlap Percentage",
|
"label": "Skin Overlap Percentage",
|
||||||
"description": "The amount of overlap between the skin and the walls as a percentage of the skin line width. A slight overlap allows the walls to connect firmly to the skin. This is a percentage of the average line widths of the skin lines and the innermost wall.",
|
"description": "Adjust the amount of overlap between the walls and (the endpoints of) the skin-centerlines, as a percentage of the line widths of the skin lines and the innermost wall. A slight overlap allows the walls to connect firmly to the skin. Note that, given an equal skin and wall line-width, any percentage over 50% may already cause any skin to go past the wall, because at that point the position of the nozzle of the skin-extruder may already reach past the middle of the wall.",
|
||||||
"unit": "%",
|
"unit": "%",
|
||||||
"type": "float",
|
"type": "float",
|
||||||
"default_value": 5,
|
"default_value": 5,
|
||||||
|
@ -1807,7 +1807,7 @@
|
||||||
"skin_overlap_mm":
|
"skin_overlap_mm":
|
||||||
{
|
{
|
||||||
"label": "Skin Overlap",
|
"label": "Skin Overlap",
|
||||||
"description": "The amount of overlap between the skin and the walls. A slight overlap allows the walls to connect firmly to the skin.",
|
"description": "Adjust the amount of overlap between the walls and (the endpoints of) the skin-centerlines. A slight overlap allows the walls to connect firmly to the skin. Note that, given an equal skin and wall line-width, any value over half the width of the wall may already cause any skin to go past the wall, because at that point the position of the nozzle of the skin-extruder may already reach past the middle of the wall.",
|
||||||
"unit": "mm",
|
"unit": "mm",
|
||||||
"type": "float",
|
"type": "float",
|
||||||
"default_value": 0.02,
|
"default_value": 0.02,
|
||||||
|
|
|
@ -1604,7 +1604,7 @@ msgstr "El patrón de relleno se mueve esta distancia a lo largo del eje X."
|
||||||
#: fdmprinter.def.json
|
#: fdmprinter.def.json
|
||||||
msgctxt "infill_offset_y label"
|
msgctxt "infill_offset_y label"
|
||||||
msgid "Infill Y Offset"
|
msgid "Infill Y Offset"
|
||||||
msgstr "Desplazamiento del relleno sobre el eje X"
|
msgstr "Desplazamiento del relleno sobre el eje Y"
|
||||||
|
|
||||||
#: fdmprinter.def.json
|
#: fdmprinter.def.json
|
||||||
msgctxt "infill_offset_y description"
|
msgctxt "infill_offset_y description"
|
||||||
|
|
|
@ -222,6 +222,7 @@
|
||||||
"monitor_text_link": [103, 160, 252, 255],
|
"monitor_text_link": [103, 160, 252, 255],
|
||||||
"monitor_icon_primary": [229, 229, 229, 255],
|
"monitor_icon_primary": [229, 229, 229, 255],
|
||||||
"monitor_icon_accent": [51, 53, 54, 255],
|
"monitor_icon_accent": [51, 53, 54, 255],
|
||||||
|
"monitor_icon_disabled": [102, 102, 102, 255],
|
||||||
|
|
||||||
"monitor_secondary_button_hover": [80, 80, 80, 255],
|
"monitor_secondary_button_hover": [80, 80, 80, 255],
|
||||||
"monitor_secondary_button": [92, 92, 92, 255],
|
"monitor_secondary_button": [92, 92, 92, 255],
|
||||||
|
|
|
@ -398,6 +398,7 @@
|
||||||
"monitor_text_link": [50, 130, 255, 255],
|
"monitor_text_link": [50, 130, 255, 255],
|
||||||
"monitor_icon_primary": [10, 8, 80, 255],
|
"monitor_icon_primary": [10, 8, 80, 255],
|
||||||
"monitor_icon_accent": [255, 255, 255, 255],
|
"monitor_icon_accent": [255, 255, 255, 255],
|
||||||
|
"monitor_icon_disabled": [238, 238, 238, 255],
|
||||||
|
|
||||||
"monitor_secondary_button_hover": [228, 228, 228, 255],
|
"monitor_secondary_button_hover": [228, 228, 228, 255],
|
||||||
"monitor_secondary_button": [240, 240, 240, 255],
|
"monitor_secondary_button": [240, 240, 240, 255],
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue