《iOS11中UIDocumentInteractionController/UIActivityViewController分享文件到三方app毫无反应》问题的解决方案

UIDocumentInteractionController和UIActivityViewController主要都是用来完成app间文件传输的功能(也就是传说中的进程间交互),很多人将他当做是一种系统原生的分享方案。具体如何实现在此就不做赘述了,很多前辈都写过相关的文章,非常详尽。

这篇文章主要解决的问题是:

通过UIDocumentInteractionController/UIActivityViewController把文件分享给第三方app,之前本来用的好好的,怎么到了iOS11就毫无反应了呢?

解决问题的关键点就是:copy文件到沙盒
对,就是这么简单,但也就是这么出人意料。实际应用中,大多数分享的文件可能都是保存在沙盒里的;但是,一般做Demo的时候,大多可能就是拖个文件到boundle中,这也就造成了后续分享文件功能毫无反应的结果。

来说说具体怎么改进吧。

  1. 首先,如果你要分享的文件是在boundle中,那么就一定要先把他copy到沙盒中,然后把沙盒路径传给UIDocumentInteractionController/UIActivityViewController;
  2. 既然做了copy,那么大多数情况下也同样需要remove掉,不然应用所占体积会越来越大,当然,如果你有特殊需求除外。但是remove需要放在哪里进行呢?
    1 - 对UIActivityViewController 来说,他有一个属性completionWithItemsHandler,这是用户完成/取消选择的回调,在回调中删除沙盒中copy过去的文件,并不影响三方app获取文件,因为此时文件已经被复制到了三方的沙盒中,默认是/Documents/Inbox中。
    2 - 对于UIDocumentInteractionController,其分享弹窗有两套方法:OpenInMenu 和 OptionsMenu,甚至连代理方法都有对应的两套:
- (void)documentInteractionControllerDidDismissOptionsMenu:(UIDocumentInteractionController *)controller;

- (void)documentInteractionControllerDidDismissOpenInMenu:(UIDocumentInteractionController *)controller;

从方法名就可以看出来,这两个方法也是分享弹窗在消失时调用的代理方法,因此根据你弹出分享弹窗的方式在相应的代理方法中实现remove文件的操作即可。

  1. 如果你的app需要接收别人分享过来的文件,那么在什么地方接收呢?
    在UIDocumentInteractionController的官方文档《Document Interaction Programming Topics for iOS》

You receive information about the file to be opened in the application:willFinishLaunchingWithOptions: or application:didFinishLaunchingWithOptions: method of your application delegate. If your application handles custom file types, you must implement this delegate method (instead of the applicationDidFinishLaunching: method) and use it to initialize your application.

建议使用application:willFinishLaunchingWithOptions:application:didFinishLaunchingWithOptions:方法来接收分享过来的文件信息,但经过我的实践发现:
无论iOS11还是iOS10,无论是UIDocumentInteractionController还是UIActivityViewController:
1.当接收app在后台时,有其他app向其分享文件,都会执行
-(BOOL)application:(UIApplication *)app openURL:(NSURL *)url options:(NSDictionary<UIApplicationOpenURLOptionsKey,id> *)options方法;
2.接收app在完全退出的情况下,有其他app向其分享文件,会先后执行
-(BOOL)application:(UIApplication *)app openURL:(NSURL *)url options:(NSDictionary<UIApplicationOpenURLOptionsKey,id> *)options

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions方法。
因此个人建议统一在openURL方法中处理即可。

还有一个很坑的地方,就是UIDocumentInteractionController的强引用问题。在iOS11之前,如果不强引用,则会直接崩溃;而在iOS11中,不崩溃了~ 我原本天真的以为,是修好了~然而我还是太年轻。。。仅仅只是把崩溃修好了而已!!!你可能没太理解我的意思吧?崩溃修好了不就是好了吗?是“只”有崩溃好了!后面的功能还是不能用!还得强引用才能正常使用!我勒个擦!以前还有报错信息可循,现在直接就是毫无反应,让人更加懵圈!简直爆炸啊!!!

还有就是这个copy文件的问题,我也没有在任何官方文档中看到,因为iOS11出来之后,和UIDocumentInteractionController还有UIActivityViewController相关的所有文档,Guide,Sample Code等等都没有过任何更新。。。感觉又是Apple自己悄悄弄的呢。。。(如果有小伙伴能找到相关的官方说明,敬请赐教,非常感谢!)

特别鸣谢
我全网只在这个帖子的9楼看到了copy的解决方法(他说的是copy到cache中,实测到Document也没有问题),鸣谢这位做好事不留名的英雄。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1.ios高性能编程 (1).内层 最小的内层平均值和峰值(2).耗电量 高效的算法和数据结构(3).初始化时...
    欧辰_OSR阅读 29,649评论 8 265
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,148评论 1 32
  • 前言 iOS中的沙盒可以让平台更加的安全,这也是沙盒给用户带来的最主要好处。不过由于沙盒的严格限制,导致程序之间共...
    不简单的风度阅读 5,556评论 0 6
  • 时间在无形中飞快的向前跑着,而自己感觉一直在重复着同样的日子,工作的压力开始增加,感觉空闲的时间越来越少,自己一直...
    嘟嘟小胖阅读 213评论 0 0
  • 春日,万物复苏,跋涉河流,顺着山岳的脚往下游走,被火车拖曳着越陌度阡,在黄土高原上生生不息。 冷风吹着荒荡的土地慢...
    半榛半柠阅读 809评论 5 5