Как вращать изображения JPEG на основе метаданных ориентации? - PullRequest
57 голосов
/ 06 мая 2011

У меня есть серверный код, который генерирует миниатюры при загрузке изображения. Проблема в том, что когда изображение было снято и камера / устройство повернуты, миниатюры поворачиваются, даже если сами изображения в натуральную величину отображаются в правильной ориентации в любом программном обеспечении для просмотра изображений. Это происходит только с jpgs.

Используя Preview в OSX, я вижу, что в jpgs встроены метаданные ориентации. Когда я использую ImageTools (Grails Plugin) для создания миниатюры, метаданные EXIF ​​отсутствуют в миниатюре, поэтому миниатюры выглядят повернутыми.

Из офлайн-разговоров я узнал, что хотя метаданные EXIF ​​относительно легко считывать, записать их нелегко, поэтому при создании эскиза jpg данные теряются.

Так что, похоже, у меня есть два варианта:

  1. Используйте ImageMagick для создания миниатюр. Недостатком является то, что на наших серверах требуется установить больше программного обеспечения.
  2. Считайте данные EXIF ​​Orientation по коду и поверните миниатюру соответствующим образом.

Кто-нибудь знает другие варианты?

Ответы [ 8 ]

58 голосов
/ 15 мая 2011

Если вы хотите повернуть ваши изображения, я бы предложил использовать библиотеку экстракторов метаданных http://code.google.com/p/metadata-extractor/. Вы можете получить информацию об изображении с помощью следующего кода:

// Inner class containing image information
public static class ImageInformation {
    public final int orientation;
    public final int width;
    public final int height;

    public ImageInformation(int orientation, int width, int height) {
        this.orientation = orientation;
        this.width = width;
        this.height = height;
    }

    public String toString() {
        return String.format("%dx%d,%d", this.width, this.height, this.orientation);
    }
}


public static ImageInformation readImageInformation(File imageFile)  throws IOException, MetadataException, ImageProcessingException {
    Metadata metadata = ImageMetadataReader.readMetadata(imageFile);
    Directory directory = metadata.getFirstDirectoryOfType(ExifIFD0Directory.class);
    JpegDirectory jpegDirectory = metadata.getFirstDirectoryOfType(JpegDirectory.class);

    int orientation = 1;
    try {
        orientation = directory.getInt(ExifIFD0Directory.TAG_ORIENTATION);
    } catch (MetadataException me) {
        logger.warn("Could not get orientation");
    }
    int width = jpegDirectory.getImageWidth();
    int height = jpegDirectory.getImageHeight();

    return new ImageInformation(orientation, width, height);
}

Затем, учитывая ориентациювы получаете, вы можете вращать и / или перевернуть изображение в правильную ориентацию.Аффинное преобразование для ориентации EXIF ​​задается следующим методом:

// Look at http://chunter.tistory.com/143 for information
public static AffineTransform getExifTransformation(ImageInformation info) {

    AffineTransform t = new AffineTransform();

    switch (info.orientation) {
    case 1:
        break;
    case 2: // Flip X
        t.scale(-1.0, 1.0);
        t.translate(-info.width, 0);
        break;
    case 3: // PI rotation 
        t.translate(info.width, info.height);
        t.rotate(Math.PI);
        break;
    case 4: // Flip Y
        t.scale(1.0, -1.0);
        t.translate(0, -info.height);
        break;
    case 5: // - PI/2 and Flip X
        t.rotate(-Math.PI / 2);
        t.scale(-1.0, 1.0);
        break;
    case 6: // -PI/2 and -width
        t.translate(info.height, 0);
        t.rotate(Math.PI / 2);
        break;
    case 7: // PI/2 and Flip
        t.scale(-1.0, 1.0);
        t.translate(-info.height, 0);
        t.translate(0, info.width);
        t.rotate(  3 * Math.PI / 2);
        break;
    case 8: // PI / 2
        t.translate(0, info.width);
        t.rotate(  3 * Math.PI / 2);
        break;
    }

    return t;
}

Поворот изображения будет выполняться следующим способом:

