内容简介:iOS 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
。
后来我就是把所有的类型都打印了一下,我发现这样的相片不属于任何一个类型。
如果有大神知道怎么解决也可以告诉小弟我。
下面来说我们怎么显示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
以上所述就是小编给大家介绍的《iOS PhotoKit 初用》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!
猜你喜欢:本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
Spring实战(第4版)
Craig Walls 沃尔斯 / 张卫滨 / 人民邮电出版社 / 2016-4-1 / CNY 89.00
《Spring实战(第4版)》是经典的、畅销的Spring学习和实践指南。 第4版针对Spring 4进行了全面更新。全书分为四部分。第1部分介绍Spring框架的核心知识。第二部分在此基础上介绍了如何使用Spring构建Web应用程序。第三部分告别前端,介绍了如何在应用程序的后端使用Spring。第四部分描述了如何使用Spring与其他的应用和服务进行集成。 《Spring实战(第4......一起来看看 《Spring实战(第4版)》 这本书的介绍吧!