人一切的痛苦,本质上都是对自己的无能的愤怒。
1.相册访问权限
2.相册中图片变化的监听
3.读取相册中的图片
4.将图片保存到相册
相册访问权限
使用系统相册的时候,首先判断是否有权限,没有权限的话,可以弹出提示框。
PHAuthorizationStatus status = [PHPhotoLibrary authorizationStatus];
if (status == PHAuthorizationStatusRestricted ||
status == PHAuthorizationStatusDenied) {
// 这里便是无访问权限
//可以弹出个提示框,叫用户去设置打开相册权限
} else {
//这里就是用权限
}
相册中图片变化的监听
相册监听中的回调方法是在 子线程 中的,因此要改变UI布局的话,必须回到主线程
[[PHPhotoLibrary sharedPhotoLibrary] registerChangeObserver:self]; //创建监听者
[[PHPhotoLibrary sharedPhotoLibrary] unregisterChangeObserver:self]; //移除监听者
//相册变化回调
- (void)photoLibraryDidChange:(PHChange *)changeInstance
{
dispatch_sync(dispatch_get_main_queue(), ^{
//修改UI
});
}
获取相册中的图片
** 几个常用的类 **
**PHAsset **: 代表照片库中的一个资源,跟 ALAsset 类似,通过 PHAsset 可以获取和保存资源。每个PHAsset就是一张图片的详细信息,包括图片、位置、时间等。
** PHFetchOptions **: 获取资源时的参数,可以传 nil,即使用系统默认值。
** PHFetchResult **: 表示一系列的资源集合,也可以是相册的集合。
** PHAssetCollection **: 表示一个相册或者一个时刻,或者是一个智能相册(系统提供的特定的一系列相册,例如:最近删除,视频列表,收藏等等)。
** PHImageManager **: 用于处理资源的加载,加载图片的过程带有缓存处理,可以通过传入一个 PHImageRequestOptions 控制资源的输出尺寸等规格。
** PHImageRequestOptions **: 如上面所说,控制加载图片时的一系列参数。
// 获取所有资源的集合,并按资源的创建时间排序
PHFetchOptions *options = [[PHFetchOptions alloc] init];
options.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"creationDate" ascending:YES]];
//获取相机胶卷所有图片
PHFetchResult *assets = [PHAsset fetchAssetsWithMediaType:PHAssetMediaTypeVideo options:nil];
PHImageRequestOptions *option = [[PHImageRequestOptions alloc] init];
//设置显示模式
/*
PHImageRequestOptionsResizeModeNone //选了这个就不会管传如的size了 ,要自己控制图片的大小,建议还是选Fast
PHImageRequestOptionsResizeModeFast //根据传入的size,迅速加载大小相匹配(略大于或略小于)的图像
PHImageRequestOptionsResizeModeExact //精确的加载与传入size相匹配的图像
*/
option.resizeMode = PHImageRequestOptionsResizeModeFast;
option.synchronous = NO;
option.deliveryMode = PHImageRequestOptionsDeliveryModeHighQualityFormat;
CGFloat scale = [UIScreen mainScreen].scale;
CGSize screenSize = [UIScreen mainScreen].bounds.size;
typeof(self)weakSelf = self;
for (PHAsset *asset in assets) {
[[PHImageManager defaultManager] requestImageForAsset:asset targetSize:CGSizeMake(screenSize.width*scale, screenSize.height*scale) contentMode:PHImageContentModeAspectFit options:option resultHandler:^(UIImage * _Nullable result, NSDictionary * _Nullable info) {
//我这里用个模型接储存了图片的 localIdentifier 和图片本身
//还有很多信息,根据自己的需求来取
CellModel * model = [CellModel new];
model.localIdentifier = asset.localIdentifier;
model.image = [UIImage imageWithData:UIImageJPEGRepresentation(result, 0.5)];
[weakSelf.cellImageArray addObject:model];
dispatch_async(dispatch_get_main_queue(), ^{
[weakSelf reloadData];
});
}];
}
将图片保存到相册
-
方法一:
- 优点:简单快捷
- 缺点:无法直接保存到自定义相册
- (IBAction)savePhoto {
// - (void)image:(UIImage *)image didFinishSavingWithError:(NSError *)error contextInfo:(void *)contextInfo;
//只能保存到相机胶卷(不能保存到自定义相册)
UIImageWriteToSavedPhotosAlbum(self.Photo.image, self, @selector(image:didFinishSavingWithError:contextInfo:), nil);
}
//回调判断保存是否成功
- (void)image:(UIImage *)image didFinishSavingWithError:(NSError *)error contextInfo:(void *)contextInfo
{
if (error) {
//保存失败
}else{
//保存成功
}
}
- ** 方法二: **
- 优点:能够保存到任何相册
- 缺点:没有第一种方法简单
// 添加图片到自己相册
- (void)savePhoto
{
[[PHPhotoLibrary sharedPhotoLibrary] performChanges:^{
// 1.创建图片请求类(创建系统相册中新的图片)PHAssetCreationRequest
// 把图片放在系统相册
PHAssetCreationRequest *assetCreationRequest = [PHAssetCreationRequest creationRequestForAssetFromImage:_imageView.image];
// 2.创建相册请求类(修改相册)PHAssetCollectionChangeRequest
PHAssetCollectionChangeRequest *assetCollectionChangeRequest = nil;
// 获取之前相册
PHAssetCollection *assetCollection = [self fetchAssetCollection:@"百思不得姐"];
// 判断是否已有相册
if (assetCollection) {
// 如果存在已有同名相册 指定这个相册,创建相册请求修改类
assetCollectionChangeRequest = [PHAssetCollectionChangeRequest changeRequestForAssetCollection:assetCollection];
} else { //不存在,创建新的相册
assetCollectionChangeRequest = [PHAssetCollectionChangeRequest creationRequestForAssetCollectionWithTitle:@"百思不得姐"];
}
// 3.把图片添加到相册中
// NSFastEnumeration:以后只要看到这个,就可以表示数组
//assetCreationRequest.placeholderForCreatedAsset 图片请求类占位符(相当于一个内存地址)
//因为creationRequestForAssetFromImage方法是异步实行的,在这里不能保证 assetCreationRequest有值
[assetCollectionChangeRequest addAssets:@[assetCreationRequest.placeholderForCreatedAsset]];
} completionHandler:^(BOOL success, NSError * _Nullable error) {
if (success) {
[SVProgressHUD showSuccessWithStatus:@"保存成功"];
} else {
[SVProgressHUD showErrorWithStatus:@"保存失败"];
}
}];
}
// 指定相册名称,获取相册
- (PHAssetCollection *)fetchAssetCollection:(NSString *)title
{
// 获取相簿中所有自定义相册
PHFetchResult *result = [PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeAlbum subtype:PHAssetCollectionSubtypeAlbumRegular options:nil];
遍历相册,判断是否存在同名的相册
for (PHAssetCollection *assetCollection in result) {
if ([title isEqualToString:assetCollection.localizedTitle]) { 存在,就返回这个相册
return assetCollection;
}
}
return nil;
}