咱们公司作了一个DLNA的投屏软件,可是iOS是不能跨应用访问数据的,因此对于局域网投屏视频和图片须要把图片或者视频写入到应用的沙盒路径下。
在我以前的前辈用的是AssetsLibrary,他是在进入界面以前写入,等到彻底都写完了才会去显示。以前拍照的照片大小不是很大,并且手机的存储空间也不大,对于用户来讲这么处理彻底是没有问题的。可是,后来有用户反馈说在本地媒体界面一直都有那个“菊花转”。后来咱们发现多是用户的本地媒体数据过于大,却是程序假死。
咱们老大说,你把这个功能优化一下,目标就是像微信那样是最好的。后来咱们就选用的PhotoKit这个框架。git
以前并无具体用过这个框架,因此就如今网上研究了一下。很幸运,我找到了一个很相似的DEMO。
咱们只要在相应的控制器界面导入#import <Photos/Photos.h>
就行。咱们还须要在info.plist中添加相册的访问权限。
github
当咱们添加受权以后再相应的界面就会出现这么一个弹框
数组
咱们须要对这个弹框的点击事件进行处理,这里咱们直接上代码:微信
- (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
须要设置一下,代码以下:async
- (PHFetchOptions *)options {
if (!_options) {
_options = [[PHFetchOptions alloc] init];
_options.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"creationDate" ascending:YES]];
}
return _options;
}复制代码
在上面获取到的相册的名称,我在控制台打出来的都是英文的
ide
依旧是在info.plist
中,以下图:
fetch
显示图片,这里用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;
}];复制代码
这里显示的只是一个缩略图,并非很清晰,若是放到满屏看就会很模糊,那么还有另外一种方式去得到清晰的图片:ui
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];
}];复制代码
就是这个方法的等待时间比较长,用户体验不太好。
这个功能出现了有一段时间,我知道的软件只有微博支持了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);复制代码
首先LivePhoto应该是属于PHAssetMediaTypeImage
的,而后是属于PHAssetMediaSubtypePhotoLive
的。
正常的理解就是这样对不对,可是在个人LivePhoto的相册中有的居然会识别不出来。我去了本地媒体相册看了一下识别不出来的那些照片的类型,左上角居然有两个标识,一个是Live
,另外一个是HDR
。
后来我就是把全部的类型都打印了一下,我发现这样的相片不属于任何一个类型。
我在控制器打了一下看了一会儿媒体类型的值,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就会自动播放一次。
我先经过上面的方法打印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