如何过滤特定应用程序以获取ACTION_SEND意图(并为每个应用程序设置不同的文本)

199
如何在使用ACTION_SEND意图时过滤特定应用程序?这个问题以各种方式被问到,但我无法根据给出的答案收集解决方案。希望有人能帮忙。 我想提供在应用内共享的功能。遵循Android Dev Alexander Lucas的建议,我希望使用意图来完成,而不是使用Facebook/Twitter API。

Sharing using ACTION_SEND intent

使用ACTION_SEND意图进行分享很棒,但问题是(1)我不想要所有的分享选项,我宁愿将其限制在FB、Twitter和电子邮件中;(2)我不想将相同的内容分享到每个分享应用程序上。例如,在我的Twitter分享中,我将包括一些提及和标签限制在140个字符以内,而Facebook分享将包括一个链接和一个特色图像。
有可能限制ACTION_SEND(共享)意图的选项吗?我看到了一些关于使用PackageManager和queryIntentActivities的东西,但还没有能够弄清楚PackageManager和ACTION_SEND意图之间的联系。
或者,与其过滤共享应用程序,我的问题也可以通过使用ACTION_SEND意图直接转到Facebook或Twitter来解决,而不是弹出对话框。如果是这样的话,那么我可以创建自己的对话框,当他们点击“Facebook”时,创建一个特定于Facebook的意图,并将他们直接发送到Facebook。Twitter也是一样。
或者不可能吗?Facebook和Twitter API是唯一的途径吗?

1
这篇博客文章似乎是完美的答案:http://hkdevtips.blogspot.com/2013/02/customize-your-actionchooser-intent.html - olshevski
2
嘿,朋友...当我点击发送按钮时,会打开共享对话框,共享对话框列表是“Gmail,电子邮件,Zapiya,Hookup”等,但不显示Facebook,WhatsApp,Facebook Messenger,Hike Hangouts等...我该如何显示它们? - GB_Bhayani ツ
如何在Android 6.0上当意图操作只有一个项目/选项时不显示选择器?这个问题在低于Android 6.0的版本上不会出现。 - zys
12个回答

330

我的规格要求用户能够选择电子邮件、Twitter、Facebook或短信,并为每个选项提供自定义文本。以下是我实现这一功能的方法:

public void onShareClick(View v) {
    Resources resources = getResources();

    Intent emailIntent = new Intent();
    emailIntent.setAction(Intent.ACTION_SEND);
    // Native email client doesn't currently support HTML, but it doesn't hurt to try in case they fix it
    emailIntent.putExtra(Intent.EXTRA_TEXT, Html.fromHtml(resources.getString(R.string.share_email_native)));
    emailIntent.putExtra(Intent.EXTRA_SUBJECT, resources.getString(R.string.share_email_subject));
    emailIntent.setType("message/rfc822");
    
    PackageManager pm = getPackageManager();
    Intent sendIntent = new Intent(Intent.ACTION_SEND);     
    sendIntent.setType("text/plain");


    Intent openInChooser = Intent.createChooser(emailIntent, resources.getString(R.string.share_chooser_text));

    List<ResolveInfo> resInfo = pm.queryIntentActivities(sendIntent, 0);
    List<LabeledIntent> intentList = new ArrayList<LabeledIntent>();        
    for (int i = 0; i < resInfo.size(); i++) {
        // Extract the label, append it, and repackage it in a LabeledIntent
        ResolveInfo ri = resInfo.get(i);
        String packageName = ri.activityInfo.packageName;
        if(packageName.contains("android.email")) {
            emailIntent.setPackage(packageName);
        } else if(packageName.contains("twitter") || packageName.contains("facebook") || packageName.contains("mms") || packageName.contains("android.gm")) {
            Intent intent = new Intent();
            intent.setComponent(new ComponentName(packageName, ri.activityInfo.name));
            intent.setAction(Intent.ACTION_SEND);
            intent.setType("text/plain");
            if(packageName.contains("twitter")) {
                intent.putExtra(Intent.EXTRA_TEXT, resources.getString(R.string.share_twitter));
            } else if(packageName.contains("facebook")) {
                // Warning: Facebook IGNORES our text. They say "These fields are intended for users to express themselves. Pre-filling these fields erodes the authenticity of the user voice."
                // One workaround is to use the Facebook SDK to post, but that doesn't allow the user to choose how they want to share. We can also make a custom landing page, and the link
                // will show the <meta content ="..."> text from that page with our link in Facebook.
                intent.putExtra(Intent.EXTRA_TEXT, resources.getString(R.string.share_facebook));
            } else if(packageName.contains("mms")) {
                intent.putExtra(Intent.EXTRA_TEXT, resources.getString(R.string.share_sms));
            } else if(packageName.contains("android.gm")) { // If Gmail shows up twice, try removing this else-if clause and the reference to "android.gm" above
                intent.putExtra(Intent.EXTRA_TEXT, Html.fromHtml(resources.getString(R.string.share_email_gmail)));
                intent.putExtra(Intent.EXTRA_SUBJECT, resources.getString(R.string.share_email_subject));               
                intent.setType("message/rfc822");
            }
            
            intentList.add(new LabeledIntent(intent, packageName, ri.loadLabel(pm), ri.icon));
        }
    }

    // convert intentList to array
    LabeledIntent[] extraIntents = intentList.toArray( new LabeledIntent[ intentList.size() ]);

    openInChooser.putExtra(Intent.EXTRA_INITIAL_INTENTS, extraIntents);
    startActivity(openInChooser);       
}

