将 svg 转换为 png 时如何包含 CSS 样式

How to Include CSS style when converting svg to png

我创建了一个简单的 SVG 元素,当点击按钮时它会下载到 png,我的解决方案类似于 here


基本思路是:
1.svg 到 canvas
2.canvas 到 dataUrl
3.trigger 从 dataUrl 下载

问题是在下载 png 文件时,它不包含应用于 svg 的 css 样式 my solution result

注意 - 我知道有一个变通解决方案,通过在元素 上移动样式 "inline" ,如 或通过挖掘 [= =60=] 树并使用 getComputedStyle(element,null);

问题:
1.what是这个问题的真正原因和解决方案。
(GPU加速是否相关?)
2.how 在 Fontface

中使用自定义字体时,我仍然克服了这个问题
 <button id="btn">svg to png</button>

  <svg id="svg" width="200" height="200">
    <circle cx="50" cy="50" r="30" />
    <text class="svgTxt" x="0" y="100">Hen's SVG Image</text>
  </svg>
  <canvas id="canvas"  width="200" height="200"></canvas>

我的CSS:

  /*adding exo2 font*/
    @font-face {
    font-family: 'exo_2black';
    src: url('./exo2font/Exo2-Black-webfont.eot');
    src: url('./exo2font/Exo2-Black-webfont.eot?#iefix') format('embedded-opentype'),
         url('./exo2font/Exo2-Black-webfont.woff') format('woff'),
         url('./exo2font/Exo2-Black-webfont.ttf') format('truetype'),
         url('./exo2font/Exo2-Black-webfont.svg#exo_2black') format('svg');
    font-weight: normal;
    font-style: normal;

}
/*change circle color depends on window size*/
@media screen and (min-width: 480px) {
    svg circle {
        fill: lightgreen;
    }
}
/*style on the svg text*/
    .svgTxt{
      font-family: 'exo_2black';
      font-size: 30px;
      fill: red;
    }

我的代码:

  //reference to elements
    var btn = document.querySelector('#btn');
    var svg = document.getElementById('svg');
    var svgTexts = svg.getElementsByTagName('text');
    var canvas = document.getElementById('canvas');
    //Style definitions for svg elements defined in stylesheets are not applied to the generated canvas. This can be patched by adding style definitions to the svg elements before calling canvg.
  //3.trigger download from dataUrl
    function triggerDownload(imgURI) {
      var evt = new MouseEvent('click', {
        view: window,
        bubbles: false,
        cancelable: true
      });

      var a = document.createElement('a');
      a.setAttribute('download', 'hen_saved_image.png');
      a.setAttribute('href', imgURI);
      a.setAttribute('target', '_blank');
      a.dispatchEvent(evt);
    }
    //btn click event
    btn.addEventListener('click', function () {
      // 1.svg to canvas
      var ctx = canvas.getContext('2d');
      var data = (new XMLSerializer()).serializeToString(svg);//serialize the svg element to string
      var DOMURL = window.URL || window.webkitURL || window;
      var img = new Image();
      var svgBlob = new Blob([data], { type: 'image/svg+xml;charset=utf-8' });//A blob object represents a chuck of bytes that holds data of a file.
      var url = DOMURL.createObjectURL(svgBlob);//creates a DOMString containing an URL representing the object given in paramete
      $('svg').append(deletedSVGText);
      img.onload = function () {
        ctx.drawImage(img, 0, 0);
        DOMURL.revokeObjectURL(url);
        // 2.canvas to dataUrl
        var imgURI = canvas
          .toDataURL('image/png')
          .replace('image/png', 'image/octet-stream');// returns a data URI containing a representation of the image in the format specified by the type parameter

        triggerDownload(imgURI);
      };
      img.src = url;
    });

问题一(前半部分): 真正的原因是什么(跟GPU加速有关系吗?)

不,GPU 加速与它无关。
最广泛的原因是 隐私

要使用 drawImage 绘制 svg,您必须将 svg 作为外部文档加载到 <img> 标签内。 SVG 可能是一种用于资源加载的非常复杂的图像格式(它实际上可以需要任何 HTML 文档可能需要的任何类型的资源)。因此,规范中已经指出,与 <iframe> 元素或 <object> 或类似元素的安全性相同的安全性应适用于 <img> 内容,甚至更严格:

<img>内容不能要求任何外部资源,也不能访问主文档。

问题一(后半部分)以及本题的解决方法

您指出了一些已经回答了它的 SO 问题,您还可以在从中创建 Blob 之前,将主文档中的所有样式表包含在已解析的 svg 节点内的 <style> 标记中。

问题二: "how i still overcome this issue when using a custom font with Fontface"

对于外部资源,您必须将其编码为 dataURI,并在创建 Blob 之前将其包含在您的 svg 节点中。特别是对于字体,您可以在 <style> 元素中设置 font-face 属性。

所以最后,你的 svg 会是这样的

<defs>
  <style>
    /* all your parsed styles in here */
    @font-face {
     font-family: foo;
     src: url('data:application/font-woff;charset=utf-8;base64,...')
    }
  </style>
</defs>

在提取其标记之前本身。