PhotoKit初用

简介: 我们公司做了一个DLNA的投屏软件,但是iOS是不能跨应用访问数据的,所以对于局域网投屏视频和图片需要把图片或者视频写入到应用的沙盒路径下。在我之前的前辈用的是AssetsLibrary,他是在进入界面之前写入,等到完全都写完了才会去显示。之前拍照的照片大小不是很大,而且手机的存储空间也不大,对于用户来说这么处理完全是没有问题的。但是,后来有用户反馈说在本地媒体界面一直都有那个“菊花转”。后来我们发现可能是用户的本地媒体数据过于大,倒是程序假死。我们老大说,你把这个功能优化一下,目标就是像微信那样是最好的。后来我们就选用的PhotoKit这个框架。

前言


我们公司做了一个DLNA的投屏软件,但是iOS是不能跨应用访问数据的,所以对于局域网投屏视频和图片需要把图片或者视频写入到应用的沙盒路径下。
在我之前的前辈用的是AssetsLibrary,他是在进入界面之前写入,等到完全都写完了才会去显示。之前拍照的照片大小不是很大,而且手机的存储空间也不大,对于用户来说这么处理完全是没有问题的。但是,后来有用户反馈说在本地媒体界面一直都有那个“菊花转”。后来我们发现可能是用户的本地媒体数据过于大,倒是程序假死。
我们老大说,你把这个功能优化一下,目标就是像微信那样是最好的。后来我们就选用的PhotoKit这个框架。


研究


之前并没有具体用过这个框架,所以就现在网上研究了一下。很幸运,我找到了一个很类似的DEMO。
我们只要在相应的控制器界面导入#import <Photos/Photos.h>就行。我们还需要在info.plist中添加相册的访问权限。
之后,我们建立一个二级控制器就可以了。


初用


弯路1:

当我们添加授权之后再相应的界面就会出现这么一个弹框
当我点击好的时候 界面并没有数据,但是当我第二期进入界面的时候数据就显示出来了。

解决1:

我们需要对这个弹框的点击事件进行处理,这里我们直接上代码:

- (void)getAuthorized{
    //判断是否有访问权限
    PHAuthorizationStatus status = [PHPhotoLibrary authorizationStatus];
    //还没有去做选择
    if (status == PHAuthorizationStatusNotDetermined) {
        [PHPhotoLibrary requestAuthorization:^(PHAuthorizationStatus status) {
            //已经授权
            if (status == PHAuthorizationStatusAuthorized) {
                dispatch_async(dispatch_get_main_queue(), ^{
              //已经授权,显示相册 或者图片
                });
            }else{
                //做一个没有授权的提示
            }
        }];
    }
    //已经授权
    else if (status == PHAuthorizationStatusAuthorized){
        dispatch_async(dispatch_get_main_queue(), ^{
          //已经授权,显示相册 或者图片
        });
    }
    //拒绝访问
    else if (status == PHAuthorizationStatusRestricted){
        dispatch_async(dispatch_get_main_queue(), ^{
        //做一个没有授权的提示
        });
    }
}

那么接了下来我们继续。
这里我们先去做了获取相册的功能:

- (void)getAllAlbums{
    PHFetchResult *smartAlbums = [PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeSmartAlbum subtype:PHAssetCollectionSubtypeAlbumRegular options:nil];
    for (PHCollection *collection in smartAlbums) {
        if ([collection isKindOfClass:[PHCollection class]]) {
            PHAssetCollection *assetCollection = (PHAssetCollection *)collection;

            switch (assetCollection.assetCollectionSubtype) {
                case PHAssetCollectionSubtypeSmartAlbumAllHidden:
                    break;
                case PHAssetCollectionSubtypeSmartAlbumUserLibrary:{
                    PHFetchResult *assetFetchResult = [PHAsset fetchAssetsInAssetCollection:assetCollection options:self.options];
                    [self.smartFetchResultArray insertObject:assetFetchResult atIndex:0];
                    [self.smartFetchResultTitlt insertObject:collection.localizedTitle atIndex:0];
                }
                    break;
                default:{
                    PHFetchResult *assetFetchResult = [PHAsset fetchAssetsInAssetCollection:assetCollection options:self.options];
                    [self.smartFetchResultTitlt addObject:collection.localizedTitle];
                    [self.smartFetchResultArray addObject:assetFetchResult];
                }
                    break;
            }
        }
    }
}

这里的self.smartFetchResultTitlt是用来存储相册标题的数组;self.smartFetchResultArray是用来存储相册内容的;self.options需要设置一下,代码如下:

- (PHFetchOptions *)options {
    if (!_options) {
        _options = [[PHFetchOptions alloc] init];
        _options.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"creationDate" ascending:YES]];
    }
    return _options;
}
弯路2:

在上面获取到的相册的名称,我在控制台打出来的都是英文的

但是我们想要的并不是英文的。

解决2:

依旧是在info.plist中:

