mirror of
https://git.mirrors.martin98.com/https://github.com/Ultimaker/Cura
synced 2025-05-02 08:44:25 +08:00
129 lines
7.5 KiB
Python
129 lines
7.5 KiB
Python
# Copyright (c) 2019 Ultimaker B.V.
|
|
# Cura is released under the terms of the LGPLv3 or higher.
|
|
|
|
from typing import Any, TYPE_CHECKING
|
|
|
|
from UM.Logger import Logger
|
|
from UM.Settings.ContainerRegistry import ContainerRegistry
|
|
from UM.Settings.Interfaces import ContainerInterface
|
|
from UM.Signal import Signal
|
|
from cura.Machines.ContainerNode import ContainerNode
|
|
from cura.Machines.QualityNode import QualityNode
|
|
|
|
if TYPE_CHECKING:
|
|
from typing import Dict
|
|
from cura.Machines.VariantNode import VariantNode
|
|
|
|
## Represents a material in the container tree.
|
|
#
|
|
# Its subcontainers are quality profiles.
|
|
class MaterialNode(ContainerNode):
|
|
def __init__(self, container_id, variant: "VariantNode") -> None:
|
|
super().__init__(container_id)
|
|
self.variant = variant
|
|
self.qualities = {} # type: Dict[str, QualityNode] # Mapping container IDs to quality profiles.
|
|
self.materialChanged = Signal() # Triggered when the material is removed or its metadata is updated.
|
|
|
|
container_registry = ContainerRegistry.getInstance()
|
|
my_metadata = container_registry.findContainersMetadata(id = container_id)[0]
|
|
self.base_file = my_metadata["base_file"]
|
|
self.material_type = my_metadata["material"]
|
|
self.guid = my_metadata["GUID"]
|
|
self._loadAll()
|
|
container_registry.containerRemoved.connect(self._onRemoved)
|
|
container_registry.containerMetaDataChanged.connect(self._onMetadataChanged)
|
|
|
|
## Finds the preferred quality for this printer with this material and this
|
|
# variant loaded.
|
|
#
|
|
# If the preferred quality is not available, an arbitrary quality is
|
|
# returned. If there is a configuration mistake (like a typo in the
|
|
# preferred quality) this returns a random available quality. If there are
|
|
# no available qualities, this will return the empty quality node.
|
|
# \return The node for the preferred quality, or any arbitrary quality if
|
|
# there is no match.
|
|
def preferredQuality(self) -> QualityNode:
|
|
for quality_id, quality_node in self.qualities.items():
|
|
if self.variant.machine.preferred_quality_type == quality_node.quality_type:
|
|
return quality_node
|
|
fallback = next(iter(self.qualities.values())) # Should only happen with empty quality node.
|
|
Logger.log("w", "Could not find preferred quality type {preferred_quality_type} for material {material_id} and variant {variant_id}, falling back to {fallback}.".format(
|
|
preferred_quality_type = self.variant.machine.preferred_quality_type,
|
|
material_id = self.container_id,
|
|
variant_id = self.variant.container_id,
|
|
fallback = fallback.container_id
|
|
))
|
|
return fallback
|
|
|
|
def _loadAll(self) -> None:
|
|
container_registry = ContainerRegistry.getInstance()
|
|
# Find all quality profiles that fit on this material.
|
|
if not self.variant.machine.has_machine_quality: # Need to find the global qualities.
|
|
qualities = container_registry.findInstanceContainersMetadata(type = "quality", definition = "fdmprinter")
|
|
elif not self.variant.machine.has_materials:
|
|
qualities = container_registry.findInstanceContainersMetadata(type="quality", definition=self.variant.machine.quality_definition)
|
|
else:
|
|
if self.variant.machine.has_variants:
|
|
# Need to find the qualities that specify a material profile with the same material type.
|
|
qualities = container_registry.findInstanceContainersMetadata(type = "quality", definition = self.variant.machine.quality_definition, variant = self.variant.variant_name, material = self.container_id) # First try by exact material ID.
|
|
else:
|
|
qualities = container_registry.findInstanceContainersMetadata(type="quality", definition=self.variant.machine.quality_definition, material=self.container_id)
|
|
if not qualities:
|
|
my_material_type = self.material_type
|
|
if self.variant.machine.has_variants:
|
|
qualities_any_material = container_registry.findInstanceContainersMetadata(type = "quality", definition = self.variant.machine.quality_definition, variant = self.variant.variant_name)
|
|
else:
|
|
qualities_any_material = container_registry.findInstanceContainersMetadata(type="quality", definition = self.variant.machine.quality_definition)
|
|
qualities_any_material = [metadata for metadata in qualities_any_material if metadata.get("global_quality", "False") != "True"]
|
|
for material_metadata in container_registry.findInstanceContainersMetadata(type = "material", material = my_material_type):
|
|
qualities.extend((quality for quality in qualities_any_material if quality["material"] == material_metadata["id"]))
|
|
if not qualities: # No quality profiles found. Go by GUID then.
|
|
my_guid = self.guid
|
|
for material_metadata in container_registry.findInstanceContainersMetadata(type = "material", guid = my_guid):
|
|
qualities.extend((quality for quality in qualities_any_material if quality["material"] == material_metadata["id"]))
|
|
|
|
for quality in qualities:
|
|
quality_id = quality["id"]
|
|
if quality_id not in self.qualities:
|
|
self.qualities[quality_id] = QualityNode(quality_id, parent = self)
|
|
if not self.qualities:
|
|
self.qualities["empty_quality"] = QualityNode("empty_quality", parent = self)
|
|
|
|
## Triggered when any container is removed, but only handles it when the
|
|
# container is removed that this node represents.
|
|
# \param container The container that was allegedly removed.
|
|
def _onRemoved(self, container: ContainerInterface) -> None:
|
|
if container.getId() == self.container_id:
|
|
# Remove myself from my parent.
|
|
if self.base_file in self.variant.materials:
|
|
del self.variant.materials[self.base_file]
|
|
if not self.variant.materials:
|
|
self.variant.materials["empty_material"] = MaterialNode("empty_material", variant = self.variant)
|
|
self.materialChanged.emit(self)
|
|
|
|
## Triggered when any metadata changed in any container, but only handles
|
|
# it when the metadata of this node is changed.
|
|
# \param container The container whose metadata changed.
|
|
# \param kwargs Key-word arguments provided when changing the metadata.
|
|
# These are ignored. As far as I know they are never provided to this
|
|
# call.
|
|
def _onMetadataChanged(self, container: ContainerInterface, **kwargs: Any) -> None:
|
|
if container.getId() != self.container_id:
|
|
return
|
|
|
|
new_metadata = container.getMetaData()
|
|
old_base_file = self.base_file
|
|
if new_metadata["base_file"] != old_base_file:
|
|
self.base_file = new_metadata["base_file"]
|
|
if old_base_file in self.variant.materials: # Move in parent node.
|
|
del self.variant.materials[old_base_file]
|
|
self.variant.materials[self.base_file] = self
|
|
|
|
old_material_type = self.material_type
|
|
self.material_type = new_metadata["material"]
|
|
old_guid = self.guid
|
|
self.guid = new_metadata["GUID"]
|
|
if self.base_file != old_base_file or self.material_type != old_material_type or self.guid != old_guid: # List of quality profiles could've changed.
|
|
self.qualities = {}
|
|
self._loadAll() # Re-load the quality profiles for this node.
|
|
self.materialChanged.emit(self) |