updated laser check
This commit is contained in:
parent
b7ac64c308
commit
cc30a8f9a4
@ -2,8 +2,67 @@
|
||||
<inkscape-extension xmlns="http://www.inkscape.org/namespace/inkscape/extension">
|
||||
<name>Laser Check</name>
|
||||
<id>fablabchemnitz.de.laser_check</id>
|
||||
<!--<param name="depth" type="int" gui-text="Max nested group depth">1</param>-->
|
||||
<effect>
|
||||
<param name="tab" type="notebook">
|
||||
<page name="tab_settings" gui-text="Checks">
|
||||
<param name="checks" type="optiongroup" appearance="combo" gui-text="Select checks">
|
||||
<option value="check_all">Check all</option>
|
||||
<option value="check_section">Check selection</option>
|
||||
</param>
|
||||
<separator/>
|
||||
<hbox>
|
||||
<vbox>
|
||||
<param name="bbox" type="bool" gui-text="Bounding box">false</param>
|
||||
<param name="bbox_offset" type="float" min="0.000" max="9999.000" precision="3" gui-text=" > Minimum required offset (mm)">5.000</param>
|
||||
<param name="groups_and_layers" type="bool" gui-text="Groups and layers">false</param>
|
||||
<param name="clones" type="bool" gui-text="Clones">false</param>
|
||||
<param name="clippaths" type="bool" gui-text="Clippings">false</param>
|
||||
<param name="images" type="bool" gui-text="Images">false</param>
|
||||
<param name="texts" type="bool" gui-text="Texts">false</param>
|
||||
<param name="lowlevelstrokes" type="bool" gui-text="Low level strokes">false</param>
|
||||
<param name="stroke_colors" type="bool" gui-text="Stroke colors">false</param>
|
||||
</vbox>
|
||||
<separator/>
|
||||
<vbox>
|
||||
<param name="stroke_widths" type="bool" gui-text="Stroke widths">false</param>
|
||||
<param name="stroke_opacities" type="bool" gui-text="Stroke opacities">false</param>
|
||||
<param name="cosmestic_dashes" type="bool" gui-text="Cosmetic dash styles">false</param>
|
||||
<param name="invisible_shapes" type="bool" gui-text="Invisible shapes">false</param>
|
||||
<param name="pointy_paths" type="bool" gui-text="Pointy paths">false</param>
|
||||
<param name="transformations" type="bool" gui-text="Transformations">false</param>
|
||||
<param name="short_paths" type="bool" gui-text="Short paths">false</param>
|
||||
<param name="short_paths_min" type="float" min="0.000" max="9999.000" precision="3" gui-text=" > Check below length (mm)">1.000</param>
|
||||
<param name="non_path_shapes" type="bool" gui-text="Non-path shapes">false</param>
|
||||
</vbox>
|
||||
</hbox>
|
||||
</page>
|
||||
<page name="tab_about" gui-text="About">
|
||||
<label appearance="header">Laser Check</label>
|
||||
<label>A utility to check for common issue potencials in vector graphics, especially regarding to laser cutting jobs.</label>
|
||||
<label>2021 / written by Mario Voigt (Stadtfabrikanten e.V. / FabLab Chemnitz)</label>
|
||||
<spacer/>
|
||||
<label appearance="header">Online Documentation</label>
|
||||
<label appearance="url">https://y.stadtfabrikanten.org/lasercheck</label>
|
||||
<spacer/>
|
||||
<label appearance="header">Contributing</label>
|
||||
<label appearance="url">https://gitea.fablabchemnitz.de/FabLab_Chemnitz/mightyscape-1.X</label>
|
||||
<label appearance="url">mailto:mario.voigt@stadtfabrikanten.org</label>
|
||||
<spacer/>
|
||||
<label appearance="header">MightyScape Extension Collection</label>
|
||||
<label>This piece of software is part of the MightyScape for Inkscape Extension Collection and is licensed under GNU GPL v3</label>
|
||||
<label appearance="url">https://y.stadtfabrikanten.org/mightyscape-overview</label>
|
||||
</page>
|
||||
<page name="tab_donate" gui-text="Donate">
|
||||
<label appearance="header">Coffee + Pizza</label>
|
||||
<label>We are the Stadtfabrikanten, running the FabLab Chemnitz since 2016. A FabLab is an open workshop that gives people access to machines and digital tools like 3D printers, laser cutters and CNC milling machines.</label>
|
||||
<spacer/>
|
||||
<label>You like our work and want to support us? You can donate to our non-profit organization by different ways:</label>
|
||||
<label appearance="url">https://y.stadtfabrikanten.org/donate</label>
|
||||
<spacer/>
|
||||
<label>Thanks for using our extension and helping us!</label>
|
||||
<image>../000_about_fablabchemnitz.svg</image>
|
||||
</page>
|
||||
</param>
|
||||
<effect needs-live-preview="false">
|
||||
<object-type>all</object-type>
|
||||
<effects-menu>
|
||||
<submenu name="FabLab Chemnitz">
|
||||
|
@ -12,11 +12,32 @@ class LaserCheck(inkex.EffectExtension):
|
||||
check for old styles which should be upgraded
|
||||
'''
|
||||
|
||||
#def add_arguments(self, pars):
|
||||
# pars.add_argument('-d', '--depth', type = int, default = 1, help = 'Convert nested group up to DEPTH layers deep')
|
||||
def add_arguments(self, pars):
|
||||
pars.add_argument('--tab')
|
||||
pars.add_argument('--checks', default="check_all")
|
||||
pars.add_argument('--bbox', type=inkex.Boolean, default=False)
|
||||
pars.add_argument('--bbox_offset', type=float, default=5.000)
|
||||
pars.add_argument('--groups_and_layers', type=inkex.Boolean, default=False)
|
||||
pars.add_argument('--clones', type=inkex.Boolean, default=False)
|
||||
pars.add_argument('--clippaths', type=inkex.Boolean, default=False)
|
||||
pars.add_argument('--images', type=inkex.Boolean, default=False)
|
||||
pars.add_argument('--texts', type=inkex.Boolean, default=False)
|
||||
pars.add_argument('--lowlevelstrokes', type=inkex.Boolean, default=False)
|
||||
pars.add_argument('--stroke_colors', type=inkex.Boolean, default=False)
|
||||
pars.add_argument('--stroke_widths', type=inkex.Boolean, default=False)
|
||||
pars.add_argument('--stroke_opacities', type=inkex.Boolean, default=False)
|
||||
pars.add_argument('--cosmestic_dashes', type=inkex.Boolean, default=False)
|
||||
pars.add_argument('--invisible_shapes', type=inkex.Boolean, default=False)
|
||||
pars.add_argument('--pointy_paths', type=inkex.Boolean, default=False)
|
||||
pars.add_argument('--transformations', type=inkex.Boolean, default=False)
|
||||
pars.add_argument('--short_paths', type=inkex.Boolean, default=False)
|
||||
pars.add_argument('--short_paths_min', type=float, default=1.000)
|
||||
pars.add_argument('--non_path_shapes', type=inkex.Boolean, default=False)
|
||||
|
||||
def effect(self):
|
||||
|
||||
so = self.options
|
||||
|
||||
def parseChildren(element):
|
||||
if element not in selected:
|
||||
selected.append(element)
|
||||
@ -37,14 +58,25 @@ class LaserCheck(inkex.EffectExtension):
|
||||
else:
|
||||
for element in self.svg.selected.values():
|
||||
parseChildren(element)
|
||||
|
||||
|
||||
namedView = self.document.getroot().find(inkex.addNS('namedview', 'sodipodi'))
|
||||
doc_units = namedView.get(inkex.addNS('document-units', 'inkscape'))
|
||||
pagecolor = namedView.get('pagecolor')
|
||||
inkex.utils.debug("---------- Default checks")
|
||||
inkex.utils.debug("Document units: {}".format(doc_units))
|
||||
nonShapes = []
|
||||
shapes = []
|
||||
for element in selected:
|
||||
if not isinstance(element, inkex.ShapeElement):
|
||||
nonShapes.append(element)
|
||||
else:
|
||||
shapes.append(element)
|
||||
inkex.utils.debug("{} shape elements in total".format(len(shapes)))
|
||||
inkex.utils.debug("{} non-shape elements in total".format(len(nonShapes)))
|
||||
for nonShape in nonShapes:
|
||||
inkex.utils.debug("non-shape id={}".format(nonShape.get('id')))
|
||||
|
||||
inkex.utils.debug("---------- Groups and layers")
|
||||
if so.checks == "check_all" or so.groups_and_layers is True:
|
||||
inkex.utils.debug("\n---------- Groups and layers")
|
||||
groups = []
|
||||
layers = []
|
||||
for element in selected:
|
||||
@ -59,33 +91,38 @@ class LaserCheck(inkex.EffectExtension):
|
||||
#check for empty groups
|
||||
for group in groups:
|
||||
if len(group) == 0:
|
||||
inkex.utils.debug("group id={} is empty".format(group.get('id')))
|
||||
inkex.utils.debug("id={} is empty group".format(group.get('id')))
|
||||
|
||||
#check for empty layers
|
||||
for layer in layers:
|
||||
if len(layer) == 0:
|
||||
inkex.utils.debug("layer id={} is empty".format(layer.get('id')))
|
||||
inkex.utils.debug("id={} is empty layer".format(layer.get('id')))
|
||||
|
||||
inkex.utils.debug("---------- Clones (svg:use) - maybe unlink")
|
||||
|
||||
if so.checks == "check_all" or so.clones is True:
|
||||
inkex.utils.debug("\n---------- Clones (svg:use) - maybe unlink")
|
||||
uses = []
|
||||
for element in selected:
|
||||
if element.tag == inkex.addNS('use','svg'):
|
||||
uses.append(element)
|
||||
inkex.utils.debug("{} svg:use clones in total".format(len(uses)))
|
||||
for use in uses:
|
||||
inkex.utils.debug("clone id={}".format(use.get('id')))
|
||||
inkex.utils.debug("id={}".format(use.get('id')))
|
||||
|
||||
inkex.utils.debug("---------- Clippings (svg:clipPath) - please replace with real cut paths")
|
||||
|
||||
if so.checks == "check_all" or so.clippaths is True:
|
||||
inkex.utils.debug("\n---------- Clippings (svg:clipPath) - please replace with real cut paths")
|
||||
clipPaths = []
|
||||
for element in selected:
|
||||
if element.tag == inkex.addNS('clipPath','svg'):
|
||||
clipPaths.append(element)
|
||||
inkex.utils.debug("{} svg:clipPath in total".format(len(clipPaths)))
|
||||
for clipPath in clipPaths:
|
||||
inkex.utils.debug("clipPath id={}".format(clipPath.get('id')))
|
||||
inkex.utils.debug("id={}".format(clipPath.get('id')))
|
||||
|
||||
|
||||
inkex.utils.debug("---------- Images (svg:image) - maybe trace to svg")
|
||||
if so.checks == "check_all" or so.images is True:
|
||||
inkex.utils.debug("\n---------- Images (svg:image) - maybe trace to svg")
|
||||
images = []
|
||||
for element in selected:
|
||||
if element.tag == inkex.addNS('image','svg'):
|
||||
@ -94,39 +131,31 @@ class LaserCheck(inkex.EffectExtension):
|
||||
for image in images:
|
||||
inkex.utils.debug("image id={}".format(image.get('id')))
|
||||
|
||||
inkex.utils.debug("---------- Low level strokes (svg:line/polyline/polygon) - maybe convert to path")
|
||||
|
||||
if so.checks == "check_all" or so.lowlevelstrokes is True:
|
||||
inkex.utils.debug("\n---------- Low level strokes (svg:line/polyline/polygon) - maybe convert to path")
|
||||
lowlevels = []
|
||||
for element in selected:
|
||||
if element.tag in (inkex.addNS('line','svg'), inkex.addNS('polyline','svg'), inkex.addNS('polygon','svg')):
|
||||
lowlevels.append(element)
|
||||
inkex.utils.debug("{} low level strokes in total".format(len(lowlevels)))
|
||||
for lowlevel in lowlevels:
|
||||
inkex.utils.debug("stroke id={}".format(lowlevel.get('id')))
|
||||
inkex.utils.debug("id={}".format(lowlevel.get('id')))
|
||||
|
||||
inkex.utils.debug("---------- Texts (should be converted to paths)")
|
||||
|
||||
if so.checks == "check_all" or so.texts is True:
|
||||
inkex.utils.debug("\n---------- Texts (should be converted to paths)")
|
||||
texts = []
|
||||
for element in selected:
|
||||
if element.tag == inkex.addNS('text','svg'):
|
||||
texts.append(element)
|
||||
inkex.utils.debug("{} svg:text in total".format(len(texts)))
|
||||
for text in texts:
|
||||
inkex.utils.debug("text id={}".format(text.get('id')))
|
||||
inkex.utils.debug("id={}".format(text.get('id')))
|
||||
|
||||
|
||||
inkex.utils.debug("---------- Shapes / Non-shapes")
|
||||
nonShapes = []
|
||||
shapes = []
|
||||
for element in selected:
|
||||
if not isinstance(element, inkex.ShapeElement):
|
||||
nonShapes.append(element)
|
||||
else:
|
||||
shapes.append(element)
|
||||
inkex.utils.debug("{} shape elements in total".format(len(shapes)))
|
||||
inkex.utils.debug("{} non-shape elements in total".format(len(nonShapes)))
|
||||
for nonShape in nonShapes:
|
||||
inkex.utils.debug("non-shape id={}".format(nonShape.get('id')))
|
||||
|
||||
inkex.utils.debug("---------- Stroke colors")
|
||||
if so.checks == "check_all" or so.stroke_colors is True:
|
||||
inkex.utils.debug("\n---------- Stroke colors")
|
||||
strokeColors = []
|
||||
for element in shapes:
|
||||
style = element.get('style')
|
||||
@ -140,7 +169,9 @@ class LaserCheck(inkex.EffectExtension):
|
||||
for strokeColor in strokeColors:
|
||||
inkex.utils.debug("stroke color {}".format(strokeColor))
|
||||
|
||||
inkex.utils.debug("---------- Borders around all elements - minimum offset 5mm from each side")
|
||||
|
||||
if so.checks == "check_all" or so.bbox is True:
|
||||
inkex.utils.debug("\n---------- Borders around all elements - minimum offset {} mm from each side".format(so.bbox_offset))
|
||||
bbox = inkex.BoundingBox()
|
||||
for element in self.document.getroot().iter(tag=etree.Element):
|
||||
if element != self.document.getroot() and isinstance(element, inkex.ShapeElement) and element.tag != inkex.addNS('use','svg') and element.get('inkscape:groupmode') != 'layer': #bbox fails for svg:use elements and layers
|
||||
@ -158,32 +189,34 @@ class LaserCheck(inkex.EffectExtension):
|
||||
|
||||
if abs(bbox.width) == math.inf or abs(bbox.height) == math.inf:
|
||||
inkex.utils.debug("bounding box could not be calculated")
|
||||
else:
|
||||
inkex.utils.debug("bounding box is {}".format(bbox))
|
||||
#else:
|
||||
# inkex.utils.debug("bounding box is {}".format(bbox))
|
||||
page_width = self.svg.unittouu(self.document.getroot().attrib['width'])
|
||||
width_height = self.svg.unittouu(self.document.getroot().attrib['height'])
|
||||
fmm = self.svg.unittouu("5mm")
|
||||
fmm = self.svg.unittouu(str(so.bbox_offset) + "mm")
|
||||
if bbox.left > fmm:
|
||||
self.msg("left ok")
|
||||
inkex.utils.debug("left border... ok")
|
||||
else:
|
||||
self.msg("left fail")
|
||||
inkex.utils.debug("left border... fail")
|
||||
|
||||
if bbox.top > fmm:
|
||||
self.msg("top ok")
|
||||
inkex.utils.debug("top border... ok")
|
||||
else:
|
||||
self.msg("top fail")
|
||||
inkex.utils.debug("top border... fail")
|
||||
|
||||
if bbox.right + fmm <= page_width:
|
||||
self.msg("right ok")
|
||||
inkex.utils.debug("right border... ok")
|
||||
else:
|
||||
self.msg("right fail")
|
||||
inkex.utils.debug("right border... fail")
|
||||
|
||||
if bbox.bottom + fmm <= width_height:
|
||||
self.msg("bottom ok")
|
||||
inkex.utils.debug("bottom border... ok")
|
||||
else:
|
||||
self.msg("bottom fail")
|
||||
inkex.utils.debug("bottom border... fail")
|
||||
|
||||
inkex.utils.debug("---------- Stroke widths")
|
||||
|
||||
if so.checks == "check_all" or so.stroke_widths is True:
|
||||
inkex.utils.debug("\n---------- Stroke widths")
|
||||
strokeWidths = []
|
||||
for element in shapes:
|
||||
style = element.get('style')
|
||||
@ -197,7 +230,9 @@ class LaserCheck(inkex.EffectExtension):
|
||||
for strokeWidth in strokeWidths:
|
||||
inkex.utils.debug("stroke width {}".format(strokeWidth))
|
||||
|
||||
inkex.utils.debug("---------- Cosmetic dashes - should be converted to paths")
|
||||
|
||||
if so.checks == "check_all" or so.cosmestic_dashes is True:
|
||||
inkex.utils.debug("\n---------- Cosmetic dashes - should be converted to paths")
|
||||
strokeDasharrays = []
|
||||
for element in shapes:
|
||||
style = element.get('style')
|
||||
@ -211,7 +246,9 @@ class LaserCheck(inkex.EffectExtension):
|
||||
for strokeDasharray in strokeDasharrays:
|
||||
inkex.utils.debug("stroke dash array {}".format(strokeDasharray))
|
||||
|
||||
inkex.utils.debug("---------- Invisible shapes")
|
||||
|
||||
if so.checks == "check_all" or so.invisible_shapes is True:
|
||||
inkex.utils.debug("\n---------- Invisible shapes")
|
||||
invisibles = []
|
||||
for element in shapes:
|
||||
if element.tag not in (inkex.addNS('tspan','svg')):
|
||||
@ -265,15 +302,17 @@ class LaserCheck(inkex.EffectExtension):
|
||||
else:
|
||||
fillOpacityVis = 1
|
||||
|
||||
#self.msg("strokeVis={}, widthVis={}, strokeOpacityVis={}, fillVis={}, fillOpacityVis={}".format(strokeVis, widthVis, strokeOpacityVis, fillVis, fillOpacityVis))
|
||||
#inkex.utils.debug("strokeVis={}, widthVis={}, strokeOpacityVis={}, fillVis={}, fillOpacityVis={}".format(strokeVis, widthVis, strokeOpacityVis, fillVis, fillOpacityVis))
|
||||
if (strokeVis == 0 or widthVis == 0 or strokeOpacityVis == 0) and (fillVis == 0 or fillOpacityVis == 0):
|
||||
if element not in invisibles:
|
||||
invisibles.append(element)
|
||||
inkex.utils.debug("{} invisible shapes in total".format(len(invisibles)))
|
||||
for invisible in invisibles:
|
||||
inkex.utils.debug("shape id={}".format(invisible.get('id')))
|
||||
inkex.utils.debug("id={}".format(invisible.get('id')))
|
||||
|
||||
inkex.utils.debug("---------- Objects with stroke transparencies (< 1.0) - should be set to 1.0")
|
||||
|
||||
if so.checks == "check_all" or so.stroke_opacities is True:
|
||||
inkex.utils.debug("\n---------- Objects with stroke transparencies < 1.0 - should be set to 1.0")
|
||||
transparencies = []
|
||||
for element in shapes:
|
||||
style = element.get('style')
|
||||
@ -283,11 +322,13 @@ class LaserCheck(inkex.EffectExtension):
|
||||
if float(stroke_opacity[0].split("stroke-opacity:")[1].split(";")[0]) < 1.0:
|
||||
if element not in transparencies:
|
||||
transparencies.append(element)
|
||||
inkex.utils.debug("{} stroke transparencies shapes in total".format(len(transparencies)))
|
||||
inkex.utils.debug("{} objects with stroke transparencies < 1.0 in total".format(len(transparencies)))
|
||||
for transparency in transparencies:
|
||||
inkex.utils.debug("shape id={}".format(transparency.get('id')))
|
||||
inkex.utils.debug("id={}".format(transparency.get('id')))
|
||||
|
||||
inkex.utils.debug("---------- Pointy paths - should be deleted")
|
||||
|
||||
if so.checks == "check_all" or so.pointy_paths is True:
|
||||
inkex.utils.debug("\n---------- Pointy paths - should be deleted")
|
||||
pointyPaths = []
|
||||
for element in shapes:
|
||||
if isinstance(element, inkex.PathElement):
|
||||
@ -300,9 +341,11 @@ class LaserCheck(inkex.EffectExtension):
|
||||
pointyPaths.append(element)
|
||||
inkex.utils.debug("{} pointy paths in total".format(len(pointyPaths)))
|
||||
for pointyPath in pointyPaths:
|
||||
inkex.utils.debug("path id={}".format(pointyPath.get('id')))
|
||||
inkex.utils.debug("id={}".format(pointyPath.get('id')))
|
||||
|
||||
inkex.utils.debug("---------- Transformations - should be applied to absolute")
|
||||
|
||||
if so.checks == "check_all" or so.transformations is True:
|
||||
inkex.utils.debug("\n---------- Transformations - should be applied to absolute")
|
||||
transformations = []
|
||||
for element in shapes:
|
||||
if element.get('transform') is not None:
|
||||
@ -311,16 +354,36 @@ class LaserCheck(inkex.EffectExtension):
|
||||
for transformation in transformations:
|
||||
inkex.utils.debug("transformation in id={}".format(transformation.get('id')))
|
||||
|
||||
inkex.utils.debug("---------- Short paths (< 1mm)")
|
||||
|
||||
if so.checks == "check_all" or so.short_paths is True:
|
||||
inkex.utils.debug("\n---------- Short paths (< {} mm)".format(so.short_paths_min))
|
||||
shortPaths = []
|
||||
totalLength = 0
|
||||
totalDropLength = 0
|
||||
for element in shapes:
|
||||
if isinstance(element, inkex.PathElement):
|
||||
slengths, stotal = csplength(element.path.transform(element.composed_transform()).to_superpath())
|
||||
if stotal < self.svg.unittouu("1mm"):
|
||||
totalLength += stotal
|
||||
if stotal < self.svg.unittouu(str(so.short_paths_min) + "mm"):
|
||||
shortPaths.append(element)
|
||||
totalDropLength += stotal
|
||||
inkex.utils.debug("{} short paths in total".format(len(shortPaths)))
|
||||
if totalLength > 0:
|
||||
inkex.utils.debug("{:0.2f}% of total ({:0.2f} mm /{:0.2f} mm)".format(totalDropLength / totalLength, totalDropLength, totalLength))
|
||||
for shortPath in shortPaths:
|
||||
inkex.utils.debug("short path id={}".format(shortPath.get('id')))
|
||||
inkex.utils.debug("id={}".format(shortPath.get('id')))
|
||||
|
||||
|
||||
if so.checks == "check_all" or so.non_path_shapes is True:
|
||||
inkex.utils.debug("\n---------- Non-path shapes - should be converted to paths")
|
||||
nonPathShapes = []
|
||||
for element in shapes:
|
||||
if not isinstance(element, inkex.PathElement) and not isinstance(element, inkex.Group):
|
||||
nonPathShapes.append(element)
|
||||
inkex.utils.debug("{} non-path shapes in total".format(len(nonPathShapes)))
|
||||
for nonPathShape in nonPathShapes:
|
||||
inkex.utils.debug("id={}".format(nonPathShape.get('id')))
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
LaserCheck().run()
|
Reference in New Issue
Block a user