美文网首页
IOS保存图片到相册

IOS保存图片到相册

作者: 怀心逝水 | 来源:发表于2017-05-07 12:14 被阅读504次
    Come On

    这篇文章主要是介绍在IOS8之后官方给出的系统框架Photos。
    首先先看一看这个框架的分类:

    PHAsset: 代表照片库中的一个资源实体,可以理解为一张照片,在打印的时候,可以清楚的看 见里面包含了照片的时间、标题等信息。
    PHFetchOptions: 获取资源时的参数,可以传 nil,即使用系统默认值
    PHFetchResult: 表示结果集,可以使相册集合,可以使照片实体集合。
    PHAssetCollection: 表示一个相册或者一个时刻,或者是一个「智能相册(系统提供的特定的一
    系列相册,例如:最近删除,视频列表,收藏等等,如下图所示)
    PHCollectionList :经过一番尝试,这个的使用和上面PHAssetCollection差不多,不做详细解释啦。
    PHImageManager: 用于处理资源的加载,加载图片的过程带有缓存处理,可以通过传入一个 PHImageRequestOptions 控制资源的输出尺寸等规格
    PHImageRequestOptions: 如上面所说,控制加载图片时的一系列参数

    由于个人开发中主要用到如何将图片储存到本地相册中,所以就只是对涉及的用法作一个简单的说明。

    大体的思路:

    1. 获取当前用户的相册授权(可以访问相册还是拒绝)。
      以下都是在允许的访问的权限下进行的:
    2. 通过PHFetchResult,并且利用图片获得PHAsset对象,从而标记需要储存的图片。
    3. 通过PHFetchResult去初始化一个利用相册名称的PHAssetCollection(存放图片集合的对象)
    4. 将图片插入到相应的图片集合中,完成全部操作。

    下面的代码部分是参考别人的,只不过个人将其封装化,这样就可以很方便的进行调用了。
    对外的调用接口:

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

    至于MBProgressHUD,可以在相应的GitHub自行下载。

    + (void)saveImageToCustomAblumWithImage:(UIImage *)image
    {
        //1 将图片保存到系统的【相机胶卷】中---调用刚才的方法
        PHFetchResult<PHAsset *> *assets = [self syncSaveImageWithPhotos:image];
        if (assets == nil)
        {
            [MBProgressHUD showError:@"保存失败"];
            return;
        }
        
        //2 拥有自定义相册(与 APP 同名,如果没有则创建)--调用刚才的方法
        PHAssetCollection *assetCollection = [self getAssetCollectionWithAppNameAndCreateIfNo];
        if (assetCollection == nil) {
            [MBProgressHUD showError:@"相册创建失败"];
            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) {
            [MBProgressHUD showError:@"保存失败"];
            return;
        }
        [MBProgressHUD showSuccess:@"保存成功"];
    }
    
    + (PHFetchResult<PHAsset *> *)syncSaveImageWithPhotos:(UIImage *)image
    {
        //--1 创建 ID 这个参数可以获取到图片保存后的 asset对象
        __block NSString *createdAssetID = nil;
        
        //--2 保存图片
        NSError *error = nil;
        [[PHPhotoLibrary sharedPhotoLibrary] performChangesAndWait:^{
            //----block 执行的时候还没有保存成功--获取占位图片的 id,通过 id 获取图片---同步
            createdAssetID = [PHAssetChangeRequest creationRequestForAssetFromImage:image].placeholderForCreatedAsset.localIdentifier;
        } error:&error];
        
        //--3 如果失败,则返回空
        if (error) {
            return nil;
        }
        
        //--4 成功后,返回对象
        //获取保存到系统相册成功后的 asset 对象集合,并返回
        PHFetchResult<PHAsset *> *assets = [PHAsset fetchAssetsWithLocalIdentifiers:@[createdAssetID] options:nil];
        return assets;
    }
    
    + (PHAssetCollection *)getAssetCollectionWithAppNameAndCreateIfNo
    {
        //1 获取以 APP 的名称
        NSDictionary *infoDictionary = [[NSBundle mainBundle] infoDictionary];
        NSString *title = [infoDictionary objectForKey:@"CFBundleDisplayName"];
        //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) {
            [MBProgressHUD showError:@"创建失败"];
            return nil;
        }else{
            [MBProgressHUD showSuccess:@"创建成功"];
            //通过 ID 获取创建完成的相册 -- 是一个数组
            return [PHAssetCollection fetchAssetCollectionsWithLocalIdentifiers:@[createID] options:nil].firstObject;
        }
    }
    
    

    这样图片就会以app的名称形式储存到本地的相册中。

    相关文章

      网友评论

          本文标题:IOS保存图片到相册

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