PhotoKit

PhotoKit的组成

PHAssetCollection:PHCollection的子类,代表一个相册或者一个时刻,或者一个智能相册(例如最近删除,视频列表,收藏等)

PHFetchResult:一系列资源的集合,也可以是相册或者图片的集合,PHAssetCollection的类方法返回的就是PHFetchResult

PHFetchOptions:获取资源时的参数,可以传nil,传nil默认使用系统的

PHAsset:代表照片库中的一个资源,跟ASset类似,通过PHAsset可以获取可保存资源

PHImageManager:用于处理资源的加载,加载资源的过程带有缓存处理,可以通过传入一个 PHImageRequestOptions 控制资源的输出尺寸等规格

PHImageRequestOptions: 如上面所说,控制加载图片时的一系列参数

ios8-photo-kit.png

PHPhotoLibrary

+ (PHPhotoLibrary *)sharedPhotoLibrary      //获得实例
+ (PHAuthorizationStatus)authorizationStatus      //获得当前用户授权情况
返回一个PHAuthorizationStatus类型的枚举:
    PHAuthorizationStatusNotDetermined 用户还未选择           
    PHAuthorizationStatusRestricted 家长模式                                       不允许
    PHAuthorizationStatusDenied 不同意访问          
    PHAuthorizationStatusAuthorized 同意访问
//在第一次获取时,系统提示选择完后的回调方法
+ (void)requestAuthorization:(void(^)(PHAuthorizationStatus status))handler;
handler:该 block 就是用户授权选择完后的回调,可获得用户选择的结果

PHAssetCollection

获得相册的集合资源
+ (PHFetchResult<PHAssetCollection *> *)fetchAssetCollectionsWithType:(PHAssetCollectionType)type subtype:(PHAssetCollectionSubtype)subtype options:(nullable PHFetchOptions *)options;
返回一个相册集合资源,返回的是PHFetchResult对象集合,集合里面是PHAssetCollection类型的对象.
type:相册类型, PHAssetCollectionType类型的枚举,
subtype:子类型, PHAssetCollectionSubtype的枚举,
PHFetchOptions: PHFetchOptions的一个实例,可以为空,主要是为了对获得资源做一些配置和排序等.可以为 nil.
PHAssetCollectionType:
    PHAssetCollectionTypeAlbum     //相册
    PHAssetCollectionTypeSmartAlbum  //智能相册
    PHAssetCollectionTypeMoment    //时刻
PHAssetCollectionSubtype:
常规的子类型
    PHAssetCollectionSubtypeAlbumRegular    常规的     
    PHAssetCollectionSubtypeAlbumSyncedEvent    使用 iTunes 同步操作过来的相册 
    PHAssetCollectionSubtypeAlbumSyncedFaces    使用 iTuens同步操作过来的人物相册
    PHAssetCollectionSubtypeAlbumSyncedAlbum     使用iTunes 同步的所有相册
    PHAssetCollectionSubtypeAlbumImported        从外界导入的相册

经分享的子类型
    PHAssetCollectionSubtypeAlbumMyPhotoStream   从相册分享得到
    PHAssetCollectionSubtypeAlbumCloudShared     从 cloud 分享得到
智能相册子类型
    PHAssetCollectionSubtypeSmartAlbumGeneric    通用的
    PHAssetCollectionSubtypeSmartAlbumPanoramas  全景
    PHAssetCollectionSubtypeSmartAlbumVideos     视屏
    PHAssetCollectionSubtypeSmartAlbumFavorites  收藏
    PHAssetCollectionSubtypeSmartAlbumTimelapses 延时视屏,也会在PHAssetCollectionSubtypeSmartAlbumVideos在出现
    PHAssetCollectionSubtypeSmartAlbumAllHidden  隐藏的
    PHAssetCollectionSubtypeSmartAlbumRecentlyAdded 最近添加
    PHAssetCollectionSubtypeSmartAlbumBursts    连拍 
    PHAssetCollectionSubtypeSmartAlbumSlomoVideos Slomo是slow motion的缩写,高速摄影慢动作解析
    PHAssetCollectionSubtypeSmartAlbumUserLibrary 用户所有的资源
    PHAssetCollectionSubtypeSmartAlbumSelfPortraits 所有前置摄像头拍的照片和视屏
    PHAssetCollectionSubtypeSmartAlbumScreenshots 所有的截屏图