我在不同的地方找到了如何做这件事的一些细节,但是我没有在其他任何地方看到全部内容。
请注意,此方法还隐藏了我不想要的所有愚蠢选项,例如通过wifi和蓝牙共享。
编辑:在评论中,有人要求我解释这段代码的作用。基本上,它为本机电子邮件客户端创建了一个ACTION_SEND意图,然后将其他意图附加到选择器上。使原始意图仅限于电子邮件可以摆脱所有额外的垃圾,例如wifi和蓝牙,然后我从纯文本类型的通用ACTION_SEND中获取我想要的其他意图,并在显示选择器之前将它们附加上去。
当我获取附加意图时,我为每个设置自定义文本。
编辑2:已经过了一段时间,情况发生了一些变化。如果您在选项列表中看到两次gmail,请尝试删除“android.gm”的特殊处理,正如@h_k在下面的评论中建议的那样。
由于这个答案几乎是我所有stackoverflow声望点数的来源,所以我必须至少尝试保持其最新状态。

1
我正在使用这段代码,但不知何故Evernote悄悄地出现在列表中。当我检查包名称时,它是com.evernote,所以我不确定为什么会发生这种情况。 - James Harpe
1
@user2249287 我建议您逐步执行代码,直到找到被跳过的消息应用程序,然后查看包名称以确定需要添加到白名单的应用程序字符串。 - dacoinminster
2
要仅过滤用户拥有的电子邮件应用程序,您可以使用此问题的第二个答案:https://dev59.com/UGoy5IYBdhLWcg3wN7YX。它不需要使用消息/rfc822数据类型,而其他应用程序(例如EverNote)则会使用该数据类型。 - mpellegr
2
@dacoinminster,你的代码非常棒,让我可以为Twitter和Whatsapp等应用定义不同的文本。为了删除重复的Gmail,我只是将“android.gm”从方程式中去掉了。我仍然可以在选择列表中获得Gmail和内置邮件应用程序,并且主题和文本仍然完好无损。 - h_k
1
@Shailesh 对不起,我以前从未尝试过使用这种方法分享图片。你可能需要创建一个新的问题来询问这个。 - dacoinminster
显示剩余21条评论

28

如果您需要定制化选项,那么就不应该依赖于Android提供的默认对话框。

相反,您需要自己编写对话框。您需要查询PackageManager以确定哪些包处理所需的操作,然后根据回复进行过滤和自定义文本。

