GlobalStack.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250
  1. # Copyright (c) 2018 Ultimaker B.V.
  2. # Cura is released under the terms of the LGPLv3 or higher.
  3. from collections import defaultdict
  4. import threading
  5. from typing import Any, Dict, Optional, Set, TYPE_CHECKING, List
  6. from PyQt5.QtCore import pyqtProperty, pyqtSlot, pyqtSignal
  7. from UM.Decorators import override
  8. from UM.MimeTypeDatabase import MimeType, MimeTypeDatabase
  9. from UM.Settings.ContainerStack import ContainerStack
  10. from UM.Settings.SettingInstance import InstanceState
  11. from UM.Settings.ContainerRegistry import ContainerRegistry
  12. from UM.Settings.Interfaces import PropertyEvaluationContext
  13. from UM.Logger import Logger
  14. from UM.Resources import Resources
  15. from UM.Platform import Platform
  16. from UM.Util import parseBool
  17. import cura.CuraApplication
  18. from . import Exceptions
  19. from .CuraContainerStack import CuraContainerStack
  20. if TYPE_CHECKING:
  21. from cura.Settings.ExtruderStack import ExtruderStack
  22. ## Represents the Global or Machine stack and its related containers.
  23. #
  24. class GlobalStack(CuraContainerStack):
  25. def __init__(self, container_id: str) -> None:
  26. super().__init__(container_id)
  27. self.setMetaDataEntry("type", "machine") # For backward compatibility
  28. self._extruders = {} # type: Dict[str, "ExtruderStack"]
  29. # This property is used to track which settings we are calculating the "resolve" for
  30. # and if so, to bypass the resolve to prevent an infinite recursion that would occur
  31. # if the resolve function tried to access the same property it is a resolve for.
  32. # Per thread we have our own resolving_settings, or strange things sometimes occur.
  33. self._resolving_settings = defaultdict(set) #type: Dict[str, Set[str]] # keys are thread names
  34. extrudersChanged = pyqtSignal()
  35. ## Get the list of extruders of this stack.
  36. #
  37. # \return The extruders registered with this stack.
  38. @pyqtProperty("QVariantMap", notify = extrudersChanged)
  39. def extruders(self) -> Dict[str, "ExtruderStack"]:
  40. return self._extruders
  41. @pyqtProperty("QVariantList", notify = extrudersChanged)
  42. def extruderList(self) -> List["ExtruderStack"]:
  43. result_tuple_list = sorted(list(self.extruders.items()), key=lambda x: int(x[0]))
  44. result_list = [item[1] for item in result_tuple_list]
  45. machine_extruder_count = self.getProperty("machine_extruder_count", "value")
  46. return result_list[:machine_extruder_count]
  47. @classmethod
  48. def getLoadingPriority(cls) -> int:
  49. return 2
  50. @classmethod
  51. def getConfigurationTypeFromSerialized(cls, serialized: str) -> Optional[str]:
  52. configuration_type = super().getConfigurationTypeFromSerialized(serialized)
  53. if configuration_type == "machine":
  54. return "machine_stack"
  55. return configuration_type
  56. def getBuildplateName(self) -> Optional[str]:
  57. name = None
  58. if self.variant.getId() != "empty_variant":
  59. name = self.variant.getName()
  60. return name
  61. @pyqtProperty(str, constant = True)
  62. def preferred_output_file_formats(self) -> str:
  63. return self.getMetaDataEntry("file_formats")
  64. ## Add an extruder to the list of extruders of this stack.
  65. #
  66. # \param extruder The extruder to add.
  67. #
  68. # \throws Exceptions.TooManyExtrudersError Raised when trying to add an extruder while we
  69. # already have the maximum number of extruders.
  70. def addExtruder(self, extruder: ContainerStack) -> None:
  71. position = extruder.getMetaDataEntry("position")
  72. if position is None:
  73. Logger.log("w", "No position defined for extruder {extruder}, cannot add it to stack {stack}", extruder = extruder.id, stack = self.id)
  74. return
  75. if any(item.getId() == extruder.id for item in self._extruders.values()):
  76. Logger.log("w", "Extruder [%s] has already been added to this stack [%s]", extruder.id, self.getId())
  77. return
  78. self._extruders[position] = extruder
  79. self.extrudersChanged.emit()
  80. Logger.log("i", "Extruder[%s] added to [%s] at position [%s]", extruder.id, self.id, position)
  81. ## Overridden from ContainerStack
  82. #
  83. # This will return the value of the specified property for the specified setting,
  84. # unless the property is "value" and that setting has a "resolve" function set.
  85. # When a resolve is set, it will instead try and execute the resolve first and
  86. # then fall back to the normal "value" property.
  87. #
  88. # \param key The setting key to get the property of.
  89. # \param property_name The property to get the value of.
  90. #
  91. # \return The value of the property for the specified setting, or None if not found.
  92. @override(ContainerStack)
  93. def getProperty(self, key: str, property_name: str, context: Optional[PropertyEvaluationContext] = None) -> Any:
  94. if not self.definition.findDefinitions(key = key):
  95. return None
  96. if context is None:
  97. context = PropertyEvaluationContext()
  98. context.pushContainer(self)
  99. # Handle the "resolve" property.
  100. #TODO: Why the hell does this involve threading?
  101. # Answer: Because if multiple threads start resolving properties that have the same underlying properties that's
  102. # related, without taking a note of which thread a resolve paths belongs to, they can bump into each other and
  103. # generate unexpected behaviours.
  104. if self._shouldResolve(key, property_name, context):
  105. current_thread = threading.current_thread()
  106. self._resolving_settings[current_thread.name].add(key)
  107. resolve = super().getProperty(key, "resolve", context)
  108. self._resolving_settings[current_thread.name].remove(key)
  109. if resolve is not None:
  110. return resolve
  111. # Handle the "limit_to_extruder" property.
  112. limit_to_extruder = super().getProperty(key, "limit_to_extruder", context)
  113. if limit_to_extruder is not None:
  114. if limit_to_extruder == -1:
  115. limit_to_extruder = int(cura.CuraApplication.CuraApplication.getInstance().getMachineManager().defaultExtruderPosition)
  116. limit_to_extruder = str(limit_to_extruder)
  117. if limit_to_extruder is not None and limit_to_extruder != "-1" and limit_to_extruder in self._extruders:
  118. if super().getProperty(key, "settable_per_extruder", context):
  119. result = self._extruders[str(limit_to_extruder)].getProperty(key, property_name, context)
  120. if result is not None:
  121. context.popContainer()
  122. return result
  123. else:
  124. Logger.log("e", "Setting {setting} has limit_to_extruder but is not settable per extruder!", setting = key)
  125. result = super().getProperty(key, property_name, context)
  126. context.popContainer()
  127. return result
  128. ## Overridden from ContainerStack
  129. #
  130. # This will simply raise an exception since the Global stack cannot have a next stack.
  131. @override(ContainerStack)
  132. def setNextStack(self, stack: CuraContainerStack, connect_signals: bool = True) -> None:
  133. raise Exceptions.InvalidOperationError("Global stack cannot have a next stack!")
  134. # protected:
  135. # Determine whether or not we should try to get the "resolve" property instead of the
  136. # requested property.
  137. def _shouldResolve(self, key: str, property_name: str, context: Optional[PropertyEvaluationContext] = None) -> bool:
  138. if property_name is not "value":
  139. # Do not try to resolve anything but the "value" property
  140. return False
  141. current_thread = threading.current_thread()
  142. if key in self._resolving_settings[current_thread.name]:
  143. # To prevent infinite recursion, if getProperty is called with the same key as
  144. # we are already trying to resolve, we should not try to resolve again. Since
  145. # this can happen multiple times when trying to resolve a value, we need to
  146. # track all settings that are being resolved.
  147. return False
  148. setting_state = super().getProperty(key, "state", context = context)
  149. if setting_state is not None and setting_state != InstanceState.Default:
  150. # When the user has explicitly set a value, we should ignore any resolve and
  151. # just return that value.
  152. return False
  153. return True
  154. ## Perform some sanity checks on the global stack
  155. # Sanity check for extruders; they must have positions 0 and up to machine_extruder_count - 1
  156. def isValid(self) -> bool:
  157. container_registry = ContainerRegistry.getInstance()
  158. extruder_trains = container_registry.findContainerStacks(type = "extruder_train", machine = self.getId())
  159. machine_extruder_count = self.getProperty("machine_extruder_count", "value")
  160. extruder_check_position = set()
  161. for extruder_train in extruder_trains:
  162. extruder_position = extruder_train.getMetaDataEntry("position")
  163. extruder_check_position.add(extruder_position)
  164. for check_position in range(machine_extruder_count):
  165. if str(check_position) not in extruder_check_position:
  166. return False
  167. return True
  168. def getHeadAndFansCoordinates(self):
  169. return self.getProperty("machine_head_with_fans_polygon", "value")
  170. def getHasMaterials(self) -> bool:
  171. return parseBool(self.getMetaDataEntry("has_materials", False))
  172. def getHasVariants(self) -> bool:
  173. return parseBool(self.getMetaDataEntry("has_variants", False))
  174. def getHasMachineQuality(self) -> bool:
  175. return parseBool(self.getMetaDataEntry("has_machine_quality", False))
  176. ## Get default firmware file name if one is specified in the firmware
  177. @pyqtSlot(result = str)
  178. def getDefaultFirmwareName(self) -> str:
  179. machine_has_heated_bed = self.getProperty("machine_heated_bed", "value")
  180. baudrate = 250000
  181. if Platform.isLinux():
  182. # Linux prefers a baudrate of 115200 here because older versions of
  183. # pySerial did not support a baudrate of 250000
  184. baudrate = 115200
  185. # If a firmware file is available, it should be specified in the definition for the printer
  186. hex_file = self.getMetaDataEntry("firmware_file", None)
  187. if machine_has_heated_bed:
  188. hex_file = self.getMetaDataEntry("firmware_hbk_file", hex_file)
  189. if not hex_file:
  190. Logger.log("w", "There is no firmware for machine %s.", self.getBottom().id)
  191. return ""
  192. try:
  193. return Resources.getPath(cura.CuraApplication.CuraApplication.ResourceTypes.Firmware, hex_file.format(baudrate=baudrate))
  194. except FileNotFoundError:
  195. Logger.log("w", "Firmware file %s not found.", hex_file)
  196. return ""
  197. ## private:
  198. global_stack_mime = MimeType(
  199. name = "application/x-cura-globalstack",
  200. comment = "Cura Global Stack",
  201. suffixes = ["global.cfg"]
  202. )
  203. MimeTypeDatabase.addMimeType(global_stack_mime)
  204. ContainerRegistry.addContainerTypeByName(GlobalStack, "global_stack", global_stack_mime.name)