iOS- 自定义相册(封装) | 干货

我们在APP中点击照片,都会显示出大图,然后在大图的上面会有个保存照片的按钮,照片直接保存到了系统的相册中,但是因为公司产品的需要,让你创建和APP同名的相册保存在里面,那么就对了,可以看下具体的代码和思路,然后把代码直接拿过去就可以用了,对,没错,站在巨人的肩膀上编程

一共有两种方法自定义相册
第一种是iOS9之后过期的 <AssetsLibrary/AssetsLibrary.h>苹果原生框架
第二种是iOS8推出的<Photos/Photos.h> 苹果原生框架,功能更多,但是只支持iOS8之后的版本
一般推荐使用<Photos/Photos.h> ,因为现在版本快iOS10了,有些版本没有升级的用户已经可以抛弃了
 1.添加图片到【相机胶卷】
 1> UIImageWriteToSavedPhotosAlbum函数
 2> AssetsLibrary框架
 3> Photos框架(推荐)
 
 2.拥有一个【自定义相册】
 1> AssetsLibrary框架
 2> Photos框架(推荐)
 
 3.将刚才添加到【相机胶卷】的图片,引用(添加)到【自定义相册】
 1> AssetsLibrary框架
 2> Photos框架(推荐)

One:

如果只需要将要将图片保存到系统的相册中,只需要下面两句代码就搞定了

#文档里已经说明这个方法是保存到用户的相机胶卷里面的
Adds the specified image to the user’s Camera Roll album.
- (void)saveBtns
{
//    /**
//     *  保存到相册的方法
//     *  @param image#>              需要保存的图片
//     *  @param completionTarget#>   哪个控制器 description#>
//     *  @param completionSelector#> 哪个方法 点击方法文档里面有提示 description#>
//     *  @param contextInfo#>        参数,传什么现实什么 description#>
//     *  @return
//     */
    UIImageWriteToSavedPhotosAlbum(self.imageView.image, self,
  @selector(image:didFinishSavingWithError:contextInfo:), @"传什么下面就调用什么什么");
}

// 需要实现下面的方法,或者传入三个参数即可
 - (void)image:(UIImage *)image didFinishSavingWithError:(NSError *)error contextInfo:(void *)contextInfo
{
    if (error) {
        [SVProgressHUD showErrorWithStatus:@"保存失败"];
    }else
    {
        [SVProgressHUD showSuccessWithStatus:@"保存成功"];
    }
}

#注意:  @selector() 方法一定要传入三个参数,不然图片虽然保存进去了,但是会崩了
因为参数越界
实参个数 > 形参个数, 就会出现参数越界
 -[NSInvocation setArgument:atIndex:]: index (2) out of bounds [-1, 1] : 

 

Two:

需要用到<AssetsLibrary/AssetsLibrary.h>这个库

官方文档给出的提示
//自定相册的类
#import <AssetsLibrary/AssetsLibrary.h>

static NSString *kGroupNameKey = @"nameKey";
static NSString *kDefaultGroupName = @"会跳舞的狮子";
/** 懒加载 只创建一次library对象 */
- (ALAssetsLibrary *)library
{
    if (!_library) {
        _library = [[ALAssetsLibrary alloc] init];
    }
    
    return _library;
}
- (NSString *)groupName
{
    //先从沙盒中取得名字
    NSString *groupName = [[NSUserDefaults standardUserDefaults] stringForKey:kGroupNameKey];
    if (groupName == nil) {//如果沙盒中取不到这个key(文件夹的名字)
        groupName = kDefaultGroupName;
        
        //储存名字到沙盒里面
        [[NSUserDefaults standardUserDefaults] setObject:groupName forKey:kGroupNameKey];
        //synchronize 立马将上面的key保存到沙盒里面
        [[NSUserDefaults standardUserDefaults]synchronize];
    }
    return groupName;
}

/**  点击保存按钮 */
- (IBAction)clickSave:(id)sender { //创建自定义相册
    
    //获得文件夹的名字 (调用从沙盒里面取的这个方法)
    __block NSString *groupName = [self groupName];
    //self的弱引用
    __weakSelf;
  //图片库
    __weak ALAssetsLibrary *weakLibrary = self.library;
    
    //创建自定义的相册
    [weakLibrary addAssetsGroupAlbumWithName:groupName resultBlock:^(ALAssetsGroup *group) {
        if (group) { //新创建的文件夹
            //直接把图片添加到文件夹中
            [weakSelf addImageToGroup:group];
        }else
        {
        //遍历相册中的每一个文件
        [weakLibrary enumerateGroupsWithTypes:ALAssetsGroupAll usingBlock:^(ALAssetsGroup *group, BOOL *stop) {
            //取出文件夹的名称
            NSString *name = [group valueForProperty:ALAssetsGroupPropertyName];
            if ([name isEqualToString:groupName]) { //如果相等,那么就是自己创建的文件夹
                //添加图片到文件夹中
                [weakSelf addImageToGroup:group];
                
                *stop = YES; // 将图片添加到了文件夹中就停止遍历
            }else if ([name isEqualToString:@"Camera Roll"])
            {
                //文件夹被用户强制删除了 (拼接一个空格,就是不同的文件夹的名字)
                groupName = [groupName stringByAppendingString:@" "];
            
                //储存新的名字
                [[NSUserDefaults standardUserDefaults] setObject:groupName forKey:kGroupNameKey];
                [[NSUserDefaults standardUserDefaults]synchronize];
                
                //创建新的文件夹中
                [weakLibrary addAssetsGroupAlbumWithName:groupName resultBlock:^(ALAssetsGroup *group) {
                    
                    //添加图片到文件夹中
                    [weakSelf addImageToGroup:group];
                    
                    
                } failureBlock:nil];
            }
        } failureBlock:nil];
        }
    } failureBlock:nil];
    
}

