summaryrefslogtreecommitdiffstats
path: root/share/extensions/hershey.py
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--share/extensions/hershey.py2040
1 files changed, 2040 insertions, 0 deletions
diff --git a/share/extensions/hershey.py b/share/extensions/hershey.py
new file mode 100644
index 0000000..fcec22a
--- /dev/null
+++ b/share/extensions/hershey.py
@@ -0,0 +1,2040 @@
+# coding=utf-8
+#
+# Copyright(C) 2021 - Windell H. Oskay, www.evilmadscientist.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.
+#
+
+"""
+Hershey Text 3.0.6, 2022-07-21
+
+Copyright 2022, Windell H. Oskay, www.evilmadscientist.com
+
+Major revisions in Hershey Text 3.0:
+
+1. Migrate font format to use SVG fonts.
+ - SVG fonts support unicode, meaning that we can use a full range of
+ characters. We are no longer limited to the ASCII range that the
+ historical Hershey font formats used.
+ - Arbitrary curves are supported within glyphs; we are no longer limited to
+ the straight line segments used in the historical Hershey format.
+ - The set of fonts can now be expanded.
+
+2. Add a mechanism for adding your own SVG fonts, either within the
+ folder containing the default fonts, or from an external file or directory.
+ This is particularly important for installations where one does not
+ have access to edit the contents of the Inkscape extensions directory.
+
+3. Support font mapping: If a given font face is used for a given block of
+ text, check first to see if a matching SVG font is present. If not,
+ substitute with the default (selected) stroke font from the list of
+ included fonts.
+
+4. Instead of entering text (one line at a time) in the extension,
+ this script now converts text (either all text, or all selected text)
+ in the document, replacing it in place. While not every possible
+ method of formatting text is supported, many are.
+
+"""
+
+import os
+import math
+
+from copy import deepcopy
+
+import inkex
+from inkex import Transform, Style, units, AbortExtension
+from inkex.localization import inkex_gettext as _
+
+from inkex import (
+ load_svg,
+ Group,
+ TextElement,
+ FlowPara,
+ SVGfont,
+ FontFace,
+ FlowSpan,
+ Glyph,
+ MissingGlyph,
+ Tspan,
+ FlowRoot,
+ Rectangle,
+ Use,
+ PathElement,
+ Defs,
+)
+
+
+class Hershey(inkex.Effect):
+
+ """
+ An extension for use with Inkscape 1.0
+ """
+
+ def __init__(self):
+ super(Hershey, self).__init__()
+
+ self.arg_parser.add_argument(
+ "--tab",
+ dest="mode",
+ default="render",
+ help="The active tab or mode when Apply was pressed",
+ )
+
+ self.arg_parser.add_argument(
+ "--fontface",
+ dest="fontface",
+ default="HersheySans1",
+ help="The selected font face when Apply was pressed",
+ )
+
+ self.arg_parser.add_argument(
+ "--otherfont",
+ dest="otherfont",
+ default="",
+ help="Optional other font name or path to use",
+ )
+
+ self.arg_parser.add_argument(
+ "--preserve",
+ type=inkex.Boolean,
+ dest="preserve_text",
+ default=False,
+ help="Preserve original text",
+ )
+
+ self.arg_parser.add_argument(
+ "--action",
+ dest="util_mode",
+ default="sample",
+ help="The utility option selected",
+ )
+
+ self.arg_parser.add_argument(
+ "--text",
+ dest="sample_text",
+ default="\nThe Quick Brown Fox Jumps Over a Lazy Dog",
+ help="Text to use for font table",
+ )
+
+ self.font_file_list = dict()
+ self.font_load_fail = False
+
+ self.svg_height = None
+ self.svg_width = None
+
+ self.output_generated = False
+
+ self.warn_unflow = False
+ self.warn_unkern = False
+ self.warn_textpath = (
+ False # For future use: Give warning about text attached to path.
+ )
+ self.font_dict = dict() # Font dictionary - Dictionary of loaded fonts
+
+ self.nodes_to_delete = [] # List of font elements to remove
+
+ self.vb_scale_factor = 0.0104166666
+
+ self.text_string = ""
+ self.text_families = [] # List of font family for characters in the string
+ self.text_heights = [] # List of font heights
+ self.text_spacings = [] # List of vertical line heights
+ self.text_aligns = [] # List of horizontal alignment values
+ self.text_x = [] # List; x-coordinate of text line start
+ self.text_y = [] # List; y-coordinate of text line start
+ self.line_number = 0
+ self.new_line = True
+ self.render_width = 1
+
+ PX_PER_INCH = 96.0
+
+ help_text = """====== Hershey Text Help ======
+
+The Hershey Text extension is designed to replace text in your document (either
+selected text or all text) with specialized "stroke" or "engraving" fonts
+designed for plotters.
+
+Whereas regular "outline" fonts (e.g., TrueType) work by filling in the region
+inside an invisible outline, stroke fonts are composed only of individual lines
+or strokes with finite width; much like human handwriting when using a physical
+pen.
+
+Stroke fonts are most often used for creating text-like paths that computer
+controlled drawing and cutting machines (from pen plotters to CNC routers) can
+efficiently follow.
+
+A full user guide for Hershey Text is available to download from
+ http://wiki.evilmadscientist.com/hershey
+
+
+ ==== Basic operation ====
+
+To use Hershey Text, start with a document that contains text objects. Select
+the "Render" tab of Hershey Text, and choose a font face from the pop-up menu.
+
+When you click Apply, it will render all text elements on your page with the
+selected stroke-based typeface. If you would like to convert only certain text
+elements, click Apply with just those elements selected.
+
+If the "Preserve original text" box is checked, then the original text elements
+on the page will be preserved even when you click Apply. If it is unchecked,
+then the original font elements will be removed once rendered.
+
+You can generate a list of available SVG fonts or a list of all glyphs available
+in a given font by using the tools available on the "Utilities" tab.
+
+
+ ==== How Hershey Text works ====
+
+Hershey Text works by performing font substitution, starting with the text in
+your document and replacing it with paths generated from the characters in the
+selected SVG font.
+
+Hershey Text uses fonts in the SVG font format. While SVG fonts are one of the
+few types that support stroke-based characters, it is important to note that
+converting an outline font to SVG format does not convert it to a stroke based
+font. Indeed, most SVG fonts are actually outline fonts.
+
+This extension *does not* convert outline fonts into stroke fonts, nor does it
+convert other fonts into SVG format. Its sole function is to replace the text
+in your document with paths from the selected SVG font.
+
+
+ ==== Using an external SVG font ====
+
+To use an external SVG font -- one not included with the distribution -- select
+"Other" for the name of the font in the pop-up menu on the "Render" tab. Then,
+do one of the following:
+
+(1) Add your SVG font file (perhaps "example.svg") to the "svg_fonts" directory
+within your Inkscape extensions directory, and enter the name of the font
+("example") in the "Other SVG font name or path" box on the "Render" tab.
+
+or
+
+(2) Place your SVG font file anywhere on your computer, and enter the full path
+to the file in the "Other SVG font name or path" box on the "Render" tab.
+A full path might, for example, look like:
+ /Users/Robin/Documents/AxiDraw/fonts/path_handwriting.svg
+
+
+ ==== Using SVG fonts: Advanced methods ====
+
+In addition to using a single SVG font for substitution, you can also use
+font name mapping to automatically use particular stroke fonts in place of
+specific font faces, to support various automated workflows and to support
+the rapid use of multiple stroke font faces within the same document.
+
+Several SVG fonts are included with this distribution, including both
+single-stroke and multi-stroke fonts. These fonts are included within the
+"svg_fonts" directory within your Inkscape extensions directory.
+
+You can select the font that you would like to use from the pop-up menu on the
+"Render" Tab. You can also make use of your own SVG fonts.
+
+Order of preference for SVG fonts:
+
+(1) If there is an SVG font with name matching that of the font for a given
+piece of text, that font will be used. For example, if the original text is in
+font "FancyScript" and there is a file in svg_fonts with name FancyScript.svg,
+then FancyScript.svg will be used to render the text.
+
+(2) Otherwise (if there is no SVG font available matching the name of the font
+for a given block of text), the face selected from the "Font face" pop-up menu
+will be used as the default font when rendering text with Hershey Text.
+
+(3) You can also enter text in the "Name/Path" box, which can represent one of
+the following: (i) a font name (for a font located in the svg_fonts directory),
+(ii) the path to a font file elsewhere on your computer, or (iii) the path to a
+directory containing (one or more) font files.
+
+(3a) Using a font name:
+If you move a custom SVG font file into your svg_fonts directory, then you can
+enter the name of the SVG font in the "Name/Path" text box and select "Other"
+from the pop-up menu. Then, the named font will be used as the default.
+
+(3b) Using a file path:
+If you enter the path to an SVG font file in the "Name/Path" text box and
+select "Other" from the pop-up menu. Then, that font will be used as the
+default. All SVG fonts located in the same directory as that font file will
+also be available for name-based font substitution. If there are multiple
+font-name matches, files in an external directory take precedence over ones in
+the svg_fonts directory.
+
+(3c) Using a directory path:
+If you enter the path to a directory containing SVG font files in the
+"Name/Path" text box, then all SVG font files files in that directory will be
+available for name-based font substitution. If there are multiple font-name
+matches, files in an external directory take precedence over ones in the
+svg_fonts directory.
+
+
+
+Tips about using these methods with your own custom fonts:
+
+(A) These methods can be used to render different text elements with different
+SVG font faces. You can even rename a font -- either your own custom one or one
+of the bundled ones -- to match the name of a font that you're using. For
+example, if you rename a script font you name a font to "Helvetica.svg",
+then all text in Helvetica will be replaced with that SVG font.
+
+(B) Using a directory path (3c) is a particularly helpful method if you do
+not have access to modify the svg_fonts directory.
+
+
+
+ ==== Limitations ====
+
+This extension renders text into non-editable paths, generated from the
+character geometry of SVG fonts. Once you have rendered the text, the resulting
+paths can be edited with path editing tools, but not text editing tools.
+
+Since this extension works by a process of font substitution, text spanning a
+single line will generally stay that way, whereas text flowed in a box (that
+may span multiple lines) will be re-flowed from scratch. Style information such
+as text size and line spacing can be lost in some cases.
+
+We recommend that you use the live preview option to achieve best results with
+this extension.
+
+
+(c) 2021 Windell H. Oskay
+Evil Mad Scientist Laboratories
+"""
+
+ def getlength_inch(self, name):
+ """
+ Get the <svg> attribute with name "name", and parse it as a length,
+ into a value and associated units. Return value in inches.
+
+ This may cause scaling issues in some circumstances. Note, for
+ example, that Adobe Illustrator uses 72 px per inch, and Inkscape
+ used 90 px per inch prior to version 0.92.
+ """
+ string_to_parse = self.document.getroot().get(name)
+ if string_to_parse:
+ value, unit = units.parse_unit(string_to_parse)
+ if value is None:
+ return None
+ bad_units = {"%", "ex", "em"} # Unsupported units
+ if unit in bad_units:
+ return None
+
+ return units.convert_unit(string_to_parse, "in")
+ return None
+
+ def units_to_userunits(self, input_string):
+ """
+ Custom replacement for the old "unittouu" routine
+
+ Parse the attribute into a value and associated units.
+ Return value in user units (typically "px").
+ Importantly, return None for malformed inputs.
+ """
+
+ value, _ = units.parse_unit(input_string)
+ if value is None:
+ return None
+
+ return units.convert_unit(input_string, "")
+
+ def vb_scale(self, viewbox, p_a_r, doc_width, doc_height):
+ """
+ Parse SVG viewbox and generate scaling parameters.
+ Reference documentation: https://www.w3.org/TR/SVG11/coords.html
+
+ Inputs:
+ viewbox: Contents of SVG viewbox attribute
+ p_a_r: Contents of SVG preserveAspectRatio attribute
+ doc_width: Width of SVG document
+ doc_height: Height of SVG document
+
+ Output: s_x, s_y, o_x, o_y
+ Scale parameters (s_x, s_y) and offset parameters (o_x, o_y)
+
+ """
+ if viewbox is None:
+ return 1, 1, 0, 0 # No viewbox; return default transform
+ vb_array = viewbox.strip().replace(", ", " ").split()
+
+ if len(vb_array) < 4:
+ return 1, 1, 0, 0 # invalid viewbox; return default transform
+
+ min_x = float(vb_array[0]) # Viewbox offset: x
+ min_y = float(vb_array[1]) # Viewbox offset: y
+ width = float(vb_array[2]) # Viewbox width
+ height = float(vb_array[3]) # Viewbox height
+
+ if width <= 0 or height <= 0:
+ return 1, 1, 0, 0 # invalid viewbox; return default transform
+
+ if doc_width is None or doc_height is None:
+ raise AbortExtension(
+ _("Width or height attribute missing on toplevel <svg> tag")
+ )
+
+ d_width = float(doc_width)
+ d_height = float(doc_height)
+
+ if d_width <= 0 or d_height <= 0:
+ return 1, 1, 0, 0 # invalid document size; return default transform
+
+ ar_doc = d_height / d_width # Document aspect ratio
+ ar_vb = height / width # Viewbox aspect ratio
+
+ # Default values of the two preserveAspectRatio parameters:
+ par_align = "xmidymid" # "align" parameter(lowercased)
+ par_mos = "meet" # "meetOrSlice" parameter
+
+ if p_a_r is not None:
+ par_array = p_a_r.strip().replace(", ", " ").lower().split()
+ if len(par_array) > 0:
+ par0 = par_array[0]
+ if par0 == "defer":
+ if len(par_array) > 1:
+ par_align = par_array[1]
+ if len(par_array) > 2:
+ par_mos = par_array[2]
+ else:
+ par_align = par0
+ if len(par_array) > 1:
+ par_mos = par_array[1]
+
+ if par_align == "none":
+ # Scale document to fill page. Do not preserve aspect ratio.
+ # This is not default behavior, nor what happens if par_align
+ # is not given; the "none" value must be _explicitly_ specified.
+
+ s_x = d_width / width
+ s_y = d_height / height
+ o_x = -min_x
+ o_y = -min_y
+ return s_x, s_y, o_x, o_y
+
+ """
+ Other than "none", all situations fall into two classes:
+
+ 1) (ar_doc >= ar_vb AND par_mos == "meet")
+ or (ar_doc < ar_vb AND par_mos == "slice")
+ -> In these cases, scale document up until VB fills doc in X.
+
+ 2) All other cases, i.e.,
+ (ar_doc < ar_vb AND par_mos == "meet")
+ or (ar_doc >= ar_vb AND par_mos == "slice")
+ -> In these cases, scale document up until VB fills doc in Y.
+
+ Note in cases where the scaled viewbox exceeds the document
+ (page) boundaries (all "slice" cases and many "meet" cases where
+ an offset value is given) that this routine does not perform
+ any clipping, but subsequent clipping to the page boundary
+ is appropriate.
+
+ Besides "none", there are 9 possible values of par_align:
+ xminymin xmidymin xmaxymin
+ xminymid xmidymid xmaxymid
+ xminymax xmidymax xmaxymax
+ """
+
+ if ((ar_doc >= ar_vb) and (par_mos == "meet")) or (
+ (ar_doc < ar_vb) and (par_mos == "slice")
+ ):
+ # Case 1: Scale document up until VB fills doc in X.
+
+ s_x = d_width / width
+ s_y = s_x # Uniform aspect ratio
+ o_x = -min_x
+
+ scaled_vb_height = ar_doc * width
+ excess_height = scaled_vb_height - height
+
+ if par_align in {"xminymin", "xmidymin", "xmaxymin"}:
+ # Case: Y-Min: Align viewbox to minimum Y of the viewport.
+ o_y = -min_y
+ # OK: tested with Tall-Meet, Wide-Slice
+
+ elif par_align in {"xminymax", "xmidymax", "xmaxymax"}:
+ # Case: Y-Max: Align viewbox to maximum Y of the viewport.
+ o_y = -min_y + excess_height
+ # OK: tested with Tall-Meet, Wide-Slice
+
+ else: # par_align in {"xminymid", "xmidymid", "xmaxymid"}:
+ # Default case: Y-Mid: Center viewbox on page in Y
+ o_y = -min_y + excess_height / 2
+ # OK: Tested with Tall-Meet, Wide-Slice
+
+ return s_x, s_y, o_x, o_y
+
+ # Case 2: Scale document up until VB fills doc in Y.
+
+ s_y = d_height / height
+ s_x = s_y # Uniform aspect ratio
+ o_y = -min_y
+
+ scaled_vb_width = height / ar_doc
+ excess_width = scaled_vb_width - width
+
+ if par_align in {"xminymin", "xminymid", "xminymax"}:
+ # Case: X-Min: Align viewbox to minimum X of the viewport.
+ o_x = -min_x
+ # OK: Tested with Tall-Slice, Wide-Meet
+
+ elif par_align in {"xmaxymin", "xmaxymid", "xmaxymax"}:
+ # Case: X-Max: Align viewbox to maximum X of the viewport.
+ o_x = -min_x + excess_width
+ # Need test: Tall-Slice, Wide-Meet
+
+ else: # par_align in {"xmidymin", "xmidymid", "xmidymax"}:
+ # Default case: X-Mid: Center viewbox on page in X
+ o_x = -min_x + excess_width / 2
+ # OK: Tested with Tall-Slice, Wide-Meet
+
+ return s_x, s_y, o_x, o_y
+
+ def strip_quotes(self, fontname):
+ """
+ A multi-word font name may have a leading and trailing
+ single or double quotes, depending on the source.
+ If so, remove those quotes.
+ """
+
+ if fontname.startswith("'") and fontname.endswith("'"):
+ return fontname[1:-1]
+ if fontname.startswith('"') and fontname.endswith('"'):
+ return fontname[1:-1]
+ return fontname
+
+ def parse_svg_font(self, node_list):
+ """
+ Parse an input svg, searching for an SVG font. If an
+ SVG font is found, parse it and return a "digest" containing
+ structured information from the font. See below for more
+ about the digest format.
+
+ If the font is not found cannot be parsed, return none.
+
+ Notable limitations:
+
+ (1) This function only parses the first font face found within the
+ tree. We may, in the future, support discovering multiple fonts
+ within an SVG file.
+
+ (2) We are only processing left-to-right and horizontal text,
+ not vertical text nor RTL.
+
+ (3) This function currently performs only certain recursive searches,
+ within the <defs> element. It will not discover fonts nested within
+ groups or other elements. So far as we know, that is not a limitation
+ in practice. (If you have a counterexample please contact Evil Mad
+ Scientist tech support and let us know!)
+
+ (4) Kerning details are not implemented yet.
+ """
+
+ digest = None
+
+ if node_list is None:
+ return None
+
+ for node in node_list:
+ if isinstance(node, Defs):
+ return self.parse_svg_font(node) # Recursive call
+
+ if isinstance(node, SVGfont):
+ """
+ === Internal structure for storing font information ===
+
+ We parse the SVG font file and create a keyed "digest"
+ from it that we can use while rendering text on the page.
+
+ This "digest" will be added to a dictionary that maps
+ each font family name to a single digest.
+
+ The digest itself is a dictionary with the following
+ keys, some of which may have empty values. This format
+ will allow us to add additional keys at a later date,
+ to support additional SVG font features.
+
+ font_id (a string)
+
+ font_family (a string)
+
+ glyphs
+ A dictionary mapping unicode points to a specific
+ dictionary for each point. See below for more about
+ the key format.
+ The dictionary for a given point will include keys:
+ glyph_name(string)
+ horiz_adv_x(numeric)
+ d(string)
+
+ missing_glyph
+ A dictionary for a single code point, with keys:
+ horiz_adv_x(numeric)
+ d(string)
+
+ geometry
+ A dictionary containing geometric data
+ Keys will include:
+ horiz_adv_x(numeric) -- Default value
+ units_per_em(numeric)
+ ascent(numeric)
+ descent(numeric)
+ x_height(numeric)
+ cap_height(numeric)
+ bbox (string)
+ underline_position(numeric)
+ scale
+ A numeric scaling factor computed from the
+ units_per_em value, which gives the overall scale
+ """
+
+ digest = dict()
+ geometry = dict()
+ glyphs = dict()
+ missing_glyph = dict()
+
+ digest["font_id"] = node.get("id")
+
+ horiz_adv_x = node.get("horiz-adv-x")
+
+ if horiz_adv_x is not None:
+ geometry["horiz_adv_x"] = float(horiz_adv_x)
+ # Note: case of no horiz_adv_x value is not handled.
+
+ for element in node:
+
+ if isinstance(element, Glyph):
+ # First, because it is the most common element
+ try:
+ uni_text = element.get("unicode")
+ except:
+ # Can't use this point if no unicode mapping.
+ continue
+
+ if uni_text is None:
+ continue
+
+ if uni_text in glyphs:
+ # Skip if that unicode point is already in the
+ # list of glyphs.(There is not currently support
+ # for alternate glyphs in the font.)
+ continue
+
+ glyph_dict = dict()
+ glyph_dict["glyph_name"] = element.get("glyph-name")
+
+ horiz_adv_x = element.get("horiz-adv-x")
+
+ if horiz_adv_x is not None:
+ glyph_dict["horiz_adv_x"] = float(horiz_adv_x)
+ else:
+ glyph_dict["horiz_adv_x"] = geometry["horiz_adv_x"]
+
+ glyph_dict["d"] = element.get("d") # SVG path data
+ glyphs[uni_text] = glyph_dict
+
+ elif isinstance(element, FontFace):
+ digest["font_family"] = element.get("font-family")
+ units_per_em = element.get("units-per-em")
+
+ if units_per_em is None:
+ # Default: 1000, per SVG specification.
+ geometry["units_per_em"] = 1000.0
+ else:
+ geometry["units_per_em"] = float(units_per_em)
+
+ ascent = element.get("ascent")
+ if ascent is not None:
+ geometry["ascent"] = float(ascent)
+
+ descent = element.get("descent")
+ if descent is not None:
+ geometry["descent"] = float(descent)
+
+ """
+ # Skip these attributes that we are not currently using
+ geometry['x_height'] = element.get('x-height')
+ geometry['cap_height'] = element.get('cap-height')
+ geometry['bbox'] = element.get('bbox')
+ geometry['underline_position'] = element.get('underline-position')
+ """
+
+ elif isinstance(element, MissingGlyph):
+ horiz_adv_x = element.get("horiz-adv-x")
+
+ if horiz_adv_x is not None:
+ missing_glyph["horiz_adv_x"] = float(horiz_adv_x)
+ else:
+ missing_glyph["horiz_adv_x"] = geometry["horiz_adv_x"]
+
+ missing_glyph["d"] = element.get("d") # SVG path data
+ digest["missing_glyph"] = missing_glyph
+
+ # Main scaling factor
+ digest["scale"] = 1.0 / geometry["units_per_em"]
+
+ digest["glyphs"] = glyphs
+ digest["geometry"] = geometry
+
+ return digest
+ return None
+
+ def load_font(self, fontname):
+ """
+ Attempt to load an SVG font from a file in our list
+ of (likely) SVG font files.
+ If we can, add the contents to the font library.
+ Otherwise, add a "None" entry to the font library.
+ """
+
+ if fontname is None:
+ return
+
+ if fontname in self.font_dict:
+ return # Awesome: The font is already loaded.
+
+ if fontname in self.font_file_list:
+ the_path = self.font_file_list[fontname]
+ else:
+ self.font_dict[fontname] = None
+ return # Font not located.
+ try:
+ """
+ Check to see if there is an SVG font file for us to read.
+
+ At present, only one font file will be read per font family;
+ the name of the file must be FONT_NAME.svg, where FONT_NAME
+ is the name of the font family.
+
+ Only the first font found in the font file will be read.
+ Multiple weights and styles within a font family are not
+ presently supported.
+ """
+ font_svg = load_svg(the_path)
+ self.font_dict[fontname] = self.parse_svg_font(font_svg.getroot())
+
+ except IOError:
+ self.font_dict[fontname] = None
+ except:
+ inkex.errormsg(_("Error parsing SVG font at {}").format(str(the_path)))
+ self.font_dict[fontname] = None
+
+ def font_table(self):
+ """
+ Generate display table of all available SVG fonts
+ """
+
+ self.options.preserve_text = False
+
+ # Embed text in group to make manipulation easier:
+ group = self.svg.get_current_layer().add(Group())
+ for fontname in self.font_file_list:
+ self.load_font(fontname)
+
+ font_size = 0.2 # in inches -- will be scaled by viewbox factor.
+ font_size_text = str(font_size / self.vb_scale_factor) + "px"
+
+ labeltext_style = Style(
+ {
+ "stroke": "none",
+ "font-size": font_size_text,
+ "fill": "black",
+ "font-family": "sans-serif",
+ "text-anchor": "end",
+ }
+ )
+
+ x_offset = font_size / self.vb_scale_factor
+ y_offset = 1.5 * x_offset
+ y = y_offset
+
+ for fontname in sorted(self.font_dict):
+ if self.font_dict[fontname] is None:
+ continue # If the SVG file did NOT contain a font, skip it.
+
+ text_attribs = {"x": "0", "y": str(y), "hershey-ignore": "true"}
+ textline = group.add(TextElement(**text_attribs))
+ textline.text = fontname
+ textline.style = labeltext_style
+ text_attribs = {"x": str(x_offset), "y": str(y)}
+
+ sampletext_style = Style(
+ {
+ "stroke": "none",
+ "font-size": font_size_text,
+ "fill": "black",
+ "font-family": fontname,
+ "text-anchor": "start",
+ }
+ )
+ sampleline = group.add(TextElement(**text_attribs))
+
+ try: # python 2
+ sampleline.text = self.options.sample_text.decode("utf-8")
+ except AttributeError: # python 3
+ sampleline.text = self.options.sample_text
+
+ sampleline.style = sampletext_style
+ y += y_offset
+ self.recursively_traverse_svg(group, self.doc_transform)
+
+ def glyph_table(self):
+ """
+ Generate display table of glyphs within the current SVG font. Sorted display of
+ all printable characters in the font _except_ missing glyph.
+ """
+
+ self.options.preserve_text = False
+
+ fontname = self.font_load_wrapper("not_a_font_name") # force load of default
+
+ if self.font_load_fail:
+ inkex.errormsg(_("Font not found; Unable to generate glyph table."))
+ return
+
+ # Embed in group to make manipulation easier:
+ group = self.svg.get_current_layer().add(Group())
+
+ # missing_glyph = self.font_dict[fontname]['missing_glyph']
+
+ glyph_count = 0
+ for glyph in self.font_dict[fontname]["glyphs"]:
+ if self.font_dict[fontname]["glyphs"][glyph]["d"] is not None:
+ glyph_count += 1
+
+ columns = int(math.floor(math.sqrt(glyph_count)))
+
+ font_size = 0.4 # in inches -- will be scaled by viewbox factor.
+ font_size_text = str(font_size / self.vb_scale_factor) + "px"
+
+ glyph_style = Style(
+ {
+ "stroke": "none",
+ "font-size": font_size_text,
+ "fill": "black",
+ "font-family": fontname,
+ "text-anchor": "start",
+ }
+ )
+
+ x_offset = 1.5 * font_size / self.vb_scale_factor
+ y_offset = x_offset
+ x = x_offset
+ y = y_offset
+
+ draw_position = 0
+
+ for glyph in sorted(self.font_dict[fontname]["glyphs"]):
+ if self.font_dict[fontname]["glyphs"][glyph]["d"] is None:
+ continue
+ y_pos, x_pos = divmod(draw_position, columns)
+ x = x_offset * (x_pos + 1)
+ y = y_offset * (y_pos + 1)
+ text_attribs = {"x": str(x), "y": str(y)}
+ sampleline = group.add(TextElement(**text_attribs))
+ sampleline.text = glyph
+ sampleline.style = glyph_style
+ draw_position = draw_position + 1
+
+ self.recursively_traverse_svg(group, self.doc_transform)
+
+ def find_font_files(self):
+ """
+ Create list of "plausible" SVG font files
+
+ List items in primary svg_fonts directory, typically located in the
+ directory where this script is being executed from.
+
+ If there is text given in the "Other name/path" input, that text may
+ represent one of the following:
+
+ (A) The name of a font file, located in the svg_fonts directory.
+ - This may be given with or without the .svg suffix.
+ - If it is a font file, and the font face selected is "other",
+ then use this as the default font face.
+
+ (B) The path to a font file, located elsewhere.
+ - If it is a font file, and the font face selected is "other",
+ then use this as the default font face.
+ - ALSO: Search the directory where that file is located for
+ any other SVG fonts.
+
+ (C) The path to a directory
+ - It may or may not have a trailing separator
+ - Search that directory for SVG fonts.
+
+ This function will create a list of available files that
+ appear to be SVG(SVG font) files. It does not parse the files.
+ We will format it as a dictionary, that maps each file name
+ (without extension) to a path.
+ """
+
+ self.font_file_list = dict()
+
+ # List contents of primary font directory:
+ font_directory_name = "svg_fonts"
+
+ font_dir = os.path.realpath(os.path.join(os.getcwd(), font_directory_name))
+ for dir_item in os.listdir(font_dir):
+ if dir_item.endswith((".svg", ".SVG")):
+ file_path = os.path.join(font_dir, dir_item)
+ if os.path.isfile(file_path): # i.e., if not a directory
+ root, _ = os.path.splitext(dir_item)
+ self.font_file_list[root] = file_path
+
+ # split off file extension(e.g., ".svg")
+ root, _ = os.path.splitext(self.options.otherfont)
+
+ # Check for case "(A)": Input text is the name
+ # of an item in the primary font directory.
+ if root in self.font_file_list:
+ # If we already have that name in our font_file_list,
+ # and "other" is selected, this is now
+ # our default font face.
+ if self.options.fontface == "other":
+ self.options.fontface = root
+ return
+
+ test_path = os.path.realpath(self.options.otherfont)
+
+ # Check for case "(B)": A file, not in primary font directory
+ if os.path.isfile(test_path):
+ directory, file_name = os.path.split(test_path)
+ root, ext = os.path.splitext(file_name)
+ self.font_file_list[root] = test_path
+
+ if self.options.fontface == "other":
+ self.options.fontface = root
+
+ # Also search the directory where that file
+ # was located for other SVG files(which may be fonts)
+
+ for dir_item in os.listdir(directory):
+ if dir_item.endswith((".svg", ".SVG")):
+ file_path = os.path.join(directory, dir_item)
+ if os.path.isfile(file_path): # i.e., if not a directory
+ root, _ = os.path.splitext(dir_item)
+ self.font_file_list[root] = file_path
+ return
+
+ # Check for case "(C)": A directory name
+ if os.path.isdir(test_path):
+ for dir_item in os.listdir(test_path):
+ if dir_item.endswith((".svg", ".SVG")):
+ file_path = os.path.join(test_path, dir_item)
+ if os.path.isfile(file_path): # i.e., if not a directory
+ root, _ = os.path.splitext(dir_item)
+ self.font_file_list[root] = file_path
+
+ def font_load_wrapper(self, fontname):
+ """
+
+ This implements the following logic:
+
+ * Check to see if the font name is in our lookup table of fonts,
+ self.font_dict
+
+ * If the font is not listed in font_dict[]:
+ * Check to see if there is a corresponding SVG font file that
+ can be opened and parsed.
+
+ * If the font can be opened and parsed:
+ * Add that font to font_dict.
+ * Otherwise
+ * Add the font name to font_dict as None.
+
+ * If the font has value None in font_dict:
+ * Try to load fallback font.
+
+ * Fallback font:
+ * If an SVG font matching that in the SVG is not available,
+ check to see if the default font is available. That font
+ is given by self.options.fontface
+
+ * If a font is loaded and available, return the font name.
+ Otherwise, return none.
+
+ """
+
+ self.load_font(fontname) # Load the font if available
+
+ """
+ It *may* be worth building one stroke font (e.g., Hershey Sans 1-stroke) as a
+ variable defined in this file so that it can be used even if no external
+ SVG font files are available.
+ """
+
+ if self.font_dict[fontname] is None:
+
+ # If we were not able to load the requested font::
+ fontname = self.options.fontface # Fallback
+ if fontname not in self.font_dict:
+ self.load_font(fontname)
+ else:
+ pass
+
+ if self.font_dict[fontname] is None:
+ self.font_load_fail = (
+ True # Set a flag so that we only generate one copy of this error.
+ )
+ return None
+ return fontname
+
+ def get_font_char(self, fontname, char):
+ """
+ Given a font face name and a character(unicode point),
+ return an SVG path, horizontal advance value,
+ and scaling factor.
+
+ If the font is not available by name, use the default font.
+ """
+
+ fontname = self.font_load_wrapper(fontname) # Load the font if available
+
+ if fontname is None:
+ return None
+
+ try:
+ scale_factor = self.font_dict[fontname]["scale"]
+ except:
+ scale_factor = 0.001 # Default: 1/1000
+
+ try:
+ if char not in self.font_dict[fontname]["glyphs"]:
+ x_adv = self.font_dict[fontname]["missing_glyph"]["horiz_adv_x"]
+
+ return (
+ self.font_dict[fontname]["missing_glyph"]["d"],
+ x_adv,
+ scale_factor,
+ )
+ x_adv = self.font_dict[fontname]["glyphs"][char]["horiz_adv_x"]
+
+ return self.font_dict[fontname]["glyphs"][char]["d"], x_adv, scale_factor
+ except:
+ return None
+
+ def handle_viewbox(self):
+ """
+ Wrapper function for processing viewbox information
+ """
+
+ self.svg_height = self.getlength_inch("height")
+ self.svg_width = self.getlength_inch("width")
+
+ self.svg = self.document.getroot()
+ viewbox = self.svg.get("viewBox")
+ if viewbox:
+ p_a_r = self.svg.get("preserveAspectRatio")
+ s_x, s_y, o_x, o_y = self.vb_scale(
+ viewbox, p_a_r, self.svg_width, self.svg_height
+ )
+ else:
+ s_x = 1.0 / float(self.PX_PER_INCH) # Handle case of no viewbox
+ s_y = s_x
+ o_x = 0.0
+ o_y = 0.0
+
+ # Initial transform of document is based on viewbox, if present:
+ self.doc_transform = Transform(scale=(s_x, s_y), translate=(o_x, o_y))
+
+ self.vb_scale_factor = (s_x + s_y) / 2.0
+ # In case of non-square aspect ratio, use average value.
+
+ def draw_svg_text(self, chardata, parent):
+ """
+ Render an individual svg glyph
+ """
+ char = chardata["char"]
+ font_family = chardata["font_family"]
+ offset = chardata["offset"]
+ vertoffset = chardata["vertoffset"]
+ font_height = chardata["font_height"]
+ font_scale = 1.0
+
+ # Stroke scale factor, including external transformations:
+ stroke_scale = chardata["stroke_scale"] * self.vb_scale_factor
+
+ try:
+ path_string, adv_x, scale_factor = self.get_font_char(font_family, char)
+ except:
+ adv_x = 0
+ path_string = None
+ scale_factor = 1.0
+
+ if self.font_load_fail:
+ return 0
+
+ font_scale *= scale_factor * font_height
+
+ h_offset = 0
+ v_offset = 0
+
+ # SVG fonts use inverted Y axis; mirror vertically
+ scale_transform = Transform(scale=(font_scale, -font_scale))
+
+ # Combine scales of external transformations with the scaling
+ # applied by this function:
+ _scale = font_scale * stroke_scale
+ if _scale == 0:
+ _scale = 1
+ stroke_width = self.render_width / _scale
+
+ # Stroke-width is a css style element; cannot use scientific notation.
+ # Thus, use variable width for encoding the stroke width factor:
+
+ log_ten = math.log10(stroke_width)
+ if log_ten > 0: # For stroke_width > 1
+ width_string = "{0:.3f}in".format(stroke_width)
+ else:
+ prec = int(math.ceil(-log_ten) + 3)
+ width_string = "{0:.{1}f}in".format(stroke_width, prec)
+
+ p_style = {"stroke-width": width_string}
+
+ the_transform = Transform(translate=(offset + h_offset, vertoffset + v_offset))
+ the_transform @= scale_transform
+
+ if path_string is not None:
+ path_element = parent.add(PathElement())
+ path_element.set_path(path_string)
+ path_element.style = p_style
+ path_element.transform = the_transform
+ self.output_generated = True
+
+ return offset + float(adv_x) * font_scale # new horizontal offset value
+
+ def recursive_get_encl_transform(self, node):
+
+ """
+ Determine the cumulative transform which node inherits from
+ its chain of ancestors.
+ """
+ node = node.getparent()
+ if node is not None:
+ parent_transform = self.recursive_get_encl_transform(node)
+ node_transform = node.get("transform", None)
+ if node_transform is None:
+ return parent_transform
+ trans = Transform(node_transform).matrix
+
+ if parent_transform is None:
+ return trans
+ return Transform(parent_transform) * Transform(trans)
+ return self.doc_transform
+
+ def recursively_parse_flowroot(self, node_list, parent_info):
+ """
+ Parse a flowroot node and its children
+ """
+
+ # By default, inherit these values from parent:
+ font_height_local = parent_info["font_height"]
+ font_family_local = parent_info["font_family"]
+ line_spacing_local = parent_info["line_spacing"]
+ text_align_local = parent_info["align"]
+
+ for node in node_list:
+ node_style = node.style
+
+ font_height = node_style("font-size")
+ try:
+ font_height_local = self.units_to_userunits(font_height)
+ except TypeError:
+ pass
+
+ font_family_local = self.strip_quotes(node_style("font-family"))
+
+ try:
+ line_spacing = node_style("line-height")
+ if "%" in line_spacing: # Handle percentage line spacing(e.g., 125%)
+ line_spacing_local = float(line_spacing.rstrip("%")) / 100.0
+ elif line_spacing == "normal":
+ line_spacing_local = 1.25 # Inkscape default line spacing
+ else:
+ line_spacing_local = self.units_to_userunits(line_spacing)
+ except TypeError:
+ pass
+
+ text_align_local = node_style("text-align") # Use text-anchor in text nodes
+
+ if node.text is not None:
+ self.text_string += node.text
+
+ for _ in node.text:
+ self.text_families.append(font_family_local)
+ self.text_heights.append(font_height_local)
+ self.text_spacings.append(line_spacing_local)
+ self.text_aligns.append(text_align_local)
+
+ if isinstance(node, (FlowPara, FlowSpan)):
+ the_style = dict()
+ the_style["font_height"] = font_height_local
+ the_style["font_family"] = font_family_local
+ the_style["line_spacing"] = line_spacing_local
+ the_style["align"] = text_align_local
+
+ self.recursively_parse_flowroot(node, the_style)
+
+ if node.tail is not None:
+ # By default, inherit these values from parent:
+ font_height_local = parent_info["font_height"]
+ font_family_local = parent_info["font_family"]
+ line_spacing_local = parent_info["line_spacing"]
+
+ text_align_local = parent_info["align"]
+ self.text_string += node.tail
+ for _ in node.tail:
+ self.text_families.append(font_family_local)
+ self.text_heights.append(font_height_local)
+ self.text_spacings.append(line_spacing_local)
+ self.text_aligns.append(text_align_local)
+
+ if isinstance(node, FlowPara):
+ self.text_string += "\n" # Conclude every flowpara with a return
+ self.text_families.append(font_family_local)
+ self.text_heights.append(font_height_local)
+ self.text_spacings.append(line_spacing_local)
+ self.text_aligns.append(text_align_local)
+
+ def recursively_parse_text(self, node, parent_info):
+ """
+ parse a text node and its children
+ """
+
+ # By default, inherit these values from parent:
+ font_height_local = parent_info["font_height"]
+ font_family_local = parent_info["font_family"]
+ anchor_local = parent_info["anchor"]
+ x_local = parent_info["x_pos"]
+ y_local = parent_info["y_pos"]
+ parent_line_spacing = parent_info["line_spacing"]
+
+ node_style = node.style
+
+ font_height = node_style("font-size")
+ try:
+ font_height_local = self.units_to_userunits(font_height)
+ except TypeError:
+ pass
+
+ font_family_local = self.strip_quotes(node_style("font-family"))
+ anchor_local = node_style("text-anchor") # Use text-anchor in text nodes
+
+ try:
+ x_temp = node.get("x")
+ if x_temp is not None:
+ x_local = x_temp
+ except ValueError:
+ pass
+
+ try:
+ y_temp = node.get("y")
+ if y_temp is not None:
+ y_local = y_temp
+ else:
+ # Special case, to handle multi-line text given by tspan
+ # elements that do not have y values
+ if y_local is None:
+ y_local = 0
+ y_local = (
+ float(y_local)
+ + self.line_number * parent_line_spacing * font_height_local
+ )
+ except ValueError:
+ pass
+
+ if node.text is not None:
+ self.text_string += node.text
+
+ for _ in node.text:
+ self.text_families.append(font_family_local)
+ self.text_heights.append(font_height_local)
+ self.text_aligns.append(anchor_local)
+ self.text_x.append(x_local)
+ self.text_y.append(y_local)
+
+ for sub_node in node:
+ # If text is located within a sub_node of this node,
+ # process that sub_node, with this very routine.
+
+ if isinstance(sub_node, Tspan):
+ # Note: There may be additional types of text tags that
+ # we should recursively search as well.
+ node_info = dict()
+ node_info["font_height"] = font_height_local
+ node_info["font_family"] = font_family_local
+ node_info["anchor"] = anchor_local
+ node_info["x_pos"] = x_local
+ node_info["y_pos"] = y_local
+ node_info["line_spacing"] = parent_line_spacing
+
+ adv_line = False
+ role = sub_node.get("sodipodi:role")
+ if role == "line":
+ adv_line = True
+
+ self.recursively_parse_text(sub_node, node_info)
+
+ # Increment line after tspan if it is labeled as a line
+ if adv_line:
+ self.line_number = self.line_number + 1
+
+ if node.tail is not None:
+ _stripped_tail = node.tail.strip()
+ if _stripped_tail is not None:
+ # By default, inherit these values from parent:
+ font_height_local = parent_info["font_height"]
+ font_family_local = parent_info["font_family"]
+ text_align_local = parent_info["anchor"]
+ x_local = parent_info["x_pos"]
+ y_local = parent_info["y_pos"]
+ self.text_string += _stripped_tail
+ for _ in _stripped_tail:
+ self.text_heights.append(font_height_local)
+ self.text_families.append(font_family_local)
+ self.text_aligns.append(text_align_local)
+ self.text_x.append(x_local)
+ self.text_y.append(y_local)
+
+ def recursively_traverse_svg(
+ self,
+ anode_list,
+ mat_current=[[1.0, 0.0, 0.0], [0.0, 1.0, 0.0]],
+ parent_visibility="visible",
+ ):
+ """
+ recursively parse the full document and its children,
+ looking for nodes that may contain text
+ """
+
+ for node in anode_list:
+
+ # Ignore invisible nodes
+ vis = node.get("visibility", parent_visibility)
+ if vis == "inherit":
+ vis = parent_visibility
+ if vis in ("hidden", "collapse"):
+ continue
+
+ # First apply the current matrix transform to this node's tranform
+ _matrix = node.transform
+ mat_new = Transform(mat_current) @ Transform(_matrix)
+
+ if isinstance(node, Group):
+
+ recurse_group = True
+ ink_label = node.get("inkscape:label")
+
+ if not ink_label:
+ pass
+ else:
+ if ink_label == "Hershey Text":
+ recurse_group = (
+ False # Do not traverse groups of rendered text.
+ )
+ if recurse_group:
+ self.recursively_traverse_svg(node, mat_new, vis)
+
+ elif isinstance(node, Use):
+ # A <use> element refers to another SVG element via an xlink:href="#blah"
+ # attribute. We will handle the element by doing an XPath search through
+ # the document, looking for the element with the matching id="blah"
+ # attribute. We then recursively process that element after applying
+ # any necessary(x, y) translation.
+ #
+ # Notes:
+ # 1. We ignore the height and width attributes as they do not apply to
+ # path-like elements, and
+ # 2. Even if the use element has visibility="hidden", SVG still calls
+ # for processing the referenced element. The referenced element is
+ # hidden only if its visibility is "inherit" or "hidden".
+
+ refnode = node.href
+ if refnode is None:
+ continue # missing reference
+
+ local_transform = Transform(_matrix)
+ x = float(node.get("x", "0"))
+ y = float(node.get("y", "0"))
+ # Note: the transform has already been applied
+ if (x != 0) or (y != 0):
+ _trans_string = "translate({0:.6E}, {1:.6E})".format(x, y)
+ ref_transform = Transform(_matrix) * Transform(_trans_string)
+ else:
+ ref_transform = local_transform
+
+ try:
+ ref_group = anode_list.add(Group()) # Add a subgroup
+ except AttributeError:
+ inkex.errormsg(
+ _("Unable to process text. Consider unlinking cloned text.")
+ )
+ continue
+
+ # Tests are not using the preset seed for this atm
+ # if 'id' not in ref_group.attrib:
+ # ref_group.set_random_id('')
+
+ ref_group.set("transform", ref_transform)
+
+ ref_group.append(deepcopy(refnode))
+
+ for sub_node in ref_group:
+ # The copied text elements should be removed at the end,
+ # or they will persist if original elements are preserved.
+ self.nodes_to_delete.append(sub_node)
+
+ # Preserve original element?
+ if not self.options.preserve_text:
+ self.nodes_to_delete.append(node)
+
+ elif isinstance(node, (TextElement, FlowRoot)):
+
+ # Flag for when we start a new line of text, for use with indents:
+ self.new_line = True
+
+ start_x = (
+ 0 # Defaults; Fail gracefully in case xy position is not given.
+ )
+ start_y = 0
+
+ # Default line spacing and font height: 125%, 16 px
+ line_spacing = self.units_to_userunits("1.25")
+ font_height = self.units_to_userunits("16px")
+
+ start_x = node.get("x") # XY Position of element
+ start_y = node.get("y")
+
+ bounding_rect = False
+ # rect_height = 100 #default size of bounding rectangle for flowroot object
+ rect_width = (
+ 100 # default size of bounding rectangle for flowroot object
+ )
+ transform = "" # transform(scale, translate, matrix, etc.)
+ text_align = "start"
+
+ try:
+ hershey_ignore = node.get("hershey-ignore")
+ if hershey_ignore is not None:
+ continue # If the attribute is present, skip this node.
+ except ValueError:
+ pass
+
+ node_style = node.style
+
+ try:
+ font_height_temp = node_style.get("font-size", 16)
+ font_height = self.units_to_userunits(font_height_temp)
+ except TypeError:
+ pass
+
+ font_family = self.strip_quotes(node_style("font-family"))
+
+ try:
+ line_spacing_temp = node_style("line-height")
+ if (
+ "%" in line_spacing_temp
+ ): # Handle percentage line spacing(e.g., 125%)
+ line_spacing = float(line_spacing_temp.rstrip("%")) / 100.0
+ elif line_spacing_temp == "normal":
+ line_spacing = 1.25 # Inkscape default line spacing
+ else:
+ line_spacing = self.units_to_userunits(line_spacing_temp)
+ except TypeError:
+ pass
+
+ try:
+ transform = node.transform
+ except ValueError:
+ pass
+
+ if transform is not None:
+ transform2 = Transform(transform).matrix
+
+ """
+ Compute estimate of transformation scale applied to
+ this element, for purposes of calculating the
+ stroke width to apply. When all transforms are applied
+ and our elements are displayed on the page, we want the
+ final visible stroke width to be reasonable.
+ Transformation matrix is [[a c e][b d f]]
+ scale_x = sqrt(a * a + b * b),
+ scale_y = sqrt(c * c + d * d)
+ Take estimated scale as the mean of the two.
+ """
+
+ scale_x = math.sqrt(
+ transform2[0][0] * transform2[0][0]
+ + transform2[1][0] * transform2[1][0]
+ )
+ scale_y = math.sqrt(
+ transform2[0][1] * transform2[0][1]
+ + transform2[1][1] * transform2[1][1]
+ )
+
+ scale_r = (scale_x + scale_y) / 2.0 # Average. ¯\_(ツ)_/¯
+ else:
+ scale_r = 1.0
+
+ the_id = node.get("id")
+
+ # Initialize text attribute lists for each top-level text object:
+ self.text_string = ""
+ self.text_families = (
+ []
+ ) # Lis of font family for characters in the string
+ self.text_heights = [] # List of font heights
+ self.text_spacings = [] # List of vertical line heights
+ self.text_aligns = [] # List of horizontal alignment values
+ self.text_x = [] # List; x-coordinate of text line start
+ self.text_y = [] # List; y-coordinate of text line start
+
+ # Group generated paths together, to make the rendered letters
+ # easier to manipulate in Inkscape once generated:
+ parent = node.getparent()
+
+ group = parent.add(Group())
+ group.label = "Hershey Text"
+
+ style = {
+ "stroke": "#000000",
+ "fill": "none",
+ "stroke-linecap": "round",
+ "stroke-linejoin": "round",
+ }
+
+ # Apply rounding to ends to improve final engraved text appearance.
+ group.style = style
+ # Some common variables used in both cases A and B:
+ str_pos = 0 # Position through the full string that we are rendering
+ i = 0 # Dummy(index) variable for looping over letters in string
+ w = 0 # Initial spacing offset
+ w_temp = 0 # Temporary variable for horizontal spacing offset
+ width_this_line = (
+ 0 # Estimated width of characters to be stored on this line
+ )
+
+ """
+ CASE A: Handle flowed text nodes
+ """
+
+ if isinstance(node, FlowRoot):
+
+ try:
+ text_align = node_style["text-align"]
+ # Use text-align, not text-anchor, in flowroot
+ except KeyError:
+ pass
+
+ # selects the flowRegion's child(svg:rect) to get @X and @Y
+ flowref = self.svg.getElement(
+ '/svg:svg//*[@id="%s"]/svg:flowRegion[1]' % the_id
+ )[0]
+
+ if isinstance(flowref, Rectangle):
+ start_x = flowref.left
+ start_y = flowref.top
+ rect_height = flowref.height
+ rect_width = flowref.width
+ bounding_rect = True
+
+ elif isinstance(flowref, Use):
+ pass
+
+ # A <use> element refers to another SVG element via an xlink:href="#blah"
+ # attribute. We will handle the element by doing an XPath search through
+ # the document, looking for the element with the matching id="blah"
+ # attribute. We then recursively process that element after applying
+ # any necessary(x, y) translation.
+ #
+ # Notes:
+ # 1. We ignore the height and width attributes as they do not apply to
+ # path-like elements, and
+ # 2. Even if the use element has visibility="hidden", SVG still calls
+ # for processing the referenced element. The referenced element is
+ # hidden only if its visibility is "inherit" or "hidden".
+ # 3. We may be able to unlink clones using the code in pathmodifier.py
+
+ # The following code can render text flowed into a rectangle object.
+ # HOWEVER, it does not handle the various transformations that could occur
+ # be present on those objects, and does not handle more general cases, such
+ # as a rotated rectangle -- for which text *should* flow in a diamond shape.
+ # For the time being, we skip these and issue a warning.
+ #
+ # refid = flowref.get('xlink:href')
+ # if refid is not None:
+ # # [1:] to ignore leading '#' in reference
+ # path = '//*[@id="%s"]' % refid[1:]
+ # refnode = flowref.xpath(path)
+ # if refnode is not None:
+ # refnode = refnode[0]
+ # if isinstance(refnode, Rectangle):
+ # start_x = refnode.get('x")
+ # start_y = refnode.get('y")
+ # rect_height = refnode.get('height")
+ # rect_width = refnode.get('width")
+ # bounding_rect = True
+
+ if not bounding_rect:
+ self.warn_unflow = True
+ continue
+
+ """
+ Recursively loop through content of the flowroot object,
+ looping through text, flowpara, and other things.
+
+ Create multiple lists: One of text content,
+ others of style that should be applied to that content.
+
+ then, loop through those lists, one line at a time,
+ finding how many words fit on a line, etc.
+ """
+
+ the_style = dict()
+ the_style["font_height"] = font_height
+ the_style["font_family"] = font_family
+ the_style["line_spacing"] = line_spacing
+ the_style["align"] = text_align
+
+ self.recursively_parse_flowroot(node, the_style)
+
+ if self.text_string == "":
+ continue # No convertable text in this SVG element.
+
+ if self.text_string.isspace():
+ continue # No convertable text in this SVG element.
+
+ # Initial vertical offset for the flowed text block:
+ v = 0
+
+ # Record that we are on the first line of the paragraph
+ # for setting the v position of the first line.
+ first_line = True
+
+ # Keep track of text height on first line, for moving entire text box:
+ y_offs_overall = 0
+
+ # Split text by lines AND make a list of how long each
+ # line is, including the newline characters.
+ # We need to keep track of this to match up styling
+ # information to the printable characters.
+
+ text_lines = self.text_string.splitlines()
+ extd_text_lines = self.text_string.splitlines(True)
+ str_pos_eol = 0 # str_pos after end of previous text_line.
+
+ nbsp = "\xa0" # Unicode non-breaking space character
+
+ for line_number, text_line in enumerate(text_lines):
+
+ line_length = len(text_line)
+ extd_line_length = len(extd_text_lines[line_number])
+
+ i = 0 # Position within this text_line.
+
+ # A given text_line may take more than one strip
+ # to render, if it overflows our box width.
+
+ line_start = 0 # Value of i when the current strip started.
+
+ if line_length == 0:
+ str_pos_temp = str_pos_eol
+ char_height = float(self.text_heights[str_pos_temp])
+ charline_spacing = float(self.text_spacings[str_pos_temp])
+ char_v_spacing = charline_spacing * char_height
+ v = v + char_v_spacing
+ else:
+ while i < line_length:
+
+ word_start = (
+ i # Value of i at beginning of the current word.
+ )
+
+ while i < line_length: # Step through the line
+ # until we reach the end of the line or word.
+ # (i.e., until we reach whitespace)
+ character = text_line[
+ i
+ ] # character is unicode(not byte string)
+ str_pos_temp = str_pos_eol + i
+
+ char_height = self.text_heights[str_pos_temp]
+ char_family = self.text_families[str_pos_temp]
+
+ try:
+ _, x_adv, scale_factor = self.get_font_char(
+ char_family, character
+ )
+ except:
+ x_adv = 0
+ scale_factor = 1
+
+ w_temp += x_adv * scale_factor * char_height
+
+ i += 1
+ if character.isspace() and not character == nbsp:
+ break # Break at space, except non-breaking
+
+ render_line = False
+ if (
+ w_temp > rect_width
+ ): # If the word will overflow the box
+ if word_start == line_start:
+ # This is the first word in the strip, so this
+ # word(alone) is wider than the box. Render it.
+ render_line = True
+ else: # Not the first word in the strip.
+ # Render the line up UNTIL this word.
+ render_line = True
+ i = word_start
+ elif i >= line_length:
+ # Render at end of text_line, if not overflowing.
+ render_line = True
+
+ if render_line:
+ # Create group for rendering a strip of text:
+ line_group = group.add(Group())
+
+ w_temp = 0
+ w = 0
+
+ self.new_line = True
+ width_this_line = 0
+ line_max_v_spacing = 0
+
+ j = line_start
+
+ while j < i: # Calculate max height for the strip:
+ str_pos_temp = str_pos_eol + j
+ char_height = float(
+ self.text_heights[str_pos_temp]
+ )
+ charline_spacing = float(
+ self.text_spacings[str_pos_temp]
+ )
+ char_v_spacing = charline_spacing * char_height
+ if char_v_spacing > line_max_v_spacing:
+ line_max_v_spacing = char_v_spacing
+ j = j + 1
+
+ v = v + line_max_v_spacing
+
+ char_data = dict()
+ char_data["vertoffset"] = v
+ char_data["stroke_scale"] = scale_r
+
+ j = line_start
+ while j < i: # Render the strip on the page
+ str_pos = str_pos_eol + j
+
+ char_height = self.text_heights[str_pos]
+ char_family = self.text_families[str_pos]
+ text_align = self.text_aligns[str_pos]
+
+ char_data["char"] = text_line[j]
+ char_data["font_height"] = char_height
+ char_data["font_family"] = char_family
+ char_data["offset"] = w
+
+ w = self.draw_svg_text(char_data, line_group)
+
+ width_this_line = w
+
+ j = j + 1
+ str_pos = str_pos + 1
+
+ line_start = i
+
+ # Alignment for the strip:
+
+ the_transform = None
+ if text_align == "center": # when using text-align
+ the_transform = Transform(
+ translate=(
+ (float(rect_width) - width_this_line)
+ / 2
+ )
+ )
+ elif text_align == "end":
+ the_transform = Transform(
+ translate=(
+ float(rect_width) - width_this_line
+ )
+ )
+ if the_transform is not None:
+ line_group.transform = the_transform
+
+ if first_line:
+ y_offs_overall = (
+ line_max_v_spacing / 3
+ ) # Heuristic
+ first_line = False
+
+ str_pos_eol = str_pos_eol + extd_line_length
+ str_pos = str_pos_eol
+
+ the_transform = Transform(
+ translate=(start_x, float(start_y) - y_offs_overall)
+ )
+
+ else: # If this is a text object, rather than a flowroot object:
+ """
+ CASE B: Handle regular(non-flowroot) text nodes
+ """
+
+ # Use text-anchor, not text-align, in text(not flowroot) elements
+ text_align = node_style("text-anchor")
+
+ """
+ Recursively loop through content of the text object,
+ looping through text, tspan, and other things as necessary.
+ (A recursive search since style elements may be nested.)
+
+ Create multiple lists: One of text content, others of the
+ style that should be applied to that content.
+
+ For each line, want to record the plain text, font size
+ per character, text alignment, and x, y start values
+ for that line)
+
+ (We may need to eventually handle additional text types and
+ tags, as when importing from other SVG sources. We should
+ try to eventually support additional formulations
+ of x, y, dx, dy, etc.
+ https://www.w3.org/TR/SVG/text.html#TSpanElement)
+
+ then, loop through those lists, one line at a time,
+ rendering text onto lines. If the x or y values changed,
+ assume we've started a new line.
+
+ Note: A text element creates a single line
+ of text; it does not create multiline text by including
+ line returns within the text itself. Multiple lines of text
+ are created with multiple text or tspan elements.
+ """
+
+ node_info = dict()
+ node_info["font_height"] = font_height
+ node_info["font_family"] = font_family
+ node_info["anchor"] = text_align
+ node_info["x_pos"] = start_x
+ node_info["y_pos"] = start_y
+ node_info["line_spacing"] = line_spacing
+
+ # Keep track of line number. Used in cases where daughter
+ # tspan elements do not have Y positions given.
+ # Reset to zero on each text element.
+ self.line_number = 0
+
+ self.recursively_parse_text(node, node_info)
+ # self.recursively_parse_text(node, font_height, text_align, start_x, start_y)
+
+ if self.text_string == "":
+ continue # No convertable text in this SVG element.
+ if self.text_string.isspace():
+ continue # No convertable text in this SVG element.
+
+ letter_vals = [q for q in self.text_string]
+ str_len = len(letter_vals)
+
+ # Use a group for each line. This starts the first:
+ line_group = group.add(Group())
+
+ i = 0
+ while i < str_len: # Loop through the entire text of the string.
+
+ try:
+ x_start_line = float(self.text_x[i]) # Start new line here.
+ except ValueError:
+ self.warn_unkern = True
+ return
+
+ y_start_line = float(self.text_y[i])
+
+ while i < str_len:
+ # Inner while loop, that we will break out of,
+ # back to the outer while loop.
+
+ q_val = letter_vals[i]
+ charfont_height = self.text_heights[i]
+
+ char_data = dict()
+ char_data["char"] = q_val
+ char_data["font_family"] = self.text_families[i]
+
+ char_data["font_height"] = charfont_height
+ char_data["offset"] = w
+ char_data["vertoffset"] = 0
+ char_data["stroke_scale"] = scale_r
+
+ w = self.draw_svg_text(char_data, line_group)
+ width_this_line = w
+ w_temp = w
+
+ # Set the alignment if(A) this is the last character in the string
+ # or if the next piece of the string is at a different position
+
+ set_alignment = False
+ i_next = i + 1
+ if i_next >= str_len: # End of the string; last character.
+ set_alignment = True
+ elif (float(self.text_x[i_next]) != x_start_line) or (
+ float(self.text_y[i_next]) != y_start_line
+ ):
+ set_alignment = True
+
+ if set_alignment:
+ text_align = self.text_aligns[i]
+ # Not currently supporting text alignment that changes in the span;
+ # Use the text alignment as of the last character.
+
+ # Left(or "start") alignment is default.
+ # if(text_align == "middle"): Center alignment
+ # if(text_align == "end"): Right alignment
+ #
+ # Strategy: Align every row (left, center, or right)
+ # as it is created.
+
+ x_shift = 0
+ if text_align == "middle": # when using text-anchor
+ x_shift = x_start_line - (width_this_line / 2)
+ elif text_align == "end":
+ x_shift = x_start_line - width_this_line
+ else:
+ x_shift = x_start_line
+
+ y_shift = y_start_line
+
+ the_transform = Transform(translate=(x_shift, y_shift))
+
+ line_group.transform = the_transform
+
+ line_group = group.add(
+ Group()
+ ) # Create new group for this line
+
+ self.new_line = True # Used for managing indent defects
+ w = 0
+ i += 1
+ break
+ i += 1 # Only executed when set_alignment is false.
+
+ the_transform = Transform()
+
+ if len(line_group) == 0:
+ parent = line_group.getparent()
+ parent.remove(line_group)
+
+ # End cases A & B. Apply transform to text/flowroot object:
+
+ if transform is not None:
+ result = Transform(transform) @ the_transform
+ else:
+ result = the_transform
+
+ group.transform = result
+
+ if not self.output_generated:
+ parent = group.getparent()
+ parent.remove(group) # remove empty group
+
+ # Preserve original element?
+ if not self.options.preserve_text and self.output_generated:
+ self.nodes_to_delete.append(node)
+
+ def effect(self):
+ """
+ Main entry point; Execute the extension's function.
+ """
+
+ # Input sanitization:
+ self.options.mode = self.options.mode.strip('"')
+ self.options.fontface = self.options.fontface.strip('"')
+ self.options.otherfont = self.options.otherfont.strip('"')
+ self.options.util_mode = self.options.util_mode.strip('"')
+ self.options.sample_text = self.options.sample_text.strip('"')
+
+ self.doc_transform = [[1.0, 0.0, 0.0], [0.0, 1.0, 0.0]]
+
+ self.find_font_files()
+
+ self.handle_viewbox()
+
+ # Calculate "ideal" effective width of rendered strokes:
+ # Default: 1/800 of page width or height, whichever is smaller
+
+ _rendered_stroke_scale = 1 / (self.PX_PER_INCH * 800.0)
+
+ if self.svg_width is not None:
+ if self.svg_width < self.svg_height:
+ self.render_width = self.svg_width * _rendered_stroke_scale
+ else:
+ self.render_width = self.svg_height * _rendered_stroke_scale
+
+ if self.options.mode == "help":
+ inkex.errormsg(self.help_text)
+ elif self.options.mode == "utilities":
+
+ if self.options.util_mode == "sample":
+ self.font_table()
+ else:
+ self.glyph_table()
+ else:
+ if self.options.ids:
+ # Traverse selected objects
+ for id_ref in self.options.ids:
+ transform = self.recursive_get_encl_transform(
+ self.svg.selection[id_ref]
+ )
+ self.recursively_traverse_svg(
+ [self.svg.selection[id_ref]], transform
+ )
+ else: # Traverse entire document
+ self.recursively_traverse_svg(
+ self.document.getroot(), self.doc_transform
+ )
+
+ for element_to_remove in self.nodes_to_delete:
+ if element_to_remove is not None:
+ parent = element_to_remove.getparent()
+ if parent is not None:
+ parent.remove(element_to_remove)
+
+ if self.font_load_fail:
+ inkex.errormsg(_("Warning: unable to load SVG stroke fonts."))
+
+ if self.warn_unkern:
+ inkex.errormsg(
+ _(
+ "Warning: unable to render text.\n"
+ + "Please use Text > Remove Manual Kerns to convert it prior to use ."
+ )
+ )
+
+ if self.warn_unflow:
+ inkex.errormsg(
+ _(
+ "Warning: unable to convert text flowed into a frame.\n"
+ + "Please use Text > Unflow to convert it prior to use.\n"
+ + "If you are unable to identify the object in question, "
+ + "please contact technical support for help."
+ )
+ )
+
+
+if __name__ == "__main__":
+ Hershey().run()