iOS14相册适配

前言: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)
             }
          }
        }
      }
    }
  }

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