WorkspaceDialog.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333
  1. # Copyright (c) 2016 Ultimaker B.V.
  2. # Cura is released under the terms of the LGPLv3 or higher.
  3. from PyQt5.QtCore import 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 = None
  25. self._result = {"machine": self._default_strategy,
  26. "quality_changes": self._default_strategy,
  27. "definition_changes": self._default_strategy,
  28. "material": self._default_strategy}
  29. self._visible = False
  30. self.showDialogSignal.connect(self.__show)
  31. self._has_quality_changes_conflict = False
  32. self._has_definition_changes_conflict = False
  33. self._has_machine_conflict = False
  34. self._has_material_conflict = False
  35. self._has_visible_settings_field = False
  36. self._num_visible_settings = 0
  37. self._num_user_settings = 0
  38. self._active_mode = ""
  39. self._quality_name = ""
  40. self._num_settings_overriden_by_quality_changes = 0
  41. self._quality_type = ""
  42. self._machine_name = ""
  43. self._machine_type = ""
  44. self._variant_type = ""
  45. self._material_labels = []
  46. self._extruders = []
  47. self._objects_on_plate = False
  48. machineConflictChanged = pyqtSignal()
  49. qualityChangesConflictChanged = pyqtSignal()
  50. definitionChangesConflictChanged = pyqtSignal()
  51. materialConflictChanged = pyqtSignal()
  52. numVisibleSettingsChanged = pyqtSignal()
  53. activeModeChanged = pyqtSignal()
  54. qualityNameChanged = pyqtSignal()
  55. hasVisibleSettingsFieldChanged = pyqtSignal()
  56. numSettingsOverridenByQualityChangesChanged = pyqtSignal()
  57. qualityTypeChanged = pyqtSignal()
  58. machineNameChanged = pyqtSignal()
  59. materialLabelsChanged = pyqtSignal()
  60. objectsOnPlateChanged = pyqtSignal()
  61. numUserSettingsChanged = pyqtSignal()
  62. machineTypeChanged = pyqtSignal()
  63. variantTypeChanged = pyqtSignal()
  64. extrudersChanged = pyqtSignal()
  65. @pyqtProperty(str, notify=variantTypeChanged)
  66. def variantType(self):
  67. return self._variant_type
  68. def setVariantType(self, variant_type):
  69. if self._variant_type != variant_type:
  70. self._variant_type = variant_type
  71. self.variantTypeChanged.emit()
  72. @pyqtProperty(str, notify=machineTypeChanged)
  73. def machineType(self):
  74. return self._machine_type
  75. def setMachineType(self, machine_type):
  76. self._machine_type = machine_type
  77. self.machineTypeChanged.emit()
  78. def setNumUserSettings(self, num_user_settings):
  79. if self._num_user_settings != num_user_settings:
  80. self._num_user_settings = num_user_settings
  81. self.numVisibleSettingsChanged.emit()
  82. @pyqtProperty(int, notify=numUserSettingsChanged)
  83. def numUserSettings(self):
  84. return self._num_user_settings
  85. @pyqtProperty(bool, notify=objectsOnPlateChanged)
  86. def hasObjectsOnPlate(self):
  87. return self._objects_on_plate
  88. def setHasObjectsOnPlate(self, objects_on_plate):
  89. if self._objects_on_plate != objects_on_plate:
  90. self._objects_on_plate = objects_on_plate
  91. self.objectsOnPlateChanged.emit()
  92. @pyqtProperty("QVariantList", notify = materialLabelsChanged)
  93. def materialLabels(self):
  94. return self._material_labels
  95. def setMaterialLabels(self, material_labels):
  96. if self._material_labels != material_labels:
  97. self._material_labels = material_labels
  98. self.materialLabelsChanged.emit()
  99. @pyqtProperty("QVariantList", notify=extrudersChanged)
  100. def extruders(self):
  101. return self._extruders
  102. def setExtruders(self, extruders):
  103. if self._extruders != extruders:
  104. self._extruders = extruders
  105. self.extrudersChanged.emit()
  106. @pyqtProperty(str, notify = machineNameChanged)
  107. def machineName(self):
  108. return self._machine_name
  109. def setMachineName(self, machine_name):
  110. if self._machine_name != machine_name:
  111. self._machine_name = machine_name
  112. self.machineNameChanged.emit()
  113. @pyqtProperty(str, notify=qualityTypeChanged)
  114. def qualityType(self):
  115. return self._quality_type
  116. def setQualityType(self, quality_type):
  117. if self._quality_type != quality_type:
  118. self._quality_type = quality_type
  119. self.qualityTypeChanged.emit()
  120. @pyqtProperty(int, notify=numSettingsOverridenByQualityChangesChanged)
  121. def numSettingsOverridenByQualityChanges(self):
  122. return self._num_settings_overriden_by_quality_changes
  123. def setNumSettingsOverridenByQualityChanges(self, num_settings_overriden_by_quality_changes):
  124. self._num_settings_overriden_by_quality_changes = num_settings_overriden_by_quality_changes
  125. self.numSettingsOverridenByQualityChangesChanged.emit()
  126. @pyqtProperty(str, notify=qualityNameChanged)
  127. def qualityName(self):
  128. return self._quality_name
  129. def setQualityName(self, quality_name):
  130. if self._quality_name != quality_name:
  131. self._quality_name = quality_name
  132. self.qualityNameChanged.emit()
  133. @pyqtProperty(str, notify=activeModeChanged)
  134. def activeMode(self):
  135. return self._active_mode
  136. def setActiveMode(self, active_mode):
  137. if active_mode == 0:
  138. self._active_mode = i18n_catalog.i18nc("@title:tab", "Recommended")
  139. else:
  140. self._active_mode = i18n_catalog.i18nc("@title:tab", "Custom")
  141. self.activeModeChanged.emit()
  142. @pyqtProperty(int, notify = hasVisibleSettingsFieldChanged)
  143. def hasVisibleSettingsField(self):
  144. return self._has_visible_settings_field
  145. def setHasVisibleSettingsField(self, has_visible_settings_field):
  146. self._has_visible_settings_field = has_visible_settings_field
  147. self.hasVisibleSettingsFieldChanged.emit()
  148. @pyqtProperty(int, constant = True)
  149. def totalNumberOfSettings(self):
  150. return len(ContainerRegistry.getInstance().findDefinitionContainers(id="fdmprinter")[0].getAllKeys())
  151. @pyqtProperty(int, notify = numVisibleSettingsChanged)
  152. def numVisibleSettings(self):
  153. return self._num_visible_settings
  154. def setNumVisibleSettings(self, num_visible_settings):
  155. if self._num_visible_settings != num_visible_settings:
  156. self._num_visible_settings = num_visible_settings
  157. self.numVisibleSettingsChanged.emit()
  158. @pyqtProperty(bool, notify = machineConflictChanged)
  159. def machineConflict(self):
  160. return self._has_machine_conflict
  161. @pyqtProperty(bool, notify=qualityChangesConflictChanged)
  162. def qualityChangesConflict(self):
  163. return self._has_quality_changes_conflict
  164. @pyqtProperty(bool, notify=definitionChangesConflictChanged)
  165. def definitionChangesConflict(self):
  166. return self._has_definition_changes_conflict
  167. @pyqtProperty(bool, notify=materialConflictChanged)
  168. def materialConflict(self):
  169. return self._has_material_conflict
  170. @pyqtSlot(str, str)
  171. def setResolveStrategy(self, key, strategy):
  172. if key in self._result:
  173. self._result[key] = strategy
  174. ## Close the backend: otherwise one could end up with "Slicing..."
  175. @pyqtSlot()
  176. def closeBackend(self):
  177. Application.getInstance().getBackend().close()
  178. def setMaterialConflict(self, material_conflict):
  179. if self._has_material_conflict != material_conflict:
  180. self._has_material_conflict = material_conflict
  181. self.materialConflictChanged.emit()
  182. def setMachineConflict(self, machine_conflict):
  183. if self._has_machine_conflict != machine_conflict:
  184. self._has_machine_conflict = machine_conflict
  185. self.machineConflictChanged.emit()
  186. def setQualityChangesConflict(self, quality_changes_conflict):
  187. if self._has_quality_changes_conflict != quality_changes_conflict:
  188. self._has_quality_changes_conflict = quality_changes_conflict
  189. self.qualityChangesConflictChanged.emit()
  190. def setDefinitionChangesConflict(self, definition_changes_conflict):
  191. if self._has_definition_changes_conflict != definition_changes_conflict:
  192. self._has_definition_changes_conflict = definition_changes_conflict
  193. self.definitionChangesConflictChanged.emit()
  194. def getResult(self):
  195. if "machine" in self._result and not self._has_machine_conflict:
  196. self._result["machine"] = None
  197. if "quality_changes" in self._result and not self._has_quality_changes_conflict:
  198. self._result["quality_changes"] = None
  199. if "definition_changes" in self._result and not self._has_definition_changes_conflict:
  200. self._result["definition_changes"] = None
  201. if "material" in self._result and not self._has_material_conflict:
  202. self._result["material"] = None
  203. # If the machine needs to be re-created, the definition_changes should also be re-created.
  204. # If the machine strategy is None, it means that there is no name conflict with existing ones. In this case
  205. # new definitions changes are created
  206. if "machine" in self._result:
  207. if self._result["machine"] == "new" or self._result["machine"] is None and self._result["definition_changes"] is None:
  208. self._result["definition_changes"] = "new"
  209. return self._result
  210. def _createViewFromQML(self):
  211. path = QUrl.fromLocalFile(os.path.join(PluginRegistry.getInstance().getPluginPath("3MFReader"), self._qml_url))
  212. self._component = QQmlComponent(Application.getInstance()._engine, path)
  213. self._context = QQmlContext(Application.getInstance()._engine.rootContext())
  214. self._context.setContextProperty("manager", self)
  215. self._view = self._component.create(self._context)
  216. if self._view is None:
  217. Logger.log("c", "QQmlComponent status %s", self._component.status())
  218. Logger.log("c", "QQmlComponent error string %s", self._component.errorString())
  219. def show(self):
  220. # Emit signal so the right thread actually shows the view.
  221. if threading.current_thread() != threading.main_thread():
  222. self._lock.acquire()
  223. # Reset the result
  224. self._result = {"machine": self._default_strategy,
  225. "quality_changes": self._default_strategy,
  226. "definition_changes": self._default_strategy,
  227. "material": self._default_strategy}
  228. self._visible = True
  229. self.showDialogSignal.emit()
  230. @pyqtSlot()
  231. ## Used to notify the dialog so the lock can be released.
  232. def notifyClosed(self):
  233. self._result = {} # The result should be cleared before hide, because after it is released the main thread lock
  234. self._visible = False
  235. try:
  236. self._lock.release()
  237. except:
  238. pass
  239. def hide(self):
  240. self._visible = False
  241. self._view.hide()
  242. try:
  243. self._lock.release()
  244. except:
  245. pass
  246. @pyqtSlot(bool)
  247. def _onVisibilityChanged(self, visible):
  248. if not visible:
  249. try:
  250. self._lock.release()
  251. except:
  252. pass
  253. @pyqtSlot()
  254. def onOkButtonClicked(self):
  255. self._view.hide()
  256. self.hide()
  257. @pyqtSlot()
  258. def onCancelButtonClicked(self):
  259. self._result = {}
  260. self._view.hide()
  261. self.hide()
  262. ## Block thread until the dialog is closed.
  263. def waitForClose(self):
  264. if self._visible:
  265. if threading.current_thread() != threading.main_thread():
  266. self._lock.acquire()
  267. self._lock.release()
  268. else:
  269. # If this is not run from a separate thread, we need to ensure that the events are still processed.
  270. while self._visible:
  271. time.sleep(1 / 50)
  272. QCoreApplication.processEvents() # Ensure that the GUI does not freeze.
  273. def __show(self):
  274. if self._view is None:
  275. self._createViewFromQML()
  276. if self._view:
  277. self._view.show()