Android调用其他应用打开各种附件(.doc/.pdf/.xls)

最近重构掌上重邮的写下载文件附件时遇到了一个问题:

附件下载完成后需要打开文件啊,不同文件打开怎么处理才优雅呢?

成功操作后踩坑记录整理于此

更新

Android N 版本适配

避雷

本文例子均为Kotlin编写

思路

首先打开文件这种跳转肯定是发intent出去,然后就是怎么传递文件信息和要打开的文件类型了

这里我发现了intent.setDataAndType(uri, type)

正文

那么整体的写法就应该是:

注意,这段代码有俩坑,后文说明

1
2
3
4
5
6
7
8
9
if (file.exists()) {
try {
startActivity(Intent(Intent.ACTION_VIEW)
.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
.setDataAndType(Uri.fromFile(file), FileTypeHelper.getMIMEType(file)))
} catch (e: Exception) {
e.printStackTrace()
}
}

这里关键的就是获取type的方法,我单独写了一个Helper来处理,因为type毕竟很多,而且需要多次判断。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
import java.io.File

/**
* Author: Hosigus
* Date: 2018/9/27 17:56
* Description: 打开对应文件需要的type
*/
object FileTypeHelper {
private val MIME_TABLE = mapOf(".doc" to "application/msword",
".docx" to "application/vnd.openxmlformats-officedocument.wordprocessingml.document",
".xls" to "application/vnd.ms-excel",
".xlsx" to "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet",
".pdf" to "application/pdf",
".pps" to "application/vnd.ms-powerpoint",
".ppt" to "application/vnd.ms-powerpoint",
".pptx" to "application/vnd.openxmlformats-officedocument.presentationml.presentation",
".z" to "application/x-compress",
".zip" to "application/x-zip-compressed")

/**
* 获取文件类型
*/
fun getMIMEType(file: File): String {
var type = "*/*"
val fName = file.name

val dotIndex = fName.lastIndexOf(".")
if (dotIndex < 0) {
return type
}

val end = fName.substring(dotIndex, fName.length).toLowerCase()
if (end.isBlank()) return type

type = MIME_TABLE[end] ?: return type
return type
}
}

有一堆映射,查找起来就特别快特别方便了,再加上一些小判断,保证传出去的type不会有问题,毕竟要保证打开文件出错不是在我的应用内嘛 (笑

看起来一切都很完美了,那么实际跑起来又如何?

android.os.FileUriExposedException

运行在Android 7.0的系统上时,会发现这个报错,Google 官方解释:

对于面向 Android 7.0 的应用,Android 框架执行的 StrictMode API 政策禁止在应用外部公开 file:// URI,即当把targetSdkVersion指定成24及之上并且在API>=24的设备上运行时,如果一项包含文件 URI 的 intent 离开应用(如分享),则应用出现故障,并出现 FileUriExposedException 异常。

也就是说,Uri.fromFile(file) 这个方法在高版本不适用,正确的解决方案是通过FileProvider 来进行操作,更具体的可以看下这篇博客.

总之,编写了ContentProvider相关配置后,原代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
if (file.exists()) {
try {
startActivity(Intent(Intent.ACTION_VIEW)
.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
.setDataAndType(file.uri, FileTypeHelper.getMIMEType(file)))
} catch (e: Exception) {
e.printStackTrace()
}
}

val File.uri: Uri
get() = if (Build.VERSION.SDK_INT >= 24) {
FileProvider.getUriForFile(context, authority, this)
} else {
Uri.fromFile(this)
}

运行之后发现成功跳转,但是!跳转目标应用却无法打开文件,查看后发现的确不是文件损坏,用文件管理器是能打开的。那么原因肯定在于代码错误。

分析发现,不仅仅在Provider提供公开文件,还需要在Flag里提供可读权限,最终代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
if (file.exists()) {
try {
startActivity(Intent(Intent.ACTION_VIEW)
.addFlags(if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_GRANT_READ_URI_PERMISSION
} else {
Intent.FLAG_ACTIVITY_NEW_TASK
})
.setDataAndType(file.uri, FileTypeHelper.getMIMEType(file)))
} catch (e: Exception) {
e.printStackTrace()
}
}

val File.uri: Uri
get() = if (Build.VERSION.SDK_INT >= 24) {
FileProvider.getUriForFile(context, authority, this)
} else {
Uri.fromFile(this)
}

至此,本篇博客结束,所以这个Android版本适配问题,还是蛮考验人经历的。

文章目录
  1. 1. 更新
  2. 2. 避雷
  3. 3. 思路
  4. 4. 正文