123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572 |
- # Copyright (c) 2016 Ultimaker B.V.
- # Cura is released under the terms of the AGPLv3 or higher.
- from PyQt5.QtCore import QObject, pyqtSlot, pyqtProperty, pyqtSignal
- from UM.Application import Application
- from UM.Preferences import Preferences
- import UM.Settings
- from UM.Settings.Validator import ValidatorState
- from UM.Settings.InstanceContainer import InstanceContainer
- from cura.PrinterOutputDevice import PrinterOutputDevice
- from UM.Settings.ContainerStack import ContainerStack
- from . import ExtruderManager
- from UM.i18n import i18nCatalog
- catalog = i18nCatalog("cura")
- class MachineManagerModel(QObject):
- def __init__(self, parent = None):
- super().__init__(parent)
- self._active_container_stack = None
- self._global_container_stack = None
- Application.getInstance().globalContainerStackChanged.connect(self._onGlobalContainerChanged)
- self._global_stack_valid = None
- self._onGlobalContainerChanged()
- ExtruderManager.ExtruderManager.getInstance().activeExtruderChanged.connect(self._onActiveExtruderStackChanged)
- self.globalContainerChanged.connect(self._onActiveExtruderStackChanged)
- self._onActiveExtruderStackChanged()
- ## When the global container is changed, active material probably needs to be updated.
- self.globalContainerChanged.connect(self.activeMaterialChanged)
- self.globalContainerChanged.connect(self.activeVariantChanged)
- self.globalContainerChanged.connect(self.activeQualityChanged)
- ExtruderManager.ExtruderManager.getInstance().activeExtruderChanged.connect(self.activeMaterialChanged)
- ExtruderManager.ExtruderManager.getInstance().activeExtruderChanged.connect(self.activeVariantChanged)
- ExtruderManager.ExtruderManager.getInstance().activeExtruderChanged.connect(self.activeQualityChanged)
- self.globalContainerChanged.connect(self.activeStackChanged)
- self.globalValueChanged.connect(self.activeStackChanged)
- ExtruderManager.ExtruderManager.getInstance().activeExtruderChanged.connect(self.activeStackChanged)
- self._empty_variant_container = UM.Settings.ContainerRegistry.getInstance().findInstanceContainers(id = "empty_variant")[0]
- self._empty_material_container = UM.Settings.ContainerRegistry.getInstance().findInstanceContainers(id = "empty_material")[0]
- self._empty_quality_container = UM.Settings.ContainerRegistry.getInstance().findInstanceContainers(id = "empty_quality")[0]
- Preferences.getInstance().addPreference("cura/active_machine", "")
- active_machine_id = Preferences.getInstance().getValue("cura/active_machine")
- Application.getInstance().getOutputDeviceManager().outputDevicesChanged.connect(self._onOutputDevicesChanged)
- if active_machine_id != "":
- # An active machine was saved, so restore it.
- self.setActiveMachine(active_machine_id)
- pass
- globalContainerChanged = pyqtSignal()
- activeMaterialChanged = pyqtSignal()
- activeVariantChanged = pyqtSignal()
- activeQualityChanged = pyqtSignal()
- activeStackChanged = pyqtSignal()
- globalValueChanged = pyqtSignal() # Emitted whenever a value inside global container is changed.
- globalValidationChanged = pyqtSignal() # Emitted whenever a validation inside global container is changed
- blurSettings = pyqtSignal() # Emitted to force fields in the advanced sidebar to un-focus, so they update properly
- outputDevicesChanged = pyqtSignal()
- def _onOutputDevicesChanged(self):
- self.outputDevicesChanged.emit()
- def _onGlobalPropertyChanged(self, key, property_name):
- if property_name == "value":
- self.globalValueChanged.emit()
- if property_name == "validationState":
- if self._global_stack_valid:
- changed_validation_state = self._active_container_stack.getProperty(key, property_name)
- if changed_validation_state in (ValidatorState.Exception, ValidatorState.MaximumError, ValidatorState.MinimumError):
- self._global_stack_valid = False
- self.globalValidationChanged.emit()
- else:
- has_errors = self._checkStackForErrors(self._active_container_stack)
- if not has_errors:
- self._global_stack_valid = True
- self.globalValidationChanged.emit()
- def _onGlobalContainerChanged(self):
- if self._global_container_stack:
- self._global_container_stack.containersChanged.disconnect(self._onInstanceContainersChanged)
- self._global_container_stack.propertyChanged.disconnect(self._onGlobalPropertyChanged)
- self._global_container_stack = Application.getInstance().getGlobalContainerStack()
- self.globalContainerChanged.emit()
- if self._global_container_stack:
- Preferences.getInstance().setValue("cura/active_machine", self._global_container_stack.getId())
- self._global_container_stack.containersChanged.connect(self._onInstanceContainersChanged)
- self._global_container_stack.propertyChanged.connect(self._onGlobalPropertyChanged)
- self._global_stack_valid = not self._checkStackForErrors(self._global_container_stack)
- def _onActiveExtruderStackChanged(self):
- self.blurSettings.emit() # Ensure no-one has focus.
- if self._active_container_stack and self._active_container_stack != self._global_container_stack:
- self._active_container_stack.containersChanged.disconnect(self._onInstanceContainersChanged)
- self._active_container_stack.propertyChanged.disconnect(self._onGlobalPropertyChanged)
- self._active_container_stack = ExtruderManager.ExtruderManager.getInstance().getActiveExtruderStack()
- if self._active_container_stack:
- self._active_container_stack.containersChanged.connect(self._onInstanceContainersChanged)
- self._active_container_stack.propertyChanged.connect(self._onGlobalPropertyChanged)
- else:
- self._active_container_stack = self._global_container_stack
- def _onInstanceContainersChanged(self, container):
- container_type = container.getMetaDataEntry("type")
- if container_type == "material":
- self.activeMaterialChanged.emit()
- elif container_type == "variant":
- self.activeVariantChanged.emit()
- elif container_type == "quality":
- self.activeQualityChanged.emit()
- @pyqtSlot(str)
- def setActiveMachine(self, stack_id):
- containers = UM.Settings.ContainerRegistry.getInstance().findContainerStacks(id = stack_id)
- if containers:
- Application.getInstance().setGlobalContainerStack(containers[0])
- @pyqtSlot(str, str)
- def addMachine(self, name, definition_id):
- definitions = UM.Settings.ContainerRegistry.getInstance().findDefinitionContainers(id = definition_id)
- if definitions:
- definition = definitions[0]
- name = self._createUniqueName("machine", "", name, definition.getName())
- new_global_stack = UM.Settings.ContainerStack(name)
- new_global_stack.addMetaDataEntry("type", "machine")
- UM.Settings.ContainerRegistry.getInstance().addContainer(new_global_stack)
- variant_instance_container = self._updateVariantContainer(definition)
- material_instance_container = self._updateMaterialContainer(definition, variant_instance_container)
- quality_instance_container = self._updateQualityContainer(definition, material_instance_container)
- current_settings_instance_container = UM.Settings.InstanceContainer(name + "_current_settings")
- current_settings_instance_container.addMetaDataEntry("machine", name)
- current_settings_instance_container.addMetaDataEntry("type", "user")
- current_settings_instance_container.setDefinition(definitions[0])
- UM.Settings.ContainerRegistry.getInstance().addContainer(current_settings_instance_container)
- # If a definition is found, its a list. Should only have one item.
- new_global_stack.addContainer(definition)
- if variant_instance_container:
- new_global_stack.addContainer(variant_instance_container)
- if material_instance_container:
- new_global_stack.addContainer(material_instance_container)
- if quality_instance_container:
- new_global_stack.addContainer(quality_instance_container)
- new_global_stack.addContainer(current_settings_instance_container)
- ExtruderManager.ExtruderManager.getInstance().addMachineExtruders(definition)
- Application.getInstance().setGlobalContainerStack(new_global_stack)
- @pyqtProperty("QVariantList", notify = outputDevicesChanged)
- def printerOutputDevices(self):
- return [printer_output_device for printer_output_device in Application.getInstance().getOutputDeviceManager().getOutputDevices() if isinstance(printer_output_device, PrinterOutputDevice)]
- ## Create a name that is not empty and unique
- # \param container_type \type{string} Type of the container (machine, quality, ...)
- # \param current_name \type{} Current name of the container, which may be an acceptable option
- # \param new_name \type{string} Base name, which may not be unique
- # \param fallback_name \type{string} Name to use when (stripped) new_name is empty
- # \return \type{string} Name that is unique for the specified type and name/id
- def _createUniqueName(self, container_type, current_name, new_name, fallback_name):
- return UM.Settings.ContainerRegistry.getInstance().createUniqueName(container_type, current_name, new_name, fallback_name)
- ## Convenience function to check if a stack has errors.
- def _checkStackForErrors(self, stack):
- if stack is None:
- return False
- for key in stack.getAllKeys():
- validation_state = stack.getProperty(key, "validationState")
- if validation_state in (ValidatorState.Exception, ValidatorState.MaximumError, ValidatorState.MinimumError):
- return True
- return False
- ## Remove all instances from the top instanceContainer (effectively removing all user-changed settings)
- @pyqtSlot()
- def clearUserSettings(self):
- if not self._active_container_stack:
- return
- self.blurSettings.emit()
- user_settings = self._active_container_stack.getTop()
- user_settings.clear()
- ## Check if the global_container has instances in the user container
- @pyqtProperty(bool, notify = activeStackChanged)
- def hasUserSettings(self):
- if not self._active_container_stack:
- return False
- user_settings = self._active_container_stack.getTop().findInstances(**{})
- return len(user_settings) != 0
- ## Check if the global profile does not contain error states
- # Note that the _global_stack_valid is cached due to performance issues
- # Calling _checkStackForErrors on every change is simply too expensive
- @pyqtProperty(bool, notify = globalValidationChanged)
- def isGlobalStackValid(self):
- return self._global_stack_valid
- @pyqtProperty(str, notify = activeStackChanged)
- def activeUserProfileId(self):
- if self._active_container_stack:
- return self._active_container_stack.getTop().getId()
- return ""
- @pyqtProperty(str, notify = globalContainerChanged)
- def activeMachineName(self):
- if self._global_container_stack:
- return self._global_container_stack.getName()
- return ""
- @pyqtProperty(str, notify = globalContainerChanged)
- def activeMachineId(self):
- if self._global_container_stack:
- return self._global_container_stack.getId()
- return ""
- @pyqtProperty(str, notify = activeMaterialChanged)
- def activeMaterialName(self):
- if self._active_container_stack:
- material = self._active_container_stack.findContainer({"type":"material"})
- if material:
- return material.getName()
- return ""
- @pyqtProperty(str, notify=activeMaterialChanged)
- def activeMaterialId(self):
- if self._active_container_stack:
- material = self._active_container_stack.findContainer({"type": "material"})
- if material:
- return material.getId()
- return ""
- @pyqtProperty(str, notify=activeQualityChanged)
- def activeQualityName(self):
- if self._active_container_stack:
- quality = self._active_container_stack.findContainer({"type": "quality"})
- if quality:
- return quality.getName()
- return ""
- @pyqtProperty(str, notify=activeQualityChanged)
- def activeQualityId(self):
- if self._active_container_stack:
- quality = self._active_container_stack.findContainer({"type": "quality"})
- if quality:
- return quality.getId()
- return ""
- ## Check if a container is read_only
- @pyqtSlot(str, result = bool)
- def isReadOnly(self, container_id):
- containers = UM.Settings.ContainerRegistry.getInstance().findInstanceContainers(id = container_id)
- if not containers or not self._active_container_stack:
- return True
- return containers[0].isReadOnly()
- @pyqtSlot(result = str)
- def newQualityContainerFromQualityAndUser(self):
- new_container_id = self.duplicateContainer(self.activeQualityId)
- if new_container_id == "":
- return
- self.blurSettings.emit()
- self.setActiveQuality(new_container_id)
- self.updateQualityContainerFromUserContainer()
- return new_container_id
- @pyqtSlot(str, result=str)
- def duplicateContainer(self, container_id):
- if not self._active_container_stack:
- return ""
- containers = UM.Settings.ContainerRegistry.getInstance().findInstanceContainers(id = container_id)
- if containers:
- new_name = self._createUniqueName("quality", "", containers[0].getName(), catalog.i18nc("@label", "Custom profile"))
- new_container = InstanceContainer("")
- ## Copy all values
- new_container.deserialize(containers[0].serialize())
- new_container.setReadOnly(False)
- new_container.setName(new_name)
- new_container._id = new_name
- UM.Settings.ContainerRegistry.getInstance().addContainer(new_container)
- return new_name
- return ""
- @pyqtSlot(str, str)
- def renameQualityContainer(self, container_id, new_name):
- containers = UM.Settings.ContainerRegistry.getInstance().findInstanceContainers(id = container_id, type = "quality")
- if containers:
- new_name = self._createUniqueName("quality", containers[0].getName(), new_name,
- catalog.i18nc("@label", "Custom profile"))
- if containers[0].getName() == new_name:
- # Nothing to do.
- return
- # As we also want the id of the container to be changed (so that profile name is the name of the file
- # on disk. We need to create a new instance and remove it (so the old file of the container is removed)
- # If we don't do that, we might get duplicates & other weird issues.
- new_container = InstanceContainer("")
- new_container.deserialize(containers[0].serialize())
- # Actually set the name
- new_container.setName(new_name)
- new_container._id = new_name # Todo: Fix proper id change function for this.
- # Add the "new" container.
- UM.Settings.ContainerRegistry.getInstance().addContainer(new_container)
- # Ensure that the renamed profile is saved -before- we remove the old profile.
- Application.getInstance().saveSettings()
- # Actually set & remove new / old quality.
- self.setActiveQuality(new_name)
- self.removeQualityContainer(containers[0].getId())
- @pyqtSlot(str)
- def removeQualityContainer(self, container_id):
- containers = UM.Settings.ContainerRegistry.getInstance().findInstanceContainers(id = container_id)
- if not containers or not self._active_container_stack:
- return
- # If the container that is being removed is the currently active container, set another machine as the active container
- activate_new_container = container_id == self.activeQualityId
- UM.Settings.ContainerRegistry.getInstance().removeContainer(container_id)
- if activate_new_container:
- definition_id = "fdmprinter" if not self.filterQualityByMachine else self.activeDefinitionId
- containers = UM.Settings.ContainerRegistry.getInstance().findInstanceContainers(type = "quality", definition = definition_id)
- if containers:
- self.setActiveQuality(containers[0].getId())
- self.activeQualityChanged.emit()
- @pyqtSlot()
- def updateQualityContainerFromUserContainer(self):
- if not self._active_container_stack:
- return
- user_settings = self._active_container_stack.getTop()
- quality = self._active_container_stack.findContainer({"type": "quality"})
- for key in user_settings.getAllKeys():
- quality.setProperty(key, "value", user_settings.getProperty(key, "value"))
- self.clearUserSettings() # As all users settings are noq a quality, remove them.
- @pyqtSlot(str)
- def setActiveMaterial(self, material_id):
- containers = UM.Settings.ContainerRegistry.getInstance().findInstanceContainers(id = material_id)
- if not containers or not self._active_container_stack:
- return
- old_material = self._active_container_stack.findContainer({"type":"material"})
- old_quality = self._active_container_stack.findContainer({"type": "quality"})
- if old_material:
- material_index = self._active_container_stack.getContainerIndex(old_material)
- self._active_container_stack.replaceContainer(material_index, containers[0])
- preferred_quality_name = None
- if old_quality:
- preferred_quality_name = old_quality.getName()
- self.setActiveQuality(self._updateQualityContainer(self._global_container_stack.getBottom(), containers[0], preferred_quality_name).id)
- @pyqtSlot(str)
- def setActiveVariant(self, variant_id):
- containers = UM.Settings.ContainerRegistry.getInstance().findInstanceContainers(id = variant_id)
- if not containers or not self._active_container_stack:
- return
- old_variant = self._active_container_stack.findContainer({"type": "variant"})
- old_material = self._active_container_stack.findContainer({"type": "material"})
- if old_variant:
- variant_index = self._active_container_stack.getContainerIndex(old_variant)
- self._active_container_stack.replaceContainer(variant_index, containers[0])
- preferred_material = None
- if old_material:
- preferred_material = old_material.getId()
- self.setActiveMaterial(self._updateMaterialContainer(self._global_container_stack.getBottom(), containers[0], preferred_material).id)
- @pyqtSlot(str)
- def setActiveQuality(self, quality_id):
- containers = UM.Settings.ContainerRegistry.getInstance().findInstanceContainers(id = quality_id)
- if not containers or not self._active_container_stack:
- return
- old_quality = self._active_container_stack.findContainer({"type": "quality"})
- if old_quality:
- quality_index = self._active_container_stack.getContainerIndex(old_quality)
- self._active_container_stack.replaceContainer(quality_index, containers[0])
- @pyqtProperty(str, notify = activeVariantChanged)
- def activeVariantName(self):
- if self._active_container_stack:
- variant = self._active_container_stack.findContainer({"type": "variant"})
- if variant:
- return variant.getName()
- return ""
- @pyqtProperty(str, notify = activeVariantChanged)
- def activeVariantId(self):
- if self._active_container_stack:
- variant = self._active_container_stack.findContainer({"type": "variant"})
- if variant:
- return variant.getId()
- return ""
- @pyqtProperty(str, notify = globalContainerChanged)
- def activeDefinitionId(self):
- if self._global_container_stack:
- definition = self._global_container_stack.getBottom()
- if definition:
- return definition.id
- return ""
- @pyqtSlot(str, str)
- def renameMachine(self, machine_id, new_name):
- containers = UM.Settings.ContainerRegistry.getInstance().findContainerStacks(id = machine_id)
- if containers:
- new_name = self._createUniqueName("machine", containers[0].getName(), new_name, containers[0].getBottom().getName())
- containers[0].setName(new_name)
- self.globalContainerChanged.emit()
- @pyqtSlot(str)
- def removeMachine(self, machine_id):
- # If the machine that is being removed is the currently active machine, set another machine as the active machine
- activate_new_machine = (self._global_container_stack and self._global_container_stack.getId() == machine_id)
- current_settings_id = machine_id + "_current_settings"
- containers = UM.Settings.ContainerRegistry.getInstance().findInstanceContainers(id = current_settings_id)
- for container in containers:
- UM.Settings.ContainerRegistry.getInstance().removeContainer(container.getId())
- UM.Settings.ContainerRegistry.getInstance().removeContainer(machine_id)
- if activate_new_machine:
- stacks = UM.Settings.ContainerRegistry.getInstance().findContainerStacks(type = "machine")
- if stacks:
- Application.getInstance().setGlobalContainerStack(stacks[0])
- @pyqtProperty(bool, notify = globalContainerChanged)
- def hasMaterials(self):
- if self._global_container_stack:
- return bool(self._global_container_stack.getMetaDataEntry("has_materials", False))
- return False
- @pyqtProperty(bool, notify = globalContainerChanged)
- def hasVariants(self):
- if self._global_container_stack:
- return bool(self._global_container_stack.getMetaDataEntry("has_variants", False))
- return False
- @pyqtProperty(bool, notify = globalContainerChanged)
- def filterMaterialsByMachine(self):
- if self._global_container_stack:
- return bool(self._global_container_stack.getMetaDataEntry("has_machine_materials", False))
- return False
- @pyqtProperty(bool, notify = globalContainerChanged)
- def filterQualityByMachine(self):
- if self._global_container_stack:
- return bool(self._global_container_stack.getMetaDataEntry("has_machine_quality", False))
- return False
- @pyqtSlot(str, result = str)
- def getDefinitionByMachineId(self, machine_id):
- containers = UM.Settings.ContainerRegistry.getInstance().findContainerStacks(id=machine_id)
- if containers:
- return containers[0].getBottom().getId()
- def _updateVariantContainer(self, definition):
- if not definition.getMetaDataEntry("has_variants"):
- return self._empty_variant_container
- containers = []
- preferred_variant = definition.getMetaDataEntry("preferred_variant")
- if preferred_variant:
- containers = UM.Settings.ContainerRegistry.getInstance().findInstanceContainers(type = "variant", definition = definition.id, id = preferred_variant)
- if not containers:
- containers = UM.Settings.ContainerRegistry.getInstance().findInstanceContainers(type = "variant", definition = definition.id)
- if containers:
- return containers[0]
- return self._empty_variant_container
- def _updateMaterialContainer(self, definition, variant_container = None, preferred_material = None):
- if not definition.getMetaDataEntry("has_materials"):
- return self._empty_material_container
- search_criteria = { "type": "material" }
- if definition.getMetaDataEntry("has_machine_materials"):
- search_criteria["definition"] = definition.id
- if definition.getMetaDataEntry("has_variants") and variant_container:
- search_criteria["variant"] = variant_container.id
- else:
- search_criteria["definition"] = "fdmprinter"
- if not preferred_material:
- preferred_material = definition.getMetaDataEntry("preferred_material")
- if preferred_material:
- search_criteria["id"] = preferred_material
- containers = UM.Settings.ContainerRegistry.getInstance().findInstanceContainers(**search_criteria)
- if containers:
- return containers[0]
- return self._empty_material_container
- def _updateQualityContainer(self, definition, material_container = None, preferred_quality_name = None):
- search_criteria = { "type": "quality" }
- if definition.getMetaDataEntry("has_machine_quality"):
- search_criteria["definition"] = definition.id
- if definition.getMetaDataEntry("has_materials") and material_container:
- search_criteria["material"] = material_container.id
- else:
- search_criteria["definition"] = "fdmprinter"
- if preferred_quality_name:
- search_criteria["name"] = preferred_quality_name
- else:
- preferred_quality = definition.getMetaDataEntry("preferred_quality")
- if preferred_quality:
- search_criteria["id"] = preferred_quality
- containers = UM.Settings.ContainerRegistry.getInstance().findInstanceContainers(**search_criteria)
- if containers:
- return containers[0]
- return self._empty_quality_container
- def createMachineManagerModel(engine, script_engine):
- return MachineManagerModel()
|