UIImageView+HBHttpCache.m 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390
  1. //
  2. // UIImageView+HBHttpCache.m
  3. // MyTest
  4. //
  5. // Created by weqia on 13-8-22.
  6. // Copyright (c) 2013年 weqia. All rights reserved.
  7. //
  8. #import "UIImageView+HBHttpCache.h"
  9. #import "UIImage+HBClass.h"
  10. #import "HBHttpRequestCache.h"
  11. @implementation UIImageView (HBHttpCache)
  12. static char operationKey='a';
  13. #pragma -mark 私有方法
  14. -(void)cancel
  15. {
  16. id<HBHttpOperationDelegate> operation=objc_getAssociatedObject(self, &operationKey);
  17. if(operation){
  18. [operation cancel];
  19. objc_setAssociatedObject(self, &operationKey, nil, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
  20. }
  21. }
  22. #pragma -mark 接口方法
  23. -(void) setImageWithURL:(NSString*)url
  24. {
  25. [self setImageWithURL:url layout:UIImageViewLayoutNone placeholderImage:nil process:nil complete:nil option:HBHttpImageDownloaderOptionUseCache|HBHttpImageDownloaderOptionRetry];
  26. }
  27. -(void) setImageWithURL:(NSString*)url
  28. layout:(UIImageViewLayoutType)layout
  29. {
  30. [self setImageWithURL:url layout:layout placeholderImage:nil process:nil complete:nil option:HBHttpImageDownloaderOptionUseCache|HBHttpImageDownloaderOptionRetry];
  31. }
  32. -(void) setImageWithURL:(NSString *)url
  33. layout:(UIImageViewLayoutType)layout
  34. placeholderImage:(UIImage*)placeholderImage
  35. {
  36. [self setImageWithURL:url layout:layout placeholderImage:placeholderImage process:nil complete:nil option:HBHttpImageDownloaderOptionUseCache|HBHttpImageDownloaderOptionRetry];
  37. }
  38. -(void) setImageWithURL:(NSString *)url
  39. layout:(UIImageViewLayoutType)layout
  40. placeholderImage:(UIImage*)placeholderImage
  41. process:(HBHttpImageDownloaderProcessBlock) process
  42. complete:(HBHttpImageDownloaderCompleteBlock)complete
  43. {
  44. [self setImageWithURL:url layout:layout placeholderImage:placeholderImage process:process complete:complete option:HBHttpImageDownloaderOptionUseCache|HBHttpImageDownloaderOptionRetry];
  45. }
  46. -(void) setImageWithURL:(NSString *)url
  47. layout:(UIImageViewLayoutType)layout
  48. placeholderImage:(UIImage*)placeholderImage
  49. process:(HBHttpImageDownloaderProcessBlock)process
  50. complete:(HBHttpImageDownloaderCompleteBlock)complete
  51. option:(HBHttpImageDownloaderOption)option
  52. {
  53. self.image=placeholderImage;
  54. /****图片处理block****/
  55. __block void(^block)(UIImage*);
  56. /****图片下载block****/
  57. void(^download)()=^{
  58. [self cancel];
  59. [[HBHttpImageDownloader shareDownlader] downBitmapWithURL:url process:process complete:^(UIImage *image, NSData *data, NSError *error, BOOL success){
  60. /****回调处理****/
  61. if(block){
  62. block(image);
  63. }
  64. if(complete){
  65. if([NSThread isMainThread]){
  66. if(complete){
  67. complete(image,data,error,success);
  68. }
  69. }else{
  70. dispatch_async(dispatch_get_main_queue(), ^{
  71. if(complete){
  72. complete(image,data,error,success);
  73. }
  74. });
  75. }
  76. }
  77. } option:option valueReturn:^(id<HBHttpOperationDelegate> operation) {
  78. /****operation 注册为属性,以便于停止下载操作****/
  79. objc_setAssociatedObject(self, &operationKey, operation, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
  80. }];
  81. };
  82. __weak UIImageView * wself=self; // 防止 UIImageView 释放时, self 变成废指针
  83. if(layout==UIImageViewLayoutNone){
  84. block=^(UIImage * image){
  85. if(!wself) return;
  86. __strong UIImageView * sself=wself; // 在操作为完成之前,UIImageView 不能被释放
  87. /*****界面操作需放在主线程*****/
  88. if([NSThread isMainThread]){
  89. sself.image=image;
  90. }else{
  91. dispatch_async(dispatch_get_main_queue(), ^{
  92. sself.image=image;
  93. });
  94. }
  95. };
  96. download(); // 执行下载操作
  97. }else if(layout==UIImageViewLayoutLimit){
  98. NSMutableString * key=[NSMutableString stringWithString:url];
  99. [key appendFormat:@".limit"];
  100. block=^(UIImage * image){
  101. if(image){
  102. if(!wself) return;
  103. __strong UIImageView * sself=wself;
  104. UIImage * limitimage=[image getLimitImage:sself.frame.size];
  105. /*****界面操作需放在主线程*****/
  106. if([NSThread isMainThread]){
  107. CGRect frame=sself.frame;
  108. frame.size=limitimage.size;
  109. sself.frame=frame;
  110. sself.image=limitimage;
  111. }else{
  112. dispatch_async(dispatch_get_main_queue(), ^{
  113. CGRect frame=sself.frame;
  114. frame.size=limitimage.size;
  115. sself.frame=frame;
  116. sself.image=limitimage;
  117. });
  118. }if(option&HBHttpImageDownloaderOptionUseCache){
  119. /***将生成的限制大小的图片缓存起来***/
  120. [[HBHttpRequestCache shareCache] storeBitmap:limitimage withKey:key complete:nil];
  121. }
  122. }
  123. };
  124. /*****如果是HBHttpImageDownloaderOptionUseCache 先从缓存中加载限制大小的图片*****/
  125. if(option&HBHttpImageDownloaderOptionUseCache){
  126. [[HBHttpRequestCache shareCache] getBitmap:key complete:^(UIImage *image) {
  127. if(!wself) return;
  128. __strong UIImageView * sself=wself;
  129. if(image){
  130. CGRect frame=sself.frame;
  131. frame.size=image.size;
  132. sself.frame=frame;
  133. sself.image=image;
  134. if(complete){
  135. complete(image,nil,nil,YES);
  136. }
  137. }else{
  138. /*****如果缓存中没有,则从已缓存的原图生成,如果没有已缓存的原图,则会下载原图并生成目标图****/
  139. download();
  140. }
  141. }];
  142. }else{
  143. download();
  144. }
  145. }else if(layout==UIImageViewLayoutClick){
  146. NSMutableString * key=[NSMutableString stringWithString:url];
  147. [key appendFormat:@".click"];
  148. block=^(UIImage * image){
  149. if(!wself) return;
  150. __strong UIImageView * sself=wself;
  151. if(image){
  152. UIImage * clickimage=[image getClickImage:sself.frame.size];
  153. /*****界面操作需放在主线程*****/
  154. if([NSThread isMainThread]){
  155. sself.image=clickimage;
  156. }else{
  157. dispatch_async(dispatch_get_main_queue(), ^{
  158. sself.image=clickimage;
  159. });
  160. }if(option&HBHttpImageDownloaderOptionUseCache){
  161. /*****将生成的裁减过的图片缓存起来****/
  162. [[HBHttpRequestCache shareCache] storeBitmap:clickimage withKey:key complete:nil];
  163. }
  164. }
  165. };
  166. /*****如果是HBHttpImageDownloaderOptionUseCache 先从缓存中加载裁减过的图片*****/
  167. if(option&HBHttpImageDownloaderOptionUseCache){
  168. [[HBHttpRequestCache shareCache] getBitmap:key complete:^(UIImage *image) {
  169. if(!wself) return;
  170. __strong UIImageView * sself=wself;
  171. if(image){
  172. sself.image=image;
  173. if(complete){
  174. complete(image,nil,nil,YES);
  175. }
  176. }else{
  177. /*****如果缓存中没有,则从已缓存的原图生成,如果没有已缓存的原图,则会下载原图并生成目标图****/
  178. download();
  179. }
  180. }];
  181. }else{
  182. download();
  183. }
  184. }
  185. }
  186. -(void) setImageWithIndirectURL:(NSString *)indirectURL
  187. {
  188. [self setImageWithURL:indirectURL layout:UIImageViewLayoutNone placeholderImage:nil process:nil complete:nil option:HBHttpImageDownloaderOptionRetry|HBHttpImageDownloaderOptionUseCache];
  189. }
  190. -(void) setImageWithIndirectURL:(NSString *)indirectURL
  191. layout:(UIImageViewLayoutType)layout
  192. {
  193. [self setImageWithURL:indirectURL layout:layout placeholderImage:nil process:nil complete:nil option:HBHttpImageDownloaderOptionRetry|HBHttpImageDownloaderOptionUseCache];
  194. }
  195. -(void) setImageWithIndirectURL:(NSString *)indirectURL
  196. layout:(UIImageViewLayoutType)layout
  197. placeholderImage:(UIImage *)placeholderImage
  198. {
  199. [self setImageWithURL:indirectURL layout:layout placeholderImage:placeholderImage process:nil complete:nil option:HBHttpImageDownloaderOptionRetry|HBHttpImageDownloaderOptionUseCache];
  200. }
  201. -(void) setImageWithIndirectURL:(NSString *)indirectURL
  202. layout:(UIImageViewLayoutType)layout
  203. placeholderImage:(UIImage *)placeholderImage
  204. process:(HBHttpImageDownloaderProcessBlock)process
  205. complete:(HBHttpImageDownloaderCompleteBlock)complete
  206. {
  207. [self setImageWithURL:indirectURL layout:layout placeholderImage:placeholderImage process:process complete:complete option:HBHttpImageDownloaderOptionRetry|HBHttpImageDownloaderOptionUseCache];
  208. }
  209. -(void) setImageWithIndirectURL:(NSString *)indirectURL
  210. layout:(UIImageViewLayoutType)layout
  211. placeholderImage:(UIImage *)placeholderImage
  212. process:(HBHttpImageDownloaderProcessBlock)process
  213. complete:(HBHttpImageDownloaderCompleteBlock)complete
  214. option:(HBHttpImageDownloaderOption)option
  215. {
  216. self.image=placeholderImage;
  217. /****图片处理block****/
  218. __block void(^block)(UIImage*);
  219. /****图片下载block****/
  220. void(^download)()=^{
  221. [self cancel];
  222. [[HBHttpImageDownloader shareDownlader] downBitmapWithIndirectURL:indirectURL process:process complete:^(UIImage * image, NSData * data, NSError * error, BOOL success) {
  223. if(block){
  224. block(image);
  225. }
  226. /****回调处理****/
  227. if(complete){
  228. complete(image,data,error,success);
  229. }
  230. } option:option valueReturn:^(id<HBHttpOperationDelegate> operation) {
  231. /****operation 注册为属性,以便于停止下载操作****/
  232. objc_setAssociatedObject(self, &operationKey, operation,OBJC_ASSOCIATION_RETAIN_NONATOMIC);
  233. }];
  234. };
  235. __weak UIImageView * wself=self; // 防止 UIImageView 释放时, self 变成废指针
  236. if(layout==UIImageViewLayoutNone){
  237. block=^(UIImage * image){
  238. if(!wself) return;
  239. __strong UIImageView * sself=wself; // 在操作为完成之前,UIImageView 不能被释放
  240. /*****界面操作需放在主线程*****/
  241. if([NSThread isMainThread]){
  242. sself.image=image;
  243. }else{
  244. dispatch_async(dispatch_get_main_queue(), ^{
  245. sself.image=image;
  246. });
  247. }
  248. };
  249. download();
  250. }else if(layout==UIImageViewLayoutLimit){
  251. NSMutableString * key=[NSMutableString stringWithString:indirectURL];
  252. [key appendFormat:@".limit"];
  253. block=^(UIImage * image){
  254. if(!wself) return;
  255. __strong UIImageView * sself=wself;
  256. if(image){
  257. UIImage * limitimage=[image getLimitImage:sself.frame.size];
  258. /*****界面操作需放在主线程*****/
  259. if([NSThread isMainThread]){
  260. CGRect frame=sself.frame;
  261. frame.size=limitimage.size;
  262. sself.frame=frame;
  263. sself.image=limitimage;
  264. }else{
  265. dispatch_async(dispatch_get_main_queue(), ^{
  266. CGRect frame=sself.frame;
  267. frame.size=limitimage.size;
  268. sself.frame=frame;
  269. sself.image=limitimage;
  270. });
  271. }if(option&HBHttpImageDownloaderOptionUseCache){
  272. /*****将生成的限制大小的图片缓存起来****/
  273. [[HBHttpRequestCache shareCache] storeBitmap:limitimage withKey:key complete:nil];
  274. }
  275. }
  276. };
  277. if(option&HBHttpImageDownloaderOptionUseCache){
  278. /*****先从缓存中加载限制大小的图片*****/
  279. [[HBHttpRequestCache shareCache] getBitmap:key complete:^(UIImage *image) {
  280. if(!wself) return;
  281. __strong UIImageView * sself=wself;
  282. if(image){
  283. CGRect frame=sself.frame;
  284. frame.size=image.size;
  285. sself.frame=frame;
  286. sself.image=image;
  287. if(complete){
  288. complete(image,nil,nil,YES);
  289. }
  290. }else{
  291. /*****如果缓存中没有,则从已缓存的原图生成,如果没有已缓存的原图,则会下载原图并生成目标图****/
  292. download();
  293. }
  294. }];
  295. }else{
  296. download();
  297. }
  298. }else if(layout==UIImageViewLayoutClick){
  299. NSMutableString * key=[NSMutableString stringWithString:indirectURL];
  300. [key appendFormat:@".click"];
  301. block=^(UIImage * image){
  302. if(!wself) return;
  303. __strong UIImageView * sself=wself;
  304. if(image){
  305. UIImage * clickimage=[image getClickImage:sself.frame.size];
  306. /*****界面操作需放在主线程*****/
  307. if([NSThread isMainThread]){
  308. sself.image=clickimage;
  309. }else{
  310. dispatch_async(dispatch_get_main_queue(), ^{
  311. sself.image=clickimage;
  312. });
  313. }if(option&HBHttpImageDownloaderOptionUseCache){
  314. /*****将生成的裁减过的图片缓存起来****/
  315. [[HBHttpRequestCache shareCache] storeBitmap:clickimage withKey:key complete:nil];
  316. }
  317. }
  318. };
  319. if(option&HBHttpImageDownloaderOptionUseCache){
  320. /*****先从缓存中加载裁减过的图片*****/
  321. [[HBHttpRequestCache shareCache] getBitmap:key complete:^(UIImage *image) {
  322. if(!wself) return;
  323. __strong UIImageView * sself=wself;
  324. if(image){
  325. sself.image=image;
  326. if(complete){
  327. complete(image,nil,nil,YES);
  328. }
  329. }else{
  330. /*****如果缓存中没有,则从已缓存的原图生成,如果没有已缓存的原图,则会下载原图并生成目标图****/
  331. download();
  332. }
  333. }];
  334. }else{
  335. download();
  336. }
  337. }
  338. }
  339. -(void) setImageWithCacheKey:(NSString *)key layout:(UIImageViewLayoutType)layout
  340. placeholderImage:(UIImage *)placeholderImage
  341. {
  342. NSMutableString * Key=[NSMutableString stringWithString:key];
  343. if(layout==UIImageViewLayoutLimit){
  344. [Key appendFormat:@".limit"];
  345. }else if(layout==UIImageViewLayoutClick){
  346. [Key appendFormat:@".click"];
  347. }
  348. UIImage * image=[[HBHttpRequestCache shareCache] getBitmapFromMemory:Key];
  349. if(image){
  350. self.image=image;
  351. if(layout==UIImageViewLayoutLimit){
  352. CGRect frame=self.frame;
  353. frame.size=image.size;
  354. self.frame=frame;
  355. }
  356. }else{
  357. self.image=placeholderImage;
  358. }
  359. }
  360. @end