summaryrefslogtreecommitdiffstats
path: root/share/extensions/dpiswitcher.py
diff options
context:
space:
mode:
Diffstat (limited to 'share/extensions/dpiswitcher.py')
-rwxr-xr-xshare/extensions/dpiswitcher.py346
1 files changed, 346 insertions, 0 deletions
diff --git a/share/extensions/dpiswitcher.py b/share/extensions/dpiswitcher.py
new file mode 100755
index 0000000..b1cf2ec
--- /dev/null
+++ b/share/extensions/dpiswitcher.py
@@ -0,0 +1,346 @@
+#!/usr/bin/env python
+# coding=utf-8
+#
+# Copyright (C) 2012 Jabiertxo Arraiza, jabier.arraiza@marker.es
+# Copyright (C) 2016 su_v, <suv-sf@users.sf.net>
+#
+# 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
+#
+"""
+Version 0.6 - DPI Switcher
+
+This extension scales a document to fit different SVG DPI -90/96-
+
+Changes since v0.5:
+ - transform all top-level containers and graphics elements
+ - support scientific notation in SVG lengths
+ - fix scaling with existing matrix()
+ - support different units for document width, height attributes
+ - improve viewBox support (syntax, offset)
+ - support common cases of text-put-on-path in SVG root
+ - support common cases of <use> references in SVG root
+ - examples from http://tavmjong.free.fr/INKSCAPE/UNITS/ tested
+
+TODO:
+ - check grids/guides created with 0.91:
+ http://tavmjong.free.fr/INKSCAPE/UNITS/units_mm_nv_90dpi.svg
+ - check <symbol> instances
+ - check more <use> and text-on-path cases (reverse scaling needed?)
+ - scale perspective of 3dboxes
+
+"""
+
+import re
+import math
+import inkex
+from inkex import Use, TextElement
+
+# globals
+SKIP_CONTAINERS = [
+ 'defs',
+ 'glyph',
+ 'marker',
+ 'mask',
+ 'missing-glyph',
+ 'pattern',
+ 'symbol',
+]
+CONTAINER_ELEMENTS = [
+ 'a',
+ 'g',
+ 'switch',
+]
+GRAPHICS_ELEMENTS = [
+ 'circle',
+ 'ellipse',
+ 'image',
+ 'line',
+ 'path',
+ 'polygon',
+ 'polyline',
+ 'rect',
+ 'text',
+ 'use',
+]
+
+
+def is_3dbox(element):
+ """Check whether element is an Inkscape 3dbox type."""
+ return element.get('sodipodi:type') == 'inkscape:box3d'
+
+
+def is_text_on_path(element):
+ """Check whether text element is put on a path."""
+ if isinstance(element, TextElement):
+ text_path = element.find('svg:textPath')
+ if text_path is not None and len(text_path):
+ return True
+ return False
+
+
+def is_sibling(element1, element2):
+ """Check whether element1 and element2 are siblings of same parent."""
+ return element2 in element1.getparent()
+
+
+def is_in_defs(doc, element):
+ """Check whether element is in defs."""
+ if element is not None:
+ defs = doc.find('defs')
+ if defs is not None:
+ return element in defs.iterdescendants()
+ return False
+
+
+def check_3dbox(svg, element, scale_x, scale_y):
+ """Check transformation for 3dbox element."""
+ skip = False
+ if skip:
+ # 3dbox elements ignore preserved transforms
+ # FIXME: manually update geometry of 3dbox?
+ pass
+ return skip
+
+
+def check_text_on_path(svg, element, scale_x, scale_y):
+ """Check whether to skip scaling a text put on a path."""
+ skip = False
+ path = element.find('textPath').href
+ if not is_in_defs(svg, path):
+ if is_sibling(element, path):
+ # skip common element scaling if both text and path are siblings
+ skip = True
+ # scale offset
+ if 'transform' in element.attrib:
+ element.transform.add_scale(scale_x, scale_y)
+ # scale font size
+ mat = inkex.Transform('scale({},{})'.format(scale_x, scale_y)).matrix
+ det = abs(mat[0][0] * mat[1][1] - mat[0][1] * mat[1][0])
+ descrim = math.sqrt(abs(det))
+ prop = 'font-size'
+ # outer text
+ sdict = dict(inkex.Style.parse_str(element.get('style')))
+ if prop in sdict:
+ sdict[prop] = float(sdict[prop]) * descrim
+ element.set('style', str(inkex.Style(sdict)))
+ # inner tspans
+ for child in element.iterdescendants():
+ if isinstance(element, inkex.Tspan):
+ sdict = dict(inkex.Style.parse_str(child.get('style')))
+ if prop in sdict:
+ sdict[prop] = float(sdict[prop]) * descrim
+ child.set('style', str(inkex.Style(sdict)))
+ return skip
+
+
+def check_use(svg, element, scale_x, scale_y):
+ """Check whether to skip scaling an instantiated element (<use>)."""
+ skip = False
+ path = element.href
+ if not is_in_defs(svg, path):
+ if is_sibling(element, path):
+ skip = True
+ # scale offset
+ if 'transform' in element.attrib:
+ element.transform.add_scale(scale_x, scale_y)
+ return skip
+
+
+class DPISwitcher(inkex.EffectExtension):
+ multi_inx = True
+ factor_a = 90.0 / 96.0
+ factor_b = 96.0 / 90.0
+ units = "px"
+
+ def add_arguments(self, pars):
+ pars.add_argument("--switcher", type=str, default="0",
+ help="Select the DPI switch you want")
+
+ # dictionaries of unit to user unit conversion factors
+ __uuconvLegacy = {
+ 'in': 90.0,
+ 'pt': 1.25,
+ 'px': 1.0,
+ 'mm': 3.5433070866,
+ 'cm': 35.433070866,
+ 'm': 3543.3070866,
+ 'km': 3543307.0866,
+ 'pc': 15.0,
+ 'yd': 3240.0,
+ 'ft': 1080.0,
+ }
+ __uuconv = {
+ 'in': 96.0,
+ 'pt': 1.33333333333,
+ 'px': 1.0,
+ 'mm': 3.77952755913,
+ 'cm': 37.7952755913,
+ 'm': 3779.52755913,
+ 'km': 3779527.55913,
+ 'pc': 16.0,
+ 'yd': 3456.0,
+ 'ft': 1152.0,
+ }
+
+ def parse_length(self, length, percent=False):
+ """Parse SVG length."""
+ if self.options.switcher == "0": # dpi90to96
+ known_units = list(self.__uuconvLegacy)
+ else: # dpi96to90
+ known_units = list(self.__uuconv)
+ if percent:
+ unitmatch = re.compile('(%s)$' % '|'.join(known_units + ['%']))
+ else:
+ unitmatch = re.compile('(%s)$' % '|'.join(known_units))
+ param = re.compile(r'(([-+]?[0-9]+(\.[0-9]*)?|[-+]?\.[0-9]+)([eE][-+]?[0-9]+)?)')
+ p = param.match(length)
+ u = unitmatch.search(length)
+ val = 100 # fallback: assume default length of 100
+ unit = 'px' # fallback: assume 'px' unit
+ if p:
+ val = float(p.string[p.start():p.end()])
+ if u:
+ unit = u.string[u.start():u.end()]
+ return val, unit
+
+ def convert_length(self, val, unit):
+ """Convert length to self.units if unit differs."""
+ doc_unit = self.units or 'px'
+ if unit != doc_unit:
+ if self.options.switcher == "0": # dpi90to96
+ val_px = val * self.__uuconvLegacy[unit]
+ val = val_px / (self.__uuconvLegacy[doc_unit] / self.__uuconvLegacy['px'])
+ unit = doc_unit
+ else: # dpi96to90
+ val_px = val * self.__uuconv[unit]
+ val = val_px / (self.__uuconv[doc_unit] / self.__uuconv['px'])
+ unit = doc_unit
+ return val, unit
+
+ def check_attr_unit(self, element, attr, unit_list):
+ """Check unit of attribute value, match to units in *unit_list*."""
+ if attr in element.attrib:
+ unit = self.parse_length(element.get(attr), percent=True)[1]
+ return unit in unit_list
+
+ def scale_attr_val(self, element, attr, unit_list, factor):
+ """Scale attribute value if unit matches one in *unit_list*."""
+ if attr in element.attrib:
+ val, unit = self.parse_length(element.get(attr), percent=True)
+ if unit in unit_list:
+ element.set(attr, '{}{}'.format(val * factor, unit))
+
+ def scale_root(self, unit_exponent=1.0):
+ """Scale all top-level elements in SVG root."""
+
+ # update viewport
+ width_num = self.parse_length(self.svg.get('width'))[0]
+ height_num = self.convert_length(*self.parse_length(self.svg.get('height')))[0]
+ width_doc = width_num * self.factor_a * unit_exponent
+ height_doc = height_num * self.factor_a * unit_exponent
+
+ svg = self.svg
+ if svg.get('height'):
+ svg.set('height', str(height_doc))
+ if svg.get('width'):
+ svg.set('width', str(width_doc))
+
+ # update viewBox
+ if svg.get('viewBox'):
+ viewboxstring = re.sub(' +|, +|,', ' ', svg.get('viewBox'))
+ viewboxlist = [float(i) for i in viewboxstring.strip().split(' ', 4)]
+ svg.set('viewBox', '{} {} {} {}'.format(*[(val * self.factor_a) for val in viewboxlist]))
+
+ # update guides, grids
+ if self.options.switcher == "1":
+ # FIXME: dpi96to90 only?
+ self.scale_guides()
+ self.scale_grid()
+
+ for element in svg: # iterate all top-level elements of SVGRoot
+
+ # init variables
+ tag = element.TAG
+ width_scale = self.factor_a
+ height_scale = self.factor_a
+
+ if tag in GRAPHICS_ELEMENTS or tag in CONTAINER_ELEMENTS:
+
+ # test for specific elements to skip from scaling
+ if is_3dbox(element):
+ if check_3dbox(svg, element, width_scale, height_scale):
+ continue
+ if is_text_on_path(element):
+ if check_text_on_path(svg, element, width_scale, height_scale):
+ continue
+ if isinstance(element, Use):
+ if check_use(svg, element, width_scale, height_scale):
+ continue
+
+ # relative units ('%') in presentation attributes
+ for attr in ['width', 'height']:
+ self.scale_attr_val(element, attr, ['%'], 1.0 / self.factor_a)
+ for attr in ['x', 'y']:
+ self.scale_attr_val(element, attr, ['%'], 1.0 / self.factor_a)
+
+ # set preserved transforms on top-level elements
+ if width_scale != 1.0 and height_scale != 1.0:
+ element.transform.add_scale(width_scale, height_scale)
+
+ def scale_element(self, elem):
+ pass # TODO: optionally scale graphics elements only?
+
+ def scale_guides(self):
+ """Scale the guidelines"""
+ for guide in self.svg.namedview.get_guides():
+ point = guide.get("position").split(",")
+ guide.set("position", str(float(point[0].strip()) * self.factor_a) + "," + str(float(point[1].strip()) * self.factor_a))
+
+ def scale_grid(self):
+ """Scale the inkscape grid"""
+ grids = self.svg.xpath('//inkscape:grid')
+ for grid in grids:
+ grid.set("units", "px")
+ if grid.get("spacingx"):
+ spacingx = str(float(re.sub("[a-zA-Z]", "", grid.get("spacingx"))) * self.factor_a) + "px"
+ grid.set("spacingx", str(spacingx))
+ if grid.get("spacingy"):
+ spacingy = str(float(re.sub("[a-zA-Z]", "", grid.get("spacingy"))) * self.factor_a) + "px"
+ grid.set("spacingy", str(spacingy))
+ if grid.get("originx"):
+ originx = str(float(re.sub("[a-zA-Z]", "", grid.get("originx"))) * self.factor_a) + "px"
+ grid.set("originx", str(originx))
+ if grid.get("originy"):
+ originy = str(float(re.sub("[a-zA-Z]", "", grid.get("originy"))) * self.factor_a) + "px"
+ grid.set("originy", str(originy))
+
+ def effect(self):
+ svg = self.svg
+ if self.options.switcher == "0":
+ self.factor_a = 96.0 / 90.0
+ self.factor_b = 90.0 / 96.0
+ svg.namedview.set('inkscape:document-units', "px")
+ self.units = self.parse_length(svg.get('width'))[1]
+ unit_exponent = 1.0
+ if self.units and self.units != "px" and self.units != "" and self.units != "%":
+ if self.options.switcher == "0":
+ unit_exponent = 1.0 / (self.factor_a / self.__uuconv[self.units])
+ else:
+ unit_exponent = 1.0 / (self.factor_a / self.__uuconvLegacy[self.units])
+ self.scale_root(unit_exponent)
+
+
+if __name__ == '__main__':
+ DPISwitcher().run()