/** 添加一张图片到某个文件夹中 */
- (void)addImageToGroup:(ALAssetsGroup *)group
{
    __weak ALAssetsLibrary *weakLibrary = self.library;
    //需要保存的图片
    CGImageRef image = self.imageView.image.CGImage;

    //添加图片到相机胶卷 (因为要先存到全部的相片里面. 在根据对应的assetURL)
    [weakLibrary writeImageToSavedPhotosAlbum:image metadata:nil completionBlock:^(NSURL *assetURL, NSError *error) {
        // asset就是一张照片
        [weakLibrary assetForURL:assetURL resultBlock:^(ALAsset *asset) {
            //将照片保存到自定的文件夹中
            [group addAsset:asset];
            [SVProgressHUD showSuccessWithStatus:@"保存成功"];

        } failureBlock:nil];
 
    }];
    
}

Three:

用苹果iOS推出<Photos/Photos.h> 这个库
先简单的介绍下,然后再上代码

Photos框架须知
 1.PHAsset : 一个PHAsset对象代表一张图片或者一个视频文件
 1> 负责查询一堆的PHAsset对象
 
 2.PHAssetCollection : 一个PHAssetCollection对象代表一个相册
 1> 负责查询一堆的PHAssetCollection对象
 
 3.PHAssetChangeRequest
 1> 负责执行对PHAsset的【增删改】操作
 2> 这个类只能放在-[PHPhotoLibrary performChanges:completionHandler:] 或者 -[PHPhotoLibrary performChangesAndWait:error:]方法的block中使用
 
 4.PHAssetCollectionChangeRequest
 1> 负责执行对PHAssetCollection的【增删改】操作
 2> 这个类只能放在-[PHPhotoLibrary performChanges:completionHandler:] 或者 -[PHPhotoLibrary performChangesAndWait:error:]方法的block中使用

错误信息
 This method can only be called from inside of -[PHPhotoLibrary performChanges:completionHandler:] or -[PHPhotoLibrary performChangesAndWait:error:]
 
 解决防范
 // Asynchronously 异步执行操作
 [[PHPhotoLibrary sharedPhotoLibrary] performChanges:^{
    [PHAssetChangeRequest creationRequestForAssetFromImage:self.imageView.image];
 } completionHandler:^(BOOL success, NSError * _Nullable error) {
    JHLog(@"保存完毕")
 }];
 
 // Synchronously 同步执行操作
 NSError *error = nil;
 [[PHPhotoLibrary sharedPhotoLibrary] performChangesAndWait:^{
    [PHAssetChangeRequest creationRequestForAssetFromImage:self.imageView.image];
 } error:&error];
 
 // 这个方法只适合加载普通的plist文件,不能用来加载Info.plist
 [NSDictionary dictionaryWithContentsOfFile:[[NSBundle mainBundle] pathForResource:@"Test" ofType:@"plist"]];


安全提示按钮选择
    // PHAuthorizationStatusNotDetermined
    // 用户还没有对当前App授权过(用户从未点击过Don't Allow或者OK按钮)
    
    // PHAuthorizationStatusRestricted
    // 因为一些系统原因导致无法访问相册(比如家长控制)
    
    // PHAuthorizationStatusDenied
    // 用户已经明显拒绝过当前App访问相片数据(用户已经点击过Don't Allow按钮)
    
    // PHAuthorizationStatusAuthorized
    // 用户已经允许过当前App访问相片数据(用户已经点击过OK按钮)

#  只有遵守了NSFastEnumeration协议的对象才能使用for-in循环

封装好的代码,可以根据公司的需求进行修改
我创建新的相册的名字就是APP的名字,

// 获取软件的名字作为相册的标题
NSString *title = [NSBundle mainBundle].infoDictionary[(NSString *)kCFBundleNameKey];

定义专属相册
#import <Photos/Photos.h>
// 在interface中写方法的声明,是为了点语法有智能提示
- (PHFetchResult<PHAsset *> *)createdAssets;
- (PHAssetCollection *)createdCollection;



/**
 *  获得刚才添加到【相机胶卷】中的图片
 */
