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

作者: 岁与禾 | 来源:发表于2016-08-17 20:14 被阅读3546次

    保存图片到自定义相册

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

    1 步骤

    • 保存到系统的相册【相机胶卷】中

        (1)C语言函数来保存
        (2)AssetsLibrary框架--系统自带,iOS9废弃
        (3)Photos框架--系统自带,iOS8即可使用,取代AssetsLibrary
      
    • 拥有自定义相册(如果没有,则创建)

        AssetsLibrary
        Photos
      
    • 将图片添加到自定义相册中

        AssetsLibrary
        Photos
      

    2 Photos 框架简单介绍

    2.1 重要的类

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

    • PHAsset 表示一个图片或者视频文件(存储在手机的照片 APP 中的)。与具体图片有关的使用这个类
    • 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:@"系统原因,无法访问相册"];
                }
            });
        }];
    }

    相关文章

      网友评论

      • 顺情风:请问有什么办法可以删除相册吗?删除照片的方法我找到了,但是每一张图片都会弹出一个弹框询问用户是否允许,这个弹框有办法去除吗?
      • PGOne爱吃饺子:楼主知道如何获取相册里的全部图片么 谢谢
      • PGOne爱吃饺子:写的真不赖,明白了,谢谢大神了
      • 心语风尚:[[PHImageManager defaultManager] requestImageForAsset:asset targetSize:size contentMode:PHImageContentModeDefault options:options resultHandler:^(UIImage * _Nullable result, NSDictionary * _Nullable info) {

        }];

        把这个方法放到 循环中 遍历 UIimage 添加到数组 作为数据源 这种方法现实吗
      • 心语风尚:为什么先保存胶卷 再保存自定义相册 而不是 直接保存自定义相册
        岁与禾:@心语风尚 不太清楚耶,你快试试,分享一下
        心语风尚:@一月二十三 还有一种方法 可以直接到自定义相册 就是通过路径 方法是 imageForUrl 传的是图片路径
        岁与禾:实际上,自定义相册中的图片并不是实际的图片,而是对系统【相机胶卷】这个相册中的图片进行了一个引用。所以将图片保存到自定义相册的第一步就是先保存到系统的【相机胶卷】中。
      • NeroXie:你好 请问有demo吗
        岁与禾:@nero_Xie 太久远了 没啦
      • b57bc247097e:想问下 保存图片时 会系统相册跟自定义相册中都保存一份 怎么只在自定义相册中保存而不在系统相册中保存呢?
      • 985c2ad14319:有几个问题请教下 1 . 对自建相册添加图片的时候 照片同时会保存到手机总的相册中对吧?
        2 下面这个方法是删除自建的相册的,包括自建相册中图片, 但是手机总的相册中图片没有被删除,如何将手机总的相册对应图片删除
        // PHAssetCollection *cr;
        // [PHAssetCollectionChangeRequest deleteAssetCollections:@[cr]];
      • 无神:首先作者写的很详细,赞一个。有点小瑕疵就是文章有错别字,希望写完文章之后能检查一下。
        岁与禾:@无神 哈哈,谢谢指出错误。
        岁与禾: @无神 thankyou
      • cdd48b9d36e0:楼主保存视频包括拍摄的视频也可以用Photos这个类吗
        cdd48b9d36e0:看到了这个creationRequestForAssetFromVideoAtFileURL貌似应该可以
      • bc35b5ddf2ef:将图片保存到自定义相册中,如何将UIImage保存到自定义相册中
        岁与禾:@bc35b5ddf2ef
        可以直接操作的,
        [PHAssetChangeRequest creationRequestForAssetFromImage:self.imageView.image];
        bc35b5ddf2ef: //1 将图片保存到系统的【相机胶卷】中---调用刚才的方法
        PHFetchResult<PHAsset *> *assets = [self syncSaveImageWithPhotos];
        这里存储的是一个PHAsset,如何变成UIImage
        岁与禾:@bc35b5ddf2ef ???就是UIImage呀

      本文标题:iOS-保存图片到自定义相册|Photos|C函数

      本文链接:https://www.haomeiwen.com/subject/paomsttx.html