Android 应用的一大优点就是他们能够相互通信和融合。如果一项功能不是你应用的核心功能,并且在其他应用中已经存在,为何还要再次开发呢?在本篇幅中,将介绍通过 Intent
对象使用 Android Sharesheet
和 Intent Resolver
在应用之间发送和接收简单的数据(例如文本、图片、文件)通用方法。
Android 通过 Intent
及其关联的附加数据让用户可以在喜爱的应用间简单快速分享信息。Android 提供了两种在应用之间分享数据的方法。
当你在构造一个 Intent
时,你必须指定一个想要 Intent
去执行的动作。Android 中使用 ACTION_SEND
动作从一个 Activity
发送数据到另一个 Activity
,甚至可以跨进程发送数据。你需要指定发送的数据及其类型,系统会自动识别能够接收数据的 Activity
并展示给用户(供用户选择一个 Activity
)。在特殊情况下,如果系统中只有一个 Activity
能够处理这个 Intent
, Activity
会立即启动。
通过 Android Sharesheet,用户只需要点击一下,就可以根据相关的应用建议将信息分享给合适的人。
我们强烈建议您使用 Android Sharesheet 在各个应用之间为您的用户打造一致的体验。应用不应显示它们自己的分享目标列表或创建它们自己的 Sharesheet 变体。Sharesheet 可以建议自定义解决方案不可用的目标,并具有一致的排名。这是因为 Sharesheet 可以考虑到只有系统可用的应用及用户 Activity 的相关信息。
Android Sharesheet 还为开发者提供了许多方便的功能。例如:
对所有类型的分享,创建一个 Intent
对象并将其动作设置为 Intent.ACTION_SEND
,要显示 Android Sharesheet界面,调用 Intent.createChooser()
方法并传入 Intent
对象,该方法返回 Intent
的一个版本,通过调用 Context.startActivity()
方法并传入该 Intent
对象,即可显示 Android ShareSheet 界面。
Android ShareSheet 一个最简单通用的用法就是从一个 Activity 向另一个 Activity 发送文本。如下示例所示:
// 定义一个 Intent 对象
val sendingIntent = Intent().apply {
action = Intent.ACTION_SEND // 设置动作
type = "text/plain" // 设置内容类型
putExtra(Intent.EXTRA_TEXT, "This is the message to send") // 添加内容
}
// 使用 Intent.createChooser() 获取 Intent 的一个版本
val intent = Intent.createChooser(sendingIntent, "Send Text")
// 打开 ShareSheet 页面
startActivity(intent)
分享数据时,
Intent
中的内容可以根据需要添加更多信息,比如发送邮件需要接收者信息(EXTRA_EMAIL
(接收者邮箱),EXTRA_CC
(抄送),EXTRA_BCC
(秘密抄送))、邮件主题(EXTRA_SUBJECT
) 等等。接收者信息可以是多个,可以使用putExtra(String, String[])
方法进行廷加。
发送二进制数据,将数据类型设定为指定类型,并将二进制文件的 URI 作为数据传入 EXTRA_STREAM
中,如下示例所示:
// 定义一个 Intent 对象
val sendingIntent = Intent().apply {
action = Intent.ACTION_SEND // 设置动作
type = "image/*" // 设置内容类型
putExtra(Intent.EXTRA_STREAM, uri) // 添加内容
}
// 使用 Intent.createChooser() 获取 Intent 的一个版本
val intent = Intent.createChooser(sendingIntent, "Send Image")
// 打开 ShareSheet 页面
startActivity(intent)
注意事项:必须保证接收数据的应用需要有权限访问传入 Uri 所指向的数据,否则会导致无法访问发送失败。
为了让接收数据的应用有权限访问 Uri 所以对应的数据(文件),可以用一下做法
ContentProvider
中,并确保其他应用有权限访问该内容太提供程序。一种简单的做法就是使用 FileProvider
辅助程序类。MediaStore
,MediaStore
主要用于存储视频、音频和图片类型的数据,不过在 Android 3.0(API Level 11)开始它也支持非媒体类型(例如文件)。可以使用文件可以插入到 MediaStore
,一旦将内容插入到 MediaStore
,其他应用就可以无需权限即可访问。使用Android ShareSheet 分享数据,必须使用正确的数据类型(MIME TYPE),这样系统才能正确分析并列出能够处理的应用程序列表。以下是常见数据类型的介绍:
text/plain
, text/rtf
, text/html
, text/json
,这些是文本类型,接收者必须注册接收 text/*
类型;image/jpg
, image/png
, image/gif
,这些是图片类型,接收者必须注册接收 image/*
类型;video/mp4
, video/3gp
,这些是视频类型,接收者必须注册接收 video/*
类型;application/[文件后缀]
,这些是跳转打开程序,例如 application/pdf
是打开pdf类型文件,接收者必须支持打开对应的文件后缀。*/*
数据类型,但是强烈建议不要这么做,因为仅仅接收必须注册了通用类型才会与之匹配。 有时候我们需要同时分享多份内容,比如分享多张图片。此时,Intent
的动作为 Intent.ACTION_SEND_MULTIPLE
,数据部分为一个 List
列表。如果多份数据类型均一致,指定明确的数据类型,如果无法保证,可以使用通配数据类型(例如:分享多张 jpeg 图片,类型为 image/jpeg
,但是无法确保多张图片均为 jpeg,也有可能为 png 的时候,类型可以使用 image/*
)。如下示例所示:
// 定义一个 Intent 对象
val sendingIntent = Intent().apply {
action = Intent.ACTION_SEND_MULTIPLE // 设置动作
type = "image/*" // 设置内容类型
putParcelableArrayListExtra(Intent.EXTRA_STREAM, uris) // 添加内容
}
// 使用 Intent.createChooser() 获取 Intent 的一个版本
val intent = Intent.createChooser(sendingIntent, "Send Image")
// 打开 ShareSheet 页面
startActivity(intent)
注意事项:必须保证接收数据的应用需要有权限访问传入 Uri 所指向的数据,否则会导致无法访问发送失败。
从Android 10(API Level 29)开始,Android ShareSheet 可以显示分享文本的预览。有时候,分享的文本内容比较难以理解(比如分享链接),此时可以添加丰富的预览内容,帮助用户更加清楚自己分享的内容。可以通过 Intent.EXTRA_TITLE
添加标题内容,或者添加缩略图等。如下示例所示:
// 定义一个 Intent 对象
val sendingIntent = Intent().apply {
action = Intent.ACTION_SEND // 设置动作
type = "text/*" // 设置内容类型
putExtra(Intent.EXTRA_TITLE, "百度一下,全是广告") // 显示标题
putExtra(Intent.EXTRA_TEXT, "https://www.baidu.com") // 添加内容
}
// 使用 Intent.createChooser() 获取 Intent 的一个版本
val intent = Intent.createChooser(sendingIntent, "Send Text")
// 打开 ShareSheet 页面
startActivity(intent)
Android ShareSheet 会自动添加可用的分享目标,供用户选择。开发者也可以添加自己的分享目标,分享目标分为两类,一种是从 ChooserTargetServices
加载的 ChooserTarget
对象(这类目标为应用内部的某个对象,比如某个具体的联系人);另一种则是目标应用项目(这类目标为应用某个 Activity
,比如聊天页面)。这两类分享目标分栏显示,并且开发者只能添加有限数量的自定义分享目标。
添加自定义分享目标很简单,只需要在调用 Intent.createChooser()
之后,在返回的 Intent
对象中传入Intent.EXTRA_CHOOSER_TARGETS
和 Intent.EXTRA_INITIAL_INTENTS
两项数据即可。前者是 ChooserTarget
数组,后者是 Intent
数组。如下示例所示:
val sendingIntent = Intent().apply {
action = Intent.ACTION_SEND // 设置动作
type = "text/*" // 设置内容类型
putExtra(Intent.EXTRA_TITLE, "百度一下,全是广告")
putExtra(Intent.EXTRA_TEXT, "https://www.baidu.com") // 添加内容
}
// 使用 Intent.createChooser() 获取 Intent 的一个版本
val intent = Intent.createChooser(sendingIntent, "Send Text").apply {
// 在 Intent.createChooser() 返回的 Intent 中添加自定义分享目标
putExtra(Intent.EXTRA_INITIAL_INTENTS, arrayOf(
Intent(applicationContext, UIActivity::class.java),
Intent(applicationContext, StorageActivity::class.java),
Intent(applicationContext, SwipeRefreshActivity::class.java),
)
)
注意事项:自定义添加的分享目标,只会显示在推荐列表中,并且显示个数有限。
Android ShareSheet 不仅可以自定义分享目标,还可以排除指定的分享目标。在调用 Intent.createChooser()
之后,在返回的 Intent
对象中传入Intent.EXTRA_EXCLUDE_COMPONENTS
传入排除的目标即可,该数据是 ComponentName
类型对象数组。如下示例所示:
// 定义一个 Intent 对象
val sendingIntent = Intent().apply {
action = Intent.ACTION_SEND // 设置动作
type = "text/*" // 设置内容类型
putExtra(Intent.EXTRA_TITLE, "百度一下,全是广告")
putExtra(Intent.EXTRA_TEXT, "https://www.baidu.com") // 添加内容
}
val pi = PendingIntent.getBroadcast(applicationContext, 10000,
Intent(applicationContext, ShareReceiver::class.java),
PendingIntent.FLAG_UPDATE_CURRENT)
// 使用 Intent.createChooser() 获取 Intent 的一个版本
val intent = Intent.createChooser(sendingIntent, "Send Text", pi.intentSender).apply {
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
// 添加排除目标
putExtra(Intent.EXTRA_EXCLUDE_COMPONENTS, arrayOf(
// 排除 Gmail
ComponentName("com.google.android.gm", "com.google.android.gm.ComposeActivityGmailExternal"),
// 排除蓝牙
ComponentName("com.android.bluetooth", "com.android.bluetooth.opp.BluetoothOppLauncherActivity"),
)
)
}
}
// 打开 ShareSheet 页面
startActivity(intent)
注意事项:
1. 隐藏分享目标必须指定正确的ComponentName
才能隐藏;
2. 隐藏分享目标只在 Android 7.0 (API Level 24)开始才可用。
在某些情况下,我们需要了解用户何时分享,点击了分享目标中的哪一个项目。Android ShareSheet 可以通过 IntentSender
获取到用户点击的分享目标的 ComponentName
。首先,创建一个 BroadcastReceiver
类,然后为 BroadcastReceiver
创建一个 PenddingIntent
对象,最后使用 PenddingIntent
对象生成一个 IntentSender
并将其作为 Intent.createChooser()
的参数传入。这样一来,就可以在 BroadcastReceiver
的 onReceive()
回调中收到点击的目标对应的 ComponentName
了。onReceive()
回调中没有动作值,数据通过 Intent
对象的 Intent.EXTRA_CHOSEN_COMPONENT
返回。如下示例所示:
// 1. 定义一个 BoradcastReceiver 类
class ShareReceiver : BroadcastReceiver() {
override fun onReceive(context: Context, intent: Intent) {
// This method is called when the BroadcastReceiver is receiving an Intent broadcast.
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP_MR1) {
val clickedComponent : ComponentName? = intent.getParcelableExtra(Intent.EXTRA_CHOSEN_COMPONENT);
}
}
}
// 2. 为 BroadcastReceiver 创建一个 PenddingIntent 对象
val pi = PendingIntent.getBroadcast(applicationContext, 10000,
Intent(applicationContext, ShareReceiver::class.java),
PendingIntent.FLAG_UPDATE_CURRENT)
// 3. PenddingIntent 对象生成一个 IntentSender 对象,并作为Intent.createChooser() 的参数传入
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP_MR1) {
// 使用 Intent.createChooser() 获取 Intent 的一个版本
val intent = Intent.createChooser(sendingIntent, "Send Text", pi.intentSender)
// 打开 ShareSheet 页面
startActivity(intent)
}
注意事项:获取点击目标的
ComponentName
信息只在 Android 5.1 (API Level 22)开始才可用。
将数据发送到一个明确的应用时,使用Android Intent 解析器是最合适的。使用 Android Intent 解析器,跟使用 ShareSheet 类似,创建一个 Intent
对象,并设置动作、数据类型以及并根据需要传入额外数据,但是不要调用 Intent.createChooser()
。然后调用 Context.startActivity()
(或者 Context.startActivityForResult()
)并将 Intent
对象传入,Intent 解析器就会解析传入的 Intent
对象。如果 系统中安装的应用中存在多个应用跟 Intent
对象的动作值和类型相匹配,系统会显示一个歧义对话框,允许用户选择一个目标处理请求;如果只有一个应用匹配,将会直接使用该应用运行处理。如下示例所示:
val sendingIntent = Intent().apply {
action = Intent.ACTION_SEND // 设置动作
type = "image/*" // 设置内容类型
putExtra(Intent.EXTRA_STREAM, uri) // 添加内容
}
startActivity(sendingIntent)
一个应用可以给其他应用发送数据,同样也可以从其他应用接收数据。通过 Android ShareSheet 或者 Intent 解析器向其他应用发送数据时,数据中会包含媒体类型。应用可以通过三种方式接收其他应用发送的数据:
AndroidManifest.xml
清单文件中有匹配的
标记的 Activity;ChooserTargetService
返回的一个或多个 ChooserTarget
对象;ChooserTarget
对象。但只有当您用在 Android 10(API 级别 29)平台上运行时才可以使用。 共享快捷方式和 ChooserTarget
对象都是深入到应用中特定 Activity 的直接共享链接。它们通常代表一个真实的人,并由 Android Sharesheet 显示。例如:社交聊天工具里的某个好友。
如果应用需要接收来自其他应用的数据,就应当配置成接收尽可能广泛的媒体类型。例如:社交聊天工具接收文本、图片、视频消息,应该配置成接收 text/*
、image/*
、video/*
。
text/*
:可接收发送者发送的类型有:text/plain
、text/rtf
,、text/html
、text/json
;image/*
:可接收发送者发送的类型有:image/jpg
、image/png
、image/gif
;video/*
:可接收发送者发送的类型有:video/mp4
、video/3gp
;application/[文件后缀]
:可接收发送者发送的打开指定文件后缀的文件的请求。注意事项:除非您的应用可以接收任何类型的数据,否则不要使用
*/*
,否则任何应用发送数据时,您的应用都会在接收列表中。
Intent 过滤器会告知系统,应用组件可以接收什么样的 Intent。跟 向其他应用发送简单的数据 中构建包含动作(例如:ACTION_SEND
)的 Intent 类似,创建 Intent 过滤器,就是为了接收指定动作的 Intent 数据。在清单文件(AndroidManifest.xml
)中的
标签内部使用
标签定义 Intent 过滤器。如下示例所示:
<activity android:name=".share.ShareDataActivity" >
<intent-filter>
<action android:name="android.intent.action.SEND" />
<category android:name="android.intent.category.DEFAULT" />
<data android:mimeType="text/*" />
intent-filter>
<intent-filter>
<action android:name="android.intent.action.SEND" />
<category android:name="android.intent.category.DEFAULT" />
<data android:mimeType="image/*" />
intent-filter>
<intent-filter>
<action android:name="android.intent.action.SEND" />
<category android:name="android.intent.category.DEFAULT" />
<data android:mimeType="video/*" />
intent-filter>
activity>
当其他应用尝试通过用指定动作类型和数据媒体类型的 Intent 对象调用 startActivity()
(或 startActivityForResult()
)分享数据时,定义了 Intent 过滤器的应用就会出现在 Android ShareSheet 或者 Intent 解析器列表中。如果用户选择了您的应用,你将会在定义了 Intent 过滤器的 Activity 将会被启动并且接收到其他应用传递过来的数据。
其他应用发过来的数据,是通过 Intent 对象运载的,在 Activity 中接收传递过来的数据,首先要先调用 getIntent()
获取 Intent 对象,然后检查 Intent 对象的内容,再决定下一步的操作。切忌,如果一个 Activity 可以被系统其他应用或者启动器启动,必须要要校验 Intent 的内容。你永远不知道另一个应用会发送什么内容给你,所以在 Activity 中对 Intent 内容进行校验,分别对 Intent 的 动作类型 和 数据媒体类型 进行校验,并进行相应的处理。如下示例所示:
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
intent?.apply {
when(action) {
Intent.ACTION_SEND -> {
if("text/plain" == type) {
// 处理 text/plain 类型数据
} else if ("image/jpeg" == type) {
// 处理 image/jpeg 类型数据
}
}
Intent.ACTION_SEND_MULTIPLE -> {
// .....
}
}
}
}