如何在IIOMetadata中嵌入ICC_Profile(Adobe98)并使用ImageIO编写PNG?

时间:2013-12-30 16:52:33

标签: java image image-processing javax.imageio

我正在尝试使用BufferedImageImageIO写入png文件,但每次向ICC_Profile添加IIOMetadata时,生成的png图片都没有{{} 1}}嵌入,并在生成的png文件上运行ICC_Profile标识会产生错误。

我需要的是添加ICC_Profile的工作代码。我承认我对ImageIO有点新手。我的想法是,我只是将配置文件添加到标题中,用于渲染图像的任何内容都应该找到配置文件并在关联的颜色空间中显示图像(如果支持)。我错过了我应该对这张照片做的其他事情吗?

另外 - 我想明确我只想使用ImageIO而不是JAI的ColorConvertOp,因为JAI还没有在7年内开发出来,在我看来,最好的方法是使用标准的JDK类。这可能和/或是否有其他人能够实现它?

我的代码:

ImageMagick

Image Magick识别输出:

Iterator i = ImageIO.getImageWritersByFormatName("PNG");
    if (i.hasNext()) {
        ImageWriter imageWriter = (ImageWriter) i.next();
        ImageWriteParam param = imageWriter.getDefaultWriteParam();
        ImageTypeSpecifier its = new ImageTypeSpecifier(bi.getColorModel(), bi.getSampleModel());
        IIOMetadata iomd = imageWriter.getDefaultImageMetadata(its, param);
        String formatName = "javax_imageio_png_1.0";
        Node node = iomd.getAsTree(formatName);

        //add resolution
        IIOMetadataNode phys = new IIOMetadataNode("pHYs");
        phys.setAttribute("unitSpecifier", "unknown");
        phys.setAttribute("pixelsPerUnitXAxis", "300");
        phys.setAttribute("pixelsPerUnitYAxis", "300");

        //add ICC PROFILE
        IIOMetadataNode iccp = new IIOMetadataNode("iCCP");
        //ICC_Profile pro = Util.getAdobeRGBICCProfile();
        iccp.setUserObject(parameters.getOriginalICC());
        iccp.setAttribute("profileName", "AdobeRGB1998");
        iccp.setAttribute("compressionMethod", "deflate");

        node.appendChild(phys);
        node.appendChild(iccp);

        try {
            iomd.setFromTree(formatName, node);
        } catch (IIOInvalidTreeException e) {
            System.out.println("IIOInvalidTreeException Occurred setting resolution on PNG EXIF Header.");
            e.printStackTrace(System.out);
        }

        IIOImage iioimage = new IIOImage(bi, null, iomd);
        try {
            imageWriter.setOutput(new FileImageOutputStream(new File(fileName)));
            imageWriter.write(iioimage);
        } catch (IOException e) {
            System.out.println("Error Occurred setting resolution on PNG.");
            e.printStackTrace(System.out);
        }
    }

非常感谢任何帮助。谢谢。

1 个答案:

答案 0 :(得分:1)

我问的原因是,我不认为您的ICC配置文件是压缩的byte[]。你说“这不是问题”,但我真的认为是。 : - )

如果我没有对ICC配置文件应用deflate压缩,我会从ExifTool获得类似的警告。

Warning                         : Error inflating iCCP
ICC Profile                     : (Binary data 526 bytes, use -b option to extract)

如果我正确地将deflate压缩应用于ICC配置文件,ICC配置文件将被正确识别为“Adobe RGB 1998”。

Profile CMM Type                : ADBE
Profile Version                 : 2.1.0
Profile Class                   : Display Device Profile
Color Space Data                : RGB
[...]
CMM Flags                       : Not Embedded, Independent
Rendering Intent                : Perceptual
Connection Space Illuminant     : 0.9642 1 0.82491
Profile Creator                 : ADBE
Profile ID                      : 0
Profile Copyright               : Copyright 2000 Adobe Systems Incorporated
Profile Description             : Adobe RGB (1998)

以下代码适用于我:

public class PNGICCProfileIssue {
    public static void main(String[] args) throws IOException {
        File input = new File(args[0]);
        BufferedImage image = ImageIO.read(input);

        Iterator<ImageWriter> writers = ImageIO.getImageWritersByFormatName("PNG");
        if (!writers.hasNext()) {
            return;
        }

        ImageWriter writer = writers.next();
        writer.setOutput(ImageIO.createImageOutputStream(new File(input.getParent(), input.getName().replace('.', '_') + "_icc.png")));

        IIOMetadata imageMetadata = writer.getDefaultImageMetadata(ImageTypeSpecifier.createFromRenderedImage(image), writer.getDefaultWriteParam());

        //add ICC PROFILE
        IIOMetadataNode iccp = new IIOMetadataNode("iCCP");
        ICC_ColorSpace colorSpace = (ICC_ColorSpace) ColorSpaces.getColorSpace(ColorSpaces.CS_ADOBE_RGB_1998);
        iccp.setUserObject(getAsDeflatedBytes(colorSpace));
        iccp.setAttribute("profileName", "AdobeRGB1998");
        iccp.setAttribute("compressionMethod", "deflate");

        Node nativeTree = imageMetadata.getAsTree(imageMetadata.getNativeMetadataFormatName());
        nativeTree.appendChild(iccp);
        imageMetadata.mergeTree(imageMetadata.getNativeMetadataFormatName(), nativeTree);

        writer.write(new IIOImage(image, null, imageMetadata));
    }

    private static byte[] getAsDeflatedBytes(ICC_ColorSpace colorSpace) throws IOException {
        byte[] data = colorSpace.getProfile().getData();

        ByteArrayOutputStream deflated = new ByteArrayOutputStream();
        DeflaterOutputStream deflater = new DeflaterOutputStream(deflated);
        deflater.write(data);
        deflater.flush();
        deflater.close();

        return deflated.toByteArray();
    }

我认为 这个要求已经记录在元数据文档中,因为我必须给你一些令人困惑的事情。

祝你好运!