DiscoveredPrintersModel.py 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178
  1. # Copyright (c) 2019 Ultimaker B.V.
  2. # Cura is released under the terms of the LGPLv3 or higher.
  3. from typing import Callable, Dict, List, Optional, TYPE_CHECKING
  4. from PyQt5.QtCore import pyqtSlot, pyqtProperty, pyqtSignal, QObject
  5. from UM.i18n import i18nCatalog
  6. from UM.Logger import Logger
  7. from UM.Util import parseBool
  8. if TYPE_CHECKING:
  9. from PyQt5.QtCore import QObject
  10. from cura.PrinterOutput.NetworkedPrinterOutputDevice import NetworkedPrinterOutputDevice
  11. catalog = i18nCatalog("cura")
  12. class DiscoveredPrinter(QObject):
  13. def __init__(self, ip_address: str, key: str, name: str, create_callback: Callable[[str], None], machine_type: str,
  14. device: "NetworkedPrinterOutputDevice", parent: Optional["QObject"] = None) -> None:
  15. super().__init__(parent)
  16. self._ip_address = ip_address
  17. self._key = key
  18. self._name = name
  19. self.create_callback = create_callback
  20. self._machine_type = machine_type
  21. self._device = device
  22. nameChanged = pyqtSignal()
  23. def getKey(self) -> str:
  24. return self._key
  25. @pyqtProperty(str, notify = nameChanged)
  26. def name(self) -> str:
  27. return self._name
  28. def setName(self, name: str) -> None:
  29. if self._name != name:
  30. self._name = name
  31. self.nameChanged.emit()
  32. machineTypeChanged = pyqtSignal()
  33. @pyqtProperty(str, notify = machineTypeChanged)
  34. def machineType(self) -> str:
  35. return self._machine_type
  36. def setMachineType(self, machine_type: str) -> None:
  37. if self._machine_type != machine_type:
  38. self._machine_type = machine_type
  39. self.machineTypeChanged.emit()
  40. # Human readable machine type string
  41. @pyqtProperty(str, notify = machineTypeChanged)
  42. def readableMachineType(self) -> str:
  43. from cura.CuraApplication import CuraApplication
  44. machine_manager = CuraApplication.getInstance().getMachineManager()
  45. # In ClusterUM3OutputDevice, when it updates a printer information, it updates the machine type using the field
  46. # "machine_variant", and for some reason, it's not the machine type ID/codename/... but a human-readable string
  47. # like "Ultimaker 3". The code below handles this case.
  48. if machine_manager.hasMachineTypeName(self._machine_type):
  49. readable_type = self._machine_type
  50. else:
  51. readable_type = machine_manager.getMachineTypeNameFromId(self._machine_type)
  52. if not readable_type:
  53. readable_type = catalog.i18nc("@label", "Unknown")
  54. return readable_type
  55. @pyqtProperty(bool, notify = machineTypeChanged)
  56. def isUnknownMachineType(self) -> bool:
  57. from cura.CuraApplication import CuraApplication
  58. machine_manager = CuraApplication.getInstance().getMachineManager()
  59. if machine_manager.hasMachineTypeName(self._machine_type):
  60. readable_type = self._machine_type
  61. else:
  62. readable_type = machine_manager.getMachineTypeNameFromId(self._machine_type)
  63. return not readable_type
  64. @pyqtProperty(QObject, constant = True)
  65. def device(self) -> "NetworkedPrinterOutputDevice":
  66. return self._device
  67. @pyqtProperty(bool, constant = True)
  68. def isHostOfGroup(self) -> bool:
  69. return getattr(self._device, "clusterSize", 1) > 0
  70. @pyqtProperty(str, constant = True)
  71. def sectionName(self) -> str:
  72. if self.isUnknownMachineType or not self.isHostOfGroup:
  73. return catalog.i18nc("@label", "The printer(s) below cannot be connected because they are part of a group")
  74. else:
  75. return catalog.i18nc("@label", "Available networked printers")
  76. #
  77. # Discovered printers are all the printers that were found on the network, which provide a more convenient way
  78. # to add networked printers (Plugin finds a bunch of printers, user can select one from the list, plugin can then
  79. # add that printer to Cura as the active one).
  80. #
  81. class DiscoveredPrintersModel(QObject):
  82. def __init__(self, parent: Optional["QObject"] = None) -> None:
  83. super().__init__(parent)
  84. self._discovered_printer_by_ip_dict = dict() # type: Dict[str, DiscoveredPrinter]
  85. discoveredPrintersChanged = pyqtSignal()
  86. @pyqtProperty(list, notify = discoveredPrintersChanged)
  87. def discoveredPrinters(self) -> List["DiscoveredPrinter"]:
  88. item_list = list(
  89. x for x in self._discovered_printer_by_ip_dict.values() if not parseBool(x.device.getProperty("temporary")))
  90. # Split the printers into 2 lists and sort them ascending based on names.
  91. available_list = []
  92. not_available_list = []
  93. for item in item_list:
  94. if item.isUnknownMachineType or getattr(item.device, "clusterSize", 1) < 1:
  95. not_available_list.append(item)
  96. else:
  97. available_list.append(item)
  98. available_list.sort(key = lambda x: x.device.name)
  99. not_available_list.sort(key = lambda x: x.device.name)
  100. return available_list + not_available_list
  101. def addDiscoveredPrinter(self, ip_address: str, key: str, name: str, create_callback: Callable[[str], None],
  102. machine_type: str, device: "NetworkedPrinterOutputDevice") -> None:
  103. if ip_address in self._discovered_printer_by_ip_dict:
  104. Logger.log("e", "Printer with ip [%s] has already been added", ip_address)
  105. return
  106. discovered_printer = DiscoveredPrinter(ip_address, key, name, create_callback, machine_type, device, parent = self)
  107. self._discovered_printer_by_ip_dict[ip_address] = discovered_printer
  108. self.discoveredPrintersChanged.emit()
  109. def updateDiscoveredPrinter(self, ip_address: str,
  110. name: Optional[str] = None,
  111. machine_type: Optional[str] = None) -> None:
  112. if ip_address not in self._discovered_printer_by_ip_dict:
  113. Logger.log("w", "Printer with ip [%s] is not known", ip_address)
  114. return
  115. item = self._discovered_printer_by_ip_dict[ip_address]
  116. if name is not None:
  117. item.setName(name)
  118. if machine_type is not None:
  119. item.setMachineType(machine_type)
  120. def removeDiscoveredPrinter(self, ip_address: str) -> None:
  121. if ip_address not in self._discovered_printer_by_ip_dict:
  122. Logger.log("w", "Key [%s] does not exist in the discovered printers list.", ip_address)
  123. return
  124. del self._discovered_printer_by_ip_dict[ip_address]
  125. self.discoveredPrintersChanged.emit()
  126. # A convenience function for QML to create a machine (GlobalStack) out of the given discovered printer.
  127. # This function invokes the given discovered printer's "create_callback" to do this.
  128. @pyqtSlot("QVariant")
  129. def createMachineFromDiscoveredPrinter(self, discovered_printer: "DiscoveredPrinter") -> None:
  130. discovered_printer.create_callback(discovered_printer.getKey())
  131. @pyqtSlot(str)
  132. def createMachineFromDiscoveredPrinterAddress(self, ip_address: str) -> None:
  133. if ip_address not in self._discovered_printer_by_ip_dict:
  134. Logger.log("i", "Key [%s] does not exist in the discovered printers list.", ip_address)
  135. return
  136. self.createMachineFromDiscoveredPrinter(self._discovered_printer_by_ip_dict[ip_address])