Изображения с камеры iPhone поворачиваются при загрузке в Интернет

Я использую UIImagePickerController, чтобы сделать снимок в портретном режиме на iphone и сохранить его в Интернете. Фотография отображается в портретной ориентации на телефоне, но поворачивается на 90 градусов в Интернете.

Если я загружаю фотографию и смотрю на нее в режиме предварительного просмотра (Mac) или Photoshop (Mac или ПК), она снова становится портретной. В Windows Picture Viewer (ПК) он повернут в альбомную ориентацию.

Нужно ли применять преобразование поворота к данным изображения перед загрузкой? Придется ли мне затем удалить метаданные, которые вращают его в Photoshop и в предварительном просмотре?


person Arrel    schedule 25.06.2009    source источник


Ответы (4)


Проблема заключалась в том, что к фотографии был добавлен поворот изображения как данные EXIF, которые не используются большинством браузеров. Есть два решения:

  1. Примените ротацию на стороне сервера. Я использовал подключаемый модуль Ruby Paperclip (от Thoughtbot), и мне просто нужно было включить опцию преобразования с автоматической ориентацией в команду has_attached_file в модели:

    has_attached_file: photo,: convert_options => {: all => '-auto-orient'}

  2. Поверните фотографию в приложении для iPhone. Это было решено в другом вопросе о переполнении стека; вызов метода scaleAndRotate заменяет метаданные вращения на преобразование изображения благодаря @Squeegy.

person Arrel    schedule 26.06.2009

Вот решение для CarrierWavew / MiniMagick: http://carrierwave.rubyforge.org/rdoc/classes/CarrierWave/MiniMagick.html

person user284046    schedule 15.05.2011
comment
Хороший! Спасибо, что обновили ответ. - person Arrel; 15.05.2011

Как только вы сделаете фотографию перед загрузкой изображения на сервер, просто передайте сделанное изображение в метод, и он обязательно сработает для вас.

#pragma mark Rotate
- (UIImage *)scaleAndRotateImage:(UIImage *)image {
    int kMaxResolution = 640; // Or whatever

    CGImageRef imgRef = image.CGImage;

    CGFloat width = CGImageGetWidth(imgRef);
    CGFloat height = CGImageGetHeight(imgRef);


    CGAffineTransform transform = CGAffineTransformIdentity;
    CGRect bounds = CGRectMake(0, 0, width, height);
    if (width > kMaxResolution || height > kMaxResolution) {
        CGFloat ratio = width/height;
        if (ratio > 1) {
            bounds.size.width = kMaxResolution;
            bounds.size.height = roundf(bounds.size.width / ratio);
        }
        else {
            bounds.size.height = kMaxResolution;
            bounds.size.width = roundf(bounds.size.height * ratio);
        }
    }

    CGFloat scaleRatio = bounds.size.width / width;
    CGSize imageSize = CGSizeMake(CGImageGetWidth(imgRef), CGImageGetHeight(imgRef));
    CGFloat boundHeight;
    UIImageOrientation orient = image.imageOrientation;
    switch(orient) {

        case UIImageOrientationUp: //EXIF = 1
            transform = CGAffineTransformIdentity;
            break;

        case UIImageOrientationUpMirrored: //EXIF = 2
            transform = CGAffineTransformMakeTranslation(imageSize.width, 0.0);
            transform = CGAffineTransformScale(transform, -1.0, 1.0);
            break;

        case UIImageOrientationDown: //EXIF = 3
            transform = CGAffineTransformMakeTranslation(imageSize.width, imageSize.height);
            transform = CGAffineTransformRotate(transform, M_PI);
            break;

        case UIImageOrientationDownMirrored: //EXIF = 4
            transform = CGAffineTransformMakeTranslation(0.0, imageSize.height);
            transform = CGAffineTransformScale(transform, 1.0, -1.0);
            break;

        case UIImageOrientationLeftMirrored: //EXIF = 5
            boundHeight = bounds.size.height;
            bounds.size.height = bounds.size.width;
            bounds.size.width = boundHeight;
            transform = CGAffineTransformMakeTranslation(imageSize.height, imageSize.width);
            transform = CGAffineTransformScale(transform, -1.0, 1.0);
            transform = CGAffineTransformRotate(transform, 3.0 * M_PI / 2.0);
            break;

        case UIImageOrientationLeft: //EXIF = 6
            boundHeight = bounds.size.height;
            bounds.size.height = bounds.size.width;
            bounds.size.width = boundHeight;
            transform = CGAffineTransformMakeTranslation(0.0, imageSize.width);
            transform = CGAffineTransformRotate(transform, 3.0 * M_PI / 2.0);
            break;

        case UIImageOrientationRightMirrored: //EXIF = 7
            boundHeight = bounds.size.height;
            bounds.size.height = bounds.size.width;
            bounds.size.width = boundHeight;
            transform = CGAffineTransformMakeScale(-1.0, 1.0);
            transform = CGAffineTransformRotate(transform, M_PI / 2.0);
            break;

        case UIImageOrientationRight: //EXIF = 8
            boundHeight = bounds.size.height;
            bounds.size.height = bounds.size.width;
            bounds.size.width = boundHeight;
            transform = CGAffineTransformMakeTranslation(imageSize.height, 0.0);
            transform = CGAffineTransformRotate(transform, M_PI / 2.0);
            break;

        default:
            [NSException raise:NSInternalInconsistencyException format:@"Invalid image orientation"];

    }

    UIGraphicsBeginImageContext(bounds.size);

    CGContextRef context = UIGraphicsGetCurrentContext();

    if (orient == UIImageOrientationRight || orient == UIImageOrientationLeft) {
        CGContextScaleCTM(context, -scaleRatio, scaleRatio);
        CGContextTranslateCTM(context, -height, 0);
    }
    else {
        CGContextScaleCTM(context, scaleRatio, -scaleRatio);
        CGContextTranslateCTM(context, 0, -height);
    }

    CGContextConcatCTM(context, transform);

    CGContextDrawImage(UIGraphicsGetCurrentContext(), CGRectMake(0, 0, width, height), imgRef);
    UIImage *imageCopy = UIGraphicsGetImageFromCurrentImageContext();
    UIGraphicsEndImageContext();

    return imageCopy;
}
person btmanikandan    schedule 07.02.2014
comment
Я считаю, что ваши комментарии неверны. Например. UIImageOrientationUp: // EXIF ​​= 0 - person jonypz; 20.11.2014

Вот простой способ вручную переопределить метаданные поворота EXIF, если изображение сохраняется в правильной ориентации в MS Windows. В проводнике Windows щелкните файл изображения правой кнопкой мыши и выберите «Повернуть по часовой стрелке». Сделайте это 4 раза, чтобы повернуть изображение полностью, и тогда изображение будет иметь правильную ориентацию для всех систем. Затем вы можете загрузить изображение на свой веб-сервер.

person BenW    schedule 26.10.2015