diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-27 16:29:01 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-27 16:29:01 +0000 |
commit | 35a96bde514a8897f6f0fcc41c5833bf63df2e2a (patch) | |
tree | 657d15a03cc46bd099fc2c6546a7a4ad43815d9f /share/extensions/text_merge.py | |
parent | Initial commit. (diff) | |
download | inkscape-35a96bde514a8897f6f0fcc41c5833bf63df2e2a.tar.xz inkscape-35a96bde514a8897f6f0fcc41c5833bf63df2e2a.zip |
Adding upstream version 1.0.2.upstream/1.0.2upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'share/extensions/text_merge.py')
-rwxr-xr-x | share/extensions/text_merge.py | 118 |
1 files changed, 118 insertions, 0 deletions
diff --git a/share/extensions/text_merge.py b/share/extensions/text_merge.py new file mode 100755 index 0000000..5f09b59 --- /dev/null +++ b/share/extensions/text_merge.py @@ -0,0 +1,118 @@ +#!/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="tb", help="direction to merge text") + pars.add_argument("-x", "--xanchor", default="center_x", help="horiz point to compare") + pars.add_argument("-y", "--yanchor", default="center_y", 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.selected: + for node in self.svg.xpath('//svg:text | //svg:flowRoot'): + self.svg.selected[node.get('id')] = node + + if not self.svg.selected: + 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.selected.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() |