123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938 |
- //
- // TZImageManager.m
- // TZImagePickerController
- //
- // Created by 谭真 on 16/1/4.
- // Copyright © 2016年 谭真. All rights reserved.
- //
- #import "TZImageManager.h"
- #import "TZAssetModel.h"
- #import "TZImagePickerController.h"
- @interface TZImageManager ()
- #pragma clang diagnostic push
- #pragma clang diagnostic ignored "-Wdeprecated-declarations"
- @end
- @implementation TZImageManager
- CGSize AssetGridThumbnailSize;
- CGFloat TZScreenWidth;
- CGFloat TZScreenScale;
- static TZImageManager *manager;
- static dispatch_once_t onceToken;
- + (instancetype)manager {
- dispatch_once(&onceToken, ^{
- manager = [[self alloc] init];
- // manager.cachingImageManager = [[PHCachingImageManager alloc] init];
- // manager.cachingImageManager.allowsCachingHighQualityImages = YES;
-
- [manager configTZScreenWidth];
- });
- return manager;
- }
- + (void)deallocManager {
- onceToken = 0;
- manager = nil;
- }
- - (void)setPhotoWidth:(CGFloat)photoWidth {
- _photoWidth = photoWidth;
- TZScreenWidth = photoWidth / 2;
- }
- - (void)setColumnNumber:(NSInteger)columnNumber {
- [self configTZScreenWidth];
- _columnNumber = columnNumber;
- CGFloat margin = 4;
- CGFloat itemWH = (TZScreenWidth - 2 * margin - 4) / columnNumber - margin;
- AssetGridThumbnailSize = CGSizeMake(itemWH * TZScreenScale, itemWH * TZScreenScale);
- }
- - (void)configTZScreenWidth {
- TZScreenWidth = [UIScreen mainScreen].bounds.size.width;
- // 测试发现,如果scale在plus真机上取到3.0,内存会增大特别多。故这里写死成2.0
- TZScreenScale = 2.0;
- if (TZScreenWidth > 700) {
- TZScreenScale = 1.5;
- }
- }
- /// Return YES if Authorized 返回YES如果得到了授权
- - (BOOL)authorizationStatusAuthorized {
- if (self.isPreviewNetworkImage) {
- return YES;
- }
- NSInteger status = [PHPhotoLibrary authorizationStatus];
- if (status == 0) {
- /**
- * 当某些情况下AuthorizationStatus == AuthorizationStatusNotDetermined时,无法弹出系统首次使用的授权alertView,系统应用设置里亦没有相册的设置,此时将无法使用,故作以下操作,弹出系统首次使用的授权alertView
- */
- [self requestAuthorizationWithCompletion:nil];
- }
-
- return status == 3;
- }
- - (void)requestAuthorizationWithCompletion:(void (^)(void))completion {
- void (^callCompletionBlock)(void) = ^(){
- dispatch_async(dispatch_get_main_queue(), ^{
- if (completion) {
- completion();
- }
- });
- };
-
- dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
- [PHPhotoLibrary requestAuthorization:^(PHAuthorizationStatus status) {
- callCompletionBlock();
- }];
- });
- }
- #pragma mark - Get Album
- /// Get Album 获得相册/相册数组
- - (void)getCameraRollAlbum:(BOOL)allowPickingVideo allowPickingImage:(BOOL)allowPickingImage needFetchAssets:(BOOL)needFetchAssets completion:(void (^)(TZAlbumModel *model))completion {
- __block TZAlbumModel *model;
- PHFetchOptions *option = [[PHFetchOptions alloc] init];
- if (!allowPickingVideo) option.predicate = [NSPredicate predicateWithFormat:@"mediaType == %ld", PHAssetMediaTypeImage];
- if (!allowPickingImage) option.predicate = [NSPredicate predicateWithFormat:@"mediaType == %ld",
- PHAssetMediaTypeVideo];
- // option.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"modificationDate" ascending:self.sortAscendingByModificationDate]];
- if (!self.sortAscendingByModificationDate) {
- option.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"creationDate" ascending:self.sortAscendingByModificationDate]];
- }
- PHFetchResult *smartAlbums = [PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeSmartAlbum subtype:PHAssetCollectionSubtypeAlbumRegular options:nil];
- for (PHAssetCollection *collection in smartAlbums) {
- // 有可能是PHCollectionList类的的对象,过滤掉
- if (![collection isKindOfClass:[PHAssetCollection class]]) continue;
- // 过滤空相册
- if (collection.estimatedAssetCount <= 0) continue;
- if ([self isCameraRollAlbum:collection]) {
- PHFetchResult *fetchResult = [PHAsset fetchAssetsInAssetCollection:collection options:option];
- model = [self modelWithResult:fetchResult name:collection.localizedTitle isCameraRoll:YES needFetchAssets:needFetchAssets];
- if (completion) completion(model);
- break;
- }
- }
- }
- - (void)getAllAlbums:(BOOL)allowPickingVideo allowPickingImage:(BOOL)allowPickingImage needFetchAssets:(BOOL)needFetchAssets completion:(void (^)(NSArray<TZAlbumModel *> *))completion{
- NSMutableArray *albumArr = [NSMutableArray array];
- PHFetchOptions *option = [[PHFetchOptions alloc] init];
- if (!allowPickingVideo) option.predicate = [NSPredicate predicateWithFormat:@"mediaType == %ld", PHAssetMediaTypeImage];
- if (!allowPickingImage) option.predicate = [NSPredicate predicateWithFormat:@"mediaType == %ld",
- PHAssetMediaTypeVideo];
- // option.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"modificationDate" ascending:self.sortAscendingByModificationDate]];
- if (!self.sortAscendingByModificationDate) {
- option.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"creationDate" ascending:self.sortAscendingByModificationDate]];
- }
- // 我的照片流 1.6.10重新加入..
- PHFetchResult *myPhotoStreamAlbum = [PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeAlbum subtype:PHAssetCollectionSubtypeAlbumMyPhotoStream options:nil];
- PHFetchResult *smartAlbums = [PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeSmartAlbum subtype:PHAssetCollectionSubtypeAlbumRegular options:nil];
- PHFetchResult *topLevelUserCollections = [PHCollectionList fetchTopLevelUserCollectionsWithOptions:nil];
- PHFetchResult *syncedAlbums = [PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeAlbum subtype:PHAssetCollectionSubtypeAlbumSyncedAlbum options:nil];
- PHFetchResult *sharedAlbums = [PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeAlbum subtype:PHAssetCollectionSubtypeAlbumCloudShared options:nil];
- NSArray *allAlbums = @[myPhotoStreamAlbum,smartAlbums,topLevelUserCollections,syncedAlbums,sharedAlbums];
- for (PHFetchResult *fetchResult in allAlbums) {
- for (PHAssetCollection *collection in fetchResult) {
- // 有可能是PHCollectionList类的的对象,过滤掉
- if (![collection isKindOfClass:[PHAssetCollection class]]) continue;
- // 过滤空相册
- if (collection.estimatedAssetCount <= 0 && ![self isCameraRollAlbum:collection]) continue;
- PHFetchResult *fetchResult = [PHAsset fetchAssetsInAssetCollection:collection options:option];
- if (fetchResult.count < 1 && ![self isCameraRollAlbum:collection]) continue;
-
- if ([self.pickerDelegate respondsToSelector:@selector(isAlbumCanSelect:result:)]) {
- if (![self.pickerDelegate isAlbumCanSelect:collection.localizedTitle result:fetchResult]) {
- continue;
- }
- }
-
- if (collection.assetCollectionSubtype == PHAssetCollectionSubtypeSmartAlbumAllHidden) continue;
- if (collection.assetCollectionSubtype == 1000000201) continue; //『最近删除』相册
- if ([self isCameraRollAlbum:collection]) {
- [albumArr insertObject:[self modelWithResult:fetchResult name:collection.localizedTitle isCameraRoll:YES needFetchAssets:needFetchAssets] atIndex:0];
- } else {
- [albumArr addObject:[self modelWithResult:fetchResult name:collection.localizedTitle isCameraRoll:NO needFetchAssets:needFetchAssets]];
- }
- }
- }
- if (completion) {
- completion(albumArr);
- }
- }
- #pragma mark - Get Assets
- /// Get Assets 获得照片数组
- - (void)getAssetsFromFetchResult:(PHFetchResult *)result completion:(void (^)(NSArray<TZAssetModel *> *))completion {
- TZImagePickerConfig *config = [TZImagePickerConfig sharedInstance];
- return [self getAssetsFromFetchResult:result allowPickingVideo:config.allowPickingVideo allowPickingImage:config.allowPickingImage completion:completion];
- }
- - (void)getAssetsFromFetchResult:(PHFetchResult *)result allowPickingVideo:(BOOL)allowPickingVideo allowPickingImage:(BOOL)allowPickingImage completion:(void (^)(NSArray<TZAssetModel *> *))completion {
- NSMutableArray *photoArr = [NSMutableArray array];
- [result enumerateObjectsUsingBlock:^(PHAsset *asset, NSUInteger idx, BOOL * _Nonnull stop) {
- TZAssetModel *model = [self assetModelWithAsset:asset allowPickingVideo:allowPickingVideo allowPickingImage:allowPickingImage];
- if (model) {
- [photoArr addObject:model];
- }
- }];
- if (completion) completion(photoArr);
- }
- /// Get asset at index 获得下标为index的单个照片
- /// if index beyond bounds, return nil in callback 如果索引越界, 在回调中返回 nil
- - (void)getAssetFromFetchResult:(PHFetchResult *)result atIndex:(NSInteger)index allowPickingVideo:(BOOL)allowPickingVideo allowPickingImage:(BOOL)allowPickingImage completion:(void (^)(TZAssetModel *))completion {
- PHAsset *asset;
- @try {
- asset = result[index];
- }
- @catch (NSException* e) {
- if (completion) completion(nil);
- return;
- }
- TZAssetModel *model = [self assetModelWithAsset:asset allowPickingVideo:allowPickingVideo allowPickingImage:allowPickingImage];
- if (completion) completion(model);
- }
- - (TZAssetModel *)assetModelWithAsset:(PHAsset *)asset allowPickingVideo:(BOOL)allowPickingVideo allowPickingImage:(BOOL)allowPickingImage {
- BOOL canSelect = YES;
- if ([self.pickerDelegate respondsToSelector:@selector(isAssetCanSelect:)]) {
- canSelect = [self.pickerDelegate isAssetCanSelect:asset];
- }
- if (!canSelect) return nil;
-
- TZAssetModel *model;
- TZAssetModelMediaType type = [self getAssetType:asset];
- if (!allowPickingVideo && type == TZAssetModelMediaTypeVideo) return nil;
- if (!allowPickingImage && type == TZAssetModelMediaTypePhoto) return nil;
- if (!allowPickingImage && type == TZAssetModelMediaTypePhotoGif) return nil;
-
- PHAsset *phAsset = (PHAsset *)asset;
- if (self.hideWhenCanNotSelect) {
- // 过滤掉尺寸不满足要求的图片
- if (![self isPhotoSelectableWithAsset:phAsset]) {
- return nil;
- }
- }
- NSString *timeLength = type == TZAssetModelMediaTypeVideo ? [NSString stringWithFormat:@"%0.0f",phAsset.duration] : @"";
- timeLength = [self getNewTimeFromDurationSecond:timeLength.integerValue];
- model = [TZAssetModel modelWithAsset:asset type:type timeLength:timeLength];
- return model;
- }
- - (TZAssetModelMediaType)getAssetType:(PHAsset *)asset {
- TZAssetModelMediaType type = TZAssetModelMediaTypePhoto;
- PHAsset *phAsset = (PHAsset *)asset;
- if (phAsset.mediaType == PHAssetMediaTypeVideo) type = TZAssetModelMediaTypeVideo;
- else if (phAsset.mediaType == PHAssetMediaTypeAudio) type = TZAssetModelMediaTypeAudio;
- else if (phAsset.mediaType == PHAssetMediaTypeImage) {
- if (@available(iOS 9.1, *)) {
- // if (asset.mediaSubtypes == PHAssetMediaSubtypePhotoLive) type = TZAssetModelMediaTypeLivePhoto;
- }
- // Gif
- if ([[phAsset valueForKey:@"filename"] hasSuffix:@"GIF"]) {
- type = TZAssetModelMediaTypePhotoGif;
- }
- }
- return type;
- }
- - (NSString *)getNewTimeFromDurationSecond:(NSInteger)duration {
- NSString *newTime;
- if (duration < 10) {
- newTime = [NSString stringWithFormat:@"0:0%zd",duration];
- } else if (duration < 60) {
- newTime = [NSString stringWithFormat:@"0:%zd",duration];
- } else {
- NSInteger min = duration / 60;
- NSInteger sec = duration - (min * 60);
- if (sec < 10) {
- newTime = [NSString stringWithFormat:@"%zd:0%zd",min,sec];
- } else {
- newTime = [NSString stringWithFormat:@"%zd:%zd",min,sec];
- }
- }
- return newTime;
- }
- /// Get photo bytes 获得一组照片的大小
- - (void)getPhotosBytesWithArray:(NSArray *)photos completion:(void (^)(NSString *totalBytes))completion {
- if (!photos || !photos.count) {
- if (completion) completion(@"0B");
- return;
- }
- __block NSInteger dataLength = 0;
- __block NSInteger assetCount = 0;
- for (NSInteger i = 0; i < photos.count; i++) {
- TZAssetModel *model = photos[i];
- PHImageRequestOptions *options = [[PHImageRequestOptions alloc] init];
- options.resizeMode = PHImageRequestOptionsResizeModeFast;
- options.networkAccessAllowed = YES;
- if (model.type == TZAssetModelMediaTypePhotoGif) {
- options.version = PHImageRequestOptionsVersionOriginal;
- }
- [[PHImageManager defaultManager] requestImageDataForAsset:model.asset options:options resultHandler:^(NSData *imageData, NSString *dataUTI, UIImageOrientation orientation, NSDictionary *info) {
- if (model.type != TZAssetModelMediaTypeVideo) dataLength += imageData.length;
- assetCount ++;
- if (assetCount >= photos.count) {
- NSString *bytes = [self getBytesFromDataLength:dataLength];
- if (completion) completion(bytes);
- }
- }];
- }
- }
- - (NSString *)getBytesFromDataLength:(NSInteger)dataLength {
- NSString *bytes;
- if (dataLength >= 0.1 * (1024 * 1024)) {
- bytes = [NSString stringWithFormat:@"%0.1fM",dataLength/1024/1024.0];
- } else if (dataLength >= 1024) {
- bytes = [NSString stringWithFormat:@"%0.0fK",dataLength/1024.0];
- } else {
- bytes = [NSString stringWithFormat:@"%zdB",dataLength];
- }
- return bytes;
- }
- #pragma mark - Get Photo
- /// Get photo 获得照片本身
- - (PHImageRequestID)getPhotoWithAsset:(PHAsset *)asset completion:(void (^)(UIImage *, NSDictionary *, BOOL isDegraded))completion {
- CGFloat fullScreenWidth = TZScreenWidth;
- if (fullScreenWidth > _photoPreviewMaxWidth) {
- fullScreenWidth = _photoPreviewMaxWidth;
- }
- return [self getPhotoWithAsset:asset photoWidth:fullScreenWidth completion:completion progressHandler:nil networkAccessAllowed:YES];
- }
- - (PHImageRequestID)getPhotoWithAsset:(PHAsset *)asset photoWidth:(CGFloat)photoWidth completion:(void (^)(UIImage *photo,NSDictionary *info,BOOL isDegraded))completion {
- return [self getPhotoWithAsset:asset photoWidth:photoWidth completion:completion progressHandler:nil networkAccessAllowed:YES];
- }
- - (PHImageRequestID)getPhotoWithAsset:(PHAsset *)asset completion:(void (^)(UIImage *photo,NSDictionary *info,BOOL isDegraded))completion progressHandler:(void (^)(double progress, NSError *error, BOOL *stop, NSDictionary *info))progressHandler networkAccessAllowed:(BOOL)networkAccessAllowed {
- CGFloat fullScreenWidth = TZScreenWidth;
- if (_photoPreviewMaxWidth > 0 && fullScreenWidth > _photoPreviewMaxWidth) {
- fullScreenWidth = _photoPreviewMaxWidth;
- }
- return [self getPhotoWithAsset:asset photoWidth:fullScreenWidth completion:completion progressHandler:progressHandler networkAccessAllowed:networkAccessAllowed];
- }
- - (PHImageRequestID)requestImageDataForAsset:(PHAsset *)asset completion:(void (^)(NSData *imageData, NSString *dataUTI, UIImageOrientation orientation, NSDictionary *info))completion progressHandler:(void (^)(double progress, NSError *error, BOOL *stop, NSDictionary *info))progressHandler {
- PHImageRequestOptions *options = [[PHImageRequestOptions alloc] init];
- options.progressHandler = ^(double progress, NSError *error, BOOL *stop, NSDictionary *info) {
- dispatch_async(dispatch_get_main_queue(), ^{
- if (progressHandler) {
- progressHandler(progress, error, stop, info);
- }
- });
- };
- options.networkAccessAllowed = YES;
- options.resizeMode = PHImageRequestOptionsResizeModeFast;
- int32_t imageRequestID = [[PHImageManager defaultManager] requestImageDataForAsset:asset options:options resultHandler:^(NSData *imageData, NSString *dataUTI, UIImageOrientation orientation, NSDictionary *info) {
- if (completion) completion(imageData,dataUTI,orientation,info);
- }];
- return imageRequestID;
- }
- - (PHImageRequestID)getPhotoWithAsset:(PHAsset *)asset photoWidth:(CGFloat)photoWidth completion:(void (^)(UIImage *photo,NSDictionary *info,BOOL isDegraded))completion progressHandler:(void (^)(double progress, NSError *error, BOOL *stop, NSDictionary *info))progressHandler networkAccessAllowed:(BOOL)networkAccessAllowed {
-
- if (asset == nil) {
- return -1;
- }
-
-
- CGSize imageSize;
- if (photoWidth < TZScreenWidth && photoWidth < _photoPreviewMaxWidth) {
- imageSize = AssetGridThumbnailSize;
- } else {
- PHAsset *phAsset = (PHAsset *)asset;
- CGFloat aspectRatio = phAsset.pixelWidth / (CGFloat)phAsset.pixelHeight;
- CGFloat pixelWidth = photoWidth * TZScreenScale;
- // 超宽图片
- if (aspectRatio > 1.8) {
- pixelWidth = pixelWidth * aspectRatio;
- }
- // 超高图片
- if (aspectRatio < 0.2) {
- pixelWidth = pixelWidth * 0.5;
- }
- CGFloat pixelHeight = pixelWidth / aspectRatio;
- imageSize = CGSizeMake(pixelWidth, pixelHeight);
- }
-
- __block UIImage *image;
- // 修复获取图片时出现的瞬间内存过高问题
- // 下面两行代码,来自hsjcom,他的github是:https://github.com/hsjcom 表示感谢
- PHImageRequestOptions *option = [[PHImageRequestOptions alloc] init];
- option.resizeMode = PHImageRequestOptionsResizeModeFast;
- int32_t imageRequestID = [[PHImageManager defaultManager] requestImageForAsset:asset targetSize:imageSize contentMode:PHImageContentModeAspectFill options:option resultHandler:^(UIImage *result, NSDictionary *info) {
- if (result) {
- image = result;
- }
- BOOL downloadFinined = (![[info objectForKey:PHImageCancelledKey] boolValue] && ![info objectForKey:PHImageErrorKey]);
- if (downloadFinined && result) {
- result = [self fixOrientation:result];
- if (completion) completion(result,info,[[info objectForKey:PHImageResultIsDegradedKey] boolValue]);
- }
- // Download image from iCloud / 从iCloud下载图片
- if ([info objectForKey:PHImageResultIsInCloudKey] && !result && networkAccessAllowed) {
- PHImageRequestOptions *options = [[PHImageRequestOptions alloc] init];
- options.progressHandler = ^(double progress, NSError *error, BOOL *stop, NSDictionary *info) {
- dispatch_async(dispatch_get_main_queue(), ^{
- if (progressHandler) {
- progressHandler(progress, error, stop, info);
- }
- });
- };
- options.networkAccessAllowed = YES;
- options.resizeMode = PHImageRequestOptionsResizeModeFast;
- [[PHImageManager defaultManager] requestImageDataForAsset:asset options:options resultHandler:^(NSData *imageData, NSString *dataUTI, UIImageOrientation orientation, NSDictionary *info) {
- UIImage *resultImage = [UIImage imageWithData:imageData];
- if (![TZImagePickerConfig sharedInstance].notScaleImage) {
- resultImage = [self scaleImage:resultImage toSize:imageSize];
- }
- if (!resultImage) {
- resultImage = image;
- }
- resultImage = [self fixOrientation:resultImage];
- if (completion) completion(resultImage,info,NO);
- }];
- }
- }];
- return imageRequestID;
- }
- /// Get postImage / 获取封面图
- - (PHImageRequestID)getPostImageWithAlbumModel:(TZAlbumModel *)model completion:(void (^)(UIImage *))completion {
- id asset = [model.result lastObject];
- if (!self.sortAscendingByModificationDate) {
- asset = [model.result firstObject];
- }
- return [[TZImageManager manager] getPhotoWithAsset:asset photoWidth:80 completion:^(UIImage *photo, NSDictionary *info, BOOL isDegraded) {
- if (completion) completion(photo);
- }];
- }
- /// Get Original Photo / 获取原图
- - (PHImageRequestID)getOriginalPhotoWithAsset:(PHAsset *)asset completion:(void (^)(UIImage *photo,NSDictionary *info))completion {
- return [self getOriginalPhotoWithAsset:asset newCompletion:^(UIImage *photo, NSDictionary *info, BOOL isDegraded) {
- if (completion) {
- completion(photo,info);
- }
- }];
- }
- - (PHImageRequestID)getOriginalPhotoWithAsset:(PHAsset *)asset newCompletion:(void (^)(UIImage *photo,NSDictionary *info,BOOL isDegraded))completion {
- return [self getOriginalPhotoWithAsset:asset progressHandler:nil newCompletion:completion];
- }
- - (PHImageRequestID)getOriginalPhotoWithAsset:(PHAsset *)asset progressHandler:(void (^)(double progress, NSError *error, BOOL *stop, NSDictionary *info))progressHandler newCompletion:(void (^)(UIImage *photo,NSDictionary *info,BOOL isDegraded))completion {
- return [self getOriginalPhotoWithAsset:asset targetSize:PHImageManagerMaximumSize contentMode:PHImageContentModeAspectFit progressHandler:progressHandler newCompletion:completion];
- }
- - (PHImageRequestID)getOriginalPhotoWithAsset:(PHAsset *)asset targetSize:(CGSize)targetSize contentMode:(PHImageContentMode)mode progressHandler:(void (^)(double progress, NSError *error, BOOL *stop, NSDictionary *info))progressHandler newCompletion:(void (^)(UIImage *photo,NSDictionary *info,BOOL isDegraded))completion {
- PHImageRequestOptions *option = [[PHImageRequestOptions alloc]init];
- option.networkAccessAllowed = YES;
- if (progressHandler) {
- [option setProgressHandler:progressHandler];
- }
- option.resizeMode = PHImageRequestOptionsResizeModeFast;
- return [[PHImageManager defaultManager] requestImageForAsset:asset targetSize:targetSize contentMode:mode options:option resultHandler:^(UIImage *result, NSDictionary *info) {
- BOOL downloadFinined = (![[info objectForKey:PHImageCancelledKey] boolValue] && ![info objectForKey:PHImageErrorKey]);
- if (downloadFinined && result) {
- result = [self fixOrientation:result];
- BOOL isDegraded = [[info objectForKey:PHImageResultIsDegradedKey] boolValue];
- if (completion) completion(result,info,isDegraded);
- }
- }];
- }
- - (PHImageRequestID)getOriginalPhotoDataWithAsset:(PHAsset *)asset completion:(void (^)(NSData *data,NSDictionary *info,BOOL isDegraded))completion {
- return [self getOriginalPhotoDataWithAsset:asset progressHandler:nil completion:completion];
- }
- - (PHImageRequestID)getOriginalPhotoDataWithAsset:(PHAsset *)asset progressHandler:(void (^)(double progress, NSError *error, BOOL *stop, NSDictionary *info))progressHandler completion:(void (^)(NSData *data,NSDictionary *info,BOOL isDegraded))completion {
- PHImageRequestOptions *option = [[PHImageRequestOptions alloc] init];
- option.networkAccessAllowed = YES;
- if ([[asset valueForKey:@"filename"] hasSuffix:@"GIF"]) {
- // if version isn't PHImageRequestOptionsVersionOriginal, the gif may cann't play
- option.version = PHImageRequestOptionsVersionOriginal;
- }
- [option setProgressHandler:progressHandler];
- option.deliveryMode = PHImageRequestOptionsDeliveryModeHighQualityFormat;
- return [[PHImageManager defaultManager] requestImageDataForAsset:asset options:option resultHandler:^(NSData *imageData, NSString *dataUTI, UIImageOrientation orientation, NSDictionary *info) {
- BOOL downloadFinined = (![[info objectForKey:PHImageCancelledKey] boolValue] && ![info objectForKey:PHImageErrorKey]);
- if (downloadFinined && imageData) {
- if (completion) completion(imageData,info,NO);
- }
- }];
- }
- #pragma mark - Save photo
- - (void)savePhotoWithImage:(UIImage *)image completion:(void (^)(PHAsset *asset, NSError *error))completion {
- [self savePhotoWithImage:image location:nil completion:completion];
- }
- - (void)savePhotoWithImage:(UIImage *)image location:(CLLocation *)location completion:(void (^)(PHAsset *asset, NSError *error))completion {
- __block NSString *localIdentifier = nil;
- [[PHPhotoLibrary sharedPhotoLibrary] performChanges:^{
- PHAssetChangeRequest *request = [PHAssetChangeRequest creationRequestForAssetFromImage:image];
- localIdentifier = request.placeholderForCreatedAsset.localIdentifier;
- if (location) {
- request.location = location;
- }
- request.creationDate = [NSDate date];
- } completionHandler:^(BOOL success, NSError *error) {
- dispatch_async(dispatch_get_main_queue(), ^{
- if (success && completion) {
- PHAsset *asset = [[PHAsset fetchAssetsWithLocalIdentifiers:@[localIdentifier] options:nil] firstObject];
- completion(asset, nil);
- } else if (error) {
- NSLog(@"保存照片出错:%@",error.localizedDescription);
- if (completion) {
- completion(nil, error);
- }
- }
- });
- }];
- }
- #pragma mark - Save video
- - (void)saveVideoWithUrl:(NSURL *)url completion:(void (^)(PHAsset *asset, NSError *error))completion {
- [self saveVideoWithUrl:url location:nil completion:completion];
- }
- - (void)saveVideoWithUrl:(NSURL *)url location:(CLLocation *)location completion:(void (^)(PHAsset *asset, NSError *error))completion {
- __block NSString *localIdentifier = nil;
- [[PHPhotoLibrary sharedPhotoLibrary] performChanges:^{
- PHAssetChangeRequest *request = [PHAssetChangeRequest creationRequestForAssetFromVideoAtFileURL:url];
- localIdentifier = request.placeholderForCreatedAsset.localIdentifier;
- if (location) {
- request.location = location;
- }
- request.creationDate = [NSDate date];
- } completionHandler:^(BOOL success, NSError *error) {
- dispatch_async(dispatch_get_main_queue(), ^{
- if (success && completion) {
- PHAsset *asset = [[PHAsset fetchAssetsWithLocalIdentifiers:@[localIdentifier] options:nil] firstObject];
- completion(asset, nil);
- } else if (error) {
- NSLog(@"保存视频出错:%@",error.localizedDescription);
- if (completion) {
- completion(nil, error);
- }
- }
- });
- }];
- }
- #pragma mark - Get Video
- /// Get Video / 获取视频
- - (void)getVideoWithAsset:(PHAsset *)asset completion:(void (^)(AVPlayerItem *, NSDictionary *))completion {
- [self getVideoWithAsset:asset progressHandler:nil completion:completion];
- }
- - (void)getVideoWithAsset:(PHAsset *)asset progressHandler:(void (^)(double progress, NSError *error, BOOL *stop, NSDictionary *info))progressHandler completion:(void (^)(AVPlayerItem *, NSDictionary *))completion {
- PHVideoRequestOptions *option = [[PHVideoRequestOptions alloc] init];
- option.networkAccessAllowed = YES;
- option.progressHandler = ^(double progress, NSError *error, BOOL *stop, NSDictionary *info) {
- dispatch_async(dispatch_get_main_queue(), ^{
- if (progressHandler) {
- progressHandler(progress, error, stop, info);
- }
- });
- };
- [[PHImageManager defaultManager] requestPlayerItemForVideo:asset options:option resultHandler:^(AVPlayerItem *playerItem, NSDictionary *info) {
- if (completion) completion(playerItem,info);
- }];
- }
- - (void)requestVideoWithAsset:(PHAsset *)asset
- startHandler:(void (^)(PHImageRequestID))startHandler
- progressHandler:(void (^)(double progress, NSError *error, BOOL *stop))progressHandler
- completion:(void (^)(AVAsset *))completion {
- PHVideoRequestOptions *option = [[PHVideoRequestOptions alloc] init];
- option.networkAccessAllowed = YES; // // 允许下载iCloud资源
- option.version = PHVideoRequestOptionsVersionOriginal;
- option.deliveryMode = PHVideoRequestOptionsDeliveryModeAutomatic;
- option.progressHandler = ^(double progress, NSError *error, BOOL *stop, NSDictionary *info) {
- dispatch_async(dispatch_get_main_queue(), ^{
- if (progressHandler) {
- progressHandler(progress, error, stop);
- }
- });
- };
- PHImageRequestID requestID = [[PHImageManager defaultManager] requestAVAssetForVideo:asset options:option resultHandler:^(AVAsset *avasset, AVAudioMix *audioMix, NSDictionary *info) {
- if (completion) {
- completion(avasset);
- }
- }];
- if (startHandler) {
- startHandler(requestID);
- }
- }
- #pragma mark - Export video
- /// Export Video / 导出视频
- - (void)getVideoOutputPathWithAsset:(PHAsset *)asset success:(void (^)(NSString *outputPath))success failure:(void (^)(NSString *errorMessage, NSError *error))failure {
- [self getVideoOutputPathWithAsset:asset presetName:AVAssetExportPreset640x480 success:success failure:failure];
- }
- - (void)getVideoOutputPathWithAsset:(PHAsset *)asset presetName:(NSString *)presetName success:(void (^)(NSString *outputPath))success failure:(void (^)(NSString *errorMessage, NSError *error))failure {
- PHVideoRequestOptions* options = [[PHVideoRequestOptions alloc] init];
- options.version = PHVideoRequestOptionsVersionOriginal;
- options.deliveryMode = PHVideoRequestOptionsDeliveryModeAutomatic;
- options.networkAccessAllowed = YES;
- [[PHImageManager defaultManager] requestAVAssetForVideo:asset options:options resultHandler:^(AVAsset* avasset, AVAudioMix* audioMix, NSDictionary* info){
- // NSLog(@"Info:\n%@",info);
- AVURLAsset *videoAsset = (AVURLAsset*)avasset;
- // NSLog(@"AVAsset URL: %@",myAsset.URL);
- [self startExportVideoWithVideoAsset:videoAsset presetName:presetName success:success failure:failure];
- }];
- }
- /// Deprecated, Use -getVideoOutputPathWithAsset:failure:success:
- - (void)getVideoOutputPathWithAsset:(PHAsset *)asset completion:(void (^)(NSString *outputPath))completion {
- [self getVideoOutputPathWithAsset:asset success:completion failure:nil];
- }
- - (void)startExportVideoWithVideoAsset:(AVURLAsset *)videoAsset presetName:(NSString *)presetName success:(void (^)(NSString *outputPath))success failure:(void (^)(NSString *errorMessage, NSError *error))failure {
- // Find compatible presets by video asset.
- NSArray *presets = [AVAssetExportSession exportPresetsCompatibleWithAsset:videoAsset];
-
- // Begin to compress video
- // Now we just compress to low resolution if it supports
- // If you need to upload to the server, but server does't support to upload by streaming,
- // You can compress the resolution to lower. Or you can support more higher resolution.
- if ([presets containsObject:presetName]) {
- AVAssetExportSession *session = [[AVAssetExportSession alloc] initWithAsset:videoAsset presetName:presetName];
- NSDateFormatter *formater = [[NSDateFormatter alloc] init];
- [formater setDateFormat:@"yyyy-MM-dd-HH:mm:ss-SSS"];
- NSString *outputPath = [NSHomeDirectory() stringByAppendingFormat:@"/tmp/video-%@.mp4", [formater stringFromDate:[NSDate date]]];
-
- // Optimize for network use.
- session.shouldOptimizeForNetworkUse = true;
-
- NSArray *supportedTypeArray = session.supportedFileTypes;
- if ([supportedTypeArray containsObject:AVFileTypeMPEG4]) {
- session.outputFileType = AVFileTypeMPEG4;
- } else if (supportedTypeArray.count == 0) {
- if (failure) {
- failure(@"该视频类型暂不支持导出", nil);
- }
- NSLog(@"No supported file types 视频类型暂不支持导出");
- return;
- } else {
- session.outputFileType = [supportedTypeArray objectAtIndex:0];
- if (videoAsset.URL && videoAsset.URL.lastPathComponent) {
- outputPath = [outputPath stringByReplacingOccurrencesOfString:@".mp4" withString:[NSString stringWithFormat:@"-%@", videoAsset.URL.lastPathComponent]];
- }
- }
- // NSLog(@"video outputPath = %@",outputPath);
- session.outputURL = [NSURL fileURLWithPath:outputPath];
-
- if (![[NSFileManager defaultManager] fileExistsAtPath:[NSHomeDirectory() stringByAppendingFormat:@"/tmp"]]) {
- [[NSFileManager defaultManager] createDirectoryAtPath:[NSHomeDirectory() stringByAppendingFormat:@"/tmp"] withIntermediateDirectories:YES attributes:nil error:nil];
- }
-
- if ([TZImagePickerConfig sharedInstance].needFixComposition) {
- AVMutableVideoComposition *videoComposition = [self fixedCompositionWithAsset:videoAsset];
- if (videoComposition.renderSize.width) {
- // 修正视频转向
- session.videoComposition = videoComposition;
- }
- }
- // Begin to export video to the output path asynchronously.
- [session exportAsynchronouslyWithCompletionHandler:^(void) {
- dispatch_async(dispatch_get_main_queue(), ^{
- switch (session.status) {
- case AVAssetExportSessionStatusUnknown: {
- NSLog(@"AVAssetExportSessionStatusUnknown");
- } break;
- case AVAssetExportSessionStatusWaiting: {
- NSLog(@"AVAssetExportSessionStatusWaiting");
- } break;
- case AVAssetExportSessionStatusExporting: {
- NSLog(@"AVAssetExportSessionStatusExporting");
- } break;
- case AVAssetExportSessionStatusCompleted: {
- NSLog(@"AVAssetExportSessionStatusCompleted");
- if (success) {
- success(outputPath);
- }
- } break;
- case AVAssetExportSessionStatusFailed: {
- NSLog(@"AVAssetExportSessionStatusFailed");
- if (failure) {
- failure(@"视频导出失败", session.error);
- }
- } break;
- case AVAssetExportSessionStatusCancelled: {
- NSLog(@"AVAssetExportSessionStatusCancelled");
- if (failure) {
- failure(@"导出任务已被取消", nil);
- }
- } break;
- default: break;
- }
- });
- }];
- } else {
- if (failure) {
- NSString *errorMessage = [NSString stringWithFormat:@"当前设备不支持该预设:%@", presetName];
- failure(errorMessage, nil);
- }
- }
- }
- - (BOOL)isCameraRollAlbum:(PHAssetCollection *)metadata {
- NSString *versionStr = [[UIDevice currentDevice].systemVersion stringByReplacingOccurrencesOfString:@"." withString:@""];
- if (versionStr.length <= 1) {
- versionStr = [versionStr stringByAppendingString:@"00"];
- } else if (versionStr.length <= 2) {
- versionStr = [versionStr stringByAppendingString:@"0"];
- }
- CGFloat version = versionStr.floatValue;
- // 目前已知8.0.0 ~ 8.0.2系统,拍照后的图片会保存在最近添加中
- if (version >= 800 && version <= 802) {
- return ((PHAssetCollection *)metadata).assetCollectionSubtype == PHAssetCollectionSubtypeSmartAlbumRecentlyAdded;
- } else {
- return ((PHAssetCollection *)metadata).assetCollectionSubtype == PHAssetCollectionSubtypeSmartAlbumUserLibrary;
- }
- }
- /// 检查照片大小是否满足最小要求
- - (BOOL)isPhotoSelectableWithAsset:(PHAsset *)asset {
- CGSize photoSize = CGSizeMake(asset.pixelWidth, asset.pixelHeight);
- if (self.minPhotoWidthSelectable > photoSize.width || self.minPhotoHeightSelectable > photoSize.height) {
- return NO;
- }
- return YES;
- }
- #pragma mark - Private Method
- - (TZAlbumModel *)modelWithResult:(PHFetchResult *)result name:(NSString *)name isCameraRoll:(BOOL)isCameraRoll needFetchAssets:(BOOL)needFetchAssets {
- TZAlbumModel *model = [[TZAlbumModel alloc] init];
- [model setResult:result needFetchAssets:needFetchAssets];
- model.name = name;
- model.isCameraRoll = isCameraRoll;
- model.count = result.count;
- return model;
- }
- /// 缩放图片至新尺寸
- - (UIImage *)scaleImage:(UIImage *)image toSize:(CGSize)size {
- if (image.size.width > size.width) {
- UIGraphicsBeginImageContext(size);
- [image drawInRect:CGRectMake(0, 0, size.width, size.height)];
- UIImage *newImage = UIGraphicsGetImageFromCurrentImageContext();
- UIGraphicsEndImageContext();
- return newImage;
-
- /* 好像不怎么管用:https://mp.weixin.qq.com/s/CiqMlEIp1Ir2EJSDGgMooQ
- CGFloat maxPixelSize = MAX(size.width, size.height);
- CGImageSourceRef sourceRef = CGImageSourceCreateWithData((__bridge CFDataRef)UIImageJPEGRepresentation(image, 0.9), nil);
- NSDictionary *options = @{(__bridge id)kCGImageSourceCreateThumbnailFromImageAlways:(__bridge id)kCFBooleanTrue,
- (__bridge id)kCGImageSourceThumbnailMaxPixelSize:[NSNumber numberWithFloat:maxPixelSize]
- };
- CGImageRef imageRef = CGImageSourceCreateImageAtIndex(sourceRef, 0, (__bridge CFDictionaryRef)options);
- UIImage *newImage = [UIImage imageWithCGImage:imageRef scale:2 orientation:image.imageOrientation];
- CGImageRelease(imageRef);
- CFRelease(sourceRef);
- return newImage;
- */
- } else {
- return image;
- }
- }
- /// 判断asset是否是视频
- - (BOOL)isVideo:(PHAsset *)asset {
- return asset.mediaType == PHAssetMediaTypeVideo;
- }
- - (TZAssetModel *)createModelWithAsset:(PHAsset *)asset {
- TZAssetModelMediaType type = [[TZImageManager manager] getAssetType:asset];
- NSString *timeLength = type == TZAssetModelMediaTypeVideo ? [NSString stringWithFormat:@"%0.0f",asset.duration] : @"";
- timeLength = [[TZImageManager manager] getNewTimeFromDurationSecond:timeLength.integerValue];
- TZAssetModel *model = [TZAssetModel modelWithAsset:asset type:type timeLength:timeLength];
- return model;
- }
- /// 获取优化后的视频转向信息
- - (AVMutableVideoComposition *)fixedCompositionWithAsset:(AVAsset *)videoAsset {
- AVMutableVideoComposition *videoComposition = [AVMutableVideoComposition videoComposition];
- // 视频转向
- int degrees = [self degressFromVideoFileWithAsset:videoAsset];
- if (degrees != 0) {
- CGAffineTransform translateToCenter;
- CGAffineTransform mixedTransform;
- videoComposition.frameDuration = CMTimeMake(1, 30);
-
- NSArray *tracks = [videoAsset tracksWithMediaType:AVMediaTypeVideo];
- AVAssetTrack *videoTrack = [tracks objectAtIndex:0];
-
- AVMutableVideoCompositionInstruction *roateInstruction = [AVMutableVideoCompositionInstruction videoCompositionInstruction];
- roateInstruction.timeRange = CMTimeRangeMake(kCMTimeZero, [videoAsset duration]);
- AVMutableVideoCompositionLayerInstruction *roateLayerInstruction = [AVMutableVideoCompositionLayerInstruction videoCompositionLayerInstructionWithAssetTrack:videoTrack];
-
- if (degrees == 90) {
- // 顺时针旋转90°
- translateToCenter = CGAffineTransformMakeTranslation(videoTrack.naturalSize.height, 0.0);
- mixedTransform = CGAffineTransformRotate(translateToCenter,M_PI_2);
- videoComposition.renderSize = CGSizeMake(videoTrack.naturalSize.height,videoTrack.naturalSize.width);
- [roateLayerInstruction setTransform:mixedTransform atTime:kCMTimeZero];
- } else if(degrees == 180){
- // 顺时针旋转180°
- translateToCenter = CGAffineTransformMakeTranslation(videoTrack.naturalSize.width, videoTrack.naturalSize.height);
- mixedTransform = CGAffineTransformRotate(translateToCenter,M_PI);
- videoComposition.renderSize = CGSizeMake(videoTrack.naturalSize.width,videoTrack.naturalSize.height);
- [roateLayerInstruction setTransform:mixedTransform atTime:kCMTimeZero];
- } else if(degrees == 270){
- // 顺时针旋转270°
- translateToCenter = CGAffineTransformMakeTranslation(0.0, videoTrack.naturalSize.width);
- mixedTransform = CGAffineTransformRotate(translateToCenter,M_PI_2*3.0);
- videoComposition.renderSize = CGSizeMake(videoTrack.naturalSize.height,videoTrack.naturalSize.width);
- [roateLayerInstruction setTransform:mixedTransform atTime:kCMTimeZero];
- }
-
- roateInstruction.layerInstructions = @[roateLayerInstruction];
- // 加入视频方向信息
- videoComposition.instructions = @[roateInstruction];
- }
- return videoComposition;
- }
- /// 获取视频角度
- - (int)degressFromVideoFileWithAsset:(AVAsset *)asset {
- int degress = 0;
- NSArray *tracks = [asset tracksWithMediaType:AVMediaTypeVideo];
- if([tracks count] > 0) {
- AVAssetTrack *videoTrack = [tracks objectAtIndex:0];
- CGAffineTransform t = videoTrack.preferredTransform;
- if(t.a == 0 && t.b == 1.0 && t.c == -1.0 && t.d == 0){
- // Portrait
- degress = 90;
- } else if(t.a == 0 && t.b == -1.0 && t.c == 1.0 && t.d == 0){
- // PortraitUpsideDown
- degress = 270;
- } else if(t.a == 1.0 && t.b == 0 && t.c == 0 && t.d == 1.0){
- // LandscapeRight
- degress = 0;
- } else if(t.a == -1.0 && t.b == 0 && t.c == 0 && t.d == -1.0){
- // LandscapeLeft
- degress = 180;
- }
- }
- return degress;
- }
- /// 修正图片转向
- - (UIImage *)fixOrientation:(UIImage *)aImage {
- if (!self.shouldFixOrientation) return aImage;
-
- // No-op if the orientation is already correct
- if (aImage.imageOrientation == UIImageOrientationUp)
- return aImage;
-
- // We need to calculate the proper transformation to make the image upright.
- // We do it in 2 steps: Rotate if Left/Right/Down, and then flip if Mirrored.
- CGAffineTransform transform = CGAffineTransformIdentity;
-
- switch (aImage.imageOrientation) {
- case UIImageOrientationDown:
- case UIImageOrientationDownMirrored:
- transform = CGAffineTransformTranslate(transform, aImage.size.width, aImage.size.height);
- transform = CGAffineTransformRotate(transform, M_PI);
- break;
-
- case UIImageOrientationLeft:
- case UIImageOrientationLeftMirrored:
- transform = CGAffineTransformTranslate(transform, aImage.size.width, 0);
- transform = CGAffineTransformRotate(transform, M_PI_2);
- break;
-
- case UIImageOrientationRight:
- case UIImageOrientationRightMirrored:
- transform = CGAffineTransformTranslate(transform, 0, aImage.size.height);
- transform = CGAffineTransformRotate(transform, -M_PI_2);
- break;
- default:
- break;
- }
-
- switch (aImage.imageOrientation) {
- case UIImageOrientationUpMirrored:
- case UIImageOrientationDownMirrored:
- transform = CGAffineTransformTranslate(transform, aImage.size.width, 0);
- transform = CGAffineTransformScale(transform, -1, 1);
- break;
-
- case UIImageOrientationLeftMirrored:
- case UIImageOrientationRightMirrored:
- transform = CGAffineTransformTranslate(transform, aImage.size.height, 0);
- transform = CGAffineTransformScale(transform, -1, 1);
- break;
- default:
- break;
- }
-
- // Now we draw the underlying CGImage into a new context, applying the transform
- // calculated above.
- CGContextRef ctx = CGBitmapContextCreate(NULL, aImage.size.width, aImage.size.height,
- CGImageGetBitsPerComponent(aImage.CGImage), 0,
- CGImageGetColorSpace(aImage.CGImage),
- CGImageGetBitmapInfo(aImage.CGImage));
- CGContextConcatCTM(ctx, transform);
- switch (aImage.imageOrientation) {
- case UIImageOrientationLeft:
- case UIImageOrientationLeftMirrored:
- case UIImageOrientationRight:
- case UIImageOrientationRightMirrored:
- // Grr...
- CGContextDrawImage(ctx, CGRectMake(0,0,aImage.size.height,aImage.size.width), aImage.CGImage);
- break;
-
- default:
- CGContextDrawImage(ctx, CGRectMake(0,0,aImage.size.width,aImage.size.height), aImage.CGImage);
- break;
- }
-
- // And now we just create a new UIImage from the drawing context
- CGImageRef cgimg = CGBitmapContextCreateImage(ctx);
- UIImage *img = [UIImage imageWithCGImage:cgimg];
- CGContextRelease(ctx);
- CGImageRelease(cgimg);
- return img;
- }
- #pragma clang diagnostic pop
- @end
- //@implementation TZSortDescriptor
- //
- //- (id)reversedSortDescriptor {
- // return [NSNumber numberWithBool:![TZImageManager manager].sortAscendingByModificationDate];
- //}
- //
- //@end
|