2008-10-24 12 views
14

Ich versuche, die dpi-Metadaten eines JPEG-Bilds in Java programmatisch einzustellen. Die Quelle des Bildes ist ein Scanner, also erhalte ich die horizontale/vertikale Auflösung von TWAIN zusammen mit den Rohdaten des Bildes. Ich möchte diese Informationen für bessere Druckergebnisse speichern.Schreibe dpi-Metadaten in ein JPEG-Bild in Java

Hier ist der Code, den ich bisher habe. Es speichert das Rohbild (ByteArray) in einer JPEG-Datei, ignoriert jedoch die X/Y-Helligkeitsinformationen, die ich über IIOMetadata angegeben habe. Irgendwelche Ratschläge, was ich falsch mache?

Jede andere Lösung (Drittanbieter-Bibliothek usw.) wäre ebenfalls willkommen.

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; 
} 

Vielen Dank!

Antwort

13

Einige Fragen, die hier nicht berücksichtigt wurden:

  1. Der Baum nicht direkt auf die IOMetaData abgebildet wird. Um Daten aus Baum anwenden, fügen Sie folgenden Aufruf nach den Dichten und Raster Parameter einstellen:

    data.setFromTree("javax_imageio_jpeg_image_1.0", tree); 
    
  2. nicht die Meta-Daten als erste Parameter in dem Schreibaufruf verwenden. Siehe JPEGImageWriter#write(IIOMetaData, IIOImage, ImageWriteParam). Wenn streamMetaData nicht NULL ist, wird eine Warnung (WARNING_STREAM_METADATA_IGNORED) generiert.

  3. stellen Sie die Metadaten als IOMetadata der IOImage. Diese Metadaten werden von JPEGImageWriter verwendet. Der richtige Schreibaufruf dann ist

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

Wo ist F_scaledImg eingestellt? Ist es dasselbe wie bufferedImage? Entschuldigung, wenn das eine dumme Frage ist ... –

+2

Ja, es ist in der Tat bufferedImage. Christoph muss das Snippet kopiert und vergessen haben, es zu bearbeiten. – dasp

1

Ich würde scheinen, das könnte ein Fehler sein.

fand ich diesen Beitrag from a few google searches

Anscheinend gibt viel mehr ist, dass Punkt zu einem auch Fehler.

Der Beitrag oben spricht über die Verwendung von JMagick als eine dritte Partei arbeiten.