iOS开发-选取、浏览和预览本地、共享或者iCloud文件

前言:

最近做了一个项目,需要实现能够选取图片或者word、pdf、txt等文档类型的文件并进行上传,并且能够实现文件的下载和预览功能,以前没有做过相关的功能,所以在网上查找相关资料之后,终于实现了,现分享下浏览和预览文件的实现过程:

一、文件浏览

实现文件浏览有以下两个类:

1、UIDocumentBrowserViewController

这个是用于浏览本地和云中存储的文件并对其执行操作,由于是iOS11之后新出的并没有使用到它,这里就不讲述这个是如何实现的。

2、UIDocumentPickerViewController

我所使用的就是这个类,打破应用沙盒的限制,提供对应用沙盒外的文件或目标的访问。具体代码如下:

创建:

//懒加载
- (UIDocumentPickerViewController *)documentPickerVC {
    if (!_documentPickerVC) {
        /**
         初始化
         
         @param allowedUTIs 支持的文件类型数组
         @param mode 支持的共享模式
         */
        self.documentPickerVC = [[UIDocumentPickerViewController alloc] initWithDocumentTypes:self.documentTypes inMode:UIDocumentPickerModeOpen];
        //设置代理
        _documentPickerVC.delegate = self;
        //设置模态弹出方式
        _documentPickerVC.modalPresentationStyle = UIModalPresentationFormSheet;
    }
    return _documentPickerVC;
}

实现代理方法:

#pragma mark - UIDocumentPickerDelegate

- (void)documentPicker:(UIDocumentPickerViewController *)controller didPickDocumentsAtURLs:(NSArray<NSURL *> *)urls {
    //获取授权
    BOOL fileUrlAuthozied = [urls.firstObject startAccessingSecurityScopedResource];
    if (fileUrlAuthozied) {
        //通过文件协调工具来得到新的文件地址,以此得到文件保护功能
        NSFileCoordinator *fileCoordinator = [[NSFileCoordinator alloc] init];
        NSError *error;

        [fileCoordinator coordinateReadingItemAtURL:urls.firstObject options:0 error:&error byAccessor:^(NSURL *newURL) {
            //读取文件
            NSString *fileName = [newURL lastPathComponent];
            NSError *error = nil;
            NSData *fileData = [NSData dataWithContentsOfURL:newURL options:NSDataReadingMappedIfSafe error:&error];
            if (error) {
                //读取出错
            } else {
                //上传
                [self uploadingWithFileData:fileData fileName:fileName fileURL:newURL];
            }

            [self dismissViewControllerAnimated:YES completion:NULL];
        }];
        [urls.firstObject stopAccessingSecurityScopedResource];
    } else {
        //授权失败
    }
}

在代理方法里可以选取得到文件,再进行上传或者其他操作。

文件浏览实现效果图:(由于是模拟器截图,所以没有任何文件,见谅!)
文件浏览效果图1-最近使用.png
文件浏览效果图2-iCloud Drive.png
文件浏览效果图3-浏览.png
PS:

至于图片选取和浏览,大家都比较熟悉,这里就不再废话了。

二、文件预览

文件预览我使用的是UIDocumentInteractionController,这个类可以预览,打开或打印其文件格式无法直接由您的应用程序处理的文件。实现代码如下:

创建对象:

//懒加载
- (UIDocumentInteractionController *)documentInteractionC {
    if (!_documentInteractionC) {
        //初始化
        self.documentInteractionC = [[UIDocumentInteractionController alloc] init];
        //设置代理
        _documentInteractionC.delegate = self;
    }
    return _documentInteractionC;
}

实现代理方法:

#pragma mark - UIDocumentInteractionControllerDelegate
//返回一个视图控制器,代表在此视图控制器弹出
- (UIViewController *)documentInteractionControllerViewControllerForPreview:(UIDocumentInteractionController *)controller {
    return self;
}
//返回一个视图,将此视图作为父视图
- (UIView *)documentInteractionControllerViewForPreview:(UIDocumentInteractionController *)controller {
    return self.view;
}
//返回一个CGRect,做为预览文件窗口的坐标和大小
- (CGRect)documentInteractionControllerRectForPreview:(UIDocumentInteractionController *)controller {
    return self.view.frame;
}

