This repository has been archived on 2023-03-25. You can view files and clone it, but cannot push or open issues or pull requests.

1460 lines
62 KiB
Python

#!/usr/bin/env python3
"""
Modified by Jay Johnson 2015, J Tech Photonics, Inc., jtechphotonics.com
modified by Adam Polak 2014, polakiumengineering.org
based on Copyright (C) 2009 Nick Drobchenko, nick@cnc-club.ru
based on gcode.py (C) 2007 hugomatic...
based on addnodes.py (C) 2005,2007 Aaron Spike, aaron@ekips.org
based on dots.py (C) 2005 Aaron Spike, aaron@ekips.org
based on interp.py (C) 2005 Aaron Spike, aaron@ekips.org
based on bezmisc.py (C) 2005 Aaron Spike, aaron@ekips.org
based on cubicsuperpath.py (C) 2005 Aaron Spike, aaron@ekips.org
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
"""
import inkex
import simpletransform
import os
import math
import bezmisc
import re
import sys
import time
import numpy
import gettext
_ = gettext.gettext
# Deprecation hack. Access the formatStyle differently for inkscape >= 1.0
target_version = 1.0
if target_version < 1.0:
# simplestyle
import simplestyle
# etree
etree = inkex.etree
# cubicsuperpath
import cubicsuperpath
parsePath = cubicsuperpath.parsePath
# Inkex.Boolean
inkex.Boolean = bool
else:
# simplestyle
# Class and method names follow the old Inkscape API for compatibility's sake.
# When support is dropped for older versions this can be ganged to follow PEP 8.
class simplestyle(object): # noqa
# I think anonymous declarations would have been cleaner. However, Python 2 doesn't like how I use them
@staticmethod
def formatStyle(a): # noqa
return str(inkex.Style(a))
@staticmethod
def parseStyle(s): # noqa
return dict(inkex.Style.parse_str(s))
# etree
from lxml import etree # noqa
# cubicsuperpath
from inkex.paths import CubicSuperPath # noqa
parsePath = CubicSuperPath
# Check if inkex has error messages. (0.46 version does not have one) Could be removed later.
if "errormsg" not in dir(inkex):
inkex.errormsg = lambda msg: sys.stderr.write((str(msg) + "\n").encode("UTF-8"))
def bezierslopeatt(xxx_todo_changeme, t):
((bx0, by0), (bx1, by1), (bx2, by2), (bx3, by3)) = xxx_todo_changeme
ax, ay, bx, by, cx, cy, x0, y0 = bezmisc.bezierparameterize(((bx0, by0), (bx1, by1), (bx2, by2), (bx3, by3)))
dx = 3 * ax * (t ** 2) + 2 * bx * t + cx
dy = 3 * ay * (t ** 2) + 2 * by * t + cy
if dx == dy == 0:
dx = 6 * ax * t + 2 * bx
dy = 6 * ay * t + 2 * by
if dx == dy == 0:
dx = 6 * ax
dy = 6 * ay
if dx == dy == 0:
print_("Slope error x = %s*t^3+%s*t^2+%s*t+%s, y = %s*t^3+%s*t^2+%s*t+%s, t = %s, dx==dy==0" % (
ax, bx, cx, dx, ay, by, cy, dy, t))
print_(((bx0, by0), (bx1, by1), (bx2, by2), (bx3, by3)))
dx, dy = 1, 1
return dx, dy
bezmisc.bezierslopeatt = bezierslopeatt
################################################################################
#
# Styles and additional parameters
#
################################################################################
math.pi2 = math.pi * 2
straight_tolerance = 0.0001
straight_distance_tolerance = 0.0001
engraving_tolerance = 0.0001
loft_lengths_tolerance = 0.0000001
options = {}
defaults = {
'header': """
G90
""",
'footer': """G1 X0 Y0
"""
}
intersection_recursion_depth = 10
intersection_tolerance = 0.00001
styles = {
"loft_style": {
'main curve': simplestyle.formatStyle(
{'stroke': '#88f', 'fill': 'none', 'stroke-width': '1', 'marker-end': 'url(#Arrow2Mend)'}),
},
"biarc_style": {
'biarc0': simplestyle.formatStyle(
{'stroke': '#88f', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '1'}),
'biarc1': simplestyle.formatStyle(
{'stroke': '#8f8', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '1'}),
'line': simplestyle.formatStyle(
{'stroke': '#f88', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '1'}),
'area': simplestyle.formatStyle(
{'stroke': '#777', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '0.1'}),
},
"biarc_style_dark": {
'biarc0': simplestyle.formatStyle(
{'stroke': '#33a', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '1'}),
'biarc1': simplestyle.formatStyle(
{'stroke': '#3a3', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '1'}),
'line': simplestyle.formatStyle(
{'stroke': '#a33', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '1'}),
'area': simplestyle.formatStyle(
{'stroke': '#222', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '0.3'}),
},
"biarc_style_dark_area": {
'biarc0': simplestyle.formatStyle(
{'stroke': '#33a', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '0.1'}),
'biarc1': simplestyle.formatStyle(
{'stroke': '#3a3', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '0.1'}),
'line': simplestyle.formatStyle(
{'stroke': '#a33', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '0.1'}),
'area': simplestyle.formatStyle(
{'stroke': '#222', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '0.3'}),
},
"biarc_style_i": {
'biarc0': simplestyle.formatStyle(
{'stroke': '#880', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '1'}),
'biarc1': simplestyle.formatStyle(
{'stroke': '#808', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '1'}),
'line': simplestyle.formatStyle(
{'stroke': '#088', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '1'}),
'area': simplestyle.formatStyle(
{'stroke': '#999', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '0.3'}),
},
"biarc_style_dark_i": {
'biarc0': simplestyle.formatStyle(
{'stroke': '#dd5', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '1'}),
'biarc1': simplestyle.formatStyle(
{'stroke': '#d5d', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '1'}),
'line': simplestyle.formatStyle(
{'stroke': '#5dd', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '1'}),
'area': simplestyle.formatStyle(
{'stroke': '#aaa', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '0.3'}),
},
"biarc_style_lathe_feed": {
'biarc0': simplestyle.formatStyle(
{'stroke': '#07f', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '.4'}),
'biarc1': simplestyle.formatStyle(
{'stroke': '#0f7', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '.4'}),
'line': simplestyle.formatStyle(
{'stroke': '#f44', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '.4'}),
'area': simplestyle.formatStyle(
{'stroke': '#aaa', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '0.3'}),
},
"biarc_style_lathe_passing feed": {
'biarc0': simplestyle.formatStyle(
{'stroke': '#07f', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '.4'}),
'biarc1': simplestyle.formatStyle(
{'stroke': '#0f7', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '.4'}),
'line': simplestyle.formatStyle(
{'stroke': '#f44', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '.4'}),
'area': simplestyle.formatStyle(
{'stroke': '#aaa', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '0.3'}),
},
"biarc_style_lathe_fine feed": {
'biarc0': simplestyle.formatStyle(
{'stroke': '#7f0', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '.4'}),
'biarc1': simplestyle.formatStyle(
{'stroke': '#f70', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '.4'}),
'line': simplestyle.formatStyle(
{'stroke': '#744', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '.4'}),
'area': simplestyle.formatStyle(
{'stroke': '#aaa', 'fill': 'none', "marker-end": "url(#DrawCurveMarker)", 'stroke-width': '0.3'}),
},
"area artefact": simplestyle.formatStyle({'stroke': '#ff0000', 'fill': '#ffff00', 'stroke-width': '1'}),
"area artefact arrow": simplestyle.formatStyle({'stroke': '#ff0000', 'fill': '#ffff00', 'stroke-width': '1'}),
"dxf_points": simplestyle.formatStyle({"stroke": "#ff0000", "fill": "#ff0000"}),
}
################################################################################
# Cubic Super Path additional functions
################################################################################
def csp_segment_to_bez(sp1, sp2):
return sp1[1:] + sp2[:2]
def csp_split(sp1, sp2, t=.5):
[x1, y1], [x2, y2], [x3, y3], [x4, y4] = sp1[1], sp1[2], sp2[0], sp2[1]
x12 = x1 + (x2 - x1) * t
y12 = y1 + (y2 - y1) * t
x23 = x2 + (x3 - x2) * t
y23 = y2 + (y3 - y2) * t
x34 = x3 + (x4 - x3) * t
y34 = y3 + (y4 - y3) * t
x1223 = x12 + (x23 - x12) * t
y1223 = y12 + (y23 - y12) * t
x2334 = x23 + (x34 - x23) * t
y2334 = y23 + (y34 - y23) * t
x = x1223 + (x2334 - x1223) * t
y = y1223 + (y2334 - y1223) * t
return [sp1[0], sp1[1], [x12, y12]], [[x1223, y1223], [x, y], [x2334, y2334]], [[x34, y34], sp2[1], sp2[2]]
def csp_curvature_at_t(sp1, sp2, t, depth=3):
ax, ay, bx, by, cx, cy, dx, dy = bezmisc.bezierparameterize(csp_segment_to_bez(sp1, sp2))
# curvature = (x'y''-y'x'') / (x'^2+y'^2)^1.5
f1x = 3 * ax * t ** 2 + 2 * bx * t + cx
f1y = 3 * ay * t ** 2 + 2 * by * t + cy
f2x = 6 * ax * t + 2 * bx
f2y = 6 * ay * t + 2 * by
d = (f1x ** 2 + f1y ** 2) ** 1.5
if d != 0:
return (f1x * f2y - f1y * f2x) / d
else:
t1 = f1x * f2y - f1y * f2x
if t1 > 0: return 1e100
if t1 < 0: return -1e100
# Use the Lapitals rule to solve 0/0 problem for 2 times...
t1 = 2 * (bx * ay - ax * by) * t + (ay * cx - ax * cy)
if t1 > 0: return 1e100
if t1 < 0: return -1e100
t1 = bx * ay - ax * by
if t1 > 0: return 1e100
if t1 < 0: return -1e100
if depth > 0:
# little hack ;^) hope it wont influence anything...
return csp_curvature_at_t(sp1, sp2, t * 1.004, depth - 1)
return 1e100
def csp_at_t(sp1, sp2, t):
ax, bx, cx, dx = sp1[1][0], sp1[2][0], sp2[0][0], sp2[1][0]
ay, by, cy, dy = sp1[1][1], sp1[2][1], sp2[0][1], sp2[1][1]
x1, y1 = ax + (bx - ax) * t, ay + (by - ay) * t
x2, y2 = bx + (cx - bx) * t, by + (cy - by) * t
x3, y3 = cx + (dx - cx) * t, cy + (dy - cy) * t
x4, y4 = x1 + (x2 - x1) * t, y1 + (y2 - y1) * t
x5, y5 = x2 + (x3 - x2) * t, y2 + (y3 - y2) * t
x, y = x4 + (x5 - x4) * t, y4 + (y5 - y4) * t
return [x, y]
def cspseglength(sp1, sp2, tolerance=0.001):
bez = (sp1[1][:], sp1[2][:], sp2[0][:], sp2[1][:])
return bezmisc.bezierlength(bez, tolerance)
# Distance calculation from point to arc
def point_to_arc_distance(p, arc):
P0, P2, c, a = arc
dist = None
p = P(p)
r = (P0 - c).mag()
if r > 0:
i = c + (p - c).unit() * r
alpha = ((i - c).angle() - (P0 - c).angle())
if a * alpha < 0:
if alpha > 0:
alpha = alpha - math.pi2
else:
alpha = math.pi2 + alpha
if between(alpha, 0, a) or min(abs(alpha), abs(alpha - a)) < straight_tolerance:
return (p - i).mag(), (i.x, i.y)
else:
d1, d2 = (p - P0).mag(), (p - P2).mag()
if d1 < d2:
return (d1, (P0.x, P0.y))
else:
return (d2, (P2.x, P2.y))
def csp_to_arc_distance(sp1, sp2, arc1, arc2, tolerance=0.01): # arc = [start,end,center,alpha]
n, i = 10, 0
d, d1, dl = (0, (0, 0)), (0, (0, 0)), 0
while i < 1 or (abs(d1[0] - dl[0]) > tolerance and i < 4):
i += 1
dl = d1 * 1
for j in range(n + 1):
t = float(j) / n
p = csp_at_t(sp1, sp2, t)
d = min(point_to_arc_distance(p, arc1), point_to_arc_distance(p, arc2))
# inkex.utils.debug("---Debug---")
# inkex.utils.debug(str(d1) + str(d))
# inkex.utils.debug(str(tuple(d1)) + str(tuple(d)))
d1 = max(tuple(d1), tuple(d))
n = n * 2
return d1[0]
################################################################################
# Common functions
################################################################################
def atan2(*arg):
if len(arg) == 1 and (type(arg[0]) == type([0., 0.]) or type(arg[0]) == type((0., 0.))):
return (math.pi / 2 - math.atan2(arg[0][0], arg[0][1])) % math.pi2
elif len(arg) == 2:
return (math.pi / 2 - math.atan2(arg[0], arg[1])) % math.pi2
else:
raise ValueError("Bad argumets for atan! (%s)" % arg)
def between(c, x, y):
return x - straight_tolerance <= c <= y + straight_tolerance or y - straight_tolerance <= c <= x + straight_tolerance
# Print arguments into specified log file
def print_(*arg):
f = open(options.log_filename, "a")
for s in arg:
s = str(str(s).encode('unicode_escape')) + " "
f.write(s)
f.write("\n")
f.close()
################################################################################
# Point (x,y) operations
################################################################################
class P:
def __init__(self, x, y=None):
if not y == None:
self.x, self.y = float(x), float(y)
else:
self.x, self.y = float(x[0]), float(x[1])
def __add__(self, other):
return P(self.x + other.x, self.y + other.y)
def __sub__(self, other):
return P(self.x - other.x, self.y - other.y)
def __neg__(self):
return P(-self.x, -self.y)
def __mul__(self, other):
if isinstance(other, P):
return self.x * other.x + self.y * other.y
return P(self.x * other, self.y * other)
__rmul__ = __mul__
def __div__(self, other):
return P(self.x / other, self.y / other)
# Added to support python 3
__floordiv__ = __div__
__truediv__ = __div__
def mag(self):
return math.hypot(self.x, self.y)
def unit(self):
h = self.mag()
if h:
return self / h
else:
return P(0, 0)
def angle(self):
return math.atan2(self.y, self.x)
def __repr__(self):
return '%f,%f' % (self.x, self.y)
def l2(self):
return self.x * self.x + self.y * self.y
################################################################################
#
# Biarc function
#
# Calculates biarc approximation of cubic super path segment
# splits segment if needed or approximates it with straight line
#
################################################################################
def biarc(sp1, sp2, z1, z2, depth=0):
def biarc_split(sp1, sp2, z1, z2, depth):
if depth < options.biarc_max_split_depth:
sp1, sp2, sp3 = csp_split(sp1, sp2)
l1, l2 = cspseglength(sp1, sp2), cspseglength(sp2, sp3)
if l1 + l2 == 0:
zm = z1
else:
zm = z1 + (z2 - z1) * l1 / (l1 + l2)
return biarc(sp1, sp2, z1, zm, depth + 1) + biarc(sp2, sp3, zm, z2, depth + 1)
else:
return [[sp1[1], 'line', 0, 0, sp2[1], [z1, z2]]]
P0, P4 = P(sp1[1]), P(sp2[1])
TS, TE, v = (P(sp1[2]) - P0), -(P(sp2[0]) - P4), P0 - P4
tsa, tea, va = TS.angle(), TE.angle(), v.angle()
if TE.mag() < straight_distance_tolerance and TS.mag() < straight_distance_tolerance:
# Both tangents are zerro - line straight
return [[sp1[1], 'line', 0, 0, sp2[1], [z1, z2]]]
if TE.mag() < straight_distance_tolerance:
TE = -(TS + v).unit()
r = TS.mag() / v.mag() * 2
elif TS.mag() < straight_distance_tolerance:
TS = -(TE + v).unit()
r = 1 / (TE.mag() / v.mag() * 2)
else:
r = TS.mag() / TE.mag()
TS, TE = TS.unit(), TE.unit()
tang_are_parallel = (
(tsa - tea) % math.pi < straight_tolerance or math.pi - (tsa - tea) % math.pi < straight_tolerance)
if (tang_are_parallel and
((
v.mag() < straight_distance_tolerance or TE.mag() < straight_distance_tolerance or TS.mag() < straight_distance_tolerance) or
1 - abs(TS * v / (TS.mag() * v.mag())) < straight_tolerance)):
# Both tangents are parallel and start and end are the same - line straight
# or one of tangents still smaller then tollerance
# Both tangents and v are parallel - line straight
return [[sp1[1], 'line', 0, 0, sp2[1], [z1, z2]]]
c, b, a = v * v, 2 * v * (r * TS + TE), 2 * r * (TS * TE - 1)
if v.mag() == 0:
return biarc_split(sp1, sp2, z1, z2, depth)
asmall, bsmall, csmall = abs(a) < 10 ** -10, abs(b) < 10 ** -10, abs(c) < 10 ** -10
if asmall and b != 0:
beta = -c / b
elif csmall and a != 0:
beta = -b / a
elif not asmall:
discr = b * b - 4 * a * c
if discr < 0: raise ValueError(a, b, c, discr)
disq = discr ** .5
beta1 = (-b - disq) / 2 / a
beta2 = (-b + disq) / 2 / a
if beta1 * beta2 > 0: raise ValueError(a, b, c, disq, beta1, beta2)
beta = max(beta1, beta2)
elif asmall and bsmall:
return biarc_split(sp1, sp2, z1, z2, depth)
alpha = beta * r
ab = alpha + beta
P1 = P0 + alpha * TS
P3 = P4 - beta * TE
P2 = (beta / ab) * P1 + (alpha / ab) * P3
def calculate_arc_params(P0, P1, P2):
D = (P0 + P2) / 2
if (D - P1).mag() == 0: return None, None
R = D - ((D - P0).mag() ** 2 / (D - P1).mag()) * (P1 - D).unit()
p0a, p1a, p2a = (P0 - R).angle() % (2 * math.pi), (P1 - R).angle() % (2 * math.pi), (P2 - R).angle() % (
2 * math.pi)
alpha = (p2a - p0a) % (2 * math.pi)
if (p0a < p2a and (p1a < p0a or p2a < p1a)) or (p2a < p1a < p0a):
alpha = -2 * math.pi + alpha
if abs(R.x) > 1000000 or abs(R.y) > 1000000 or (R - P0).mag() < .1:
return None, None
else:
return R, alpha
R1, a1 = calculate_arc_params(P0, P1, P2)
R2, a2 = calculate_arc_params(P2, P3, P4)
if R1 == None or R2 == None or (R1 - P0).mag() < straight_tolerance or (
R2 - P2).mag() < straight_tolerance: return [[sp1[1], 'line', 0, 0, sp2[1], [z1, z2]]]
d = csp_to_arc_distance(sp1, sp2, [P0, P2, R1, a1], [P2, P4, R2, a2])
if d > 1 and depth < options.biarc_max_split_depth:
return biarc_split(sp1, sp2, z1, z2, depth)
else:
if R2.mag() * a2 == 0:
zm = z2
else:
zm = z1 + (z2 - z1) * (abs(R1.mag() * a1)) / (abs(R2.mag() * a2) + abs(R1.mag() * a1))
return [[sp1[1], 'arc', [R1.x, R1.y], a1, [P2.x, P2.y], [z1, zm]],
[[P2.x, P2.y], 'arc', [R2.x, R2.y], a2, [P4.x, P4.y], [zm, z2]]]
################################################################################
# Polygon class
################################################################################
class Polygon:
def __init__(self, polygon=None):
self.polygon = [] if polygon == None else polygon[:]
def add(self, add):
if type(add) == type([]):
self.polygon += add[:]
else:
self.polygon += add.polygon[:]
class ArrangementGenetic:
# gene = [fittness, order, rotation, xposition]
# spieces = [gene]*shapes count
# population = [spieces]
def __init__(self, polygons, material_width):
self.population = []
self.genes_count = len(polygons)
self.polygons = polygons
self.width = material_width
self.mutation_factor = 0.1
self.order_mutate_factor = 1.
self.move_mutate_factor = 1.
################################################################################
###
### Gcodetools class
###
################################################################################
class LaserGcode(inkex.Effect):
def export_gcode(self, gcode):
gcode_pass = gcode
for x in range(1, self.options.passes):
gcode += "G91\nG1 Z-" + self.options.pass_depth + "\nG90\n" + gcode_pass
f = open(self.options.directory + self.options.file, "w")
f.write(
self.options.laser_off_command + " S0" + "\n" + self.header +
"G1 F" + self.options.travel_speed + "\n" + gcode + self.footer)
f.close()
def add_arguments_old(self):
add_option = self.OptionParser.add_option
for arg in self.arguments:
# Stringify add_option arguments
action = arg["action"] if "action" in arg else "store"
arg_type = {str: "str", int: "int", bool: "inkbool"}[arg["type"]]
default = arg["type"](arg["default"])
add_option("", arg["name"], action=action, type=arg_type, dest=arg["dest"],
default=default, help=arg["help"])
def add_arguments_new(self):
add_argument = self.arg_parser.add_argument
for arg in self.arguments:
# Not using kwargs unpacking for clarity, flexibility and constancy with add_arguments_old
action = arg["action"] if "action" in arg else "store"
add_argument(arg["name"], action=action, type=arg["type"], dest=arg["dest"],
default=arg["default"], help=arg["help"])
def __init__(self):
inkex.Effect.__init__(self)
# Define command line arguments, inkex will use these to interface with the GUI defined in laser.ini
self.arguments = [
{"name": "--directory", "type": str, "dest": "directory",
"default": "", "help": "Output directory"},
{"name": "--filename", "type": str, "dest": "file",
"default": "output.gcode", "help": "File name"},
{"name": "--add-numeric-suffix-to-filename", "type": inkex.Boolean,
"dest": "add_numeric_suffix_to_filename", "default": False,
"help": "Add numeric suffix to file name"},
{"name": "--laser-command", "type": str, "dest": "laser_command",
"default": "M03", "help": "Laser gcode command"},
{"name": "--laser-off-command", "type": str, "dest": "laser_off_command",
"default": "M05", "help": "Laser gcode end command"},
{"name": "--laser-speed", "type": int, "dest": "laser_speed", "default": 750,
"help": "Laser speed (mm/min},"},
{"name": "--travel-speed", "type": str, "dest": "travel_speed",
"default": "3000", "help": "Travel speed (mm/min},"},
{"name": "--laser-power", "type": int, "dest": "laser_power", "default": 255,
"help": "S# is 256 or 10000 for full power"},
{"name": "--passes", "type": int, "dest": "passes", "default": 1,
"help": "Quantity of passes"},
{"name": "--pass-depth", "type": str, "dest": "pass_depth", "default": 1,
"help": "Depth of laser cut"},
{"name": "--power-delay", "type": str, "dest": "power_delay",
"default": "0", "help": "Laser power-on delay (ms},"},
{"name": "--suppress-all-messages", "type": inkex.Boolean,
"dest": "suppress_all_messages", "default": True,
"help": "Hide messages during g-code generation"},
{"name": "--create-log", "type": bool, "dest": "log_create_log",
"default": False, "help": "Create log files"},
{"name": "--log-filename", "type": str, "dest": "log_filename",
"default": '', "help": "Create log files"},
{"name": "--engraving-draw-calculation-paths", "type": inkex.Boolean,
"dest": "engraving_draw_calculation_paths", "default": False,
"help": "Draw additional graphics to debug engraving path"},
{"name": "--unit", "type": str, "dest": "unit",
"default": "G21 (All units in mm},", "help": "Units either mm or inches"},
{"name": "--active-tab", "type": str, "dest": "active_tab", "default": "",
"help": "Defines which tab is active"},
{"name": "--biarc-max-split-depth", "type": int,
"dest": "biarc_max_split_depth", "default": "4",
"help": "Defines maximum depth of splitting while approximating using biarcs."}
]
if target_version < 1.0:
self.add_arguments_old()
else:
self.add_arguments_new()
# Another hack to maintain support across different Inkscape versions
if target_version < 1.0:
self.selected_hack = self.selected
else:
self.selected_hack = self.svg.selected
def parse_curve(self, p, layer, w=None, f=None):
c = []
if len(p) == 0:
return []
p = self.transform_csp(p, layer)
# Sort to reduce Rapid distance
k = list(range(1, len(p)))
keys = [0]
while len(k) > 0:
end = p[keys[-1]][-1][1]
dist = (float('-inf'), float('-inf'))
for i in range(len(k)):
start = p[k[i]][0][1]
dist = max((-((end[0] - start[0]) ** 2 + (end[1] - start[1]) ** 2), i), dist)
keys += [k[dist[1]]]
del k[dist[1]]
for k in keys:
subpath = p[k]
c += [[[subpath[0][1][0], subpath[0][1][1]], 'move', 0, 0]]
for i in range(1, len(subpath)):
sp1 = [[subpath[i - 1][j][0], subpath[i - 1][j][1]] for j in range(3)]
sp2 = [[subpath[i][j][0], subpath[i][j][1]] for j in range(3)]
c += biarc(sp1, sp2, 0, 0) if w == None else biarc(sp1, sp2, -f(w[k][i - 1]), -f(w[k][i]))
# l1 = biarc(sp1,sp2,0,0) if w==None else biarc(sp1,sp2,-f(w[k][i-1]),-f(w[k][i]))
# print_((-f(w[k][i-1]),-f(w[k][i]), [i1[5] for i1 in l1]) )
c += [[[subpath[-1][1][0], subpath[-1][1][1]], 'end', 0, 0]]
print_("Curve: " + str(c))
return c
def draw_curve(self, curve, layer, group=None, style=styles["biarc_style"]):
self.get_defs()
# Add marker to defs if it does not exist
if "DrawCurveMarker" not in self.defs:
defs = etree.SubElement(self.document.getroot(), inkex.addNS("defs", "svg"))
marker = etree.SubElement(defs, inkex.addNS("marker", "svg"),
{"id": "DrawCurveMarker", "orient": "auto", "refX": "-8",
"refY": "-2.41063", "style": "overflow:visible"})
etree.SubElement(marker, inkex.addNS("path", "svg"),
{
"d": "m -6.55552,-2.41063 0,0 L -13.11104,0 c 1.0473,-1.42323 1.04126,-3.37047 0,-4.82126",
"style": "fill:#000044; fill-rule:evenodd;stroke-width:0.62500000;stroke-linejoin:round;"}
)
if "DrawCurveMarker_r" not in self.defs:
defs = etree.SubElement(self.document.getroot(), inkex.addNS("defs", "svg"))
marker = etree.SubElement(defs, inkex.addNS("marker", "svg"),
{"id": "DrawCurveMarker_r", "orient": "auto", "refX": "8",
"refY": "-2.41063", "style": "overflow:visible"})
etree.SubElement(marker, inkex.addNS("path", "svg"),
{
"d": "m 6.55552,-2.41063 0,0 L 13.11104,0 c -1.0473,-1.42323 -1.04126,-3.37047 0,-4.82126",
"style": "fill:#000044; fill-rule:evenodd;stroke-width:0.62500000;stroke-linejoin:round;"}
)
for i in [0, 1]:
style['biarc%s_r' % i] = simplestyle.parseStyle(style['biarc%s' % i])
style['biarc%s_r' % i]["marker-start"] = "url(#DrawCurveMarker_r)"
del (style['biarc%s_r' % i]["marker-end"])
style['biarc%s_r' % i] = simplestyle.formatStyle(style['biarc%s_r' % i])
if group is None:
group = etree.SubElement(self.layers[min(1, len(self.layers) - 1)], inkex.addNS('g', 'svg'),
{"gcodetools": "Preview group"})
s, arcn = '', 0
a, b, c = [0., 0.], [1., 0.], [0., 1.]
k = (b[0] - a[0]) * (c[1] - a[1]) - (c[0] - a[0]) * (b[1] - a[1])
a, b, c = self.transform(a, layer, True), self.transform(b, layer, True), self.transform(c, layer, True)
if ((b[0] - a[0]) * (c[1] - a[1]) - (c[0] - a[0]) * (b[1] - a[1])) * k > 0:
reverse_angle = 1
else:
reverse_angle = -1
for sk in curve:
si = sk[:]
si[0], si[2] = self.transform(si[0], layer, True), (
self.transform(si[2], layer, True) if type(si[2]) == type([]) and len(si[2]) == 2 else si[2])
if s != '':
if s[1] == 'line':
etree.SubElement(group, inkex.addNS('path', 'svg'),
{
'style': style['line'],
'd': 'M %s,%s L %s,%s' % (s[0][0], s[0][1], si[0][0], si[0][1]),
"gcodetools": "Preview",
}
)
elif s[1] == 'arc':
arcn += 1
sp = s[0]
c = s[2]
s[3] = s[3] * reverse_angle
a = ((P(si[0]) - P(c)).angle() - (P(s[0]) - P(c)).angle()) % math.pi2 # s[3]
if s[3] * a < 0:
if a > 0:
a = a - math.pi2
else:
a = math.pi2 + a
r = math.sqrt((sp[0] - c[0]) ** 2 + (sp[1] - c[1]) ** 2)
a_st = (math.atan2(sp[0] - c[0], - (sp[1] - c[1])) - math.pi / 2) % (math.pi * 2)
st = style['biarc%s' % (arcn % 2)][:]
if a > 0:
a_end = a_st + a
st = style['biarc%s' % (arcn % 2)]
else:
a_end = a_st * 1
a_st = a_st + a
st = style['biarc%s_r' % (arcn % 2)]
etree.SubElement(group, inkex.addNS('path', 'svg'),
{
'style': st,
inkex.addNS('cx', 'sodipodi'): str(c[0]),
inkex.addNS('cy', 'sodipodi'): str(c[1]),
inkex.addNS('rx', 'sodipodi'): str(r),
inkex.addNS('ry', 'sodipodi'): str(r),
inkex.addNS('start', 'sodipodi'): str(a_st),
inkex.addNS('end', 'sodipodi'): str(a_end),
inkex.addNS('open', 'sodipodi'): 'true',
inkex.addNS('type', 'sodipodi'): 'arc',
"gcodetools": "Preview",
})
s = si
def check_dir(self):
if self.options.directory[-1] not in ["/", "\\"]:
if "\\" in self.options.directory:
self.options.directory += "\\"
else:
self.options.directory += "/"
print_("Checking direcrory: '%s'" % self.options.directory)
if (os.path.isdir(self.options.directory)):
if (os.path.isfile(self.options.directory + 'header')):
f = open(self.options.directory + 'header', 'r')
self.header = f.read()
f.close()
else:
self.header = defaults['header']
if (os.path.isfile(self.options.directory + 'footer')):
f = open(self.options.directory + 'footer', 'r')
self.footer = f.read()
f.close()
else:
self.footer = defaults['footer']
if self.options.unit == "G21 (All units in mm)":
self.header += "G21\n"
elif self.options.unit == "G20 (All units in inches)":
self.header += "G20\n"
else:
self.error(_("Directory does not exist! Please specify existing directory at options tab!"), "error")
return False
if self.options.add_numeric_suffix_to_filename:
dir_list = os.listdir(self.options.directory)
if "." in self.options.file:
r = re.match(r"^(.*)(\..*)$", self.options.file)
ext = r.group(2)
name = r.group(1)
else:
ext = ""
name = self.options.file
max_n = 0
for s in dir_list:
r = re.match(r"^%s_0*(\d+)%s$" % (re.escape(name), re.escape(ext)), s)
if r:
max_n = max(max_n, int(r.group(1)))
filename = name + "_" + ("0" * (4 - len(str(max_n + 1))) + str(max_n + 1)) + ext
self.options.file = filename
print_("Testing writing rights on '%s'" % (self.options.directory + self.options.file))
try:
f = open(self.options.directory + self.options.file, "w")
f.close()
except:
self.error(_("Can not write to specified file!\n%s" % (self.options.directory + self.options.file)),
"error")
return False
return True
################################################################################
#
# Generate Gcode
#
# Curve definition
# [start point, type = {'arc','line','move','end'}, arc center, arc angle, end point, [zstart, zend]]
#
################################################################################
def generate_gcode(self, curve, layer, depth):
tool = self.tools
print_("Tool in g-code generator: " + str(tool))
def c(c):
c = [c[i] if i < len(c) else None for i in range(6)]
if c[5] == 0: c[5] = None
s = [" X", " Y", " Z", " I", " J", " K"]
r = ''
for i in range(6):
if c[i] != None:
r += s[i] + ("%f" % (round(c[i], 4))).rstrip('0')
return r
if len(curve) == 0: return ""
try:
self.last_used_tool == None
except:
self.last_used_tool = None
print_("working on curve")
print_("Curve: " + str(curve))
g = ""
lg, f = 'G00', "F%f" % tool['penetration feed']
penetration_feed = "F%s" % tool['penetration feed']
current_a = 0
for i in range(1, len(curve)):
# Creating Gcode for curve between s=curve[i-1] and si=curve[i] start at s[0] end at s[4]=si[0]
s, si = curve[i - 1], curve[i]
feed = f if lg not in ['G01', 'G02', 'G03'] else ''
if s[1] == 'move':
g += "G1 " + c(si[0]) + "\n" + tool['gcode before path'] + "\n"
lg = 'G00'
elif s[1] == 'end':
g += tool['gcode after path'] + "\n"
lg = 'G00'
elif s[1] == 'line':
if lg == "G00": g += "G1 " + feed + "\n"
g += "G1 " + c(si[0]) + "\n"
lg = 'G01'
elif s[1] == 'arc':
r = [(s[2][0] - s[0][0]), (s[2][1] - s[0][1])]
if lg == "G00": g += "G1 " + feed + "\n"
if (r[0] ** 2 + r[1] ** 2) > .1:
r1, r2 = (P(s[0]) - P(s[2])), (P(si[0]) - P(s[2]))
if abs(r1.mag() - r2.mag()) < 0.001:
g += ("G2" if s[3] < 0 else "G3") + c(
si[0] + [None, (s[2][0] - s[0][0]), (s[2][1] - s[0][1])]) + "\n"
else:
r = (r1.mag() + r2.mag()) / 2
g += ("G2" if s[3] < 0 else "G3") + c(si[0]) + " R%f" % (r) + "\n"
lg = 'G02'
else:
g += "G1 " + c(si[0]) + " " + feed + "\n"
lg = 'G01'
if si[1] == 'end':
g += tool['gcode after path'] + "\n"
return g
def get_transforms(self, g):
root = self.document.getroot()
trans = []
while (g != root):
if 'transform' in list(g.keys()):
t = g.get('transform')
t = simpletransform.parseTransform(t)
trans = simpletransform.composeTransform(t, trans) if trans != [] else t
print_(trans)
g = g.getparent()
return trans
def apply_transforms(self, g, csp):
trans = self.get_transforms(g)
if trans != []:
simpletransform.applyTransformToPath(trans, csp)
return csp
def transform(self, source_point, layer, reverse=False):
if layer == None:
layer = self.current_layer if self.current_layer is not None else self.document.getroot()
if layer not in self.transform_matrix:
for i in range(self.layers.index(layer), -1, -1):
if self.layers[i] in self.orientation_points:
break
print_(str(self.layers))
print_(str("I: " + str(i)))
print_("Transform: " + str(self.layers[i]))
if self.layers[i] not in self.orientation_points:
self.error(_(
"Orientation points for '%s' layer have not been found! Please add orientation points using Orientation tab!") % layer.get(
inkex.addNS('label', 'inkscape')), "no_orientation_points")
elif self.layers[i] in self.transform_matrix:
self.transform_matrix[layer] = self.transform_matrix[self.layers[i]]
else:
orientation_layer = self.layers[i]
if len(self.orientation_points[orientation_layer]) > 1:
self.error(
_("There are more than one orientation point groups in '%s' layer") % orientation_layer.get(
inkex.addNS('label', 'inkscape')), "more_than_one_orientation_point_groups")
points = self.orientation_points[orientation_layer][0]
if len(points) == 2:
points += [[[(points[1][0][1] - points[0][0][1]) + points[0][0][0],
-(points[1][0][0] - points[0][0][0]) + points[0][0][1]],
[-(points[1][1][1] - points[0][1][1]) + points[0][1][0],
points[1][1][0] - points[0][1][0] + points[0][1][1]]]]
if len(points) == 3:
print_("Layer '%s' Orientation points: " % orientation_layer.get(inkex.addNS('label', 'inkscape')))
for point in points:
print_(point)
# Zcoordinates definition taken from Orientatnion point 1 and 2
self.Zcoordinates[layer] = [max(points[0][1][2], points[1][1][2]),
min(points[0][1][2], points[1][1][2])]
matrix = numpy.array([
[points[0][0][0], points[0][0][1], 1, 0, 0, 0, 0, 0, 0],
[0, 0, 0, points[0][0][0], points[0][0][1], 1, 0, 0, 0],
[0, 0, 0, 0, 0, 0, points[0][0][0], points[0][0][1], 1],
[points[1][0][0], points[1][0][1], 1, 0, 0, 0, 0, 0, 0],
[0, 0, 0, points[1][0][0], points[1][0][1], 1, 0, 0, 0],
[0, 0, 0, 0, 0, 0, points[1][0][0], points[1][0][1], 1],
[points[2][0][0], points[2][0][1], 1, 0, 0, 0, 0, 0, 0],
[0, 0, 0, points[2][0][0], points[2][0][1], 1, 0, 0, 0],
[0, 0, 0, 0, 0, 0, points[2][0][0], points[2][0][1], 1]
])
if numpy.linalg.det(matrix) != 0:
m = numpy.linalg.solve(matrix,
numpy.array(
[[points[0][1][0]], [points[0][1][1]], [1], [points[1][1][0]],
[points[1][1][1]], [1], [points[2][1][0]], [points[2][1][1]], [1]]
)
).tolist()
self.transform_matrix[layer] = [[m[j * 3 + i][0] for i in range(3)] for j in range(3)]
else:
self.error(_(
"Orientation points are wrong! (if there are two orientation points they sould not be the same. If there are three orientation points they should not be in a straight line.)"),
"wrong_orientation_points")
else:
self.error(_(
"Orientation points are wrong! (if there are two orientation points they sould not be the same. If there are three orientation points they should not be in a straight line.)"),
"wrong_orientation_points")
self.transform_matrix_reverse[layer] = numpy.linalg.inv(self.transform_matrix[layer]).tolist()
print_("\n Layer '%s' transformation matrixes:" % layer.get(inkex.addNS('label', 'inkscape')))
print_(self.transform_matrix)
print_(self.transform_matrix_reverse)
# Zautoscale is absolute
self.Zauto_scale[layer] = 1
print_("Z automatic scale = %s (computed according orientation points)" % self.Zauto_scale[layer])
x, y = source_point[0], source_point[1]
if not reverse:
t = self.transform_matrix[layer]
else:
t = self.transform_matrix_reverse[layer]
return [t[0][0] * x + t[0][1] * y + t[0][2], t[1][0] * x + t[1][1] * y + t[1][2]]
def transform_csp(self, csp_, layer, reverse=False):
csp = [[[csp_[i][j][0][:], csp_[i][j][1][:], csp_[i][j][2][:]] for j in range(len(csp_[i]))] for i in
range(len(csp_))]
for i in range(len(csp)):
for j in range(len(csp[i])):
for k in range(len(csp[i][j])):
csp[i][j][k] = self.transform(csp[i][j][k], layer, reverse)
return csp
################################################################################
# Errors handling function, notes are just printed into Logfile,
# warnings are printed into log file and warning message is displayed but
# extension continues working, errors causes log and execution is halted
# Notes, warnings adn errors could be assigned to space or comma or dot
# sepparated strings (case is ignoreg).
################################################################################
def error(self, s, type_="Warning"):
notes = "Note "
warnings = """
Warning tools_warning
bad_orientation_points_in_some_layers
more_than_one_orientation_point_groups
more_than_one_tool
orientation_have_not_been_defined
tool_have_not_been_defined
selection_does_not_contain_paths
selection_does_not_contain_paths_will_take_all
selection_is_empty_will_comupe_drawing
selection_contains_objects_that_are_not_paths
"""
errors = """
Error
wrong_orientation_points
area_tools_diameter_error
no_tool_error
active_layer_already_has_tool
active_layer_already_has_orientation_points
"""
if type_.lower() in re.split("[\s\n,\.]+", errors.lower()):
print_(s)
inkex.errormsg(s + "\n")
sys.exit()
elif type_.lower() in re.split("[\s\n,\.]+", warnings.lower()):
print_(s)
if not self.options.suppress_all_messages:
inkex.errormsg(s + "\n")
elif type_.lower() in re.split("[\s\n,\.]+", notes.lower()):
print_(s)
else:
print_(s)
inkex.errormsg(s)
sys.exit()
################################################################################
# Get defs from svg
################################################################################
def get_defs(self):
self.defs = {}
def recursive(g):
for i in g:
if i.tag == inkex.addNS("defs", "svg"):
for j in i:
self.defs[j.get("id")] = i
if i.tag == inkex.addNS("g", 'svg'):
recursive(i)
recursive(self.document.getroot())
################################################################################
#
# Get Gcodetools info from the svg
#
################################################################################
def get_info(self):
self.selected_paths = {}
self.paths = {}
self.orientation_points = {}
self.layers = [self.document.getroot()]
self.Zcoordinates = {}
self.transform_matrix = {}
self.transform_matrix_reverse = {}
self.Zauto_scale = {}
def recursive_search(g, layer, selected=False):
items = g.getchildren()
items.reverse()
for i in items:
if selected:
self.selected_hack[i.get("id")] = i
if i.tag == inkex.addNS("g", 'svg') and i.get(inkex.addNS('groupmode', 'inkscape')) == 'layer':
self.layers += [i]
recursive_search(i, i)
elif i.get('gcodetools') == "Gcodetools orientation group":
points = self.get_orientation_points(i)
if points != None:
self.orientation_points[layer] = self.orientation_points[layer] + [
points[:]] if layer in self.orientation_points else [points[:]]
print_("Found orientation points in '%s' layer: %s" % (
layer.get(inkex.addNS('label', 'inkscape')), points))
else:
self.error(_(
"Warning! Found bad orientation points in '%s' layer. Resulting Gcode could be corrupt!") % layer.get(
inkex.addNS('label', 'inkscape')), "bad_orientation_points_in_some_layers")
elif i.tag == inkex.addNS('path', 'svg'):
if "gcodetools" not in list(i.keys()):
self.paths[layer] = self.paths[layer] + [i] if layer in self.paths else [i]
if i.get("id") in self.selected_hack:
self.selected_paths[layer] = self.selected_paths[layer] + [
i] if layer in self.selected_paths else [i]
elif i.tag == inkex.addNS("g", 'svg'):
recursive_search(i, layer, (i.get("id") in self.selected_hack))
elif i.get("id") in self.selected_hack:
self.error(_(
"This extension works with Paths and Dynamic Offsets and groups of them only! All other objects will be ignored!\nSolution 1: press Path->Object to path or Shift+Ctrl+C.\nSolution 2: Path->Dynamic offset or Ctrl+J.\nSolution 3: export all contours to PostScript level 2 (File->Save As->.ps) and File->Import this file."),
"selection_contains_objects_that_are_not_paths")
recursive_search(self.document.getroot(), self.document.getroot())
def get_orientation_points(self, g):
items = g.getchildren()
items.reverse()
p2, p3 = [], []
p = None
for i in items:
if i.tag == inkex.addNS("g", 'svg') and i.get("gcodetools") == "Gcodetools orientation point (2 points)":
p2 += [i]
if i.tag == inkex.addNS("g", 'svg') and i.get("gcodetools") == "Gcodetools orientation point (3 points)":
p3 += [i]
if len(p2) == 2:
p = p2
elif len(p3) == 3:
p = p3
if p == None: return None
points = []
for i in p:
point = [[], []]
for node in i:
if node.get('gcodetools') == "Gcodetools orientation point arrow":
point[0] = self.apply_transforms(node, parsePath(node.get("d")))[0][0][1]
if node.get('gcodetools') == "Gcodetools orientation point text":
r = re.match(
r'(?i)\s*\(\s*(-?\s*\d*(?:,|\.)*\d*)\s*;\s*(-?\s*\d*(?:,|\.)*\d*)\s*;\s*(-?\s*\d*(?:,|\.)*\d*)\s*\)\s*',
node.text)
point[1] = [float(r.group(1)), float(r.group(2)), float(r.group(3))]
if point[0] != [] and point[1] != []: points += [point]
if len(points) == len(p2) == 2 or len(points) == len(p3) == 3:
return points
else:
return None
################################################################################
#
# dxfpoints
#
################################################################################
def dxfpoints(self):
if self.selected_paths == {}:
self.error(_(
"Noting is selected. Please select something to convert to drill point (dxfpoint) or clear point sign."),
"warning")
for layer in self.layers:
if layer in self.selected_paths:
for path in self.selected_paths[layer]:
if self.options.dxfpoints_action == 'replace':
path.set("dxfpoint", "1")
r = re.match("^\s*.\s*(\S+)", path.get("d"))
if r != None:
print_(("got path=", r.group(1)))
path.set("d",
"m %s 2.9375,-6.343750000001 0.8125,1.90625 6.843748640396,-6.84374864039 0,0 0.6875,0.6875 -6.84375,6.84375 1.90625,0.812500000001 z" % r.group(
1))
path.set("style", styles["dxf_points"])
if self.options.dxfpoints_action == 'save':
path.set("dxfpoint", "1")
if self.options.dxfpoints_action == 'clear' and path.get("dxfpoint") == "1":
path.set("dxfpoint", "0")
################################################################################
#
# Laser
#
################################################################################
def laser(self):
def get_boundaries(points):
minx, miny, maxx, maxy = None, None, None, None
out = [[], [], [], []]
for p in points:
if minx == p[0]:
out[0] += [p]
if minx == None or p[0] < minx:
minx = p[0]
out[0] = [p]
if miny == p[1]:
out[1] += [p]
if miny == None or p[1] < miny:
miny = p[1]
out[1] = [p]
if maxx == p[0]:
out[2] += [p]
if maxx == None or p[0] > maxx:
maxx = p[0]
out[2] = [p]
if maxy == p[1]:
out[3] += [p]
if maxy == None or p[1] > maxy:
maxy = p[1]
out[3] = [p]
return out
def remove_duplicates(points):
i = 0
out = []
for p in points:
for j in range(i, len(points)):
if p == points[j]: points[j] = [None, None]
if p != [None, None]: out += [p]
i += 1
return (out)
def get_way_len(points):
l = 0
for i in range(1, len(points)):
l += math.sqrt((points[i][0] - points[i - 1][0]) ** 2 + (points[i][1] - points[i - 1][1]) ** 2)
return l
def sort_dxfpoints(points):
points = remove_duplicates(points)
ways = [
# l=0, d=1, r=2, u=3
[3, 0], # ul
[3, 2], # ur
[1, 0], # dl
[1, 2], # dr
[0, 3], # lu
[0, 1], # ld
[2, 3], # ru
[2, 1], # rd
]
minimal_way = []
minimal_len = None
minimal_way_type = None
for w in ways:
tpoints = points[:]
cw = []
for j in range(0, len(points)):
p = get_boundaries(get_boundaries(tpoints)[w[0]])[w[1]]
tpoints.remove(p[0])
cw += p
curlen = get_way_len(cw)
if minimal_len == None or curlen < minimal_len:
minimal_len = curlen
minimal_way = cw
minimal_way_type = w
return minimal_way
if self.selected_paths == {}:
paths = self.paths
self.error(_("No paths are selected! Trying to work on all available paths."), "warning")
else:
paths = self.selected_paths
self.check_dir()
gcode = ""
biarc_group = etree.SubElement(
list(self.selected_paths.keys())[0] if len(list(self.selected_paths.keys())) > 0 else self.layers[0],
inkex.addNS('g', 'svg'))
print_(("self.layers=", self.layers))
print_(("paths=", paths))
for layer in self.layers:
if layer in paths:
print_(("layer", layer))
p = []
dxfpoints = []
for path in paths[layer]:
print_(str(layer))
if "d" not in list(path.keys()):
self.error(_(
"Warning: One or more paths dont have 'd' parameter, try to Ungroup (Ctrl+Shift+G) and Object to Path (Ctrl+Shift+C)!"),
"selection_contains_objects_that_are_not_paths")
continue
csp = parsePath(path.get("d"))
csp = self.apply_transforms(path, csp)
if path.get("dxfpoint") == "1":
tmp_curve = self.transform_csp(csp, layer)
x = tmp_curve[0][0][0][0]
y = tmp_curve[0][0][0][1]
print_("got dxfpoint (scaled) at (%f,%f)" % (x, y))
dxfpoints += [[x, y]]
else:
p += csp
dxfpoints = sort_dxfpoints(dxfpoints)
curve = self.parse_curve(p, layer)
self.draw_curve(curve, layer, biarc_group)
gcode += self.generate_gcode(curve, layer, 0)
self.export_gcode(gcode)
################################################################################
#
# Orientation
#
################################################################################
def orientation(self, layer=None):
print_("entering orientations")
if layer == None:
layer = self.current_layer if self.current_layer is not None else self.document.getroot()
if layer in self.orientation_points:
self.error(_("Active layer already has orientation points! Remove them or select another layer!"),
"active_layer_already_has_orientation_points")
orientation_group = etree.SubElement(layer, inkex.addNS('g', 'svg'),
{"gcodetools": "Gcodetools orientation group"})
# translate == ['0', '-917.7043']
if layer.get("transform") != None:
translate = layer.get("transform").replace("translate(", "").replace(")", "").split(",")
else:
translate = [0, 0]
# doc height in pixels (38 mm == 143.62204724px)
doc_height = self.svg.unittouu(self.document.getroot().xpath('@height', namespaces=inkex.NSS)[0])
if self.document.getroot().get('height') == "100%":
doc_height = 1052.3622047
print_("Overriding height from 100 percents to %s" % doc_height)
print_("Document height: " + str(doc_height));
if self.options.unit == "G21 (All units in mm)":
points = [[0., 0., 0.], [100., 0., 0.], [0., 100., 0.]]
orientation_scale = 1
print_("orientation_scale < 0 ===> switching to mm units=%0.10f" % orientation_scale)
elif self.options.unit == "G20 (All units in inches)":
points = [[0., 0., 0.], [5., 0., 0.], [0., 5., 0.]]
orientation_scale = 90
print_("orientation_scale < 0 ===> switching to inches units=%0.10f" % orientation_scale)
points = points[:2]
print_(("using orientation scale", orientation_scale, "i=", points))
for i in points:
# X == Correct!
# si == x,y coordinate in px
# si have correct coordinates
# if layer have any transform it will be in translate so lets add that
si = [i[0] * orientation_scale, (i[1] * orientation_scale) + float(translate[1])]
g = etree.SubElement(orientation_group, inkex.addNS('g', 'svg'),
{'gcodetools': "Gcodetools orientation point (2 points)"})
etree.SubElement(g, inkex.addNS('path', 'svg'),
{
'style': "stroke:none;fill:#000000;",
'd': 'm %s,%s 2.9375,-6.343750000001 0.8125,1.90625 6.843748640396,-6.84374864039 0,0 0.6875,0.6875 -6.84375,6.84375 1.90625,0.812500000001 z z' % (
si[0], -si[1] + doc_height),
'gcodetools': "Gcodetools orientation point arrow"
})
t = etree.SubElement(g, inkex.addNS('text', 'svg'),
{
'style': "font-size:10px;font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;fill:#000000;fill-opacity:1;stroke:none;",
inkex.addNS("space", "xml"): "preserve",
'x': str(si[0] + 10),
'y': str(-si[1] - 10 + doc_height),
'gcodetools': "Gcodetools orientation point text"
})
t.text = "(%s; %s; %s)" % (i[0], i[1], i[2])
################################################################################
#
# Effect
#
# Main function of Gcodetools class
#
################################################################################
def effect(self):
global options
options = self.options
options.self = self
options.doc_root = self.document.getroot()
# define print_ function
global print_
if self.options.log_create_log:
try:
if os.path.isfile(self.options.log_filename): os.remove(self.options.log_filename)
f = open(self.options.log_filename, "a")
f.write("Gcodetools log file.\nStarted at %s.\n%s\n" % (
time.strftime("%d.%m.%Y %H:%M:%S"), options.log_filename))
f.write("%s tab is active.\n" % self.options.active_tab)
f.close()
except:
print_ = lambda *x: None
else:
print_ = lambda *x: None
self.get_info()
if self.orientation_points == {}:
self.error(_(
"Orientation points have not been defined! A default set of orientation points has been automatically added."),
"warning")
self.orientation(self.layers[min(0, len(self.layers) - 1)])
self.get_info()
self.tools = {
"name": "Laser Engraver",
"id": "Laser Engraver",
"penetration feed": self.options.laser_speed,
"feed": self.options.laser_speed,
"gcode before path": ("G4 P0 \n" + self.options.laser_command + " S" + str(
int(self.options.laser_power)) + "\nG4 P" + self.options.power_delay),
"gcode after path": (
"G4 P0 \n" + self.options.laser_off_command + " S0" + "\n" + "G1 F" + self.options.travel_speed),
}
self.get_info()
self.laser()
e = LaserGcode()
if target_version < 1.0:
e.affect()
else:
e.run()