diff options
Diffstat (limited to 'share/extensions/jitternodes.py')
-rwxr-xr-x | share/extensions/jitternodes.py | 107 |
1 files changed, 107 insertions, 0 deletions
diff --git a/share/extensions/jitternodes.py b/share/extensions/jitternodes.py new file mode 100755 index 0000000..7fcc69d --- /dev/null +++ b/share/extensions/jitternodes.py @@ -0,0 +1,107 @@ +#!/usr/bin/env python +# coding=utf-8 +# +# Copyright (C) 2012 Juan Pablo Carbajal ajuanpi-dev@gmail.com +# 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 3 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. +# + +import math +import random +import inkex + + +class JitterNodes(inkex.EffectExtension): + """Jiggle nodes around""" + + def add_arguments(self, pars): + pars.add_argument("--tab") + pars.add_argument("--radiusx", type=float, default=10.0, help="Randum radius X") + pars.add_argument("--radiusy", type=float, default=10.0, help="Randum radius Y") + pars.add_argument( + "--ctrl", type=inkex.Boolean, default=False, help="Randomize ctrl points" + ) + pars.add_argument( + "--end", type=inkex.Boolean, default=True, help="Randomize nodes" + ) + pars.add_argument( + "--dist", + type=self.arg_method("dist"), + default=self.dist_uniform, + help="Distribution of displacement", + ) + + def effect(self): + for node in self.svg.selection.filter(inkex.PathElement): + path = node.path.to_superpath() + for subpath in path: + closed = subpath[0] == subpath[-1] + for index, csp in enumerate(subpath): + if closed and index == len(subpath) - 1: + subpath[index] = subpath[0] + break + if self.options.end: + delta = self.randomize([0, 0]) + csp[0][0] += delta[0] + csp[0][1] += delta[1] + csp[1][0] += delta[0] + csp[1][1] += delta[1] + csp[2][0] += delta[0] + csp[2][1] += delta[1] + if self.options.ctrl: + csp[0] = self.randomize(csp[0]) + csp[2] = self.randomize(csp[2]) + node.path = path + + def randomize(self, pos): + """Randomise the given position [x, y] as set in the options""" + delta = self.options.dist(self.options.radiusx, self.options.radiusy) + return [pos[0] + delta[0], pos[1] + delta[1]] + + @staticmethod + def dist_gaussian(x, y): + """Gaussian distribution""" + return random.gauss(0.0, x), random.gauss(0.0, y) + + @staticmethod + def dist_pareto(x, y): + """Pareto distribution""" + # sign is used to fake a double sided pareto distribution. + # for parameter value between 1 and 2 the distribution has infinite variance + # I truncate the distribution to a high value and then normalize it. + # The idea is to get spiky distributions, any distribution with long-tails is + # good (ideal would be Levy distribution). + sign = random.uniform(-1.0, 1.0) + return x * math.copysign( + min(random.paretovariate(1.0), 20.0) / 20.0, sign + ), y * math.copysign(min(random.paretovariate(1.0), 20.0) / 20.0, sign) + + @staticmethod + def dist_lognorm(x, y): + """Log Norm distribution""" + sign = random.uniform(-1.0, 1.0) + return x * math.copysign( + random.lognormvariate(0.0, 1.0) / 3.5, sign + ), y * math.copysign(random.lognormvariate(0.0, 1.0) / 3.5, sign) + + @staticmethod + def dist_uniform(x, y): + """Uniform distribution""" + return random.uniform(-x, x), random.uniform(-y, y) + + +if __name__ == "__main__": + JitterNodes().run() |