iOS 技巧

关于Photos库的简单应用,筛选、获取、GIF、livePho

2021-03-17  本文已影响0人  boyka_yang

写在最开始

由于本项目中对UI还原度要求较高,交互要求完全还原,用别人封装的改起来总归是有些别扭;为了后续方便自己实现定制UI交互等,决定自己从系统API开始封装一套相册资源选择器。

而AL用起来则到处报被弃用的⚠️,想逼死我这个强迫症啊~
然后就选择了PH,总的来说和AL比较类似,但是很多东西实现起来却是缺这少那的。虽说磨了有段时间,但目前用起来效果&性能尚可;下边分享点小坑和核心代码。


一、 其实一开始就遇到了个问题,不能同时获取照片和视频。。。

最后为分别拉出加到同一个数组之后,进行按时间排序🤡。

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时再做些筛选的话,就会碰到各种问题了😄。


/*
         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测试,可能会有遗漏之类(期待大家的验证,欢迎验证通过的朋友在评论区留下一笔)。

  1. 判断PHAsset是否GIF第二种:
[[asset valueForKey:@"filename"] hasSuffix:@"GIF"]

判断后缀是否为GIF / gif。也通过了我那一百多张GIF的测试。

  1. 取PHAsset私有属性uniformTypeIdentifier,和第二种类似,但似乎更靠谱一点
[[asset valueForKey:@"uniformTypeIdentifier"] isEqual:@"com.compuserve.gif"]
  1. 判断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略有不同,自行解决。

上一篇下一篇

猜你喜欢

热点阅读