场景
当我们的应用需求里面,需要发送邮件的时候。我们通常并不会自己去实现一个邮件App,而是通过隐式Intent的方式转发到其他App,我们只是需要添加一些相应的数据而已。Android的官方文档告诉我们,发送邮件我们可以通过如下三个Action:
ACTION_SENDTO(适用于不带附件)
ACTION_SEND(适用于带一个附件)
ACTION_SEND_MULTIPLE`(适用于带多个附件)
通过ACTION_SENDTO和"mailto:"的如下组合,系统会为我们筛选出所有邮箱App。
public void composeEmail(String[] addresses, String subject) {
Intent intent = new Intent(Intent.ACTION_SENDTO);
intent.setData(Uri.parse("mailto:")); // only email apps should handle this
intent.putExtra(Intent.EXTRA_EMAIL, addresses);
intent.putExtra(Intent.EXTRA_SUBJECT, subject);
if (intent.resolveActivity(getPackageManager()) != null) {
startActivity(Intent.createChooser(intent, "Send mail..."));
}
}
这是我们所期望的效果,然而,当我们希望添加附件。比如,以下代码,系统就会弹出很多包括并非邮箱App的第三方软件供我们选择:
public void composeEmail(String[] addresses, String subject, Uri attachment) {
Intent intent = new Intent(Intent.ACTION_SEND);
intent.setType("*/*");
intent.putExtra(Intent.EXTRA_EMAIL, addresses);
intent.putExtra(Intent.EXTRA_SUBJECT, subject);
intent.putExtra(Intent.EXTRA_STREAM, attachment);
if (intent.resolveActivity(getPackageManager()) != null) {
startActivity(Intent.createChooser(intent, "Send mail..."));
}
}
通过自定义CreateChooser拣选出我们希望的App
上面的这种结果肯定不是我们所希望的,我们希望我们的应用选择弹窗干净,不会出现qq、微信、文件管理器等所有能够接受ACTION_SEND的额外应用。解决方案,就是自定义我们自己的CreateChooser。
方法就是如下代码,代码中添加了详细的解释。
public void sendLog(String email, String subject, Uri contentUri) {
// 第一步,构建一个添加了所有邮件信息的Intent,
// 理论上,调用该Intent直接去创建**CreateChooser**的
// 话会弹出非常多的备选应用,包括我们不需要的。
Intent emailIntent = new Intent(Intent.ACTION_SEND);
emailIntent.setType("*/*");
emailIntent.putExtra(Intent.EXTRA_EMAIL, new String[]{email});
emailIntent.putExtra(Intent.EXTRA_SUBJECT, subject);
emailIntent.putExtra(Intent.EXTRA_STREAM, contentUri);
// 第二步,创建一个ACTION_SENDTO和"mailto:"组合的Intent,
// 并使用该组合去找到支持邮件发送的应用的ResolveInfo的集合。
// ResolveInfo是解析manifest文件中的Intent过滤器所得到的信息。
Intent queryIntent = new Intent(Intent.ACTION_SENDTO, Uri.parse("mailto:"));
List<ResolveInfo> resolveInfos = mContext.getPackageManager().queryIntentActivities(queryIntent,
PackageManager.MATCH_DEFAULT_ONLY
| PackageManager.GET_RESOLVED_FILTER);
// 第三步,构建我们的目标Intent,通过迭代resolveInfos,
// 创建所有的目标Intent。这些Intent不仅仅包含了上面emailIntent的信息,
// 同时,由于还设置了Component,这也就相当于一个显示的Intent,
// 因此这些Intent都被固定了。
ArrayList<Intent> targetIntents = new ArrayList<>();
for (ResolveInfo info : resolveInfos) {
ActivityInfo ai = info.activityInfo;
Intent intent = new Intent(emailIntent);
intent.setPackage(ai.packageName);
intent.setComponent(new ComponentName(ai.packageName, ai.name));
targetIntents.add(intent);
}
// 第四步,通过上面的目标Intent集合,构建CreateChooser。
// 此处之所以需要targetIntents.remove(0)是因为,
// 创建该CreateChooser的Intent的时候,传入的Intent就会在里面。
// 因此需要从集合中remove掉,否则在下面通过EXTRA_INITIAL_INTENTS的方式传入目标Intent数组时,
// 第一个Intent会多出一个。
Intent chooser = Intent.createChooser(targetIntents.remove(0), subject);
chooser.putExtra(
Intent.EXTRA_INITIAL_INTENTS, targetIntents.toArray(new Parcelable[]{}));
mContext.startActivity(chooser);
}