2020-05-16 21:12:06 +00:00
|
|
|
import logging
|
|
|
|
from copy import copy
|
2020-04-19 16:38:28 +00:00
|
|
|
|
2020-05-16 21:12:06 +00:00
|
|
|
from shapely.geometry import LineString, MultiPolygon, Polygon
|
|
|
|
from shapely.ops import polygonize, unary_union
|
2020-04-19 16:38:28 +00:00
|
|
|
|
|
|
|
import inkex
|
|
|
|
|
2020-05-16 21:12:06 +00:00
|
|
|
from ..elements import EmbroideryElement
|
2020-04-19 16:38:28 +00:00
|
|
|
from ..i18n import _
|
|
|
|
from ..svg import get_correction_transform
|
2020-05-16 21:12:06 +00:00
|
|
|
from ..svg.tags import SVG_PATH_TAG
|
2020-04-19 16:38:28 +00:00
|
|
|
from .base import InkstitchExtension
|
|
|
|
|
|
|
|
|
|
|
|
class BreakApart(InkstitchExtension):
|
2020-05-16 21:12:06 +00:00
|
|
|
'''
|
|
|
|
This will break apart fill areas into separate elements.
|
|
|
|
'''
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
InkstitchExtension.__init__(self, *args, **kwargs)
|
|
|
|
self.OptionParser.add_option("-m", "--method", type="int", default=1, dest="method")
|
2020-04-19 16:38:28 +00:00
|
|
|
|
2020-05-16 21:12:06 +00:00
|
|
|
def effect(self):
|
2020-04-19 16:38:28 +00:00
|
|
|
if not self.selected:
|
|
|
|
inkex.errormsg(_("Please select one or more fill areas to break apart."))
|
|
|
|
return
|
|
|
|
|
2020-05-16 21:12:06 +00:00
|
|
|
elements = []
|
|
|
|
nodes = self.get_nodes()
|
|
|
|
for node in nodes:
|
|
|
|
if node.tag in SVG_PATH_TAG:
|
|
|
|
elements.append(EmbroideryElement(node))
|
|
|
|
|
|
|
|
for element in elements:
|
|
|
|
if not element.get_style("fill", "black"):
|
2020-04-19 16:38:28 +00:00
|
|
|
continue
|
2020-05-16 21:12:06 +00:00
|
|
|
|
|
|
|
# we don't want to touch valid elements
|
|
|
|
paths = element.flatten(element.parse_path())
|
|
|
|
paths.sort(key=lambda point_list: Polygon(point_list).area, reverse=True)
|
|
|
|
polygon = MultiPolygon([(paths[0], paths[1:])])
|
|
|
|
if self.geom_is_valid(polygon):
|
2020-04-19 16:38:28 +00:00
|
|
|
continue
|
|
|
|
|
2020-05-16 21:12:06 +00:00
|
|
|
polygons = self.break_apart_paths(paths)
|
|
|
|
polygons = self.ensure_minimum_size(polygons, 5)
|
|
|
|
if self.options.method == 1:
|
|
|
|
polygons = self.combine_overlapping_polygons(polygons)
|
|
|
|
polygons = self.recombine_polygons(polygons)
|
|
|
|
if polygons:
|
|
|
|
self.polygons_to_nodes(polygons, element)
|
2020-04-19 16:38:28 +00:00
|
|
|
|
2020-05-16 21:12:06 +00:00
|
|
|
def break_apart_paths(self, paths):
|
|
|
|
polygons = []
|
|
|
|
for path in paths:
|
|
|
|
linestring = LineString(path)
|
|
|
|
polygon = Polygon(path).buffer(0)
|
|
|
|
if not linestring.is_simple:
|
|
|
|
linestring = unary_union(linestring)
|
|
|
|
for polygon in polygonize(linestring):
|
|
|
|
polygons.append(polygon)
|
|
|
|
else:
|
|
|
|
polygons.append(polygon)
|
|
|
|
return polygons
|
|
|
|
|
|
|
|
def combine_overlapping_polygons(self, polygons):
|
|
|
|
for polygon in polygons:
|
|
|
|
for other in polygons:
|
|
|
|
if polygon == other:
|
|
|
|
continue
|
|
|
|
if polygon.overlaps(other):
|
|
|
|
diff = polygon.symmetric_difference(other)
|
|
|
|
if diff.geom_type == 'MultiPolygon':
|
|
|
|
polygons.remove(other)
|
|
|
|
polygons.remove(polygon)
|
|
|
|
for p in diff:
|
|
|
|
polygons.append(p)
|
|
|
|
# it is possible, that a polygons overlap with multiple
|
|
|
|
# polygons, this means, we need to start all over again
|
|
|
|
polygons = self.combine_overlapping_polygons(polygons)
|
|
|
|
return polygons
|
|
|
|
return polygons
|
2020-04-19 16:38:28 +00:00
|
|
|
|
2020-05-16 21:12:06 +00:00
|
|
|
def geom_is_valid(self, geom):
|
|
|
|
# Don't complain about invalid shapes, we just want to know
|
|
|
|
logger = logging.getLogger('shapely.geos')
|
|
|
|
level = logger.level
|
|
|
|
logger.setLevel(logging.CRITICAL)
|
|
|
|
valid = geom.is_valid
|
|
|
|
logger.setLevel(level)
|
|
|
|
return valid
|
2020-04-19 16:38:28 +00:00
|
|
|
|
2020-05-16 21:12:06 +00:00
|
|
|
def ensure_minimum_size(self, polygons, size):
|
|
|
|
for polygon in polygons:
|
|
|
|
if polygon.area < size:
|
|
|
|
polygons.remove(polygon)
|
|
|
|
return polygons
|
|
|
|
|
|
|
|
def recombine_polygons(self, polygons):
|
|
|
|
polygons.sort(key=lambda polygon: polygon.area, reverse=True)
|
|
|
|
multipolygons = []
|
|
|
|
holes = []
|
|
|
|
for polygon in polygons:
|
|
|
|
if polygon in holes:
|
|
|
|
continue
|
|
|
|
polygon_list = [polygon]
|
|
|
|
for other in polygons:
|
|
|
|
if polygon == other:
|
2020-04-19 16:38:28 +00:00
|
|
|
continue
|
2020-05-16 21:12:06 +00:00
|
|
|
if polygon.contains(other) and other not in holes:
|
|
|
|
if any(p.contains(other) or p.intersects(other) for p in polygon_list[1:]):
|
|
|
|
continue
|
|
|
|
holes.append(other)
|
|
|
|
# if possible let's make the hole a tiny little bit smaller, just in case, it hits the edge
|
|
|
|
# and would lead therefore to an invalid shape
|
|
|
|
o = other.buffer(-0.01)
|
|
|
|
if not o.is_empty and o.geom_type == 'Polygon':
|
|
|
|
other = o
|
|
|
|
polygon_list.append(other)
|
|
|
|
multipolygons.append(polygon_list)
|
|
|
|
return multipolygons
|
|
|
|
|
|
|
|
def polygons_to_nodes(self, polygon_list, element):
|
|
|
|
# reverse the list of polygons, we don't want to cover smaller shapes
|
|
|
|
polygon_list = polygon_list[::-1]
|
|
|
|
index = element.node.getparent().index(element.node)
|
|
|
|
for polygons in polygon_list:
|
|
|
|
if polygons[0].area < 5:
|
|
|
|
continue
|
|
|
|
el = copy(element.node)
|
2020-04-19 16:38:28 +00:00
|
|
|
d = ""
|
|
|
|
for polygon in polygons:
|
2020-05-16 21:12:06 +00:00
|
|
|
# update element id
|
|
|
|
if len(polygon_list) > 1:
|
|
|
|
node_id = self.uniqueId(el.get('id') + '_')
|
|
|
|
el.set('id', node_id)
|
2020-04-19 16:38:28 +00:00
|
|
|
d += "M"
|
|
|
|
for x, y in polygon.exterior.coords:
|
|
|
|
d += "%s,%s " % (x, y)
|
|
|
|
d += " "
|
|
|
|
d += "Z"
|
2020-05-16 21:12:06 +00:00
|
|
|
el.set('d', d)
|
|
|
|
el.set('transform', get_correction_transform(element.node))
|
|
|
|
element.node.getparent().insert(index, el)
|
2020-04-19 16:38:28 +00:00
|
|
|
element.node.getparent().remove(element.node)
|