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

保存图片到自定义相册

<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:@"系统原因,无法访问相册"];
            }
        });
    }];
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,839评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,543评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,116评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,371评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,384评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,111评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,416评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,053评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,558评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,007评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,117评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,756评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,324评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,315评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,539评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,578评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,877评论 2 345

推荐阅读更多精彩内容