"""
This page is in the table of contents.
The svg.py script is an import translator plugin to get a carving from an svg file.
An import plugin is a script in the import_plugins folder which has the function getCarving. It is meant to be run from the interpret tool. To ensure that the plugin works on platforms which do not handle file capitalization properly, give the plugin a lower case name.
The getCarving function takes the file name of an svg file and returns the carving.
This example gets a carving for the svg file Screw Holder Bottom.svg. This example is run in a terminal in the folder which contains Screw Holder Bottom.svg and svg.py.
> python
Python 2.5.1 (r251:54863, Sep 22 2007, 01:43:31)
[GCC 4.2.1 (SUSE Linux)] on linux2
Type "help", "copyright", "credits" or "license" for more information.
>>> import svg
>>> svg.getCarving()
0.20000000298, 999999999.0, -999999999.0, [8.72782748851e-17, None
..
many more lines of the carving
..
"""
from __future__ import absolute_import
#Init has to be imported first because it has code to workaround the python bug where relative imports don't work if the module is imported as a main module.
import __init__
from skeinforge_tools.skeinforge_utilities.vector3 import Vector3
from skeinforge_tools.skeinforge_utilities import euclidean
from skeinforge_tools.skeinforge_utilities import gcodec
__author__ = "Enrique Perez (perez_enrique@yahoo.com)"
__credits__ = 'Nophead \nArt of Illusion '
__date__ = "$Date: 2008/21/04 $"
__license__ = "GPL 3.0"
def addPathData( line, loops ):
"Add the data from the path line."
line = line.replace( '"', ' ' )
splitLine = line.split()
if splitLine[ 1 ] != 'transform=':
return
line = line.lower()
line = line.replace( 'm', ' ' )
line = line.replace( 'l', ' ' )
line = line.replace( '/>', ' ' )
splitLine = line.split()
if 'd=' not in splitLine:
return
splitLine = splitLine[ splitLine.index( 'd=' ) + 1 : ]
pathSequence = []
for word in splitLine:
if word == 'z':
loop = []
for pathSequenceIndex in xrange( 0, len( pathSequence ), 2 ):
coordinate = complex( pathSequence[ pathSequenceIndex ], pathSequence[ pathSequenceIndex + 1 ] )
loop.append( coordinate )
loops.append( loop )
pathSequence = []
else:
pathSequence.append( float( word ) )
def addTextData( line, rotatedBoundaryLayers ):
"Add the data from the text line."
if line.find( 'layerThickness' ) != - 1:
return
line = line.replace( '>', ' ' )
line = line.replace( '<', ' ' )
line = line.replace( ',', ' ' )
splitLine = line.split()
if 'Layer' not in splitLine:
return
splitLine = splitLine[ splitLine.index( 'Layer' ) + 1 : ]
if 'z' not in splitLine:
return
zIndex = splitLine.index( 'z' )
rotatedBoundaryLayer = euclidean.RotatedLoopLayer( float( splitLine[ zIndex + 1 ] ) )
rotatedBoundaryLayers.append( rotatedBoundaryLayer )
def getCarving( fileName = '' ):
"Get the triangle mesh for the gts file."
if fileName == '':
unmodified = gcodec.getFilesWithFileTypeWithoutWords( 'gts' )
if len( unmodified ) == 0:
print( "There is no gts file in this folder." )
return None
fileName = unmodified[ 0 ]
carving = SVGCarving()
carving.parseSVG( gcodec.getFileText( fileName ) )
return carving
def getValueInQuotes( name, text, value ):
"Get value in quotes after the name."
nameAndQuote = name + '="'
nameIndexStart = text.find( nameAndQuote )
if nameIndexStart == - 1:
return value
valueStartIndex = nameIndexStart + len( nameAndQuote )
nameIndexEnd = text.find( '"', valueStartIndex )
if nameIndexEnd == - 1:
return value
return float( text[ valueStartIndex : nameIndexEnd ] )
class SVGCarving:
"An svg carving."
def __init__( self ):
"Add empty lists."
self.maximumZ = - 999999999.0
self.minimumZ = 999999999.0
self.layerThickness = None
self.rotatedBoundaryLayers = []
def __repr__( self ):
"Get the string representation of this carving."
return str( self.rotatedBoundaryLayers )
def getCarveCornerMaximum( self ):
"Get the corner maximum of the vertices."
return self.cornerMaximum
def getCarveCornerMinimum( self ):
"Get the corner minimum of the vertices."
return self.cornerMinimum
def getCarveLayerThickness( self ):
"Get the layer thickness."
return self.layerThickness
def getCarveRotatedBoundaryLayers( self ):
"Get the rotated boundary layers."
self.cornerMaximum = Vector3( - 999999999.0, - 999999999.0, self.maximumZ )
self.cornerMinimum = Vector3( 999999999.0, 999999999.0, self.minimumZ )
for rotatedBoundaryLayer in self.rotatedBoundaryLayers:
for loop in rotatedBoundaryLayer.loops:
for point in loop:
pointVector3 = Vector3( point.real, point.imag, rotatedBoundaryLayer.z )
self.cornerMaximum = euclidean.getPointMaximum( self.cornerMaximum, pointVector3 )
self.cornerMinimum = euclidean.getPointMinimum( self.cornerMinimum, pointVector3 )
return self.rotatedBoundaryLayers
def parseSVG( self, svgText ):
"Parse SVG text and store the layers."
if svgText == '':
return None
svgText = svgText.replace( '\t', ' ' )
svgText = svgText.replace( ';', ' ' )
self.lines = gcodec.getTextLines( svgText )
self.parseInitialization()
for lineIndex in xrange( self.lineIndex, len( self.lines ) ):
self.parseLine( lineIndex )
def parseInitialization( self ):
"Parse gcode initialization and store the parameters."
for self.lineIndex in xrange( len( self.lines ) ):
line = self.lines[ self.lineIndex ].lstrip()
self.layerThickness = getValueInQuotes( 'layerThickness', line, self.layerThickness )
self.maximumZ = getValueInQuotes( 'maxZ', line, self.maximumZ )
self.minimumZ = getValueInQuotes( 'minZ', line, self.minimumZ )
if line.find( '' ) != - 1:
return
self.lineIndex = 2
def parseLine( self, lineIndex ):
"Parse a gcode line and add it to the inset skein."
line = self.lines[ lineIndex ].lstrip()
splitLine = line.split()
if len( splitLine ) < 1:
return
firstWord = splitLine[ 0 ]
if firstWord == '