kopia lustrzana https://github.com/inkstitch/inkstitch
97 wiersze
2.8 KiB
Python
97 wiersze
2.8 KiB
Python
import inkex
|
|
from lxml import etree
|
|
|
|
from .tags import SVG_GROUP_TAG, SVG_LINK_TAG
|
|
from .units import get_viewbox_transform
|
|
|
|
|
|
def apply_transforms(path, node):
|
|
transform = get_node_transform(node)
|
|
|
|
# apply the combined transform to this node's path
|
|
path = path.transform(transform)
|
|
|
|
return path
|
|
|
|
|
|
def compose_parent_transforms(node, mat):
|
|
# This is adapted from Inkscape's simpletransform.py's composeParents()
|
|
# function. That one can't handle nodes that are detached from a DOM.
|
|
|
|
trans = node.get('transform')
|
|
if trans:
|
|
mat = inkex.transforms.Transform(trans) * mat
|
|
if node.getparent() is not None:
|
|
if node.getparent().tag in [SVG_GROUP_TAG, SVG_LINK_TAG]:
|
|
mat = compose_parent_transforms(node.getparent(), mat)
|
|
return mat
|
|
|
|
|
|
def get_node_transform(node):
|
|
"""
|
|
if getattr(node, "composed_transform", None):
|
|
return node.composed_transform()
|
|
"""
|
|
|
|
# start with the identity transform
|
|
transform = inkex.transforms.Transform([[1.0, 0.0, 0.0], [0.0, 1.0, 0.0]])
|
|
|
|
# this if is because sometimes inkscape likes to create paths outside of a layer?!
|
|
if node.getparent() is not None:
|
|
# combine this node's transform with all parent groups' transforms
|
|
transform = compose_parent_transforms(node, transform)
|
|
|
|
# add in the transform implied by the viewBox
|
|
viewbox_transform = get_viewbox_transform(node.getroottree().getroot())
|
|
transform = viewbox_transform * transform
|
|
|
|
return transform
|
|
|
|
|
|
def get_correction_transform(node, child=False):
|
|
"""Get a transform to apply to new siblings or children of this SVG node"""
|
|
|
|
# if we want to place our new nodes in the same group/layer as this node,
|
|
# then we'll need to factor in the effects of any transforms set on
|
|
# the parents of this node.
|
|
|
|
if child:
|
|
transform = get_node_transform(node)
|
|
else:
|
|
# we can ignore the transform on the node itself since it won't apply
|
|
# to the objects we add
|
|
transform = get_node_transform(node.getparent())
|
|
|
|
# now invert it, so that we can position our objects in absolute
|
|
# coordinates
|
|
transform = -transform
|
|
|
|
return str(transform)
|
|
|
|
|
|
def line_strings_to_csp(line_strings):
|
|
return point_lists_to_csp(ls.coords for ls in line_strings)
|
|
|
|
|
|
def point_lists_to_csp(point_lists):
|
|
csp = []
|
|
|
|
for point_list in point_lists:
|
|
subpath = []
|
|
for point in point_list:
|
|
# cubicsuperpath is very particular that these must be lists, not tuples
|
|
point = list(point)
|
|
# create a straight line as a degenerate bezier
|
|
subpath.append([point, point, point])
|
|
csp.append(subpath)
|
|
|
|
return csp
|
|
|
|
|
|
def line_strings_to_path(line_strings):
|
|
csp = line_strings_to_csp(line_strings)
|
|
|
|
return etree.Element("path", {
|
|
"d": str(inkex.paths.CubicSuperPath(csp))
|
|
})
|