From ff28f33967f069a85dc86098ecfc9d14feb71c8b Mon Sep 17 00:00:00 2001 From: Kevin O'Connor Date: Wed, 22 Apr 2020 14:30:52 -0400 Subject: [PATCH] gcode_arcs: Simplify parameter parsing Use the normal gcode.get_float() mechanism for extracting parameters from the g-code command. Don't register descriptions for the G2/G3 commands as the convention is to only use descriptions for "extended g-code commands". Signed-off-by: Kevin O'Connor --- config/example-extras.cfg | 5 +- klippy/extras/gcode_arcs.py | 121 +++++++++++------------------------- 2 files changed, 39 insertions(+), 87 deletions(-) diff --git a/config/example-extras.cfg b/config/example-extras.cfg index bcc90fe3..a81ab014 100644 --- a/config/example-extras.cfg +++ b/config/example-extras.cfg @@ -502,14 +502,13 @@ #unretract_speed: 10 # The speed of unretraction, in mm/s. The default is 10 mm/s. -# enables arc (G2/G3) commands. Only IJ version is supported -# example: "G2 X125 Y32 Z10 E5 I10.5 J10.5" +# Support for gcode arc (G2/G3) commands. #[gcode_arcs] #resolution: 1.0 # An arc will be split into segments. Each segment's length will equal # the resolution in mm set above. Lower values will produce a finer arc, # but also more work for your machine. Arcs smaller than the configured -# value will become straight lines. +# value will become straight lines. The default is 1mm. # Enable the "M118" and "RESPOND" extended commands. # [respond] diff --git a/klippy/extras/gcode_arcs.py b/klippy/extras/gcode_arcs.py index 1f6f0040..219c76d8 100644 --- a/klippy/extras/gcode_arcs.py +++ b/klippy/extras/gcode_arcs.py @@ -6,92 +6,56 @@ # Copyright (C) 2011 Camiel Gubbels / Erik van der Zalm # # This file may be distributed under the terms of the GNU GPLv3 license. +import math - -# uses the plan_arc function from marlin which does steps in mm rather then -# in degrees. # Coordinates created by this are converted into G1 commands. +# Coordinates created by this are converted into G1 commands. # # note: only IJ version available -import math -import re - class ArcSupport: def __init__(self, config): self.printer = config.get_printer() - self.mm_per_arc_segment = config.getfloat('resolution', 1, above=0.0) + self.mm_per_arc_segment = config.getfloat('resolution', 1., above=0.0) self.gcode = self.printer.lookup_object('gcode') - self.gcode.register_command("G2", self.cmd_G2, desc=self.cmd_G2_help) - self.gcode.register_command("G3", self.cmd_G2, desc=self.cmd_G3_help) - - cmd_G2_help = "Counterclockwise rotation move" - cmd_G3_help = "Clockwise rotaion move" + self.gcode.register_command("G2", self.cmd_G2) + self.gcode.register_command("G3", self.cmd_G2) def cmd_G2(self, params): - # set vars - currentPos = self.gcode.get_status(None)['gcode_position'] + currentPos = self.gcode.get_status(None)['gcode_position'] - asX = params.get("X", None) - asY = params.get("Y", None) - asZ = params.get("Z", None) + # Parse parameters + asX = self.gcode.get_float("X", params) + asY = self.gcode.get_float("Y", params) + asZ = self.gcode.get_float("Z", params, None) + if self.gcode.get_float("R", params, None) is not None: + raise self.gcode.error("G2/G3 does not support R moves") + asI = self.gcode.get_float("I", params, 0.) + asJ = self.gcode.get_float("J", params, 0.) + if not asI and not asJ: + raise self.gcode.error("G2/G3 neither I nor J given") + asE = self.gcode.get_float("E", params, None) + asF = self.gcode.get_float("F", params, None) + clockwise = (params['#command'] == 'G2') - asR = float(params.get("R", 0.)) #radius - asI = float(params.get("I", 0.)) - asJ = float(params.get("J", 0.)) - - asE = float(params.get("E", 0.)) - asF = float(params.get("F", -1)) - - # --------- health checks of code ----------- - if (asX is None or asY is None): - raise self.gcode.error("g2/g3: Coords missing") - - elif asR == 0 and asI == 0 and asJ==0: - raise self.gcode.error("g2/g3: neither R nor I and J given") - - elif asR > 0 and (asI !=0 or asJ!=0): - raise self.gcode.error("g2/g3: R, I and J were given. Invalid") - else: # -------- execute conversion ----------- - coords = [] - clockwise = params['#command'].lower().startswith("g2") - asY = float(asY) - asX = float(asX) - - # use radius - # if asR > 0: - # not sure if neccessary since R barely seems to be used - - # use IJK - - if asI != 0 or asJ!=0: - coords = self.planArc(currentPos, - [asX,asY,0.,0.], - [asI, asJ], - clockwise) - ############################### - # converting coords into G1 codes (lazy aproch) - if len(coords)>0: - - # build dict and call cmd_G1 - for coord in coords: - g1_params = {'X': coord[0], 'Y': coord[1]} - if asZ!=None: - g1_params['Z']= float(asZ) - if asE>0: - g1_params['E']= float(asE)/len(coords) - if asF>0: - g1_params['F']= asF - - self.gcode.cmd_G1(g1_params) - - - - - else: - self.gcode.respond_info( - "could not tranlate from '" + params['#original'] + "'") + # Build list of linear coordinates to move to + coords = self.planArc(currentPos, [asX, asY, 0., 0.], [asI, asJ], + clockwise) + if not coords: + self.gcode.respond_info("G2/G3 could not translate '%s'" + % (params['#original'],)) + return + # Convert coords into G1 commands + for coord in coords: + g1_params = {'X': coord[0], 'Y': coord[1]} + if asZ is not None: + g1_params['Z'] = asZ + if asE is not None: + g1_params['E'] = asE / len(coords) + if asF is not None: + g1_params['F'] = asF + self.gcode.cmd_G1(g1_params) # function planArc() originates from marlin plan_arc() # https://github.com/MarlinFirmware/Marlin @@ -99,13 +63,7 @@ class ArcSupport: # The arc is approximated by generating many small linear segments. # The length of each segment is configured in MM_PER_ARC_SEGMENT # Arcs smaller then this value, will be a Line only - - def planArc( - self, - currentPos, - targetPos=[0.,0.,0.,0.], - offset=[0.,0.], - clockwise=False): + def planArc(self, currentPos, targetPos, offset, clockwise): # todo: sometimes produces full circles coords = [] MM_PER_ARC_SEGMENT = self.mm_per_arc_segment @@ -137,7 +95,6 @@ class ArcSupport: and currentPos[Y_AXIS] == targetPos[Y_AXIS]): angular_travel = math.radians(360) - flat_mm = radius * angular_travel mm_of_travel = linear_travel if(mm_of_travel == linear_travel): @@ -145,7 +102,6 @@ class ArcSupport: else: mm_of_travel = math.abs(flat_mm) - if (mm_of_travel < 0.001): return coords @@ -153,7 +109,6 @@ class ArcSupport: if(segments<1): segments=1 - raw = [0.,0.,0.,0.] theta_per_segment = float(angular_travel / segments) linear_per_segment = float(linear_travel / segments) @@ -161,7 +116,6 @@ class ArcSupport: # Initialize the linear axis raw[Z_AXIS] = currentPos[Z_AXIS]; - for i in range(1,segments+1): cos_Ti = math.cos(i * theta_per_segment) sin_Ti = math.sin(i * theta_per_segment) @@ -176,6 +130,5 @@ class ArcSupport: return coords - def load_config(config): return ArcSupport(config)