|  |  |  | // | 
					
						
							|  |  |  | //  Copyright (c) 2018 Open Whisper Systems. All rights reserved. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #import "UIImage+OWS.h" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @implementation UIImage (normalizeImage) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - (UIImage *)normalizedImage | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (self.imageOrientation == UIImageOrientationUp) { | 
					
						
							|  |  |  |         return self; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     UIGraphicsBeginImageContextWithOptions(self.size, NO, self.scale); | 
					
						
							|  |  |  |     [self drawInRect:(CGRect){ { 0, 0 }, self.size }]; | 
					
						
							|  |  |  |     UIImage *normalizedImage = UIGraphicsGetImageFromCurrentImageContext(); | 
					
						
							|  |  |  |     UIGraphicsEndImageContext(); | 
					
						
							|  |  |  |     return normalizedImage; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - (UIImage *)resizedWithQuality:(CGInterpolationQuality)quality rate:(CGFloat)rate | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     UIImage *resized = nil; | 
					
						
							|  |  |  |     CGFloat width = self.size.width * rate; | 
					
						
							|  |  |  |     CGFloat height = self.size.height * rate; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     UIGraphicsBeginImageContext(CGSizeMake(width, height)); | 
					
						
							|  |  |  |     CGContextRef context = UIGraphicsGetCurrentContext(); | 
					
						
							|  |  |  |     CGContextSetInterpolationQuality(context, quality); | 
					
						
							|  |  |  |     [self drawInRect:CGRectMake(0, 0, width, height)]; | 
					
						
							|  |  |  |     resized = UIGraphicsGetImageFromCurrentImageContext(); | 
					
						
							|  |  |  |     UIGraphicsEndImageContext(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return resized; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Source: https://github.com/AliSoftware/UIImage-Resize | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - (UIImage *)resizedImageToSize:(CGSize)dstSize | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CGImageRef imgRef = self.CGImage; | 
					
						
							|  |  |  |     // the below values are regardless of orientation : for UIImages from Camera, width>height (landscape) | 
					
						
							|  |  |  |     CGSize srcSize = CGSizeMake(CGImageGetWidth(imgRef), | 
					
						
							|  |  |  |         CGImageGetHeight(imgRef)); // not equivalent to self.size (which is dependant on the imageOrientation)! | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Don't resize if we already meet the required destination size. */ | 
					
						
							|  |  |  |     if (CGSizeEqualToSize(srcSize, dstSize)) { | 
					
						
							|  |  |  |         return self; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     CGFloat scaleRatio = dstSize.width / srcSize.width; | 
					
						
							|  |  |  |     UIImageOrientation orient = self.imageOrientation; | 
					
						
							|  |  |  |     CGAffineTransform transform = CGAffineTransformIdentity; | 
					
						
							|  |  |  |     switch (orient) { | 
					
						
							|  |  |  |         case UIImageOrientationUp: // EXIF = 1 | 
					
						
							|  |  |  |             transform = CGAffineTransformIdentity; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case UIImageOrientationUpMirrored: // EXIF = 2 | 
					
						
							|  |  |  |             transform = CGAffineTransformMakeTranslation(srcSize.width, 0.0); | 
					
						
							|  |  |  |             transform = CGAffineTransformScale(transform, -1.0, 1.0); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case UIImageOrientationDown: // EXIF = 3 | 
					
						
							|  |  |  |             transform = CGAffineTransformMakeTranslation(srcSize.width, srcSize.height); | 
					
						
							|  |  |  |             transform = CGAffineTransformRotate(transform, (CGFloat)M_PI); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case UIImageOrientationDownMirrored: // EXIF = 4 | 
					
						
							|  |  |  |             transform = CGAffineTransformMakeTranslation(0.0, srcSize.height); | 
					
						
							|  |  |  |             transform = CGAffineTransformScale(transform, 1.0, -1.0); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case UIImageOrientationLeftMirrored: // EXIF = 5 | 
					
						
							|  |  |  |             dstSize = CGSizeMake(dstSize.height, dstSize.width); | 
					
						
							|  |  |  |             transform = CGAffineTransformMakeTranslation(srcSize.height, srcSize.width); | 
					
						
							|  |  |  |             transform = CGAffineTransformScale(transform, -1.0, 1.0); | 
					
						
							|  |  |  |             transform = CGAffineTransformRotate(transform, (CGFloat)(3.0f * M_PI_2)); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case UIImageOrientationLeft: // EXIF = 6 | 
					
						
							|  |  |  |             dstSize = CGSizeMake(dstSize.height, dstSize.width); | 
					
						
							|  |  |  |             transform = CGAffineTransformMakeTranslation(0.0, srcSize.width); | 
					
						
							|  |  |  |             transform = CGAffineTransformRotate(transform, (CGFloat)(3.0 * M_PI_2)); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case UIImageOrientationRightMirrored: // EXIF = 7 | 
					
						
							|  |  |  |             dstSize = CGSizeMake(dstSize.height, dstSize.width); | 
					
						
							|  |  |  |             transform = CGAffineTransformMakeScale(-1.0, 1.0); | 
					
						
							|  |  |  |             transform = CGAffineTransformRotate(transform, (CGFloat)M_PI_2); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case UIImageOrientationRight: // EXIF = 8 | 
					
						
							|  |  |  |             dstSize = CGSizeMake(dstSize.height, dstSize.width); | 
					
						
							|  |  |  |             transform = CGAffineTransformMakeTranslation(srcSize.height, 0.0); | 
					
						
							|  |  |  |             transform = CGAffineTransformRotate(transform, (CGFloat)M_PI_2); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             OWSRaiseException(NSInternalInconsistencyException, @"Invalid image orientation"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ///////////////////////////////////////////////////////////////////////////// | 
					
						
							|  |  |  |     // The actual resize: draw the image on a new context, applying a transform matrix | 
					
						
							|  |  |  |     UIGraphicsBeginImageContextWithOptions(dstSize, NO, self.scale); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     CGContextRef context = UIGraphicsGetCurrentContext(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!context) { | 
					
						
							|  |  |  |         return nil; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (orient == UIImageOrientationRight || orient == UIImageOrientationLeft) { | 
					
						
							|  |  |  |         CGContextScaleCTM(context, -scaleRatio, scaleRatio); | 
					
						
							|  |  |  |         CGContextTranslateCTM(context, -srcSize.height, 0); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         CGContextScaleCTM(context, scaleRatio, -scaleRatio); | 
					
						
							|  |  |  |         CGContextTranslateCTM(context, 0, -srcSize.height); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     CGContextConcatCTM(context, transform); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // we use srcSize (and not dstSize) as the size to specify is in user space (and we use the CTM to apply a | 
					
						
							|  |  |  |     // scaleRatio) | 
					
						
							|  |  |  |     CGContextDrawImage(UIGraphicsGetCurrentContext(), CGRectMake(0, 0, srcSize.width, srcSize.height), imgRef); | 
					
						
							|  |  |  |     UIImage *resizedImage = UIGraphicsGetImageFromCurrentImageContext(); | 
					
						
							|  |  |  |     UIGraphicsEndImageContext(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return resizedImage; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - (UIImage *)resizedImageToFillPixelSize:(CGSize)dstSize | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     OWSAssert(dstSize.width > 0); | 
					
						
							|  |  |  |     OWSAssert(dstSize.height > 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     UIImage *normalized = [self normalizedImage]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Get the size in pixels, not points. | 
					
						
							|  |  |  |     CGSize srcSize = CGSizeMake(CGImageGetWidth(normalized.CGImage), CGImageGetHeight(normalized.CGImage)); | 
					
						
							|  |  |  |     OWSAssert(srcSize.width > 0); | 
					
						
							|  |  |  |     OWSAssert(srcSize.height > 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     CGFloat widthRatio = srcSize.width / dstSize.width; | 
					
						
							|  |  |  |     CGFloat heightRatio = srcSize.height / dstSize.height; | 
					
						
							|  |  |  |     CGRect drawRect = CGRectZero; | 
					
						
							|  |  |  |     if (widthRatio > heightRatio) { | 
					
						
							|  |  |  |         drawRect.origin.y = 0; | 
					
						
							|  |  |  |         drawRect.size.height = dstSize.height; | 
					
						
							|  |  |  |         drawRect.size.width = dstSize.height * srcSize.width / srcSize.height; | 
					
						
							|  |  |  |         OWSAssert(drawRect.size.width > dstSize.width); | 
					
						
							|  |  |  |         drawRect.origin.x = (drawRect.size.width - dstSize.width) * -0.5f; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         drawRect.origin.x = 0; | 
					
						
							|  |  |  |         drawRect.size.width = dstSize.width; | 
					
						
							|  |  |  |         drawRect.size.height = dstSize.width * srcSize.height / srcSize.width; | 
					
						
							|  |  |  |         OWSAssert(drawRect.size.height >= dstSize.height); | 
					
						
							|  |  |  |         drawRect.origin.y = (drawRect.size.height - dstSize.height) * -0.5f; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     UIGraphicsBeginImageContextWithOptions(dstSize, NO, 1.f); | 
					
						
							|  |  |  |     CGContextRef context = UIGraphicsGetCurrentContext(); | 
					
						
							|  |  |  |     CGContextSetInterpolationQuality(context, kCGInterpolationHigh); | 
					
						
							|  |  |  |     [self drawInRect:drawRect]; | 
					
						
							|  |  |  |     UIImage *dstImage = UIGraphicsGetImageFromCurrentImageContext(); | 
					
						
							|  |  |  |     UIGraphicsEndImageContext(); | 
					
						
							|  |  |  |     return dstImage; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | + (UIImage *)imageWithColor:(UIColor *)color | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     OWSAssertIsOnMainThread(); | 
					
						
							|  |  |  |     OWSAssert(color); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return [self imageWithColor:color size:CGSizeMake(1.f, 1.f)]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | + (UIImage *)imageWithColor:(UIColor *)color size:(CGSize)size | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     OWSAssertIsOnMainThread(); | 
					
						
							|  |  |  |     OWSAssert(color); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     CGRect rect = CGRectMake(0.0f, 0.0f, size.width, size.height); | 
					
						
							|  |  |  |     UIGraphicsBeginImageContextWithOptions(rect.size, NO, 1.f); | 
					
						
							|  |  |  |     CGContextRef context = UIGraphicsGetCurrentContext(); | 
					
						
							|  |  |  |     CGContextClearRect(context, rect); | 
					
						
							|  |  |  |     CGContextSetFillColorWithColor(context, [color CGColor]); | 
					
						
							|  |  |  |     CGContextFillRect(context, rect); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     UIImage *image = UIGraphicsGetImageFromCurrentImageContext(); | 
					
						
							|  |  |  |     UIGraphicsEndImageContext(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return image; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @end |