具体来说,请查看PackageManager类的方法queryIntentActivities。您需要构建将启动默认对话框(ACTION_SEND意图)的意图,并将其传递给此方法,然后您将收到一个包含有关可以处理该意图的活动的信息的对象列表。使用它,您可以选择自己需要的活动。

一旦您构建了要显示的软件包列表,就需要构建自己的列表对话框(最好是对话框主题的活动),用于显示该列表。

需要注意的一点是很难让自定义对话框看起来像默认的对话框。问题在于该对话框中使用的主题是内部主题,无法由您的应用程序使用。您可以尝试使其看起来与原生对话框尽可能相似,或者选择完全自定义外观(许多应用程序都这样做,例如图库应用等)。


1
将此答案标记为正确,因为它最接近原始问题的答案,尽管我最终采取了不同的方法(请参见我的答案)。谢谢。 - Kyle Clegg

24

在stackoverflow查找到了一个适用于我的解决方案,可以在这里查看(请看第一个回答的第三个评论)。该代码寻找有效的Twitter客户端并使用其发布推文。注意:它不会给您Intent并允许您选择不同的Twitter客户端。

使用Twitter分享:

Intent shareIntent = findTwitterClient(); 
shareIntent.putExtra(Intent.EXTRA_TEXT, "test");
startActivity(Intent.createChooser(shareIntent, "Share"));

调用这个方法:

public Intent findTwitterClient() {
    final String[] twitterApps = {
            // package // name - nb installs (thousands)
            "com.twitter.android", // official - 10 000
            "com.twidroid", // twidroid - 5 000
            "com.handmark.tweetcaster", // Tweecaster - 5 000
            "com.thedeck.android" }; // TweetDeck - 5 000 };
    Intent tweetIntent = new Intent();
    tweetIntent.setType("text/plain");
    final PackageManager packageManager = getPackageManager();
    List<ResolveInfo> list = packageManager.queryIntentActivities(
            tweetIntent, PackageManager.MATCH_DEFAULT_ONLY);

    for (int i = 0; i < twitterApps.length; i++) {
        for (ResolveInfo resolveInfo : list) {
            String p = resolveInfo.activityInfo.packageName;
            if (p != null && p.startsWith(twitterApps[i])) {
                tweetIntent.setPackage(p);
                return tweetIntent;
            }
        }
    }

    return null;
}

使用 "com.facebook.katana",Facebook将是类似的,尽管你仍然无法设置消息文本(已于2011年7月停止支持)。

代码来源:在Android上打开Twitter客户端的意图


4
我不喜欢这个答案,因为它依赖于知道所有 Twitter 应用程序的包名称。另一种方法请参见https://dev59.com/XGw15IYBdhLWcg3wGn9c#9229654。 - Ed Burnette
我同意你的观点,尽管你链接的答案也存在类似的问题。我从不喜欢依赖字符串比较,特别是当我无法控制或保证字符串不会改变时。 - Kyle Clegg

22

试一下只分享三个应用程序的方法- Facebook、Twitter、KakaoStory。

public void onShareClick(View v){
    List<Intent> targetShareIntents=new ArrayList<Intent>();
    Intent shareIntent=new Intent();
    shareIntent.setAction(Intent.ACTION_SEND);
    shareIntent.setType("text/plain");
    List<ResolveInfo> resInfos=getPackageManager().queryIntentActivities(shareIntent, 0);
    if(!resInfos.isEmpty()){
        System.out.println("Have package");
        for(ResolveInfo resInfo : resInfos){
            String packageName=resInfo.activityInfo.packageName;
            Log.i("Package Name", packageName);
            if(packageName.contains("com.twitter.android") || packageName.contains("com.facebook.katana") || packageName.contains("com.kakao.story")){
                Intent intent=new Intent();
                intent.setComponent(new ComponentName(packageName, resInfo.activityInfo.name));
                intent.setAction(Intent.ACTION_SEND);
                intent.setType("text/plain");
                intent.putExtra(Intent.EXTRA_TEXT, "Text");
                intent.putExtra(Intent.EXTRA_SUBJECT, "Subject");
                intent.setPackage(packageName);
                targetShareIntents.add(intent);
            }
        }
        if(!targetShareIntents.isEmpty()){
            System.out.println("Have Intent");
            Intent chooserIntent=Intent.createChooser(targetShareIntents.remove(0), "Choose app to share");
            chooserIntent.putExtra(Intent.EXTRA_INITIAL_INTENTS, targetShareIntents.toArray(new Parcelable[]{}));
            startActivity(chooserIntent);
        }else{
            System.out.println("Do not Have Intent");
            showDialaog(this);
        }
    }
}

