Android系统 设置第三方应用为默认Launcher实现和原理分析
  HvTJUzsxOBtS 2023年11月18日 47 0


前言
Android系统中,launcher是用户与系统交互的主要界面,它负责显示桌面、应用列表、小部件等内容。Android系统允许用户安装第三方的launcher应用,以替换系统自带的launcher。

但如何满足客制化需求让第三方的launcher应用成为默认的launcher呢?本文将从源码的角度,分析Android系统是如何处理launcher应用的启动和切换的,以及如何通过修改源码来实现设置第三方应用为默认launcher的功能。

launcher应用的启动和切换
在Android系统中,当用户按下Home键时,系统会发送一个包含Intent.CATEGORY_HOME类别的隐式Intent,该Intent的作用是启动一个能够显示主屏幕的Activity。系统会根据该Intent,在已安装的应用中查找匹配的Activity,并显示一个选择器让用户选择要启动的launcher应用。如果用户选择了某个应用,并勾选了“始终”选项,则该应用会被设置为默认的launcher,并保存在系统设置中。以后每次按下Home键时,系统都会直接启动该应用,而不再显示选择器(部分平台系统不会保存就算你选择了始终也,重启也会弹出选择)。

系统是如何保存和读取默认的launcher应用的呢?答案就在RootWindowContainer类中。该类是窗口管理服务(WindowManagerService)中最顶层的容器类,它负责管理所有显示内容(DisplayContent)和任务栈(TaskStack)。在该类中,有一个方法叫做resolveHomeActivity,它的作用是根据一个包含Intent.CATEGORY_HOME类别的Intent,解析出对应的ActivityInfo对象,并返回给调用者。该方法会首先从系统设置中读取默认的launcher组件名(ComponentName),如果存在,则直接使用该组件名创建一个显式Intent,并通过包管理服务(PackageManagerService)获取对应的ActivityInfo对象;如果不存在,则使用隐式Intent查询匹配的Activity,并返回第一个匹配结果(通常是系统自带的launcher)。

当用户在选择器中选择了某个launcher应用,并勾选了“始终”选项时,系统会调用ActivityManagerService中的setHomeActivity方法,将用户选择的launcher组件名保存在系统设置中。这样,下次再按下Home键时,就会直接启动该组件对应的Activity。

设置第三方应用为默认launcher
有了上面的分析,我们就可以知道如何通过修改源码来实现设置第三方应用为默认launcher的功能。我们只需要修改RootWindowContainer类中的resolveHomeActivity方法,让它不再从系统设置中读取默认的launcher组件名,而是从我们指定的地方获取。例如,我们可以使用一个系统属性(SystemProperty)来存储我们想要设置为默认launcher的应用包名(PackageName),然后在该方法中根据该包名查询匹配的Activity,并返回其ActivityInfo对象。这样,我们就可以通过修改系统属性来控制默认的launcher应用。

具体来说,我们可以按照以下步骤来修改源码:
vi frameworks/base/services/core/java/com/android/server/wm/RootWindowContainer.java

1.在RootWindowContainer类中导入以下几个类:
 

import android.os.SystemProperties;
 import android.text.TextUtils;
 import android.content.ComponentName;
 import android.content.Intent;
 import android.content.pm.PackageManager;
 import android.content.pm.ResolveInfo;

2.在resolveHomeActivity方法中添加以下代码:

// 从系统属性中获取Home包名
 String homePackageName = SystemProperties.get("persist.home.package", null);// 如果没有设置Home包名,则使用默认的"com.android.launcher3"
 if (TextUtils.isEmpty(homePackageName)) {
     homePackageName = "com.android.launcher3"; // 默认launcher包名
 } try {
     Intent it = new Intent(Intent.ACTION_MAIN);
     List<ResolveInfo> list = AppGlobals.getPackageManager().queryIntentActivities(homeIntent,
             homeIntent.resolveTypeIfNeeded(mService.mContext.getContentResolver()), ActivityManagerService.STOCK_PM_FLAGS, userId).getList();
     final int count = list.size();
     for (int i = 0; i < count; i++) {
         ResolveInfo r = list.get(i);
         if (homePackageName.equals(r.activityInfo.packageName)) {
             Slog.d(TAG, "3rd launcher: " + r.activityInfo.packageName + "@" + r.activityInfo.name);
             comp = new ComponentName(homePackageName, r.activityInfo.name);
             aInfo = r.activityInfo;
             break;
         }
     }
 } catch (Exception e) {
     e.printStackTrace();
 }if (aInfo == null) {
     // 如果指定的Home包名应用未安装或找不到指定的Activity,启动默认的Launcher
     PackageManager packageManager = mService.mContext.getPackageManager();
     homeIntent = new Intent(Intent.ACTION_MAIN).addCategory(Intent.CATEGORY_HOME);
     comp = homeIntent.resolveActivity(packageManager);
     if (comp != null) {
         try {
             aInfo = packageManager.getActivityInfo(comp, flags);
         } catch (PackageManager.NameNotFoundException e) {
             e.printStackTrace();
         }
     }
 }if (aInfo == null) {
     Slog.wtf(TAG, "No home screen found for " + homeIntent, new Throwable());
     return null;
 }

3. 注释掉原来的代码:

/*@VisibleForTesting
ActivityInfo resolveHomeActivity(int userId, Intent homeIntent) {
    final int flags = ActivityManagerService.STOCK_PM_FLAGS;
    final ComponentName comp = homeIntent.getComponent();
    ActivityInfo aInfo;
    if (comp != null) {
        try {
            aInfo = AppGlobals.getPackageManager().getActivityInfo(comp, flags, userId);
        } catch (RemoteException e) {
            return null;
        }
    } else {
        ResolveInfo info = AppGlobals.getPackageManager().resolveIntent(homeIntent,
                homeIntent.resolveTypeIfNeeded(mService.mContext.getContentResolver()), flags, userId);
        if (info != null) {
            aInfo = info.activityInfo;
        } else {
            Slog.wtf(TAG, "No home screen found for " + homeIntent, new Throwable());
            return null;
        }
    }

    aInfo = new ActivityInfo(aInfo);
    aInfo.applicationInfo = mService.getAppInfoForUser(aInfo.applicationInfo, userId);
    return aInfo;
}*/

重新编译并刷入系统 , 我的验证结果是 :
如果默认没有预装第三方launcher , 则走默认launcher3 。
如果装了第三方launcher 则走第三方的 , 并且按home和重启不会再次弹框。
总结
本文介绍了Android系统中launcher应用的启动和切换的原理,以及如何通过修改源码来实现设置第三方应用为默认launcher的功能。通过这个例子,我们可以了解Android系统中隐式Intent和显式Intent的区别,以及如何使用系统属性和包管理服务来控制应用的启动。
 


【版权声明】本文内容来自摩杜云社区用户原创、第三方投稿、转载,内容版权归原作者所有。本网站的目的在于传递更多信息,不拥有版权,亦不承担相应法律责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@moduyun.com

  1. 分享:
最后一次编辑于 2023年11月18日 0

暂无评论

推荐阅读
HvTJUzsxOBtS