美文网首页iOS 实用iOS Developer小知识点
自定义相机、相册(相册篇)

自定义相机、相册(相册篇)

作者: iccpeng | 来源:发表于2016-11-20 16:01 被阅读795次

    工作之余,研究了一下相机与相册的自定义,在这里整理成篇仅供参考学习,希望可以给大家带来些许帮助,也期待大家的批评指正。

    自定义相册

    GIF 示例:


    ccpcamera.gif

    DEMO下载地址:https://github.com/IMCCP/CCPCustomCamera

    在 iOS 设备中,照片是相当重要的一部分。在 iOS8.0之前,开发者只能使用 AssetsLibrary 框架来访问设备的照片库。而在 iOS8 之后,苹果提供了一个名为 PhotoKit 的框架,一个可以让应用更好地与设备照片库对接的框架.由于市面上有一部分应用还支持iOS7,同时为了更加全面的学习,在这里将整理AssetsLibrary 框架与 PhotoKit 框架的相关知识,供大家参考学习.

    一、AssetsLibrary 基本介绍

    AssetsLibrary: 代表整个设备中的资源库(照片库),通过 AssetsLibrary 可以获取和包括设备中的照片和视频

    ALAssetsGroup: 映射照片库中的一个相册,通过 ALAssetsGroup 可以获取某个相册的信息,相册下的资源,同时也可以对某个相册添加资源。

    ALAsset: 映射照片库中的一个照片或视频,通过 ALAsset 可以获取某个照片或视频的详细信息,或者保存照片和视频。

    ALAssetRepresentation: ALAssetRepresentation 是对 ALAsset 的封装(但不是其子类),可以更方便地获取 ALAsset 中的资源信息,每个 ALAsset 都有至少有一个 ALAssetRepresentation 对象,可以通过 defaultRepresentation 获取。

    二、PhotoKit 基本介绍

    PhotoKit 是一套比 AssetsLibrary 更完整也更高效的库,对资源的处理跟 AssetsLibrary 也有很大的不同。

    PhotoKit 基本构成的介绍:

    PHAsset: 代表照片库中的一个资源,跟 ALAsset 类似,通过 PHAsset 可以获取和保存资源
    PHFetchOptions: 获取资源时的参数,可以传 nil,即使用系统默认值
    PHAssetCollection: PHCollection 的子类,表示一个相册或者一个时刻,或者是一个「智能相册(系统提供的特定的一系列相册,例如:最近删除,视频列表,收藏等等,如下图所示)
    PHFetchResult: 表示一系列的资源结果集合,也可以是相册的集合,从PHCollection 的类方法中获得
    PHImageManager: 用于处理资源的加载,加载图片的过程带有缓存处理,可以通过传入一个 PHImageRequestOptions 控制资源的输出尺寸等规格
    PHImageRequestOptions: 如上面所说,控制加载图片时的一系列参数

    三、主要功能

    1.获取相册图片资源;
    2.自定义相册功能;
    3.图片浏览器功能;
    4.主要 工具类 介绍:

    a.CCPPhotoAlbumViewController 控制器 主要要来进行相册资源的读取;

    - (void)viewDidLoad {
        [super viewDidLoad];
        [self makeUI];
        //进行系统型号的判断,调用不同的图库处理框架
        if ([self iOSIsbefore_iOS8]) {
            [self iOSBefore_iOS8];
        } else {
            [self iOSAfter_iOS8];
        }
    }
    
    //系统版本小于8.0
    - (void) iOSBefore_iOS8 {
        //提示
        NSString *tipTitle = nil;
        //相册的访问状态
        ALAuthorizationStatus authorizationStatus = [ALAssetsLibrary authorizationStatus];
        /* 获取当前应用对照片的访问授权状态
         ALAuthorizationStatusNotDetermined = 0, // 用户还没有做出选择这个应用程序的问候
         ALAuthorizationStatusRestricted,        // 这个应用程序没有被授权访问照片数据。当前用户不能改变应用程序的状态,是受限制的。如家长控制权限
         ALAuthorizationStatusDenied,            // 用户已拒绝该应用程序访问照片数据
         ALAuthorizationStatusAuthorized         // 用户已授权该应用可以访问
         */
        
        // 如果没有获取访问授权,或者访问授权状态已经被明确禁止,则显示提示语,引导用户开启授权
        if (authorizationStatus == ALAuthorizationStatusRestricted || authorizationStatus == ALAuthorizationStatusDenied) {
            NSDictionary *mainInfoDictionary = [[NSBundle mainBundle] infoDictionary];
            NSString *myAppName = [mainInfoDictionary objectForKey:@"CFBundleDisplayName"];
            tipTitle = [NSString stringWithFormat:@"请在设备的\"设置-隐私-照片\"选项中,允许%@访问你的手机相册", myAppName];
        } else {
            
            [self.assetsLibrary enumerateGroupsWithTypes:ALAssetsGroupAll usingBlock:^(ALAssetsGroup *group, BOOL *stop) {
                
                if (group) {
                    [group setAssetsFilter:[ALAssetsFilter allPhotos]];
                    if (group.numberOfAssets > 0) {
                        // 把相册储存到数组中,方便后面展示相册时使用
                        [self.assetsArray addObject:group];
                        [self.nameArray addObject:[group valueForProperty:ALAssetsGroupPropertyName]];
                        //获取相册封面图
                        UIImage *posterImage =  [UIImage imageWithCGImage:[group posterImage]];
                        [self.posterImageArray addObject:posterImage];
                        
                    }
                    
                } else {
                    
                    if ([self.assetsArray count] > 0) {
                        // 把所有的相册储存完毕,可以展示相册列表
                    } else {
                        // 没有任何有资源的相册,输出提示
                    }
                }
                
                
            } failureBlock:^(NSError *error) {
                
                NSLog(@"Asset group not found!\n");
                
            }];
            
            dispatch_async(dispatch_get_main_queue(), ^{
                [self.showTableView reloadData];
            });
            
        }
    }
    
    //系统版本号大于等于8.0
    - (void) iOSAfter_iOS8 {
        /*
         PHAssetCollectionTypeAlbum      自建相册
         PHAssetCollectionTypeSmartAlbum 智能相册
         PHAssetCollectionTypeMoment     时刻相册
         
         智能相册子类型
         PHAssetCollectionSubtypeSmartAlbumGeneric    通用的
         PHAssetCollectionSubtypeSmartAlbumPanoramas  全景
         PHAssetCollectionSubtypeSmartAlbumVideos     视屏
         PHAssetCollectionSubtypeSmartAlbumFavorites  收藏
         PHAssetCollectionSubtypeSmartAlbumTimelapses 延时视屏,也会在PHAssetCollectionSubtypeSmartAlbumVideos在出现
         PHAssetCollectionSubtypeSmartAlbumAllHidden  隐藏的
         PHAssetCollectionSubtypeSmartAlbumRecentlyAdded 最近添加
         PHAssetCollectionSubtypeSmartAlbumBursts    连拍
         PHAssetCollectionSubtypeSmartAlbumSlomoVideos Slomo是slow motion的缩写,高速摄影慢动作解析
         PHAssetCollectionSubtypeSmartAlbumUserLibrary 用户所有的资源
         PHAssetCollectionSubtypeSmartAlbumSelfPortraits 所有前置摄像头拍的照片和视屏
         PHAssetCollectionSubtypeSmartAlbumScreenshots 所有的截屏图
         */
        //权限
        PHAuthorizationStatus status = [PHPhotoLibrary authorizationStatus];
        if (status == PHAuthorizationStatusDenied || status == PHAuthorizationStatusRestricted) {
            [self noticeAlerPhotos];
        }else{
            [PHPhotoLibrary requestAuthorization:^(PHAuthorizationStatus status) {
                if (status == PHAuthorizationStatusAuthorized) {
                    
                    PHFetchOptions *option = [[PHFetchOptions alloc] init];
                    //排序方式
                    option.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"modificationDate" ascending:NO]];
                    // 列出所有相册智能相册
                    PHFetchResult *smartAlbums = [PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeSmartAlbum subtype:PHAssetCollectionSubtypeAlbumRegular options:nil];
                    
                    for (NSInteger i = 0; i < smartAlbums.count; i++) {
                        // 获取一个相册(PHAssetCollection)
                        PHCollection *collection = smartAlbums[i];
                        if ([collection isKindOfClass:[PHAssetCollection class]]) {
                            PHAssetCollection *assetCollection = (PHAssetCollection *)collection;
                            // 从每一个智能相册中获取到的 PHFetchResult 中包含的才是真正的资源(PHAsset)
                            PHFetchResult *fetchResult = [PHAsset fetchAssetsInAssetCollection:assetCollection options:nil];
                            
                            if (fetchResult.count > 0) {
                                
                                [self.assetsArray addObject:fetchResult];
                                
                                [self.nameArray addObject:assetCollection.localizedTitle];
                                //获取封面图片,就是第一张图片
                                PHAsset *asset = (PHAsset *)fetchResult.firstObject;
                                PHImageRequestOptions *options = [[PHImageRequestOptions alloc] init];
                                //默认的是异步加载,这里选择了同步 
                                options.synchronous = YES;
                               //PHImageManagerMaximumSize:获取原图,占用很大内存 建议不要使用
                                [[PHImageManager defaultManager] requestImageForAsset:asset targetSize:PHImageManagerMaximumSize contentMode:PHImageContentModeAspectFill options:options resultHandler:^(UIImage * _Nullable result, NSDictionary * _Nullable info)
                                 {
                                     [self.posterImageArray addObject:result];
                                 }];
                            }
                        }else {
                            
                            NSAssert(NO, @"Fetch collection not PHCollection: %@", collection);
                        }
                        
                        dispatch_async(dispatch_get_main_queue(), ^{
                            [self.showTableView reloadData];
                        });
                    }
                }
            }];
            
        }
        
        // 获取所有资源的集合,并按资源的创建时间排序
        //    PHFetchOptions *options = [[PHFetchOptions alloc] init];
        //    options.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"creationDate" ascending:YES]];
        //    PHFetchResult *assetsFetchResults = [PHAsset fetchAssetsWithOptions:options];
        //
        //    // 这时 assetsFetchResults 中包含的,应该就是各个资源(PHAsset)
        //    for (NSInteger i = 0; i < assetsFetchResults.count; i++) {
        //        // 获取一个资源(PHAsset)
        //        PHAsset *asset = assetsFetchResults[i];
        //    }
        //
        //    NSLog(@"%@",assetsFetchResults);
        
    }
    

    b.CCPShowPhotoVC 控制器 主要用于获取到的对应照片的展示
    如图:

    CCPShowPhotoVC.jpg

    c.XLPhotoBrowser 主要用于图片的浏览
    本着不重复造轮子的原则,demo中图片浏览器使用了 XLPhotoBrowser
    XLPhotoBrowser下载地址:https://github.com/Shannoon/XLPhotoBrowser

    在这里对框架作者表示感谢!

    四、参考:

    a.http://kayosite.com/ios-development-and-detail-of-photo-framework.html

    b.[http://www.jianshu.com/p/535bfe3c328f](http://www.jianshu.com/p/535bfe3c3

    c.http://www.jianshu.com/p/cc85282fac5e

    d. http://www.cnblogs.com/Jenaral/p/5580497.html

    在这里对blog作者表示感谢!

    由于篇幅有限,在这里不再详细的展开,demo中对一些坑点都做了详细的注释,如果您有需要可下载demo进行查看.

    感谢您的阅读,期待您的 Star,如果在使用中您有任何问题,可以在 github issue,我会尽自己能力给您答复 。

    相关文章

      网友评论

        本文标题:自定义相机、相册(相册篇)

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