mirror of
https://github.com/Swiftgram/Telegram-iOS.git
synced 2025-09-01 18:33:10 +00:00
309 lines
8.8 KiB
Plaintext
309 lines
8.8 KiB
Plaintext
/* Copyright (c) 2014-present, Facebook, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This source code is licensed under the BSD-style license found in the
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
*/
|
|
|
|
#import "ASBasicImageDownloader.h"
|
|
|
|
#import <objc/runtime.h>
|
|
|
|
#import <UIKit/UIKit.h>
|
|
|
|
#import "ASBasicImageDownloaderInternal.h"
|
|
#import "ASThread.h"
|
|
|
|
|
|
#pragma mark -
|
|
/**
|
|
* Collection of properties associated with a download request.
|
|
*/
|
|
|
|
typedef void (^ASBasicImageDownloaderContextProgressBlock)(CGFloat);
|
|
typedef void (^ASBasicImageDownloaderContextCompletionBlock)(CGImageRef, NSError *);
|
|
|
|
NSString * const kASBasicImageDownloaderContextCallbackQueue = @"kASBasicImageDownloaderContextCallbackQueue";
|
|
NSString * const kASBasicImageDownloaderContextProgressBlock = @"kASBasicImageDownloaderContextProgressBlock";
|
|
NSString * const kASBasicImageDownloaderContextCompletionBlock = @"kASBasicImageDownloaderContextCompletionBlock";
|
|
|
|
@interface ASBasicImageDownloaderContext ()
|
|
{
|
|
BOOL _invalid;
|
|
ASDN::RecursiveMutex _propertyLock;
|
|
}
|
|
|
|
@property (nonatomic, strong) NSMutableArray *callbackDatas;
|
|
|
|
@end
|
|
|
|
@implementation ASBasicImageDownloaderContext
|
|
|
|
static NSMutableDictionary *currentRequests = nil;
|
|
static ASDN::RecursiveMutex currentRequestsLock;
|
|
|
|
+ (ASBasicImageDownloaderContext *)contextForURL:(NSURL *)URL
|
|
{
|
|
ASDN::MutexLocker l(currentRequestsLock);
|
|
if (!currentRequests) {
|
|
currentRequests = [[NSMutableDictionary alloc] init];
|
|
}
|
|
ASBasicImageDownloaderContext *context = currentRequests[URL];
|
|
if (!context) {
|
|
context = [[ASBasicImageDownloaderContext alloc] initWithURL:URL];
|
|
currentRequests[URL] = context;
|
|
}
|
|
return context;
|
|
}
|
|
|
|
+ (void)cancelContextWithURL:(NSURL *)URL
|
|
{
|
|
ASDN::MutexLocker l(currentRequestsLock);
|
|
if (currentRequests) {
|
|
[currentRequests removeObjectForKey:URL];
|
|
}
|
|
}
|
|
|
|
- (instancetype)initWithURL:(NSURL *)URL
|
|
{
|
|
if (self = [super init]) {
|
|
_URL = URL;
|
|
_callbackDatas = [NSMutableArray array];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (void)cancel
|
|
{
|
|
ASDN::MutexLocker l(_propertyLock);
|
|
|
|
NSURLSessionTask *sessionTask = self.sessionTask;
|
|
if (sessionTask) {
|
|
[sessionTask cancel];
|
|
self.sessionTask = nil;
|
|
}
|
|
|
|
_invalid = YES;
|
|
[self.class cancelContextWithURL:self.URL];
|
|
}
|
|
|
|
- (BOOL)isCancelled
|
|
{
|
|
ASDN::MutexLocker l(_propertyLock);
|
|
return _invalid;
|
|
}
|
|
|
|
- (void)addCallbackData:(NSDictionary *)callbackData
|
|
{
|
|
ASDN::MutexLocker l(_propertyLock);
|
|
[self.callbackDatas addObject:callbackData];
|
|
}
|
|
|
|
- (void)performProgressBlocks:(CGFloat)progress
|
|
{
|
|
ASDN::MutexLocker l(_propertyLock);
|
|
for (NSDictionary *callbackData in self.callbackDatas) {
|
|
ASBasicImageDownloaderContextProgressBlock progressBlock = callbackData[kASBasicImageDownloaderContextProgressBlock];
|
|
dispatch_queue_t callbackQueue = callbackData[kASBasicImageDownloaderContextCallbackQueue];
|
|
|
|
if (progressBlock) {
|
|
dispatch_async(callbackQueue, ^{
|
|
progressBlock(progress);
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
- (void)completeWithImage:(UIImage *)image error:(NSError *)error
|
|
{
|
|
ASDN::MutexLocker l(_propertyLock);
|
|
for (NSDictionary *callbackData in self.callbackDatas) {
|
|
ASBasicImageDownloaderContextCompletionBlock completionBlock = callbackData[kASBasicImageDownloaderContextCompletionBlock];
|
|
dispatch_queue_t callbackQueue = callbackData[kASBasicImageDownloaderContextCallbackQueue];
|
|
|
|
if (completionBlock) {
|
|
dispatch_async(callbackQueue, ^{
|
|
completionBlock(image.CGImage, error);
|
|
});
|
|
}
|
|
}
|
|
|
|
self.sessionTask = nil;
|
|
[self.callbackDatas removeAllObjects];
|
|
}
|
|
|
|
- (NSURLSessionTask *)createSessionTaskIfNecessaryWithBlock:(NSURLSessionTask *(^)())creationBlock {
|
|
{
|
|
ASDN::MutexLocker l(_propertyLock);
|
|
|
|
if (self.isCancelled) {
|
|
return nil;
|
|
}
|
|
|
|
if (self.sessionTask && (self.sessionTask.state == NSURLSessionTaskStateRunning)) {
|
|
return nil;
|
|
}
|
|
}
|
|
|
|
NSURLSessionTask *newTask = creationBlock();
|
|
|
|
{
|
|
ASDN::MutexLocker l(_propertyLock);
|
|
|
|
if (self.isCancelled) {
|
|
return nil;
|
|
}
|
|
|
|
if (self.sessionTask && (self.sessionTask.state == NSURLSessionTaskStateRunning)) {
|
|
return nil;
|
|
}
|
|
|
|
self.sessionTask = newTask;
|
|
|
|
return self.sessionTask;
|
|
}
|
|
}
|
|
|
|
@end
|
|
|
|
|
|
#pragma mark -
|
|
/**
|
|
* NSURLSessionDownloadTask lacks a `userInfo` property, so add this association ourselves.
|
|
*/
|
|
@interface NSURLRequest (ASBasicImageDownloader)
|
|
@property (nonatomic, strong) ASBasicImageDownloaderContext *asyncdisplaykit_context;
|
|
@end
|
|
|
|
@implementation NSURLRequest (ASBasicImageDownloader)
|
|
static const char *kContextKey = NSStringFromClass(ASBasicImageDownloaderContext.class).UTF8String;
|
|
- (void)setAsyncdisplaykit_context:(ASBasicImageDownloaderContext *)asyncdisplaykit_context
|
|
{
|
|
objc_setAssociatedObject(self, kContextKey, asyncdisplaykit_context, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
|
|
}
|
|
- (ASBasicImageDownloader *)asyncdisplaykit_context
|
|
{
|
|
return objc_getAssociatedObject(self, kContextKey);
|
|
}
|
|
@end
|
|
|
|
|
|
#pragma mark -
|
|
@interface ASBasicImageDownloader () <NSURLSessionDownloadDelegate>
|
|
{
|
|
NSOperationQueue *_sessionDelegateQueue;
|
|
NSURLSession *_session;
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation ASBasicImageDownloader
|
|
|
|
#pragma mark Lifecycle.
|
|
|
|
- (instancetype)init
|
|
{
|
|
if (!(self = [super init]))
|
|
return nil;
|
|
|
|
_sessionDelegateQueue = [[NSOperationQueue alloc] init];
|
|
_session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]
|
|
delegate:self
|
|
delegateQueue:_sessionDelegateQueue];
|
|
|
|
return self;
|
|
}
|
|
|
|
|
|
#pragma mark ASImageDownloaderProtocol.
|
|
|
|
- (id)downloadImageWithURL:(NSURL *)URL
|
|
callbackQueue:(dispatch_queue_t)callbackQueue
|
|
downloadProgressBlock:(void (^)(CGFloat))downloadProgressBlock
|
|
completion:(void (^)(CGImageRef, NSError *))completion
|
|
{
|
|
ASBasicImageDownloaderContext *context = [ASBasicImageDownloaderContext contextForURL:URL];
|
|
|
|
// NSURLSessionDownloadTask will do file I/O to create a temp directory. If called on the main thread this will
|
|
// cause significant performance issues.
|
|
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
|
|
// associate metadata with it
|
|
NSMutableDictionary *callbackData = [NSMutableDictionary dictionary];
|
|
callbackData[kASBasicImageDownloaderContextCallbackQueue] = callbackQueue ?: dispatch_get_main_queue();
|
|
|
|
if (downloadProgressBlock) {
|
|
callbackData[kASBasicImageDownloaderContextProgressBlock] = [downloadProgressBlock copy];
|
|
}
|
|
|
|
if (completion) {
|
|
callbackData[kASBasicImageDownloaderContextCompletionBlock] = [completion copy];
|
|
}
|
|
|
|
[context addCallbackData:[NSDictionary dictionaryWithDictionary:callbackData]];
|
|
|
|
// Create new task if necessary
|
|
NSURLSessionDownloadTask *task = (NSURLSessionDownloadTask *)[context createSessionTaskIfNecessaryWithBlock:^(){return [_session downloadTaskWithURL:URL];}];
|
|
|
|
if (task) {
|
|
task.originalRequest.asyncdisplaykit_context = context;
|
|
|
|
// start downloading
|
|
[task resume];
|
|
}
|
|
});
|
|
|
|
return context;
|
|
}
|
|
|
|
- (void)cancelImageDownloadForIdentifier:(id)downloadIdentifier
|
|
{
|
|
if (!downloadIdentifier) {
|
|
return;
|
|
}
|
|
|
|
ASDisplayNodeAssert([downloadIdentifier isKindOfClass:ASBasicImageDownloaderContext.class], @"unexpected downloadIdentifier");
|
|
ASBasicImageDownloaderContext *context = (ASBasicImageDownloaderContext *)downloadIdentifier;
|
|
|
|
[context cancel];
|
|
}
|
|
|
|
|
|
#pragma mark NSURLSessionDownloadDelegate.
|
|
|
|
- (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask
|
|
didWriteData:(int64_t)bytesWritten
|
|
totalBytesWritten:(int64_t)totalBytesWritten
|
|
totalBytesExpectedToWrite:(int64_t)totalBytesExpectedToWrite
|
|
{
|
|
ASBasicImageDownloaderContext *context = downloadTask.originalRequest.asyncdisplaykit_context;
|
|
[context performProgressBlocks:(CGFloat)totalBytesWritten / (CGFloat)totalBytesExpectedToWrite];
|
|
}
|
|
|
|
// invoked if the download succeeded with no error
|
|
- (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask
|
|
didFinishDownloadingToURL:(NSURL *)location
|
|
{
|
|
ASBasicImageDownloaderContext *context = downloadTask.originalRequest.asyncdisplaykit_context;
|
|
if ([context isCancelled]) {
|
|
return;
|
|
}
|
|
|
|
if (context) {
|
|
UIImage *image = [UIImage imageWithData:[NSData dataWithContentsOfURL:location]];
|
|
[context completeWithImage:image error:nil];
|
|
}
|
|
}
|
|
|
|
// invoked unconditionally
|
|
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionDownloadTask *)task
|
|
didCompleteWithError:(NSError *)error
|
|
{
|
|
ASBasicImageDownloaderContext *context = task.originalRequest.asyncdisplaykit_context;
|
|
if (context && error) {
|
|
[context completeWithImage:nil error:error];
|
|
}
|
|
}
|
|
|
|
@end
|