Android:如何以编程方式选择默认启动器

9

我想弹出一个对话框,让用户选择一个启动器,并带有“设置为默认值”选项。我尝试了

Intent home = new Intent(Intent.ACTION_DEFAULT);
home.addCategory(Intent.CATEGORY_LAUNCHER);
Intent chooser = Intent.createChooser(home, "Launcher");
context.startActivity(chooser);

但是这个对话框没有设置默认选项。如果已经设置了默认的启动器,下面的代码将不会弹出对话框。
Intent startMain = new Intent(Intent.ACTION_MAIN);
startMain.addCategory(Intent.CATEGORY_HOME);
startMain.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(startMain);

如何实现这一功能?
3个回答

23

请尝试使用以下内容:

Intent startMain = new Intent(Intent.ACTION_MAIN);
startMain.addCategory(Intent.CATEGORY_HOME);
startMain.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(startMain);

如果已经设置了默认操作(您的),则可以先调用:

getPackageManager().clearPackagePreferredActivities(getPackageName());

如果默认操作不是你的话,你不能以编程方式清除它,你可以做的是检查其他应用程序是否被设置为默认,并显示消息。

private boolean isMyLauncherDefault() {
    PackageManager localPackageManager = getPackageManager();
    Intent intent = new Intent("android.intent.action.MAIN");
    intent.addCategory("android.intent.category.HOME");
    String str = localPackageManager.resolveActivity(intent,
          PackageManager.MATCH_DEFAULT_ONLY).activityInfo.packageName;
    return str.equals(getPackageName());
}

如果其他应用程序被设置为默认应用程序,您可以通过创建一个虚拟 主屏幕并安装它(这将强制系统清除默认应用程序),然后再卸载它来解决问题...

Manifest.xml

<activity
        android:name="FakeHome"  android:enabled="false">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />
            <category android:name="android.intent.category.HOME"/>
            <category android:name="android.intent.category.DEFAULT" />
        </intent-filter>
    </activity>

FakeHome.java

public class FakeHome extends Activity {

}

某处

if (!isMyLauncherDefault()) {           
    PackageManager p = getPackageManager();
    ComponentName cN = new ComponentName(getApplicationContext(), FakeHome.class);
    p.setComponentEnabledSetting(cN, PackageManager.COMPONENT_ENABLED_STATE_ENABLED, PackageManager.DONT_KILL_APP);

    Intent selector = new Intent(Intent.ACTION_MAIN);
    selector.addCategory(Intent.CATEGORY_HOME);            
    startActivity(selector);

    p.setComponentEnabledSetting(cN, PackageManager.COMPONENT_ENABLED_STATE_DISABLED, PackageManager.DONT_KILL_APP);            
}

2
如果已经设置了默认启动器,此代码将不会弹出对话框。 - Vishnu Mohan G
getPreferredActivities()filters 参数是一个输出参数。在调用之前向列表中添加过滤项没有任何效果。 - Christian d'Heureuse
@Nermeen,这个isMyLauncherDefault()函数在哪里被调用?是在FakeHome Activity还是其他的Activity中?而且要把上述意图传递到哪里呢? - Rajesh
这个很好用。有没有办法使用ROLE_HOME让它工作? - android developer

5

从API 29开始,使用RoleManager官方支持此功能。

以下是可以从任何活动中调用的非常简单的Kotlin示例方法:

fun showLauncherSelector(activity: AppCompatActivity, requestCode : Int) {
    val roleManager = activity.getSystemService(Context.ROLE_SERVICE) as RoleManager
    if(roleManager.isRoleAvailable(RoleManager.ROLE_HOME)){
        val intent = roleManager.createRequestRoleIntent(RoleManager.ROLE_HOME)
        activity.startActivityForResult(intent, requestCode)
    }
}

然后您可以在调用者活动中检查错误:

override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
    if (requestCode == MY_REQUEST_CODE) {
        if (resultCode != Activity.RESULT_OK) {
            // Something went wrong
        }
    }
}

2
我真的不明白为什么他们要创建 Kotlin,只是为了使类似的答案变得无用。 - W.M.
遗憾的是,这仅在您拥有能够处理意图的启用活动时才有效。如果没有,什么也不会发生。 - android developer

0

试试这个:

    Intent intent = new Intent("android.intent.action.MAIN");
    intent.addCategory("android.intent.category.HOME");
    intent.addFlags(Intent.FLAG_RECEIVER_FOREGROUND);
    getContext().startActivity(intent);

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