iOS中的沙盒可以让平台更加的安全,这也是沙盒给用户带来的最主要好处。不过由于沙盒的严格限制,导致程序之间共享数据比较麻烦。一般在程序间共享文档可以通过UIDocumentInteractionController(该类经常被开发者忽略)。本文中,我将介绍如何使用这个类在其它程序(已经安装在设备中的程序)中预览和打开文档。
UIDocumentInteractionController在iOS 3.2中就已经存在了,使用起来非常灵活,功能也比较强大。它除了支持同设备上app之间的文档分享外,还可以实现文档的预览、打印、发邮件以及复制。
UIDocumentInteractionController的使用非常简单。首先通过调用它唯一的类方法 interactionControllerWithURL:,并传入一个URL(NSURL),来初始化一个实例对象。之后设置一下这个view controller的delegate属性,并实现一些恰当的delegate方法。
注意,UIDocumentInteractionController并不是UIViewController的子类,所以必须通知document interaction controller使用哪个view controller来预览文档。
Step 1: 设置项目
在同设备app之间打开文档非常有用,一个常见的实例是在图片编辑器中打开图片,比如iPhoto。
在 Xcode中创建一个新项目,选择“Single View Application”模板。命名文档,键入公司标识符,Device选择iPhone,设备下边选择“Use Automatic Reference Counting”,其他两项不选。然后点击“Next”,保存项目,点击“Creat”按钮。
Step 2: 创建用户界面
这个程序的用户界面包括两个按钮,一个是用于在其他app中预览PDF文档,另一个是用户在其他app中打开PDF文档。创建用户界面之前,在viewcontroller执行文件中为每个按钮赋予一个动作,如下:
1.- (IBAction)previewDocument:(id)sender {
2.}
1.- (IBAction)openDocument:(id)sender {
2.}
选择MTViewController.xib,我们需要从右边view controller视图中拖拽两个UIButton实例。选择左边的File’s Owner objectobject,打开Connections Inspector,把先前创建的动作和按钮连接起来。
Step 3:预览文档
现在支持的是PDF文档,你可以使用任何PDF文档,但是在关于这个技巧的源文件中,我已经包含了一个PDF例子,就是苹果的iOS编程指南,也可以在线获得。把文档拖至你的项目中,选中“ Copy items into destination group’s folder (ifneeded)”这个复选框,最后确保文件已经添加至下边的“Documents target”中。
使用UIDocumentInteractionController类注意事项:1. 你需要保存着document interation controller的实例。2.需要实现UIDocumentInteractionControllerDelegate协议。
首先更新viewcontroller的头文件(如下所示)来告诉compiler,MTViewController类遵照UIDocumentInteractionControllerDelegate协议。
1.#import <UIKit/UIKit.h>
2.@interface MTViewController : UIViewController<UIDocumentInteractionControllerDelegate>
3.@end
在view controller的实现文件中,添加一个私有属性,类型为UIDocumentInteractionController,并将名称命名为 documentInteractionController。这个属性存储着document interaction controller,之后会用着。
看看previewDocument:方法的实现,首先获得文档的URL (NSURL) ,由于文档是app的一部分,因此通过NSBundle类获得文档的(NSURL)非常容易,如下:
1.- (IBAction)previewDocument:(id)sender {
2. NSURL *URL = [[NSBundle mainBundle] URLForResource:@"sample"withExtension:@"pdf"];
3. if (URL) {
4. // Initialize Document InteractionController
5. self.documentInteractionController= [UIDocumentInteractionController
interactionControllerWithURL:URL];
6. // Configure Document InteractionController
7. [self.documentInteractionControllersetDelegate:self];
8. // Preview PDF
9. [self.documentInteractionController presentPreviewAnimated:YES];
10. }
11.}
如果返回了一个有效的URL,我们初始化一个UIDocumentInteractionController的实例,并且通过文档的URL。在 documentInteractionController的属性中我们存储了一个document interaction controller的引用。view controller将会充当document interaction controller的delegate
如果现在运行app,你会注意到点击预览按钮时,什么都没有发生。因为这里我们首先要实现一个delegate method。
前边提到,UIDocumentInteractionController类并不是UIViewController的子类,而是继承自 NSObject。我们需要通知documentinteraction controller使用哪个view controller进行文档预览。
UIDocumentInteractionControllerDelegate中有一个名documentInteractionControllerViewControllerForPreview:的delegate方法,该方法请求获得一个用于显示(预览)文档的viewcontroller。
我们希望在main viewcontroller中显示预览,所以可简单的返回self,如下代码所示。它的意思是document interfation controller将使用我们的view controller来预览PDF文档——以modal view的方式显示文档。
1.- (UIViewController *)documentInteractionControllerViewControllerForPreview:
(UIDocumentInteractionController *) controller {
2. return self;
3.
当然你可以简化documentInteractionControllerViewControllerForPreview:的实现以满足你的需要。执行委托方法的同时,你可以首次运行app,试试看这个效果,你可以通过邮件分享这个文档,可以打印或者复制。另外,也可以在支持该文档类型的其他app中打开文档,在图中点击右边的按钮,看看我说的什么意思。
Step 4: 打开文档
为了实现这一目的我们需要实现openDocument:方法。在openDocument:方法中,获取到在程序bundle中一个PDF文件的url,用这个url初始化一个UIDocumentInteractionController。
之后设置一下UIDocumentInteractionController的 delegate,在这个UIDocumentInteractionController中调用 presentOpenInMenuFromRect:inView:方法显示一个菜单。传入的第一个参数CGRect是button的frame,如下所示:
1.- (IBAction)openDocument:(id)sender {
2. UIButton *button = (UIButton *)sender;
3. NSURL *URL = [[NSBundle mainBundle]URLForResource:@"sample" withExtension:@"pdf"];
4. if (URL) {
5. // Initialize Document InteractionController
6. self.documentInteractionController= [UIDocumentInteractionController
interactionControllerWithURL:URL];
7. // Configure Document InteractionController
8. [self.documentInteractionController setDelegate:self];
9. // Present Open In Menu
10. [self.documentInteractionControllerpresentOpenInMenuFromRect:[button frame] inView:self.view animated:YES];
11. }
12.}
为了测试openDocument:方法,在真机上运行实例app非常重要。原因很简单,操作系统要检查安装的app是否支持我们要打开的文件类型(UTI)。如果没有找到支持相应文件类型的app,那么菜单中就不会有打开的提示,而这个不能通过iOS模拟器进行测试。
为了测试这个功能,首先要在真机上安装支持PDF的app,比如Dropbox或者Amazon的Kindle app。
总结
使用UIDocumentInteractionController这个类可以简单地实现app之间文档的预览和打开。建议你去看看这个类的参考文档,特别是UIDocumentInteractionControllerDelegate协议——这里面有许多delegate
方法,当遇到大文档或者复杂的工作流时,这些方法都非常的方便。
源文件: