iOS开发技术分享首页投稿(暂停使用,暂停投稿)傲视苍穹iOS《Objective-C》VIP专题

iOS-保存图片到自定义相册|Photos|C函数

2016-08-17  本文已影响3546人  岁与禾

保存图片到自定义相册

<b style="color:red">
实际上,自定义相册中的图片并不是实际的图片,而是对系统【相机胶卷】这个相册中的图片进行了一个引用。所以将图片保存到自定义相册的第一步就是先保存到系统的【相机胶卷】中。
</b>

1 步骤

2 Photos 框架简单介绍

2.1 重要的类

该框架有几个非常重要的类:PHAsset、PHAssetCollection 和 PHLibrary。

<b style="color:red">
只要与单个图片相关,使用 PHAsset。只要与相册相关,使用 PHAssetCollection
</b>

2.2 查询操作

查询操作,直接使用 PHAsset 和 PHAssetCollection 类本身的方法

//1 获取相册中的图片--传入相册图片的 ID---返回一组图片
[PHAsset fetchAssetsWithLocalIdentifiers:@[ID] options:nil];

//2 查询手机中所有的相册列表(分为系统相册和自定义相册,通过控制传入的参数来确定)---返回相册组--类似数组-forin遍历即可
[PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeAlbum subtype:PHAssetCollectionSubtypeAlbumRegular options:nil];
2.3 增删改操作(除了获取之外所有的操作)

<b style="color:red">
1 如果做查询之外的操作,比如说保存图片、创建自定义相册、向自定义相册中添加图片等,都需要使用另外两个类:PHAssetChangeRequest 和 PHAssetCollectionChangeRequest

2 这些操作必须在 [[PHPhotoLibrary sharedPhotoLibrary]performChange...]的 block 中间调用

</b>

//1 保存图片
[PHAssetChangeRequest creationRequestForAssetFromImage:self.imageView.image];

//2 创建相册
[PHAssetCollectionChangeRequest creationRequestForAssetCollectionWithTitle:@"自定义相册"];   

3 将图片保存到系统相册【相机胶卷】中

3.1 C语言函数保存

点击保存按钮后的代码:

//1 把图片保存到系统相册中,结束后调用 image:didFinishSavingWithError:contextInfo:方法(回调方法)
//2 回调方法的格式有要求,可以进入头文件查看
UIImageWriteToSavedPhotosAlbum(self.imageView.image, self, @selector(image:didFinishSavingWithError:contextInfo:), nil);

实现回调方法

-(void)image:(UIImage *)image didFinishSavingWithError:(NSError *)error contextInfo:(void *)contextInfo
{
    if(error)
    {
        NSLog(@"保存图片失败");
        return;
    }
    NSLog(@"保存图片成功");
}
3.2 Photos 框架保存图片到系统相册

Photos 框架保存图片 --- 使用 PHAssetChangeRequest 类 方法

有两种方式;异步方式和同步方式,但是保存图片的操作性能消耗不大,所以可以直接使用同步方式

3.2.1 异步方式保存图片
//异步保存图片
-(void)asyncSaveImageWithPhotos
{
    //1 必须在 block 中调用
    [[PHPhotoLibrary sharedPhotoLibrary] performChanges:^{
        //2 异步执行保存图片操作
        [PHAssetChangeRequest creationRequestForAssetFromImage:self.imageView.image];

    } completionHandler:^(BOOL success, NSError * _Nullable error) {
        //3 保存结束后,回调
        if (error) {
            [SVProgressHUD showErrorWithStatus:@"保存失败"];
        }else
            [SVProgressHUD showSuccessWithStatus:@"保存成功"];
    }];
}

#######3.2.2 同步方式保存图片

下面的例子是通过保存时刻的占位 id 来获取图像,其实也可以直接返回占位图片。后面的操作可以直接进使用占位图片代替图片

/**同步方式保存图片到系统的相机胶卷中---返回的是当前保存成功后相册图片对象集合*/
-(PHFetchResult<PHAsset *> *)syncSaveImageWithPhotos
{
    //--1 创建 ID 这个参数可以获取到图片保存后的 asset对象
    __block NSString *createdAssetID = nil;

    //--2 保存图片
    NSError *error = nil;
    [[PHPhotoLibrary sharedPhotoLibrary] performChangesAndWait:^{
        //----block 执行的时候还没有保存成功--获取占位图片的 id,通过 id 获取图片---同步
        createdAssetID = [PHAssetChangeRequest          creationRequestForAssetFromImage:self.imageView.image].placeholderForCreatedAsset.localIdentifier;
    } error:&error];

    //--3 如果失败,则返回空
    if (error) {
        return nil;
    }

    //--4 成功后,返回对象
    //获取保存到系统相册成功后的 asset 对象集合,并返回
    PHFetchResult<PHAsset *> *assets = [PHAsset fetchAssetsWithLocalIdentifiers:@[createdAssetID] options:nil];
    return assets;

}

4 拥有自定义相册(如果没有,则创建)

下面的例子是:如果没有,则常见跟当前 APP 同名的自定义相册

实现思路:

① 获取当前的 APP 的 BundleName

② 使用PHAssetCollection的fetchAssetCollectionsWithType:subType:options方法,通过传入类型,获取所有的自定义相册列表

③ 遍历获取的自定义相册列表,与APP的名称进行比对,匹配后返回当前同名的自定义相册 PHAssetCollection对象

④ 如果没有找到,则开始创建,在PHPhotoLibrary单例对象的perfromChange方法中执行创建自定义相册操作

⑤ block中,保存待创建相册的占位标识符---其实这个时刻,相册根本没创建完成

⑥ 通过error判断是否创建成功,

⑦ 如果创建成功,通过PHAssetCollection的fetchAssetCollectionsWithLocalIdentifiers:options来获取当前创建相册对象PHAssetCollection

/**拥有与 APP 同名的自定义相册--如果没有则创建*/
-(PHAssetCollection *)getAssetCollectionWithAppNameAndCreateIfNo
{
    //1 获取以 APP 的名称
    NSString *title = [NSBundle mainBundle].infoDictionary[(__bridge NSString *)kCFBundleNameKey];
    //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) {
        [SVProgressHUD showErrorWithStatus:@"创建失败"];
        return nil;
    }else{
        [SVProgressHUD showSuccessWithStatus:@"创建成功"];
        //通过 ID 获取创建完成的相册 -- 是一个数组
        return [PHAssetCollection fetchAssetCollectionsWithLocalIdentifiers:@[createID] options:nil].firstObject;
    }

}

5 将图片对象添加到自定义相册中

实现思路

① 使用获取的自定义相册来创建PHAssetCollection对象

② 将刚才保存到系统相册的PHAsset保存到相册中

/**将图片保存到自定义相册中*/
-(void)saveImageToCustomAblum
{
    //1 将图片保存到系统的【相机胶卷】中---调用刚才的方法
    PHFetchResult<PHAsset *> *assets = [self syncSaveImageWithPhotos];
    if (assets == nil)
    {
        [SVProgressHUD showErrorWithStatus:@"保存失败"];
        return;
    }

    //2 拥有自定义相册(与 APP 同名,如果没有则创建)--调用刚才的方法
    PHAssetCollection *assetCollection = [self getAssetCollectionWithAppNameAndCreateIfNo];
    if (assetCollection == nil) {
        [SVProgressHUD showErrorWithStatus:@"相册创建失败"];
        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) {
        [SVProgressHUD showErrorWithStatus:@"保存失败"];
        return;
    }
    [SVProgressHUD showSuccessWithStatus:@"保存成功"];
}

6 相册的授权访问

当第一次使用APP的时候,或者第一次访问相册的时候,系统会弹出授权选择对话框询问用户。所以我们在保存图片到相册的时候,需要判断当前是否授权。

6.1 权限分类
PHAuthorizationStatusNotDetermined ,---用户之前还未决定
PHAuthorizationStatusRestricted, ---系统问题,用户没有权限决定--比如家长控制器模式
PHAuthorizationStatusDenied,---用户之前拒绝过
PHAuthorizationStatusAuthorized --用户允许
6.2 请求权限的方式

可以使用NSPhotoLibrary的类方法requestAuthorization来查看权限,或者请求权限。如果用户之前没做决定,则弹出系统对话框请求权限;如果用户做过决定,则调用该类方法的block。

/*
 1 block 调用时刻---这个实在子线程中调用的
 --1.1 如果用户第一次打开 APP,之前决定过权限,则弹出系统框,让用户选择权限。---选择之后才会调用 block,并把刚才选择的结果一并传入
 --1.2 如果用户之前已经决定过权限,则直接调用 block,并把之前选择的结果传入
 2 state 类型
    PHAuthorizationStatusNotDetermined ,---用户之前还未决定,直接弹出系统对话框,这个 state 不会传给 block,会传入用户选择的结果
    PHAuthorizationStatusRestricted, ---系统问题,用户没有权限决定--比如家长控制器模式
    PHAuthorizationStatusDenied,---用户之前拒绝过
    PHAuthorizationStatusAuthorized --用户允许,直接调用 block,传入该状态
 **/

[PHPhotoLibrary requestAuthorization:^(PHAuthorizationStatus status) {
    //当前的block的调用是在子线程,需要回到主线程来操作
}
6.3 例子

点击保存图片按钮后的操作---一个完整的将图片保存到自定义相册的操作

-(void)save
{
    //(1) 获取当前的授权状态
    PHAuthorizationStatus lastStatus = [PHPhotoLibrary authorizationStatus];
    
    //(2) 请求授权
    [PHPhotoLibrary requestAuthorization:^(PHAuthorizationStatus status) {
        //回到主线程
        dispatch_async(dispatch_get_main_queue(), ^{
        
            if(status == PHAuthorizationStatusDenied) //用户拒绝(可能是之前拒绝的,有可能是刚才在系统弹框中选择的拒绝)
            {
                if (lastStatus == PHAuthorizationStatusNotDetermined) {
                    //说明,用户之前没有做决定,在弹出授权框中,选择了拒绝
                    [SVProgressHUD showErrorWithStatus:@"保存失败"];
                    return;
                }
                // 说明,之前用户选择拒绝过,现在又点击保存按钮,说明想要使用该功能,需要提示用户打开授权
                [SVProgressHUD showInfoWithStatus:@"失败!请在系统设置中开启访问相册权限"];
            
            }
            else if(status == PHAuthorizationStatusAuthorized) //用户允许
            {
                //保存图片---调用上面封装的方法
                [self saveImageToCustomAblum];
            }
            else if (status == PHAuthorizationStatusRestricted)
            {
                [SVProgressHUD showErrorWithStatus:@"系统原因,无法访问相册"];
            }
        });
    }];
}
上一篇下一篇

猜你喜欢

热点阅读