summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorGeorge Steel <george.steel@gmail.com>2022-11-24 23:50:00 -0500
committerGeorge Steel <george.steel@gmail.com>2022-11-24 23:50:00 -0500
commit495a22ea55234e032ac526450fd7969d9aa799c8 (patch)
tree15b58fd0fbb116e9529d8b0663d656109e710be0
parentac655910eb4d283e3c00b27260fd1d2dbecd903e (diff)
parent72b001e7c227131a01d3bf3ae8678b704127de77 (diff)
Merge commit '72b001e7' into george-steel/random-base-satin
-rw-r--r--icons/randomize_20x20.pngbin0 -> 697 bytes
-rw-r--r--lib/elements/satin_column.py100
-rw-r--r--lib/extensions/params.py25
-rw-r--r--lib/svg/tags.py7
4 files changed, 126 insertions, 6 deletions
diff --git a/icons/randomize_20x20.png b/icons/randomize_20x20.png
new file mode 100644
index 00000000..40d8e88b
--- /dev/null
+++ b/icons/randomize_20x20.png
Binary files differ
diff --git a/lib/elements/satin_column.py b/lib/elements/satin_column.py
index e4d69361..3a183ab0 100644
--- a/lib/elements/satin_column.py
+++ b/lib/elements/satin_column.py
@@ -2,17 +2,16 @@
#
# Copyright (c) 2010 Authors
# Licensed under the GNU GPL version 3.0 or later. See the file LICENSE for details.
-
+import random
from copy import deepcopy
from itertools import chain
import numpy as np
+from inkex import paths
from shapely import affinity as shaffinity
from shapely import geometry as shgeo
from shapely.ops import nearest_points
-from inkex import paths
-
from ..i18n import _
from ..stitch_plan import StitchGroup
from ..svg import line_strings_to_csp, point_lists_to_csp
@@ -101,6 +100,46 @@ class SatinColumn(EmbroideryElement):
return self.get_float_param("max_stitch_length_mm") or None
@property
+ @param('random_split_factor',
+ _('Random Split Factor'),
+ tooltip=_('randomize position for split stitches.'),
+ type='int', unit="%", sort_index=70)
+ def random_split_factor(self):
+ return min(max(self.get_int_param("random_split_factor", 0), 0), 100)
+
+ @property
+ @param('random_first_rail_factor_in',
+ _('First Rail Random Factor inside'),
+ tooltip=_('shorten stitch around first rail at most this percent.'),
+ type='int', unit="%", sort_index=60)
+ def random_first_rail_factor_in(self):
+ return min(max(self.get_int_param("random_first_rail_factor_in", 0), 0), 100)
+
+ @property
+ @param('random_first_rail_factor_out',
+ _('First Rail Random Factor outside'),
+ tooltip=_('lengthen stitch around first rail at most this percent.'),
+ type='int', unit="%", sort_index=61)
+ def random_first_rail_factor_out(self):
+ return max(self.get_int_param("random_first_rail_factor_out", 0), 0)
+
+ @property
+ @param('random_second_rail_factor_in',
+ _('Second Rail Random Factor inside'),
+ tooltip=_('shorten stitch around second rail at most this percent.'),
+ type='int', unit="%", sort_index=62)
+ def random_second_rail_factor_in(self):
+ return min(max(self.get_int_param("random_second_rail_factor_in", 0), 0), 100)
+
+ @property
+ @param('random_second_rail_factor_out',
+ _('Second Rail Random Factor outside'),
+ tooltip=_('lengthen stitch around second rail at most this percent.'),
+ type='int', unit="%", sort_index=63)
+ def random_second_rail_factor_out(self):
+ return max(self.get_int_param("random_second_rail_factor_out", 0), 0)
+
+ @property
@param('short_stitch_inset',
_('Short stitch inset'),
tooltip=_('Stitches in areas with high density will be shortened by this amount.'),
@@ -134,6 +173,15 @@ class SatinColumn(EmbroideryElement):
return max(self.get_float_param("zigzag_spacing_mm", 0.4), 0.01)
@property
+ @param('random_zigzag_spacing',
+ _('Zig-zag spacing randomness(peak-to-peak)'),
+ tooltip=_('percentage of randomness of Peak-to-peak distance between zig-zags.'),
+ type='int', unit="%", sort_index=64)
+ def random_zigzag_spacing(self):
+ # peak-to-peak distance between zigzags
+ return max(self.get_int_param("random_zigzag_spacing", 0), 0)
+
+ @property
@param(
'pull_compensation_percent',
_('Pull compensation percentage'),
@@ -306,6 +354,10 @@ class SatinColumn(EmbroideryElement):
return self.get_float_param("zigzag_underlay_max_stitch_length_mm") or None
@property
+ def use_seed(self):
+ return self.get_int_param("use_seed", 0)
+
+ @property
@cache
def shape(self):
# This isn't used for satins at all, but other parts of the code
@@ -552,6 +604,22 @@ class SatinColumn(EmbroideryElement):
for rung in self.rungs:
point_lists.append(self.flatten_subpath(rung))
+ # If originally there were only two subpaths (no rungs) with same number of rails, we may the rails may now
+ # have two rails with different number of points, and still no rungs, let's add one.
+
+ if not self.rungs:
+ rails = [shgeo.LineString(reversed(self.flatten_subpath(rail))) for rail in self.rails]
+ rails.reverse()
+ path_list = rails
+
+ rung_start = path_list[0].interpolate(0.1)
+ rung_end = path_list[1].interpolate(0.1)
+ rung = shgeo.LineString((rung_start, rung_end))
+ # make it a bit bigger so that it definitely intersects
+ rung = shaffinity.scale(rung, 1.1, 1.1)
+ path_list.append(rung)
+ return (self._path_list_to_satins(path_list))
+
return self._csp_to_satin(point_lists_to_csp(point_lists))
def apply_transform(self):
@@ -830,8 +898,13 @@ class SatinColumn(EmbroideryElement):
old_center = new_center
if to_travel <= 0:
- add_pair(pos0, pos1)
- to_travel = spacing
+
+ decalage0 = random.uniform(-self.random_first_rail_factor_in, self.random_first_rail_factor_out) / 100
+ decalage1 = random.uniform(-self.random_second_rail_factor_in, self.random_second_rail_factor_out) / 100
+
+ add_pair(pos0 + (pos0 - pos1) * decalage0, pos1 + (pos1 - pos0) * decalage1)
+
+ to_travel = spacing * (random.uniform(1, 1 + self.random_zigzag_spacing/100))
if to_travel > 0:
add_pair(pos0, pos1)
@@ -1009,7 +1082,12 @@ class SatinColumn(EmbroideryElement):
split_count = count or int(-(-distance // max_stitch_length))
for i in range(split_count):
line = shgeo.LineString((left, right))
- split_point = line.interpolate((i+1)/split_count, normalized=True)
+
+ random_move = 0
+ if self.random_split_factor and i != split_count-1:
+ random_move = random.uniform(-self.random_split_factor / 100, self.random_split_factor / 100)
+
+ split_point = line.interpolate((i + 1 + random_move) / split_count, normalized=True)
points.append(Point(split_point.x, split_point.y))
return [points, split_count]
@@ -1034,6 +1112,16 @@ class SatinColumn(EmbroideryElement):
# beziers. The boundary points between beziers serve as "checkpoints",
# allowing the user to control how the zigzags flow around corners.
+ # If no seed is defined, compute one randomly using time to seed, otherwise, use stored seed
+
+ if self.use_seed == 0:
+ random.seed()
+ x = random.randint(1, 10000)
+ random.seed(x)
+ self.set_param("use_seed", x)
+ else:
+ random.seed(self.use_seed)
+
patch = StitchGroup(color=self.color)
if self.center_walk_underlay:
diff --git a/lib/extensions/params.py b/lib/extensions/params.py
index df62128f..7d38b5ff 100644
--- a/lib/extensions/params.py
+++ b/lib/extensions/params.py
@@ -6,6 +6,7 @@
# -*- coding: UTF-8 -*-
import os
+import random
import sys
from collections import defaultdict
from copy import copy
@@ -78,6 +79,9 @@ class ParamsTab(ScrolledPanel):
self.pencil_icon = wx.Image(os.path.join(get_resource_dir(
"icons"), "pencil_20x20.png")).ConvertToBitmap()
+ self.randomize_icon = wx.Image(os.path.join(get_resource_dir(
+ "icons"), "randomize_20x20.png")).ConvertToBitmap()
+
self.__set_properties()
self.__do_layout()
@@ -187,6 +191,16 @@ class ParamsTab(ScrolledPanel):
return values
+ def on_change_seed(self, event):
+
+ for node in self.nodes:
+ random.seed()
+ new_seed = random.randint(1, 10000)
+ node.set_param("use_seed", new_seed)
+ if self.on_change_hook:
+ self.on_change_hook(self)
+ event.Skip()
+
def apply(self):
values = self.get_values()
for node in self.nodes:
@@ -379,6 +393,17 @@ class ParamsTab(ScrolledPanel):
self.inputs_to_params = {v: k for k, v in self.param_inputs.items()}
box.Add(self.settings_grid, proportion=1, flag=wx.ALL, border=10)
+
+ add_seed_button = False
+ for param in self.params:
+ if param.name[:6] == "random":
+ add_seed_button = True
+ if add_seed_button:
+ self.change_seed_button = wx.Button(self, wx.ID_ANY, _("Change Seed"))
+ self.change_seed_button.Bind(wx.EVT_BUTTON, self.on_change_seed)
+ self.change_seed_button.SetBitmap(self.randomize_icon)
+ box.Add(self.change_seed_button, proportion=0, flag=wx.ALIGN_CENTER_HORIZONTAL, border=10)
+
self.SetSizer(box)
self.update_choice_widgets()
diff --git a/lib/svg/tags.py b/lib/svg/tags.py
index 850256ee..4743438b 100644
--- a/lib/svg/tags.py
+++ b/lib/svg/tags.py
@@ -123,6 +123,13 @@ inkstitch_attribs = [
'pull_compensation_mm',
'pull_compensation_percent',
'stroke_first',
+ 'random_split_factor',
+ 'random_first_rail_factor_in',
+ 'random_first_rail_factor_out',
+ 'random_second_rail_factor_in',
+ 'random_second_rail_factor_out',
+ 'random_zigzag_spacing',
+ 'use_seed',
# stitch_plan
'invisible_layers',
# Legacy