Pergunta

Estou tentando definir programaticamente os metadados de dpi de uma imagem jpeg em Java.A origem da imagem é um scanner, então obtenho a resolução horizontal/vertical do TWAIN, junto com os dados brutos da imagem.Gostaria de salvar essas informações para obter melhores resultados de impressão.

Aqui está o código que tenho até agora.Ele salva a imagem bruta (byteArray) em um arquivo JPEG, mas ignora as informações de densidade X/Y especificadas por meio de IIOMetadata.Algum conselho sobre o que estou fazendo de errado?

Qualquer outra solução (biblioteca de terceiros, etc.) também seria bem-vinda.

import java.awt.image.BufferedImage;
import java.awt.image.WritableRaster;
import java.io.File;

import javax.imageio.IIOImage;
import javax.imageio.ImageIO;
import javax.imageio.ImageTypeSpecifier;
import javax.imageio.metadata.IIOMetadata;
import javax.imageio.plugins.jpeg.JPEGImageWriteParam;
import javax.imageio.stream.ImageOutputStream    

import org.w3c.dom.Element;    
import com.sun.imageio.plugins.jpeg.JPEGImageWriter;

public boolean saveJpeg(int[] byteArray, int width, int height, int dpi, String file)
{
    BufferedImage bufferedImage = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);

    WritableRaster wr = bufferedImage.getRaster();
    wr.setPixels(0, 0, width, height, byteArray);

    try
    {           
        // Image writer 
        JPEGImageWriter imageWriter = (JPEGImageWriter) ImageIO.getImageWritersBySuffix("jpeg").next();
        ImageOutputStream ios = ImageIO.createImageOutputStream(new File(file));
        imageWriter.setOutput(ios);

        // Compression
        JPEGImageWriteParam jpegParams = (JPEGImageWriteParam) imageWriter.getDefaultWriteParam();
        jpegParams.setCompressionMode(JPEGImageWriteParam.MODE_EXPLICIT);
        jpegParams.setCompressionQuality(0.85f);

        // Metadata (dpi)
        IIOMetadata data = imageWriter.getDefaultImageMetadata(new ImageTypeSpecifier(bufferedImage), jpegParams);
        Element tree = (Element)data.getAsTree("javax_imageio_jpeg_image_1.0");
        Element jfif = (Element)tree.getElementsByTagName("app0JFIF").item(0);
        jfif.setAttribute("Xdensity", Integer.toString(dpi));
        jfif.setAttribute("Ydensity", Integer.toString(dpi));
        jfif.setAttribute("resUnits", "1"); // density is dots per inch         

        // Write and clean up
        imageWriter.write(data, new IIOImage(bufferedImage, null, null), jpegParams);
        ios.close();
        imageWriter.dispose();
    }
    catch (Exception e)
    {
       return false;
    }

    return true;
}

Obrigado!

Foi útil?

Solução

Algumas questões que não foram consideradas aqui:

  1. A árvore não está mapeada diretamente para o IOMetaData.Para aplicar dados da árvore, adicione a seguinte chamada após definir as densidades e os parâmetros raster:

    data.setFromTree("javax_imageio_jpeg_image_1.0", tree);
    
  2. não use os metadados como primeiro parâmetro na chamada de gravação.Ver JPEGImageWriter#write(IIOMetaData, IIOImage, ImageWriteParam).Se streamMetaData não for NULL, um aviso (WARNING_STREAM_METADATA_IGNORED) será gerado.

  3. defina os metadados como IOMetadata do IOImage.Esses metadados são usados ​​pelo JPEGImageWriter.A chamada de gravação correta então é

    imageWriter.write(null, new IIOImage(F_scaledImg, null, data), jpegParams);
    

Outras dicas

Eu parece que isso poderia ser um bug.

Eu encontrei este post De algumas pesquisas no Google

Aparentemente, existem muito mais que apontam para um bug também.

O post acima fala sobre o uso de Jmagick como um trabalho de terceiros.

Licenciado em: CC-BY-SA com atribuição
Não afiliado a StackOverflow
scroll top