前言:iOS14相册隐私权限适配方案,苹果官方推荐PHPicker,网上也有很多优质文章。本文是项目还没有升级适配到Xcode12,还在Xcode11上使用UIImagePickerController的临时的适配方案。
优质文章链接:
【淘系技术】iOS14 隐私适配及部分解决方案
iOS 14 相册适配指南
PhotoKit 新变化:认识新的照片选择器
1.问题
项目用的系统的UIImagePickerController,会获取本地相册,展示到自定义相册列表。但是在iOS14系统手机上,会多弹出一个系统级别的隐私权限弹窗,并且选取照片/视频后,无法得到回调(didFinishPickingMediaWithInfo),因此找不到刷新数据源的时机。在查阅了苹果官方文档及网上文章后,建议PHPicker,但是此API需要Xcode12才可以调用,目前项目还未迁移到Xcode12。
2.思路
在Xcode11运行项目时,当弹出隐私弹窗,可以看到视图层级结构最顶层是UIImagePickerController类,因此想根据查找当时topViewController是不是UIImagePickerController,并且所选的媒体资源数据有没有变化来决定要不要刷新数据源。
3.临时解决方式
在自定义相册类,开启定时器查找当前topViewController是不是UIImagePickerController,如果不是则刷新PHAsset数据,代替回调拿到数据并展示。解决了问题。
4.等项目完全适配完Xcode12后,还是用苹果官方推荐的PHPicker来适配。
5.代码
var checkTimer: Timer?
override func viewDidLoad() {
super.viewDidLoad()
//iOS14,相册权限适配,临时解决方案
if #available(iOS 14.0, *) {
// 因为在Xcode11中的UIImagePickerController无法拿到iOS14隐私相册的选中回调。临时解决办法:轮询监听UIImagePickerController是否不是topVC,不是就刷新Assets数据
checkImagePicker()
}
}
// 因为在Xcode11中的UIImagePickerController无法拿到iOS14隐私相册的选中回调。临时解决办法:轮询监听UIImagePickerController是否不是topVC,不是就刷新Assets数据
func checkImagePicker() {
if (checkTimer != nil) {
checkTimer?.invalidate()
checkTimer = nil
}
checkTimer = Timer.scheduledTimer(timeInterval: 1, target: self, selector: #selector(checkTopViewController), userInfo: nil, repeats: true)
if let timer = checkTimer {
RunLoop.main.add(timer, forMode: .common)
}
}
@objc func checkTopViewController() {
let appDelegate = UIApplication.shared.delegate as? AppDelegate
let topViewController = appDelegate?.findTopVisibleViewController()
if topViewController?.isKind(of: UIImagePickerController.self) ?? false {
} else {
self.reloadAssets()
}
}
func reloadAssets() {
showPhotoAuthorization{ [weak self] b in
DispatchQueue.main.async {
if b {
let options = PHFetchOptions()
options.sortDescriptors = [NSSortDescriptor(key: "creationDate", ascending: false)] // 按照时间排序
let assetsCount = self?.assets?.count
self?.assets = PHAsset.fetchAssets(with: self?.mediaType ?? .image, options: options)
// iOS14,相册权限适配,临时解决方案,如果选中数量发生变化时,才去刷新当前collectionView
if assetsCount != self?.assets?.count {
DispatchQueue.main.async {
self?.collectionView.reloadData()
}
}
} else {
DispatchQueue.main.async {
self?.view.setupBlankView(with: .typePhotos, hasData: false, hasError: true, reloadButtonBlock: nil)
self?.navView.rightView?.isHidden = true
if let navview = self?.navView {
self?.view.bringSubviewToFront(navview)
}
}
}
}
}
}