
用户喜欢图片、视频和其他富有表现力的内容,但在应用中插入和移动这些内容并非易事。为了让应用能够更轻松地接收富媒体内容,Android 12(API 级别 31)引入了一个统一 API,便于应用接受来自任何来源(剪贴板粘贴、键盘输入或拖放操作)的内容。
您可以向界面组件附加接口 OnReceiveContentListener
,并在通过任何机制插入内容时获得回调。此回调会成为您的代码接收所有内容(从纯文本和样式文本到标记、图片、视频、音频文件等)的唯一位置。
为了向后兼容以前的 Android 版本,此 API 也可在 AndroidX 中使用(从 Core 1.7 和 Appcompat 1.4 开始),我们建议您在实现此功能时使用该 API。
概览
使用其他现有 API 时,每个界面机制(例如长按菜单或拖放)都有各自对应的 API。这意味着您必须单独与每个 API 集成,并为每种插入内容的机制添加类似的代码:
OnReceiveContentListener
API 会通过创建一个要实现的单一 API 来整合这些不同的代码路径,这样您就可以专注于应用特定的逻辑,而让平台处理其余的工作:
此外,这种方法还意味着,在向平台添加插入内容的新方法时,您无需对代码做出其他更改即可在应用中启用相关支持。如果您的应用需要针对特定用例实现完全自定义,您仍然可以使用现有的 API,这些 API 将继续以相同的方式工作。
实现
该 API 是一个监听器接口,包含一种方法,即 OnReceiveContentListener
。为了支持较低版本的 Android 平台,我们建议您使用 AndroidX Core 库中的匹配 OnReceiveContentListener
接口。
如需使用该 API,请先指定您的应用可以处理哪些类型的内容,以开始实现该监听器:
public class MyReceiver implements OnReceiveContentListener {
public static final String[] MIME_TYPES = new String[] {"image/*", "video/*"};
// ...
指定您的应用支持的所有内容 MIME 类型后,实现该监听器的其余部分:
public class MyReceiver implements OnReceiveContentListener {
public static final String[] MIME_TYPES = new String[] {"image/*", "video/*"};
@Override
public ContentInfoCompat onReceiveContent(View view, ContentInfoCompat contentInfo) {
Pair<ContentInfoCompat, ContentInfoCompat> split = contentInfo.partition(
item -> item.getUri() != null);
ContentInfo uriContent = split.first;
ContentInfo remaining = split.second;
if (uriContent != null) {
// App-specific logic to handle the URI(s) in uriContent...
}
// Return anything that your app didn't handle. This preserves the default platform
// behavior for text and anything else that you aren't implementing custom handling for.
return remaining;
}
}
如果您的应用已经支持与 intent 共享,您可以重复使用应用特定逻辑来处理内容 URI。将任何剩余数据返回以委派给平台进行处理。
实现监听器后,在应用的相应界面元素上设置该监听器:
public class MyActivity extends Activity {
@Override
public void onCreate(Bundle savedInstanceState) {
// ...
AppCompatEditText myInput = findViewById(R.id.my_input);
ViewCompat.setOnReceiveContentListener(myInput, MyReceiver.MIME_TYPES, new MyReceiver());
}
}
URI 权限
对于传递给 OnReceiveContentListener
的载荷中的任何内容 URI,平台都会自动授予和释放读取权限。
通常,应用应在 Service 或 Activity 中处理内容 URI。对于长时间运行的处理任务,请使用 WorkManager。 实现此功能时,您应使用 Intent.setClipData 传递内容,并设置标记 FLAG_GRANT_READ_URI_PERMISSION,从而将权限扩展到目标 Service 或 Activity。
或者,您也可以在当前上下文中使用后台线程来处理内容。在这种情况下,您必须保持对监听器收到的 payload
对象的引用,以确保平台不会提前撤消权限。
自定义 View
如果您的应用使用自定义 View
子类,您需要注意确保系统不会绕过 OnReceiveContentListener
。
如果 View
类会替换 onCreateInputConnection
方法,请使用 Jetpack API InputConnectionCompat.createWrapper
配置 InputConnection
。
如果 View
类会替换 onTextContextMenuItem
方法,请确保在菜单项为 R.id.paste
或 R.id.pasteAsPlainText
时委托给 super。
与键盘图片 API 相比
您可以将 OnReceiveContentListener
API 视为现有键盘图片 API 的下一个版本。此统一 API 支持键盘图片 API 的功能,以及一些其他功能。设备和功能兼容性因您使用 Jetpack 库还是 Android SDK 中的原生 API 而异。
支持的功能和 API 级别:Jetpack
操作或功能 | 受键盘图片 API 支持 | 受统一 API 支持 |
---|---|---|
通过键盘插入 | 是(API 级别 13 及更高) | 是(API 级别 13 及更高) |
通过从长按菜单粘贴插入 | 否 | 是 |
通过拖放操作插入 | 否 | 是(API 级别 24 及更高) |
支持的功能和 API 级别:原生 API
操作或功能 | 受键盘图片 API 支持 | 受统一 API 支持 |
---|---|---|
通过键盘插入 | 是(API 级别 25 及更高) | 是(Android 12 及更高版本) |
通过从长按菜单粘贴插入 | 否 | |
通过拖放操作插入 | 否 |