专栏首页ios技术安装为什么不推荐使用PHPicker
原创

为什么不推荐使用PHPicker

WWDC 20 过去已经有好几个月了, iOS 14 正式版也发布了,这篇文章写的有点晚了,因为有些 API 没有彻底弄懂,所以一直拖到了现在(奇怪我怎么感觉去年也说过一样的话: doge)其实过了这么多个月,大家应该或多或少都看过一些别人写文章,介绍相册的变化,介绍 PHPicker,但是有一些点没讲清楚,比如怎么用 PHPicker 获取视频?PHPicker 有什么不足?那么下面让我们一起看看什么是 PHPicker 以及 iOS 14 相册有什么新的变化。

PHPicker

iOS 14 中系统新增了一个图片选择器 PHPicker,官方建议使用 PHPicker 来替代原有的 API 进行图片选择,下面我们来看看 PHPicker 的优点:

  • 支持多选
  • 支持搜索
  • 独立的进程
  • 内置隐私
    • 不需要直接访问用户相册
    • 不会弹出访问相册提示
    • 仅提供用户选择的照片和视频(App 无法获取其他照片)

如何调用 PHPicker

我们先来看下 PHPicker 的流程图,首先声明 PHPickerConfiguration,进行配置,再传给 PHPickerViewController,完成调用环节,代码如下:

var config = PHPickerConfiguration()
// 可选择的资源数量,0表示不设限制,默认为1
config.selectionLimit = 0
// 可选择的资源类型
// 只显示图片(注:images 包含 livePhotos)
config.filter = .images
// 显示 Live Photos 和视频(注:livePhotos 不包含 images)
config.filter = .any(of: [.livePhotos, .videos])
// 如果要获取视频,最好设置该属性,避免系统对视频进行转码
config.preferredAssetRepresentationMode = .current

let picker = PHPickerViewController(configuration: config)
picker.delegate = self
present(picker, animated: true, completion: nil)复制代码

处理 PHPicker 的回调

PHPicker 的代理方法只有一个,声明如下:

@available(iOS 14, *)
public protocol PHPickerViewControllerDelegate : AnyObject {
    func picker(_ picker: PHPickerViewController, didFinishPicking results: [PHPickerResult])
}复制代码

注意: 取消选择也会触发代理方法,会返回空的 results

如何获取照片

PHPicker 获取图片的方法还是比较简单的,代码如下:

func picker(_ picker: PHPickerViewController, didFinishPicking results: [PHPickerResult]) {
    // 首先需要 dismiss picker
    picker.dismiss(animated: true, completion: nil)
    for result in results {
        // 判断类型是否为 UIImage
        if result.itemProvider.canLoadObject(ofClass: UIImage.self) {
            // 确认类型后,调用 loadObject 方法获取图片
            result.itemProvider.loadObject(ofClass: UIImage.self) { (data, error) in
                // 回调结果是在异步线程,展示时需要切换到主线程
                if let image = data as? UIImage {
                    DispatchQueue.main.async {
                        self.showImage(image)
                    }
                }
            }
        }
    }
}复制代码

如何获取视频

其他文章中都没有介绍 PHPicker 如何获取视频,其实获取视频的方法在官方的 Demo 以及视频中都没有介绍,这也是我迟迟没有写文章的原因,因为之前我也不知道怎么获取,那么下面让我们一起来看下怎么获取视频。

func picker(_ picker: PHPickerViewController, didFinishPicking results: [PHPickerResult]) {
    // 首先需要 dismiss picker
    picker.dismiss(animated: true, completion: nil)
    for result in results {
        if result.itemProvider.canLoadObject(ofClass: UIImage.self) {
            // 判断类型是否为 UIImage
            ...
        } else {
            // 类型为 Video
            // 调用 loadFileRepresentation 方法获取视频的 url
            // 这里 Type Identifier 我们用 UTType.movie.identifier (“public.movie”) 这个 UTI 可以获取所有格式的视频
            result.itemProvider.loadFileRepresentation(forTypeIdentifier: UTType.movie.identifier) { (url, error) in
                if let error = error {
                    print(error)
                    return
                }
                // 系统会将视频文件存放到 tmp 文件夹下
                // 我们必须在这个回调结束前,将视频拷贝出去,一旦回调结束,系统就会把视频删掉
                // 所以一定要确定拷贝结束后,再切换到主线程做 UI 操作
                // 另外不用担心视频过大而导致拷贝的时间很久,系统将创建一个 APFS 的克隆项,因此拷贝的速度会非常快
                guard let url = url else { return }
                let fileName = "\(Int(Date().timeIntervalSince1970)).\(url.pathExtension)"
                let newUrl = URL(fileURLWithPath: NSTemporaryDirectory() + fileName)
                try? FileManager.default.copyItem(at: url, to: newUrl)
                DispatchQueue.main.async {
                    self.playVideo(newUrl)
                }
            }
        }
    }
}复制代码

