CCGraphics.m 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481
  1. //
  2. // CCGraphics.m
  3. // Nextcloud iOS
  4. //
  5. // Created by Marino Faggiana on 04/02/16.
  6. // Copyright (c) 2017 TWS. All rights reserved.
  7. //
  8. // Author Marino Faggiana <m.faggiana@twsweb.it>
  9. //
  10. // This program is free software: you can redistribute it and/or modify
  11. // it under the terms of the GNU General Public License as published by
  12. // the Free Software Foundation, either version 3 of the License, or
  13. // (at your option) any later version.
  14. //
  15. // This program is distributed in the hope that it will be useful,
  16. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  17. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  18. // GNU General Public License for more details.
  19. //
  20. // You should have received a copy of the GNU General Public License
  21. // along with this program. If not, see <http://www.gnu.org/licenses/>.
  22. //
  23. #import "CCGraphics.h"
  24. #import "CCUtility.h"
  25. #import "NSString+TruncateToWidth.h"
  26. #import "NCBridgeSwift.h"
  27. @implementation CCGraphics
  28. + (UIImage *)thumbnailImageForVideo:(NSURL *)videoURL atTime:(NSTimeInterval)time
  29. {
  30. AVURLAsset *asset = [[AVURLAsset alloc] initWithURL:videoURL options:nil];
  31. NSParameterAssert(asset);
  32. AVAssetImageGenerator *assetIG =
  33. [[AVAssetImageGenerator alloc] initWithAsset:asset];
  34. assetIG.appliesPreferredTrackTransform = YES;
  35. assetIG.apertureMode = AVAssetImageGeneratorApertureModeEncodedPixels;
  36. CGImageRef thumbnailImageRef = NULL;
  37. CFTimeInterval thumbnailImageTime = time;
  38. NSError *igError = nil;
  39. thumbnailImageRef =
  40. [assetIG copyCGImageAtTime:CMTimeMake(thumbnailImageTime, 60) actualTime:NULL error:&igError];
  41. if (!thumbnailImageRef) NSLog(@"[LOG] thumbnailImageGenerationError %@", igError );
  42. UIImage *thumbnailImage = thumbnailImageRef ? [[UIImage alloc] initWithCGImage:thumbnailImageRef] : nil;
  43. return thumbnailImage;
  44. }
  45. // mix two image
  46. + (UIImage *)overlayImage:(UIImage *)backgroundImage watermarkImage:(UIImage *)watermarkImage where:(NSString *)where
  47. {
  48. // example watermarkImage = [UIImage imageNamed:@"lock"];
  49. UIGraphicsBeginImageContext(backgroundImage.size);
  50. [backgroundImage drawInRect:CGRectMake(0, 0, backgroundImage.size.width, backgroundImage.size.height)];
  51. if ([where isEqualToString:@"right"]) [watermarkImage drawInRect:CGRectMake(backgroundImage.size.width - watermarkImage.size.width, backgroundImage.size.height - watermarkImage.size.height, watermarkImage.size.width, watermarkImage.size.height)];
  52. if ([where isEqualToString:@"left"]) [watermarkImage drawInRect:CGRectMake(0, backgroundImage.size.height - watermarkImage.size.height, backgroundImage.size.width - watermarkImage.size.width, backgroundImage.size.height - watermarkImage.size.height)];
  53. UIImage *result = UIGraphicsGetImageFromCurrentImageContext();
  54. UIGraphicsEndImageContext();
  55. return result;
  56. }
  57. + (UIImage *)generateImageFromVideo:(NSString *)videoPath
  58. {
  59. NSURL *url = [NSURL fileURLWithPath:videoPath];
  60. NSError *error = NULL;
  61. AVURLAsset* asset = [AVURLAsset URLAssetWithURL:url options:nil];
  62. AVAssetImageGenerator* imageGenerator = [AVAssetImageGenerator assetImageGeneratorWithAsset:asset];
  63. imageGenerator.appliesPreferredTrackTransform = YES;
  64. // CMTime time = CMTimeMake(1, 65);
  65. CGImageRef cgImage = [imageGenerator copyCGImageAtTime:CMTimeMake(0, 1) actualTime:nil error:&error];
  66. if(error) return nil;
  67. UIImage* image = [UIImage imageWithCGImage:cgImage];
  68. CGImageRelease(cgImage);
  69. return image;
  70. }
  71. + (UIImage *)scaleImage:(UIImage *)image toSize:(CGSize)targetSize
  72. {
  73. //If scaleFactor is not touched, no scaling will occur
  74. CGFloat scaleFactor = 1.0;
  75. //Deciding which factor to use to scale the image (factor = targetSize / imageSize)
  76. if (image.size.width > targetSize.width || image.size.height > targetSize.height)
  77. if (!((scaleFactor = (targetSize.width / image.size.width)) > (targetSize.height / image.size.height))) //scale to fit width, or
  78. scaleFactor = targetSize.height / image.size.height; // scale to fit heigth.
  79. UIGraphicsBeginImageContext(targetSize);
  80. //Creating the rect where the scaled image is drawn in
  81. CGRect rect = CGRectMake((targetSize.width - image.size.width * scaleFactor) / 2,
  82. (targetSize.height - image.size.height * scaleFactor) / 2,
  83. image.size.width * scaleFactor, image.size.height * scaleFactor);
  84. //Draw the image into the rect
  85. [image drawInRect:rect];
  86. //Saving the image, ending image context
  87. UIImage *scaledImage = UIGraphicsGetImageFromCurrentImageContext();
  88. UIGraphicsEndImageContext();
  89. return scaledImage;
  90. }
  91. + (UIImage *)scaleImage:(UIImage *)image toSize:(CGSize)targetSize isAspectRation:(BOOL)aspect
  92. {
  93. CGFloat originRatio = image.size.width / image.size.height;
  94. CGFloat newRatio = targetSize.width / targetSize.height;
  95. CGSize sz;
  96. CGFloat scale = 1.0;
  97. if (!aspect) {
  98. sz = targetSize;
  99. }else {
  100. if (originRatio < newRatio) {
  101. sz.height = targetSize.height;
  102. sz.width = targetSize.height * originRatio;
  103. }else {
  104. sz.width = targetSize.width;
  105. sz.height = targetSize.width / originRatio;
  106. }
  107. }
  108. sz.width /= scale;
  109. sz.height /= scale;
  110. UIGraphicsBeginImageContextWithOptions(sz, NO, scale);
  111. [image drawInRect:CGRectMake(0, 0, sz.width, sz.height)];
  112. UIImage *newImage = UIGraphicsGetImageFromCurrentImageContext();
  113. UIGraphicsEndImageContext();
  114. return newImage;
  115. }
  116. + (UIImage *)createNewImageFrom:(NSString *)fileNameView fileID:(NSString *)fileID extension:(NSString *)extension size:(NSString *)size imageForUpload:(BOOL)imageForUpload typeFile:(NSString *)typeFile writeImage:(BOOL)writeImage optimizedFileName:(BOOL)optimizedFileName
  117. {
  118. UIImage *originalImage;
  119. UIImage *scaleImage;
  120. CGRect rect;
  121. CGFloat width, height;
  122. NSString *ext = [extension lowercaseString];
  123. NSString *fileNamePath = [CCUtility getDirectoryProviderStorageFileID:fileID fileNameView:fileNameView];
  124. if (![CCUtility fileProviderStorageExists:fileID fileNameView:fileNameView]) return nil;
  125. // only viedo / image
  126. if (![typeFile isEqualToString: k_metadataTypeFile_image] && ![typeFile isEqualToString: k_metadataTypeFile_video]) return nil;
  127. if ([typeFile isEqualToString: k_metadataTypeFile_image]) {
  128. originalImage = [UIImage imageWithContentsOfFile:fileNamePath];
  129. }
  130. if ([typeFile isEqualToString: k_metadataTypeFile_video]) {
  131. // create symbolik link for read video file in temp
  132. [[NSFileManager defaultManager] removeItemAtPath:[NSTemporaryDirectory() stringByAppendingString:@"tempvideo.mp4"] error:nil];
  133. [[NSFileManager defaultManager] linkItemAtPath:fileNamePath toPath:[NSTemporaryDirectory() stringByAppendingString:@"tempvideo.mp4"] error:nil];
  134. originalImage = [self generateImageFromVideo:[NSTemporaryDirectory() stringByAppendingString:@"tempvideo.mp4"]];
  135. }
  136. if ([size isEqualToString:@"xs"]) scaleImage = [self scaleImage:originalImage toSize:CGSizeMake(32, 32)];
  137. if ([size isEqualToString:@"s"]) scaleImage = [self scaleImage:originalImage toSize:CGSizeMake(64, 64)];
  138. if ([size isEqualToString:@"m"]) scaleImage = [self scaleImage:originalImage toSize:CGSizeMake(128, 128)];
  139. if ([size isEqualToString:@"l"]) scaleImage = [self scaleImage:originalImage toSize:CGSizeMake(640, 640)];
  140. if ([size isEqualToString:@"xl"]) scaleImage = [self scaleImage:originalImage toSize:CGSizeMake(1024, 1024)];
  141. scaleImage = [UIImage imageWithData:UIImageJPEGRepresentation(scaleImage, 0.5f)];
  142. // it is request write photo ?
  143. if (writeImage && scaleImage) {
  144. if (imageForUpload) {
  145. // if it is preview for Upload then trasform it in gray scale
  146. scaleImage = [self grayscale:scaleImage];
  147. [UIImagePNGRepresentation(scaleImage) writeToFile:[CCUtility getDirectoryProviderStorageIconFileID:fileID fileNameView:fileNameView] atomically: YES];
  148. } else {
  149. [UIImagePNGRepresentation(scaleImage) writeToFile:[CCUtility getDirectoryProviderStorageIconFileID:fileID fileNameView:fileNameView] atomically: YES];
  150. }
  151. }
  152. // Optimized photos resolution
  153. // Resize image 640 x 480 ( with proportion : 1,333)
  154. if (originalImage.size.height < originalImage.size.width) {
  155. // (lanscape)
  156. width = 640;
  157. height = 480;
  158. } else {
  159. // (portrait)
  160. height = 640;
  161. width = 480;
  162. }
  163. // Optimized photos resolution
  164. if ([typeFile isEqualToString: k_metadataTypeFile_image] && [ext isEqualToString:@"gif"] == NO && optimizedFileName && scaleImage && (originalImage.size.width > width || originalImage.size.height > height)) {
  165. // conversion scale proportion
  166. if (height > width) {
  167. float proportion = originalImage.size.height / originalImage.size.width;
  168. width = height / proportion;
  169. } else {
  170. float proportion = originalImage.size.width / originalImage.size.height;
  171. height = width / proportion;
  172. }
  173. rect = CGRectMake(0,0,width,height);
  174. UIGraphicsBeginImageContext(rect.size);
  175. [originalImage drawInRect:rect];
  176. UIImage *resizeImage = UIGraphicsGetImageFromCurrentImageContext();
  177. UIGraphicsEndImageContext();
  178. resizeImage = [UIImage imageWithData:UIImageJPEGRepresentation(resizeImage, 0.5f)];
  179. if (resizeImage)
  180. [UIImagePNGRepresentation(resizeImage) writeToFile:[CCUtility getDirectoryProviderStorageFileID:fileID fileNameView:fileNameView] atomically: YES];
  181. }
  182. return scaleImage;
  183. }
  184. + (UIColor *)colorFromHexString:(NSString *)hexString
  185. {
  186. unsigned rgbValue = 0;
  187. NSScanner *scanner = [NSScanner scannerWithString:hexString];
  188. [scanner setScanLocation:1]; // bypass '#' character
  189. [scanner scanHexInt:&rgbValue];
  190. return [UIColor colorWithRed:((rgbValue & 0xFF0000) >> 16)/255.0 green:((rgbValue & 0xFF00) >> 8)/255.0 blue:(rgbValue & 0xFF)/255.0 alpha:1.0];
  191. }
  192. + (UIImage *)changeThemingColorImage:(UIImage *)image color:(UIColor *)color
  193. {
  194. CGRect rect = CGRectMake(0, 0, image.size.width*2, image.size.height*2);
  195. UIGraphicsBeginImageContext(rect.size);
  196. CGContextRef context = UIGraphicsGetCurrentContext();
  197. CGContextClipToMask(context, rect, image.CGImage);
  198. CGContextSetFillColorWithColor(context, [color CGColor]);
  199. CGContextFillRect(context, rect);
  200. UIImage *img = UIGraphicsGetImageFromCurrentImageContext();
  201. UIGraphicsEndImageContext();
  202. return [UIImage imageWithCGImage:img.CGImage scale:2.0 orientation: UIImageOrientationDownMirrored];
  203. }
  204. + (UIImage*)drawText:(NSString*)text inImage:(UIImage*)image colorText:(UIColor *)colorText sizeOfFont:(CGFloat)sizeOfFont
  205. {
  206. NSDictionary* attributes = @{NSFontAttributeName: [UIFont systemFontOfSize:sizeOfFont], NSForegroundColorAttributeName:colorText};
  207. NSAttributedString* attributedString = [[NSAttributedString alloc] initWithString:text attributes:attributes];
  208. int x = image.size.width/2 - attributedString.size.width/2;
  209. int y = image.size.height/2 - attributedString.size.height/2;
  210. UIGraphicsBeginImageContext(image.size);
  211. [image drawInRect:CGRectMake(0,0,image.size.width,image.size.height)];
  212. CGRect rect = CGRectMake(x, y, image.size.width, image.size.height);
  213. [[UIColor whiteColor] set];
  214. [text drawInRect:CGRectIntegral(rect) withAttributes:attributes];
  215. UIImage *newImage = UIGraphicsGetImageFromCurrentImageContext();
  216. newImage = [UIImage imageWithCGImage:newImage.CGImage scale:2 orientation:UIImageOrientationUp];
  217. UIGraphicsEndImageContext();
  218. return newImage;
  219. }
  220. // ------------------------------------------------------------------------------------------------------
  221. // MARK: Blur Image
  222. // ------------------------------------------------------------------------------------------------------
  223. + (UIImage *)blurryImage:(UIImage *)image withBlurLevel:(CGFloat)blur toSize:(CGSize)toSize
  224. {
  225. CIImage *inputImage = [CIImage imageWithCGImage:image.CGImage];
  226. CIFilter *filter = [CIFilter filterWithName:@"CIGaussianBlur" keysAndValues:kCIInputImageKey, inputImage, @"inputRadius", @(blur), nil];
  227. CIImage *outputImage = filter.outputImage;
  228. CIContext *context = [CIContext contextWithOptions:nil];
  229. CGImageRef outImage = [context createCGImage:outputImage fromRect:[outputImage extent]];
  230. UIImage *blurImage = [UIImage imageWithCGImage:outImage];
  231. return [CCGraphics scaleImage:blurImage toSize:toSize isAspectRation:YES];
  232. }
  233. // ------------------------------------------------------------------------------------------------------
  234. // MARK: Is Light Color
  235. // ------------------------------------------------------------------------------------------------------
  236. /*
  237. Color visibility can be determined according to the following algorithm:
  238. (This is a suggested algorithm that is still open to change.)
  239. Two colors provide good color visibility if the brightness difference and the color difference between the two colors are greater than a set range.
  240. Color brightness is determined by the following formula:
  241. ((Red value X 299) + (Green value X 587) + (Blue value X 114)) / 1000
  242. Note: This algorithm is taken from a formula for converting RGB values to YIQ values. This brightness value gives a perceived brightness for a color.
  243. Color difference is determined by the following formula:
  244. (maximum (Red value 1, Red value 2) - minimum (Red value 1, Red value 2)) + (maximum (Green value 1, Green value 2) - minimum (Green value 1, Green value 2)) + (maximum (Blue value 1, Blue value 2) - minimum (Blue value 1, Blue value 2))
  245. */
  246. + (BOOL)isLight:(UIColor *)color
  247. {
  248. const CGFloat *componentColors = CGColorGetComponents(color.CGColor);
  249. CGFloat colorBrightness = ((componentColors[0] * 299) + (componentColors[1] * 587) + (componentColors[2] * 114)) / 1000;
  250. if (colorBrightness < 0.8) { // STD 0.5
  251. return false;
  252. } else {
  253. return true;
  254. }
  255. }
  256. + (UIImage *)grayscale:(UIImage *)sourceImage
  257. {
  258. /* const UInt8 luminance = (red * 0.2126) + (green * 0.7152) + (blue * 0.0722); // Good luminance value */
  259. /// Create a gray bitmap context
  260. const size_t width = (size_t)sourceImage.size.width;
  261. const size_t height = (size_t)sourceImage.size.height;
  262. const int kNyxNumberOfComponentsPerGreyPixel = 3;
  263. CGRect imageRect = CGRectMake(0, 0, sourceImage.size.width, sourceImage.size.height);
  264. CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceGray();
  265. CGContextRef bmContext = CGBitmapContextCreate(NULL, width, height, 8/*Bits per component*/, width * kNyxNumberOfComponentsPerGreyPixel, colorSpace, kCGImageAlphaNone);
  266. CGColorSpaceRelease(colorSpace);
  267. if (!bmContext)
  268. return nil;
  269. /// Image quality
  270. CGContextSetShouldAntialias(bmContext, false);
  271. CGContextSetInterpolationQuality(bmContext, kCGInterpolationHigh);
  272. /// Draw the image in the bitmap context
  273. CGContextDrawImage(bmContext, imageRect, sourceImage.CGImage);
  274. /// Create an image object from the context
  275. CGImageRef grayscaledImageRef = CGBitmapContextCreateImage(bmContext);
  276. UIImage *grayscaled = [UIImage imageWithCGImage:grayscaledImageRef scale:sourceImage.scale orientation:sourceImage.imageOrientation];
  277. /// Cleanup
  278. CGImageRelease(grayscaledImageRef);
  279. CGContextRelease(bmContext);
  280. return grayscaled;
  281. }
  282. + (UIImage *)generateSinglePixelImageWithColor:(UIColor *)color
  283. {
  284. CGSize imageSize = CGSizeMake(1.0f, 1.0f);
  285. UIGraphicsBeginImageContextWithOptions(imageSize, NO, 0.0f);
  286. CGContextRef theContext = UIGraphicsGetCurrentContext();
  287. CGContextSetFillColorWithColor(theContext, color.CGColor);
  288. CGContextFillRect(theContext, CGRectMake(0.0f, 0.0f, imageSize.width, imageSize.height));
  289. CGImageRef theCGImage = CGBitmapContextCreateImage(theContext);
  290. UIImage *theImage;
  291. if ([[UIImage class] respondsToSelector:@selector(imageWithCGImage:scale:orientation:)]) {
  292. theImage = [UIImage imageWithCGImage:theCGImage scale:[UIScreen mainScreen].scale orientation:UIImageOrientationUp];
  293. } else {
  294. theImage = [UIImage imageWithCGImage:theCGImage];
  295. }
  296. CGImageRelease(theCGImage);
  297. return theImage;
  298. }
  299. + (void)addImageToTitle:(NSString *)title colorTitle:(UIColor *)colorTitle imageTitle:(UIImage *)imageTitle navigationItem:(UINavigationItem *)navigationItem
  300. {
  301. UIView *navView = [UIView new];
  302. UILabel *label = [UILabel new];
  303. label.text = title;
  304. [label sizeToFit];
  305. label.center = navView.center;
  306. label.textColor = colorTitle;
  307. label.textAlignment = NSTextAlignmentCenter;
  308. CGFloat correct = 6;
  309. UIImageView *image = [UIImageView new];
  310. image.image = imageTitle;
  311. CGFloat imageAspect = image.image.size.width/image.image.size.height;
  312. image.frame = CGRectMake(label.frame.origin.x-label.frame.size.height*imageAspect, label.frame.origin.y+correct/2, label.frame.size.height*imageAspect-correct, label.frame.size.height-correct);
  313. image.contentMode = UIViewContentModeScaleAspectFit;
  314. [navView addSubview:label];
  315. [navView addSubview:image];
  316. navigationItem.titleView = navView;
  317. [navView sizeToFit];
  318. }
  319. + (void)settingThemingColor:(NSString *)themingColor themingColorElement:(NSString *)themingColorElement themingColorText:(NSString *)themingColorText
  320. {
  321. UIColor *newColor, *newColorElement, *newColorText;
  322. // COLOR
  323. if (themingColor.length == 7) {
  324. newColor = [CCGraphics colorFromHexString:themingColor];
  325. } else {
  326. newColor = [NCBrandColor sharedInstance].customer;
  327. }
  328. // COLOR TEXT
  329. if (themingColorText.length == 7) {
  330. newColorText = [CCGraphics colorFromHexString:themingColorText];
  331. } else {
  332. newColorText = [NCBrandColor sharedInstance].customerText;
  333. }
  334. // COLOR ELEMENT
  335. if (themingColorElement.length == 7) {
  336. newColorElement = [CCGraphics colorFromHexString:themingColorElement];
  337. } else {
  338. if ([themingColorText isEqualToString:@"#000000"])
  339. newColorElement = [UIColor blackColor];
  340. else
  341. newColorElement = newColor;
  342. }
  343. [NCBrandColor sharedInstance].brand = newColor;
  344. [NCBrandColor sharedInstance].brandElement = newColorElement;
  345. [NCBrandColor sharedInstance].brandText = newColorText;
  346. }
  347. @end
  348. // ------------------------------------------------------------------------------------------------------
  349. // MARK: Avatar
  350. // ------------------------------------------------------------------------------------------------------
  351. @implementation CCAvatar
  352. - (id)initWithImage:(UIImage *)image borderColor:(UIColor*)borderColor borderWidth:(float)borderWidth
  353. {
  354. self = [super initWithImage:image];
  355. float cornerRadius = self.frame.size.height/2.0f;
  356. CALayer *layer = [self layer];
  357. [layer setMasksToBounds:YES];
  358. [layer setCornerRadius: cornerRadius];
  359. [layer setBorderWidth: borderWidth];
  360. [layer setBorderColor:[borderColor CGColor]];
  361. return self;
  362. }
  363. @end