WorkspaceDialog.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286
  1. # Copyright (c) 2016 Ultimaker B.V.
  2. # Cura is released under the terms of the AGPLv3 or higher.
  3. from PyQt5.QtCore import Qt, QUrl, pyqtSignal, QObject, pyqtProperty, QCoreApplication
  4. from UM.FlameProfiler import pyqtSlot
  5. from PyQt5.QtQml import QQmlComponent, QQmlContext
  6. from UM.PluginRegistry import PluginRegistry
  7. from UM.Application import Application
  8. from UM.Logger import Logger
  9. from UM.i18n import i18nCatalog
  10. from UM.Settings.ContainerRegistry import ContainerRegistry
  11. import os
  12. import threading
  13. import time
  14. i18n_catalog = i18nCatalog("cura")
  15. class WorkspaceDialog(QObject):
  16. showDialogSignal = pyqtSignal()
  17. def __init__(self, parent = None):
  18. super().__init__(parent)
  19. self._component = None
  20. self._context = None
  21. self._view = None
  22. self._qml_url = "WorkspaceDialog.qml"
  23. self._lock = threading.Lock()
  24. self._default_strategy = "override"
  25. self._result = {"machine": self._default_strategy,
  26. "quality_changes": self._default_strategy,
  27. "material": self._default_strategy}
  28. self._visible = False
  29. self.showDialogSignal.connect(self.__show)
  30. self._has_quality_changes_conflict = False
  31. self._has_machine_conflict = False
  32. self._has_material_conflict = False
  33. self._num_visible_settings = 0
  34. self._num_user_settings = 0
  35. self._active_mode = ""
  36. self._quality_name = ""
  37. self._num_settings_overriden_by_quality_changes = 0
  38. self._quality_type = ""
  39. self._machine_name = ""
  40. self._machine_type = ""
  41. self._variant_type = ""
  42. self._material_labels = []
  43. self._extruders = []
  44. self._objects_on_plate = False
  45. machineConflictChanged = pyqtSignal()
  46. qualityChangesConflictChanged = pyqtSignal()
  47. materialConflictChanged = pyqtSignal()
  48. numVisibleSettingsChanged = pyqtSignal()
  49. activeModeChanged = pyqtSignal()
  50. qualityNameChanged = pyqtSignal()
  51. numSettingsOverridenByQualityChangesChanged = pyqtSignal()
  52. qualityTypeChanged = pyqtSignal()
  53. machineNameChanged = pyqtSignal()
  54. materialLabelsChanged = pyqtSignal()
  55. objectsOnPlateChanged = pyqtSignal()
  56. numUserSettingsChanged = pyqtSignal()
  57. machineTypeChanged = pyqtSignal()
  58. variantTypeChanged = pyqtSignal()
  59. extrudersChanged = pyqtSignal()
  60. @pyqtProperty(str, notify=variantTypeChanged)
  61. def variantType(self):
  62. return self._variant_type
  63. def setVariantType(self, variant_type):
  64. if self._variant_type != variant_type:
  65. self._variant_type = variant_type
  66. self.variantTypeChanged.emit()
  67. @pyqtProperty(str, notify=machineTypeChanged)
  68. def machineType(self):
  69. return self._machine_type
  70. def setMachineType(self, machine_type):
  71. self._machine_type = machine_type
  72. self.machineTypeChanged.emit()
  73. def setNumUserSettings(self, num_user_settings):
  74. if self._num_user_settings != num_user_settings:
  75. self._num_user_settings = num_user_settings
  76. self.numVisibleSettingsChanged.emit()
  77. @pyqtProperty(int, notify=numUserSettingsChanged)
  78. def numUserSettings(self):
  79. return self._num_user_settings
  80. @pyqtProperty(bool, notify=objectsOnPlateChanged)
  81. def hasObjectsOnPlate(self):
  82. return self._objects_on_plate
  83. def setHasObjectsOnPlate(self, objects_on_plate):
  84. if self._objects_on_plate != objects_on_plate:
  85. self._objects_on_plate = objects_on_plate
  86. self.objectsOnPlateChanged.emit()
  87. @pyqtProperty("QVariantList", notify = materialLabelsChanged)
  88. def materialLabels(self):
  89. return self._material_labels
  90. def setMaterialLabels(self, material_labels):
  91. if self._material_labels != material_labels:
  92. self._material_labels = material_labels
  93. self.materialLabelsChanged.emit()
  94. @pyqtProperty("QVariantList", notify=extrudersChanged)
  95. def extruders(self):
  96. return self._extruders
  97. def setExtruders(self, extruders):
  98. if self._extruders != extruders:
  99. self._extruders = extruders
  100. self.extrudersChanged.emit()
  101. @pyqtProperty(str, notify = machineNameChanged)
  102. def machineName(self):
  103. return self._machine_name
  104. def setMachineName(self, machine_name):
  105. if self._machine_name != machine_name:
  106. self._machine_name = machine_name
  107. self.machineNameChanged.emit()
  108. @pyqtProperty(str, notify=qualityTypeChanged)
  109. def qualityType(self):
  110. return self._quality_type
  111. def setQualityType(self, quality_type):
  112. if self._quality_type != quality_type:
  113. self._quality_type = quality_type
  114. self.qualityTypeChanged.emit()
  115. @pyqtProperty(int, notify=numSettingsOverridenByQualityChangesChanged)
  116. def numSettingsOverridenByQualityChanges(self):
  117. return self._num_settings_overriden_by_quality_changes
  118. def setNumSettingsOverridenByQualityChanges(self, num_settings_overriden_by_quality_changes):
  119. self._num_settings_overriden_by_quality_changes = num_settings_overriden_by_quality_changes
  120. self.numSettingsOverridenByQualityChangesChanged.emit()
  121. @pyqtProperty(str, notify=qualityNameChanged)
  122. def qualityName(self):
  123. return self._quality_name
  124. def setQualityName(self, quality_name):
  125. if self._quality_name != quality_name:
  126. self._quality_name = quality_name
  127. self.qualityNameChanged.emit()
  128. @pyqtProperty(str, notify=activeModeChanged)
  129. def activeMode(self):
  130. return self._active_mode
  131. def setActiveMode(self, active_mode):
  132. if active_mode == 0:
  133. self._active_mode = i18n_catalog.i18nc("@title:tab", "Recommended")
  134. else:
  135. self._active_mode = i18n_catalog.i18nc("@title:tab", "Custom")
  136. self.activeModeChanged.emit()
  137. @pyqtProperty(int, constant = True)
  138. def totalNumberOfSettings(self):
  139. return len(ContainerRegistry.getInstance().findDefinitionContainers(id="fdmprinter")[0].getAllKeys())
  140. @pyqtProperty(int, notify = numVisibleSettingsChanged)
  141. def numVisibleSettings(self):
  142. return self._num_visible_settings
  143. def setNumVisibleSettings(self, num_visible_settings):
  144. if self._num_visible_settings != num_visible_settings:
  145. self._num_visible_settings = num_visible_settings
  146. self.numVisibleSettingsChanged.emit()
  147. @pyqtProperty(bool, notify = machineConflictChanged)
  148. def machineConflict(self):
  149. return self._has_machine_conflict
  150. @pyqtProperty(bool, notify=qualityChangesConflictChanged)
  151. def qualityChangesConflict(self):
  152. return self._has_quality_changes_conflict
  153. @pyqtProperty(bool, notify=materialConflictChanged)
  154. def materialConflict(self):
  155. return self._has_material_conflict
  156. @pyqtSlot(str, str)
  157. def setResolveStrategy(self, key, strategy):
  158. if key in self._result:
  159. self._result[key] = strategy
  160. ## Close the backend: otherwise one could end up with "Slicing..."
  161. @pyqtSlot()
  162. def closeBackend(self):
  163. Application.getInstance().getBackend().close()
  164. def setMaterialConflict(self, material_conflict):
  165. if self._has_material_conflict != material_conflict:
  166. self._has_material_conflict = material_conflict
  167. self.materialConflictChanged.emit()
  168. def setMachineConflict(self, machine_conflict):
  169. if self._has_machine_conflict != machine_conflict:
  170. self._has_machine_conflict = machine_conflict
  171. self.machineConflictChanged.emit()
  172. def setQualityChangesConflict(self, quality_changes_conflict):
  173. if self._has_quality_changes_conflict != quality_changes_conflict:
  174. self._has_quality_changes_conflict = quality_changes_conflict
  175. self.qualityChangesConflictChanged.emit()
  176. def getResult(self):
  177. if "machine" in self._result and not self._has_machine_conflict:
  178. self._result["machine"] = None
  179. if "quality_changes" in self._result and not self._has_quality_changes_conflict:
  180. self._result["quality_changes"] = None
  181. if "material" in self._result and not self._has_material_conflict:
  182. self._result["material"] = None
  183. return self._result
  184. def _createViewFromQML(self):
  185. path = QUrl.fromLocalFile(os.path.join(PluginRegistry.getInstance().getPluginPath("3MFReader"), self._qml_url))
  186. self._component = QQmlComponent(Application.getInstance()._engine, path)
  187. self._context = QQmlContext(Application.getInstance()._engine.rootContext())
  188. self._context.setContextProperty("manager", self)
  189. self._view = self._component.create(self._context)
  190. if self._view is None:
  191. Logger.log("c", "QQmlComponent status %s", self._component.status())
  192. Logger.log("c", "QQmlComponent error string %s", self._component.errorString())
  193. def show(self):
  194. # Emit signal so the right thread actually shows the view.
  195. if threading.current_thread() != threading.main_thread():
  196. self._lock.acquire()
  197. # Reset the result
  198. self._result = {"machine": self._default_strategy,
  199. "quality_changes": self._default_strategy,
  200. "material": self._default_strategy}
  201. self._visible = True
  202. self.showDialogSignal.emit()
  203. @pyqtSlot()
  204. ## Used to notify the dialog so the lock can be released.
  205. def notifyClosed(self):
  206. self._result = {}
  207. self._visible = False
  208. self._lock.release()
  209. def hide(self):
  210. self._visible = False
  211. self._lock.release()
  212. self._view.hide()
  213. @pyqtSlot()
  214. def onOkButtonClicked(self):
  215. self._view.hide()
  216. self.hide()
  217. @pyqtSlot()
  218. def onCancelButtonClicked(self):
  219. self._view.hide()
  220. self.hide()
  221. self._result = {}
  222. ## Block thread until the dialog is closed.
  223. def waitForClose(self):
  224. if self._visible:
  225. if threading.current_thread() != threading.main_thread():
  226. self._lock.acquire()
  227. self._lock.release()
  228. else:
  229. # If this is not run from a separate thread, we need to ensure that the events are still processed.
  230. while self._visible:
  231. time.sleep(1 / 50)
  232. QCoreApplication.processEvents() # Ensure that the GUI does not freeze.
  233. def __show(self):
  234. if self._view is None:
  235. self._createViewFromQML()
  236. if self._view:
  237. self._view.show()