Repository URL to install this package:
Version:
5.3.0 ▾
|
enable
/
svg.py
|
---|
# (C) Copyright 2005-2022 Enthought, Inc., Austin, TX # All rights reserved. # # This software is provided without warranty under the terms of the BSD # license included in LICENSE.txt and may be redistributed only under # the conditions described in the aforementioned license. The license # is also available online at http://www.enthought.com/licenses/BSD.txt # # Thanks for using Enthought open source! """ Chaco's SVG backend :Copyright: ActiveState :License: BSD Style :Author: David Ascher (davida@activestate.com) :Version: $Revision: 1.5 $ """ #### # # Known limitations # # * BUG: Weird behavior with compound plots # * Limitation: text widths are lousy if reportlab is not installed # * Missing feature: rotated text """ Miscellaneous notes: * the way to do links: <a xlink:href="http://www.w3.org"> <ellipse cx="2.5" cy="1.5" rx="2" ry="1" fill="red" /> </a> """ from base64 import b64encode from io import BytesIO, StringIO import os import warnings from numpy import arange, ndarray, ravel # Local, relative Kiva imports from . import affine from . import basecore2d from . import constants from .constants import FILL, FILL_STROKE, EOF_FILL_STROKE, EOF_FILL, STROKE def _strpoints(points): c = StringIO() for x, y in points: c.write("%3.2f,%3.2f " % (x, y)) return c.getvalue() def _mkstyle(kw): return "; ".join([str(k) + ":" + str(v) for k, v in kw.items()]) def default_filter(kw1): kw = {} for (k, v) in kw1.items(): if isinstance(v, tuple): if v[0] != v[1]: kw[k] = v[0] else: kw[k] = v return kw line_cap_map = { constants.CAP_ROUND: "round", constants.CAP_SQUARE: "square", constants.CAP_BUTT: "butt", } line_join_map = { constants.JOIN_ROUND: "round", constants.JOIN_BEVEL: "bevel", constants.JOIN_MITER: "miter", } xmltemplate = """<?xml version="1.0"?> <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.0//EN" "http://www.w3.org/TR/2001/REC-SVG-20010904/DTD/svg10.dtd"> <svg xmlns="http://www.w3.org/2000/svg" xmlns:text="http://xmlns.graougraou.com/svg/text/" xmlns:a3="http://ns.adobe.com/AdobeSVGViewerExtensions/3.0/" xmlns:xlink="http://www.w3.org/1999/xlink" a3:scriptImplementation="Adobe" width="%(width)f" height="%(height)f" viewBox="0 0 %(width)f %(height)f" > <g transform="translate(0,%(height)f)"> <g transform="scale(1,-1)"> %(contents)s </g> </g> </svg> """ htmltemplate = """<html xmlns:svg="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <object id="AdobeSVG" CLASSID="clsid:78156a80-c6a1-4bbf-8e6a-3cd390eeb4e2"> </object> <?import namespace="svg" implementation="#AdobeSVG"?> <body> <svg:svg width="100%%" height="100%%" viewBox="0 0 %(width)f %(height)f"> %(contents)s </svg:svg> </body> </html> """ try: # expensive way of computing string widths import reportlab.pdfbase.pdfmetrics as pdfmetrics import reportlab.pdfbase._fontdata as _fontdata _reportlab_loaded = 1 except ImportError: from . import pdfmetrics from . import _fontdata _reportlab_loaded = 0 # This backend has no compiled path object, yet. class CompiledPath(object): pass _clip_counter = 0 class GraphicsContext(basecore2d.GraphicsContextBase): def __init__(self, size, *args, **kwargs): super().__init__(self, size, *args, **kwargs) self.size = size self._height = size[1] self.contents = StringIO() self._clipmap = {} def render(self, format): assert format == "svg" height, width = self.size contents = ( self.contents.getvalue() .replace("<svg:", "<") .replace("</svg:", "</") ) return xmltemplate % locals() def clear(self): self.contents = StringIO() def width(self): return self.size[0] def height(self): return self.size[1] def save(self, filename, file_format=None, pil_options=None): with open(filename, "w", encoding="utf-8") as f: ext = os.path.splitext(filename)[1] if ext == ".svg": template = xmltemplate width, height = self.size contents = ( self.contents.getvalue() .replace("<svg:", "<") .replace("</svg:", "</") ) elif ext == ".html": width, height = self.size[0] * 3, self.size[1] * 3 contents = self.contents.getvalue() template = htmltemplate else: raise ValueError("don't know how to write a %s file" % ext) f.write(template % locals()) # Text handling code def set_font(self, font): self.face_name = font.face_name self.font_size = font.size # actual implementation =) def device_show_text(self, text): ttm = self.get_text_matrix() ctm = self.get_ctm() # not device_ctm!! m = affine.concat(ctm, ttm) # height = self.get_full_text_extent(text)[1] a, b, c, d, tx, ty = affine.affine_params(m) transform = ( "matrix(%(a)f,%(b)f,%(c)f,%(d)f,%(tx)f,%(ty)f) scale(1,-1)" % locals() ) self._emit( "text", contents=text, kw={ "font-family": self.face_name, "font-size": str(self.font_size), "xml:space": "preserve", "transform": transform, }, ) def get_full_text_extent(self, text): ascent, descent = _fontdata.ascent_descent[self.face_name] descent = (-descent) * self.font_size / 1000.0 ascent = ascent * self.font_size / 1000.0 height = ascent + descent width = pdfmetrics.stringWidth(text, self.face_name, self.font_size) return ( width, height, descent, height * 1.2, ) # assume leading of 1.2*height def device_draw_image(self, img, rect): """ draw_image(img_gc, rect=(x,y,w,h)) Draws another gc into this one. If 'rect' is not provided, then the image gc is drawn into this one, rooted at (0,0) and at full pixel size. If 'rect' is provided, then the image is resized into the (w,h) given and drawn into this GC at point (x,y). img_gc is either a Numeric array (WxHx3 or WxHx4) or a PIL Image. Requires the Python Imaging Library (PIL). """ from PIL import Image # We turn img into a PIL object, since that is what ReportLab # requires. if isinstance(img, ndarray): # From numpy array pil_img = Image.fromarray(img) elif isinstance(img, Image.Image): pil_img = img elif hasattr(img, "bmp_array"): # An offscreen kiva agg context if hasattr(img, "convert_pixel_format"): img = img.convert_pixel_format("rgba32", inplace=0) pil_img = Image.fromarray(img.bmp_array) else: warnings.warn( "Cannot render image of type %r into SVG context." % type(img) ) return if rect is None: rect = (0, 0, pil_img.width, pil_img.height) left, top, width, height = rect if width != pil_img.width or height != pil_img.height: # This is not strictly required. pil_img = pil_img.resize((int(width), int(height)), Image.NEAREST) png_buffer = BytesIO() pil_img.save(png_buffer, "png") b64_img_data = b64encode(png_buffer.getvalue()).decode('utf8') png_buffer.close() # Draw the actual image. m = self.get_ctm() # Place the image on the page. # Using bottom instead of top here to account for the y-flip. m = affine.translate(m, left, height + top) transform = ( "matrix(%f,%f,%f,%f,%f,%f) scale(1,-1)" % affine.affine_params(m) ) # Flip y to reverse the flip at the start of the document. image_data = "data:image/png;base64," + b64_img_data self._emit( "image", transform=transform, width=str(width), height=str(height), preserveAspectRatio="none", kw={"xlink:href": image_data}, ) def device_fill_points(self, points, mode): points = self._fixpoints(points) if mode in (FILL, FILL_STROKE, EOF_FILL_STROKE): fill = self._color(self.state.fill_color) else: fill = "none" if mode in (STROKE, FILL_STROKE, EOF_FILL_STROKE): stroke = self._color(self.state.line_color) else: stroke = "none" if mode in (EOF_FILL_STROKE, EOF_FILL): rule = "evenodd" else: rule = "nonzero" linecap = line_cap_map[self.state.line_cap] linejoin = line_join_map[self.state.line_join] dasharray = self._dasharray() width = "%3.3f" % self.state.line_width clip_id = getattr(self.state, "_clip_id", None) if clip_id: clip = "url(#" + clip_id + ")" else: clip = None a, b, c, d, tx, ty = affine.affine_params(self.get_ctm()) transform = "matrix(%(a)f,%(b)f,%(c)f,%(d)f,%(tx)f,%(ty)f)" % locals() if mode == STROKE: opacity = "%1.3f" % self.state.line_color[-1] self._emit( "polyline", transform=transform, points=_strpoints(points), kw=default_filter({"clip-path": (clip, None)}), style=_mkstyle( default_filter( { "opacity": (opacity, "1.000"), "stroke": stroke, "fill": "none", "stroke-width": (width, "1.000"), "stroke-linejoin": (linejoin, "miter"), "stroke-linecap": (linecap, "butt"), "stroke-dasharray": (dasharray, "none"), } ) ), ) else: opacity = "%1.3f" % self.state.fill_color[-1] self._emit( "polygon", transform=transform, points=_strpoints(points), kw=default_filter({"clip-path": (clip, None)}), style=_mkstyle( default_filter( { "opacity": (opacity, "1.000"), "stroke-width": (width, "1.000"), "fill": fill, "fill-rule": rule, "stroke": stroke, "stroke-linejoin": (linejoin, "miter"), "stroke-linecap": (linecap, "butt"), "stroke-dasharray": (dasharray, "none"), } ) ), ) def device_stroke_points(self, points, mode): # handled by device_fill_points pass def _build(self, elname, contents=None, **kw): x = "<" + elname + " " for k, v in kw.items(): if isinstance(v, float): v = "%3.3f" % v elif isinstance(v, int): v = "%d" % v else: v = "%s" % str(v) x += k + '="' + v + '" ' if contents is None: x += "/>\n" else: x += ">" if elname != "text": x += "\n" x += contents x += "</" + elname + ">\n" return x def _debug_draw_clipping_path(self, x, y, width, height): a, b, c, d, tx, ty = affine.affine_params(self.get_ctm()) transform = "matrix(%(a)f,%(b)f,%(c)f,%(d)f,%(tx)f,%(ty)f)" % locals() self._emit( "rect", x=x, y=y, width=width, height=height, transform=transform, style=_mkstyle( {"stroke-width": 5, "fill": "none", "stroke": "green"} ), ) def device_set_clipping_path(self, x, y, width, height): # self._debug_draw_clipping_path(x, y, width, height) # return global _clip_counter self.state._clip_id = "clip_%d" % _clip_counter _clip_counter += 1 x, y = self._fixpoints([[x, y]])[0] a, b, c, d, tx, ty = affine.affine_params(self.get_ctm()) transform = "matrix(%(a)f,%(b)f,%(c)f,%(d)f,%(tx)f,%(ty)f)" % locals() rect = self._build("rect", x=x, y=y, width=width, height=height) clippath = self._build( "clipPath", contents=rect, id=self.state._clip_id ) self._emit("g", transform=transform, contents=clippath) def device_destroy_clipping_path(self): self.state._clip_id = None # utility routines def _fixpoints(self, points): return points # convert lines from Kiva coordinate space to PIL coordinate space # XXX I suspect this is the location of the bug w.r.t. compound graphs # and "global" sizing. # XXX this should be made more efficient for NumPy arrays np = [] for (x, y) in points: np.append((x, self._height - y)) return np def _emit(self, name, contents=None, kw={}, **otherkw): self.contents.write("<svg:%(name)s " % locals()) for k, v in kw.items(): self.contents.write('%(k)s="%(v)s" ' % locals()) for k, v in otherkw.items(): self.contents.write('%(k)s="%(v)s" ' % locals()) if contents is None: self.contents.write("/>\n") else: self.contents.write(">") if name != "text": self.contents.write("\n") self.contents.write(contents) self.contents.write("</svg:" + name + ">\n") def _color(self, color): r, g, b, a = color return "#%02x%02x%02x" % (int(r * 255), int(g * 255), int(b * 255)) def _dasharray(self): dasharray = "" for x in self.state.line_dash: if type(x) == type(arange(3)): # why is this so hard? x = ravel(x)[0] dasharray += " " + "%3.2f" % x if not dasharray or dasharray == " 0.00 0.00": dasharray = "none" return dasharray # noops which seem to be needed def device_update_line_state(self): pass def device_update_fill_state(self): pass def _repr_svg_(self): """ Return a the current contents of the context as SVG text. This provides Jupyter and IPython compatibility, so that the graphics context can be displayed in the Jupyter Notebook or the IPython Qt console. Returns ------- svg : str The contents of the context as an SVG string. """ return self.render('svg') def font_metrics_provider(): return GraphicsContext((1, 1)) SVGGC = GraphicsContext # for b/w compatibility