不关心子类型时的全部资源
    PHAssetCollectionSubtypeAny = NSIntegerMax

PHFetchResult

count   相册中图片的数量
//遍历相册资源中每个相册组的信息
- (void)enumerateObjectsUsingBlock:(void (^)(ObjectType obj, NSUInteger idx, BOOL *stop))block;

PHFetchOptions

对使用 PHAsset, PHCollection, PHAssetCollection, 和 PHCollectionLis 的方法时出入的参数,主要对获取到资源做一些配置和排序等,一般为 nil, 默认使用系统的.

sortDescriptors   排序
includeHiddenAssets      是否显示隐藏的相册,默认不显示
includeAssetSourceTypes;     //获取到相册的类型
PHAssetSourceType类型的枚举,默认PHAssetSourceTypeNone
    PHAssetSourceTypeNone   都没有,就获得到就是常规的         
    PHAssetSourceTypeUserLibrary     用户所有的
    PHAssetSourceTypeCloudShared     分享的    
    PHAssetSourceTypeiTunesSynced    iTunes 同步的

PHAsset

PHAsset:每个照片的详细信息

mediaType:资源的类型
//遍历获得PHAsset的集合
+ (PHFetchResult<PHAsset *> *)fetchAssetsInAssetCollection:(PHAssetCollection *)assetCollection options:(nullable PHFetchOptions *)options;

PHImageManager

PHImageManager:管理PHAsset的一个类,对资源进行管理和筛选

+ (PHImageManager *)defaultManager;   //获得该实例.
//经删选和限制条件获得具体的资源UIImage.
- (PHImageRequestID)requestImageForAsset:(PHAsset *)asset 
                              targetSize:(CGSize)targetSize 
                             contentMode:(PHImageContentMode)contentMode
                                 options:(nullable PHImageRequestOptions *)options resultHandler:(void (^)(UIImage *__nullable result, NSDictionary *__nullable info))resultHandler;
返回值: PHImageRequestID,是个常量,定义为:static const PHImageRequestID PHInvalidImageRequestID = 0;

asset:想要获得信息的PHAsset的对象,

targetSize:获得图片的尺寸大小,这里的大小是pixel(即像素)所以换算需要用自己自己想要的尺寸乘以[UIScreen mainScreen].scale   
如果想要原图的尺寸,直接传入PHImageManagerMaximumSize.很大很大的尺寸,系统会默认返回原图的尺寸,要注意的是传入PHImageManagerMaximumSize时,则 contentMode 无论传入什么值都会被视为PHImageContentModeDefault.,另外这个尺寸只是理想的,其实返回的图片不是这个尺寸,需要进行处理,对图片尽行裁剪,变成我们需要的尺寸。

contentMode:想要图片的裁剪方式, PHImageContentMode的枚举:
    PHImageContentModeAspectFit  适合的
    PHImageContentModeAspectFill 铺满的
    PHImageContentModeDefault = PHImageContentModeAspectFit

options: PHImageRequestOptions的实例,包括控制图片版本,质量,裁剪参数等的一个类.

resultHandler:成功回调block,

result:获取到的具体图片,

info:关于图片的一些信息,如是否来自 cloud, 是否是原图等.

PHCachingImageManager

PHCachingImageManager是PHImageManager的子类,主要作用在获取图片的时候做缓存和清理的一个类

//缓存图片
- (void)startCachingImagesForAssets:(NSArray<PHAsset *> *)assets targetSize:(CGSize)targetSize contentMode:(PHImageContentMode)contentMode options:(nullable PHImageRequestOptions *)options;

assets:要缓存获取 PHAsset 类型对象的集合.
targetSize:想要缓存的尺寸.
contentMode:裁剪方法,
options:传入的控制参数类.
取消缓存操作.
- (void)stopCachingImagesForAssets:(NSArray<PHAsset *> *)assets targetSize:(CGSize)targetSize contentMode:(PHImageContentMode)contentMode options:(nullable PHImageRequestOptions *)options;

assets:要缓存获取 PHAsset 类型对象的集合.
targetSize:缓存的尺寸.
contentMode:裁剪方法,
options:传入的控制参数类.

PHImageRequestOptions

deliveryMode    //控制图片质量和获取速度的 api

