GCodeReader.py 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. # Copyright (c) 2016 Aleph Objects, Inc.
  2. # Cura is released under the terms of the AGPLv3 or higher.
  3. from UM.Mesh.MeshReader import MeshReader
  4. import os
  5. from UM.Scene.SceneNode import SceneNode
  6. from UM.Scene.Iterator.DepthFirstIterator import DepthFirstIterator
  7. from UM.Math.Vector import Vector
  8. from UM.Math.AxisAlignedBox import AxisAlignedBox
  9. from UM.Application import Application
  10. from UM.Message import Message
  11. from UM.Logger import Logger
  12. from UM.Backend.Backend import BackendState
  13. from UM.i18n import i18nCatalog
  14. catalog = i18nCatalog("cura")
  15. from cura import LayerDataBuilder
  16. from cura import LayerDataDecorator
  17. from cura.LayerPolygon import LayerPolygon
  18. import numpy
  19. import math
  20. import re
  21. class GCodeReader(MeshReader):
  22. def __init__(self):
  23. super(GCodeReader, self).__init__()
  24. self._supported_extensions = [".gcode", ".g"]
  25. Application.getInstance().hideMessageSignal.connect(self._onHideMessage)
  26. self._cancelled = False
  27. self._message = None
  28. self._scene_node = None
  29. @staticmethod
  30. def _getValue(line, code):
  31. n = line.find(code) + len(code)
  32. if n < 1:
  33. return None
  34. pattern = re.compile("[;\s]")
  35. match = pattern.search(line, n)
  36. m = match.start() if math is not None else -1
  37. try:
  38. if m < 0:
  39. return line[n:]
  40. return line[n:m]
  41. except:
  42. return None
  43. def _getInt(self, line, code):
  44. value = self._getValue(line, code)
  45. try:
  46. return int(value)
  47. except:
  48. return None
  49. def _getFloat(self, line, code):
  50. value = self._getValue(line, code)
  51. try:
  52. return float(value)
  53. except:
  54. return None
  55. def _onHideMessage(self, message):
  56. if message == self._message:
  57. self._cancelled = True
  58. def _onParentChanged(self, node):
  59. if self._scene_node is not None and self._scene_node.getParent() is None:
  60. self._scene_node = None
  61. Application.getInstance().getBackend().continueSlicing()
  62. Application.getInstance().setHideSettings(False)
  63. Application.getInstance().getPrintInformation().setPreSliced(False)
  64. @staticmethod
  65. def _getNullBoundingBox():
  66. return AxisAlignedBox(minimum=Vector(0, 0, 0), maximum=Vector(10, 10, 10))
  67. @staticmethod
  68. def _createPolygon(layer_data, path, layer_id, extruder, thickness):
  69. countvalid = 0
  70. for point in path:
  71. if point[3] > 0:
  72. countvalid += 1
  73. if countvalid < 2:
  74. return False
  75. try:
  76. layer_data.addLayer(layer_id)
  77. layer_data.setLayerHeight(layer_id, path[0][1])
  78. layer_data.setLayerThickness(layer_id, thickness)
  79. this_layer = layer_data.getLayer(layer_id)
  80. except ValueError:
  81. return False
  82. count = len(path)
  83. line_types = numpy.empty((count - 1, 1), numpy.int32)
  84. line_widths = numpy.empty((count - 1, 1), numpy.float32)
  85. line_widths[:, 0] = 0.5
  86. points = numpy.empty((count, 3), numpy.float32)
  87. i = 0
  88. for point in path:
  89. points[i, 0] = point[0]
  90. points[i, 1] = point[2]
  91. points[i, 2] = -point[1]
  92. if i > 0:
  93. if point[3] == LayerPolygon.Inset0Type:
  94. line_types[i - 1] = extruder + 1
  95. else:
  96. line_types[i - 1] = point[3]
  97. i += 1
  98. this_poly = LayerPolygon(layer_data, extruder, line_types, points, line_widths)
  99. this_poly.buildCache()
  100. this_layer.polygons.append(this_poly)
  101. return True
  102. def read(self, file_name):
  103. Logger.log("d", "Preparing to load %s" % file_name)
  104. self._cancelled = False
  105. self._scene_node = SceneNode()
  106. self._scene_node.getBoundingBox = self._getNullBoundingBox # Manually set bounding box, because mesh doesn't have mesh data
  107. self._scene_node.gcode = True
  108. self._scene_node.parentChanged.connect(self._onParentChanged)
  109. Application.getInstance().getBackend().pauseSlicing()
  110. glist = []
  111. Application.getInstance().getController().getScene().gcode_list = glist
  112. Logger.log("d", "Opening file %s" % file_name)
  113. layer_data_builder = LayerDataBuilder.LayerDataBuilder()
  114. with open(file_name, "r") as file:
  115. file_lines = 0
  116. current_line = 0
  117. for line in file:
  118. file_lines += 1
  119. file.seek(0)
  120. file_step = max(math.floor(file_lines / 100), 1)
  121. current_extruder = 0
  122. current_path = []
  123. current_x = 0
  124. current_y = 0
  125. current_z = 0
  126. current_e = 0
  127. current_layer = 0
  128. prev_z = 0
  129. self._message = Message(catalog.i18nc("@info:status", "Parsing GCODE"), lifetime=0)
  130. self._message.setProgress(0)
  131. self._message.show()
  132. Logger.log("d", "Parsing %s" % file_name)
  133. for line in file:
  134. if self._cancelled:
  135. Logger.log("w", "Parsing %s cancelled" % file_name)
  136. return None
  137. current_line += 1
  138. if current_line % file_step == 0:
  139. self._message.setProgress(math.floor(current_line / file_lines * 100))
  140. if len(line) == 0:
  141. continue
  142. if line[0] == ";":
  143. continue
  144. G = self._getInt(line, "G")
  145. if G is not None:
  146. if G == 0 or G == 1:
  147. x = self._getFloat(line, "X")
  148. y = self._getFloat(line, "Y")
  149. z = self._getFloat(line, "Z")
  150. e = self._getFloat(line, "E")
  151. z_changed = False
  152. if x is not None:
  153. current_x = x
  154. if y is not None:
  155. current_y = y
  156. if z is not None:
  157. if not current_z == z:
  158. z_changed = True
  159. prev_z = current_z
  160. current_z = z
  161. if e is not None:
  162. if e > current_e:
  163. current_path.append([current_x, current_y, current_z, LayerPolygon.Inset0Type]) # extrusion
  164. else:
  165. current_path.append([current_x, current_y, current_z, LayerPolygon.MoveRetractionType]) # retraction
  166. current_e = e
  167. else:
  168. current_path.append([current_x, current_y, current_z, LayerPolygon.MoveCombingType])
  169. if z_changed:
  170. if len(current_path) > 1 and current_z > 0:
  171. if self._createPolygon(layer_data_builder, current_path, current_layer, current_extruder, math.fabs(current_z - prev_z)):
  172. current_layer += 1
  173. current_path.clear()
  174. else:
  175. current_path.clear()
  176. elif G == 28:
  177. x = self._getFloat(line, "X")
  178. y = self._getFloat(line, "Y")
  179. if x is not None:
  180. current_x = x
  181. if y is not None:
  182. current_y = y
  183. current_z = 0
  184. elif G == 92:
  185. x = self._getFloat(line, "X")
  186. y = self._getFloat(line, "Y")
  187. z = self._getFloat(line, "Z")
  188. e = self._getFloat(line, "E")
  189. if x is not None:
  190. current_x = x
  191. if y is not None:
  192. current_y = y
  193. if z is not None:
  194. current_z = z
  195. if e is not None:
  196. current_e = e
  197. T = self._getInt(line, "T")
  198. if T is not None:
  199. current_extruder = T
  200. if len(current_path) > 1 and current_z > 0:
  201. if self._createPolygon(layer_data_builder, current_path, current_layer, current_extruder, math.fabs(current_z - prev_z)):
  202. current_layer += 1
  203. current_path.clear()
  204. else:
  205. current_path.clear()
  206. if len(current_path) > 1 and current_z > 0:
  207. if self._createPolygon(layer_data_builder, current_path, current_layer, current_extruder, math.fabs(current_z - prev_z)):
  208. current_layer += 1
  209. current_path.clear()
  210. layer_mesh = layer_data_builder.build()
  211. decorator = LayerDataDecorator.LayerDataDecorator()
  212. decorator.setLayerData(layer_mesh)
  213. self._scene_node.removeDecorator("LayerDataDecorator")
  214. self._scene_node.addDecorator(decorator)
  215. Logger.log("d", "Finished parsing %s" % file_name)
  216. self._message.hide()
  217. if current_layer == 0:
  218. Logger.log("w", "File %s doesn't contain any valid layers" % file_name)
  219. Application.getInstance().getPrintInformation().setPreSliced(True)
  220. Application.getInstance().setHideSettings(True)
  221. settings = Application.getInstance().getGlobalContainerStack()
  222. machine_width = settings.getProperty("machine_width", "value")
  223. machine_depth = settings.getProperty("machine_depth", "value")
  224. self._scene_node.setPosition(Vector(-machine_width / 2, 0, machine_depth / 2))
  225. Logger.log("d", "Loaded %s" % file_name)
  226. return self._scene_node