这篇文章主要是介绍在IOS8之后官方给出的系统框架Photos。
首先先看一看这个框架的分类:
PHAsset: 代表照片库中的一个资源实体,可以理解为一张照片,在打印的时候,可以清楚的看 见里面包含了照片的时间、标题等信息。
PHFetchOptions: 获取资源时的参数,可以传 nil,即使用系统默认值
PHFetchResult: 表示结果集,可以使相册集合,可以使照片实体集合。
PHAssetCollection: 表示一个相册或者一个时刻,或者是一个「智能相册(系统提供的特定的一
系列相册,例如:最近删除,视频列表,收藏等等,如下图所示)
PHCollectionList :经过一番尝试,这个的使用和上面PHAssetCollection差不多,不做详细解释啦。
PHImageManager: 用于处理资源的加载,加载图片的过程带有缓存处理,可以通过传入一个 PHImageRequestOptions 控制资源的输出尺寸等规格
PHImageRequestOptions: 如上面所说,控制加载图片时的一系列参数
由于个人开发中主要用到如何将图片储存到本地相册中,所以就只是对涉及的用法作一个简单的说明。
大体的思路:
- 获取当前用户的相册授权(可以访问相册还是拒绝)。
以下都是在允许的访问的权限下进行的: - 通过PHFetchResult,并且利用图片获得PHAsset对象,从而标记需要储存的图片。
- 通过PHFetchResult去初始化一个利用相册名称的PHAssetCollection(存放图片集合的对象)
- 将图片插入到相应的图片集合中,完成全部操作。
下面的代码部分是参考别人的,只不过个人将其封装化,这样就可以很方便的进行调用了。
对外的调用接口:
+ (void)saveImage:(UIImage *)image
{
//(1) 获取当前的授权状态
PHAuthorizationStatus lastStatus = [PHPhotoLibrary authorizationStatus];
//(2) 请求授权
[PHPhotoLibrary requestAuthorization:^(PHAuthorizationStatus status) {
//回到主线程
dispatch_async(dispatch_get_main_queue(), ^{
if(status == PHAuthorizationStatusDenied) //用户拒绝(可能是之前拒绝的,有可能是刚才在系统弹框中选择的拒绝)
{
if (lastStatus == PHAuthorizationStatusNotDetermined) {
//说明,用户之前没有做决定,在弹出授权框中,选择了拒绝
[MBProgressHUD showError:@"保存失败"];
return;
}
// 说明,之前用户选择拒绝过,现在又点击保存按钮,说明想要使用该功能,需要提示用户打开授权
[MBProgressHUD showMessage:@"失败!请在系统设置中开启访问相册权限"];
}
else if(status == PHAuthorizationStatusAuthorized) //用户允许
{
//保存图片---调用上面封装的方法
[self saveImageToCustomAblumWithImage:image];
}
else if (status == PHAuthorizationStatusRestricted)
{
[MBProgressHUD showError:@"系统原因,无法访问相册"];
}
});
}];
}
至于MBProgressHUD,可以在相应的GitHub自行下载。
+ (void)saveImageToCustomAblumWithImage:(UIImage *)image
{
//1 将图片保存到系统的【相机胶卷】中---调用刚才的方法
PHFetchResult<PHAsset *> *assets = [self syncSaveImageWithPhotos:image];
if (assets == nil)
{
[MBProgressHUD showError:@"保存失败"];
return;
}
//2 拥有自定义相册(与 APP 同名,如果没有则创建)--调用刚才的方法
PHAssetCollection *assetCollection = [self getAssetCollectionWithAppNameAndCreateIfNo];
if (assetCollection == nil) {
[MBProgressHUD showError:@"相册创建失败"];
return;
}
//3 将刚才保存到相机胶卷的图片添加到自定义相册中 --- 保存带自定义相册--属于增的操作,需要在PHPhotoLibrary的block中进行
NSError *error = nil;
[[PHPhotoLibrary sharedPhotoLibrary] performChangesAndWait:^{
//--告诉系统,要操作哪个相册
PHAssetCollectionChangeRequest *collectionChangeRequest = [PHAssetCollectionChangeRequest changeRequestForAssetCollection:assetCollection];
//--添加图片到自定义相册--追加--就不能成为封面了
//--[collectionChangeRequest addAssets:assets];
//--插入图片到自定义相册--插入--可以成为封面
[collectionChangeRequest insertAssets:assets atIndexes:[NSIndexSet indexSetWithIndex:0]];
} error:&error];
if (error) {
[MBProgressHUD showError:@"保存失败"];
return;
}
[MBProgressHUD showSuccess:@"保存成功"];
}
+ (PHFetchResult<PHAsset *> *)syncSaveImageWithPhotos:(UIImage *)image
{
//--1 创建 ID 这个参数可以获取到图片保存后的 asset对象
__block NSString *createdAssetID = nil;
//--2 保存图片
NSError *error = nil;
[[PHPhotoLibrary sharedPhotoLibrary] performChangesAndWait:^{
//----block 执行的时候还没有保存成功--获取占位图片的 id,通过 id 获取图片---同步
createdAssetID = [PHAssetChangeRequest creationRequestForAssetFromImage:image].placeholderForCreatedAsset.localIdentifier;
} error:&error];
//--3 如果失败,则返回空
if (error) {
return nil;
}
//--4 成功后,返回对象
//获取保存到系统相册成功后的 asset 对象集合,并返回
PHFetchResult<PHAsset *> *assets = [PHAsset fetchAssetsWithLocalIdentifiers:@[createdAssetID] options:nil];
return assets;
}
+ (PHAssetCollection *)getAssetCollectionWithAppNameAndCreateIfNo
{
//1 获取以 APP 的名称
NSDictionary *infoDictionary = [[NSBundle mainBundle] infoDictionary];
NSString *title = [infoDictionary objectForKey:@"CFBundleDisplayName"];
//2 获取与 APP 同名的自定义相册
PHFetchResult<PHAssetCollection *> *collections = [PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeAlbum subtype:PHAssetCollectionSubtypeAlbumRegular options:nil];
for (PHAssetCollection *collection in collections) {
//遍历
if ([collection.localizedTitle isEqualToString:title]) {
//找到了同名的自定义相册--返回
return collection;
}
}
//说明没有找到,需要创建
NSError *error = nil;
__block NSString *createID = nil; //用来获取创建好的相册
[[PHPhotoLibrary sharedPhotoLibrary] performChangesAndWait:^{
//发起了创建新相册的请求,并拿到ID,当前并没有创建成功,待创建成功后,通过 ID 来获取创建好的自定义相册
PHAssetCollectionChangeRequest *request = [PHAssetCollectionChangeRequest creationRequestForAssetCollectionWithTitle:title];
createID = request.placeholderForCreatedAssetCollection.localIdentifier;
} error:&error];
if (error) {
[MBProgressHUD showError:@"创建失败"];
return nil;
}else{
[MBProgressHUD showSuccess:@"创建成功"];
//通过 ID 获取创建完成的相册 -- 是一个数组
return [PHAssetCollection fetchAssetCollectionsWithLocalIdentifiers:@[createID] options:nil].firstObject;
}
}
这样图片就会以app的名称形式储存到本地的相册中。