summaryrefslogtreecommitdiffstats
path: root/share/extensions/jitternodes.py
diff options
context:
space:
mode:
Diffstat (limited to '')
-rwxr-xr-xshare/extensions/jitternodes.py107
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()