注意: 如果你遇到了部分资源可以加载,而部分资源无法加载的话,那么有可能是设备没有连接到 iCloud,只能加载本地资源,而无法加载 iCould 上的资源。

被废弃的 API

有新的 API 出现,也会有一些 API 被废弃,在 UIImagePickerController 中有三个 sourceType,现在有两个被废弃,只留下 camera

public enum SourceType : Int {
    @available(iOS, introduced: 2, deprecated: 100000, message: "Will be removed in a future release, use PHPicker.")
    case photoLibrary = 0
    case camera = 1
    @available(iOS, introduced: 2, deprecated: 100000, message: "Will be removed in a future release, use PHPicker.")
    case savedPhotosAlbum = 2
}复制代码

另外 AssetsLibrary 早在几年前被废弃,如果还在使用 AssetsLibrary 请尽快使用新的 API。

PHPicker 的缺点

为什么不推荐使用 PHPicker,虽然说 PHPicker 有一些优点,但同时也有一些缺点:

  • 加载 iCloud 资源时没有进度回调
  • 不支持图片编辑(比如选择头像要将图片裁剪成正方形)

有没有其他的解决方案?

有的,如果你不能接受 PHPicker 的缺点,同时又想保护用户的隐私,目前有 Picker、Editor、Capture 三个模块,支持图片/视频选择、编辑、拍摄功能,支持 SPM、CocoaPods 方式引入。

新增权限

iOS 14 中相册新增了一个 “Limited Photos Library” 模式,在授权时多了一个 “选择照片” 的选项。点击之后系统会弹出 PHPickerController 用户可以选择指定的照片让 App 读取。

当用户选择了 limited 模式后,系统将在 App 每次启动后首次触发相册时弹出提示,允许用户修改需要授权给 App 的照片。

当然这个弹窗是可以关闭的,如果你希望手动控制 PHPickerController 弹出的时机也是有办法的。

我们需要在 Info.plist 中添加 PHPhotoLibraryPreventAutomaticLimitedAccessAlert 字段,并设置为 YES,设置后系统将不再弹出访问提示。

然后我们可以在合适的时机调用以下这个 API 来推出 PHPickerController

let viewController = self
PHPhotoLibrary.shared().presentLimitedLibraryPicker(from: viewController)复制代码

我们可以看到,当用户选择 limited 模式后,底部出现了一段提示:“无法查看相册全部照片,点击选择更多照片”。当点击这个提示后,将会推出 PHPickerController,此时用户可以修改授权给 App 的照片。同时我们会监听相册的变化,当用户修改授权的照片后,会立即刷新相册,用户可以继续进行选择照片的流程。

监听相册变化

配合手动调用 PHPickerController,我们还需要监听用户添加/删除了哪些照片。

注意: 这组 API 并不是新出的,从 iOS 8 开始就支持了。

let viewController = self
// 开始监听
PHPhotoLibrary.shared().register(viewController) 
// 结束监听
PHPhotoLibrary.shared().unregisterChangeObserver(viewController)复制代码

处理监听回调:

/// 回调方法
func photoLibraryDidChange(_ changeInstance: PHChange) {
    // Your code
}复制代码

由于这是一组旧的 API,所以就不介绍细节了(比如判断是新增还是删除),感兴趣的朋友可以去了解一下。

新增的 API

PHAccessLevel

在 iOS 14 中新增了权限等级枚举 PHAccessLevel,有两个 case,分别是 “只读” 和 “读写”。

public enum PHAccessLevel : Int {
    case addOnly = 1
    case readWrite = 2
}复制代码

对应新增了一组获取/查看权限的 API:

let level: PHAccessLevel = .readWrite
// 获取权限
PHPhotoLibrary.requestAuthorization(for: level) { status in
  // Your code
}
// 查看权限
let status: PHAuthorizationStatus = PHPhotoLibrary.authorizationStatus(for: level)复制代码

PHAuthorizationStatus

PHAuthorizationStatus 新增了一个 case limited

public enum PHAuthorizationStatus : Int {
    case notDetermined = 0
    case restricted = 1
    case denied = 2
    case authorized = 3
    @available(iOS 14, *)
    case limited = 4
}复制代码

当用户在授权时选择了 “选择照片” 的选项时:

  • 使用新 API 将会返回 limited case
  • 使用旧 API 将会返回 authorized case

注意: limited case 仅在 PHAccessLevel = .readWrite 时会返回。

总结

新出的 PHPicker 个人觉得一般,如果对 Picker 要求不多的朋友可以考虑使用。然后是新出的 “Limited Photos Library” 模式,这个非常棒,如果有自定义 Picker 的朋友建议跟进一下。如果没有自定义 Picker 的朋友可以考虑使用我们做的第三方图片选择框架 AnyImageKit。

以上就是 iOS 14 相册的改动以及 PHPicker 的全部内容,如有错误欢迎指出。

