3

I've got some sliced models that represent the right-side arms and legs of a robot. I'm really happy with how they printed, so now I'd like to print the left-side arms and legs.

I was thinking it would be pretty trivial to parse the G-code file using Python and change the value of all the Xn commands from n to 2*h - n, where h is in the middle of the bed, say 110 or 120 mm for an Ender 3.

Before I fire up my favorite IDE, are there any major gotchas I might encounter from such a naïve approach to mirroring the G-code like this? I originally sliced in Cura 4.9.1.

agarza
  • 1,334
  • 2
  • 10
  • 30
Ron Jensen
  • 163
  • 8
  • There are g-code visualizers which will allow you to open your newly modified file to observe obvious problems. Is your reference to the middle of the bed valid? I've not examined a g-code file sufficiently enough to note if the origin is the bed center or a specific corner. – fred_dot_u Jul 03 '21 at 00:09
  • @fred_dot_u good point, I suppose I could find the bounding box of the print while I'm parsing the file. At least on the Ender 3 the home position seems to be at (0,0,0). – Ron Jensen Jul 03 '21 at 01:41
  • often times you can scale with a -1 value to mirror an axis if your slicer does not have an explicit mirror option. – John Meacham Jul 03 '21 at 03:32

3 Answers3

4

If your slicer does not have a mirror operation or a scale that allows negative values then mirroring in the G-code should be straightforward.

As long as your printer doesn't have certain specific tool change or homing or purge positions that are done in the G-code you can just transform it, otherwise you would want to skip these sections and just do the model data (it should be obvious looking at the code where model data starts).

In order to mirror it you just need to swap out the X coordinates in your G-code, If (0,0) is the center of your bed, as is often (but not always) the case for delta printers you will just want to negate the X, so G1 X30 Y-3 Z2 becomes G1 X-30 Y-3 Z2. If your coordinates have (0,0) in a corner (often the case for orthogonal printers) then you want to subtract X from the maximum X value. for instance if your bed is 250 mm wide then in G1 X30 Y10 Z3 X30 becomes X(250-30) or G1 X220 Y10 Z3.

There is only one caveat, some slicers will switch to relative movement using G91 for certain operations and then back to absolute with G90, so you will want to look out for these. Between a G91 and a G90 you will want to negate the X, no matter where the origin of your printer is.

When writing your script, I'd keep track of the minimum and maximum values encountered and the new minimum and maximum values and print them at the end as a sanity check so you can see if anything is wonky.

Greenonline
  • 5,831
  • 7
  • 30
  • 60
John Meacham
  • 156
  • 2
3

Based on @John Mecham's comprehensive answer, I whipped up a quick proof of concept. In the image below, the left arrow (top) is the original and the right arrow (bottom) is the reversed clone. Cura does generate a little relative offset code at the end, I think I handled it correctly.

Left and right arrows

import re

data_start = re.compile('^;LAYER_COUNT:[0-9]+')

ifilename = 'c:/Users/jentron128/Downloads/ThingVerse/Tools/arrow.gcode'
ofilename = 'c:/Users/jentron128/Downloads/ThingVerse/Tools/rev_arrow.gcode'

new_data =[]
BEDX = float(235)
h = BEDX # Absolute Positioning is default

mode='Skip'

with open(ifilename, 'r') as ifp:
    for d in ifp:
        new_d = ''
        tokens = d.split()
        if len(tokens) == 0:
            pass
        elif mode == 'Skip':
            new_d = d[0:-1]
            if data_start .match(d):
                mode = 'Go'
        else:
            if tokens[0] == 'G91': # Relative Positioning
                h = 0 
            elif tokens[0] == 'G90':# Absolute Positioning
                h = BEDX
                
            for t in tokens:
                if t.startswith('X'):
                    if len(t) > 1:
                        x = h - float(t[1:])
                        t = 'X'+f'{x:.3f}'
                new_d += t+' '
        
        new_data.append(new_d)

