From 36d55fccfc1e9aa2ba8824f2f0241704ce1759b6 Mon Sep 17 00:00:00 2001
From: Folkert de Vries <flokkievids@gmail.com>
Date: Wed, 18 Feb 2015 19:22:10 +0100
Subject: [PATCH] Fix T43635:  'fill contours' fails when using multiple render
 layers There was a problem with the order of stroke and fill elements.
 Additionally, sometimes names of linesets collide between render layers. this
 has been fixed by prepending the name of the render layer to a layer's id.

---
 render_freestyle_svg.py | 75 +++++++++++++++++++++--------------------
 1 file changed, 39 insertions(+), 36 deletions(-)

diff --git a/render_freestyle_svg.py b/render_freestyle_svg.py
index 6423847fa..492672429 100644
--- a/render_freestyle_svg.py
+++ b/render_freestyle_svg.py
@@ -63,6 +63,7 @@ from bpy.props import (
         )
 from bpy.app.handlers import persistent
 from collections import OrderedDict
+from functools import partial
 from mathutils import Vector
 
 
@@ -78,6 +79,14 @@ namespaces = {
     "svg": "http://www.w3.org/2000/svg",
     }
 
+# wrap XMLElem.find, so the namespaces don't need to be given as an argument
+def find_xml_elem(obj, search, namespaces, *, all=False):
+    if all:
+        return obj.findall(search, namespaces=namespaces)
+    return obj.find(search, namespaces=namespaces)
+
+find_svg_elem = partial(find_xml_elem, namespaces=namespaces)
+
 
 def render_height(scene):
     return int(scene.render.resolution_y * scene.render.resolution_percentage / 100)
@@ -87,6 +96,7 @@ def render_width(scene):
     return int(scene.render.resolution_x * scene.render.resolution_percentage / 100)
 
 
+# stores the state of the render, used to differ between animation and single frame renders.
 class RenderState:
     # Note that this flag is set to False only after the first frame
     # has been written to file.
@@ -213,10 +223,10 @@ def write_animation(filepath, frame_begin, fps):
     tree = et.parse(filepath)
     root = tree.getroot()
 
-    linesets = tree.findall(".//svg:g[@inkscape:groupmode='lineset']", namespaces=namespaces)
+    linesets = find_svg_elem(tree, ".//svg:g[@inkscape:groupmode='lineset']", all=True)
     for i, lineset in enumerate(linesets):
         name = lineset.get('id')
-        frames = lineset.findall(".//svg:g[@inkscape:groupmode='frame']", namespaces=namespaces)
+        frames = find_svg_elem(lineset, ".//svg:g[@inkscape:groupmode='frame']", all=True)
         n_of_frames = len(frames)
         keyTimes = ";".join(str(round(x / n_of_frames, 3)) for x in range(n_of_frames)) + ";1"
 
@@ -313,8 +323,9 @@ class SVGPathShader(StrokeShader):
         name = self._name
         scene = bpy.context.scene
 
-        # make <g> for lineset as a whole (don't overwrite)
-        lineset_group = tree.find(".//svg:g[@id='{}']".format(name), namespaces=namespaces)
+        # create <g> for lineset as a whole (don't overwrite)
+        # when rendering an animation, frames will be nested in here, otherwise a group of strokes and optionally fills.
+        lineset_group = find_svg_elem(tree, ".//svg:g[@id='{}']".format(name))
         if lineset_group is None:
             lineset_group = et.XML('<g/>')
             lineset_group.attrib = {
@@ -323,15 +334,11 @@ class SVGPathShader(StrokeShader):
                 'inkscape:groupmode': 'lineset',
                 'inkscape:label': name,
                 }
-            root.insert(0, lineset_group)
+            root.append(lineset_group)
 
-        # make <g> for the current frame
+        # create <g> for the current frame
         id = "frame_{:04n}".format(self.frame_current)
 
