summaryrefslogtreecommitdiffstats
path: root/share/extensions/inkex/utils.py
diff options
context:
space:
mode:
Diffstat (limited to 'share/extensions/inkex/utils.py')
-rw-r--r--share/extensions/inkex/utils.py290
1 files changed, 290 insertions, 0 deletions
diff --git a/share/extensions/inkex/utils.py b/share/extensions/inkex/utils.py
new file mode 100644
index 0000000..dea5308
--- /dev/null
+++ b/share/extensions/inkex/utils.py
@@ -0,0 +1,290 @@
+# coding=utf-8
+#
+# Copyright (C) 2010 Nick Drobchenko, nick@cnc-club.ru
+# Copyright (C) 2005 Aaron Spike, aaron@ekips.org
+#
+# 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+#
+"""
+Basic common utility functions for calculated things
+"""
+from __future__ import absolute_import, print_function, unicode_literals
+
+import re
+import os
+import sys
+import shutil
+
+from itertools import tee
+from collections import defaultdict
+from argparse import ArgumentTypeError
+
+# When python2 support is gone, enable tempfile's version
+# from tempfile import TemporaryDirectory
+
+# All the names that get added to the inkex API itself.
+__all__ = ('AbortExtension', 'DependencyError', 'Boolean', 'errormsg', 'addNS', 'NSS')
+
+ABORT_STATUS = -5
+
+(X, Y) = range(2)
+PY3 = sys.version_info[0] == 3
+
+if PY3:
+ unicode = str # pylint: disable=redefined-builtin,invalid-name
+
+# a dictionary of all of the xmlns prefixes in a standard inkscape doc
+NSS = {
+ 'sodipodi': 'http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd',
+ 'cc': 'http://creativecommons.org/ns#',
+ 'ccOLD': 'http://web.resource.org/cc/',
+ 'svg': 'http://www.w3.org/2000/svg',
+ 'dc': 'http://purl.org/dc/elements/1.1/',
+ 'rdf': 'http://www.w3.org/1999/02/22-rdf-syntax-ns#',
+ 'inkscape': 'http://www.inkscape.org/namespaces/inkscape',
+ 'xlink': 'http://www.w3.org/1999/xlink',
+ 'xml': 'http://www.w3.org/XML/1998/namespace'
+}
+SSN = dict((b, a) for (a, b) in NSS.items())
+
+class KeyDict(dict):
+ """
+ A normal dictionary, except asking for anything not in the dictionary
+ always returns the key itself. This is used for translation dictionaries.
+ """
+ def __getitem__(self, key):
+ try:
+ return super(KeyDict, self).__getitem__(key)
+ except KeyError:
+ return key
+
+class TemporaryDirectory(object): # pylint: disable=too-few-public-methods
+ """Tiny replacement for python3's version."""
+ def __init__(self, suffix="", prefix="tmp"):
+ self.suffix = suffix
+ self.prefix = prefix
+ self.path = None
+ def __enter__(self):
+ from tempfile import mkdtemp
+ self.path = mkdtemp(self.suffix, self.prefix, None)
+ return self.path
+ def __exit__(self, exc, value, traceback):
+ if os.path.isdir(self.path):
+ shutil.rmtree(self.path)
+
+def Boolean(value):
+ """ArgParser function to turn a boolean string into a python boolean"""
+ if value.upper() == 'TRUE':
+ return True
+ elif value.upper() == 'FALSE':
+ return False
+ return None
+
+def to_bytes(content):
+ """Ensures the content is bytes"""
+ if isinstance(content, bytes):
+ return content
+ return str(content).encode("utf8")
+
+def debug(what):
+ """Print debug message if debugging is switched on"""
+ errormsg(what)
+ return what
+
+def do_nothing(*args, **kwargs): # pylint: disable=unused-argument
+ """A blank function to do nothing"""
+ pass
+
+def errormsg(msg):
+ """Intended for end-user-visible error messages.
+
+ (Currently just writes to stderr with an appended newline, but could do
+ something better in future: e.g. could add markup to distinguish error
+ messages from status messages or debugging output.)
+
+ Note that this should always be combined with translation:
+
+ import inkex
+ ...
+ inkex.errormsg(_("This extension requires two selected paths."))
+ """
+ try:
+ sys.stderr.write(msg)
+ except TypeError:
+ sys.stderr.write(unicode(msg))
+ except UnicodeEncodeError:
+ # Python 2:
+ # Fallback for cases where sys.stderr.encoding is not Unicode.
+ # Python 3:
+ # This will not work as write() does not accept byte strings, but AFAIK
+ # we should never reach this point as the default error handler is
+ # 'backslashreplace'.
+
+ # This will be None by default if stderr is piped, so use ASCII as a
+ # last resort.
+ encoding = sys.stderr.encoding or 'ascii'
+ sys.stderr.write(msg.encode(encoding, 'backslashreplace'))
+
+ # Write '\n' separately to avoid dealing with different string types.
+ sys.stderr.write('\n')
+
+
+class AbortExtension(Exception):
+ """Raised to print a message to the user without backtrace"""
+
+ def __init__(self, message=""):
+ self.message = message
+
+ def write(self):
+ """write the error message out to the user"""
+ errormsg(self.message)
+
+
+class DependencyError(NotImplementedError):
+ """Raised when we need an external python module that isn't available"""
+
+class FragmentError(Exception):
+ """Raised when trying to do rooty things on an xml fragment"""
+
+class InitSubClassPy3(type):
+ """Provide a poly-fill for python3 __init_subclass__()"""
+ def __init__(cls, name, bases, dct):
+ if '__metaclass__' not in cls.__dict__:
+ if hasattr(cls, '__init_subclass__'):
+ cls.__init_subclass__()
+ super(InitSubClassPy3, cls).__init__(name, bases, dct)
+
+def to(kind): # pylint: disable=invalid-name
+ """
+ Decorator which will turn a generator into a list, tuple or other object type.
+ """
+
+ def _inner(call):
+ def _outer(*args, **kw):
+ return kind(call(*args, **kw))
+
+ return _outer
+
+ return _inner
+
+
+def strargs(string, kind=float):
+ """Returns a list of floats from a string with commas or space separators,
+ also splits at -(minus) signs by adding a space in front of the - sign
+ """
+ return [kind(val) for val in string.replace(',', ' ').replace('-', ' -').replace('e ', 'e').split()]
+
+
+def addNS(tag, ns=None): # pylint: disable=invalid-name
+ """Add a known namespace to a name for use with lxml"""
+ if tag.startswith('{') and ns:
+ _, tag = removeNS(tag)
+ if not tag.startswith('{'):
+ tag = tag.replace('__', ':')
+ if ':' in tag:
+ (ns, tag) = tag.rsplit(':', 1)
+ if ns in NSS:
+ ns = NSS[ns]
+ if ns is not None:
+ return "{%s}%s" % (ns, tag)
+ return tag
+
+
+def removeNS(name): # pylint: disable=invalid-name
+ """The reverse of addNS, finds any namespace and returns tuple (ns, tag)"""
+ if name[0] == '{':
+ (url, tag) = name[1:].split('}', 1)
+ return SSN.get(url, 'svg'), tag
+ if ':' in name:
+ return name.rsplit(':', 1)
+ return 'svg', name
+
+def splitNS(name): # pylint: disable=invalid-name
+ """Like removeNS, but returns a url instead of a prefix"""
+ (prefix, tag) = removeNS(name)
+ return (NSS[prefix], tag)
+
+class classproperty(object): # pylint: disable=invalid-name, too-few-public-methods
+ """Combine classmethod and property decorators"""
+
+ def __init__(self, func):
+ self.func = func
+
+ def __get__(self, obj, owner):
+ return self.func(owner)
+
+
+def filename_arg(name):
+ """Existing file to read or option used in script arguments"""
+ filename = os.path.abspath(os.path.expanduser(name))
+ if not os.path.isfile(filename):
+ raise ArgumentTypeError("File not found: {}".format(name))
+ return filename
+
+def pairwise(iterable, start=True):
+ "Iterate over a list with overlapping pairs (see itertools recipes)"
+ first, then = tee(iterable)
+ starter = [(None, next(then, None))]
+ if not start:
+ starter = []
+ return starter + list(zip(first, then))
+
+class CloningVat(object):
+ """
+ When modifying defs, sometimes we want to know if every backlink would have
+ needed changing, or it was just some of them.
+
+ This tracks the def elements, their promises and creates clones if needed.
+ """
+ def __init__(self, svg):
+ self.svg = svg
+ self.tracks = defaultdict(set)
+ self.set_ids = defaultdict(list)
+
+ def track(self, elem, parent, set_id=None, **kwargs):
+ """Track the element and connected parent"""
+ elem_id = elem.get('id')
+ parent_id = parent.get('id')
+ self.tracks[elem_id].add(parent_id)
+ self.set_ids[elem_id].append((set_id, kwargs))
+
+ def process(self, process, types=(), make_clones=True, **kwargs):
+ """
+ Process each tracked item if the backlinks match the parents
+
+ Optionally make clones, process the clone and set the new id.
+ """
+ for elem_id in list(self.tracks):
+ parents = self.tracks[elem_id]
+ elem = self.svg.getElementById(elem_id)
+ backlinks = set([blk.get('id') for blk in elem.backlinks(*types)])
+ if backlinks == parents:
+ # No need to clone, we're processing on-behalf of all parents
+ process(elem, **kwargs)
+ elif make_clones:
+ clone = elem.copy()
+ elem.getparent().append(clone)
+ clone.set_random_id()
+ for update, upkw in self.set_ids.get(elem_id, ()):
+ update(elem.get('id'), clone.get('id'), **upkw)
+ process(clone, **kwargs)
+
+def fullmatch(regex, string, flags=0):
+ """Emulate python-3.4 re.fullmatch()."""
+ if hasattr(regex, 'fullmatch'):
+ return regex.fullmatch(string, flags)
+ if hasattr(regex, 'pattern'):
+ regex = regex.pattern
+ return re.match("(?:" + regex + r")\Z", string, flags=flags)