得到文件的URL,指定URL并打开预览窗口:

NSURL *URL = self.fileURLArray[indexPath.row];
    if ([URL isFileURL]) {
        if ([[[NSFileManager alloc] init] fileExistsAtPath:URL.path]) {
            //指定预览文件的URL
            self.documentInteractionC.URL = URL;
            //弹出预览文件窗口
            [self.documentInteractionC presentPreviewAnimated:YES];
        } else {
            [SVProgressHUD showErrorWithStatus:@"该文件不存在!"];
        }
    } else {
        if ([URL.scheme containsString:@"http"]) {
            UIAlertController *alertC = [UIAlertController alertControllerWithTitle:@"提示" message:@"不存在本地文件,需要下载,是否下载?" preferredStyle:UIAlertControllerStyleAlert];
            [alertC addAction:[UIAlertAction actionWithTitle:@"确定" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
                //文件存储路径
                NSString *filePathString = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES).firstObject stringByAppendingPathComponent:URL.absoluteString.lastPathComponent];
                NSURLSessionDownloadTask *downloadTask = [self.manager downloadTaskWithRequest:[NSURLRequest requestWithURL:URL] progress:^(NSProgress * _Nonnull downloadProgress) {
                    
                } destination:^NSURL * _Nonnull(NSURL * _Nonnull targetPath, NSURLResponse * _Nonnull response) {
                    //返回文件存储路径
                    return [NSURL fileURLWithPath:filePathString];
                } completionHandler:^(NSURLResponse * _Nonnull response, NSURL * _Nullable filePath, NSError * _Nullable error) {
                    
                    //下载完成后,替换原来的URL,下次不用再下载,直接打开
                    [self.fileURLArray replaceObjectAtIndex:indexPath.row withObject:filePath];
                    //指定预览文件的URL
                    self.documentInteractionC.URL = filePath;
                    //弹出预览文件窗口
                    [self.documentInteractionC presentPreviewAnimated:YES];
                    
                }];
                [MBProgressHUD showHUDAddedTo:self.view animated:YES];
                MBProgressHUD *hud = [MBProgressHUD HUDForView:self.view];
                hud.label.text = @"正在下载文件";
                //启动下载文件任务
                [downloadTask resume];
            }]];
            [alertC addAction:[UIAlertAction actionWithTitle:@"取消" style:UIAlertActionStyleCancel handler:^(UIAlertAction * _Nonnull action) {
                
            }]];
            [self presentViewController:alertC animated:YES completion:^{
                
            }];
        } else {
            [SVProgressHUD showErrorWithStatus:@"该文件链接不存在!"];
        }
    }
文件预览实现效果图:
文件预览效果图.png

到这里就简单的实现文件的选取、浏览和预览了,如果写得有什么错误的地方,欢迎大家批评指正!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,271评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,275评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,151评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,550评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,553评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,559评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,924评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,580评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,826评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,578评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,661评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,363评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,940评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,926评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,156评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,872评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,391评论 2 342

推荐阅读更多精彩内容

  • 2016年12月7日星期三 8点15分 好大的雪,粉妆玉砌,原来适用于此种境界。 我走向校园的时候,脚底是松软的积...
    悦者阅读 645评论 -1 3
  • 践行的结果一:我开始坚持写作。 放两篇日记有二个目的,第一个是通过日记可以看出我缓慢成长的过程,可以看出我的变化。...
    认知成长笔记阅读 201评论 0 0
  • 今天是星期五了,我要关闭我的周一到周五模式,开启周六周日模式。一般在周五我对家里某一部分进行大清理,平...
    静起风云阅读 231评论 1 0
  • 当我意识到自己犯下大错时,我的生活已成了一团乱麻。每次我跟我的丈夫说“我们为什么不能再作进一步尝试呢”或“我...
    不懒得妞阅读 326评论 0 2