From ee4f321d36b845f6ee196b2195658231417e72f0 Mon Sep 17 00:00:00 2001 From: DreamPiggy Date: Sun, 4 Feb 2024 18:04:52 +0800 Subject: [PATCH] Fix the issue when EXIF orientation does not keep during the libwebp encoding Using transform currently --- .../Classes/SDImageWebPCoder.m | 33 ++++++++++++++++++- 1 file changed, 32 insertions(+), 1 deletion(-) diff --git a/SDWebImageWebPCoder/Classes/SDImageWebPCoder.m b/SDWebImageWebPCoder/Classes/SDImageWebPCoder.m index ead8fff..1940f05 100644 --- a/SDWebImageWebPCoder/Classes/SDImageWebPCoder.m +++ b/SDWebImageWebPCoder/Classes/SDImageWebPCoder.m @@ -8,6 +8,7 @@ #import "SDImageWebPCoder.h" #import "SDWebImageWebPCoderDefine.h" +#import "SDInternalMacros.h" #import #import #import @@ -784,7 +785,14 @@ - (NSData *)encodedDataWithFrames:(NSArray *)frames loopCount:(N BOOL encodeFirstFrame = [options[SDImageCoderEncodeFirstFrameOnly] boolValue]; if (encodeFirstFrame || frames.count <= 1) { // for static single webp image + // Keep EXIF orientation +#if SD_UIKIT || SD_WATCH + CGImagePropertyOrientation orientation = [SDImageCoderHelper exifOrientationFromImageOrientation:image.imageOrientation]; +#else + CGImagePropertyOrientation orientation = kCGImagePropertyOrientationUp; +#endif data = [self sd_encodedWebpDataWithImage:imageRef + orientation:orientation quality:compressionQuality maxPixelSize:maxPixelSize maxFileSize:maxFileSize @@ -797,7 +805,15 @@ - (NSData *)encodedDataWithFrames:(NSArray *)frames loopCount:(N } for (size_t i = 0; i < frames.count; i++) { SDImageFrame *currentFrame = frames[i]; - NSData *webpData = [self sd_encodedWebpDataWithImage:currentFrame.image.CGImage + UIImage *currentImage = currentFrame.image; + // Keep EXIF orientation +#if SD_UIKIT || SD_WATCH + CGImagePropertyOrientation orientation = [SDImageCoderHelper exifOrientationFromImageOrientation:currentImage.imageOrientation]; +#else + CGImagePropertyOrientation orientation = kCGImagePropertyOrientationUp; +#endif + NSData *webpData = [self sd_encodedWebpDataWithImage:currentImage.CGImage + orientation:orientation quality:compressionQuality maxPixelSize:maxPixelSize maxFileSize:maxFileSize @@ -838,6 +854,7 @@ - (NSData *)encodedDataWithFrames:(NSArray *)frames loopCount:(N } - (nullable NSData *)sd_encodedWebpDataWithImage:(nullable CGImageRef)imageRef + orientation:(CGImagePropertyOrientation)orientation quality:(double)quality maxPixelSize:(CGSize)maxPixelSize maxFileSize:(NSUInteger)maxFileSize @@ -847,6 +864,20 @@ - (nullable NSData *)sd_encodedWebpDataWithImage:(nullable CGImageRef)imageRef if (!imageRef) { return nil; } + // Seems libwebp has no convenient EXIF orientation API ? + // Use transform to apply ourselves. Need to release before return + // TODO: Use `WebPMuxSetChunk` API to write/read EXIF data, see: https://developers.google.com/speed/webp/docs/riff_container#extended_file_format + __block CGImageRef rotatedCGImage = NULL; + @onExit { + if (rotatedCGImage) { + CGImageRelease(rotatedCGImage); + } + }; + if (orientation != kCGImagePropertyOrientationUp) { + rotatedCGImage = [SDImageCoderHelper CGImageCreateDecoded:imageRef orientation:orientation]; + NSCParameterAssert(rotatedCGImage); + imageRef = rotatedCGImage; + } size_t width = CGImageGetWidth(imageRef); size_t height = CGImageGetHeight(imageRef);