显示图片,这里用PhototKit自身的PHCachingImageManager做显示就可以了。
我是用了一个collectionView做显示。方法如下:

    [self.imageManager requestImageForAsset:asset targetSize:CGSizeMake(self.bounds.size.width, self.bounds.size.width) contentMode:PHImageContentModeAspectFit options:nil resultHandler:^(UIImage * _Nullable result, NSDictionary * _Nullable info) {
        weakSelf.imageView.image = result;
    }];

这里显示的只是一个缩略图,并不是很清晰,如果放到满屏看就会很模糊,那么还有另一种方式去获得清晰的图片:

WeakSelf(weakSelf);
    [[PHImageManager defaultManager]requestImageDataForAsset:self.asset                                                             options:nil
                                               resultHandler:^(NSData * _Nullable imageData, NSString * _Nullable dataUTI, UIImageOrientation orientation, NSDictionary * _Nullable info) {

                                                   UIImage *selectedImage = [UIImage imageWithData:imageData];
                                                   weakSelf.imageView.image = selectedImage;
                                                   weakSelf.title = [[info objectForKey:@"PHImageFileURLKey"] lastPathComponent];
                                               }];

这样不仅获得了图片,还可以获得这个照片的名字。


上面说的都是图片,接下来说说视频。

我么可以用下面这个方法去获取视频:

PHVideoRequestOptions *phVideoRequestOptions = [[PHVideoRequestOptions alloc]init];
        phVideoRequestOptions.version = PHImageRequestOptionsVersionCurrent;
        phVideoRequestOptions.deliveryMode = PHVideoRequestOptionsDeliveryModeAutomatic;
        PHImageManager *manager = [PHImageManager defaultManager];
        [manager requestAVAssetForVideo:asset options:phVideoRequestOptions resultHandler:^(AVAsset * _Nullable asset, AVAudioMix * _Nullable audioMix, NSDictionary * _Nullable info) {
            ShowVIdeoViewController *shouwVideoVC = [[ShowVIdeoViewController alloc]init];
            shouwVideoVC.asset = asset;
            shouwVideoVC.fileName =[[info objectForKey:@"PHImageFileSandboxExtensionTokenKey"] lastPathComponent];
            [weakSelf.navigationController pushViewController:shouwVideoVC animated:YES];
        }];
弯路3:

就是这个方法的等待时间比较长,用户体验不太好。


Live Photo

这个功能出现了有一段时间,我知道的软件只有微博支持了Live Photo的功能(小人可能见识短浅,有盆友知到别的软件也支持的话可以 私聊告诉我),我看PhotoKit支持Live Photo,那么我就小小的研究了一下。
但是在这个过程中我也是遇到了一些问题的,
我们看源码中给本地媒体分了很多种类型:

typedef NS_ENUM(NSInteger, PHAssetMediaType) {
    PHAssetMediaTypeUnknown = 0,
    PHAssetMediaTypeImage   = 1,
    PHAssetMediaTypeVideo   = 2,
    PHAssetMediaTypeAudio   = 3,
} PHOTOS_ENUM_AVAILABLE_IOS_TVOS(8_0, 10_0);

typedef NS_OPTIONS(NSUInteger, PHAssetMediaSubtype) {
    PHAssetMediaSubtypeNone               = 0,

    // Photo subtypes
    PHAssetMediaSubtypePhotoPanorama      = (1UL << 0),
    PHAssetMediaSubtypePhotoHDR           = (1UL << 1),
    PHAssetMediaSubtypePhotoScreenshot PHOTOS_AVAILABLE_IOS_TVOS(9_0, 10_0) = (1UL << 2),
    PHAssetMediaSubtypePhotoLive PHOTOS_AVAILABLE_IOS_TVOS(9_1, 10_0) = (1UL << 3),
    PHAssetMediaSubtypePhotoDepthEffect PHOTOS_AVAILABLE_IOS_TVOS(10_2, 10_1) = (1UL << 4),

    // Video subtypes
    PHAssetMediaSubtypeVideoStreamed      = (1UL << 16),
    PHAssetMediaSubtypeVideoHighFrameRate = (1UL << 17),
    PHAssetMediaSubtypeVideoTimelapse     = (1UL << 18),
} PHOTOS_AVAILABLE_IOS_TVOS(8_0, 10_0);
弯路4:

首先LivePhoto应该是属于PHAssetMediaTypeImage的,然后是属于PHAssetMediaSubtypePhotoLive的。
正常的理解就是这样对不对,但是在我的LivePhoto的相册中有的竟然会识别不出来。我去了本地媒体相册看了一下识别不出来的那些照片的类型,左上角竟然有两个标识,一个是Live,另一个是HDR
后来我就是把所有的类型都打印了一下,我发现这样的相片不属于任何一个类型。

解决4:

我在控制器打了一下看了一下子媒体类型的值,LivePhoto的类型值是8,既是LivePhoto又是HDR的类型值是10。我之前的媒体类型判断是:

asset.mediaSubtypes == PHAssetMediaSubtypePhotoLive

