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

时间:2010-08-10 21:13:14

标签: java svg batik

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

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

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

1 个答案:

答案 0 :(得分:10)

我曾经使用过一种技术在Blogger帖子中嵌入SVG图像 哪个可能适用于此。基本上,这是一个两步过程:

  1. 您序列化要嵌入的SVG,并对其进行URL编码。
  2. 然后使用URL编码的字符串作为SVG使用元素的xlink:href属性中的数据URI。
  3. 这是我用Batik测试过的一个工作示例。假设您要嵌入以下SVG文档circle.svg:

    <?xml version="1.0" standalone="no"?>
    <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" 
      "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
    <svg width="4in" height="4in" id="the_svg"
         viewBox="0 0 4 4" version="1.1"
         xmlns="http://www.w3.org/2000/svg">
        <circle cx="1" cy="1" r="1" fill="blue" stroke="none" id="the_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文档:

    <html>
        <head>
        </head>
        <body>
            <object data="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" width="400" height="400"></object>
        </body>
    </html>
    

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

    <?xml version="1.0" standalone="no"?>
    <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" 
      "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
    <svg width="4in" height="4in" id="the_svg"
         viewBox="0 0 4 4" version="1.1"
         xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
        <use x="0" y="0" width="4" height="4" xlink:href="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#the_svg"/>
    </svg>
    

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