- (PHFetchResult<PHAsset *> *)createdAssets
{
    __block NSString *createdAssetId = nil;
    
    // 添加图片到【相机胶卷】
    [[PHPhotoLibrary sharedPhotoLibrary] performChangesAndWait:^{
        createdAssetId = [PHAssetChangeRequest creationRequestForAssetFromImage:self.imageView.image].placeholderForCreatedAsset.localIdentifier;
    } error:nil];
    
    if (createdAssetId == nil) return nil;
    
    // 在保存完毕后取出图片
    return [PHAsset fetchAssetsWithLocalIdentifiers:@[createdAssetId] options:nil];
}

/**
 *  获得【自定义相册】
 */
- (PHAssetCollection *)createdCollection
{
    // 获取软件的名字作为相册的标题
    NSString *title = [NSBundle mainBundle].infoDictionary[(NSString *)kCFBundleNameKey];
    
    // 获得所有的自定义相册
    PHFetchResult<PHAssetCollection *> *collections = [PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeAlbum subtype:PHAssetCollectionSubtypeAlbumRegular options:nil];
    for (PHAssetCollection *collection in collections) {
        if ([collection.localizedTitle isEqualToString:title]) {
            return collection;
        }
    }
    
    // 代码执行到这里,说明还没有自定义相册
    
    __block NSString *createdCollectionId = nil;
    
    // 创建一个新的相册
    [[PHPhotoLibrary sharedPhotoLibrary] performChangesAndWait:^{
        createdCollectionId = [PHAssetCollectionChangeRequest creationRequestForAssetCollectionWithTitle:title].placeholderForCreatedAssetCollection.localIdentifier;
    } error:nil];
    
    if (createdCollectionId == nil) return nil;
    
    // 创建完毕后再取出相册
    return [PHAssetCollection fetchAssetCollectionsWithLocalIdentifiers:@[createdCollectionId] options:nil].firstObject;
}

/**
 *  保存图片到相册
 */
- (void)saveImageIntoAlbum
{
    // 获得相片
    PHFetchResult<PHAsset *> *createdAssets = self.createdAssets;
    
    // 获得相册
    PHAssetCollection *createdCollection = self.createdCollection;
    
    if (createdAssets == nil || createdCollection == nil) {
        [SVProgressHUD showErrorWithStatus:@"保存失败!"];
        return;
    }
    
    // 将相片添加到相册
    NSError *error = nil;
    [[PHPhotoLibrary sharedPhotoLibrary] performChangesAndWait:^{
        PHAssetCollectionChangeRequest *request = [PHAssetCollectionChangeRequest changeRequestForAssetCollection:createdCollection];
        [request insertAssets:createdAssets atIndexes:[NSIndexSet indexSetWithIndex:0]];
    } error:&error];
    
    // 保存结果
    if (error) {
        [SVProgressHUD showErrorWithStatus:@"保存失败!"];
    } else {
        [SVProgressHUD showSuccessWithStatus:@"保存成功!"];
    }
}

- (IBAction)save {
    /*
     requestAuthorization方法的功能
     1.如果用户还没有做过选择,这个方法就会弹框让用户做出选择
     1> 用户做出选择以后才会回调block
     
     2.如果用户之前已经做过选择,这个方法就不会再弹框,直接回调block,传递现在的授权状态给block
     */
    
    PHAuthorizationStatus oldStatus = [PHPhotoLibrary authorizationStatus];
    
    [PHPhotoLibrary requestAuthorization:^(PHAuthorizationStatus status) {
        dispatch_async(dispatch_get_main_queue(), ^{
            switch (status) {
                case PHAuthorizationStatusAuthorized: {
                    //  保存图片到相册
                    [self saveImageIntoAlbum];
                    break;
                }
                    
                case PHAuthorizationStatusDenied: {
                    if (oldStatus == PHAuthorizationStatusNotDetermined) return;
                    
                    JHLog(@"提醒用户打开相册的访问开关")
                    break;
                }
                    
                case PHAuthorizationStatusRestricted: {
                    [SVProgressHUD showErrorWithStatus:@"因系统原因,无法访问相册!"];
                    break;
                }
                    
                default:
                    break;
            }
        });
    }];

分享知识,感谢简书平台

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,520评论 25 707
  • 两个人在一起总会有比较,心里的暗斗,无论是知识,家境。 她总是对我和别人不一样,她可以认真仔细的给另一个女孩子讲一...
    2e70bcbd39b5阅读 162评论 0 0
  • 文章只是简单的对视频处理(开始、暂停、快进、快退、声音、循环播放等)。音视频方法属性方法基本相同。想要实现更多操作...
    被时光移动的城阅读 978评论 0 1
  • 文/爱上时光的猫 第61天 在人生的路上,不断地发现并寻找更好的自己。 从年初参加呆萌写作营,迄今为止...
    木棉纪阅读 587评论 4 50
  • 今天是年三十,大家族的老老小小们照例中午一起吃年饭,给小辈的发了红包,给长辈们敬酒,照了全家福,吃完年饭分成...
    Catherin77阅读 295评论 0 0