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

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

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

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

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

更新

Android N 版本适配

避雷

本文例子均为Kotlin编写

思路

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

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

正文

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

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

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毕竟很多,而且需要多次判断。

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相关配置后,原代码如下:

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里提供可读权限,最终代码如下:

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版本适配问题,还是蛮考验人经历的。

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

推荐阅读更多精彩内容