org.sejda.impl.sambox.component.image.ExifHelper Maven / Gradle / Ivy
Show all versions of sejda-sambox Show documentation
/*
* Copyright 2017 by Eduard Weissmann ([email protected]).
*
* This file is part of the Sejda source code
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see .
*/
package org.sejda.impl.sambox.component.image;
import com.drew.imaging.ImageMetadataReader;
import com.drew.imaging.ImageProcessingException;
import com.drew.metadata.Directory;
import com.drew.metadata.Metadata;
import com.drew.metadata.MetadataException;
import com.drew.metadata.exif.ExifIFD0Directory;
import org.sejda.model.exception.TaskIOException;
import org.sejda.model.input.FileSource;
import org.sejda.model.input.Source;
import org.sejda.model.input.SourceDispatcher;
import org.sejda.model.input.StreamSource;
import java.io.IOException;
/**
* Reads exif orientation of an image and determines if the image should be rotated or not.
*
* Based on https://stackoverflow.com/questions/5905868/how-to-rotate-jpeg-images-based-on-the-orientation-metadata
*/
public class ExifHelper {
public static int getRotationBasedOnExifOrientation(Source> imageSource) {
try {
return imageSource.dispatch(new SourceDispatcher() {
@Override
public Integer dispatch(FileSource source) throws TaskIOException {
try {
int orientation = readExifOrientation(ImageMetadataReader.readMetadata(source.getSource()));
return getRotation(orientation);
} catch (IOException | MetadataException | ImageProcessingException e) {
return 0;
}
}
@Override
public Integer dispatch(StreamSource source) throws TaskIOException {
try {
int orientation = readExifOrientation(ImageMetadataReader.readMetadata(source.getSource()));
return getRotation(orientation);
} catch (IOException | MetadataException | ImageProcessingException e) {
return 0;
}
}
});
} catch(Exception e) {
return 0;
}
}
private static int readExifOrientation(Metadata metadata) throws MetadataException {
Directory directory = metadata.getFirstDirectoryOfType(ExifIFD0Directory.class);
return directory.getInt(ExifIFD0Directory.TAG_ORIENTATION);
}
/**
* Returns a rotation that corresponds to the given exif orientation.
* If the exif orientation involves a flip, thus not being achievable only with a rotation, then 0 is returned (unsupported)
*
* See http://sylvana.net/jpegcrop/exif_orientation.html
*/
private static int getRotation(int orientation) {
switch (orientation) {
case 1:
return 0;
case 2: // Flip X
// unsupported
return 0;
case 3: // PI rotation
return 180;
case 4: // Flip Y
// unsupported
return 0;
case 5: // - PI/2 and Flip X
// unsupported
return 0;
case 6: // -PI/2 and -width
return 90;
case 7: // PI/2 and Flip
// unsupported
return 0;
case 8: // PI / 2
return 270;
}
return 0;
}
}