diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-07 18:24:48 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-07 18:24:48 +0000 |
commit | cca66b9ec4e494c1d919bff0f71a820d8afab1fa (patch) | |
tree | 146f39ded1c938019e1ed42d30923c2ac9e86789 /share/extensions/text_merge.py | |
parent | Initial commit. (diff) | |
download | inkscape-cca66b9ec4e494c1d919bff0f71a820d8afab1fa.tar.xz inkscape-cca66b9ec4e494c1d919bff0f71a820d8afab1fa.zip |
Adding upstream version 1.2.2.upstream/1.2.2upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to '')
-rwxr-xr-x | share/extensions/text_merge.py | 129 |
1 files changed, 129 insertions, 0 deletions
diff --git a/share/extensions/text_merge.py b/share/extensions/text_merge.py new file mode 100755 index 0000000..6085647 --- /dev/null +++ b/share/extensions/text_merge.py @@ -0,0 +1,129 @@ +#!/usr/bin/env python +# coding=utf-8 +# +# Copyright (C) 2013 Nicolas Dufour (jazzynico) +# Direction code from the Restack extension, by Rob Antonishen +# +# Permission is hereby granted, free of charge, to any person obtaining a copy +# of this software and associated documentation files (the "Software"), to deal +# in the Software without restriction, including without limitation the rights +# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +# copies of the Software, and to permit persons to whom the Software is +# furnished to do so, subject to the following conditions: +# +# The above copyright notice and this permission notice shall be included in +# all copies or substantial portions of the Software. +# +# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +# THE SOFTWARE. +# +""" +Merge text blocks together. +""" + +import inkex +from inkex.utils import KeyDict +from inkex import Rectangle, FlowRoot, FlowPara, FlowRegion, TextElement, Tspan + +# Old settings, supported because users click 'ok' without looking. +XAN = KeyDict({"l": "left", "r": "right", "m": "center_x"}) +YAN = KeyDict({"t": "top", "b": "bottom", "m": "center_y"}) + + +class Merge(inkex.EffectExtension): + """Merge text blocks together""" + + def add_arguments(self, pars): + pars.add_argument( + "-d", "--direction", default="lr", help="direction to merge text" + ) + pars.add_argument( + "-x", "--xanchor", default="left", help="horiz point to compare" + ) + pars.add_argument( + "-y", "--yanchor", default="top", help="vertical point to compare" + ) + pars.add_argument("-k", "--keepstyle", type=inkex.Boolean, help="keep format") + pars.add_argument( + "-t", + "--flowtext", + type=inkex.Boolean, + help="use a flow text structure instead of a normal text element", + ) + + def effect(self): + if not self.svg.selection: + for node in self.svg.xpath("//svg:text | //svg:flowRoot"): + self.svg.selection[node.get("id")] = node + + if not self.svg.selection: + return + + parentnode = self.svg.get_current_layer() + + if self.options.flowtext: + text_element = FlowRoot + text_span = FlowPara + else: + text_element = TextElement + text_span = Tspan + + text_root = parentnode.add(text_element()) + text_root.set("xml:space", "preserve") + text_root.style = { + "font-size": "20px", + "font-style": "normal", + "font-weight": "normal", + "line-height": "125%", + "letter-spacing": "0px", + "word-spacing": "0px", + "fill": "#000000", + "fill-opacity": 1, + "stroke": "none", + } + + for node in sorted(self.svg.selection.values(), key=self._sort): + self.recurse(text_span, node, text_root) + + if self.options.flowtext: + region = text_root.add(FlowRegion()) + region.set("xml:space", "preserve") + rect = region.add(Rectangle()) + rect.set("xml:space", "preserve") + rect.set("height", 200) + rect.set("width", 200) + + def _sort(self, node): + return node.bounding_box().get_anchor( + self.options.xanchor, self.options.yanchor, self.options.direction + ) + + def recurse(self, text_span, node, span): + """Recursively go through each node self calling on child nodes""" + if not isinstance(node, FlowRegion): + + newspan = span.add(text_span()) + newspan.set("xml:space", "preserve") + + newspan.set("sodipodi:role", node.get("sodipodi:role")) + if isinstance(node, (TextElement, FlowPara)): + newspan.set("sodipodi:role", "line") + + if self.options.keepstyle: + newspan.style = node.style + + if node.text is not None: + newspan.text = node.text + for child in node: + self.recurse(text_span, child, newspan) + if node.tail and not isinstance(node, TextElement): + newspan.tail = node.tail + + +if __name__ == "__main__": + Merge().run() |