-        if scene.svg_export.mode == 'ANIMATION':
-            frame_group = et.XML("<g/>")
-            frame_group.attrib = {'id': id, 'inkscape:groupmode': 'frame', 'inkscape:label': id}
-
         stroke_group = et.XML("<g/>")
         stroke_group.attrib = {'xmlns:inkscape': namespaces["inkscape"],
                                'inkscape:groupmode': 'layer',
@@ -340,13 +347,15 @@ class SVGPathShader(StrokeShader):
         # nest the structure
         stroke_group.extend(self.elements)
         if scene.svg_export.mode == 'ANIMATION':
+            frame_group = et.XML("<g/>")
+            frame_group.attrib = {'id': id, 'inkscape:groupmode': 'frame', 'inkscape:label': id}
             frame_group.append(stroke_group)
             lineset_group.append(frame_group)
         else:
             lineset_group.append(stroke_group)
 
         # write SVG to file
-        print("SVG Export: writing to ", self.filepath)
+        print("SVG Export: writing to", self.filepath)
         indent_xml(root)
         tree.write(self.filepath, encoding='ascii', xml_declaration=True)
 
@@ -391,6 +400,7 @@ class SVGFillShader(StrokeShader):
         root = tree.getroot()
         name = self._name
         scene = bpy.context.scene
+        lineset_group = find_svg_elem(tree, ".//svg:g[@id='{}']".format(name))
 
         # create XML elements from the acquired data
         elems = []
@@ -400,39 +410,30 @@ class SVGFillShader(StrokeShader):
             for stroke in strokes:
                 elems.append(et.XML("".join(self.pathgen((sv.point for sv in stroke), p, self.h))))
 
-        # make <g> for lineset as a whole (don't overwrite)
-        lineset_group = tree.find(".//svg:g[@id='{}']".format(name), namespaces=namespaces)
-        if lineset_group is None:
-            lineset_group = et.XML('<g/>')
-            lineset_group.attrib = {
-                'id': name,
-                'xmlns:inkscape': namespaces["inkscape"],
-                'inkscape:groupmode': 'lineset',
-                'inkscape:label': name,
-                }
-            root.insert(0, lineset_group)
-
         if scene.svg_export.mode == 'ANIMATION':
             # add the fills to the <g> of the current frame
-            frame_group = tree.find(".//svg:g[@id='frame_{:04n}']".format(scene.frame_current), namespaces=namespaces)
+            frame_group = find_svg_elem(lineset_group, ".//svg:g[@id='frame_{:04n}']".format(scene.frame_current))
             if frame_group is None:
                 # something has gone very wrong
                 raise RuntimeError("SVGFillShader: frame_group is None")
 
-        # add <g> for the strokes of the current frame
-        stroke_group = et.XML("<g/>")
-        stroke_group.attrib = {'xmlns:inkscape': namespaces["inkscape"],
-                               'inkscape:groupmode': 'layer',
-                               'inkscape:label': 'fills',
-                               'id': 'fills'}
 
-        # reverse fills to get the correct order
-        stroke_group.extend(reversed(elems))
+        # <g> for the fills of the current frame
+        fill_group = et.XML('<g/>')
+        fill_group.attrib = {
+            'xmlns:inkscape': namespaces["inkscape"],
+           'inkscape:groupmode': 'layer',
+           'inkscape:label': 'fills',
+           'id': 'fills'
+           }
 
+        fill_group.extend(reversed(elems))
         if scene.svg_export.mode == 'ANIMATION':
-            frame_group.insert(0, stroke_group)
+            frame_group.insert(0, fill_group)
         else:
-            lineset_group.insert(0, stroke_group)
+            # get the current lineset group. if it's None we're in trouble, so may as well error hard.
+            lineset_group = tree.find(".//svg:g[@id='{}']".format(name), namespaces=namespaces)
+            lineset_group.insert(0, fill_group)
 
         # write SVG to file
         indent_xml(root)
@@ -461,7 +462,7 @@ class SVGPathShaderCallback(ParameterEditorCallback):
         split = scene.svg_export.split_at_invisible
         cls.shader = SVGPathShader.from_lineset(
                 lineset, create_path(scene),
-                render_height(scene), split, scene.frame_current)
+                render_height(scene), split, scene.frame_current, name=layer.name + '_' + lineset.name)
         return [cls.shader]
 
     @classmethod
@@ -489,8 +490,9 @@ class SVGFillShaderCallback(ParameterEditorCallback):
         # sort according to the distance from camera
         Operators.sort(pyZBP1D())
         # render and write fills
-        shader = SVGFillShader(create_path(scene), render_height(scene), lineset.name)
+        shader = SVGFillShader(create_path(scene), render_height(scene), layer.name + '_' + lineset.name)
         Operators.create(TrueUP1D(), [shader, ])
+
         shader.write()
 
 
@@ -559,3 +561,4 @@ def unregister():
 
 if __name__ == "__main__":
     register()
+
-- 
GitLab