如果您想与特定应用程序共享,则此代码可以完美运行。 - Orcun Sevsay

15
感谢@dacoinminster的答案。我对他的答案进行了一些修改,包括流行应用程序的包名称和这些应用程序的排序。
List<Intent> targetShareIntents = new ArrayList<Intent>();
Intent shareIntent = new Intent();
shareIntent.setAction(Intent.ACTION_SEND);
shareIntent.setType("text/plain");
PackageManager pm = getActivity().getPackageManager();
List<ResolveInfo> resInfos = pm.queryIntentActivities(shareIntent, 0);
if (!resInfos.isEmpty()) {
    System.out.println("Have package");
    for (ResolveInfo resInfo : resInfos) {
        String packageName = resInfo.activityInfo.packageName;
        Log.i("Package Name", packageName);

        if (packageName.contains("com.twitter.android") || packageName.contains("com.facebook.katana")
                || packageName.contains("com.whatsapp") || packageName.contains("com.google.android.apps.plus")
                || packageName.contains("com.google.android.talk") || packageName.contains("com.slack")
                || packageName.contains("com.google.android.gm") || packageName.contains("com.facebook.orca")
                || packageName.contains("com.yahoo.mobile") || packageName.contains("com.skype.raider")
                || packageName.contains("com.android.mms")|| packageName.contains("com.linkedin.android")
                || packageName.contains("com.google.android.apps.messaging")) {
            Intent intent = new Intent();

            intent.setComponent(new ComponentName(packageName, resInfo.activityInfo.name));
            intent.putExtra("AppName", resInfo.loadLabel(pm).toString());
            intent.setAction(Intent.ACTION_SEND);
            intent.setType("text/plain");
            intent.putExtra(Intent.EXTRA_TEXT, "https://website.com/");
            intent.putExtra(Intent.EXTRA_SUBJECT, getString(R.string.share_text));
            intent.setPackage(packageName);
            targetShareIntents.add(intent);
        }
    }
    if (!targetShareIntents.isEmpty()) {
        Collections.sort(targetShareIntents, new Comparator<Intent>() {
            @Override
            public int compare(Intent o1, Intent o2) {
                return o1.getStringExtra("AppName").compareTo(o2.getStringExtra("AppName"));
            }
        });
        Intent chooserIntent = Intent.createChooser(targetShareIntents.remove(0), "Select app to share");
        chooserIntent.putExtra(Intent.EXTRA_INITIAL_INTENTS, targetShareIntents.toArray(new Parcelable[]{}));
        startActivity(chooserIntent);
    } else {
        Toast.makeText(getActivity(), "No app to share.", Toast.LENGTH_LONG).show();
    }
}

1
谢谢@Oguz,这对我有用,第一个答案对我不起作用。 - Sanjayrajsinh

11

你可以尝试下面的代码,它完美地工作。

这里我们分享一些特定的应用程序,它们是Facebook、Messenger、Twitter、Google Plus和Gmail。