public static BufferedImage transformImage(BufferedImage image, AffineTransform transform) throws Exception {

    AffineTransformOp op = new AffineTransformOp(transform, AffineTransformOp.TYPE_BICUBIC);

    BufferedImage destinationImage = op.createCompatibleDestImage(image, (image.getType() == BufferedImage.TYPE_BYTE_GRAY) ? image.getColorModel() : null );
    Graphics2D g = destinationImage.createGraphics();
    g.setBackground(Color.WHITE);
    g.clearRect(0, 0, destinationImage.getWidth(), destinationImage.getHeight());
    destinationImage = op.filter(image, destinationImage);
    return destinationImage;
}

В среде сервера,не забудьте запустить с -Djava.awt.headless=true

14 голосов
/ 01 октября 2014

Библиотека Thumbnailator поддерживает флаги ориентации EXIF. Чтобы прочитать изображение в полном размере с правильной ориентацией:

BufferedImage image = Thumbnails.of(inputStream).scale(1).asBufferedImage();
7 голосов
/ 24 января 2014

Это можно сделать на удивление легко, используя часть изображения базовой библиотеки JavaXT :

// Browsers today can't handle images with Exif Orientation tag
Image image = new Image(uploadedFilename);
// Auto-rotate based on Exif Orientation tag, and remove all Exif tags
image.rotate(); 
image.saveAs(permanentFilename);

Вот и все!

Я пробовал Apache Commons Imaging, но это был беспорядок. JavaXT намного элегантнее.

3 голосов
/ 14 мая 2011

Exif, кажется, трудно писать из-за запатентованных вещей в нем. Тем не менее, вы можете рассмотреть другой вариант

Чтение оригинала, но только пиктограмму ориентации для миниатюр.

В Apache Sanselan, похоже, есть хорошая коллекция инструментов для этого.

http://commons.apache.org/proper/commons-imaging/

Посмотрите на класс ExifRewriter, например.

1 голос
/ 12 мая 2011

Если вы просто хотите, чтобы это выглядело правильно. Вы можете просто добавить «поворот» -PI / 2 (-90 градусов), PI / 2 (90 градусов) или PI (+180 градусов) по мере необходимости в зависимости от «ориентации», которую вы уже извлекли. Браузер или любая другая программа правильно отобразит изображение, так как будет применена ориентация и метаданные будут удалены из вывода эскизов.

0 голосов
/ 06 мая 2019

Мое решение - это сочетание ответа @ PerLindberg и ответа @ AntoineMartin. Я попробовал другие ответы с Java 8 на Windows 10, и ни один из них не помог. Решение @ AntoinMartin com.drew.imaging было медленным, и изображение получилось черно-белым и полным артефактов. Решение JavaXT от PerLindberg не считывало данные Exif 2.2.

1) Используйте com.drew.imaging для чтения exif-информации:

// Inner class containing image information
public static class ImageInformation {
    public final int orientation;
    public final int width;
    public final int height;

    public ImageInformation(int orientation, int width, int height) {
        this.orientation = orientation;
        this.width = width;
        this.height = height;
    }

    public String toString() {
        return String.format("%dx%d,%d", this.width, this.height, this.orientation);
    }
}

public ImageInformation readImageInformation(File imageFile)  throws IOException, MetadataException, ImageProcessingException {
    Metadata metadata = ImageMetadataReader.readMetadata(imageFile);
    Directory directory = metadata.getFirstDirectoryOfType(ExifIFD0Directory.class);
    JpegDirectory jpegDirectory = metadata.getFirstDirectoryOfType(JpegDirectory.class);

    int orientation = 1;
    if (directory != null) {
        try {
            orientation = directory.getInt(ExifIFD0Directory.TAG_ORIENTATION);
        } catch (MetadataException me) {
            logger.warn("Could not get orientation");
        }
        int width = jpegDirectory.getImageWidth();
        int height = jpegDirectory.getImageHeight();

        return new ImageInformation(orientation, width, height);
    } else {
        return null;
    }
}

2) Используйте JavaXT для выполнения поворота на основе данных Exif.

public void rotateMyImage(String imageDownloadFilenme);
    File imageDownloadFile =  new File(imgageDownloadFilenme);
    Image image = new Image(imgageDownloadFilenme);
    ImageInformation imageInformation = readImageInformation(imageDownloadFile);
    if (imageInformation != null) {
        rotate(imageInformation, image);
    }
    image.saveAs(imgageDownloadFilenme);
}

