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<UIImage *> *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<PHPickerResult *> *)results
{
    [picker dismissViewControllerAnimated:YES completion:nil];
    if (!results || !results.count) {
        return;
    }
    NSLog(@"didFinishPicking");
}

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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 160,165评论 4 364
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,720评论 1 298
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,849评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,245评论 0 213
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,596评论 3 288
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,747评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,977评论 2 315
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,708评论 0 204
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,448评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,657评论 2 249
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,141评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,493评论 3 258
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,153评论 3 238
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,108评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,890评论 0 198
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,799评论 2 277
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,685评论 2 272

推荐阅读更多精彩内容