PHImageRequestOptionsDeliveryMode类型的枚举,只有synchronous属性设置为 YES,即异步获取有限
    PHImageRequestOptionsDeliveryModeOpportunistic  图片质量和获取速度均衡
    PHImageRequestOptionsDeliveryModeHighQualityFormat 获取高质量图片,不保证获取速度
    PHImageRequestOptionsDeliveryModeFastFormat 快速获得,不保证质量
resizeMode    //裁剪的方式

PHImageRequestOptionsResizeMode类型的枚举:
PHImageRequestOptionsResizeModeNone 不设置   PHImageRequestOptionsResizeModeFast  返回的图像可能和目标大小不一样并且质量较低,但效率高.
PHImageRequestOptionsResizeModeExact 返回图像必须和目标大小相匹配,并且图像质量也为高质量图像

代码范例:

PHFetchOptions
        PHFetchOptions *option = [[PHFetchOptions alloc]init];
        option.predicate = [NSPredicate predicateWithFormat:@"mediaType == %ld", PHAssetMediaTypeImage];
        option.predicate = [NSPredicate predicateWithFormat:@"mediaType == %ld", PHAssetMediaTypeVideo];
        option.sortDescriptors = @[[[NSSortDescriptor alloc]initWithKey:@"createDate" ascending:YES]];
PHAssetCollection(获取相册列表)
        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 *result in allAlbums)
        {
            for (PHAssetCollection *collection in result) {
                //过滤掉非PHAssetCollection
                if (![collection isKindOfClass:[PHAssetCollection class]]) {
                    continue;
                }
                PHFetchResult *assetResult = [PHAsset fetchAssetsInAssetCollection:collection options:option];
                //如果相册里面没有资源过滤掉
                if (assetResult.count <= 0) {
                    continue;
                }
                //过滤掉最近删除的
                if ([collection.localizedTitle containsString:@"Deleted"] || [collection.localizedTitle isEqualToString:@"最近删除"])
                {
                   continue;
                }
                [_albumArray addObject:collection];
            }
        }
    }

PHImageManager
 PHImageRequestOptions *option = [[PHImageRequestOptions alloc]init];
    option.resizeMode = PHImageRequestOptionsResizeModeFast;
    [[PHImageManager defaultManager]requestImageForAsset:asset targetSize:fsize contentMode:PHImageContentModeAspectFill options:option resultHandler:^(UIImage * _Nullable result, NSDictionary * _Nullable info) {
        //这一块为什么要加BOOL值来判断呢,是因为有时候图片会从iCloud下载,这个时候这个block会回调多次,此时在block里面做的操作也会被执行多次,于是需要进行判断,当图片被取消或者错误或者低清图的时候不执行回调
        BOOL downloadFinined = ![[info objectForKey:PHImageCancelledKey] boolValue] && ![info objectForKey:PHImageErrorKey] && ![[info objectForKey:PHImageResultIsDegradedKey] boolValue];
        if (downloadFinined && result) {
            if (noCut) {
                 completion ? completion(result) : nil;
            }else
            {
                //这一块拿到的图片其实是原先的图片等比列缩小的,因为原先的图片宽和高不一定相等,所以图片会变形,需要进行处理
                result = [self customImage:result];
                 completion ? completion(result) : nil;
            }
           
        }
    }];
//将不合适的图片裁剪成合适的缩略图
-(UIImage*)customImage:(UIImage*)image
{
    CGSize size;
    if (image.size.width*1.0/image.size.height < 1) { //高比宽大,已宽为基准
        size.width = image.size.width;
        size.height = image.size.width;
        //下面这段代码的意思是:因为高大于宽,所以y从中间裁剪
        CGImageRef imageref = CGImageCreateWithImageInRect(image.CGImage, CGRectMake(0, fabs((image.size.height - image.size.width)/2.0), size.width, size.height));
        return  [UIImage imageWithCGImage:imageref];
    }else   //宽比高大
    {
        size.width = image.size.height;
        size.height = image.size.height;
        //下面这段代码的意思是:因为高大于宽,所以y从中间裁剪
        CGImageRef imageref = CGImageCreateWithImageInRect(image.CGImage, CGRectMake( fabs((image.size.width - image.size.height)/2.0),0, size.width, size.height));
        return  [UIImage imageWithCGImage:imageref];
    }
}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335

推荐阅读更多精彩内容