2021-10-26 20:37:24 +02:00
#!/usr/bin/env python3
import sys
import os
import inkex
import tempfile
import subprocess
from subprocess import Popen , PIPE
from lxml import etree
from inkex import Transform
class PlyCutter ( inkex . EffectExtension ) :
def add_arguments ( self , pars ) :
pars . add_argument ( " --tab " )
pars . add_argument ( " --infile " )
pars . add_argument ( " --resizetoimport " , type = inkex . Boolean , default = True , help = " Resize the canvas to the imported drawing ' s bounding box " )
pars . add_argument ( " --extraborder " , type = float , default = 0.0 )
pars . add_argument ( " --extraborder_units " , default = " mm " )
pars . add_argument ( " --thickness " , type = float , default = 6.000 , help = " Set the thickness of sheets to find. " )
pars . add_argument ( " --debug " , type = inkex . Boolean , default = False , help = " Turn on debugging " )
pars . add_argument ( " --min_finger_width " , type = float , default = 3.000 , help = " Set minimum width for generated fingers. " )
pars . add_argument ( " --max_finger_width " , type = float , default = 5.000 , help = " Set maximum width for generated fingers. " )
pars . add_argument ( " --support_radius " , type = float , default = 12.000 , help = " Set maximum range for generating material on a sheet where neither surface is visible " )
pars . add_argument ( " --final_dilation " , default = 0.05 , type = float , help = " Final dilation (laser cutter kerf compensation) " )
pars . add_argument ( " --random_seed " , type = int , default = 42 , help = " Random seed for pseudo-random heuristics " )
def effect ( self ) :
stl_input = self . options . infile
if not os . path . exists ( stl_input ) :
inkex . utils . debug ( " The input file does not exist. Please select a proper file and try again. " )
exit ( 1 )
# Prepare output
2021-10-31 21:31:13 +01:00
basename = os . path . splitext ( os . path . basename ( stl_input ) ) [ 0 ]
svg_output = os . path . join ( tempfile . gettempdir ( ) , basename + " .svg " )
2021-10-26 20:37:24 +02:00
# Clean up possibly previously generated output file from plycutter
2021-10-31 21:31:13 +01:00
if os . path . exists ( svg_output ) :
2021-10-26 20:37:24 +02:00
try :
2021-10-31 21:31:13 +01:00
os . remove ( svg_output )
2021-10-26 20:37:24 +02:00
except OSError as e :
inkex . utils . debug ( " Error while deleting previously generated output file " + stl_input )
# Run PlyCutter
plycutter_cmd = " plycutter "
plycutter_cmd + = " --thickness " + str ( self . options . thickness ) + " "
if self . options . debug == True : plycutter_cmd + = " --debug "
plycutter_cmd + = " --min_finger_width " + str ( self . options . min_finger_width ) + " "
plycutter_cmd + = " --max_finger_width " + str ( self . options . max_finger_width ) + " "
plycutter_cmd + = " --support_radius " + str ( self . options . support_radius ) + " "
plycutter_cmd + = " --final_dilation " + str ( self . options . final_dilation ) + " "
plycutter_cmd + = " --random_seed " + str ( self . options . random_seed ) + " "
2021-10-31 21:31:13 +01:00
plycutter_cmd + = " --format svg " #static
plycutter_cmd + = " -o \" " + svg_output + " \" "
2021-10-26 20:37:24 +02:00
plycutter_cmd + = " \" " + stl_input + " \" "
#print command
2021-10-31 21:31:13 +01:00
#inkex.utils.debug(plycutter_cmd)
#create a new env for subprocess which does not contain extensions dir because there's a collision with "rtree.py"
pypath = ' '
for d in sys . path :
if d != ' /usr/share/inkscape/extensions ' :
pypath = pypath + d + ' ; '
neutral_env = os . environ . copy ( )
neutral_env [ ' PYTHONPATH ' ] = pypath
2021-10-26 20:37:24 +02:00
2021-10-31 21:31:13 +01:00
p = Popen ( plycutter_cmd , shell = True , stdout = PIPE , stderr = PIPE , env = neutral_env )
2021-10-26 20:37:24 +02:00
stdout , stderr = p . communicate ( )
2021-10-31 21:31:13 +01:00
2021-10-26 20:37:24 +02:00
p . wait ( )
if p . returncode != 0 :
inkex . utils . debug ( " PlyCutter failed: %d %s %s " % ( p . returncode ,
str ( stdout ) . replace ( ' \\ n ' , ' \n ' ) . replace ( ' \\ t ' , ' \t ' ) ,
str ( stderr ) . replace ( ' \\ n ' , ' \n ' ) . replace ( ' \\ t ' , ' \t ' ) )
)
exit ( 1 )
2021-10-31 21:31:13 +01:00
elif self . options . debug is True :
inkex . utils . debug ( " PlyCutter debug output: %d %s %s " % ( p . returncode ,
str ( stdout ) . replace ( ' \\ n ' , ' \n ' ) . replace ( ' \\ t ' , ' \t ' ) ,
str ( stderr ) . replace ( ' \\ n ' , ' \n ' ) . replace ( ' \\ t ' , ' \t ' ) )
)
2021-10-26 20:37:24 +02:00
# Write the generated SVG into InkScape's canvas
try :
stream = open ( svg_output , ' r ' )
except FileNotFoundError as e :
2021-10-31 21:31:13 +01:00
inkex . utils . debug ( " There was no SVG output generated by PlyCutter. Please check your model file. " )
2021-10-26 20:37:24 +02:00
exit ( 1 )
p = etree . XMLParser ( huge_tree = True )
doc = etree . parse ( stream , parser = etree . XMLParser ( huge_tree = True ) ) . getroot ( )
stream . close ( )
2021-10-31 21:31:13 +01:00
g = inkex . Group ( id = self . svg . get_unique_id ( " plycutter- " ) )
g . insert ( 0 , inkex . Desc ( " Imported file: {} " . format ( self . options . infile ) ) )
self . svg . get_current_layer ( ) . add ( g )
for element in doc . iter ( " { http://www.w3.org/2000/svg}path " ) :
g . append ( element )
2021-10-26 20:37:24 +02:00
#Adjust viewport and width/height to have the import at the center of the canvas
if self . options . resizetoimport :
2021-11-02 14:13:04 +01:00
#push some calculation of all bounding boxes. seems to refresh something in the background which makes the bbox calculation working at the bottom
for element in self . document . getroot ( ) . iter ( " * " ) :
try :
element . bounding_box ( )
except :
pass
bbox = g . bounding_box ( ) #only works because we process bounding boxes previously. see top
2021-10-26 20:37:24 +02:00
if bbox is not None :
root = self . svg . getElement ( ' //svg:svg ' ) ;
offset = self . svg . unittouu ( str ( self . options . extraborder ) + self . options . extraborder_units )
root . set ( ' viewBox ' , ' %f %f %f %f ' % ( bbox . left - offset , bbox . top - offset , bbox . width + 2 * offset , bbox . height + 2 * offset ) )
root . set ( ' width ' , bbox . width + 2 * offset )
root . set ( ' height ' , bbox . height + 2 * offset )
else :
self . msg ( " Error resizing to bounding box. " )
if __name__ == ' __main__ ' :
PlyCutter ( ) . run ( )