由于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
?
好的,我找到了绕过问题的解决方案。幸运的是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
?