使用带有元素的另一个svg图像创建一个带有嵌入式base64字符串的SVG图像

我有一个SVG文件,其中包含几个元素(如路径,圆形,矩形等)。

我想将该SVG文件转换为具有嵌入式base64数据而非多个元素的SVG。 可以用Batik做到吗?

我正在进行的项目要求我只使用Java库。

有一种技术我曾经在Bloggerpost中嵌入SVG图像,这可能对此有用。 基本上,这是一个两步过程:

  1. 您序列化要嵌入的SVG,并对其进行URL编码。
  2. 然后,使用URL编码的字符串作为SVG use元素的xlink:href属性中的数据URI。

这是我用Batik测试过的一个工作示例。 假设您要嵌入以下SVG文档circle.svg:

     

您可以通过将其路径传递给以下小型Rhino脚本对其进行URL编码:

 #!/usr/bin/env rhino print(escape(readFile(arguments[0]))) 

当然,如果您希望以编程方式在Java中执行此操作,那么您将需要一种特定于Java的方法来序列化SVG文档并对字符串进行URL编码。

这会将文档作为URL编码的字符串:

 %3C%3Fxml%20version%3D%221.0%22%20standalone%3D%22no%22%3F%3E%0A%3C%21DOCTYPE%20svg%20PUBLIC%20%22-//W3C//DTD%20SVG%201.1//EN%22%20%0A%20%20%22http%3A//www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd%22%3E%0A%3Csvg%20width%3D%224in%22%20height%3D%224in%22%20id%3D%22the_svg%22%0A%20%20%20%20%20viewBox%3D%220%200%204%204%22%20version%3D%221.1%22%0A%20%20%20%20%20xmlns%3D%22http%3A//www.w3.org/2000/svg%22%3E%0A%09%3Ccircle%20cx%3D%221%22%20cy%3D%221%22%20r%3D%221%22%20fill%3D%22blue%22%20stroke%3D%22none%22%20id%3D%22the_circle%22/%3E%0A%3C/svg%3E%0A%0A 

然后,您可以在数据URI中使用它来嵌入此文档,如下所示:

 data:image/svg+xml,%3C%3Fxml%20version%3D%221.0%22%20standalone%3D%22no%22%3F%3E%0A%3C%21DOCTYPE%20svg%20PUBLIC%20%22-//W3C//DTD%20SVG%201.1//EN%22%20%0A%20%20%22http%3A//www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd%22%3E%0A%3Csvg%20width%3D%224in%22%20height%3D%224in%22%20id%3D%22the_svg%22%0A%20%20%20%20%20viewBox%3D%220%200%204%204%22%20version%3D%221.1%22%0A%20%20%20%20%20xmlns%3D%22http%3A//www.w3.org/2000/svg%22%3E%0A%09%3Ccircle%20cx%3D%221%22%20cy%3D%221%22%20r%3D%221%22%20fill%3D%22blue%22%20stroke%3D%22none%22%20id%3D%22the_circle%22/%3E%0A%3C/svg%3E%0A%0A 

例如,以下HTML文档使用对象标记和数据URI来嵌入SVG文档:

        

您可以使用SVG“use”元素的xlink:href属性执行相同的操作,但有一点需要注意:引用完整文档是非法的。 相反,您需要通过其id引用文档中的元素,并且该元素将被深度克隆到SVG主机文档中。 在此示例中,SVG文档根元素由其id“the_svg”引用(请注意URI末尾的哈希标记)。

      

仅供参考,这在Batik 1.7(在Squiggle浏览器中测试)中效果很好,但不适用于Chromium或Firefox。