iOS14 适配相册Limited模式

前言

苹果在iOS14继续加强了对用户隐私的保护,有时需求只是想选择一张相册中的图片,但是需要对App开发整个照片库的权限,一些私密照片也可以被App读取到,这样很不合理!因此iOS14中对相册权限新增了“Limited Photo Library Access” 模式,这样用户可以控制App允许访问的照片。下面简单介绍下如何适配iOS14相册新增的功能。

相册相关库

iOS8以后苹果逐渐使用Photos代替AssetsLibrary,这里主要使用Photos实现访问系统相册。如果还在使用AssetsLibrary请尽快使用新的 API实现相册相关功能。项目中需要引入Photos和PhotoUI库。

权限弹窗变化

  • Select Photo(选择照片): 限制访问,点击之后会弹出系统的图片选择界面选择资源,APP 只能访问用户选择的资源。
  • Allow Access to All Photos(允许访问所有照片): 可以访问所有的资源。
  • Don't Allow(不允许): 不允许访问资源
图片

PHPhotoLibrary新增API

PHPhotoLibrary用于获取查看相册权限,处理相册变化,注册监听相册变化,监听用户添加/删除了哪些照片。

  • 权限枚举

PHAuthorizationStatusNotDetermined 用户未作出选择

PHAuthorizationStatusRestricted 此App无权限访问照片数据

PHAuthorizationStatusDenied 用户已明确拒绝此应用程序访问照片数据

PHAuthorizationStatusAuthorized 用户已授权此应用程序访问照片数据

PHAuthorizationStatusLimited 用户已授权此应用程序进行有限照片库访问(iOS14新增)

  • 权限等级枚举

PHAccessLevelAddOnly 仅允许添加

PHAccessLevelReadWrite 读写

注意:PHAuthorizationStatusLimited 权限只在 accessLevel 为 PHAccessLevelReadWrite 时生效

  • 权限获取

新权限获取:增加了权限等级

+ (PHAuthorizationStatus)authorizationStatusForAccessLevel:(PHAccessLevel)accessLevel
+ (void)requestAuthorizationForAccessLevel:(PHAccessLevel)accessLevel handler:(void(^)(PHAuthorizationStatus status))handler 

旧权限获取:在iOS14中已经废弃,建议使用上面的新API

+ (PHAuthorizationStatus)authorizationStatus 
+ (void)requestAuthorization:(void(^)(PHAuthorizationStatus status))handler 

注意:如果仍使用旧的API未适配iOS14新特性,这时获取相册权限状态,就算在Limited 模式下也会返回Authorized

  • 新增PHPicker

iOS 14 中系统新增了一个图片选择器PHPicker(iOS14以上使用),官方建议使用 PHPicker 来替代原有的UIImagePickerController(iOS14以下使用)进行图片选择 。UIImagePickerController只能选中一张图片已经不符合需求了,将逐渐被废弃替换。

怎样使用PHPicker

图片

1、使用PHPickerConfiguration配置PHPicker,键selectionLimit 设置为0表示多选,设置为大于1表示只可选中一张图片,默认值为1;使用filter设置想要的相册资源类型,包括imagesFilter、videosFilter、livePhotosFilter,亦可以设置为数组@[videoFilter,livePhotosFilter]显示多种类型.
2、设置PHPickerViewControllerDelegate代理,接收选中照片后的回调;
3、在代理回调piscker:didFinishPicking: 中处理返回结果PHPickerResult;

PHPicker优缺点

优点:

  1. 支持多选,可以设置选择一个资源,还是多个资源;
  2. 支持按 image,video,livePhotos 类型进行选择;
  3. 只是资源搜索,在页面上有搜索框;
  4. 独立进程,不会影响App性能,如何体现呢:在设置→照片→照片权限设置中选择点击”选中的照片“后也会弹出PHPicker,并且可以为App添加允许访问的照片;
  5. 内置隐私:不需要直接访问用户相册;不会弹出访问相册提示;仅为用户提供选择的照片和视频(App 无法获取其他照片);

缺点:

  1. 不支持选中图片的编辑,例如选中后裁剪成正方形,需要自定义实现了;

plist设置

NSPhotoLibraryAddUsageDescription

用户存入相册时的提示信息。

NSPhotoLibraryUsageDescription

相册访问权限信息,必须有此项,不然访问相册的时候 APP 会 Crash。

PHPhotoLibraryPreventAutomaticLimited

如果未适配,App在每次冷启动时都会触发询问用户是否需要修改照片权限,添加可供App访问的图片。

图片

