前言
苹果在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优缺点
优点:
- 支持多选,可以设置选择一个资源,还是多个资源;
- 支持按 image,video,livePhotos 类型进行选择;
- 只是资源搜索,在页面上有搜索框;
- 独立进程,不会影响App性能,如何体现呢:在设置→照片→照片权限设置中选择点击”选中的照片“后也会弹出PHPicker,并且可以为App添加允许访问的照片;
- 内置隐私:不需要直接访问用户相册;不会弹出访问相册提示;仅为用户提供选择的照片和视频(App 无法获取其他照片);
缺点:
- 不支持选中图片的编辑,例如选中后裁剪成正方形,需要自定义实现了;
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返回为空