美文网首页
iOS14相册适配

iOS14相册适配

作者: 彩色大猩猩 | 来源:发表于2020-09-29 15:05 被阅读0次

    前言: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相册适配

          本文链接:https://www.haomeiwen.com/subject/deseuktx.html