2013-03-10 184 views
1

我在使用下面的代碼將一些SVG渲染爲PNG數據時遇到了一些麻煩。由於該圖像最終旨在用作HTTP響應的主體,所以使用了StringIO編寫器。使用pyrsvg和Cairo在Python中渲染SVG的問題

結果是正確尺寸的PNG,但它是完全透明的。

import rsvg 
import cairo 
import StringIO 

def render_svg_to_png(svg_data): 
    # Render 
    svg = rsvg.Handle() 
    svg.write(buffer=svg_data) 

    img = cairo.ImageSurface(cairo.FORMAT_ARGB32, 
          svg.props.width, 
          svg.props.height) 
    ctx = cairo.Context(img) 
    svg.render_cairo(ctx) 

    # Write to StringIO 
    png_io = StringIO.StringIO() 
    img.write_to_png(png_io) 
    img.finish() 

    return png_io.getvalue() 

所呈現的SVG數據如下:

<?xml version="1.0" encoding="UTF-8" standalone="no"?> 
    <!-- Created with Inkscape (http://www.inkscape.org/) --> 

<svg 
    xmlns:dc="http://purl.org/dc/elements/1.1/" 
    xmlns:cc="http://creativecommons.org/ns#" 
    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" 
    xmlns:svg="http://www.w3.org/2000/svg" 
    xmlns="http://www.w3.org/2000/svg" 
    xmlns:xlink="http://www.w3.org/1999/xlink" 
    version="1.1" 
    width="40" 
    height="40" 
    id="svg2"> 
    <defs 
     id="defs4"> 
     <linearGradient 
     id="linearGradient3769"> 
     <stop 
      id="stop3771" 
      style="stop-color:#ffffff;stop-opacity:1" 
      offset="0" /> 
     <stop 
      id="stop3773" 
      style="stop-color:#000000;stop-opacity:1" 
      offset="1" /> 
     </linearGradient> 
     <radialGradient 
     cx="13.895907" 
     cy="15.277355" 
     r="19.6875" 
     fx="13.895907" 
     fy="15.277355" 
     id="radialGradient3775" 
     xlink:href="#linearGradient3769" 
     gradientUnits="userSpaceOnUse" 
     gradientTransform="matrix(0.96746783,-0.96746783,0.95701248,0.95701249,-12.884457,12.690532)" /> 
    </defs> 
    <metadata 
     id="metadata7"> 
     <rdf:RDF> 
     <cc:Work 
      rdf:about=""> 
      <dc:format>image/svg+xml</dc:format> 
      <dc:type 
      rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> 
      <dc:title></dc:title> 
     </cc:Work> 
     </rdf:RDF> 
    </metadata> 
    <g 
     transform="translate(0,-1012.3622)" 
     id="layer1"> 
     <path 
      d="m 39.375,19.776785 a 19.6875,19.6875 0 1 1 -39.375,0 19.6875,19.6875 0 1 1 39.375,0 z" 
      transform="matrix(1.015873,0,0,1.015873,0,1012.2715)" 
      id="path2991" 
      style="fill:url(#radialGradient3775);fill-opacity:1;fill-rule:nonzero;stroke:none" /> 
    </g> 
</svg> 

回答

1

爲我工作好這裏:

import rsvg 
import cairo 
import StringIO 

def render_svg_to_png(svg_data): 
    # Render 
    svg = rsvg.Handle(data=svg_data) 
    img = cairo.ImageSurface(cairo.FORMAT_ARGB32, 
     svg.props.width, 
     svg.props.height) 
    ctx = cairo.Context(img) 
    svg.render_cairo(ctx) 

    # Write to StringIO 
    png_io = StringIO.StringIO() 
    img.write_to_png(png_io) 

    return png_io.getvalue() 

svg_data = open('test.svg', 'r').read() 
print render_svg_to_png(svg_data) 

與輸出重定向冉到一個.png文件,並證實了它想出了在圖像查看程序中可以。

+0

問題實際上是使用ElementTree來修改XML。來自ET的XML輸出的命名空間已經搞亂了。這並沒有打擾Chrome,我用它來測試代碼,但似乎讓rsvg感到不安。我沒有注意到這一點,直到我更仔細地查看ET的輸出。切換到minidom解決了這個問題。 – dlp 2013-03-13 14:20:27