之前写了一个仿 Snapseed 的 ImagePicker,见 Demo 和之前的 文章。
之后发现有人说第一次未授权时 UICCollectionview
会加载不出照片,发现没有在合适时候调用权限管理。iOS 10 的这个权限管理是 info.plist 的一个键值,没有回调方法。查询 PhotosKit 方法里,发现有一个权限回调方法,就想封装一个方法来友好的询问用户权限。
权限 Tips
查询文档,相册权限 PHAuthorizationStatus
枚举类型有这么几种。
typedef NS_ENUM(NSInteger, PHAuthorizationStatus) {
PHAuthorizationStatusNotDetermined = 0, // User has not yet made a choice with regards to this application 用户未决定
PHAuthorizationStatusRestricted, // This application is not authorized to access photo data. 一般是家长权限之类的拒绝
// The user cannot change this application’s status, possibly due to active restrictions
// such as parental controls being in place.
PHAuthorizationStatusDenied, // User has explicitly denied this application access to photos data. 用户拒绝
PHAuthorizationStatusAuthorized // User has authorized this application to access photos data. 用户允许
} PHOTOS_AVAILABLE_IOS_TVOS(8_0, 10_0);
其中,我们不希望用户去拒绝,达到 PHAuthorizationStatusDenied
这个状态,这样的话用户再要获取权限就要到设置里的隐私里找到我们的应用再手动打开,而没办法在 App 内去请求权限更改。那么我们希望暂时拒绝后的状态应该是 PHAuthorizationStatusNotDetermined
的,思路就是提示一个 UIAlertViewController
去询问,用户取消则不去调用真正的询问权限(相当于忽略),以便下次提醒在询问。用户同意则调用系统询问权限,这时一般用户也会再允许了。流程示意图如下。
代码很简单,直接贴上来
- (void)showPermissionAlertInController{
UIAlertController *alert = [UIAlertController alertControllerWithTitle:nil message:@"需要你的图库的权限" preferredStyle:UIAlertControllerStyleAlert];
UIAlertAction *cancelAction = [UIAlertAction actionWithTitle:@"取消" style:UIAlertActionStyleCancel handler:^(UIAlertAction *action) {
//do something
}];
UIAlertAction *requestAction = [UIAlertAction actionWithTitle:@"同意" style:UIAlertActionStyleDefault handler:^(UIAlertAction *action) {
dispatch_async(dispatch_get_global_queue(0, 0), ^{
[PHPhotoLibrary requestAuthorization:^(PHAuthorizationStatus status) {
if (status == PHAuthorizationStatusAuthorized) {
NSLog(@"用户同意授权相册");
}else {
NSLog(@"用户拒绝授权相册");
}
dispatch_async(dispatch_get_main_queue(), ^{
//do something
});
}];
});
}];
[alert addAction:cancelAction];
[alert addAction:requestAction];
[self presentViewController:alert animated:YES completion:nil];
}
再在需要用到权限的地方加上判断就好
if ([PHPhotoLibrary authorizationStatus] == PHAuthorizationStatusAuthorized)
相册变化
后来发现如果调用选择器时在外部改变了照片(增加,删除)后 Collectionview 不会改变,就研究了一下 PhotosKit,发现有个 - (void)photoLibraryDidChange:(PHChange *)changeInfo
检测相册变动,但之前需要注册为观察者并实现协议
PHPhotoLibraryChangeObserver
- (void)viewDidLoad{
[super viewDidLoad];
[[PHPhotoLibrary sharedPhotoLibrary]registerChangeObserver:self];
//other code
}
- (void)dealloc{
[[PHPhotoLibrary sharedPhotoLibrary]unregisterChangeObserver:self];
}
然后实现方法,更新 UI 的话要在主线程里。
- (void)photoLibraryDidChange:(PHChange *)changeInfo {
dispatch_async(dispatch_get_main_queue(), ^{
PHFetchResultChangeDetails *changes = [changeInfo changeDetailsForFetchResult:self.imageAssetsResult];
if (changes) {
self.photosDataSource.itemArray = [self getImageAssets];
[self.photosView reloadData];
}
});
}
这个方法默认所有相册的变更都会通知,但是有些变动是我们不需要的,比如用户新建了一个新空相册,删除了一个旧相册等,这时我们可以用 PHChange
里的 ChangeDetails 去区分,比如 changeDetailsForFetchResult:(PHFetchResult *)object;
就是传入一组 FetchResult,如果这组 Result 有变化,比如这组 Result 是 Asset 的集合,Asset 增加减少,那么就会返回一个 ChangeDetails,可以用此区分,但是如果是 Asset 内部的变化,就不会返回一个 ChangeDetails。同样的如果这组 Result 是 Collection 的集合,那么 Collection 内部的变化,比如 Collection 里的 Asset 对象增加减少,就不会有返回。
具体可以见苹果官方的 例子。
最后
这篇文章篇幅不多,作为一些补充,希望能帮到有需要的人。