summaryrefslogtreecommitdiffstats
path: root/share/extensions/text_split.py
diff options
context:
space:
mode:
Diffstat (limited to 'share/extensions/text_split.py')
-rwxr-xr-xshare/extensions/text_split.py191
1 files changed, 191 insertions, 0 deletions
diff --git a/share/extensions/text_split.py b/share/extensions/text_split.py
new file mode 100755
index 0000000..069c797
--- /dev/null
+++ b/share/extensions/text_split.py
@@ -0,0 +1,191 @@
+#!/usr/bin/env python
+# coding=utf-8
+#
+# Copyright (C) 2009 Karlisson Bezerra, contato@nerdson.com
+#
+# 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.
+#
+
+import inkex
+from inkex import (
+ TextElement, FlowRoot, FlowPara, Tspan, TextPath, Rectangle
+)
+
+class TextSplit(inkex.EffectExtension):
+ """Split text up."""
+ def add_arguments(self, pars):
+ pars.add_argument("--tab", help="The selected UI-tab when OK was pressed")
+ pars.add_argument("-s", "--splittype", default="word", help="type of split")
+ pars.add_argument("-p", "--preserve", type=inkex.Boolean, default=True,\
+ help="Preserve original")
+
+ def split_lines(self, node):
+ """Returns a list of lines"""
+ lines = []
+ count = 1
+
+ for elem in node:
+ if isinstance(elem, TextPath):
+ inkex.errormsg("Text on path isn't supported. First remove text from path.")
+ break
+ elif not isinstance(elem, (FlowPara, Tspan)):
+ continue
+
+ text = TextElement(**node.attrib)
+
+ # handling flowed text nodes
+ if isinstance(node, FlowRoot):
+ fontsize = node.style.get("font-size", "12px")
+ fs = self.svg.unittouu(fontsize)
+
+ # selects the flowRegion's child (svg:rect) to get @X and @Y
+ flowref = node.findone('svg:flowRegion')[0]
+
+ if isinstance(flowref, Rectangle):
+ text.set("x", flowref.get("x"))
+ text.set("y", str(float(flowref.get("y")) + fs * count))
+ count += 1
+ else:
+ inkex.debug("This type of text element isn't supported. First unflow text.")
+ break
+
+ # now let's convert flowPara into tspan
+ tspan = Tspan()
+ tspan.set("sodipodi:role", "line")
+ tspan.text = elem.text
+ text.append(tspan)
+
+ else:
+ from copy import copy
+ x = elem.get("x") or node.get("x")
+ y = elem.get("y") or node.get("y")
+
+ text.set("x", x)
+ text.set("y", y)
+ text.append(copy(elem))
+
+ lines.append(text)
+
+ return lines
+
+ def split_words(self, node):
+ """Returns a list of words"""
+ words = []
+
+ # Function to recursively extract text
+ def plain_str(elem):
+ words = []
+ if elem.text:
+ words.append(elem.text)
+ for n in elem:
+ words.extend(plain_str(n))
+ if n.tail:
+ words.append(n.tail)
+ return words
+
+ # if text has more than one line, iterates through elements
+ lines = self.split_lines(node)
+ if not lines:
+ return words
+
+ for line in lines:
+ # gets the position of text node
+ x = float(line.get("x"))
+ y = line.get("y")
+
+ # gets the font size. if element doesn't have a style attribute, it assumes font-size = 12px
+ fontsize = line.style.get("font-size", "12px")
+ fs = self.svg.unittouu(fontsize)
+
+ # extract and returns a list of words
+ words_list = "".join(plain_str(line)).split()
+ prev_len = 0
+
+ # creates new text nodes for each string in words_list
+ for word in words_list:
+ tspan = Tspan()
+ tspan.text = word
+
+ text = TextElement(**line.attrib)
+ tspan.set('sodipodi:role', "line")
+
+ # positioning new text elements
+ x = x + prev_len * fs
+ prev_len = len(word)
+ text.set("x", str(x))
+ text.set("y", str(y))
+
+ text.append(tspan)
+ words.append(text)
+
+ return words
+
+ def split_letters(self, node):
+ """Returns a list of letters"""
+
+ letters = []
+
+ words = self.split_words(node)
+ if not words:
+ return letters
+
+ for word in words:
+
+ x = float(word.get("x"))
+ y = word.get("y")
+
+ # gets the font size. If element doesn't have a style attribute, it assumes font-size = 12px
+ fontsize = word.style.get("font-size", "12px")
+ fs = self.svg.unittouu(fontsize)
+
+ # for each letter in element string
+ for letter in word[0].text:
+ tspan = Tspan()
+ tspan.text = letter
+
+ text = TextElement(**node.attrib)
+ text.set("x", str(x))
+ text.set("y", str(y))
+ x += fs
+
+ text.append(tspan)
+ letters.append(text)
+ return letters
+
+ def effect(self):
+ """Applies the effect"""
+
+ split_type = self.options.splittype
+ preserve = self.options.preserve
+
+ # checks if the selected elements are text nodes
+ for elem in self.svg.selection.get(TextElement, FlowRoot).values():
+ if split_type == "line":
+ nodes = self.split_lines(elem)
+ elif split_type == "word":
+ nodes = self.split_words(elem)
+ elif split_type == "letter":
+ nodes = self.split_letters(elem)
+
+ for child in nodes:
+ elem.getparent().append(child)
+
+ # preserve original element
+ if not preserve and nodes:
+ parent = elem.getparent()
+ parent.remove(elem)
+
+if __name__ == '__main__':
+ TextSplit().run()