iOS之相册新框架<Photos/Photos.h>

获取相册中的图片


几个常用的类:
======

PHAsset : 代表照片库中的一个资源,跟 ALAsset 类似,通过 PHAsset 可以获取和保存资源。每个PHAsset就是一张图片的详细信息,包括图片、位置、时间等。

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

PHFetchResult : 表示一系列的资源集合,也可以是相册的集合。

PHAssetCollection : 表示一个相册或者一个时刻,或者是一个智能相册(系统提供的特定的一系列相册,例如:最近删除,视频列表,收藏等等)。

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

使用系统相册的时候,首先判断是否有权限,没有权限的话,可以弹出提示框。

PHImageRequestOptions : 如上面所说,控制加载图片时的一系列参数。
PHAuthorizationStatus status = [PHPhotoLibrary authorizationStatus];  
if (status == PHAuthorizationStatusRestricted ||  
    status == PHAuthorizationStatusDenied) {  
    // 这里便是无访问权限  
    //可以弹出个提示框,叫用户去设置打开相册权限
}  else {
    //这里就是用权限
}

获得相册的所有图片

// 获取所有资源的集合,并按资源的创建时间排序
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;
 }

获得自定义的所有相簿

// 获得所有的自定义相簿
PHFetchResult<PHAssetCollection *> *assetCollections = [PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeAlbum subtype:PHAssetCollectionSubtypeAlbumRegular options:nil];
// 遍历所有的自定义相簿
for (PHAssetCollection *assetCollection in assetCollections) {

}

获得相机胶卷相簿

// 获得相机胶卷
PHAssetCollection *cameraRoll = [PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeSmartAlbum subtype:PHAssetCollectionSubtypeSmartAlbumUserLibrary options:nil].lastObject;
获得某个相簿的缩略图

 PHImageRequestOptions *options = [[PHImageRequestOptions alloc] init];
 // 同步获得图片, 只会返回1张图片
 options.synchronous = YES;

// 获得某个相簿中的所有PHAsset对象
PHFetchResult<PHAsset *> *assets = [PHAsset fetchAssetsInAssetCollection:assetCollection options:nil];
for (PHAsset *asset in assets) {
CGSize size = CGSizeZero;

// 从asset中获得图片
[[PHImageManager defaultManager] requestImageForAsset:asset targetSize:size contentMode:PHImageContentModeDefault options:options resultHandler:^(UIImage * _Nullable result, NSDictionary * _Nullable info) {
    NSLog(@"%@", result);
}];
}
获得某个相簿的原图

PHImageRequestOptions *options = [[PHImageRequestOptions alloc] init];
// 同步获得图片, 只会返回1张图片
options.synchronous = YES;

// 获得某个相簿中的所有PHAsset对象
PHFetchResult<PHAsset *> *assets = [PHAsset fetchAssetsInAssetCollection:assetCollection options:nil];
 for (PHAsset *asset in assets) {
CGSize size = CGSizeMake(asset.pixelWidth, asset.pixelHeight);

// 从asset中获得图片
[[PHImageManager defaultManager] requestImageForAsset:asset targetSize:size contentMode:PHImageContentModeDefault options:options resultHandler:^(UIImage * _Nullable result, NSDictionary * _Nullable info) {
    NSLog(@"%@", result);
}];
}
利用UIImagePickerController挑选图片

// UIImagePickerController : 可以从系统自带的App(照片\相机)中获得图片

// 判断相册是否可以打开
if (![UIImagePickerController isSourceTypeAvailable:UIImagePickerControllerSourceTypePhotoLibrary]) return;

UIImagePickerController *ipc = [[UIImagePickerController alloc] init];
// 打开照片应用(显示所有相簿)
ipc.sourceType = UIImagePickerControllerSourceTypePhotoLibrary;
// 打开照片应用(只显示"时刻"这个相簿)
// ipc.sourceType = UIImagePickerControllerSourceTypeSavedPhotosAlbum;
// 照相机
// ipc.sourceType = UIImagePickerControllerSourceTypeCamera;
ipc.delegate = self;
[self presentViewController:ipc animated:YES completion:nil];

#pragma mark - <UIImagePickerControllerDelegate>
 - (void)imagePickerController:(UIImagePickerController *)picker didFinishPickingMediaWithInfo:(NSDictionary<NSString *,id> *)info
{
// 销毁控制器
[picker dismissViewControllerAnimated:YES completion:nil];

// 设置图片
self.imageView.image = info[UIImagePickerControllerOriginalImage];
}
NaN错误

