Code

made it callable by other extensions. the workaround is sorta ugly, but it should...
[inkscape.git] / share / extensions / measure.py
index e8089ca055a1f92c03cb994d580cd1e3cbf5e05e..d19dde93166e144ca2a6a8653c90d629f191b632 100644 (file)
@@ -1,5 +1,10 @@
 #!/usr/bin/env python 
 '''
+This extension module can measure arbitrary path and object length
+It adds a text to the selected path containing the length in a
+given unit.
+
+Copyright (C) 2006 Georg Wiora
 Copyright (C) 2006 Nathan Hurst
 Copyright (C) 2005 Aaron Spike, aaron@ekips.org
 
@@ -16,91 +21,152 @@ 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
+
+TODO:
+ * should use the standard attributes for text
+ * Implement option to keep text orientation upright 
+    1. Find text direction i.e. path tangent,
+    2. check direction >90 or <-90 Degrees
+    3. rotate by 180 degrees around text center
 '''
-import inkex, simplestyle, simplepath,sys,cubicsuperpath, bezmisc
+import inkex, simplestyle, simplepath,sys,cubicsuperpath, bezmisc, locale
+# Set current system locale
+locale.setlocale(locale.LC_ALL, '')
 
 def numsegs(csp):
-       return sum([len(p)-1 for p in csp])
+    return sum([len(p)-1 for p in csp])
 def interpcoord(v1,v2,p):
-       return v1+((v2-v1)*p)
+    return v1+((v2-v1)*p)
 def interppoints(p1,p2,p):
-       return [interpcoord(p1[0],p2[0],p),interpcoord(p1[1],p2[1],p)]
+    return [interpcoord(p1[0],p2[0],p),interpcoord(p1[1],p2[1],p)]
 def pointdistance((x1,y1),(x2,y2)):
-       return math.sqrt(((x2 - x1) ** 2) + ((y2 - y1) ** 2))
+    return math.sqrt(((x2 - x1) ** 2) + ((y2 - y1) ** 2))
 def bezlenapprx(sp1, sp2):
-       return pointdistance(sp1[1], sp1[2]) + pointdistance(sp1[2], sp2[0]) + pointdistance(sp2[0], sp2[1])
+    return pointdistance(sp1[1], sp1[2]) + pointdistance(sp1[2], sp2[0]) + pointdistance(sp2[0], sp2[1])
 def tpoint((x1,y1), (x2,y2), t = 0.5):
-       return [x1+t*(x2-x1),y1+t*(y2-y1)]
+    return [x1+t*(x2-x1),y1+t*(y2-y1)]
 def cspbezsplit(sp1, sp2, t = 0.5):
-       m1=tpoint(sp1[1],sp1[2],t)
-       m2=tpoint(sp1[2],sp2[0],t)
-       m3=tpoint(sp2[0],sp2[1],t)
-       m4=tpoint(m1,m2,t)
-       m5=tpoint(m2,m3,t)
-       m=tpoint(m4,m5,t)
-       return [[sp1[0][:],sp1[1][:],m1], [m4,m,m5], [m3,sp2[1][:],sp2[2][:]]]
+    m1=tpoint(sp1[1],sp1[2],t)
+    m2=tpoint(sp1[2],sp2[0],t)
+    m3=tpoint(sp2[0],sp2[1],t)
+    m4=tpoint(m1,m2,t)
+    m5=tpoint(m2,m3,t)
+    m=tpoint(m4,m5,t)
+    return [[sp1[0][:],sp1[1][:],m1], [m4,m,m5], [m3,sp2[1][:],sp2[2][:]]]
 def cspbezsplitatlength(sp1, sp2, l = 0.5, tolerance = 0.001):
-       bez = (sp1[1][:],sp1[2][:],sp2[0][:],sp2[1][:])
-       t = bezmisc.beziertatlength(bez, l, tolerance)
-       return cspbezsplit(sp1, sp2, t)
+    bez = (sp1[1][:],sp1[2][:],sp2[0][:],sp2[1][:])
+    t = bezmisc.beziertatlength(bez, l, tolerance)
+    return cspbezsplit(sp1, sp2, t)
 def cspseglength(sp1,sp2, tolerance = 0.001):
-       bez = (sp1[1][:],sp1[2][:],sp2[0][:],sp2[1][:])
-       return bezmisc.bezierlength(bez, tolerance)     
+    bez = (sp1[1][:],sp1[2][:],sp2[0][:],sp2[1][:])
+    return bezmisc.bezierlength(bez, tolerance)    
 def csplength(csp):
-       total = 0
-       lengths = []
-       for sp in csp:
-               lengths.append([])
-               for i in xrange(1,len(sp)):
-                       l = cspseglength(sp[i-1],sp[i])
-                       lengths[-1].append(l)
-                       total += l                      
-       return lengths, total
+    total = 0
+    lengths = []
+    for sp in csp:
+        lengths.append([])
+        for i in xrange(1,len(sp)):
+            l = cspseglength(sp[i-1],sp[i])
+            lengths[-1].append(l)
+            total += l            
+    return lengths, total
 
 class Length(inkex.Effect):
