1
0
Fork 0
inkscape/share/extensions/spirograph.py
Daniel Baumann 02d935e272
Adding upstream version 1.4.
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
2025-06-22 23:40:13 +02:00

151 lines
4.8 KiB
Python
Executable file

#! /usr/bin/env python3
# coding=utf-8
#
# Copyright (C) 2007 Joel Holdsworth joel@airwebreathe.org.uk
#
# 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.
#
import math
import inkex
class Spirograph(inkex.EffectExtension):
def add_arguments(self, pars):
pars.add_argument(
"--primaryr", type=float, default=100.0, help="The radius of the outer gear"
)
pars.add_argument(
"--secondaryr",
type=float,
default=60.0,
help="The radius of the inner gear",
)
pars.add_argument(
"--penr",
type=float,
default=50.0,
help="The distance of the pen from the inner gear",
)
pars.add_argument(
"--gearplacement",
default="inside",
help="Selects whether the gear is inside or outside the ring",
)
pars.add_argument(
"--rotation",
type=float,
default=0.0,
help="The number of degrees to rotate the image by",
)
pars.add_argument(
"--quality",
type=int,
default=16,
help="The quality of the calculated output",
)
def effect(self):
self.options.primaryr = self.svg.unittouu(str(self.options.primaryr) + "px")
self.options.secondaryr = self.svg.unittouu(str(self.options.secondaryr) + "px")
self.options.penr = self.svg.unittouu(str(self.options.penr) + "px")
if self.options.secondaryr == 0:
return
if self.options.quality == 0:
return
if self.options.gearplacement.strip(" ").lower().startswith("outside"):
a = self.options.primaryr + self.options.secondaryr
flip = -1
else:
a = self.options.primaryr - self.options.secondaryr
flip = 1
ratio = a / self.options.secondaryr
if ratio == 0:
return
scale = 2 * math.pi / (ratio * self.options.quality)
rotation = -math.pi * self.options.rotation / 180
new = inkex.PathElement()
new.style = inkex.Style(stroke="#000000", fill="none", stroke_width="1.0")
path_string = ""
maxPointCount = 1000
for i in range(maxPointCount):
theta = i * scale
view_center = self.svg.namedview.center
x = (
a * math.cos(theta + rotation)
+ self.options.penr * math.cos(ratio * theta + rotation) * flip
+ view_center[0]
)
y = (
a * math.sin(theta + rotation)
- self.options.penr * math.sin(ratio * theta + rotation)
+ view_center[1]
)
dx = (
(
-a * math.sin(theta + rotation)
- ratio
* self.options.penr
* math.sin(ratio * theta + rotation)
* flip
)
* scale
/ 3
)
dy = (
(
a * math.cos(theta + rotation)
- ratio * self.options.penr * math.cos(ratio * theta + rotation)
)
* scale
/ 3
)
if i <= 0:
path_string += "M {},{} C {},{} ".format(
str(x), str(y), str(x + dx), str(y + dy)
)
else:
path_string += "{},{} {},{}".format(
str(x - dx), str(y - dy), str(x), str(y)
)
if (
math.fmod(i / ratio, self.options.quality) == 0
and i % self.options.quality == 0
):
path_string += "Z"
break
else:
if i == maxPointCount - 1:
pass # we reached the allowed maximum of points, stop here
else:
path_string += " C {},{} ".format(str(x + dx), str(y + dy))
new.path = path_string
self.svg.get_current_layer().append(new)
if __name__ == "__main__":
Spirograph().run()