- 错误起因:0被当做除数, 比如 10 / 0
最简单的方法保存图片到相机胶卷

 UIImageWriteToSavedPhotosAlbum(self.imageView.image, self, @selector(image:didFinishSavingWithError:contextInfo:), nil);

 /**
 *  通过UIImageWriteToSavedPhotosAlbum函数写入图片完毕后就会调用这个方法
 *
 *  @param image       写入的图片
 *  @param error       错误信息
 *  @param contextInfo UIImageWriteToSavedPhotosAlbum函数的最后一个参数
 */
  - (void)image:(UIImage *)image didFinishSavingWithError:(NSError *)error contextInfo:(void *)contextInfo
 {
if (error) {
    [SVProgressHUD showErrorWithStatus:@"图片保存失败!"];
} else {
    [SVProgressHUD showSuccessWithStatus:@"图片保存成功!"];
}
}
保存图片到自定义相册

 - (IBAction)save {
/*
 PHAuthorizationStatusNotDetermined,     用户还没有做出选择
 PHAuthorizationStatusDenied,            用户拒绝当前应用访问相册(用户当初点击了"不允许")
 PHAuthorizationStatusAuthorized         用户允许当前应用访问相册(用户当初点击了"好")
 PHAuthorizationStatusRestricted,        因为家长控制, 导致应用无法方法相册(跟用户的选择没有关系)
 */

// 判断授权状态
PHAuthorizationStatus status = [PHPhotoLibrary authorizationStatus];
if (status == PHAuthorizationStatusRestricted) { // 因为家长控制, 导致应用无法方法相册(跟用户的选择没有关系)
    [SVProgressHUD showErrorWithStatus:@"因为系统原因, 无法访问相册"];
} else if (status == PHAuthorizationStatusDenied) { // 用户拒绝当前应用访问相册(用户当初点击了"不允许")
    NSLog(@"提醒用户去[设置-隐私-照片-xxx]打开访问开关");
} else if (status == PHAuthorizationStatusAuthorized) { // 用户允许当前应用访问相册(用户当初点击了"好")
    [self saveImage];
} else if (status == PHAuthorizationStatusNotDetermined) { // 用户还没有做出选择
    // 弹框请求用户授权
    [PHPhotoLibrary requestAuthorization:^(PHAuthorizationStatus status) {
        if (status == PHAuthorizationStatusAuthorized) { // 用户点击了好
            [self saveImage];
        }
    }];
}
 }

 // 保存图片
 - (void)saveImage
{
// PHAsset : 一个资源, 比如一张图片\一段视频
// PHAssetCollection : 一个相簿

// PHAsset的标识, 利用这个标识可以找到对应的PHAsset对象(图片对象)
__block NSString *assetLocalIdentifier = nil;

// 如果想对"相册"进行修改(增删改), 那么修改代码必须放在[PHPhotoLibrary sharedPhotoLibrary]的performChanges方法的block中
[[PHPhotoLibrary sharedPhotoLibrary] performChanges:^{
    // 1.保存图片A到"相机胶卷"中
    // 创建图片的请求
    assetLocalIdentifier = [PHAssetCreationRequest creationRequestForAssetFromImage:self.imageView.image].placeholderForCreatedAsset.localIdentifier;
} completionHandler:^(BOOL success, NSError * _Nullable error) {
    if (success == NO) {
        [self showError:@"保存图片失败!"];
        return;
    }

    // 2.获得相簿
    PHAssetCollection *createdAssetCollection = [self createdAssetCollection];
    if (createdAssetCollection == nil) {
        [self showError:@"创建相簿失败!"];
        return;
    }

    [[PHPhotoLibrary sharedPhotoLibrary] performChanges:^{
        // 3.添加"相机胶卷"中的图片A到"相簿"D中

        // 获得图片
        PHAsset *asset = [PHAsset fetchAssetsWithLocalIdentifiers:@[assetLocalIdentifier] options:nil].lastObject;

        // 添加图片到相簿中的请求
        PHAssetCollectionChangeRequest *request = [PHAssetCollectionChangeRequest changeRequestForAssetCollection:createdAssetCollection];

        // 添加图片到相簿
        [request addAssets:@[asset]];
    } completionHandler:^(BOOL success, NSError * _Nullable error) {
        if (success == NO) {
            [self showError:@"保存图片失败!"];;
        } else {
            [self showSuccess:@"保存图片成功!"];;
        }
    }];
}];
}

/**
 *  获得相簿
 */
 - (PHAssetCollection *)createdAssetCollection
{
// 从已存在相簿中查找这个应用对应的相簿
PHFetchResult<PHAssetCollection *> *assetCollections = [PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeAlbum subtype:PHAssetCollectionSubtypeAlbumRegular options:nil];
for (PHAssetCollection *assetCollection in assetCollections) {
    if ([assetCollection.localizedTitle isEqualToString:XMGAssetCollectionTitle]) {
        return assetCollection;
    }
}

// 没有找到对应的相簿, 得创建新的相簿

// 错误信息
NSError *error = nil;

// PHAssetCollection的标识, 利用这个标识可以找到对应的PHAssetCollection对象(相簿对象)
__block NSString *assetCollectionLocalIdentifier = nil;
[[PHPhotoLibrary sharedPhotoLibrary] performChangesAndWait:^{
    // 创建相簿的请求
    assetCollectionLocalIdentifier = [PHAssetCollectionChangeRequest creationRequestForAssetCollectionWithTitle:XMGAssetCollectionTitle].placeholderForCreatedAssetCollection.localIdentifier;
} error:&error];

// 如果有错误信息
if (error) return nil;

// 获得刚才创建的相簿
return [PHAssetCollection fetchAssetCollectionsWithLocalIdentifiers:@[assetCollectionLocalIdentifier] options:nil].lastObject;
}

- (void)showSuccess:(NSString *)text
{
dispatch_async(dispatch_get_main_queue(), ^{
    [SVProgressHUD showSuccessWithStatus:text];
});
}

- (void)showError:(NSString *)text
{
dispatch_async(dispatch_get_main_queue(), ^{
    [SVProgressHUD showErrorWithStatus:text];
});
}
Xcode插件的安装路径

/Users/用户名/Library/Application Support/Developer/Shared/Xcode/Plug-ins
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值