CuraFormulaFunctions.py 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. # Copyright (c) 2018 Ultimaker B.V.
  2. # Cura is released under the terms of the LGPLv3 or higher.
  3. from typing import Any, List, Optional, TYPE_CHECKING
  4. from UM.Settings.PropertyEvaluationContext import PropertyEvaluationContext
  5. from UM.Settings.SettingFunction import SettingFunction
  6. from UM.Logger import Logger
  7. if TYPE_CHECKING:
  8. from cura.CuraApplication import CuraApplication
  9. from cura.Settings.CuraContainerStack import CuraContainerStack
  10. #
  11. # This class contains all Cura-related custom functions that can be used in formulas. Some functions requires
  12. # information such as the currently active machine, so this is made into a class instead of standalone functions.
  13. #
  14. class CuraFormulaFunctions:
  15. def __init__(self, application: "CuraApplication") -> None:
  16. self._application = application
  17. # ================
  18. # Custom Functions
  19. # ================
  20. # Gets the default extruder position of the currently active machine.
  21. def getDefaultExtruderPosition(self) -> str:
  22. machine_manager = self._application.getMachineManager()
  23. return machine_manager.defaultExtruderPosition
  24. # Gets the given setting key from the given extruder position.
  25. def getValueInExtruder(self, extruder_position: int, property_key: str,
  26. context: Optional["PropertyEvaluationContext"] = None) -> Any:
  27. machine_manager = self._application.getMachineManager()
  28. if extruder_position == -1:
  29. extruder_position = int(machine_manager.defaultExtruderPosition)
  30. global_stack = machine_manager.activeMachine
  31. try:
  32. extruder_stack = global_stack.extruderList[int(extruder_position)]
  33. except IndexError:
  34. if extruder_position != 0:
  35. Logger.log("w", "Value for %s of extruder %s was requested, but that extruder is not available. Returning the result from extruder 0 instead" % (property_key, extruder_position))
  36. # This fixes a very specific fringe case; If a profile was created for a custom printer and one of the
  37. # extruder settings has been set to non zero and the profile is loaded for a machine that has only a single extruder
  38. # it would cause all kinds of issues (and eventually a crash).
  39. # See https://github.com/Ultimaker/Cura/issues/5535
  40. return self.getValueInExtruder(0, property_key, context)
  41. Logger.log("w", "Value for %s of extruder %s was requested, but that extruder is not available. " % (property_key, extruder_position))
  42. return None
  43. value = extruder_stack.getRawProperty(property_key, "value", context = context)
  44. if isinstance(value, SettingFunction):
  45. value = value(extruder_stack, context = context)
  46. if isinstance(value, str):
  47. value = value.lower()
  48. return value
  49. def _getActiveExtruders(self, context: Optional["PropertyEvaluationContext"] = None) -> List[str]:
  50. machine_manager = self._application.getMachineManager()
  51. extruder_manager = self._application.getExtruderManager()
  52. global_stack = machine_manager.activeMachine
  53. result = []
  54. for extruder in extruder_manager.getActiveExtruderStacks():
  55. if not extruder.isEnabled:
  56. continue
  57. # only include values from extruders that are "active" for the current machine instance
  58. if int(extruder.getMetaDataEntry("position")) >= global_stack.getProperty("machine_extruder_count", "value", context = context):
  59. continue
  60. result.append(extruder)
  61. return result
  62. # Gets all extruder values as a list for the given property.
  63. def getValuesInAllExtruders(self, property_key: str,
  64. context: Optional["PropertyEvaluationContext"] = None) -> List[Any]:
  65. global_stack = self._application.getMachineManager().activeMachine
  66. result = []
  67. for extruder in self._getActiveExtruders(context):
  68. value = extruder.getRawProperty(property_key, "value", context = context)
  69. if value is None:
  70. continue
  71. if isinstance(value, SettingFunction):
  72. value = value(extruder, context = context)
  73. result.append(value)
  74. if not result:
  75. result.append(global_stack.getProperty(property_key, "value", context = context))
  76. return result
  77. # Get the first extruder that adheres to a specific (boolean) property, like 'material_is_support_material'.
  78. def getAnyExtruderPositionWithOrDefault(self, filter_key: str,
  79. context: Optional["PropertyEvaluationContext"] = None) -> str:
  80. for extruder in self._getActiveExtruders(context):
  81. value = extruder.getRawProperty(filter_key, "value", context=context)
  82. if value is None or not value:
  83. continue
  84. return str(extruder.position)
  85. # Get the first extruder with material that adheres to a specific (boolean) property, like 'material_is_support_material'.
  86. def getExtruderPositionWithMaterial(self, filter_key: str,
  87. context: Optional["PropertyEvaluationContext"] = None) -> str:
  88. for extruder in self._getActiveExtruders(context):
  89. material_container = extruder.material
  90. value = material_container.getProperty(filter_key, "value", context)
  91. if value is not None:
  92. return str(extruder.position)
  93. return self.getDefaultExtruderPosition()
  94. # Get the resolve value or value for a given key.
  95. def getResolveOrValue(self, property_key: str, context: Optional["PropertyEvaluationContext"] = None) -> Any:
  96. machine_manager = self._application.getMachineManager()
  97. global_stack = machine_manager.activeMachine
  98. resolved_value = global_stack.getProperty(property_key, "value", context = context)
  99. return resolved_value
  100. # Gets the default setting value from given extruder position. The default value is what excludes the values in
  101. # the user_changes container.
  102. def getDefaultValueInExtruder(self, extruder_position: int, property_key: str) -> Any:
  103. machine_manager = self._application.getMachineManager()
  104. global_stack = machine_manager.activeMachine
  105. try:
  106. extruder_stack = global_stack.extruderList[extruder_position]
  107. except IndexError:
  108. Logger.log("w", "Unable to find extruder on in index %s", extruder_position)
  109. else:
  110. context = self.createContextForDefaultValueEvaluation(extruder_stack)
  111. return self.getValueInExtruder(extruder_position, property_key, context = context)
  112. # Gets all default setting values as a list from all extruders of the currently active machine.
  113. # The default values are those excluding the values in the user_changes container.
  114. def getDefaultValuesInAllExtruders(self, property_key: str) -> List[Any]:
  115. machine_manager = self._application.getMachineManager()
  116. global_stack = machine_manager.activeMachine
  117. context = self.createContextForDefaultValueEvaluation(global_stack)
  118. return self.getValuesInAllExtruders(property_key, context = context)
  119. # Gets the resolve value or value for a given key without looking the first container (user container).
  120. def getDefaultResolveOrValue(self, property_key: str) -> Any:
  121. machine_manager = self._application.getMachineManager()
  122. global_stack = machine_manager.activeMachine
  123. context = self.createContextForDefaultValueEvaluation(global_stack)
  124. return self.getResolveOrValue(property_key, context = context)
  125. # Gets the value for the given setting key starting from the given container index.
  126. def getValueFromContainerAtIndex(self, property_key: str, container_index: int,
  127. context: Optional["PropertyEvaluationContext"] = None) -> Any:
  128. machine_manager = self._application.getMachineManager()
  129. global_stack = machine_manager.activeMachine
  130. context = self.createContextForDefaultValueEvaluation(global_stack)
  131. context.context["evaluate_from_container_index"] = container_index
  132. return global_stack.getProperty(property_key, "value", context = context)
  133. # Gets the extruder value for the given setting key starting from the given container index.
  134. def getValueFromContainerAtIndexInExtruder(self, extruder_position: int, property_key: str, container_index: int,
  135. context: Optional["PropertyEvaluationContext"] = None) -> Any:
  136. machine_manager = self._application.getMachineManager()
  137. global_stack = machine_manager.activeMachine
  138. if extruder_position == -1:
  139. extruder_position = int(machine_manager.defaultExtruderPosition)
  140. global_stack = machine_manager.activeMachine
  141. try:
  142. extruder_stack = global_stack.extruderList[int(extruder_position)]
  143. except IndexError:
  144. Logger.log("w", "Value for %s of extruder %s was requested, but that extruder is not available. " % (property_key, extruder_position))
  145. return None
  146. context = self.createContextForDefaultValueEvaluation(extruder_stack)
  147. context.context["evaluate_from_container_index"] = container_index
  148. return self.getValueInExtruder(extruder_position, property_key, context)
  149. # Creates a context for evaluating default values (skip the user_changes container).
  150. def createContextForDefaultValueEvaluation(self, source_stack: "CuraContainerStack") -> "PropertyEvaluationContext":
  151. context = PropertyEvaluationContext(source_stack)
  152. context.context["evaluate_from_container_index"] = 1 # skip the user settings container
  153. context.context["override_operators"] = {
  154. "extruderValue": self.getDefaultValueInExtruder,
  155. "extruderValues": self.getDefaultValuesInAllExtruders,
  156. "resolveOrValue": self.getDefaultResolveOrValue,
  157. }
  158. return context