public void shareIntentSpecificApps() {
        List<Intent> intentShareList = new ArrayList<Intent>();
        Intent shareIntent = new Intent();
        shareIntent.setAction(Intent.ACTION_SEND);
        shareIntent.setType("text/plain");
        List<ResolveInfo> resolveInfoList = getPackageManager().queryIntentActivities(shareIntent, 0);

        for (ResolveInfo resInfo : resolveInfoList) {
            String packageName = resInfo.activityInfo.packageName;
            String name = resInfo.activityInfo.name;
            Log.d(TAG, "Package Name : " + packageName);
            Log.d(TAG, "Name : " + name);

            if (packageName.contains("com.facebook") ||
                    packageName.contains("com.twitter.android") ||
                    packageName.contains("com.google.android.apps.plus") ||
                    packageName.contains("com.google.android.gm")) {

                if (name.contains("com.twitter.android.DMActivity")) {
                    continue;
                }

                Intent intent = new Intent();
                intent.setComponent(new ComponentName(packageName, name));
                intent.setAction(Intent.ACTION_SEND);
                intent.setType("text/plain");
                intent.putExtra(Intent.EXTRA_SUBJECT, "Your Subject");
                intent.putExtra(Intent.EXTRA_TEXT, "Your Content");
                intentShareList.add(intent);
            }
        }

        if (intentShareList.isEmpty()) {
            Toast.makeText(MainActivity.this, "No apps to share !", Toast.LENGTH_SHORT).show();
        } else {
            Intent chooserIntent = Intent.createChooser(intentShareList.remove(0), "Share via");
            chooserIntent.putExtra(Intent.EXTRA_INITIAL_INTENTS, intentShareList.toArray(new Parcelable[]{}));
            startActivity(chooserIntent);
        }
    }

“if(name.contains("com.twitter.android.DMActivity")) { continue ; }” 的原因是什么? - isJulian00

8
这个方案展示了一个类似于选择器的ListView对话框,其中包含应用程序列表: screenshot 你需要完成以下任务: 1.获取相关应用程序包列表 2.根据包名调用相应的意图
该适配器类:
import java.util.List;

import android.content.Context;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageManager;
import android.content.pm.PackageManager.NameNotFoundException;
import android.graphics.drawable.Drawable;
import android.util.TypedValue;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ArrayAdapter;
import android.widget.TextView;

public class ChooserArrayAdapter extends ArrayAdapter<String> {
    PackageManager mPm;
    int mTextViewResourceId;
    List<String> mPackages;

    public ChooserArrayAdapter(Context context, int resource, int textViewResourceId, List<String> packages) {
        super(context, resource, textViewResourceId, packages);
        mPm = context.getPackageManager();
        mTextViewResourceId = textViewResourceId;
        mPackages = packages;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        String pkg = mPackages.get(position);
        View view = super.getView(position, convertView, parent);

        try {
            ApplicationInfo ai = mPm.getApplicationInfo(pkg, 0);

            CharSequence appName = mPm.getApplicationLabel(ai);
            Drawable appIcon = mPm.getApplicationIcon(pkg);

            TextView textView = (TextView) view.findViewById(mTextViewResourceId);
            textView.setText(appName);
            textView.setCompoundDrawablesWithIntrinsicBounds(appIcon, null, null, null);
            textView.setCompoundDrawablePadding((int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 12, getContext().getResources().getDisplayMetrics()));
        } catch (NameNotFoundException e) {
            e.printStackTrace();
        }

        return view;
    }

}

及其使用:

    void doXxxButton() {
        final List<String> packages = ...;
        if (packages.size() > 1) {
            ArrayAdapter<String> adapter = new ChooserArrayAdapter(MyActivity.this, android.R.layout.select_dialog_item, android.R.id.text1, packages);

            new AlertDialog.Builder(MyActivity.this)
            .setTitle(R.string.app_list_title)
            .setAdapter(adapter, new DialogInterface.OnClickListener() {
                public void onClick(DialogInterface dialog, int item ) {
                    invokeApplication(packages.get(item));
                }
            })
            .show();
        } else if (packages.size() == 1) {
            invokeApplication(packages.get(0));
        }
    }

    void invokeApplication(String packageName) {
        // given a package name, create an intent and fill it with data
        ...
        startActivityForResult(intent, rq);
    }

