Java 使用蜡染读取SVG文件时使用自定义元素工厂

Java 使用蜡染读取SVG文件时使用自定义元素工厂,java,svg,batik,Java,Svg,Batik,我试图读取从旧程序导出的更大(1MB)的svg文件。我没有访问此程序或数据的权限。我只有这个导出的svg文件,可能需要定期从这个源导入一个新导出的svg文件。我发现使用蜡染的第一个问题是,读取此文件时非常严格。例如,Firefox读取和显示此文件没有问题。蜡染(包括Squiggle)无法显示,因为文档中有一些“自定义”标记。我得到的例外是 org.w3c.dom.DOMException: The current document is unable to create an element o

我试图读取从旧程序导出的更大(1MB)的svg文件。我没有访问此程序或数据的权限。我只有这个导出的svg文件,可能需要定期从这个源导入一个新导出的svg文件。我发现使用蜡染的第一个问题是,读取此文件时非常严格。例如,Firefox读取和显示此文件没有问题。蜡染(包括Squiggle)无法显示,因为文档中有一些“自定义”标记。我得到的例外是

org.w3c.dom.DOMException: The current document is unable to create an element of the requested type (namespace: http://www.w3.org/2000/svg, name: menu).
    at org.apache.batik.dom.AbstractNode.createDOMException(AbstractNode.java:408)
    at org.apache.batik.dom.svg.SVGDOMImplementation.createElementNS(SVGDOMImplementation.java:211)
    at org.apache.batik.dom.svg.SVGOMDocument.createElementNS(SVGOMDocument.java:372)
    at org.apache.batik.dom.util.SAXDocumentFactory.startElement(SAXDocumentFactory.java:625)
    at com.sun.org.apache.xerces.internal.parsers.AbstractSAXParser.startElement(AbstractSAXParser.java:501)
    at com.sun.org.apache.xerces.internal.impl.XMLNSDocumentScannerImpl.scanStartElement(XMLNSDocumentScannerImpl.java:400)
    at com.sun.org.apache.xerces.internal.impl.XMLDocumentFragmentScannerImpl$FragmentContentDriver.next(XMLDocumentFragmentScannerImpl.java:2756)
    at com.sun.org.apache.xerces.internal.impl.XMLDocumentScannerImpl.next(XMLDocumentScannerImpl.java:647)
    at com.sun.org.apache.xerces.internal.impl.XMLNSDocumentScannerImpl.next(XMLNSDocumentScannerImpl.java:140)
    at com.sun.org.apache.xerces.internal.impl.XMLDocumentFragmentScannerImpl.scanDocument(XMLDocumentFragmentScannerImpl.java:511)
    at com.sun.org.apache.xerces.internal.parsers.XML11Configuration.parse(XML11Configuration.java:808)
    at com.sun.org.apache.xerces.internal.parsers.XML11Configuration.parse(XML11Configuration.java:737)
    at com.sun.org.apache.xerces.internal.parsers.XMLParser.parse(XMLParser.java:119)
    at com.sun.org.apache.xerces.internal.parsers.AbstractSAXParser.parse(AbstractSAXParser.java:1205)
    at com.sun.org.apache.xerces.internal.jaxp.SAXParserImpl$JAXPSAXParser.parse(SAXParserImpl.java:522)
    at org.apache.batik.dom.util.SAXDocumentFactory.createDocument(SAXDocumentFactory.java:431)
    at org.apache.batik.dom.util.SAXDocumentFactory.createDocument(SAXDocumentFactory.java:349)
    at org.apache.batik.dom.svg.SAXSVGDocumentFactory.createDocument(SAXSVGDocumentFactory.java:200)
    at com.samsix.nrg.io.SvgFileImporter.importFile(SvgFileImporter.java:74)
。。。因此,使用这个不完全有用的链接,我写了以下内容

    public ImportReport importFile( final String    uri )
    throws
        IOException
{
    String parser = XMLResourceDescriptor.getXMLParserClassName();
    SAXSVGDocumentFactory factory = new SAXSVGDocumentFactory(parser);

    SVGDOMImplementation    implementation = (SVGDOMImplementation) factory.getDOMImplementation( null );

    implementation.registerCustomElementFactory( "http://www.w3.org/2000/svg",
                                                 "menu",
                                                 new ExtensibleDOMImplementation.ElementFactory() {
                                                    @Override
                                                    public Element create( final String prefix,
                                                                           final Document doc )
                                                    {
                                                        System.out.println( "Element.create: " + prefix );
                                                        return new GenericElement( "prefix",
                                                                                   (org.apache.batik.dom.AbstractDocument) doc );
                                                    }
                                                } );

    SVGDocument    doc = (SVGDocument) factory.createDocument( uri );

    System.out.println( doc.getDocumentURI() );
}

但是我得到了同样的错误,System.out.println()语句从未命中,因此它显然没有正确注册我的工厂。另一方面,是否有必要降低Batik的严格程度,以便它像Firefox一样跳过它不理解的内容?

调查表明,不允许您创建拦截SVG命名空间中元素的DOM扩展;它会在异常点抛出异常,否则它会跳转到查找已注册工厂的代码。我不知道这是否是故意的。 有趣的是,在当前版本的中似乎根本没有定义
元素;它要么长期被弃用,要么是一个被放置在错误命名空间中的扩展。啊


最简单的解决方法可能是使用XSLT样式表将
元素重新映射到另一个名称空间-这是一个非常简单的样式表IIRC-然后注册自定义处理程序,以便根据您的选择处理该问题(如果有必要的话).

现在这是一个很好的解决方案,因为我可以通过这个过滤器运行文件。在不久的将来,我希望能够让用户自己上传svg文件,然后需要一个更健壮的解决方案。这是我创建的xslt


这很奇怪;我找不到在当前SVG规范中定义的
菜单
元素…抱歉,我不知道如何用代码位扩展您的答案,除非我自己做了一个答案。没关系。我很高兴我们已经设法找到了一种解决方法(没有深入研究产生似乎有缺陷的SVG文件的代码,这是我的第一个爱好)。
<xsl:stylesheet version="1.0" xmlns:svg="http://www.w3.org/2000/svg" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
    <xsl:output indent="yes"/>
    <xsl:strip-space elements="*"/>

    <!--Identity Transform.-->
    <xsl:template match="node()|@*">
        <xsl:copy>
            <xsl:apply-templates select="node()|@*"/>
        </xsl:copy>
    </xsl:template>

    <xsl:template match="svg:menu"/>
</xsl:stylesheet>
java -jar /usr/share/java/saxon.jar original.svg svgfix.xslt > fixed.svg