10 from Cura.util import profile
12 def getEngineFilename():
13 if platform.system() == 'Windows':
14 if os.path.exists('C:/Software/Cura_SteamEngine/_bin/Release/Cura_SteamEngine.exe'):
15 return 'C:/Software/Cura_SteamEngine/_bin/Release/Cura_SteamEngine.exe'
16 return os.path.abspath(os.path.join(os.path.dirname(__file__), '../..', 'SteamEngine.exe'))
17 return os.path.abspath(os.path.join(os.path.dirname(__file__), '../..', 'SteamEngine'))
19 def getTempFilename():
20 warnings.simplefilter('ignore')
21 ret = os.tempnam(None, "Cura_Tmp")
22 warnings.simplefilter('default')
26 def __init__(self, progressCallback):
29 self._callback = progressCallback
30 self._binaryStorageFilename = getTempFilename()
31 self._exportFilename = getTempFilename()
32 self._progressSteps = ['inset', 'skin', 'export']
34 def abortSlicer(self):
35 if self._process is not None:
36 self._process.terminate()
39 def getGCodeFilename(self):
40 return self._exportFilename
42 def runSlicer(self, scene):
44 self._callback(0.0, False)
46 commandList = [getEngineFilename(), '-vv']
47 for k, v in self._engineSettings().iteritems():
48 commandList += ['-s', '%s=%s' % (k, str(v))]
49 commandList += ['-o', self._exportFilename]
50 commandList += ['-b', self._binaryStorageFilename]
52 with open(self._binaryStorageFilename, "wb") as f:
53 for obj in scene._objectList:
54 if not scene.checkPlatform(obj):
56 for mesh in obj._meshList:
57 n = numpy.array(mesh.vertexCount, numpy.int32)
59 f.write(mesh.vertexes.tostring())
60 pos = obj.getPosition() * 1000
61 pos += numpy.array([profile.getPreferenceFloat('machine_width') * 1000 / 2, profile.getPreferenceFloat('machine_depth') * 1000 / 2])
62 commandList += ['-m', ','.join(map(str, obj._matrix.getA().flatten()))]
63 commandList += ['-s', 'posx=%d' % int(pos[0]), '-s', 'posy=%d' % int(pos[1]), '#']
65 if self._objCount > 0:
66 print ' '.join(commandList)
68 self._process = self._runSliceProcess(commandList)
69 self._thread = threading.Thread(target=self._watchProcess)
70 self._thread.daemon = True
75 def _watchProcess(self):
76 self._callback(0.0, False)
77 line = self._process.stdout.readline()
80 if line.startswith('Progress:'):
81 line = line.split(':')
82 progressValue = float(line[2]) / float(line[3])
83 progressValue /= len(self._progressSteps)
84 progressValue += 1.0 / len(self._progressSteps) * self._progressSteps.index(line[1])
85 self._callback(progressValue, False)
87 print '#', line.strip()
88 line = self._process.stdout.readline()
89 for line in self._process.stderr:
91 returnCode = self._process.wait()
94 self._callback(1.0, True)
96 self._callback(0.0, False)
99 def _engineSettings(self):
101 'layerThickness': int(profile.getProfileSettingFloat('layer_height') * 1000),
102 'initialLayerThickness': int(profile.getProfileSettingFloat('bottom_thickness') * 1000),
103 'filamentDiameter': int(profile.getProfileSettingFloat('filament_diameter') * 1000),
104 'extrusionWidth': int(profile.calculateEdgeWidth() * 1000),
105 'insetCount': int(profile.calculateLineCount()),
106 'downSkinCount': int(profile.calculateSolidLayerCount()),
107 'upSkinCount': int(profile.calculateSolidLayerCount()),
108 'sparseInfillLineDistance': int(100 * profile.calculateEdgeWidth() * 1000 / profile.getProfileSettingFloat('fill_density')),
109 'skirtDistance': int(profile.getProfileSettingFloat('skirt_gap') * 1000),
110 'skirtLineCount': int(profile.getProfileSettingFloat('skirt_line_count')),
111 'initialSpeedupLayers': int(4),
112 'initialLayerSpeed': int(profile.getProfileSettingFloat('bottom_layer_speed')),
113 'printSpeed': int(profile.getProfileSettingFloat('print_speed')),
114 'moveSpeed': int(profile.getProfileSettingFloat('travel_speed')),
115 'fanOnLayerNr': int(profile.getProfileSettingFloat('fan_layer')),
116 'supportAngle': int(-1) if profile.getProfileSetting('support') == 'None' else int(60),
117 'supportEverywhere': int(1) if profile.getProfileSetting('support') == 'Everywhere' else int(0),
118 'retractionAmount': int(profile.getProfileSettingFloat('retraction_amount') * 1000),
119 'retractionSpeed': int(profile.getProfileSettingFloat('retraction_speed')),
120 'objectSink': int(profile.getProfileSettingFloat('object_sink') * 1000),
123 def _runSliceProcess(self, cmdList):
125 if subprocess.mswindows:
126 su = subprocess.STARTUPINFO()
127 su.dwFlags |= subprocess.STARTF_USESHOWWINDOW
128 su.wShowWindow = subprocess.SW_HIDE
129 kwargs['startupinfo'] = su
130 kwargs['creationflags'] = 0x00004000 #BELOW_NORMAL_PRIORITY_CLASS
131 return subprocess.Popen(cmdList, stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE, **kwargs)