而改成:

asset.mediaSubtypes >= PHAssetMediaSubtypePhotoLive

就可以了


下面来说我们怎么显示Live Photo:

PHLivePhotoRequestOptions *options = [[PHLivePhotoRequestOptions alloc]init];
    options.deliveryMode = PHImageRequestOptionsDeliveryModeHighQualityFormat;
    options.networkAccessAllowed = YES;
    options.progressHandler = ^(double progress, NSError * _Nullable error, BOOL * _Nonnull stop, NSDictionary * _Nullable info) {
        NSLog(@"progress = %f",progress);
    };
    [[PHImageManager defaultManager]requestLivePhotoForAsset:self.asset targetSize:self.livePhotoView.bounds.size contentMode:PHImageContentModeAspectFill options:options resultHandler:^(PHLivePhoto * _Nullable livePhoto, NSDictionary * _Nullable info) {
        self.livePhotoView.livePhoto = livePhoto;
        NSLog(@"info = %@",info);
    }];

这里的self.livephotoview 是PHLivePhotoView这个类 ,就和普通的View初始化一样。
这样你现实出来的LivePhoto就可以了,这时你只要按住照片,照片就会动起来,这里你也可以设置你的播放设置,我是这样设置的:

 [self.livePhotoView startPlaybackWithStyle:PHLivePhotoViewPlaybackStyleHint];

这样设置的效果就是进来界面,livePhoto就会自动播放一次。

弯路5:

我先通过上面的方法打印info里面的信息,但是控制台给我这样的信息:

error reading settings archive file: <ISRootSettings: /var/mobile/Containers/Data/Application/BCAA7EBA-543E-4B9E-B945-D8C4C509C491/Documents/com.C4ibD3.PhotoKitDemo.settings/ISRootSettings_10.plist>
2017-06-16 09:00:16.433082+0800 PhotoKitDemo[1444:307243] info = {
}

不知道是为什么?


传送门

github:PhotoKitDemo

相关文章
|
4月前
源码拾贝三则
文章分享了三则源码示例,包括:1) 枚举类型的新型使用方式;2) Eigen库中LDLT分解的实现;3) Eigen库中访问者模式的应用。
|
7月前
|
SQL 前端开发 Java
2024考古之还在用原始JDBC开发 手搓 案例 实现一个模块的增删改
2024考古之还在用原始JDBC开发 手搓 案例 实现一个模块的增删改
46 0
|
8月前
|
SQL 前端开发 JavaScript
终于弄懂了Layui表格重载数据(中途被女朋友劝反附聊天记录)
终于弄懂了Layui表格重载数据(中途被女朋友劝反附聊天记录)
183 0
|
安全 测试技术
不会写测试用例咋办?牢记这5点,你也能写出高逼格案例
不会写测试用例咋办?牢记这5点,你也能写出高逼格案例
167 1
|
存储 算法 iOS开发
ZPhotoBrowser (基于之前的那篇文章PhotoKit初用)
作为一个iOS开发人员,我已经不知不觉的在帝都这个地方上干了块两年了。前一阵,由于公司的发展方向的问题,我被迫加入了找工作的大军之中。这可把我担心坏了,因为我之前的一个同事找了好久都没有找到工作,之后他就选择回老家发展了。做过iOS这行的都知道现在是什么行情了,我就不多说了。不过还好我找了一周左右吧,面试不少。但是现在招人的公司,真的不知道要招什么样的开发人员,面试草草了事的偏多。还有就是自认为大牛的比较多,我记得我面了一家智能家居的科技公司。那个面试我的面试官,看了我做过的产品。就给我说了一句:“你就是个调接口和写TableView的啊。”这话听起来真的让人难受,用一句很流行的话说,那就是“
|
Java 数据库连接 Spring
参数校验别再写满屏的 if/else 了,差点被劝退……(上)
参数校验别再写满屏的 if/else 了,差点被劝退…(上)
131 0
参数校验别再写满屏的 if/else 了,差点被劝退……(上)
参数校验别再写满屏的 if/else 了,差点被劝退……(下)
参数校验别再写满屏的 if/else 了,差点被劝退……(下)
119 0
|
Go 数据安全/隐私保护 开发者
流程控制课后练习说明 | 学习笔记
简介:快速学习流程控制课后练习说明
|
设计模式 架构师 Java
为什么有些蛮厉害的人,后来都不咋样了
写这篇文章目的是之前在一篇文章中谈到,我实习那会有个老哥很牛皮,业务能力嘎嘎厉害,但是后面发展一般般,这引起我的思考,最近有个同事发了篇腾讯pcg的同学关于review 相关的文章,里面也谈到架构师的层次,也再次引起我关于架构师的相关思考,接下来我们展开聊聊吧~
161 0
|
Android开发
被后台杀死后,Android应用如何重新走闪屏逻辑
被后台杀死后,Android应用如何重新走闪屏逻辑
620 0
被后台杀死后,Android应用如何重新走闪屏逻辑