with open(ofilename, 'w') as ofp:
    for d in new_data:
        ofp.write(d+'\n') # how does writelines not support a line separator?
    
Ron Jensen
  • 163
  • 8
  • ```#!/usr/bin/env python3 import re from sys import argv data_start = re.compile('^;LAYER_COUNT:[0-9]+') ifilename = argv[1]; ofilename = ifilename.replace('.gcode', '-reversed.gcode'); ``` – Onza Oct 07 '22 at 17:15
  • I tried to comment to make it a reusable script that you can run against any file, but the comment got mangled; basically use sys to import argv and use the argv[1] for the filename :) – Onza Oct 07 '22 at 17:16
  • It didn't work for me nevertheless, not with a file generated by prusa. – Onza Oct 07 '22 at 17:25
  • At least on Windows I never run Python from a command line. If you have a (preferably small) prusa gcode file I will take a look to see if I can modify the code to work with it. – Ron Jensen Oct 07 '22 at 17:52
  • I think I found the issue, you are looking for a header that seemed to be specific to Cura, I made a new version based on yours which is correctly working for the prusa slicer files; I took all the math you used overall, it seems to work correctly now; honestly I am not sure, I have never touched gcode until now, but I think the G1 is universal. – Onza Oct 07 '22 at 18:07
  • Yes, my code has a state-machine built into it, and it seems the switch is based on a Cura specific comment. Anything before the first ";LAYER:0" comment is output as read because it's all printer setup, then the state switches to "Go" mode and processes ALL X directives after that, regardless of G code. It treats G90/G91 separately by setting h = 0 for RELATIVE positioning and h = bed width for ABSOLUTE positioning. – Ron Jensen Oct 07 '22 at 18:20
  • The easiest way around this is just change mode='Skip' to mode='Go' and then the script will process all the lines. – Ron Jensen Oct 07 '22 at 18:27
  • However we (or at least I) wasn't sure wether that was the correct assumption, I simply took all G1 token output and flipped that one, assuming those are the instructions; I wrote the code below as you can see. – Onza Oct 08 '22 at 07:12
0

Usage: python3 [filename.gcode] [bed-width]

#!/usr/bin/env python3

import re
from sys import argv

#first arg is the file, second arg is the bed x width
ifilename = argv[1];
ofilename = ifilename.replace('.gcode', '-reversed.gcode');

new_data =[]
BEDX = float(argv[2])
h = BEDX # Absolute Positioning is default

with open(ifilename, 'r') as ifp:
    for d in ifp:
        new_d = ''
        tokens = d.split()
        if len(tokens) == 0:
            pass
        else:
            if tokens[0] == 'G91': # Relative Positioning
                h = 0
                new_d += d
            elif tokens[0] == 'G90':# Absolute Positioning
                h = BEDX
                new_d += d
            elif tokens[0] == 'G1':
                for t in tokens:
                    if t.startswith('X'):
                        if len(t) > 1:
                            x = h - float(t[1:])
                            t = 'X'+f'{x:.3f}'
                    new_d += t+' '
            else:
                new_d += d

        new_data.append(new_d)

with open(ofilename, 'w') as ofp:
    for d in new_data:
        ofp.write(d+'\n') # how does writelines not support a line separator?

For those that have a problem with the script above, I think this fixes it.

Also there seems to be a bug with not writting G90 or G91 whenever it finds it.

Onza
  • 101
  • 1
  • The way you generate `ofilename` is not very safe. If the filename doesn't end in `.gcode` (e.g. if it ends in `.GCODE` or `.gco`) the replace will be a no-op and it looks like you'll overwrite the input. – R.. GitHub STOP HELPING ICE Oct 10 '22 at 14:48
  • You are right, make a third version or something; I hacked this quickly on the spot, maybe using argv[2] for output name. – Onza Oct 20 '22 at 07:05