由于 SVG 是常规 XML 文件,并且 ImageTranscoder.transcode() API 接受 org.w3c.dom.Document,因此相应的 TranscoderInput 构造函数接受 org.w3c.dom.Document;人们期望使用 Java 通用 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 解析器来完成,Batik XML 解析器过早地崩溃了,在到达此代码之前,在生成 Document 之前。但是当我使用库存 XML 解析器时,进行版本修复,然后 Batik Transcoder(光栅化器)不喜欢它。所以我在这里碰壁了。
是否有一个由现有 XML 解析器生成的转换器 org.w3c.dom.Document
和 batik 兼容org.w3c.dom.svg.SVGDocument
?
最佳答案
好的,我找到了绕过该问题的解决方案。幸运的是,类 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
?
关于java - 使用普通 W3C 文档和 Apache Batik SVG 光栅器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44029119/