如何在 Android 中取消将我的短信应用设置为默认应用

How to unset my sms app as default app in Android

我想在 android 中取消设置我的短信应用程序是否为默认应用程序。我正在学习本教程:

http://android-developers.blogspot.com/2013/10/getting-your-sms-apps-ready-for-kitkat.html

我可以通过以下代码将我的短信应用设置为默认短信应用:

Intent intent = new Intent(context, Sms.Intents.ACTION_CHANGE_DEFAULT);
intent.putExtra(Sms.Intents.EXTRA_PACKAGE_NAME, context.getPackageName());
startActivity(intent);

但我想取消将我的短信应用设置为默认应用。我该怎么做?

这里需要注意一点:我已经安装了消息传递经典应用程序。从那个应用程序,我可以取消我的短信应用程序的默认设置。

在你费心阅读这个答案之前,你可能会看看(我希望在做这个之前我已经想到了) .如果您想让用户选择要设置的应用程序,而不是让系统恢复到它决定的任何应用程序,这个答案仍然很方便。


要取消select您的应用程序作为默认短信应用程序,您可以让用户选择另一个符合条件的应用程序代替您的应用程序作为默认应用程序,然后触发 ACTION_CHANGE_DEFAULT Intent 用那个包名。

为了简化这个过程,我写了selectDefaultSmsPackage()方法,它会找到所有接受"android.provider.Telephony.SMS_DELIVER"广播的应用程序(不包括当前包),并提示用户选择一个列表.这是一个相当幼稚的过滤标准,但最糟糕的情况是 selected 应用程序不会成功设置为默认应用程序。

从列表中 select 下载所需的应用程序后,通常的 yes/no 验证对话框将出现。 Activity 收到结果后,将 selected 应用程序的程序包名称与当前设置为默认值的程序包名称进行比较,以确定是否成功。由于一些用户反映在这种情况下结果代码不可靠,因此检查当前默认值是保证正确成功结果的唯一方法。

我们将使用自定义 Dialog 来列出符合条件的应用程序的显示名称和图标。 Activity 创建 AppsDialog 必须实现其 OnAppSelectedListener 接口。

public class MainActivity extends Activity
    implements AppsDialog.OnAppSelectedListener {
    ...

    private static final int DEF_SMS_REQ = 0;
    private AppInfo selectedApp;

    private void selectDefaultSmsPackage() {
        final List<ResolveInfo> receivers = getPackageManager().
            queryBroadcastReceivers(new Intent(Sms.Intents.SMS_DELIVER_ACTION), 0);

        final ArrayList<AppInfo> apps = new ArrayList<>();
        for (ResolveInfo info : receivers) {
            final String packageName = info.activityInfo.packageName;

            if (!packageName.equals(getPackageName())) {
                final String appName = getPackageManager()
                    .getApplicationLabel(info.activityInfo.applicationInfo)
                    .toString();
                final Drawable icon = getPackageManager()
                    .getApplicationIcon(info.activityInfo.applicationInfo);

                apps.add(new AppInfo(packageName, appName, icon));
            }
        }

        Collections.sort(apps, new Comparator<AppInfo>() {
                @Override
                public int compare(AppInfo app1, AppInfo app2) {
                    return app1.appName.compareTo(app2.appName);
                }
            }
        );

        new AppsDialog(this, apps).show();
    }

    @Override
    public void onAppSelected(AppInfo selectedApp) {
        this.selectedApp = selectedApp;

        Intent intent = new Intent(Sms.Intents.ACTION_CHANGE_DEFAULT);
        intent.putExtra(Sms.Intents.EXTRA_PACKAGE_NAME, selectedApp.packageName);
        startActivityForResult(intent, DEF_SMS_REQ);
    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        switch (requestCode) {
            case DEF_SMS_REQ:
                String currentDefault = Sms.getDefaultSmsPackage(this);
                boolean isDefault = selectedApp.packageName.equals(currentDefault);

                String msg = selectedApp.appName + (isDefault ?
                    " successfully set as default" :
                    " not set as default");

                Toast.makeText(this, msg, Toast.LENGTH_SHORT).show();

                break;
            ...
        }
    }

我们需要下面的POJOclass来保存相关的应用信息。

public class AppInfo {
    String appName;
    String packageName;
    Drawable icon;

    public AppInfo(String packageName, String appName, Drawable icon) {
        this.packageName = packageName;
        this.appName = appName;
        this.icon = icon;
    }

    @Override
    public String toString() {
        return appName;
    }
}

AppsDialog class 创建可用默认值的简单 ListView,并通过接口将 selection 传回 Activity

public class AppsDialog extends Dialog
    implements OnItemClickListener {

    public interface OnAppSelectedListener {
        public void onAppSelected(AppInfo selectedApp);
    }

    private final Context context;
    private final List<AppInfo> apps;

    public AppsDialog(Context context, List<AppInfo> apps) {
        super(context);

        if (!(context instanceof OnAppSelectedListener)) {
            throw new IllegalArgumentException(
                "Activity must implement OnAppSelectedListener interface");
        }

        this.context = context;
        this.apps = apps;
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        setTitle("Select default SMS app");

        final ListView listView = new ListView(context);
        listView.setAdapter(new AppsAdapter(context, apps));
        listView.setOnItemClickListener(this);
        setContentView(listView);
    }

    @Override
    public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
        ((OnAppSelectedListener) context).onAppSelected(apps.get(position));
        dismiss();
    }

    private class AppsAdapter extends ArrayAdapter<AppInfo> {
        public AppsAdapter(Context context, List<AppInfo> list) {
            super(context, R.layout.list_item, R.id.text, list);
        }

        public View getView(int position, View convertView, ViewGroup parent) {
            final AppInfo item = getItem(position);

            View v = super.getView(position, convertView, parent); 
            ((ImageView) v.findViewById(R.id.icon)).setImageDrawable(item.icon);

            return v;
        }
    }
}

ArrayAdapter 使用以下项目布局,list_item

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:gravity="center_vertical"
    android:paddingTop="1dp"
    android:paddingBottom="1dp"
    android:paddingStart="8dp"
    android:paddingEnd="8dp">

    <ImageView android:id="@+id/icon"
        android:layout_width="36dp"
        android:layout_height="36dp" />

    <TextView android:id="@+id/text"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:gravity="center_vertical"
        android:textAppearance="?android:attr/textAppearanceListItemSmall"
        android:paddingStart="?android:attr/listPreferredItemPaddingStart"
        android:paddingEnd="?android:attr/listPreferredItemPaddingEnd"
        android:minHeight="?android:attr/listPreferredItemHeightSmall"
        android:ellipsize="marquee" />

</LinearLayout>