赞
踩
Android系统中,launcher是用户与系统交互的主要界面,它负责显示桌面、应用列表、小部件等内容。Android系统允许用户安装第三方的launcher应用,以替换系统自带的launcher。
但如何满足客制化需求让第三方的launcher应用成为默认的launcher呢?本文将从源码的角度,分析Android系统是如何处理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的功能。我们只需要修改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;
}*/
本文介绍了Android系统中launcher应用的启动和切换的原理,以及如何通过修改源码来实现设置第三方应用为默认launcher的功能。通过这个例子,我们可以了解Android系统中隐式Intent和显式Intent的区别,以及如何使用系统属性和包管理服务来控制应用的启动。
希望本文对你有所帮助,如果你有任何问题或建议,欢迎在评论区留言。谢谢!
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。