4
最清晰的方法是复制以下类:ShareActionProvider、ActivityChooserView、ActivityChooserModel,并添加在ActivityChooserModel中筛选意图的能力,以及在ShareActionProvider中相应的支持方法。我已经创建了必要的类,您可以将它们复制到您的项目中(https://gist.github.com/saulpower/10557956)。这不仅可以添加筛选您想共享的应用程序的能力(如果您知道包名称),还可以关闭历史记录。
private final String[] INTENT_FILTER = new String[] {
    "com.twitter.android",
    "com.facebook.katana"
};

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    MenuInflater inflater = getMenuInflater();
    inflater.inflate(R.menu.journal_entry_menu, menu);

    // Set up ShareActionProvider's default share intent
    MenuItem shareItem = menu.findItem(R.id.action_share);

    if (shareItem instanceof SupportMenuItem) {
        mShareActionProvider = new ShareActionProvider(this);
        mShareActionProvider.setShareIntent(ShareUtils.share(mJournalEntry));
        mShareActionProvider.setIntentFilter(Arrays.asList(INTENT_FILTER));
        mShareActionProvider.setShowHistory(false);
        ((SupportMenuItem) shareItem).setSupportActionProvider(mShareActionProvider);
    }

    return super.onCreateOptionsMenu(menu);
}

如何添加包含剩余应用程序的Google+和其他选项 - Sunishtha Singh

3
Intent emailIntent = new Intent(Intent.ACTION_SENDTO, 
    Uri.fromParts("mailto", "android@gmail.com", null));
emailIntent.putExtra(Intent.EXTRA_SUBJECT, text);
startActivity(Intent.createChooser(emailIntent, "Send email..."));

3

我改进了@dacoinminster的答案,并提供了一个示例,以分享您的应用程序:

// Intents with SEND action
PackageManager packageManager = context.getPackageManager();
Intent sendIntent = new Intent(Intent.ACTION_SEND);
sendIntent.setType("text/plain");
List<ResolveInfo> resolveInfoList = packageManager.queryIntentActivities(sendIntent, 0);

List<LabeledIntent> intentList = new ArrayList<LabeledIntent>();
Resources resources = context.getResources();

for (int j = 0; j < resolveInfoList.size(); j++) {
    ResolveInfo resolveInfo = resolveInfoList.get(j);
    String packageName = resolveInfo.activityInfo.packageName;
    Intent intent = new Intent();
    intent.setAction(Intent.ACTION_SEND);
    intent.setComponent(new ComponentName(packageName,
    resolveInfo.activityInfo.name));
    intent.setType("text/plain");

    if (packageName.contains("twitter")) {
        intent.putExtra(Intent.EXTRA_TEXT, resources.getString(R.string.twitter) + "https://play.google.com/store/apps/details?id=" + context.getPackageName());
    } else {
        // skip android mail and gmail to avoid adding to the list twice
        if (packageName.contains("android.email") || packageName.contains("android.gm")) {
            continue;
        }
        intent.putExtra(Intent.EXTRA_TEXT, resources.getString(R.string.largeTextForFacebookWhatsapp) + "https://play.google.com/store/apps/details?id=" + context.getPackageName());
    }

    intentList.add(new LabeledIntent(intent, packageName, resolveInfo.loadLabel(packageManager), resolveInfo.icon));
}

Intent emailIntent = new Intent(Intent.ACTION_SENDTO, Uri.parse("mailto:"));
emailIntent.putExtra(Intent.EXTRA_SUBJECT, resources.getString(R.string.subjectForMailApps));
emailIntent.putExtra(Intent.EXTRA_TEXT, resources.getString(R.string.largeTextForMailApps) + "https://play.google.com/store/apps/details?id=" + context.getPackageName());

context.startActivity(Intent.createChooser(emailIntent, resources.getString(R.string.compartirEn)).putExtra(Intent.EXTRA_INITIAL_INTENTS, intentList.toArray(new LabeledIntent[intentList.size()])));

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接