美文网首页
《iOS11中UIDocumentInteractionCont

《iOS11中UIDocumentInteractionCont

作者: XTShow | 来源:发表于2018-06-02 16:02 被阅读233次

    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也没有问题),鸣谢这位做好事不留名的英雄。

    相关文章

      网友评论

          本文标题:《iOS11中UIDocumentInteractionCont

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