前言
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
- 在
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;
- 在
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; }
- 注释掉原来的代码:
/*@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的区别,以及如何使用系统属性和包管理服务来控制应用的启动。
希望本文对你有所帮助,如果你有任何问题或建议,欢迎在评论区留言。谢谢!