ThreeMFReader.py 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125
  1. # Copyright (c) 2015 Ultimaker B.V.
  2. # Cura is released under the terms of the AGPLv3 or higher.
  3. from UM.Mesh.MeshReader import MeshReader
  4. from UM.Mesh.MeshData import MeshData
  5. from UM.Logger import Logger
  6. from UM.Math.Matrix import Matrix
  7. from UM.Math.Vector import Vector
  8. from UM.Scene.SceneNode import SceneNode
  9. from UM.Scene.GroupDecorator import GroupDecorator
  10. from UM.Math.Quaternion import Quaternion
  11. import os
  12. import struct
  13. import math
  14. from os import listdir
  15. import zipfile
  16. import xml.etree.ElementTree as ET
  17. ## Base implementation for reading 3MF files. Has no support for textures. Only loads meshes!
  18. class ThreeMFReader(MeshReader):
  19. def __init__(self):
  20. super(ThreeMFReader, self).__init__()
  21. self._supported_extension = ".3mf"
  22. self._namespaces = {
  23. "3mf": "http://schemas.microsoft.com/3dmanufacturing/core/2015/02",
  24. "cura": "http://software.ultimaker.com/xml/cura/3mf/2015/10"
  25. }
  26. def read(self, file_name):
  27. result = None
  28. extension = os.path.splitext(file_name)[1]
  29. if extension.lower() == self._supported_extension:
  30. result = SceneNode()
  31. # The base object of 3mf is a zipped archive.
  32. archive = zipfile.ZipFile(file_name, 'r')
  33. try:
  34. root = ET.parse(archive.open("3D/3dmodel.model"))
  35. # There can be multiple objects, try to load all of them.
  36. objects = root.findall("./3mf:resources/3mf:object", self._namespaces)
  37. for object in objects:
  38. mesh = MeshData()
  39. node = SceneNode()
  40. vertex_list = []
  41. #for vertex in object.mesh.vertices.vertex:
  42. for vertex in object.findall(".//3mf:vertex", self._namespaces):
  43. vertex_list.append([vertex.get("x"), vertex.get("y"), vertex.get("z")])
  44. triangles = object.findall(".//3mf:triangle", self._namespaces)
  45. mesh.reserveFaceCount(len(triangles))
  46. #for triangle in object.mesh.triangles.triangle:
  47. for triangle in triangles:
  48. v1 = int(triangle.get("v1"))
  49. v2 = int(triangle.get("v2"))
  50. v3 = int(triangle.get("v3"))
  51. mesh.addFace(vertex_list[v1][0],vertex_list[v1][2],vertex_list[v1][1],vertex_list[v2][0],vertex_list[v2][2],vertex_list[v2][1],vertex_list[v3][0],vertex_list[v3][2],vertex_list[v3][1])
  52. #TODO: We currently do not check for normals and simply recalculate them.
  53. mesh.calculateNormals()
  54. node.setMeshData(mesh)
  55. node.setSelectable(True)
  56. Logger.log("d", "Loaded a mesh with %s vertices", mesh.getVertexCount())
  57. transformation = root.findall("./3mf:build/3mf:item[@objectid='{0}']".format(object.get("id")), self._namespaces)
  58. if transformation:
  59. transformation = transformation[0]
  60. if transformation.get("transform"):
  61. splitted_transformation = transformation.get("transform").split()
  62. ## Transformation is saved as:
  63. ## M00 M01 M02 0.0
  64. ## M10 M11 M12 0.0
  65. ## M20 M21 M22 0.0
  66. ## M30 M31 M32 1.0
  67. ## We switch the row & cols as that is how everyone else uses matrices!
  68. temp_mat = Matrix()
  69. # Rotation & Scale
  70. temp_mat._data[0,0] = splitted_transformation[0]
  71. temp_mat._data[1,0] = splitted_transformation[1]
  72. temp_mat._data[2,0] = splitted_transformation[2]
  73. temp_mat._data[0,1] = splitted_transformation[3]
  74. temp_mat._data[1,1] = splitted_transformation[4]
  75. temp_mat._data[2,1] = splitted_transformation[5]
  76. temp_mat._data[0,2] = splitted_transformation[6]
  77. temp_mat._data[1,2] = splitted_transformation[7]
  78. temp_mat._data[2,2] = splitted_transformation[8]
  79. # Translation
  80. temp_mat._data[0,3] = splitted_transformation[9]
  81. temp_mat._data[1,3] = splitted_transformation[10]
  82. temp_mat._data[2,3] = splitted_transformation[11]
  83. node.setPosition(Vector(temp_mat.at(0,3), temp_mat.at(1,3), temp_mat.at(2,3)))
  84. temp_quaternion = Quaternion()
  85. temp_quaternion.setByMatrix(temp_mat)
  86. node.setOrientation(temp_quaternion)
  87. # Magical scale extraction
  88. S2 = temp_mat.getTransposed().multiply(temp_mat)
  89. scale_x = math.sqrt(S2.at(0,0))
  90. scale_y = math.sqrt(S2.at(1,1))
  91. scale_z = math.sqrt(S2.at(2,2))
  92. node.setScale(Vector(scale_x,scale_y,scale_z))
  93. # We use a different coordinate frame, so rotate.
  94. #rotation = Quaternion.fromAngleAxis(-0.5 * math.pi, Vector(1,0,0))
  95. #node.rotate(rotation)
  96. result.addChild(node)
  97. #If there is more then one object, group them.
  98. try:
  99. if len(objects) > 1:
  100. group_decorator = GroupDecorator()
  101. result.addDecorator(group_decorator)
  102. except:
  103. pass
  104. except Exception as e:
  105. Logger.log("e" ,"exception occured in 3mf reader: %s" , e)
  106. return result