关于Photos库的简单应用,筛选、获取、GIF、livePho
写在最开始:
由于本项目中对UI还原度要求较高,交互要求完全还原,用别人封装的改起来总归是有些别扭;为了后续方便自己实现定制UI交互等,决定自己从系统API开始封装一套相册资源选择器。
而AL用起来则到处报被弃用的⚠️,想逼死我这个强迫症啊~
然后就选择了PH,总的来说和AL比较类似,但是很多东西实现起来却是缺这少那的。虽说磨了有段时间,但目前用起来效果&性能尚可;下边分享点小坑和核心代码。
-
一、简单实现拉取所有相册资源(照片视频等),并保持创建时间排序
-
二、筛选大小,剔除不符合规则视频
-
三、gif区分、处理以及一些猜测(希望有朋友能帮忙验证最后的猜测)
-
四、网络GIF图存储到相册
一、 其实一开始就遇到了个问题,不能同时获取照片和视频。。。
最后为分别拉出加到同一个数组之后,进行按时间排序🤡。
NSMutableArray<PHAsset *> *assets = [NSMutableArray array];
PHFetchOptions *option = [[PHFetchOptions alloc] init];
//ascending 为YES时,按照照片的创建时间升序排列;为NO时,则降序排列
option.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"creationDate" ascending:YES]];
PHFetchResult *result = [PHAsset fetchAssetsWithMediaType:PHAssetMediaTypeImage options:option];
WS(ws);
[result enumerateObjectsWithOptions:NSEnumerationReverse usingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
PHAsset *asset = (PHAsset *)obj;
[ws.allAssetItemModelArr addObject:[YTMediaSelectorItemModel initWithAssetType:1 itemAsset:asset withAssetLocalIdentifier:asset.localIdentifier]];
[assets addObject:asset];
}];
PHFetchOptions *option = [[PHFetchOptions alloc] init];
option.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"creationDate" ascending:YES]];
PHFetchResult *result = [PHAsset fetchAssetsWithMediaType:PHAssetMediaTypeVideo options:option];
NSLog(@"拉取到 %ld 个视频资源\n", result.count);
WS(ws);
[result enumerateObjectsWithOptions:NSEnumerationReverse usingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
PHAsset *asset = (PHAsset *)obj;
[ws.allAssetItemModelArr addObject:[YTMediaSelectorItemModel initWithAssetType:2 itemAsset:asset withAssetLocalIdentifier:asset.localIdentifier]];
}];
- (NSArray *)comparaAssetsModelArr:(NSMutableArray <YTMediaSelectorItemModel *>*)assetModels{
NSComparator cmptr = ^(YTMediaSelectorItemModel *obj1, YTMediaSelectorItemModel *obj2){
if ([obj1.itemAsset.creationDate timeIntervalSince1970] < [obj2.itemAsset.creationDate timeIntervalSince1970]) {
return (NSComparisonResult)NSOrderedDescending;
}
if ([obj1.itemAsset.creationDate timeIntervalSince1970] > [obj2.itemAsset.creationDate timeIntervalSince1970]) {
return (NSComparisonResult)NSOrderedAscending;
}
return (NSComparisonResult)NSOrderedSame;
};
self.allAssetItemModelArr = [[assetModels sortedArrayUsingComparator:cmptr] mutableCopy];
return self.allAssetItemModelArr;
}
YTMediaSelectorItemModel为自己包装的对象,方便本地做标记和取用。
@interface YTMediaSelectorItemModel : NSObject
/// 1 image 2 video 3 gif
@property (nonatomic, assign) NSInteger assetType;
@property (nonatomic, strong) PHAsset *itemAsset;
@property (nonatomic, copy) NSString *astLocalIdentifier;
@property (nonatomic, strong) UIImage *thumbnailImg;
/** 是否被用户勾选 */
@property (nonatomic, assign) BOOL isSelected;
/** 当前元素 被选中之后的标号 */
@property (nonatomic, copy) NSString *currentItemSelectedFlage;
@property (nonatomic, assign) long long videoTimeLength;
+ (YTMediaSelectorItemModel *)initWithAssetType:(NSInteger)assetType itemAsset:(PHAsset *)asset withAssetLocalIdentifier:(NSString *)localIdentifier;
@end
二、 这么搞完一套,就取出了所有的资源,但是若想在构造YTMediaSelectorItemModel时再做些筛选的话,就会碰到各种问题了😄。
- 比如想筛选视频大小,限制一个范围的时候
此时我们肯定会去看下Asset里有无fileSize之类的属性,我看完伤心了,没有。。。
PHAsset头文件
也就是说无法直接取其属性进行计算比对了,多番查找发现有个PHAssetResource类,ta有fileSize的私有属性:
/*
PHAssetResource:
type:
uti:
filename:
asset:
locallyAvailable:
fileURL:
width:
height:
fileSize: 1924730
analysisType: never-download
cplResourceType: Original
isCurrent: YES
isInCloud: NO
*/
继续查找怎么获取这个asset的Resource有如此一行代码:
[[PHAssetResource assetResourcesForAsset:asset] firstObject]
此时我们 valueForKey一下就拿到了想要的数据,暂时实现了既定需求。
如此就结束了吗???没这么简单。。。
打印log发现,每执行100次左右的assetResourcesForAsset:就会花大概一秒的时间!!!😅尝试有无别的方式获取fileSize无果,我妥协了、向现实低了头,我做了当需要筛选fileSize的时候进行分批回调。。。
__block int count = 0;
[result enumerateObjectsWithOptions:NSEnumerationReverse usingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
count ++;
if (count > 0 && count % 50 == 0) {
NSLog(@"在筛选fileSize时,分批回调,先回调第%d批\n",count / 50);
if (ws.ytSelectorTakedAllMediaBlock) {
ws.ytSelectorTakedAllMediaBlock(ws.allAssetItemModelArr);
}
}
}];
如此,筛选fileSize时也能有类似秒开秒显的感觉了,勉强算是实现需求吧~~~
以上是19年写的代码,感觉没啥东西好说;而今(2021.03.17)需要对GIF做些处理,查找GIF相关知识点时发现都是东拉西扯,基本没有关于PHAsset拿到之后构造显示数据源时就做好标记进行区分的文章,尝试自己搞搞~~~并分享下吧,希望能帮助到一些后来者.
三、 下面开始正经的说下怎么通过PHAsset区分GIF、livePhoto等
先说livePhoto、HDR、截图之类,有对应的mediaSubtypes字段返回一个枚举值:
typedef NS_OPTIONS(NSUInteger, PHAssetMediaSubtype) {
PHAssetMediaSubtypeNone = 0,
// Photo subtypes
PHAssetMediaSubtypePhotoPanorama = (1UL << 0),
PHAssetMediaSubtypePhotoHDR = (1UL << 1),
PHAssetMediaSubtypePhotoScreenshot API_AVAILABLE(ios(9)) = (1UL << 2),
PHAssetMediaSubtypePhotoLive API_AVAILABLE(ios(9.1)) = (1UL << 3),
PHAssetMediaSubtypePhotoDepthEffect API_AVAILABLE(macos(10.12.2), ios(10.2), tvos(10.1)) = (1UL << 4),
// Video subtypes
PHAssetMediaSubtypeVideoStreamed = (1UL << 16),
PHAssetMediaSubtypeVideoHighFrameRate = (1UL << 17),
PHAssetMediaSubtypeVideoTimelapse = (1UL << 18),
};
UL » 无符号long类型
1.下面捋捋对应关系:
None = 0,
全景 = (1UL << 0) 1,
HDR = (1UL << 1) 2,
截图 = (1UL << 2) 4,
实况照片(livePhoto) = (1UL << 3) 8,
景深效果(DepthEffect) = (1UL << 4) 16,
**** = 32,
gif = 64,
视频的 1UL << 16 开始,我不得不猜测可以用这个mediaSubtypes == 64来判断是否为gif。当然,这个办法只经过我一百来张三个渠道的GIF测试,可能会有遗漏之类(期待大家的验证,欢迎验证通过的朋友在评论区留下一笔)。
- 判断PHAsset是否GIF第二种:
[[asset valueForKey:@"filename"] hasSuffix:@"GIF"]
判断后缀是否为GIF / gif。也通过了我那一百多张GIF的测试。
- 取PHAsset私有属性uniformTypeIdentifier,和第二种类似,但似乎更靠谱一点
[[asset valueForKey:@"uniformTypeIdentifier"] isEqual:@"com.compuserve.gif"]
- 判断PHAsset是否GIF第四种:
[[PHAssetResource assetResourcesForAsset:asset].firstObject.uniformTypeIdentifier isEqualToString:@"com.compuserve.gif"]
也是我认为最正经的一种办法,但是又面临一个执行耗时的问题。也是100次耗时1秒左右!!!
四种方式:1 2 3都不影响速度,4看着最正经但是费时。。。最后我选择了第三种方式😄
期待用了第一种方式的朋友和我互通有无问题
四、 关于存储GIF动图到相册
最常见的把image存入相册的操作莫过于 UIImageWriteToSavedPhotosAlbum
这个API了吧?然鹅、GIF图通过SDWebImage的loadImageWithURL获取到的为image对象,直接writeToSavedPhotosAlbum将保存一张静态图。怎么办呢?去找了下PHotos里的API,找到了如下代码:
NSError *err = nil;
[[PHPhotoLibrary sharedPhotoLibrary] performChangesAndWait:^{
PHAssetResourceCreationOptions *options = [[PHAssetResourceCreationOptions alloc] init];
[[PHAssetCreationRequest creationRequestForAsset] addResourceWithType:PHAssetResourceTypePhoto data:imgData options:options];
} error:&error];
以为这个问题到这就解决了,实际还有个小问题:SDWebImage的loadImageWithURL加载图片结束的回调里并不会返回data,其值为nil;而PHPhotoLibrary里的API需要data,尝试调整SDWebImageOptions字段,发现并不能实现直接回调image对应的data。
*一个好的程序员当以解决问题为第一要务,想办法取SD缓存到本地的data!有了如下一行:
NSData *imgData = [[SDImageCache sharedImageCache] diskImageDataForKey:imageUrl];
此处我的SDWebImage版本为5.0以后版本,之前版本取本地data略有不同,自行解决。