原创声明,本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

登录 后参与评论
0 条评论

相关文章

  • 为什么 MySQL 不推荐使用 join?

    作者:李博 , 链接: https://cnblogs.com/liboware/p/12740901.html

    Java小咖秀
  • 为什么不推荐使用存储过程?

    之所以有这个题目,我既不是故意吸引眼球,也不想在本文对存储过程进行教科书般论述。最近项目中遇到的存储过程问题,让我想起了去年在武汉出差时一位同事的发问:

    芋道源码
  • 为什么IDEA不推荐你使用@Autowired ?

    @Autowired注解相信每个Spring开发者都不陌生了!在DD的Spring Boot基础教程(https://blog.didispace.com/sp...

    程序猿DD
  • 为什么IDEA不推荐你使用@Autowired?

    但是当我们使用IDEA写代码的时候,经常会发现@Autowired注解下面是有小黄线的,我们把小鼠标悬停在上面,可以看到这个如下图所示的警告信息:

    SerMs
  • 为什么不推荐使用汉字作为密码?

    日常生活中,密码的使用十分常见。基本上,登录APP、手机支付、开机解锁,都需要使用密码。密码的形式也多种多样:数字密码,指纹密码,字母密码等,却唯独没有汉字,这...

    不脱发的程序猿
  • 为什么不推荐数据库使用外键?

    我的经验告诉我,很多数据库(大多数我曾经使用的)不包含外键时并不总是一件坏事。在这篇文章中,我想把重点放在为什么的原因上。

    JavaFish
  • 为什么阿里不推荐使用MySQL分区表?

    在示例表插入两条记录,按分区规则,记录分别落在p_2018和p_2019分区。 可见,该表包含了一个.frm文件和4个.ibd文件,每个分区对应一个.ibd...

    JavaEdge
  • 为什么MySQL不推荐使用子查询和join

    1.对于mysql,不推荐使用子查询和join是因为本身join的效率就是硬伤,一旦数据量很大效率就很难保证,强烈推荐分别根据索引单表取数据,然后在程序里面做j...

    JAVA葵花宝典
  • 为什么不推荐使用BeanUtils属性转换工具

    1 背景 之前在专栏中讲过“不推荐使用属性拷贝工具”,推荐直接定义转换类和方法使用 IDEA 插件自动填充 get / set 函数。

    崔笑颜
  • 为什么MySQL不推荐使用uuid作为主键?

    在mysql中设计表的时候,mysql官方推荐不要使用uuid或者不连续不重复的雪花id(long形且唯一,单机递增),而是推荐连续自增的主键id,官方的推荐是...

    程序员追风
  • 为什么不推荐使用BeanUtils属性转换工具

    之前在专栏中讲过“不推荐使用属性拷贝工具”,推荐直接定义转换类和方法使用 IDEA 插件自动填充 get / set 函数。

    明明如月学长
  • 为什么有的程序员不推荐使用Lombok!

    我有个学弟,在一家小型互联网公司做Java后端开发,最近他们公司新来了一个技术总监,这位技术总监对技术细节很看重,一来公司之后就推出了很多"政策",比如定义了很...

    Java3y
  • 为什么不推荐使用 stop、suspend 方法中断线程?

    来源 | https://blog.csdn.net/qq_40400960/article/details/112651249

    程序猿DD
  • 为什么不推荐使用 stop、suspend 方法中断线程?

    我们知道像stop、suspend这几种中断或者阻塞线程的方法在较高java版本中已经被标记上了@Deprecated过期标签,那么为什么她们曾经登上了java...

    Java技术栈
  • 为什么不推荐Selenium写爬虫

    最近在群里经常会看到有些朋友说,使用Selenium去采集网站,我看到其实内心是很难受的,哎!为什么要用Selenium呢? 我想说下自己的看法,欢迎各位大佬批...

    小歪
  • 为什么我不推荐你使用RabbitMQ的消息转换功能

    发送消息与订阅消息取消使用amqp提供的消息序列化与反序列化功能,使用String类型,发送消息时手动转化为json字符串再发送,消费消息时手动json反序列化...

    Java艺术
  • 为什么MySQL不推荐使用uuid或者雪花id作为主键?

    在mysql中设计表的时候,mysql官方推荐不要使用uuid或者不连续不重复的雪花id(long形且唯一,单机递增),而是推荐连续自增的主键id,官方的推荐是...

    业余草
  • 不推荐使用Spring Boot 2.2.0

    目前 Spring Cloud Hoxton 未发布 RELEASE 版本,官方计划 本月发布

    冷冷
  • 为什么不推荐大家去外包公司

    其实大家心里都像明镜一样地明白“低技术含量的外包没有前途”,好多人都认为外包没有技术,其实这种思想不对,国外有大量的高端技术外包业务等着我们干,但是反问一句“你...

    JAVA高级架构开发

扫码关注腾讯云开发者

领取腾讯云代金券