美文网首页iOS 技巧
关于Photos库的简单应用,筛选、获取、GIF、livePho

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

作者: boyka_yang | 来源:发表于2021-03-17 16:26 被阅读0次

    写在最开始

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

    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略有不同,自行解决。

    相关文章

      网友评论

        本文标题:关于Photos库的简单应用,筛选、获取、GIF、livePho

        本文链接:https://www.haomeiwen.com/subject/mcxecltx.html