#!/usr/bin/env python ''' Copyright (C) 2006 Jean-Francois Barraud, barraud@math.univ-lille1.fr 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 barraud@math.univ-lille1.fr ''' import inkex, cubicsuperpath, bezmisc, simplestyle import copy, math, re, random def parseTransform(transf,mat=[[1.0,0.0,0.0],[0.0,1.0,0.0]]): if transf=="": return(mat) result=re.match("(translate|scale|rotate|skewX|skewY|matrix)\(([^)]*)\)",transf) #-- translate -- if result.group(1)=="translate": args=result.group(2).split(",") dx=float(args[0]) if len(args)==1: dy=0.0 else: dy=float(args[1]) matrix=[[1,0,dx],[0,1,dy]] #-- scale -- if result.groups(1)=="scale": args=result.group(2).split(",") sx=float(args[0]) if len(args)==1: sy=sx else: sy=float(args[1]) matrix=[[sx,0,0],[0,sy,0]] #-- rotate -- if result.groups(1)=="rotate": args=result.group(2).split(",") a=float(args[0])*math.pi/180 if len(args)==1: cx,cy=(0.0,0.0) else: cx,cy=args[1:] matrix=[[math.cos(a),-math.sin(a),cx],[math.sin(a),math.cos(a),cy]] #-- skewX -- if result.groups(1)=="skewX": a=float(result.group(2))*math.pi/180 matrix=[[1,math.tan(a),0],[0,1,0]] #-- skewX -- if result.groups(1)=="skewX": a=float(result.group(2))*math.pi/180 matrix=[[1,0,0],[math.tan(a),1,0]] #-- matrix -- if result.group(1)=="matrix": a11,a21,a12,a22,v1,v2=result.group(2).split(",") matrix=[[float(a11),float(a12),float(v1)],[float(a21),float(a22),float(v2)]] matrix=composeTransform(mat,matrix) if result.end() should it be given an id? #seems to work without this!?! myid = node.tag.split('}')[-1] clone.set("id", self.uniqueId(myid)) node.getparent().append(clone) clones[clone.get("id")]=clone return(clones) def uniqueId(self, prefix): id="%s%04i"%(prefix,random.randint(0,9999)) while len(self.document.getroot().xpath('//*[@id="%s"]' % id,inkex.NSS)): id="%s%04i"%(prefix,random.randint(0,9999)) return(id) def expandGroups(self,aList,transferTransform=True): for id, node in aList.items(): if node.tag == inkex.addNS('g','svg') or node.tag=='g': mat=parseTransform(node.get("transform")) for child in node: if transferTransform: applyTransformToNode(mat,child) aList.update(self.expandGroups({child.get('id'):child})) if transferTransform: del node.attrib["transform"] del aList[id] return(aList) def expandGroupsUnlinkClones(self,aList,transferTransform=True,doReplace=True): for id in aList.keys()[:]: node=aList[id] if node.tag == inkex.addNS('g','svg') or node.tag=='g': self.expandGroups(aList,transferTransform) self.expandGroupsUnlinkClones(aList,transferTransform,doReplace) #Hum... not very efficient if there are many clones of groups... elif node.tag == inkex.addNS('use','svg') or node.tag=='use': refid=node.get(inkex.addNS('href','xlink')) path = '//*[@id="%s"]' % refid[1:] refnode = self.document.getroot().xpath(path,inkex.NSS) newnode=copy.deepcopy(refnode) self.recursNewIds(newnode) s = node.get('style') if s: style=simplestyle.parseStyle(s) refstyle=simplestyle.parseStyle(refnode.get('style')) style.update(refstyle) newnode.set('style',simplestyle.formatStyle(style)) applyTransformToNode(parseTransform(node.get('transform')),newnode) if doReplace: parent=node.getparent() parent.insert(node.index,newnode) parent.remove(node) del aList[id] newid=newnode.get('id') aList.update(self.expandGroupsUnlinkClones({newid:newnode},transferTransform,doReplace)) return aList def recursNewIds(self,node): if node.get('id'): node.set('id',self.uniqueId(node.tag)) for child in node: self.recursNewIds(child) # Had not been rewritten for ElementTree # def makeClonesReal(self,aList,doReplace=True,recursivelytransferTransform=True): # for id in aList.keys(): # node=aList[id] # if node.tagName == 'g': # childs={} # for child in node.childNodes: # if child.nodeType==child.ELEMENT_NODE: # childid=child.getAttributeNS(None,'id') # del aList[childid] # aList.update(self.makeClonesReal({childid:child},doReplace)) # elif node.tagName == 'use': # refid=node.getAttributeNS(inkex.NSS[u'xlink'],'href') # path = '//*[@id="%s"]' % refid[1:] # refnode = xml.xpath.Evaluate(path,document)[0] # clone=refnode.cloneNode(True) # cloneid=self.uniqueId(clone.tagName) # clone.setAttributeNS(None,'id', cloneid) # style=simplestyle.parseStyle(node.getAttributeNS(None,u'style')) # refstyle=simplestyle.parseStyle(refnode.getAttributeNS(None,u'style')) # style.update(refstyle) # clone.setAttributeNS(None,'style',simplestyle.formatStyle(style)) # applyTransformToNode(parseTransform(node.getAttributeNS(None,'transform')),clone) # if doReplace: # parent=node.parentNode # parent.insertBefore(clone,node) # parent.removeChild(node) # del aList[id] # aList.update(self.expandGroupsUnlinkClones({cloneid:clone})) # return aList ################################ #-- Object conversion ---------- ################################ def rectToPath(self,node,doReplace=True): if node.tag == inkex.addNS('rect','svg'): x =float(node.get('x')) y =float(node.get('y')) #FIXME: no exception anymore and sometimes just one try: rx=float(node.get('rx')) ry=float(node.get('ry')) except: rx=0 ry=0 w =float(node.get('width' )) h =float(node.get('height')) d ='M %f,%f '%(x+rx,y) d+='L %f,%f '%(x+w-rx,y) d+='A %f,%f,%i,%i,%i,%f,%f '%(rx,ry,0,0,1,x+w,y+ry) d+='L %f,%f '%(x+w,y+h-ry) d+='A %f,%f,%i,%i,%i,%f,%f '%(rx,ry,0,0,1,x+w-rx,y+h) d+='L %f,%f '%(x+rx,y+h) d+='A %f,%f,%i,%i,%i,%f,%f '%(rx,ry,0,0,1,x,y+h-ry) d+='L %f,%f '%(x,y+ry) d+='A %f,%f,%i,%i,%i,%f,%f '%(rx,ry,0,0,1,x+rx,y) newnode=inkex.etree.Element('path') newnode.set('d',d) newnode.set('id', self.uniqueId('path')) newnode.set('style',node.get('style')) nnt = node.get('transform') if nnt: newnode.set('transform',nnt) fuseTransform(newnode) if doReplace: parent=node.getparent() parent.insert(node.index,newnode) parent.remove(node) return newnode def objectToPath(self,node,doReplace=True): #--TODO: support other object types!!!! #--TODO: make sure cubicsuperpath supports A and Q commands... if node.tag == inkex.addNS('rect','svg'): return(self.rectToPath(node,doReplace)) elif node.tag == inkex.addNS('path','svg') or node.tag == 'path': attributes = node.keys() for attName in attributes: uri = None if attName[0] == '{': uri,attName = attName.split('}') uri = uri[1:] if uri in [inkex.NSS[u'sodipodi'],inkex.NSS[u'inkscape']]: #if attName not in ["d","id","style","transform"]: del node.attrib[inkex.addNS(attName,uri)] fuseTransform(node) return node else: inkex.debug("Please first convert objects to paths!...(got '%s')"%node.tag) return None def objectsToPaths(self,aList,doReplace=True): newSelection={} for id,node in aList.items(): newnode=self.objectToPath(node,self.document) del aList[id] aList[newnode.get('id')]=newnode ################################ #-- Action ---------- ################################ #-- overwrite this method in subclasses... def effect(self): #self.duplicateNodes(self.selected) self.expandGroupsUnlinkClones(self.selected, True) self.objectsToPaths(self.selected, True) self.bbox=self.computeBBox(self.selected) for id, node in self.selected.iteritems(): if node.tag == inkex.addNS('path','svg'): d = node.get('d') p = cubicsuperpath.parsePath(d) #do what ever you want with p! node.set('d',cubicsuperpath.formatPath(p)) class Diffeo(PathModifier): def __init__(self): inkex.Effect.__init__(self) def applyDiffeo(self,bpt,vects=()): ''' bpt is a base point and for v in vectors, v'=v-p is a tangent vector at bpt. Defaults to identity! ''' for v in vects: v[0]-=bpt[0] v[1]-=bpt[1] #-- your transformations go here: #x,y=bpt #bpt[0]=f(x,y) #bpt[1]=g(x,y) #for v in vects: # vx,vy=v # v[0]=df/dx(x,y)*vx+df/dy(x,y)*vy # v[1]=dg/dx(x,y)*vx+dg/dy(x,y)*vy # #-- !caution! y-axis is pointing downward! for v in vects: v[0]+=bpt[0] v[1]+=bpt[1] def effect(self): #self.duplicateNodes(self.selected) self.expandGroupsUnlinkClones(self.selected, True) self.expandGroups(self.selected, True) self.objectsToPaths(self.selected, True) self.bbox=self.computeBBox(self.selected) for id, node in self.selected.iteritems(): if node.tag == inkex.addNS('path','svg') or node.tag=='path': d = node.get('d') p = cubicsuperpath.parsePath(d) for sub in p: for ctlpt in sub: self.applyDiffeo(ctlpt[1],(ctlpt[0],ctlpt[2])) node.set('d',cubicsuperpath.formatPath(p)) #e = Diffeo() #e.affect()