Grave metadados de dpi em uma imagem jpeg em Java
-
04-07-2019 - |
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!
Solução
Algumas questões que não foram consideradas aqui:
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);
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.defina os metadados como
IOMetadata
doIOImage
.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.