由于SVG是常规XML文件而ImageTranscoder.transcode()API接受org.w3c.dom.Document,因此相应的TranscoderInput构造函数接受org.w3c.dom.Document;可以预期用Java stock XML解析器加载和解析文件会起作用:
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)
注意:类BufferedImageTranscoder
是我的类,按照Batik蓝图创建,扩展ImageTranscoder
,后者又扩展了上面堆栈跟踪中提到的SVGAbstractTranscoder
。
不幸的是我无法使用Batik自己的解析器SAXSVGDocumentFactory
:
String parser = XMLResourceDescriptor.getXMLParserClassName();
SAXSVGDocumentFactory f = new SAXSVGDocumentFactory(parser);
svgDocument = (SVGDocument) f.createDocument(..);
我正在尝试呈现Papirus SVG icons,但他们都有<svg ... version="1">
,而SAXSVGDocumentFactory不喜欢这样,并且在createDocument(..)
Unsupport SVG version '1'
上失败。它们可能意味着不受支持的。
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"
可以解决问题,并且可以很好地为我呈现图标。但是有数百(数千)个图标,修复它们都很乏味,我会有效地创建一个项目的端口。这对我来说不是一个前进的方法。使用DOM API更容易在运行时进行修复:
Element svgDocumentNode = svgDocument.getDocumentElement();
String svgVersion = svgDocumentNode.getAttribute("version");
if (svgVersion.equals("1")) {
svgDocumentNode.setAttribute("version", "1.0");
}
但是,只有使用库存Java XML解析器才能完成,在生成Document之前,Batik XML解析器会在达到此代码之前过早地破坏。但是,当我使用stock XML解析器,进行版本修复时,Batik Transcoder(rasterizer)不喜欢它。所以我在这里碰到了一堵墙。
是否有来自库存XML解析器的转换器生成org.w3c.dom.Document
和Batik兼容org.w3c.dom.svg.SVGDocument
?
答案 0 :(得分:1)
好的,我找到了解决问题的解决方案。幸运的是,类SAXSVGDocumentFactory
可以很容易地成为子类和关键方法
getDOMImplementation()
覆盖。
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
?