我正在使用Apache BatikJava库将. svg
矢量图像文件转换为.png
文件。问题是生成的.png
图像的字体颜色都变黑了。这是我用来进行转换的代码:
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import org.apache.batik.transcoder.TranscoderInput;
import org.apache.batik.transcoder.TranscoderOutput;
import org.apache.batik.transcoder.image.PNGTranscoder;
public class SVGHelperDesktop extends SVGHelper {
@Override
public byte[] convertSvgToPng(String svgXml, float png_width)
throws SVGConversionException {
byte[] resultPngBytes = null;
try {
ByteArrayInputStream inputSvgBytes = new
ByteArrayInputStream(svgXml.getBytes());
TranscoderInput input_svg_image = new
TranscoderInput(inputSvgBytes);
ByteArrayOutputStream outputPngBytes = new ByteArrayOutputStream();
TranscoderOutput output_png_image = new TranscoderOutput(outputPngBytes);
PNGTranscoder svgToPngConverter = new PNGTranscoder();
svgToPngConverter.addTranscodingHint(PNGTranscoder.KEY_WIDTH, png_width);
svgToPngConverter.transcode(input_svg_image, output_png_image);
resultPngBytes = outputPngBytes.toByteArray();
outputPngBytes.flush();
outputPngBytes.close();
} catch (Exception e) {
throw new SVGConversionException("Error converting SVG to PNG", e);
}
return resultPngBytes;
}
}
在同一个. svg
文件上使用AndroidSVG
库生成具有正确颜色的正确.png
图像。
另一个注意事项;使用inkscape
(我用来创建矢量图形的程序)的默认字体解决了这个问题。使用任何其他字体脸都会导致蜡染将其颜色更改为黑色。
这是我的SVG文件的链接。
Inkscape在与自定义字体文本关联的style
属性中添加了自定义CSS属性-inkscape-font-规范
。以下是摘录:
style="...;-inkscape-font-specification:'Aharoni, Bold';..."
已知的Batikbug阻止呈现以连字符(-
)开头的CSS属性。
一个简单的修复方法是编辑文件,然后删除“-inkscape-font-规范”属性。
或者,您可以使用Batik的API创建一个新的CSS样式解析器,删除有问题的CSS样式类。例如:
/**
* <a href="https://issues.apache.org/jira/browse/BATIK-1112">Bug fix</a>
*/
public static final class InkscapeCssParser extends Parser {
public void parseStyleDeclaration( final String source )
throws CSSException, IOException {
super.parseStyleDeclaration(
source.replaceAll( "-inkscape-font-specification:[^;\"]*;", "" )
);
}
}
static {
XMLResourceDescriptor.setCSSParserClassName(
InkscapeCssParser.class.getName()
);
}
公认的答案非常有用地指向蜡染bug1112,表明“-inkscape-font-规范”没有正确解析。bug报告有一个解决方法,它通过重新实现Batik的CSS解析器的一部分来工作。为了扩展这一点并使其在这里可用,这里是我的简化和天真版本的解决方法。
首先,需要一个CSS解析器类来覆盖Batik中的解析器类:
import org.apache.batik.css.parser.ExtendedParser;
import org.apache.batik.css.parser.Parser;
import org.w3c.css.sac.CSSException;
import java.io.IOException;
public class InkscapeCssParser extends Parser {
public void parseStyleDeclaration(String source)
throws CSSException, IOException {
source = source.replaceAll(";-inkscape", ";inkscape");
super.parseStyleDeclaration(source);
}
}
请注意,source
参数是特定SVG元素的CSS样式,而不是完整的SVG源代码。
虽然Batikbug中提供的解决方法重新实现了一大块parseStyle声明
,但我只是做了一个简单的字符串替换来消除";"之后的冒犯"-",因为这是问题的核心。
要使用它,请在加载文档之前设置CSS解析器类名:
JSVGCanvas mSvgCanvas;
XMLResourceDescriptor.setCSSParserClassName(InkscapeCssParser.class.getName());
mSvgCanvas.setURI(svgUrl.toString());