隐藏系统弹出的选择图片Alert

在首次启动访问相册权限,并且选择了Limited权限后,再次冷启动的时候会自动弹出权限选择Alert,要求用户选择图片。

图片

在info.plist中加入PHPhotoLibraryPreventAutomaticLimited = YES关闭系统自动弹窗。并且使用下面的API主动调用控制弹出PHPickerViewController 进行照片选择。(在应该使用的地方使用)

  [[PHPhotoLibrary sharedPhotoLibrary] presentLimitedLibraryPickerFromViewController:self];

使用示例

  • 查询权限
PHAuthorizationStatus status = [PHPhotoLibrary authorizationStatusForAccessLevel:PHAccessLevelReadWrite];
  switch (status) {
      case PHAuthorizationStatusLimited:
          NSLog(@"limited");
          break;
      case PHAuthorizationStatusDenied:
          NSLog(@"denied");
          break;
      case PHAuthorizationStatusAuthorized:
          NSLog(@"authorized");
          break;
      default:
          break;
}
  • 请求权限
[PHPhotoLibrary requestAuthorizationForAccessLevel:PHAccessLevelReadWrite handler:^(PHAuthorizationStatus status) {
            switch (status) {
                case PHAuthorizationStatusLimited:
                {
                    //用户选择Limited模式,限制App访问有限的相册资源
                    NSMutableArray *images = [NSMutableArray array];
                    //获取可访问的图片配置选项
                    PHFetchOptions *option = [[PHFetchOptions alloc] init];
                    //根据图片的创建时间升序排序返回
                    option.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"creationDate" ascending:YES]];
                    //获取类型为image的资源
                    PHFetchResult *result = [PHAsset fetchAssetsWithMediaType:PHAssetMediaTypeImage options:option];
                     //遍历出每个PHAsset资源对象
                    [result enumerateObjectsUsingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
                        PHAsset *asset = (PHAsset *)obj;
                        //将PHAsset解析为image的配置选项
                        PHImageRequestOptions *requestOptions = [[PHImageRequestOptions alloc] init];
                        //图像缩放模式
                        requestOptions.resizeMode = PHImageRequestOptionsResizeModeExact;
                        //图片质量
                        requestOptions.deliveryMode = PHImageRequestOptionsDeliveryModeHighQualityFormat;
                        //PHImageManager解析图片
                        [[PHImageManager defaultManager] requestImageForAsset:asset targetSize:PHImageManagerMaximumSize contentMode:PHImageContentModeDefault options:requestOptions resultHandler:^(UIImage * _Nullable result, NSDictionary * _Nullable info) {
                            NSLog(@"图片 %@",result);
                            //在这里可以自定义一个显示可访问相册资源的viewController.
                            [images addObject:result];
                        }];
                    }];
                    break;
                }
                case PHAuthorizationStatusDenied:
                {
                    NSLog(@"denied");
                }
                    break;
                case PHAuthorizationStatusAuthorized:
                {
                    //用户选择"允许访问所有照片",调用PHPickerViewController显示图片选择器
                    dispatch_async(dispatch_get_main_queue(), ^{
                        PHPickerConfiguration *configuration = [[PHPickerConfiguration alloc] init];
                        //只获取image类型资源
                        configuration.filter = [PHPickerFilter imagesFilter];
                         //可以多选
                        configuration.selectionLimit = 0;
                        PHPickerViewController *pickerVC = [[PHPickerViewController alloc] initWithConfiguration:configuration];
                        pickerVC.delegate = self;
                        pickerVC.modalPresentationStyle = UIModalPresentationFullScreen;
                        [self presentViewController:pickerVC animated:YES completion:^{
                        }];
                    });  
                }
                    break;
                default:
                    break;
            }
        }];

返回结果:选中了三张图片允许App访问


图片

注意: 我们通过requestImageForAsset获取到允许访问的图片PHAsset对象后,解析为UIImage后,需要开发者自定义一个显示可访问相册资源的viewController.

  • 主动弹出选择照片PHPickerViewController
[[PHPhotoLibrary sharedPhotoLibrary] presentLimitedLibraryPickerFromViewController:self];
  • 选中图片后的回调
- (void)picker:(PHPickerViewController *)picker didFinishPicking:(NSArray *)results
{
    [picker dismissViewControllerAnimated:YES completion:nil];
    if (!results || !results.count) {
        return;
    }
    NSLog(@"didFinishPicking");
}

注意:无论我们点击完成还是取消都会调用这个回调,当点击取消时results返回为空

你可能感兴趣的:(iOS14 适配相册Limited模式)