public void rotate(ImageInformation info, Image image) {

    switch(info.orientation) {
        case 1:
            return;
        case 2:
            image.flip();
            break;
        case 3:
            image.rotate(180.0D);
            break;
        case 4:
            image.flip();
            image.rotate(180.0D);
            break;
        case 5:
            image.flip();
            image.rotate(270.0D);
            break;
        case 6:
            image.rotate(90.0D);
            break;
        case 7:
            image.flip();
            image.rotate(90.0D);
            break;
        case 8:
            image.rotate(270.0D);
    }

}
0 голосов
/ 10 сентября 2018

Поскольку dnault , упомянутое в предыдущем комментарии, библиотека Thumbnaliator решает проблему. Но вы должны использовать правильные форматы ввода / вывода, чтобы избежать изменения цвета при этом автоматическом вращении.

ByteArrayOutputStream baos = new ByteArrayOutputStream();
ByteArrayInputStream in = new ByteArrayInputStream(file.getContents());
Thumbnails.of(in)
    .scale(1)
    .toOutputStream(baos);
byte[] bytes = baos.toByteArray();
0 голосов
/ 22 июля 2017

Основываясь на ответах Антуана Мартина, я создал собственный класс для корректировки ориентации данного изображения JPEG (в моем случае в качестве входного потока) на основе информации exif изображения. С его решением у меня была проблема, что цвета получающегося изображения были неправильными, поэтому я создал это. Для извлечения метаданных изображения я использовал библиотеку metadata-extractor .

Надеюсь, это поможет некоторым людям.

public class ImageOrientationUtil {

/**
 * Checks the orientation of the image and corrects it if necessary.
 * <p>If the orientation of the image does not need to be corrected, no operation will be performed.</p>
 * @param inputStream
 * @return
 * @throws ImageProcessingException
 * @throws IOException
 * @throws MetadataException
 */
public static BufferedImage correctOrientation(InputStream inputStream) throws ImageProcessingException, IOException, MetadataException {
    Metadata metadata = ImageMetadataReader.readMetadata(inputStream);
    if(metadata != null) {
        if(metadata.containsDirectoryOfType(ExifIFD0Directory.class)) {
            // Get the current orientation of the image
            Directory directory = metadata.getFirstDirectoryOfType(ExifIFD0Directory.class);
            int orientation = directory.getInt(ExifIFD0Directory.TAG_ORIENTATION);

            // Create a buffered image from the input stream
            BufferedImage bimg = ImageIO.read(inputStream);


            // Get the current width and height of the image
            int[] imageSize = {bimg.getWidth(), bimg.getHeight()};
            int width = imageSize[0];
            int height = imageSize[1];

            // Determine which correction is needed
            AffineTransform t = new AffineTransform();
            switch(orientation) {
            case 1:
                // no correction necessary skip and return the image
                return bimg;
            case 2: // Flip X
                t.scale(-1.0, 1.0);
                t.translate(-width, 0);
                return transform(bimg, t);
            case 3: // PI rotation 
                t.translate(width, height);
                t.rotate(Math.PI);
                return transform(bimg, t);
            case 4: // Flip Y
                t.scale(1.0, -1.0);
                t.translate(0, -height);
                return transform(bimg, t);
            case 5: // - PI/2 and Flip X
                t.rotate(-Math.PI / 2);
                t.scale(-1.0, 1.0);
                return transform(bimg, t);
            case 6: // -PI/2 and -width
                t.translate(height, 0);
                t.rotate(Math.PI / 2);
                return transform(bimg, t);
            case 7: // PI/2 and Flip
                t.scale(-1.0, 1.0);
                t.translate(height, 0);
                t.translate(0, width);
                t.rotate(  3 * Math.PI / 2);
                return transform(bimg, t);
            case 8: // PI / 2
                t.translate(0, width);
                t.rotate(  3 * Math.PI / 2);
                return transform(bimg, t);
            }
        }
    }

    return null;
}

/**
 * Performs the tranformation
 * @param bimage
 * @param transform
 * @return
 * @throws IOException
 */
private static BufferedImage transform(BufferedImage bimage, AffineTransform transform) throws IOException {
    // Create an transformation operation
    AffineTransformOp op = new AffineTransformOp(transform, AffineTransformOp.TYPE_BICUBIC);

    // Create an instance of the resulting image, with the same width, height and image type than the referenced one
    BufferedImage destinationImage = new BufferedImage( bimage.getWidth(), bimage.getHeight(), bimage.getType() );
    op.filter(bimage, destinationImage);

   return destinationImage;
}
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...