diff options
Diffstat (limited to 'share/extensions/synfig_prepare.py')
-rwxr-xr-x | share/extensions/synfig_prepare.py | 323 |
1 files changed, 323 insertions, 0 deletions
diff --git a/share/extensions/synfig_prepare.py b/share/extensions/synfig_prepare.py new file mode 100755 index 0000000..9899eec --- /dev/null +++ b/share/extensions/synfig_prepare.py @@ -0,0 +1,323 @@ +#!/usr/bin/env python +# coding=utf-8 +# +# Copyright (C) 2011 Nikita Kitaev +# +# 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 +# +""" +Simplifies SVG files in preparation for sif export. +""" + +import os +import tempfile +from subprocess import PIPE, Popen +from inkex.command import inkscape, write_svg, ProgramRunError +from inkex.localization import inkex_gettext as _ +from inkex.base import TempDirMixin + +import inkex +from inkex import ( + load_svg, + Group, + PathElement, + ShapeElement, + Anchor, + Switch, + SvgDocumentElement, + Transform, +) + +###### Utility Classes #################################### + + +class MalformedSVGError(Exception): + """Raised when the SVG document is invalid or contains unsupported features""" + + def __init__(self, value): + self.value = value + + def __str__(self): + return ( + _( + """SVG document is invalid or contains unsupported features + +Error message: %s + +The SVG to Synfig converter is designed to handle SVG files that were created using Inkscape. Unsupported features are most likely to occur in SVG files written by other programs. +""" + ) + % repr(self.value) + ) + + +###### Utility Functions ################################## + +### Path related + + +def fuse_subpaths(path: inkex.Path): + """Fuse subpaths of a path. Should only be used on unstroked paths. + The idea is to replace every moveto by a lineto, and then walk all the extra lines + backwards to get the same fill. For unfilled paths, this gives visually good results + in cases with not-too-complex paths, i.e. no intersections. + There may be extra zero-length Lineto commands.""" + + result = inkex.Path() + return_stack = [] + for i, seg in enumerate(path.proxy_iterator()): + if seg.letter not in "zZmM": + result.append(seg.command) + elif i == 0: + result.append(seg.command) + first = seg.end_point + else: + # Add a line instead of the ZoneClose / Move command, and store + # the initial position. + return_to = seg.previous_end_point + if seg.letter in "mM": + if seg.previous_end_point != first: # only close if needed + # all paths must be closed for this algorithm to work. Since + # the path doesn't have a stroke, it's visually irrellevant. + result.append(inkex.paths.Line(*first)) + return_to = first + + return_stack += [return_to] + result.append(inkex.paths.Line(*seg.end_point)) + + first = seg.end_point + + # also close the last subpath + return_stack += [first] + + # now apply the return stack backwards + for point in return_stack[::-1]: + result.append(inkex.paths.Line(*point)) + + return result + + +def split_fill_and_stroke(path_node): + """Split a path into two paths, one filled and one stroked + + Returns a the list [fill, stroke], where each is the XML element of the + fill or stroke, or None. + """ + style = dict(inkex.Style.parse_str(path_node.get("style", ""))) + + # If there is only stroke or only fill, don't split anything + if "fill" in style and style["fill"] == "none": + if "stroke" not in style or style["stroke"] == "none": + return [None, None] # Path has neither stroke nor fill + else: + return [None, path_node] + if "stroke" not in style.keys() or style["stroke"] == "none": + return [path_node, None] + + group = Group() + fill = group.add(PathElement()) + stroke = group.add(PathElement()) + + d = path_node.pop("d") + if d is None: + raise AssertionError("Cannot split stroke and fill of non-path element") + + nodetypes = path_node.pop("sodipodi:nodetypes", None) + path_id = path_node.pop("id", str(id(path_node))) + transform = path_node.pop("transform", None) + path_node.pop("style") + + # Pass along all remaining attributes to the group + for attrib_name, attrib_value in path_node.attrib.items(): + group.set(attrib_name, attrib_value) + + group.set("id", path_id) + + # Next split apart the style attribute + style_group = {} + style_fill = {"stroke": "none", "fill": "#000000"} + style_stroke = {"fill": "none", "stroke": "none"} + + for key in style.keys(): + if key.startswith("fill"): + style_fill[key] = style[key] + elif key.startswith("stroke"): + style_stroke[key] = style[key] + elif key.startswith("marker"): + style_stroke[key] = style[key] + elif key.startswith("filter"): + style_group[key] = style[key] + else: + style_fill[key] = style[key] + style_stroke[key] = style[key] + + if len(style_group) != 0: + group.set("style", str(inkex.Style(style_group))) + + fill.set("style", str(inkex.Style(style_fill))) + stroke.set("style", str(inkex.Style(style_stroke))) + + # Finalize the two paths + fill.set("d", d) + stroke.set("d", d) + if nodetypes is not None: + fill.set("sodipodi:nodetypes", nodetypes) + stroke.set("sodipodi:nodetypes", nodetypes) + fill.set("id", path_id + "-fill") + stroke.set("id", path_id + "-stroke") + if transform is not None: + fill.set("transform", transform) + stroke.set("transform", transform) + + # Replace the original node with the group + path_node.getparent().replace(path_node, group) + + return [fill, stroke] + + +### Object related + + +def propagate_attribs( + node, parent_style={}, parent_transform=[[1.0, 0.0, 0.0], [0.0, 1.0, 0.0]] +): + """Propagate style and transform to remove inheritance""" + + # Don't enter non-graphical portions of the document + if not isinstance(node, (ShapeElement, SvgDocumentElement)): + return + + # Compose the transformations + if isinstance(node, SvgDocumentElement) and node.get("viewBox"): + vx, vy, vw, vh = [get_dimension(x) for x in node.get_viewbox()] + dw = get_dimension(node.get("width", vw)) + dh = get_dimension(node.get("height", vh)) + this_transform = Transform(translate=(-vx, -vy), scale=(dw / vw, dh / vh)) + del node.attrib["viewBox"] + else: + this_transform = Transform(parent_transform) + + this_transform @= node.transform + + # Compose the style attribs + this_style = dict(inkex.Style.parse_str(node.get("style", ""))) + remaining_style = {} # Style attributes that are not propagated + + non_propagated = ["filter"] # Filters should remain on the topmost ancestor + for key in non_propagated: + if key in this_style.keys(): + remaining_style[key] = this_style[key] + del this_style[key] + + # Create a copy of the parent style, and merge this style into it + parent_style_copy = parent_style.copy() + parent_style_copy.update(this_style) + this_style = parent_style_copy + + # Merge in any attributes outside of the style + style_attribs = ["fill", "stroke"] + for attrib in style_attribs: + if node.get(attrib): + this_style[attrib] = node.get(attrib) + del node.attrib[attrib] + + if isinstance(node, (SvgDocumentElement, Group, Anchor, Switch)): + # Leave only non-propagating style attributes + if remaining_style: + node.style = remaining_style + else: + if "style" in node.keys(): + del node.attrib["style"] + + # Remove the transform attribute + if "transform" in node.keys(): + del node.attrib["transform"] + + # Continue propagating on subelements + for child in node.iterchildren(): + propagate_attribs(child, this_style, this_transform) + else: + # This element is not a container + + # Merge remaining_style into this_style + this_style.update(remaining_style) + + # Set the element's style and transform attribs + node.style = this_style + node.transform = this_transform + + +### Style related + + +def get_dimension(s="1024"): + """Convert an SVG length string from arbitrary units to pixels""" + return inkex.units.convert_unit(s, "px") + + +###### Main Class ######################################### +class SynfigPrep(TempDirMixin, inkex.EffectExtension): + def effect(self): + """Transform document in preparation for exporting it into the Synfig format""" + + self.preprocess() + + # Remove inheritance of attributes + propagate_attribs(self.document.getroot()) + + # Fuse multiple subpaths in fills + for node in self.document.getroot().xpath("//svg:path"): + if node.get("d", "").lower().count("m") > 1: + # There are multiple subpaths + fill = split_fill_and_stroke(node)[0] + if fill is not None: + fill.path = fuse_subpaths(fill.path) + + def preprocess(self): + + actions = [ + "unlock-all", + # Flow roots contain rectangles inside them, so they need to be + # converted to paths separately from other shapes + "select-by-element:flowRoot", + "object-to-path", + "select-clear", + ] + + # Now convert all non-paths to paths + elements = ["rect", "circle", "ellipse", "line", "polyline", "polygon", "text"] + actions += ["select-by-element:" + i for i in elements] + actions += ["object-to-path", "select-clear"] + # unlink clones + actions += ["select-by-element:use", "object-unlink-clones"] + # save and overwrite + actions += ["export-overwrite", "export-do"] + + infile = os.path.join(self.tempdir, "input.svg") + write_svg(self.document, infile) + try: + inkscape(infile, actions=";".join(actions)) + except ProgramRunError as err: + inkex.errormsg(_("An error occurred during document preparation")) + inkex.errormsg(err.stderr.decode("utf-8")) + + with open(infile, "r") as stream: + self.document = load_svg(stream) + + return self.document + + +if __name__ == "__main__": + SynfigPrep().run() |