-       def __init__(self):
-               inkex.Effect.__init__(self)
-               self.OptionParser.add_option("-f", "--fontsize",
-                                               action="store", type="string", 
-                                               dest="fontsize", default="20",
-                                               help="Size of node label numbers")      
-               self.OptionParser.add_option("-o", "--offset",
-                                               action="store", type="string", 
-                                               dest="offset", default="-4",
-                                               help="The distance above the curve")    
-       def effect(self):
-               for id, node in self.selected.iteritems():
-                       if node.tagName == 'path':
-                               self.group = self.document.createElement('svg:g')
-                               node.parentNode.appendChild(self.group)
-                               
-                               try:
-                                       t = node.attributes.getNamedItem('transform').value
-                                       self.group.setAttribute('transform', t)
-                               except AttributeError:
-                                       pass
+    def __init__(self):
+        inkex.Effect.__init__(self)
+        self.OptionParser.add_option("-f", "--fontsize",
+                        action="store", type="int", 
+                        dest="fontsize", default=20,
+                        help="Size of length lable text in px")
+        self.OptionParser.add_option("-o", "--offset",
+                        action="store", type="float", 
+                        dest="offset", default=-6,
+                        help="The distance above the curve")
+        self.OptionParser.add_option("-u", "--unit",
+                        action="store", type="string", 
+                        dest="unit", default="mm",
+                        help="The unit of the measurement")
+        self.OptionParser.add_option("-p", "--precision",
+                        action="store", type="int", 
+                        dest="precision", default=2,
+                        help="Number of significant digits after decimal point")
+        self.OptionParser.add_option("-s", "--scale",
+                        action="store", type="float", 
+                        dest="scale", default=1,
+                        help="The distance above the curve")
+        self.OptionParser.add_option("-r", "--orient",
+                        action="store", type="inkbool", 
+                        dest="orient", default=True,
+                        help="Keep orientation of text upright")
+        self.OptionParser.add_option("--tab",
+                        action="store", type="string", 
+                        dest="tab", default="sampling",
+                        help="The selected UI-tab when OK was pressed") 
+        self.OptionParser.add_option("--measurehelp",
+                        action="store", type="string", 
+                        dest="measurehelp", default="",
+                        help="dummy") 
+                        
+    def effect(self):
+        # get number of digits
+        prec = int(self.options.precision)
+        # loop over all selected pathes
+        for id, node in self.selected.iteritems():
+            if node.tagName == 'path':
+                # self.group = self.document.createElement('svg:g')
+                self.group = self.document.createElement('svg:text')
+                node.parentNode.appendChild(self.group)
+                
+                try:
+                    t = node.attributes.getNamedItem('transform').value
+                    self.group.setAttribute('transform', t)
+                except AttributeError:
+                    pass
+
+                a =[]
+                p = cubicsuperpath.parsePath(node.attributes.getNamedItem('d').value)
+                num = 1
+                slengths, stotal = csplength(p)
+                ''' Wio: Umrechnung in unit '''
+                if self.options.unit=="mm":
+                    factor=0.2822219  # px->mm
+                elif self.options.unit=="pt":
+                    factor=0.80       # px->pt
+                elif self.options.unit=="cm":
+                    factor=0.02822219 # px->cm
+                elif self.options.unit=="m":
+                    factor=0.0002822219 # px->m
+                elif self.options.unit=="km":
+                    factor=0.0000002822219 # px->km
+                elif self.options.unit=="in":
+                    factor=0.2822219/25.4 # px->in
+                else :
+                    ''' Default unit is px'''
+                    factor=1
+                    self.options.unit="px"
+                    
+                # Format the length as string
+                lenstr = locale.format("%(len)25."+str(prec)+"f",{'len':round(stotal*factor*self.options.scale,prec)}).strip()
+                self.addTextOnPath(self.group,0, 0,lenstr+' '+self.options.unit, id, self.options.offset)
 
-                               a =[]
-                               p = cubicsuperpath.parsePath(node.attributes.getNamedItem('d').value)
-                               num = 1
-                               slengths, stotal = csplength(p)
-                                self.addTextOnPath(self.group,0, 0,str(stotal), id, self.options.offset)
 
-                               
-       def addTextOnPath(self,node,x,y,text, id,dy=0):
-                               new = self.document.createElement('svg:text')
-                               tp = self.document.createElement('svg:textPath')
-                               s = {'font-size': self.options.fontsize, 'fill-opacity': '1.0', 'stroke': 'none',
-                                       'font-weight': 'normal', 'font-style': 'normal', 'fill': '#000000'}
-                               new.setAttribute('style', simplestyle.formatStyle(s))
-                               new.setAttribute('x', str(x))
-                               new.setAttribute('y', str(y))
-                               tp.setAttributeNS('http://www.w3.org/1999/xlink','xlink:href', '#'+id)
-                               tp.setAttribute('startOffset', "50%")
-                               #tp.setAttribute('dy', dy) # dubious merit
-                               new.appendChild(tp)
-                                tp.appendChild(self.document.createTextNode(str(text)))
-                               node.appendChild(new)
+    def addTextOnPath(self,node,x,y,text, id,dy=0):
+                #new = self.document.createElement('svg:text')
+                new = self.document.createElement('svg:textPath')
+                s = {'text-align': 'center', 'vertical-align': 'bottom',
+                    'text-anchor': 'middle', 'font-size': str(self.options.fontsize),
+                    'fill-opacity': '1.0', 'stroke': 'none',
+                    'font-weight': 'normal', 'font-style': 'normal', 'fill': '#000000'}
+                new.setAttribute('style', simplestyle.formatStyle(s))
+                node.setAttribute('x', str(x))
+                node.setAttribute('y', str(y))
+                #node.setAttribute('transform','rotate(180,'+str(-x)+','+str(-y)+')')
+                new.setAttributeNS('http://www.w3.org/1999/xlink','xlink:href', '#'+id)
+                new.setAttribute('startOffset', "50%")
+                new.setAttribute('dy', str(dy)) # dubious merit
+                #new.appendChild(tp)
+                new.appendChild(self.document.createTextNode(str(text)))
+                node.appendChild(new)
 
 e = Length()
 e.affect()