UIDocumentInteractionController和UIActivityViewController主要都是用来完成app间文件传输的功能(也就是传说中的进程间交互),很多人将他当做是一种系统原生的分享方案。具体如何实现在此就不做赘述了,很多前辈都写过相关的文章,非常详尽。
这篇文章主要解决的问题是:
通过UIDocumentInteractionController/UIActivityViewController把文件分享给第三方app,之前本来用的好好的,怎么到了iOS11就毫无反应了呢?
解决问题的关键点就是:copy文件到沙盒。
对,就是这么简单,但也就是这么出人意料。实际应用中,大多数分享的文件可能都是保存在沙盒里的;但是,一般做Demo的时候,大多可能就是拖个文件到boundle中,这也就造成了后续分享文件功能毫无反应的结果。
来说说具体怎么改进吧。
- 首先,如果你要分享的文件是在boundle中,那么就一定要先把他copy到沙盒中,然后把沙盒路径传给UIDocumentInteractionController/UIActivityViewController;
- 既然做了copy,那么大多数情况下也同样需要remove掉,不然应用所占体积会越来越大,当然,如果你有特殊需求除外。但是remove需要放在哪里进行呢?
1 - 对UIActivityViewController 来说,他有一个属性completionWithItemsHandler,这是用户完成/取消选择的回调,在回调中删除沙盒中copy过去的文件,并不影响三方app获取文件,因为此时文件已经被复制到了三方的沙盒中,默认是/Documents/Inbox中。
2 - 对于UIDocumentInteractionController,其分享弹窗有两套方法:OpenInMenu 和 OptionsMenu,甚至连代理方法都有对应的两套:
- (void)documentInteractionControllerDidDismissOptionsMenu:(UIDocumentInteractionController *)controller;
- (void)documentInteractionControllerDidDismissOpenInMenu:(UIDocumentInteractionController *)controller;
从方法名就可以看出来,这两个方法也是分享弹窗在消失时调用的代理方法,因此根据你弹出分享弹窗的方式在相应的代理方法中实现remove文件的操作即可。
- 如果你的app需要接收别人分享过来的文件,那么在什么地方接收呢?
在UIDocumentInteractionController的官方文档《Document Interaction Programming Topics for iOS》中
You receive information about the file to be opened in the
application:willFinishLaunchingWithOptions:
orapplication:didFinishLaunchingWithOptions:
method of your application delegate. If your application handles custom file types, you must implement this delegate method (instead of theapplicationDidFinishLaunching:
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也没有问题),鸣谢这位做好事不留名的英雄。