2016-06-04
Added the Lasercut settings to the pronterface options dialog
(default settings when used without pronterwindow)
--- a/printrun-src/printrun/gui/widgets.py Sat Jun 04 10:01:58 2016 +0200 +++ b/printrun-src/printrun/gui/widgets.py Sat Jun 04 12:41:32 2016 +0200 @@ -120,7 +120,8 @@ "UI": _("User interface"), "Viewer": _("Viewer"), "Colors": _("Colors"), - "External": _("External commands")} + "External": _("External commands"), + "Laser": "Lasercut options"} class PronterOptionsDialog(wx.Dialog): """Options editor"""
--- a/printrun-src/printrun/laser.py Sat Jun 04 10:01:58 2016 +0200 +++ b/printrun-src/printrun/laser.py Sat Jun 04 12:41:32 2016 +0200 @@ -6,32 +6,14 @@ """ LASERCUT SETTINGS -TODO: move to printrun settings +Will be overridden from pronterface settings """ -ENGRAVE_SPEED = 10 * 60 # mm/min -# 30mm/min works for wood (regulate the output power to something between 10-30%) -# 30mm/min for black anodized aluminum to get a light engraving @ 100% power -# 10mm/min for black anodized aluminum to get more "silver" @ 100% power - -TRAVEL_SPEED = 120 * 60 E_FACTOR = 0.5 -# BITMAP: -DPI = 300 -GREY_THRESHOLD = 0 -CHANGE_DIRECTION = True -INVERT_CUT = True +from PIL import Image +import sys +import math -""" -STATIC DEFINITIONS -DO NOT CHANGE WORLD's RULES! -""" -INCH = 25.4 # mm -MM_PIXEL = round(INCH / DPI, 4) -STEPS_PIXEL = MM_PIXEL * 80 # mine is 80 steps/mm on XY - -# FOR HPGL: -SCALE_FACTOR = 1.0 / 40.0 # 40 plotter units # GENERAL HEADER AND FOOTER GCODE GCODE_HEAD = """ @@ -48,27 +30,39 @@ M209 S0 ; disable firmware retraction, we dont want to burn holes... M302 ; Allow cold extrudes - doesnt matter because we hack the extruder physically off with the M571 E mod M571 S1 E1 ; Activate Laser output on extrusion, but block real motor movement! -G0 X0 Y0 F%d ; Set moving speed TRAVEL_SPEED -G1 X0 Y0 F%d ; Set linear engraving speed ENGRAVE_SPEED - -""" % (TRAVEL_SPEED, ENGRAVE_SPEED) +""" GCODE_FOOT = """G0 X0 Y0 F%.4f M400 ; Wait for all moves to finish M571 S0 E0 M42 P28 S0 ; Force laser off! M501 ; undo all settings made -""" % (TRAVEL_SPEED) +""" % (100*60) -from PIL import Image -import sys -# Imports for SVG -import xml.etree.ElementTree as ET -import math -from svg2gcode import shapes as shapes_pkg -from svg2gcode.shapes import point_generator +class LasercutterSettings: + """ + Default settings object + """ + def __init__(self): + self.lc_engrave_speed = 10 + # 30mm/min works for wood (regulate the output power to something between 10-30%) + # 30mm/min for black anodized aluminum to get a light engraving @ 100% power + # 10mm/min for black anodized aluminum to get more "silver" @ 100% power + self.lc_travel_speed = 120 + # BITMAP: + self.lc_bitmap_speed_factor = 1.0 + self.lc_dpi = 300 + self.lc_grey_threshold = 0 + self.lc_change_dir = True + self.lc_invert_cut = True + + # HPGL: + self.lc_hpgl_speed_factor = 1.0 + + # SVG: + self.lc_svg_speed_factor = 1.0 class Lasercutter: """ @@ -78,24 +72,32 @@ def __init__(self, pronterwindow = None): if pronterwindow: self.pronterwindow = pronterwindow + self.settings = pronterwindow.settings #self.log = pronterwindow.log self.log = self.log_print self.pronterwindow.clear_log(None) else: self.pronterwindow = None + self.settings = LasercutterSettings() self.log = lambda : None - self.log("Lasercutter library initialized\n%d DPI (%f mm/pixel)" % (DPI, MM_PIXEL)) - if STEPS_PIXEL <= 5: - self.log("WARNING: STEPS PER PIXEL NEEDS TO BE > 5 (otherwise marlin joins lines): %f" % STEPS_PIXEL) + + # STATIC DEFINITIONS, DO NOT CHANGE WORLD's RULES! + self.INCH = 25.4 # mm + self.MM_PIXEL = round(self.INCH / self.settings.lc_dpi, 4) + self.STEPS_PIXEL = self.MM_PIXEL * 80 # mine is 80 steps/mm on XY + + self.log("Lasercutter library initialized\n%d DPI (%f mm/pixel)" % (self.settings.lc_dpi, self.MM_PIXEL)) + if self.STEPS_PIXEL <= 5: + self.log("WARNING: STEPS PER PIXEL NEEDS TO BE > 5 (otherwise marlin joins lines): %f" % self.STEPS_PIXEL) self.log("Travel/Engrave speed: %d mm/sec, %d mm/sec" % ( - TRAVEL_SPEED / 60, ENGRAVE_SPEED / 60) ) + self.settings.lc_travel_speed, self.settings.lc_engrave_speed) ) self.log("") def log_print(self, msg): print(msg) - def pixel2bit(self, pixel, threshold=128): + def pixel2bit(self, pixel): """Convert the pixel value to a bit.""" # some really weird stuff here ;-P @@ -105,14 +107,14 @@ if isinstance(pixel, tuple): #rgb pixel = pixel[0]*0.2989 + pixel[1]*0.5870 + pixel[2]*0.1140 - threshold = 128 - if pixel > threshold: + if pixel > self.settings.lc_grey_threshold: return 1 else: return 0 # color palette - if pixel <= threshold: + # TODO: get the grey value of the palette index instead of using pixel which is the palette index? + if pixel <= self.settings.lc_grey_threshold: return 1 else: return 0 @@ -140,12 +142,15 @@ fo.write("; Filename: %s\n%s" % (filename, GCODE_HEAD)) fo.write(";Start engraving the raster image: %dx%d points @ %d DPI = %.0fx%.0f mm\n\n" % ( - im.size[0], im.size[1], DPI, im.size[0]*MM_PIXEL, im.size[1]*MM_PIXEL) ) + im.size[0], im.size[1], self.settings.lc_dpi, im.size[0] * self.MM_PIXEL, im.size[1] * self.MM_PIXEL) ) - INVERT_Y = MM_PIXEL * (im.size[1] -1) * (-1) + INVERT_Y = self.MM_PIXEL * (im.size[1] -1) * (-1) DIR = 1 + travel_speed = self.settings.lc_travel_speed * 60 + engrave_speed = self.settings.lc_engrave_speed * 60 * self.settings.lc_bitmap_speed_factor for X in range(im.size[0]): - fo.write("; X=%d printing row: direction %i\n" % (X, DIR)) + # TODO: Skip empty rows!!! + fo.write("M400 ; X=%d printing row: direction %i\n" % (X, DIR)) fo.write("G92 E0\n") E = 0 last_bit = 1 # we engrave on black pixel = 0 @@ -158,11 +163,11 @@ range_stop = -1 for Y in range(range_start, range_stop, DIR): - YMM = abs((Y * MM_PIXEL) + INVERT_Y) - XMM = X * MM_PIXEL + YMM = abs((Y * self.MM_PIXEL) + INVERT_Y) + XMM = X * self.MM_PIXEL #print "X %d Y %d" % (X, Y) - bit = self.pixel2bit(pix[X, Y], GREY_THRESHOLD) - if INVERT_CUT: + bit = self.pixel2bit(pix[X, Y]) + if self.settings.lc_invert_cut: if bit == 0: bit = 1 else: @@ -177,25 +182,25 @@ if (Y == (im.size[1] - 1)) or (Y == 0): # draw line if DIR > 0: - E = E + MM_PIXEL * (Y - START_Y) + E = E + self.MM_PIXEL * (Y - START_Y) else: - E = E + MM_PIXEL * (START_Y - Y) - fo.write("G1 X%.4f Y%.4f E%.4f\n" % (XMM, YMM, E * E_FACTOR)) + E = E + self.MM_PIXEL * (START_Y - Y) + fo.write("G1 X%.4f Y%.4f E%.4f F%.4f\n" % (XMM, YMM, E * E_FACTOR, engrave_speed)) else: # bit value has changed! if bit == 0: # jump to start of line to write START_Y = Y - fo.write("G0 X%.4f Y%.4f\n" % (XMM, YMM)) + fo.write("G0 X%.4f Y%.4f F%.4f\n" % (XMM, YMM, travel_speed)) else: # end of line to write if DIR > 0: - E = E + (MM_PIXEL * (Y - START_Y)) + E = E + (self.MM_PIXEL * (Y - START_Y)) else: - E = E + (MM_PIXEL * (START_Y - Y)) - fo.write("G1 X%.4f Y%.4f E%.4f\n" % (XMM, YMM, E * E_FACTOR)) + E = E + (self.MM_PIXEL * (START_Y - Y)) + fo.write("G1 X%.4f Y%.4f E%.4f F%.4f\n" % (XMM, YMM, E * E_FACTOR, engrave_speed)) last_bit = bit - if CHANGE_DIRECTION: + if self.settings.lc_change_dir: DIR = DIR * (-1) # change y direction on every X fo.write(GCODE_FOOT) @@ -206,6 +211,8 @@ self.pronterwindow.load_gcode_async(filename + '.g') def hpgl2gcode(self, filename): + # FOR HPGL: + SCALE_FACTOR = 1.0 / 40.0 # 40 plotter units OFFSET_X = 0.0 OFFSET_Y = 0.0 @@ -221,6 +228,10 @@ last_coord = [0.0,0.0] last_cmd = "" + travel_speed = self.settings.lc_travel_speed * 60 + engrave_speed = self.settings.lc_engrave_speed * 60 * self.settings.lc_hpgl_speed_factor + + for line in fi.readlines(): for action in line.split(";"): action = action.strip() @@ -242,9 +253,9 @@ if LASER_STATE: EN = " E%.4f F%.4f" % ( E_FACTOR * math.hypot(coord[0] - last_coord[0], coord[1] - last_coord[1]), - ENGRAVE_SPEED * 0.5 ) # 1/2 engraving speed + engrave_speed) else: - EN = " F%.4f" % TRAVEL_SPEED + EN = " F%.4f" % travel_speed fo.write("G%d X%.4f Y%.4f%s\n" % ( LASER_STATE, coord[0], coord[1], EN) ) @@ -265,22 +276,15 @@ self.log("") self.pronterwindow.load_gcode_async(filename + '.g') - def svg2gcode(self, filename, bed_max_x = 50, bed_max_y = 50, smoothness = 0.2): - self.log("Generating paths from SVG...") - - shape_preamble = "G92 E0\n" - shape_postamble = "" + # Imports for SVG + import xml.etree.ElementTree as ET + from svg2gcode import shapes as shapes_pkg + from svg2gcode.shapes import point_generator - """ - Used to control the smoothness/sharpness of the curves. - Smaller the value greater the sharpness. Make sure the - value is greater than 0.1 - """ + self.log("Generating paths from SVG, alternative lib (outlines only)...") if smoothness < 0.1: smoothness = 0.1 - svg_shapes = set(['rect', 'circle', 'ellipse', 'line', 'polyline', 'polygon', 'path']) - tree = ET.parse(filename) root = tree.getroot() @@ -295,8 +299,8 @@ self.log("Unable to get width and height for the svg!") return False - width = float(width.replace("px", "")) - height = float(height.replace("px", "")) + width = float(width.replace("px", "").replace("pt", "")) + height = float(height.replace("px", "").replace("pt", "")) scale_x = bed_max_x / max(width, height) scale_y = bed_max_y / max(width, height) @@ -306,6 +310,9 @@ fo = open(filename + ".g", "w") fo.write("; Filename: %s\n%s" % (filename, GCODE_HEAD)) + travel_speed = self.settings.lc_travel_speed * 60 + engrave_speed = self.settings.lc_engrave_speed * 60 * self.settings.lc_svg_speed_factor + for elem in root.iter(): try: _, tag_suffix = elem.tag.split('}') @@ -319,11 +326,11 @@ m = shape_obj.transformation_matrix() if d: - fo.write("M400 ; wait for moves finish, then printing shape: %s\n" % (tag_suffix)) + fo.write("M400 ; start %s\n" % (tag_suffix)) + fo.write("G92 E0\n") E = 0 xo = 0 yo = 0 - fo.write(shape_preamble) p = point_generator(d, m, smoothness) start = True for x,y,pen in p: @@ -335,7 +342,7 @@ if not pen: start = True if xs >= 0 and xs <= bed_max_x and ys >= 0 and ys <= bed_max_y: if start: - fo.write("G0 X%0.2f Y%0.2f F%.4f ; Move to start of shape\n" % (xs, ys, TRAVEL_SPEED)) + fo.write("G0 X%0.2f Y%0.2f F%.4f ; Move to start of shape\n" % (xs, ys, travel_speed)) start = False xo = xs yo = ys @@ -346,17 +353,15 @@ xo = xs yo = ys E = E + (e_distance) - fo.write("G1 X%0.2f Y%0.2f E%.4f F%.4f\n" % (xs, ys, E * E_FACTOR, ENGRAVE_SPEED)) + fo.write("G1 X%0.2f Y%0.2f E%.4f F%.4f\n" % (xs, ys, E * E_FACTOR, engrave_speed)) else: self.log("Position outside print dimension: %d, %d" % (xs, ys)) if shape_obj.xml_node.get('fill'): # Close the polygon e_distance = math.hypot(object_xs - xo, object_ys - yo) E = E + (e_distance) - fo.write("G1 X%0.2f Y%0.2f E%.4f F%.4f ; Close the object polygon\n" % (object_xs, object_ys, E * E_FACTOR, ENGRAVE_SPEED)) - print "connecting polycommon path end to start" - - fo.write(shape_postamble) + fo.write("G1 X%0.2f Y%0.2f E%.4f F%.4f ; Close the object polygon\n" % (object_xs, object_ys, E * E_FACTOR, engrave_speed)) + print "connecting filled path end to start" fo.write(GCODE_FOOT) fo.close()
--- a/printrun-src/printrun/pronterface.py Sat Jun 04 10:01:58 2016 +0200 +++ b/printrun-src/printrun/pronterface.py Sat Jun 04 12:41:32 2016 +0200 @@ -838,6 +838,7 @@ info.SetLicence(licence) info.AddDeveloper('Kliment Yanev') info.AddDeveloper('Guillaume Seguin') + info.AddDeveloper('Malte Bayer') wx.AboutBox(info) @@ -845,7 +846,30 @@ # Settings & command line handling (including update callbacks) # -------------------------------------------------------------- + def update_lc_settings(self, key, value): + return True + def _add_settings(self, size): + + # first add the lasercutter options + self.settings._add(StaticTextSetting("separator_lc_general", "General laser settings", "", group = "Laser")) + self.settings._add(SpinSetting("lc_travel_speed", 120, 10, 300, "Travel speed in mm/s", "", "Laser"), self.update_lc_settings) + self.settings._add(SpinSetting("lc_engrave_speed", 10, 10, 300, "Engrave speed in mm/s", "", "Laser"), self.update_lc_settings) + + self.settings._add(StaticTextSetting("separator_lc_bitmap", "PNG Bitmap processing", "", group = "Laser")) + self.settings._add(FloatSpinSetting("lc_bitmap_speed_factor", 1.0, 0.1, 2.0, "Engrave speed factor", "", "Laser"), self.update_lc_settings) + self.settings._add(SpinSetting("lc_dpi", 300, 25, 300, "Image DPI", "Image resolution for scaling", "Laser"), self.update_lc_settings) + self.settings._add(SpinSetting("lc_grey_threshold", 0, 0, 255, "Grey threshold value for RGB", "", "Laser"), self.update_lc_settings) + self.settings._add(BooleanSetting("lc_invert_cut", True, "PNG: Invert grey threshold", "Invert laser on/off logic", "Laser"), self.update_lc_settings) + self.settings._add(BooleanSetting("lc_change_dir", True, "PNG: Change direction", "Engrave in both directions on Y Axis", "Laser"), self.update_lc_settings) + + self.settings._add(StaticTextSetting("separator_lc_hpgl", "HPGL processing", "", group = "Laser")) + self.settings._add(FloatSpinSetting("lc_hpgl_speed_factor", 1.0, 0.1, 2.0, "Engrave speed factor", "", "Laser"), self.update_lc_settings) + + self.settings._add(StaticTextSetting("separator_lc_svg", "SVG processing", "", group = "Laser")) + self.settings._add(FloatSpinSetting("lc_svg_speed_factor", 1.0, 0.1, 2.0, "Engrave speed factor", "", "Laser"), self.update_lc_settings) + + self.settings._add(BooleanSetting("monitor", True, _("Monitor printer status"), _("Regularly monitor printer temperatures (required to have functional temperature graph or gauges)"), "Printer"), self.update_monitor) self.settings._add(StringSetting("simarrange_path", "", _("Simarrange command"), _("Path to the simarrange binary to use in the STL plater"), "External")) self.settings._add(BooleanSetting("circular_bed", False, _("Circular build platform"), _("Draw a circular (or oval) build platform instead of a rectangular one"), "Printer"), self.update_bed_viz)