Java 如何将ICC_配置文件(Adobe98)嵌入IIOMetadata并使用ImageIO写入PNG?

Java 如何将ICC_配置文件(Adobe98)嵌入IIOMetadata并使用ImageIO写入PNG?,java,image,image-processing,javax.imageio,Java,Image,Image Processing,Javax.imageio,我试图使用ImageIO将buffereImage写入png文件,但每次我向iimetadata添加ICC_配置文件时,生成的png图像没有嵌入ICC_配置文件,并且在生成的png文件上运行ImageMagick标识会产生错误 我需要的是添加ICC_配置文件的工作代码。我承认我对ImageIO有点生疏。我的想法是,我只需将配置文件添加到标题中,用于渲染图像的任何内容都应该找到配置文件,并在相关的颜色空间中显示图像(如果支持)。我是否错过了我应该对这张图片做的其他事情 另外,我想明确一点,我只想使

我试图使用
ImageIO
buffereImage
写入png文件,但每次我向
iimetadata
添加
ICC_配置文件时,生成的png图像没有嵌入
ICC_配置文件
,并且在生成的png文件上运行
ImageMagick
标识会产生错误

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

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

我的代码:

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);
        }
    }
图像Magick标识输出:

Image: /adobe.png
Format: PNG (Portable Network Graphics)
Class: DirectClass
Geometry: 399x508+0+0
Resolution: 300x300
Print size: 1.33x1.69333
Units: Undefined
Type: TrueColorAlpha
Endianess: Undefined
Colorspace: sRGB
Depth: 8-bit
Channel depth:
red: 8-bit
green: 8-bit
blue: 8-bit
alpha: 1-bit
Channel statistics:
Red:
  min: 0 (0)
  max: 255 (1)
  mean: 34.4178 (0.134972)
  standard deviation: 82.6482 (0.324111)
  kurtosis: 2.12664
  skewness: 2.01838
Green:
  min: 0 (0)
  max: 252 (0.988235)
  mean: 27.8842 (0.10935)
  standard deviation: 71.8833 (0.281895)
  kurtosis: 4.11874
  skewness: 2.41137
Blue:
  min: 0 (0)
  max: 254 (0.996078)
  mean: 28.1182 (0.110267)
  standard deviation: 73.4184 (0.287915)
  kurtosis: 4.25472
  skewness: 2.44667
Alpha:
  min: 0 (0)
  max: 255 (1)
  mean: 38.5321 (0.151106)
  standard deviation: 91.3288 (0.358152)
  kurtosis: 1.79587
  skewness: -1.9483
Image statistics:
Overall:
  min: 0 (0)
  max: 255 (1)
  mean: 76.722 (0.300871)
  standard deviation: 80.2016 (0.314516)
  kurtosis: 4.15993
  skewness: 2.43672
Alpha: none   #00000000
Rendering intent: Perceptual
Gamma: 0.454545
Chromaticity:
red primary: (0.64,0.33)
green primary: (0.3,0.6)
blue primary: (0.15,0.06)
white point: (0.3127,0.329)
Background color: white
Border color: srgba(223,223,223,1)
Matte color: grey74
Transparent color: none
Interlace: None
Intensity: Undefined
Compose: Over
Page geometry: 399x508+0+0
Dispose: Undefined
Iterations: 0
Compression: Zip
Orientation: Undefined
Properties:
date:create: 2013-12-30T10:23:05-06:00
date:modify: 2013-12-30T10:23:05-06:00
png:IHDR.bit-depth-orig: 8
png:IHDR.bit_depth: 8
png:IHDR.color-type-orig: 6
png:IHDR.color_type: 6 (RGBA)
png:IHDR.interlace_method: 0 (Not interlaced)
png:IHDR.width,height: 399, 508
png:pHYs: x_res=300, y_res=300, units=0
png:sRGB: intent=0 (Perceptual Intent)
signature: f03e01954623f646c9b594d0b2234ca444bb5c8282e923c8af520e08753fed0d
Artifacts:
filename: /adobe.png
verbose: true
Tainted: False
Filesize: 68.3KB
Number pixels: 203K
Pixels per second: 20.27MB
User time: 0.000u
Elapsed time: 0:01.009
Version: ImageMagick 6.8.6-6 2013-10-12 Q16 http://www.imagemagick.org
identify: iCCP: Data error in compressed datastream `/adobe.png' @ warning/png.c   /MagickPNGWarningHandler/1830.
identify: Profile size field missing from iCCP chunk `/adobe.png' @ warning/png.c/MagickPNGWarningHandler/1830.

任何帮助都将不胜感激。谢谢。

我之所以这么问,是因为我不认为您的ICC配置文件是压缩的
字节[]
。你说“这不是问题”,但我真的认为它是。:-)

如果我没有对ICC配置文件应用deflate压缩,我会从ExifTool收到与您从identify收到的类似警告

Warning                         : Error inflating iCCP
ICC Profile                     : (Binary data 526 bytes, use -b option to extract)
如果我正确地将deflate压缩应用于ICC配置文件,ICC配置文件将正确标识为“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();
    }
公共类PNGICCProfileIssue{
公共静态void main(字符串[]args)引发IOException{
文件输入=新文件(args[0]);
BuffereImage image=ImageIO.read(输入);
迭代器编写器=ImageIO.getImageWritersByFormatName(“PNG”);
如果(!writers.hasNext()){
返回;
}
ImageWriter writer=writers.next();
writer.setOutput(ImageIO.createImageOutputStream(新文件(input.getParent(),input.getName().replace('.',''.''.'''.'+''.'icc.png')));
IIOMetadata imageMetadata=writer.getDefaultImageMetadata(ImageTypeSpecifier.CreateFromRenderImage(图像),writer.getDefaultWriteParam());
//添加ICC配置文件
IIOMetadataNode iccp=新IIOMetadataNode(“iccp”);
ICC_ColorSpace ColorSpace=(ICC_ColorSpace)ColorSpace.getColorSpace(ColorSpace.CS_ADOBE_RGB_1998);
setUserObject(getAsDeflatedBytes(colorSpace));
iccp.setAttribute(“profileName”、“AdobeRGB1998”);
iccp.setAttribute(“压缩方法”、“放气”);
节点nativeTree=imageMetadata.getAsTree(imageMetadata.getNativeMetadataFormatName());
nativeTree.appendChild(iccp);
imageMetadata.mergeTree(imageMetadata.getNativeMetadataFormatName(),nativeTree);
write(新IIOImage(image,null,imageMetadata));
}
私有静态字节[]getAsDeflatedBytes(ICC_ColorSpace ColorSpace)引发IOException{
byte[]data=colorSpace.getProfile().getData();
ByteArrayOutputStream deflated=新的ByteArrayOutputStream();
DeflaterOutputStream deflater=新的DeflaterOutputStream(已放气);
平减指数.写入(数据);
flash();
deflater.close();
返回deflated.toByteArray();
}
我认为这个要求应该在元数据文档中记录下来,因为我不得不告诉你这有点让人困惑


祝你好运!

什么是
parameters.getOriginalICC()
?一个
ICC_配置文件
实例,一个
byte[]
或其他什么?你确定它是压缩的吗?这只是一种使用ICC_配置文件从本地文件获取byte[]的方便方法。getInstance(pathToFile)。这不是问题。而deflate compression是默认设置,也是您可以根据设置在标题中设置此字段的唯一方法。您是正确的。我不应该对压缩不屑一顾。谢谢。
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();
    }