提问者:小点点

使用vanilla W3C Document和Apache Batik SVG光栅化器


由于SVG是一个常规的XML文件,ImageTranscoder. transcode()API接受org.w3c.dom.Document,相应的TranscoderInput构造函数接受org.w3c.dom.Document;人们会期望使用JavaXML解析器加载和解析文件可以工作:

TranscoderInput input = new TranscoderInput(loadSvgDocument(new FileInputStream(svgFile)));
BufferedImageTranscoder t = new BufferedImageTranscoder();
t.transcode(input, null);

其中loadSvgDocument()方法定义为:

Document loadSvgDocument(String svgFileName, InputStream is) {
    DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();
    // using stock Java 8 XML parser
    Document document;
    try {
        DocumentBuilder db = dbf.newDocumentBuilder(); 
        document = db.parse(is);
    } catch (...) {...}
    return document;
}

它不工作。我得到奇怪的铸造例外。

Exception in thread "main" java.lang.ClassCastException: org.apache.batik.dom.GenericElement cannot be cast to org.w3c.dom.svg.SVGSVGElement
at org.apache.batik.anim.dom.SVGOMDocument.getRootElement(SVGOMDocument.java:235)
at org.apache.batik.transcoder.SVGAbstractTranscoder.transcode(SVGAbstractTranscoder.java:193)
at org.apache.batik.transcoder.image.ImageTranscoder.transcode(ImageTranscoder.java:92)
at org.apache.batik.transcoder.XMLAbstractTranscoder.transcode(XMLAbstractTranscoder.java:142)
at org.apache.batik.transcoder.SVGAbstractTranscoder.transcode(SVGAbstractTranscoder.java:156)

注意:classBufferedImageTranscoder是我的类,根据Batik蓝图创建,扩展ImageTranscoder进而扩展上面堆栈跟踪中提到的SVGAbstractTranscoder

不幸的是,我不能使用Batik自己的解析器,SAXSVGDocumentFactory

String parser = XMLResourceDescriptor.getXMLParserClassName();
SAXSVGDocumentFactory f = new SAXSVGDocumentFactory(parser);
svgDocument = (SVGDocument) f.createDocument(..);

我正在尝试渲染Papirus SVG图标,但它们都有

Exception in thread "main" java.lang.RuntimeException: Unsupport SVG version '1'
at org.apache.batik.anim.dom.SAXSVGDocumentFactory.getDOMImplementation(SAXSVGDocumentFactory.java:327)
at org.apache.batik.dom.util.SAXDocumentFactory.startElement(SAXDocumentFactory.java:640)
. . .
at org.apache.batik.anim.dom.SAXSVGDocumentFactory.createDocument(SAXSVGDocumentFactory.java:225)

将文件中的version="1"更改为version="1.0"可以解决问题,并且图标呈现得很好。但是有成百上千个图标,修复它们很乏味,我会有效地为他们的项目创建一个端口。这对我来说不是一个前进的方向。更容易的是在运行时修复,使用DOMAPI:

Element svgDocumentNode = svgDocument.getDocumentElement();
String svgVersion = svgDocumentNode.getAttribute("version");
if (svgVersion.equals("1")) {
    svgDocumentNode.setAttribute("version", "1.0");
}

但是那只能用存量JavaXML解析器来完成,BatikXML解析器吹得太早,在这段代码可以到达之前,在Document生成之前。但是当我使用存量XML解析器,进行版本修复时,然后Batik Transcoder(光栅化器)不喜欢它。所以我在这里碰壁了。

是否有来自股票XML解析器的转换器org. w3c.dom.Document和Batik兼容org.w3c.dom.svg.SVGDocument


共1个答案

匿名用户

好的,我找到了绕过问题的解决方案。幸运的是classSAXSVGDocumentFactory可以很容易地子类和关键方法getDOM实现()覆盖。

protected Document loadSvgDocument(InputStream is) {
    String parser = XMLResourceDescriptor.getXMLParserClassName();
    SAXSVGDocumentFactory f = new LenientSaxSvgDocumentFactory(parser);
    SVGDocument svgDocument;
    try {
        svgDocument = (SVGDocument) f.createDocument("aaa", is);
    } catch (...) {
        ...
    }
    return svgDocument;
}

static class LenientSaxSvgDocumentFactory extends SAXSVGDocumentFactory {
    public LenientSaxSvgDocumentFactory(String parser) {
        super(parser);
    }

    @Override
    public DOMImplementation getDOMImplementation(String ver) {
        // code is mostly rip-off from original Apache Batik 1.9 code
        // only the condition was extended to accept also "1" string
        if (ver == null || ver.length() == 0
                || ver.equals("1.0") || ver.equals("1.1") || ver.equals("1")) {
            return SVGDOMImplementation.getDOMImplementation();
        } else if (ver.equals("1.2")) {
            return SVG12DOMImplementation.getDOMImplementation();
        }
        throw new RuntimeException("Unsupported SVG version '" + ver + "'");
    }
}

这次我很幸运,但是主要问题仍然存在:是否有一个来自股票XML解析器的转换器产生org. w3c.dom.Document和Batik兼容org.w3c.dom.svg.SVGDocument