MaterialManager.py 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535
  1. # Copyright (c) 2018 Ultimaker B.V.
  2. # Cura is released under the terms of the LGPLv3 or higher.
  3. from collections import defaultdict, OrderedDict
  4. import copy
  5. import uuid
  6. from typing import Optional, TYPE_CHECKING
  7. from PyQt5.Qt import QTimer, QObject, pyqtSignal, pyqtSlot
  8. from UM.Application import Application
  9. from UM.ConfigurationErrorMessage import ConfigurationErrorMessage
  10. from UM.Logger import Logger
  11. from UM.Settings.ContainerRegistry import ContainerRegistry
  12. from UM.Settings.SettingFunction import SettingFunction
  13. from UM.Util import parseBool
  14. from .MaterialNode import MaterialNode
  15. from .MaterialGroup import MaterialGroup
  16. if TYPE_CHECKING:
  17. from UM.Settings.DefinitionContainer import DefinitionContainer
  18. from UM.Settings.InstanceContainer import InstanceContainer
  19. from cura.Settings.GlobalStack import GlobalStack
  20. from cura.Settings.ExtruderStack import ExtruderStack
  21. #
  22. # MaterialManager maintains a number of maps and trees for material lookup.
  23. # The models GUI and QML use are now only dependent on the MaterialManager. That means as long as the data in
  24. # MaterialManager gets updated correctly, the GUI models should be updated correctly too, and the same goes for GUI.
  25. #
  26. # For now, updating the lookup maps and trees here is very simple: we discard the old data completely and recreate them
  27. # again. This means the update is exactly the same as initialization. There are performance concerns about this approach
  28. # but so far the creation of the tables and maps is very fast and there is no noticeable slowness, we keep it like this
  29. # because it's simple.
  30. #
  31. class MaterialManager(QObject):
  32. materialsUpdated = pyqtSignal() # Emitted whenever the material lookup tables are updated.
  33. def __init__(self, container_registry, parent = None):
  34. super().__init__(parent)
  35. self._application = Application.getInstance()
  36. self._container_registry = container_registry # type: ContainerRegistry
  37. self._fallback_materials_map = dict() # material_type -> generic material metadata
  38. self._material_group_map = dict() # root_material_id -> MaterialGroup
  39. self._diameter_machine_variant_material_map = dict() # approximate diameter str -> dict(machine_definition_id -> MaterialNode)
  40. # We're using these two maps to convert between the specific diameter material id and the generic material id
  41. # because the generic material ids are used in qualities and definitions, while the specific diameter material is meant
  42. # i.e. generic_pla -> generic_pla_175
  43. self._material_diameter_map = defaultdict(dict) # root_material_id -> approximate diameter str -> root_material_id for that diameter
  44. self._diameter_material_map = dict() # material id including diameter (generic_pla_175) -> material root id (generic_pla)
  45. # This is used in Legacy UM3 send material function and the material management page.
  46. self._guid_material_groups_map = defaultdict(list) # GUID -> a list of material_groups
  47. # The machine definition ID for the non-machine-specific materials.
  48. # This is used as the last fallback option if the given machine-specific material(s) cannot be found.
  49. self._default_machine_definition_id = "fdmprinter"
  50. self._default_approximate_diameter_for_quality_search = "3"
  51. # When a material gets added/imported, there can be more than one InstanceContainers. In those cases, we don't
  52. # want to react on every container/metadata changed signal. The timer here is to buffer it a bit so we don't
  53. # react too many time.
  54. self._update_timer = QTimer(self)
  55. self._update_timer.setInterval(300)
  56. self._update_timer.setSingleShot(True)
  57. self._update_timer.timeout.connect(self._updateMaps)
  58. self._container_registry.containerMetaDataChanged.connect(self._onContainerMetadataChanged)
  59. self._container_registry.containerAdded.connect(self._onContainerMetadataChanged)
  60. self._container_registry.containerRemoved.connect(self._onContainerMetadataChanged)
  61. def initialize(self):
  62. # Find all materials and put them in a matrix for quick search.
  63. material_metadatas = {metadata["id"]: metadata for metadata in self._container_registry.findContainersMetadata(type = "material")}
  64. self._material_group_map = dict()
  65. # Map #1
  66. # root_material_id -> MaterialGroup
  67. for material_id, material_metadata in material_metadatas.items():
  68. # We don't store empty material in the lookup tables
  69. if material_id == "empty_material":
  70. continue
  71. root_material_id = material_metadata.get("base_file")
  72. if root_material_id not in self._material_group_map:
  73. self._material_group_map[root_material_id] = MaterialGroup(root_material_id, MaterialNode(material_metadatas[root_material_id]))
  74. self._material_group_map[root_material_id].is_read_only = self._container_registry.isReadOnly(root_material_id)
  75. group = self._material_group_map[root_material_id]
  76. #Store this material in the group of the appropriate root material.
  77. if material_id != root_material_id:
  78. new_node = MaterialNode(material_metadata)
  79. group.derived_material_node_list.append(new_node)
  80. # Order this map alphabetically so it's easier to navigate in a debugger
  81. self._material_group_map = OrderedDict(sorted(self._material_group_map.items(), key = lambda x: x[0]))
  82. # Map #1.5
  83. # GUID -> material group list
  84. self._guid_material_groups_map = defaultdict(list)
  85. for root_material_id, material_group in self._material_group_map.items():
  86. guid = material_group.root_material_node.metadata["GUID"]
  87. self._guid_material_groups_map[guid].append(material_group)
  88. # Map #2
  89. # Lookup table for material type -> fallback material metadata, only for read-only materials
  90. grouped_by_type_dict = dict()
  91. material_types_without_fallback = set()
  92. for root_material_id, material_node in self._material_group_map.items():
  93. if not self._container_registry.isReadOnly(root_material_id):
  94. continue
  95. material_type = material_node.root_material_node.metadata["material"]
  96. if material_type not in grouped_by_type_dict:
  97. grouped_by_type_dict[material_type] = {"generic": None,
  98. "others": []}
  99. material_types_without_fallback.add(material_type)
  100. brand = material_node.root_material_node.metadata["brand"]
  101. if brand.lower() == "generic":
  102. to_add = True
  103. if material_type in grouped_by_type_dict:
  104. diameter = material_node.root_material_node.metadata.get("approximate_diameter")
  105. if diameter != self._default_approximate_diameter_for_quality_search:
  106. to_add = False # don't add if it's not the default diameter
  107. if to_add:
  108. grouped_by_type_dict[material_type] = material_node.root_material_node.metadata
  109. material_types_without_fallback.remove(material_type)
  110. # Remove the materials that have no fallback materials
  111. for material_type in material_types_without_fallback:
  112. del grouped_by_type_dict[material_type]
  113. self._fallback_materials_map = grouped_by_type_dict
  114. # Map #3
  115. # There can be multiple material profiles for the same material with different diameters, such as "generic_pla"
  116. # and "generic_pla_175". This is inconvenient when we do material-specific quality lookup because a quality can
  117. # be for either "generic_pla" or "generic_pla_175", but not both. This map helps to get the correct material ID
  118. # for quality search.
  119. self._material_diameter_map = defaultdict(dict)
  120. self._diameter_material_map = dict()
  121. # Group the material IDs by the same name, material, brand, and color but with different diameters.
  122. material_group_dict = dict()
  123. keys_to_fetch = ("name", "material", "brand", "color")
  124. for root_material_id, machine_node in self._material_group_map.items():
  125. if not self._container_registry.isReadOnly(root_material_id):
  126. continue
  127. root_material_metadata = machine_node.root_material_node.metadata
  128. key_data = []
  129. for key in keys_to_fetch:
  130. key_data.append(root_material_metadata.get(key))
  131. key_data = tuple(key_data)
  132. if key_data not in material_group_dict:
  133. material_group_dict[key_data] = dict()
  134. approximate_diameter = root_material_metadata.get("approximate_diameter")
  135. material_group_dict[key_data][approximate_diameter] = root_material_metadata["id"]
  136. # Map [root_material_id][diameter] -> root_material_id for this diameter
  137. for data_dict in material_group_dict.values():
  138. for root_material_id1 in data_dict.values():
  139. if root_material_id1 in self._material_diameter_map:
  140. continue
  141. diameter_map = data_dict
  142. for root_material_id2 in data_dict.values():
  143. self._material_diameter_map[root_material_id2] = diameter_map
  144. default_root_material_id = data_dict.get(self._default_approximate_diameter_for_quality_search)
  145. if default_root_material_id is None:
  146. default_root_material_id = list(data_dict.values())[0] # no default diameter present, just take "the" only one
  147. for root_material_id in data_dict.values():
  148. self._diameter_material_map[root_material_id] = default_root_material_id
  149. # Map #4
  150. # "machine" -> "variant_name" -> "root material ID" -> specific material InstanceContainer
  151. # Construct the "machine" -> "variant" -> "root material ID" -> specific material InstanceContainer
  152. self._diameter_machine_variant_material_map = dict()
  153. for material_metadata in material_metadatas.values():
  154. # We don't store empty material in the lookup tables
  155. if material_metadata["id"] == "empty_material":
  156. continue
  157. root_material_id = material_metadata["base_file"]
  158. definition = material_metadata["definition"]
  159. approximate_diameter = material_metadata["approximate_diameter"]
  160. if approximate_diameter not in self._diameter_machine_variant_material_map:
  161. self._diameter_machine_variant_material_map[approximate_diameter] = {}
  162. machine_variant_material_map = self._diameter_machine_variant_material_map[approximate_diameter]
  163. if definition not in machine_variant_material_map:
  164. machine_variant_material_map[definition] = MaterialNode()
  165. machine_node = machine_variant_material_map[definition]
  166. variant_name = material_metadata.get("variant_name")
  167. if not variant_name:
  168. # if there is no variant, this material is for the machine, so put its metadata in the machine node.
  169. machine_node.material_map[root_material_id] = MaterialNode(material_metadata)
  170. else:
  171. # this material is variant-specific, so we save it in a variant-specific node under the
  172. # machine-specific node
  173. if variant_name not in machine_node.children_map:
  174. machine_node.children_map[variant_name] = MaterialNode()
  175. variant_node = machine_node.children_map[variant_name]
  176. if root_material_id in variant_node.material_map: #We shouldn't have duplicated variant-specific materials for the same machine.
  177. ConfigurationErrorMessage.getInstance().addFaultyContainers(root_material_id)
  178. continue
  179. variant_node.material_map[root_material_id] = MaterialNode(material_metadata)
  180. self.materialsUpdated.emit()
  181. def _updateMaps(self):
  182. Logger.log("i", "Updating material lookup data ...")
  183. self.initialize()
  184. def _onContainerMetadataChanged(self, container):
  185. self._onContainerChanged(container)
  186. def _onContainerChanged(self, container):
  187. container_type = container.getMetaDataEntry("type")
  188. if container_type != "material":
  189. return
  190. # update the maps
  191. self._update_timer.start()
  192. def getMaterialGroup(self, root_material_id: str) -> Optional[MaterialGroup]:
  193. return self._material_group_map.get(root_material_id)
  194. def getRootMaterialIDForDiameter(self, root_material_id: str, approximate_diameter: str) -> str:
  195. return self._material_diameter_map.get(root_material_id, {}).get(approximate_diameter, root_material_id)
  196. def getRootMaterialIDWithoutDiameter(self, root_material_id: str) -> str:
  197. return self._diameter_material_map.get(root_material_id)
  198. def getMaterialGroupListByGUID(self, guid: str) -> Optional[list]:
  199. return self._guid_material_groups_map.get(guid)
  200. #
  201. # Return a dict with all root material IDs (k) and ContainerNodes (v) that's suitable for the given setup.
  202. #
  203. def getAvailableMaterials(self, machine_definition: "DefinitionContainer", extruder_variant_name: Optional[str],
  204. diameter: float) -> dict:
  205. # round the diameter to get the approximate diameter
  206. rounded_diameter = str(round(diameter))
  207. if rounded_diameter not in self._diameter_machine_variant_material_map:
  208. Logger.log("i", "Cannot find materials with diameter [%s] (rounded to [%s])", diameter, rounded_diameter)
  209. return dict()
  210. machine_definition_id = machine_definition.getId()
  211. # If there are variant materials, get the variant material
  212. machine_variant_material_map = self._diameter_machine_variant_material_map[rounded_diameter]
  213. machine_node = machine_variant_material_map.get(machine_definition_id)
  214. default_machine_node = machine_variant_material_map.get(self._default_machine_definition_id)
  215. variant_node = None
  216. if extruder_variant_name is not None and machine_node is not None:
  217. variant_node = machine_node.getChildNode(extruder_variant_name)
  218. nodes_to_check = [variant_node, machine_node, default_machine_node]
  219. # Fallback mechanism of finding materials:
  220. # 1. variant-specific material
  221. # 2. machine-specific material
  222. # 3. generic material (for fdmprinter)
  223. machine_exclude_materials = machine_definition.getMetaDataEntry("exclude_materials", [])
  224. material_id_metadata_dict = dict()
  225. for node in nodes_to_check:
  226. if node is not None:
  227. for material_id, node in node.material_map.items():
  228. fallback_id = self.getFallbackMaterialIdByMaterialType(node.metadata["material"])
  229. if fallback_id in machine_exclude_materials:
  230. Logger.log("d", "Exclude material [%s] for machine [%s]",
  231. material_id, machine_definition.getId())
  232. continue
  233. if material_id not in material_id_metadata_dict:
  234. material_id_metadata_dict[material_id] = node
  235. return material_id_metadata_dict
  236. #
  237. # A convenience function to get available materials for the given machine with the extruder position.
  238. #
  239. def getAvailableMaterialsForMachineExtruder(self, machine: "GlobalStack",
  240. extruder_stack: "ExtruderStack") -> Optional[dict]:
  241. variant_name = None
  242. if extruder_stack.variant.getId() != "empty_variant":
  243. variant_name = extruder_stack.variant.getName()
  244. diameter = extruder_stack.approximateMaterialDiameter
  245. # Fetch the available materials (ContainerNode) for the current active machine and extruder setup.
  246. return self.getAvailableMaterials(machine.definition, variant_name, diameter)
  247. #
  248. # Gets MaterialNode for the given extruder and machine with the given material name.
  249. # Returns None if:
  250. # 1. the given machine doesn't have materials;
  251. # 2. cannot find any material InstanceContainers with the given settings.
  252. #
  253. def getMaterialNode(self, machine_definition_id: str, extruder_variant_name: Optional[str],
  254. diameter: float, root_material_id: str) -> Optional["InstanceContainer"]:
  255. # round the diameter to get the approximate diameter
  256. rounded_diameter = str(round(diameter))
  257. if rounded_diameter not in self._diameter_machine_variant_material_map:
  258. Logger.log("i", "Cannot find materials with diameter [%s] (rounded to [%s]) for root material id [%s]",
  259. diameter, rounded_diameter, root_material_id)
  260. return None
  261. # If there are variant materials, get the variant material
  262. machine_variant_material_map = self._diameter_machine_variant_material_map[rounded_diameter]
  263. machine_node = machine_variant_material_map.get(machine_definition_id)
  264. variant_node = None
  265. # Fallback for "fdmprinter" if the machine-specific materials cannot be found
  266. if machine_node is None:
  267. machine_node = machine_variant_material_map.get(self._default_machine_definition_id)
  268. if machine_node is not None and extruder_variant_name is not None:
  269. variant_node = machine_node.getChildNode(extruder_variant_name)
  270. # Fallback mechanism of finding materials:
  271. # 1. variant-specific material
  272. # 2. machine-specific material
  273. # 3. generic material (for fdmprinter)
  274. nodes_to_check = [variant_node, machine_node,
  275. machine_variant_material_map.get(self._default_machine_definition_id)]
  276. material_node = None
  277. for node in nodes_to_check:
  278. if node is not None:
  279. material_node = node.material_map.get(root_material_id)
  280. if material_node:
  281. break
  282. return material_node
  283. #
  284. # Gets MaterialNode for the given extruder and machine with the given material type.
  285. # Returns None if:
  286. # 1. the given machine doesn't have materials;
  287. # 2. cannot find any material InstanceContainers with the given settings.
  288. #
  289. def getMaterialNodeByType(self, global_stack: "GlobalStack", extruder_variant_name: str, material_guid: str) -> Optional["MaterialNode"]:
  290. node = None
  291. machine_definition = global_stack.definition
  292. if parseBool(machine_definition.getMetaDataEntry("has_materials", False)):
  293. material_diameter = machine_definition.getProperty("material_diameter", "value")
  294. if isinstance(material_diameter, SettingFunction):
  295. material_diameter = material_diameter(global_stack)
  296. # Look at the guid to material dictionary
  297. root_material_id = None
  298. for material_group in self._guid_material_groups_map[material_guid]:
  299. if material_group.is_read_only:
  300. root_material_id = material_group.root_material_node.metadata["id"]
  301. break
  302. if not root_material_id:
  303. Logger.log("i", "Cannot find materials with guid [%s] ", material_guid)
  304. return None
  305. node = self.getMaterialNode(machine_definition.getId(), extruder_variant_name,
  306. material_diameter, root_material_id)
  307. return node
  308. #
  309. # Used by QualityManager. Built-in quality profiles may be based on generic material IDs such as "generic_pla".
  310. # For materials such as ultimaker_pla_orange, no quality profiles may be found, so we should fall back to use
  311. # the generic material IDs to search for qualities.
  312. #
  313. # An example would be, suppose we have machine with preferred material set to "filo3d_pla" (1.75mm), but its
  314. # extruders only use 2.85mm materials, then we won't be able to find the preferred material for this machine.
  315. # A fallback would be to fetch a generic material of the same type "PLA" as "filo3d_pla", and in this case it will
  316. # be "generic_pla". This function is intended to get a generic fallback material for the given material type.
  317. #
  318. # This function returns the generic root material ID for the given material type, where material types are "PLA",
  319. # "ABS", etc.
  320. #
  321. def getFallbackMaterialIdByMaterialType(self, material_type: str) -> Optional[str]:
  322. # For safety
  323. if material_type not in self._fallback_materials_map:
  324. Logger.log("w", "The material type [%s] does not have a fallback material" % material_type)
  325. return None
  326. fallback_material = self._fallback_materials_map[material_type]
  327. if fallback_material:
  328. return self.getRootMaterialIDWithoutDiameter(fallback_material["id"])
  329. else:
  330. return None
  331. def getDefaultMaterial(self, global_stack: "GlobalStack", extruder_variant_name: Optional[str]) -> Optional["MaterialNode"]:
  332. node = None
  333. machine_definition = global_stack.definition
  334. if parseBool(global_stack.getMetaDataEntry("has_materials", False)):
  335. material_diameter = machine_definition.getProperty("material_diameter", "value")
  336. if isinstance(material_diameter, SettingFunction):
  337. material_diameter = material_diameter(global_stack)
  338. approximate_material_diameter = str(round(material_diameter))
  339. root_material_id = machine_definition.getMetaDataEntry("preferred_material")
  340. root_material_id = self.getRootMaterialIDForDiameter(root_material_id, approximate_material_diameter)
  341. node = self.getMaterialNode(machine_definition.getId(), extruder_variant_name,
  342. material_diameter, root_material_id)
  343. return node
  344. def removeMaterialByRootId(self, root_material_id: str):
  345. material_group = self.getMaterialGroup(root_material_id)
  346. if not material_group:
  347. Logger.log("i", "Unable to remove the material with id %s, because it doesn't exist.", root_material_id)
  348. return
  349. nodes_to_remove = [material_group.root_material_node] + material_group.derived_material_node_list
  350. for node in nodes_to_remove:
  351. self._container_registry.removeContainer(node.metadata["id"])
  352. #
  353. # Methods for GUI
  354. #
  355. #
  356. # Sets the new name for the given material.
  357. #
  358. @pyqtSlot("QVariant", str)
  359. def setMaterialName(self, material_node: "MaterialNode", name: str):
  360. root_material_id = material_node.metadata["base_file"]
  361. if self._container_registry.isReadOnly(root_material_id):
  362. Logger.log("w", "Cannot set name of read-only container %s.", root_material_id)
  363. return
  364. material_group = self.getMaterialGroup(root_material_id)
  365. if material_group:
  366. material_group.root_material_node.getContainer().setName(name)
  367. #
  368. # Removes the given material.
  369. #
  370. @pyqtSlot("QVariant")
  371. def removeMaterial(self, material_node: "MaterialNode"):
  372. root_material_id = material_node.metadata["base_file"]
  373. self.removeMaterialByRootId(root_material_id)
  374. #
  375. # Creates a duplicate of a material, which has the same GUID and base_file metadata.
  376. # Returns the root material ID of the duplicated material if successful.
  377. #
  378. @pyqtSlot("QVariant", result = str)
  379. def duplicateMaterial(self, material_node, new_base_id = None, new_metadata = None) -> Optional[str]:
  380. root_material_id = material_node.metadata["base_file"]
  381. material_group = self.getMaterialGroup(root_material_id)
  382. if not material_group:
  383. Logger.log("i", "Unable to duplicate the material with id %s, because it doesn't exist.", root_material_id)
  384. return None
  385. base_container = material_group.root_material_node.getContainer()
  386. if not base_container:
  387. return None
  388. # Ensure all settings are saved.
  389. self._application.saveSettings()
  390. # Create a new ID & container to hold the data.
  391. new_containers = []
  392. if new_base_id is None:
  393. new_base_id = self._container_registry.uniqueName(base_container.getId())
  394. new_base_container = copy.deepcopy(base_container)
  395. new_base_container.getMetaData()["id"] = new_base_id
  396. new_base_container.getMetaData()["base_file"] = new_base_id
  397. if new_metadata is not None:
  398. for key, value in new_metadata.items():
  399. new_base_container.getMetaData()[key] = value
  400. new_containers.append(new_base_container)
  401. # Clone all of them.
  402. for node in material_group.derived_material_node_list:
  403. container_to_copy = node.getContainer()
  404. if not container_to_copy:
  405. continue
  406. # Create unique IDs for every clone.
  407. new_id = new_base_id
  408. if container_to_copy.getMetaDataEntry("definition") != "fdmprinter":
  409. new_id += "_" + container_to_copy.getMetaDataEntry("definition")
  410. if container_to_copy.getMetaDataEntry("variant_name"):
  411. variant_name = container_to_copy.getMetaDataEntry("variant_name")
  412. new_id += "_" + variant_name.replace(" ", "_")
  413. new_container = copy.deepcopy(container_to_copy)
  414. new_container.getMetaData()["id"] = new_id
  415. new_container.getMetaData()["base_file"] = new_base_id
  416. if new_metadata is not None:
  417. for key, value in new_metadata.items():
  418. new_container.getMetaData()[key] = value
  419. new_containers.append(new_container)
  420. for container_to_add in new_containers:
  421. container_to_add.setDirty(True)
  422. self._container_registry.addContainer(container_to_add)
  423. return new_base_id
  424. #
  425. # Create a new material by cloning Generic PLA for the current material diameter and generate a new GUID.
  426. #
  427. @pyqtSlot(result = str)
  428. def createMaterial(self) -> str:
  429. from UM.i18n import i18nCatalog
  430. catalog = i18nCatalog("cura")
  431. # Ensure all settings are saved.
  432. self._application.saveSettings()
  433. machine_manager = self._application.getMachineManager()
  434. extruder_stack = machine_manager.activeStack
  435. approximate_diameter = str(extruder_stack.approximateMaterialDiameter)
  436. root_material_id = "generic_pla"
  437. root_material_id = self.getRootMaterialIDForDiameter(root_material_id, approximate_diameter)
  438. material_group = self.getMaterialGroup(root_material_id)
  439. # Create a new ID & container to hold the data.
  440. new_id = self._container_registry.uniqueName("custom_material")
  441. new_metadata = {"name": catalog.i18nc("@label", "Custom Material"),
  442. "brand": catalog.i18nc("@label", "Custom"),
  443. "GUID": str(uuid.uuid4()),
  444. }
  445. self.duplicateMaterial(material_group.root_material_node,
  446. new_base_id = new_id,
  447. new_metadata = new_metadata)
  448. return new_id