Android 使用 Intent 及其关联的额外数据,让用户能够使用他们最喜欢的应用快速轻松地共享信息。
Android 提供了两种方法供用户在应用之间共享数据
- Android 共享表单主要用于将内容发送到您的应用外部和/或直接发送到另一个用户。例如,与朋友共享 URL。
- Android Intent 解析器最适合将数据传递到定义良好的任务的下一阶段。例如,从您的应用中打开 PDF 并让用户选择他们首选的查看器。
构建 Intent 时,您需要指定 Intent 要执行的操作。Android 使用操作 ACTION_SEND
将数据从一个活动发送到另一个活动,甚至跨越进程边界。您需要指定数据及其类型。系统会自动识别可以接收数据的兼容活动,并将它们显示给用户。在 Intent 解析器的情况下,如果只有一个活动可以处理 Intent,则该活动会立即启动。
为什么要使用 Android 共享表单
我们强烈建议使用 Android 共享表单,以便为您的用户在各个应用中创建一致性。不要显示您应用自己的共享目标列表或创建您自己的共享表单变体。
Android 共享表单允许用户与合适的人员共享信息,并提供相关的应用建议,所有这些只需轻点一下即可。共享表单可以建议自定义解决方案无法使用的目标,并使用一致的排名。这是因为共享表单可以考虑有关应用和用户活动的信息,而这些信息仅对系统可用。
Android 共享表单还为开发者提供了许多方便的功能。例如,您可以执行以下操作
使用 Android 共享表单
对于所有类型的共享,请创建一个 Intent 并将其操作设置为 Intent.ACTION_SEND
。要显示 Android 共享表单,请调用 Intent.createChooser()
,并将您的 Intent
对象传递给它。它会返回您的 Intent 的一个版本,该版本始终显示 Android 共享表单。
发送文本内容
Android 共享表单最直接和最常见的用法是从一个活动向另一个活动发送文本内容。例如,大多数浏览器可以将当前显示页面的 URL 作为文本与另一个应用共享。这对于通过电子邮件或社交网络与朋友共享文章或网站很有用。以下是如何执行此操作的示例
Kotlin
val sendIntent: Intent = Intent().apply { action = Intent.ACTION_SEND putExtra(Intent.EXTRA_TEXT, "This is my text to send.") type = "text/plain" } val shareIntent = Intent.createChooser(sendIntent, null) startActivity(shareIntent)
Java
Intent sendIntent = new Intent(); sendIntent.setAction(Intent.ACTION_SEND); sendIntent.putExtra(Intent.EXTRA_TEXT, "This is my text to send."); sendIntent.setType("text/plain"); Intent shareIntent = Intent.createChooser(sendIntent, null); startActivity(shareIntent);
或者,您可以添加额外信息以包含更多信息,例如电子邮件收件人 (EXTRA_EMAIL
、EXTRA_CC
、EXTRA_BCC
)、电子邮件主题 (EXTRA_SUBJECT
) 等。
注意:某些电子邮件应用(例如 Gmail)需要 String[]
才能使用 EXTRA_EMAIL
和 EXTRA_CC
等额外信息。使用 putExtra(String, String[])
将这些信息添加到您的 Intent 中。
发送二进制内容
使用 ACTION_SEND
操作共享二进制数据。设置适当的 MIME 类型,并将指向数据的 URI 放入额外的 EXTRA_STREAM
中,如下例所示。这通常用于共享图像,但可以用于共享任何类型的二进制内容。
Kotlin
val shareIntent: Intent = Intent().apply { action = Intent.ACTION_SEND // Example: content://com.google.android.apps.photos.contentprovider/... putExtra(Intent.EXTRA_STREAM, uriToImage) type = "image/jpeg" } startActivity(Intent.createChooser(shareIntent, null))
Java
Intent shareIntent = new Intent(); shareIntent.setAction(Intent.ACTION_SEND); // Example: content://com.google.android.apps.photos.contentprovider/... shareIntent.putExtra(Intent.EXTRA_STREAM, uriToImage); shareIntent.setType("image/jpeg"); startActivity(Intent.createChooser(shareIntent, null));
接收应用程序需要权限才能访问 Uri
指向的数据。建议使用两种方法来实现这一点
- 将数据存储在您自己的
ContentProvider
中,确保其他应用程序具有访问您提供程序的正确权限。提供访问权限的首选机制是使用 每个 URI 权限,这些权限是临时的,并且仅授予接收应用程序访问权限。创建此类ContentProvider
的一种简单方法是使用FileProvider
辅助类。 - 使用系统
MediaStore
。MediaStore
主要用于视频、音频和图像 MIME 类型。但是,从 Android 3.0(API 级别 11)开始,它也可以存储非媒体类型。有关更多信息,请参阅MediaStore.Files
。可以使用scanFile()
将文件插入到MediaStore
中,之后会将适合共享的content://
样式Uri
传递给提供的onScanCompleted()
回调。请注意,一旦添加到系统MediaStore
中,任何设备上的应用程序都可以访问该内容。
使用正确的 MIME 类型
为要发送的数据提供最具体的 MIME 类型。例如,在共享纯文本时使用 text/plain
。以下是在 Android 中发送简单数据时的一些常见 MIME 类型
接收方注册 | 发送方发送 |
---|---|
text/* |
|
`image/*` |
|
video/* |
|
支持的文件扩展名 | application/pdf |
有关 MIME 类型的更多信息,请参阅 IANA MIME 媒体类型的官方注册表。
Android 共享表单可能会显示内容预览,具体取决于提供的 MIME 类型。某些预览功能仅适用于特定类型。
共享多个内容片段
要共享多个内容片段,请将 ACTION_SEND_MULTIPLE
操作与指向内容的 URI 列表一起使用。MIME 类型根据您共享的内容组合而异。例如,如果您共享三个 JPEG 图像,则使用类型 "image/jpg"
。对于图像类型的混合,请使用 "image/*"
以匹配处理任何类型图像的活动。虽然可以共享多种类型,但我们强烈建议不要这样做,因为接收方不清楚想要发送的内容。如果必须发送多种类型,请使用 "*/*"
。接收应用程序负责解析和处理您的数据。这是一个示例
Kotlin
val imageUris: ArrayList<Uri> = arrayListOf( // Add your image URIs here imageUri1, imageUri2 ) val shareIntent = Intent().apply { action = Intent.ACTION_SEND_MULTIPLE putParcelableArrayListExtra(Intent.EXTRA_STREAM, imageUris) type = "image/*" } startActivity(Intent.createChooser(shareIntent, null))
Java
ArrayList<Uri> imageUris = new ArrayList<Uri>(); imageUris.add(imageUri1); // Add your image URIs here imageUris.add(imageUri2); Intent shareIntent = new Intent(); shareIntent.setAction(Intent.ACTION_SEND_MULTIPLE); shareIntent.putParcelableArrayListExtra(Intent.EXTRA_STREAM, imageUris); shareIntent.setType("image/*"); startActivity(Intent.createChooser(shareIntent, null));
确保提供的 Uri
对象指向接收应用程序可以访问的数据。
向文本预览添加丰富的内容
从 Android 10(API 级别 29)开始,Android 共享表单显示共享文本的预览。在某些情况下,共享的文本可能难以理解。请考虑共享一个复杂的 URL,例如 https://www.google.com/search?ei=2rRVXcLkJajM0PEPoLy7oA4
。更丰富的预览可以向您的用户保证正在共享的内容。
如果您正在预览文本,则可以设置标题、缩略图图像或两者兼而有之。在调用 Intent.createChooser()
之前,将说明添加到 Intent.EXTRA_TITLE
中,并使用 ClipData
添加相关的缩略图。
注意:图像内容 URI 来自 FileProvider
,通常来自已配置的 <cache-path>
。有关更多信息,请参阅 共享文件。请务必授予共享表单读取任何您想用作缩略图的图像的权限。有关更多信息,请参阅 Intent.FLAG_GRANT_READ_URI_PERMISSION
。
这是一个示例
Kotlin
val share = Intent.createChooser(Intent().apply { action = Intent.ACTION_SEND putExtra(Intent.EXTRA_TEXT, "https://developer.android.com/training/sharing/") // (Optional) Here you're setting the title of the content putExtra(Intent.EXTRA_TITLE, "Introducing content previews") // (Optional) Here you're passing a content URI to an image to be displayed data = contentUri flags = Intent.FLAG_GRANT_READ_URI_PERMISSION }, null) startActivity(share)
Java
Intent sendIntent = new Intent(Intent.ACTION_SEND); sendIntent.putExtra(Intent.EXTRA_TEXT, "https://developer.android.com/training/sharing/"); // (Optional) Here you're setting the title of the content sendIntent.putExtra(Intent.EXTRA_TITLE, "Introducing content previews"); // (Optional) Here you're passing a content URI to an image to be displayed sendIntent.setData(contentUri); sendIntent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); // Show the Sharesheet startActivity(Intent.createChooser(sendIntent, null));
预览如下所示
向共享表单添加自定义操作
在 Android 14(API 级别 34)及更高版本上,应用程序可以向 Android 共享表单添加自定义操作。自定义操作显示为 Android 共享表单顶部的较小操作图标,并且应用程序可以指定任何 Intent
作为单击图标时调用的操作。
要向 Android 共享表单添加自定义操作,首先使用 ChooserAction.Builder
创建一个 ChooserAction
。您可以指定一个 PendingIntent
作为单击图标时调用的操作。创建一个包含所有自定义操作的数组,并将其指定为共享 Intent
的 EXTRA_CHOOSER_CUSTOM_ACTIONS
。
Kotlin
val sendIntent = Intent(Intent.ACTION_SEND) .setType("text/plain") .putExtra(Intent.EXTRA_TEXT, text) val shareIntent = Intent.createChooser(sendIntent, null) val customActions = arrayOf( ChooserAction.Builder( Icon.createWithResource(context, R.drawable.ic_custom_action), "Custom", PendingIntent.getBroadcast( context, 1, Intent(Intent.ACTION_VIEW), PendingIntent.FLAG_IMMUTABLE or PendingIntent.FLAG_CANCEL_CURRENT ) ).build() ) shareIntent.putExtra(Intent.EXTRA_CHOOSER_CUSTOM_ACTIONS, customActions) context.startActivity(shareIntent)
Java
Intent sendIntent = new Intent(Intent.ACTION_SEND) .setType("text.plain") .putExtra(Intent.EXTRA_TEXT, text); Intent shareIntent = Intent.createChooser(sendIntent, null); ChooserAction[] actions = new ChooserAction[]{ new ChooserAction.Builder( Icon.createWithResource(context, R.drawable.ic_custom_action), "Custom", PendingIntent.getBroadcast( context, 1, new Intent(Intent.ACTION_VIEW), PendingIntent.FLAG_IMMUTABLE | PendingIntent.FLAG_CANCEL_CURRENT ) ).build() }; shareIntent.putExtra(Intent.EXTRA_CHOOSER_CUSTOM_ACTIONS, actions); context.startActivity(shareIntent);
添加自定义目标
Android 共享表单允许您指定最多两个 ChooserTarget
对象,这些对象显示在共享快捷方式和从 ChooserTargetServices
加载的目标选择器目标之前。您还可以指定最多两个指向活动的 Intent,这些活动列在应用程序建议之前
在调用 Intent.createChooser()
*之后*,将 Intent.EXTRA_CHOOSER_TARGETS
和 Intent.EXTRA_INITIAL_INTENTS
添加到您的共享 Intent 中
Kotlin
val share = Intent.createChooser(myShareIntent, null).apply { putExtra(Intent.EXTRA_CHOOSER_TARGETS, myChooserTargetArray) putExtra(Intent.EXTRA_INITIAL_INTENTS, myInitialIntentArray) }
Java
Intent shareIntent = Intent.createChooser(sendIntent, null); share.putExtra(Intent.EXTRA_CHOOSER_TARGETS, myChooserTargetArray); share.putExtra(Intent.EXTRA_INITIAL_INTENTS, myInitialIntentArray);
谨慎使用此功能。您添加的每个自定义 Intent
和 ChooserTarget
都会减少系统建议的数量。我们通常不建议添加自定义目标。添加 Intent.EXTRA_INITIAL_INTENTS
的一个常见适当示例是提供用户可以在共享内容上执行的其他操作。例如,用户共享图像,并且 Intent.EXTRA_INITIAL_INTENTS
用于让他们改为发送链接。添加 Intent.EXTRA_CHOOSER_TARGETS
的一个常见适当示例是显示您的应用程序提供的相关人员或设备。
按组件排除特定目标
您可以通过提供 Intent.EXTRA_EXCLUDE_COMPONENTS
来排除特定目标。仅在删除受您控制的目标时才执行此操作。一个常见的用例是在用户从您的应用程序内部共享时隐藏您应用程序的共享目标,因为他们的意图可能是共享到您的应用程序外部。
在调用 Intent.createChooser()
后,将 Intent.EXTRA_EXCLUDE_COMPONENTS
添加到您的 Intent 中
Kotlin
val share = Intent.createChooser(Intent(), null).apply { // Only use for components you have control over val excludedComponentNames = arrayOf(ComponentName("com.example.android", "ExampleClass")) putExtra(Intent.EXTRA_EXCLUDE_COMPONENTS, excludedComponentNames) }
Java
Intent shareIntent = Intent.createChooser(new Intent(), null); // Only use for components you have control over ComponentName[] excludedComponentNames = { new ComponentName("com.example.android", "ExampleClass") }; shareIntent.putExtra(Intent.EXTRA_EXCLUDE_COMPONENTS, excludedComponentNames);
获取有关共享的信息
了解用户何时共享以及他们选择哪个目标可能很有用。Android 共享表单允许您通过使用 IntentSender
提供用户选择的目标的 ComponentName
来获取此信息。
首先为 BroadcastReceiver
创建一个 PendingIntent
,并在 Intent.createChooser()
中提供其 IntentSender
Kotlin
var share = Intent(Intent.ACTION_SEND) // ... val pi = PendingIntent.getBroadcast( myContext, requestCode, Intent(myContext, MyBroadcastReceiver::class.java), PendingIntent.FLAG_MUTABLE or PendingIntent.FLAG_UPDATE_CURRENT ) share = Intent.createChooser(share, null, pi.intentSender)
Java
Intent share = new Intent(ACTION_SEND); ... PendingIntent pi = PendingIntent.getBroadcast(myContext, requestCode, new Intent(myContext, MyBroadcastReceiver.class), PendingIntent.FLAG_MUTABLE | PendingIntent.FLAG_UPDATE_CURRENT); share = Intent.createChooser(share, null, pi.getIntentSender());
在 MyBroadcastReceiver
中接收回调,并在 Intent.EXTRA_CHOSEN_COMPONENT
中查看
Kotlin
override fun onReceive(context: Context, intent: Intent) { ... val clickedComponent : ComponentName = intent.getParcelableExtra(EXTRA_CHOSEN_COMPONENT); }
Java
@Override public void onReceive(Context context, Intent intent) { ... ComponentName clickedComponent = intent.getParcelableExtra(EXTRA_CHOSEN_COMPONENT); }
向共享表单添加自定义操作
在 Android 14(API 级别 34)及更高版本上,应用程序可以向 Android 共享表单添加自定义操作。使用 ChooserAction.Builder
创建一个 ChooserAction
。您可以指定一个 PendingIntent
作为单击图标时调用的操作。创建一个包含所有自定义操作的数组,并将其指定为共享 Intent
的 EXTRA_CHOOSER_CUSTOM_ACTIONS
。
Kotlin
val sendIntent = Intent(Intent.ACTION_SEND) .setType("text/plain") .putExtra(Intent.EXTRA_TEXT, text) val shareIntent = Intent.createChooser(sendIntent, null) val customActions = arrayOf( ChooserAction.Builder( Icon.createWithResource(context, R.drawable.ic_custom_action), "Custom", PendingIntent.getBroadcast( context, 1, Intent(Intent.ACTION_VIEW), PendingIntent.FLAG_IMMUTABLE or PendingIntent.FLAG_CANCEL_CURRENT ) ).build() ) shareIntent.putExtra(Intent.EXTRA_CHOOSER_CUSTOM_ACTIONS, customActions) context.startActivity(shareIntent)
Java
Intent sendIntent = new Intent(Intent.ACTION_SEND) .setType("text.plain") .putExtra(Intent.EXTRA_TEXT, text); Intent shareIntent = Intent.createChooser(sendIntent, null); ChooserAction[] actions = new ChooserAction[]{ new ChooserAction.Builder( Icon.createWithResource(context, R.drawable.ic_custom_action), "Custom", PendingIntent.getBroadcast( context, 1, new Intent(Intent.ACTION_VIEW), PendingIntent.FLAG_IMMUTABLE | PendingIntent.FLAG_CANCEL_CURRENT ) ).build() }; shareIntent.putExtra(Intent.EXTRA_CHOOSER_CUSTOM_ACTIONS, actions); context.startActivity(shareIntent);
使用 Android Intent 解析器
当将数据作为定义明确的任务流的一部分发送到另一个应用程序时,最好使用 Android Intent 解析器。
要使用 Android Intent 解析器,请创建一个 Intent 并添加额外信息,就像调用 Android 共享表单一样。但是,*不要*调用 Intent.createChooser()
。
如果有多个已安装的应用程序的过滤器与 ACTION_SEND
和 MIME 类型匹配,则系统会显示一个称为*Intent 解析器*的歧义对话框,允许用户选择要共享到的目标。如果只有一个应用程序匹配,则它将运行。
以下是如何使用 Android Intent 解析器发送文本的示例
Kotlin
val sendIntent: Intent = Intent().apply { action = Intent.ACTION_SEND putExtra(Intent.EXTRA_TEXT, "This is my text to send.") type = "text/plain" } startActivity(sendIntent)
Java
Intent sendIntent = new Intent(); sendIntent.setAction(Intent.ACTION_SEND); sendIntent.putExtra(Intent.EXTRA_TEXT, "This is my text to send."); sendIntent.setType("text/plain"); startActivity(sendIntent);
了解更多
有关发送数据的更多信息,请参阅 Intent 和 Intent 过滤器。