| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| |
|
| | import FreeCAD |
| | from FreeCAD import Units |
| | import Path |
| | import Path.Base.Util as PathUtil |
| | import Path.Post.Utils as PostUtils |
| | import PathScripts.PathUtils as PathUtils |
| | import argparse |
| | import datetime |
| | import shlex |
| | import re |
| | from builtins import open as pyopen |
| |
|
| |
|
| | TOOLTIP = """ |
| | Generate g-code from a Path that is compatible with the grbl controller. |
| | import grbl_legacy_post |
| | grbl_legacy_post.export(object, "/path/to/file.ncc") |
| | """ |
| |
|
| |
|
| | |
| | |
| | |
| |
|
| | |
| | OUTPUT_COMMENTS = True |
| | OUTPUT_HEADER = True |
| | OUTPUT_LINE_NUMBERS = False |
| | OUTPUT_BCNC = False |
| | SHOW_EDITOR = True |
| | PRECISION = 3 |
| | TRANSLATE_DRILL_CYCLES = True |
| | PREAMBLE = """G17 G90 |
| | """ |
| | POSTAMBLE = """M5 |
| | G17 G90 |
| | M2 |
| | """ |
| |
|
| | SPINDLE_WAIT = 0 |
| | RETURN_TO = None |
| |
|
| | |
| | MODAL = False |
| | LINENR = 100 |
| | LINEINCR = 10 |
| | OUTPUT_TOOL_CHANGE = ( |
| | False |
| | ) |
| | DRILL_RETRACT_MODE = ( |
| | "G98" |
| | ) |
| | MOTION_MODE = "G90" |
| | UNITS = "G21" |
| | UNIT_FORMAT = "mm" |
| | UNIT_SPEED_FORMAT = "mm/min" |
| | PRE_OPERATION = """""" |
| | POST_OPERATION = """""" |
| | TOOL_CHANGE = """""" |
| |
|
| | |
| | |
| | |
| |
|
| | |
| | parser = argparse.ArgumentParser(prog="grbl", add_help=False) |
| | parser.add_argument("--comments", action="store_true", help="output comment (default)") |
| | parser.add_argument("--no-comments", action="store_true", help="suppress comment output") |
| | parser.add_argument("--header", action="store_true", help="output headers (default)") |
| | parser.add_argument("--no-header", action="store_true", help="suppress header output") |
| | parser.add_argument("--line-numbers", action="store_true", help="prefix with line numbers") |
| | parser.add_argument( |
| | "--no-line-numbers", |
| | action="store_true", |
| | help="don't prefix with line numbers (default)", |
| | ) |
| | parser.add_argument( |
| | "--show-editor", |
| | action="store_true", |
| | help="pop up editor before writing output (default)", |
| | ) |
| | parser.add_argument( |
| | "--no-show-editor", |
| | action="store_true", |
| | help="don't pop up editor before writing output", |
| | ) |
| | parser.add_argument("--precision", default="3", help="number of digits of precision, default=3") |
| | parser.add_argument( |
| | "--translate_drill", |
| | action="store_true", |
| | help="translate drill cycles G81, G82 & G83 in G0/G1 movements", |
| | ) |
| | parser.add_argument( |
| | "--no-translate_drill", |
| | action="store_true", |
| | help="don't translate drill cycles G81, G82 & G83 in G0/G1 movements (default)", |
| | ) |
| | parser.add_argument( |
| | "--preamble", |
| | help='set commands to be issued before the first command, default="G17 G90\\n"', |
| | ) |
| | parser.add_argument( |
| | "--postamble", |
| | help='set commands to be issued after the last command, default="M5\\nG17 G90\\nM2\\n"', |
| | ) |
| | parser.add_argument( |
| | "--inches", action="store_true", help="Convert output for US imperial mode (G20)" |
| | ) |
| | parser.add_argument("--tool-change", action="store_true", help="Insert M6 for all tool changes") |
| | parser.add_argument( |
| | "--wait-for-spindle", |
| | type=int, |
| | default=0, |
| | help="Wait for spindle to reach desired speed after M3 / M4, default=0", |
| | ) |
| | parser.add_argument( |
| | "--return-to", |
| | default="", |
| | help="Move to the specified coordinates at the end, e.g. --return-to=0,0", |
| | ) |
| | parser.add_argument( |
| | "--bcnc", |
| | action="store_true", |
| | help="Add Job operations as bCNC block headers. Consider suppressing existing comments: Add argument --no-comments", |
| | ) |
| | parser.add_argument( |
| | "--no-bcnc", action="store_true", help="suppress bCNC block header output (default)" |
| | ) |
| | TOOLTIP_ARGS = parser.format_help() |
| |
|
| |
|
| | |
| | |
| | |
| | MOTION_COMMANDS = [ |
| | "G0", |
| | "G00", |
| | "G1", |
| | "G01", |
| | "G2", |
| | "G02", |
| | "G3", |
| | "G03", |
| | ] |
| | RAPID_MOVES = ["G0", "G00"] |
| | SUPPRESS_COMMANDS = [] |
| | COMMAND_SPACE = " " |
| | |
| | CURRENT_X = 0 |
| | CURRENT_Y = 0 |
| | CURRENT_Z = 0 |
| |
|
| |
|
| | def processArguments(argstring): |
| |
|
| | global OUTPUT_HEADER |
| | global OUTPUT_COMMENTS |
| | global OUTPUT_LINE_NUMBERS |
| | global SHOW_EDITOR |
| | global PRECISION |
| | global PREAMBLE |
| | global POSTAMBLE |
| | global UNITS |
| | global UNIT_SPEED_FORMAT |
| | global UNIT_FORMAT |
| | global TRANSLATE_DRILL_CYCLES |
| | global OUTPUT_TOOL_CHANGE |
| | global SPINDLE_WAIT |
| | global RETURN_TO |
| | global OUTPUT_BCNC |
| |
|
| | try: |
| | args = parser.parse_args(shlex.split(argstring)) |
| | if args.no_header: |
| | OUTPUT_HEADER = False |
| | if args.header: |
| | OUTPUT_HEADER = True |
| | if args.no_comments: |
| | OUTPUT_COMMENTS = False |
| | if args.comments: |
| | OUTPUT_COMMENTS = True |
| | if args.no_line_numbers: |
| | OUTPUT_LINE_NUMBERS = False |
| | if args.line_numbers: |
| | OUTPUT_LINE_NUMBERS = True |
| | if args.no_show_editor: |
| | SHOW_EDITOR = False |
| | if args.show_editor: |
| | SHOW_EDITOR = True |
| | PRECISION = args.precision |
| | if args.preamble is not None: |
| | PREAMBLE = args.preamble.replace("\\n", "\n") |
| | if args.postamble is not None: |
| | POSTAMBLE = args.postamble.replace("\\n", "\n") |
| | if args.no_translate_drill: |
| | TRANSLATE_DRILL_CYCLES = False |
| | if args.translate_drill: |
| | TRANSLATE_DRILL_CYCLES = True |
| | if args.inches: |
| | UNITS = "G20" |
| | UNIT_SPEED_FORMAT = "in/min" |
| | UNIT_FORMAT = "in" |
| | PRECISION = 4 |
| | if args.tool_change: |
| | OUTPUT_TOOL_CHANGE = True |
| | if args.wait_for_spindle > 0: |
| | SPINDLE_WAIT = args.wait_for_spindle |
| | if args.return_to != "": |
| | RETURN_TO = [int(v) for v in args.return_to.split(",")] |
| | if len(RETURN_TO) != 2: |
| | RETURN_TO = None |
| | print("--return-to coordinates must be specified as <x>,<y>, ignoring") |
| | if args.bcnc: |
| | OUTPUT_BCNC = True |
| | if args.no_bcnc: |
| | OUTPUT_BCNC = False |
| |
|
| | except Exception as e: |
| | return False |
| |
|
| | return True |
| |
|
| |
|
| | |
| | def dump(obj): |
| | for attr in dir(obj): |
| | print("obj.%s = %s" % (attr, getattr(obj, attr))) |
| |
|
| |
|
| | def export(objectslist, filename, argstring): |
| |
|
| | if not processArguments(argstring): |
| | return None |
| |
|
| | global UNITS |
| | global UNIT_FORMAT |
| | global UNIT_SPEED_FORMAT |
| | global MOTION_MODE |
| | global SUPPRESS_COMMANDS |
| |
|
| | print("Post Processor: " + __name__ + " postprocessing...") |
| | gcode = "" |
| |
|
| | |
| | if OUTPUT_HEADER: |
| | gcode += linenumber() + "(Exported by FreeCAD)\n" |
| | gcode += linenumber() + "(Post Processor: " + __name__ + ")\n" |
| | gcode += linenumber() + "(Output Time:" + str(datetime.datetime.now()) + ")\n" |
| |
|
| | |
| | if TRANSLATE_DRILL_CYCLES: |
| | if len(SUPPRESS_COMMANDS) == 0: |
| | SUPPRESS_COMMANDS = ["G99", "G98", "G80"] |
| | else: |
| | SUPPRESS_COMMANDS += ["G99", "G98", "G80"] |
| |
|
| | |
| | if OUTPUT_COMMENTS: |
| | gcode += linenumber() + "(Begin preamble)\n" |
| | for line in PREAMBLE.splitlines(): |
| | gcode += linenumber() + line + "\n" |
| | |
| | if "G90" in PREAMBLE: |
| | MOTION_MODE = "G90" |
| | elif "G91" in PREAMBLE: |
| | MOTION_MODE = "G91" |
| | else: |
| | gcode += linenumber() + MOTION_MODE + "\n" |
| | if "G21" in PREAMBLE: |
| | UNITS = "G21" |
| | UNIT_FORMAT = "mm" |
| | UNIT_SPEED_FORMAT = "mm/min" |
| | elif "G20" in PREAMBLE: |
| | UNITS = "G20" |
| | UNIT_FORMAT = "in" |
| | UNIT_SPEED_FORMAT = "in/min" |
| | else: |
| | gcode += linenumber() + UNITS + "\n" |
| |
|
| | for obj in objectslist: |
| | |
| | |
| | |
| | |
| | if not hasattr(obj, "Path"): |
| | print( |
| | "The object " + obj.Name + " is not a path. Please select only path and Compounds." |
| | ) |
| | return |
| |
|
| | |
| | if not PathUtil.activeForOp(obj): |
| | continue |
| |
|
| | |
| | if OUTPUT_BCNC: |
| | gcode += linenumber() + "(Block-name: " + obj.Label + ")\n" |
| | gcode += linenumber() + "(Block-expand: 0)\n" |
| | gcode += linenumber() + "(Block-enable: 1)\n" |
| | if OUTPUT_COMMENTS: |
| | gcode += linenumber() + "(Begin operation: " + obj.Label + ")\n" |
| | for line in PRE_OPERATION.splitlines(True): |
| | gcode += linenumber() + line |
| |
|
| | |
| | coolantMode = PathUtil.coolantModeForOp(obj) |
| |
|
| | |
| | if OUTPUT_COMMENTS: |
| | if not coolantMode == "None": |
| | gcode += linenumber() + "(Coolant On:" + coolantMode + ")\n" |
| | if coolantMode == "Flood": |
| | gcode += linenumber() + "M8" + "\n" |
| | if coolantMode == "Mist": |
| | gcode += linenumber() + "M7" + "\n" |
| |
|
| | |
| | gcode += parse(obj) |
| |
|
| | |
| | if OUTPUT_COMMENTS: |
| | gcode += linenumber() + "(Finish operation: " + obj.Label + ")\n" |
| | for line in POST_OPERATION.splitlines(True): |
| | gcode += linenumber() + line |
| |
|
| | |
| | if not coolantMode == "None": |
| | if OUTPUT_COMMENTS: |
| | gcode += linenumber() + "(Coolant Off:" + coolantMode + ")\n" |
| | gcode += linenumber() + "M9" + "\n" |
| |
|
| | if RETURN_TO: |
| | gcode += linenumber() + "G0 X%s Y%s\n" % tuple(RETURN_TO) |
| |
|
| | |
| | if OUTPUT_BCNC: |
| | gcode += linenumber() + "(Block-name: post_amble)\n" |
| | gcode += linenumber() + "(Block-expand: 0)\n" |
| | gcode += linenumber() + "(Block-enable: 1)\n" |
| | if OUTPUT_COMMENTS: |
| | gcode += linenumber() + "(Begin postamble)\n" |
| | for line in POSTAMBLE.splitlines(): |
| | gcode += linenumber() + line + "\n" |
| |
|
| | |
| | if FreeCAD.GuiUp and SHOW_EDITOR: |
| | dia = PostUtils.GCodeEditorDialog() |
| | dia.editor.setPlainText(gcode) |
| | result = dia.exec_() |
| | if result: |
| | final = dia.editor.toPlainText() |
| | else: |
| | final = gcode |
| | else: |
| | final = gcode |
| |
|
| | print("Done postprocessing.") |
| |
|
| | |
| | if filename != "-": |
| | with pyopen(filename, "w") as gfile: |
| | gfile.write(final) |
| |
|
| | return final |
| |
|
| |
|
| | def linenumber(): |
| | if not OUTPUT_LINE_NUMBERS: |
| | return "" |
| | global LINENR |
| | global LINEINCR |
| | s = "N" + str(LINENR) + " " |
| | LINENR += LINEINCR |
| | return s |
| |
|
| |
|
| | def format_outstring(strTable): |
| | global COMMAND_SPACE |
| | |
| | s = "" |
| | for w in strTable: |
| | s += w + COMMAND_SPACE |
| | s = s.strip() |
| | return s |
| |
|
| |
|
| | def parse(pathobj): |
| |
|
| | global DRILL_RETRACT_MODE |
| | global MOTION_MODE |
| | global CURRENT_X |
| | global CURRENT_Y |
| | global CURRENT_Z |
| |
|
| | out = "" |
| | lastcommand = None |
| | precision_string = "." + str(PRECISION) + "f" |
| |
|
| | params = [ |
| | "X", |
| | "Y", |
| | "Z", |
| | "A", |
| | "B", |
| | "C", |
| | "U", |
| | "V", |
| | "W", |
| | "I", |
| | "J", |
| | "K", |
| | "F", |
| | "S", |
| | "T", |
| | "Q", |
| | "R", |
| | "L", |
| | "P", |
| | ] |
| |
|
| | if hasattr(pathobj, "Group"): |
| | if OUTPUT_COMMENTS: |
| | out += linenumber() + "(Compound: " + pathobj.Label + ")\n" |
| | for p in pathobj.Group: |
| | out += parse(p) |
| | return out |
| |
|
| | else: |
| | if not hasattr(pathobj, "Path"): |
| | return out |
| |
|
| | if OUTPUT_COMMENTS: |
| | out += linenumber() + "(Path: " + pathobj.Label + ")\n" |
| |
|
| | for c in PathUtils.getPathWithPlacement(pathobj).Commands: |
| | outstring = [] |
| | command = c.Name |
| |
|
| | outstring.append(command) |
| |
|
| | |
| | if MODAL: |
| | if command == lastcommand: |
| | outstring.pop(0) |
| |
|
| | |
| | for param in params: |
| | if param in c.Parameters: |
| | if param == "F": |
| | if command not in RAPID_MOVES: |
| | speed = Units.Quantity(c.Parameters["F"], FreeCAD.Units.Velocity) |
| | if speed.getValueAs(UNIT_SPEED_FORMAT) > 0.0: |
| | outstring.append( |
| | param |
| | + format( |
| | float(speed.getValueAs(UNIT_SPEED_FORMAT)), |
| | precision_string, |
| | ) |
| | ) |
| | elif param in ["T", "H", "S"]: |
| | outstring.append(param + str(int(c.Parameters[param]))) |
| | elif param in ["D", "P", "L"]: |
| | outstring.append(param + str(c.Parameters[param])) |
| | elif param in ["A", "B", "C"]: |
| | outstring.append(param + format(c.Parameters[param], precision_string)) |
| | else: |
| | pos = Units.Quantity(c.Parameters[param], FreeCAD.Units.Length) |
| | outstring.append( |
| | param + format(float(pos.getValueAs(UNIT_FORMAT)), precision_string) |
| | ) |
| |
|
| | |
| | lastcommand = command |
| |
|
| | |
| | if command in MOTION_COMMANDS: |
| | if "X" in c.Parameters: |
| | CURRENT_X = Units.Quantity(c.Parameters["X"], FreeCAD.Units.Length) |
| | if "Y" in c.Parameters: |
| | CURRENT_Y = Units.Quantity(c.Parameters["Y"], FreeCAD.Units.Length) |
| | if "Z" in c.Parameters: |
| | CURRENT_Z = Units.Quantity(c.Parameters["Z"], FreeCAD.Units.Length) |
| |
|
| | if command in ("G98", "G99"): |
| | DRILL_RETRACT_MODE = command |
| |
|
| | if command in ("G90", "G91"): |
| | MOTION_MODE = command |
| |
|
| | if TRANSLATE_DRILL_CYCLES: |
| | if command in ("G81", "G82", "G83"): |
| | out += drill_translate(outstring, command, c.Parameters) |
| | |
| | outstring = [] |
| |
|
| | if SPINDLE_WAIT > 0: |
| | if command in ("M3", "M03", "M4", "M04"): |
| | out += linenumber() + format_outstring(outstring) + "\n" |
| | out += linenumber() + format_outstring(["G4", "P%s" % SPINDLE_WAIT]) + "\n" |
| | outstring = [] |
| |
|
| | |
| | if command in ("M6", "M06"): |
| | if OUTPUT_COMMENTS: |
| | out += linenumber() + "(Begin toolchange)\n" |
| | if not OUTPUT_TOOL_CHANGE: |
| | outstring.insert(0, "(") |
| | outstring.append(")") |
| | else: |
| | for line in TOOL_CHANGE.splitlines(True): |
| | out += linenumber() + line |
| |
|
| | if command == "message": |
| | if OUTPUT_COMMENTS is False: |
| | out = [] |
| | else: |
| | outstring.pop(0) |
| |
|
| | if command in SUPPRESS_COMMANDS: |
| | outstring.insert(0, "(") |
| | outstring.append(")") |
| |
|
| | |
| | if len(outstring) >= 1: |
| | out += linenumber() + format_outstring(outstring) + "\n" |
| |
|
| | |
| | m = re.match(r"^\(MC_RUN_COMMAND: ([^)]+)\)$", command) |
| | if m: |
| | raw_command = m.group(1) |
| | out += linenumber() + raw_command + "\n" |
| |
|
| | return out |
| |
|
| |
|
| | def drill_translate(outstring, cmd, params): |
| | global DRILL_RETRACT_MODE |
| | global MOTION_MODE |
| | global CURRENT_X |
| | global CURRENT_Y |
| | global CURRENT_Z |
| | global UNITS |
| | global UNIT_FORMAT |
| | global UNIT_SPEED_FORMAT |
| |
|
| | strFormat = "." + str(PRECISION) + "f" |
| | trBuff = "" |
| |
|
| | if OUTPUT_COMMENTS: |
| | outstring[0] = "(" + outstring[0] |
| | outstring[-1] = outstring[-1] + ")" |
| | trBuff += linenumber() + format_outstring(outstring) + "\n" |
| |
|
| | |
| | |
| | |
| | param_X = Units.Quantity(params["X"], FreeCAD.Units.Length) |
| | param_Y = Units.Quantity(params["Y"], FreeCAD.Units.Length) |
| | param_Z = Units.Quantity(params["Z"], FreeCAD.Units.Length) |
| | param_R = Units.Quantity(params["R"], FreeCAD.Units.Length) |
| | |
| | if param_R < param_Z: |
| | trBuff += linenumber() + "(drill cycle error: R less than Z )\n" |
| | return trBuff |
| |
|
| | if MOTION_MODE == "G91": |
| | param_X += CURRENT_X |
| | param_Y += CURRENT_Y |
| | param_Z += CURRENT_Z |
| | param_R += param_Z |
| |
|
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| |
|
| | if DRILL_RETRACT_MODE == "G99": |
| | clear_Z = param_R |
| | if DRILL_RETRACT_MODE == "G98" and CURRENT_Z >= param_R: |
| | clear_Z = CURRENT_Z |
| | else: |
| | clear_Z = param_R |
| |
|
| | strG0_clear_Z = "G0 Z" + format(float(clear_Z.getValueAs(UNIT_FORMAT)), strFormat) + "\n" |
| | strG0_param_R = "G0 Z" + format(float(param_R.getValueAs(UNIT_FORMAT)), strFormat) + "\n" |
| |
|
| | |
| | drill_feedrate = Units.Quantity(params["F"], FreeCAD.Units.Velocity) |
| | strF_Feedrate = " F" + format(float(drill_feedrate.getValueAs(UNIT_SPEED_FORMAT)), ".2f") + "\n" |
| |
|
| | if cmd == "G83": |
| | drill_Step = Units.Quantity(params["Q"], FreeCAD.Units.Length) |
| | a_bit = ( |
| | drill_Step * 0.05 |
| | ) |
| | elif cmd == "G82": |
| | drill_DwellTime = params["P"] |
| |
|
| | |
| | try: |
| | if MOTION_MODE == "G91": |
| | trBuff += linenumber() + "G90\n" |
| |
|
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| |
|
| | if CURRENT_Z < param_R: |
| | trBuff += linenumber() + strG0_param_R |
| | trBuff += ( |
| | linenumber() |
| | + "G0 X" |
| | + format(float(param_X.getValueAs(UNIT_FORMAT)), strFormat) |
| | + " Y" |
| | + format(float(param_Y.getValueAs(UNIT_FORMAT)), strFormat) |
| | + "\n" |
| | ) |
| | if CURRENT_Z > param_R: |
| | trBuff += linenumber() + strG0_param_R |
| |
|
| | last_Stop_Z = param_R |
| |
|
| | |
| | if cmd in ("G81", "G82"): |
| | trBuff += ( |
| | linenumber() |
| | + "G1 Z" |
| | + format(float(param_Z.getValueAs(UNIT_FORMAT)), strFormat) |
| | + strF_Feedrate |
| | ) |
| | |
| | if cmd == "G82": |
| | trBuff += linenumber() + "G4 P" + str(drill_DwellTime) + "\n" |
| | trBuff += linenumber() + strG0_clear_Z |
| | else: |
| | if params["Q"] != 0: |
| | while 1: |
| | if last_Stop_Z != clear_Z: |
| | clearance_depth = ( |
| | last_Stop_Z + a_bit |
| | ) |
| | trBuff += ( |
| | linenumber() |
| | + "G0 Z" |
| | + format( |
| | float(clearance_depth.getValueAs(UNIT_FORMAT)), |
| | strFormat, |
| | ) |
| | + "\n" |
| | ) |
| | next_Stop_Z = last_Stop_Z - drill_Step |
| | if next_Stop_Z > param_Z: |
| | trBuff += ( |
| | linenumber() |
| | + "G1 Z" |
| | + format(float(next_Stop_Z.getValueAs(UNIT_FORMAT)), strFormat) |
| | + strF_Feedrate |
| | ) |
| | trBuff += linenumber() + strG0_clear_Z |
| | last_Stop_Z = next_Stop_Z |
| | else: |
| | trBuff += ( |
| | linenumber() |
| | + "G1 Z" |
| | + format(float(param_Z.getValueAs(UNIT_FORMAT)), strFormat) |
| | + strF_Feedrate |
| | ) |
| | trBuff += linenumber() + strG0_clear_Z |
| | break |
| |
|
| | except Exception as e: |
| | pass |
| |
|
| | if MOTION_MODE == "G91": |
| | trBuff += linenumber() + "G91" |
| |
|
| | return trBuff |
| |
|
| |
|
| | |
| |
|