Android系统 设置第三方应用为默认Launcher实现和原理分析

简介: Android系统 设置第三方应用为默认Launcher实现和原理分析

前言

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;
  1. 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;
}
  1. 注释掉原来的代码:
/*@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;
}*/
  1. 重新编译并刷入系统 , 我的验证结果是 :
  • 如果默认没有预装第三方launcher , 则走默认launcher3 。
  • 如果装了第三方launcher 则走第三方的 , 并且按home和重启不会再次弹框。

总结

本文介绍了Android系统中launcher应用的启动和切换的原理,以及如何通过修改源码来实现设置第三方应用为默认launcher的功能。通过这个例子,我们可以了解Android系统中隐式Intent和显式Intent的区别,以及如何使用系统属性和包管理服务来控制应用的启动。

希望本文对你有所帮助,如果你有任何问题或建议,欢迎在评论区留言。谢谢!

相关文章
|
11月前
|
存储 Android开发
如何查看Flutter应用在Android设备上已被撤销的权限?
如何查看Flutter应用在Android设备上已被撤销的权限?
582 64
|
10月前
|
Android开发 开发者
Android设置View是否可用
在Android开发中,有时需要将布局设置为不可点击状态(失去焦点)。常见的解决方法是使用`setOnClickListener(null)`,但本文介绍一种更通用的方式:通过封装`setViewEnabled`方法实现。该方法可递归设置View及其子View的启用状态,支持传入目标View和布尔值(`true`为可用,`false`为禁用)。例如,调用`setViewEnabled(edittext, false)`即可禁用EditText。文章附有源码及示例动图,帮助开发者快速理解与应用。
238 1
|
6月前
|
Linux 测试技术 语音技术
【车载Android】模拟Android系统的高负载环境
本文介绍如何将Linux压力测试工具Stress移植到Android系统,用于模拟高负载环境下的CPU、内存、IO和磁盘压力,帮助开发者优化车载Android应用在多任务并发时的性能问题,提升系统稳定性与用户体验。
502 6
|
6月前
|
Java 数据库 Android开发
基于Android的电子记账本系统
本项目研究开发一款基于Java与Android平台的开源电子记账系统,采用SQLite数据库和Gradle工具,实现高效、安全、便捷的个人财务管理,顺应数字化转型趋势。
|
10月前
|
Android开发 开发者
Android中Dialog位置+样式的设置
本文介绍了在Android开发中如何设置Dialog的位置和样式。通过自定义`MyDialog`类,可以灵活调整Dialog的显示位置,例如将其固定在屏幕底部,并设置宽度匹配父布局。同时,文章还展示了如何模仿Android原生Dialog样式,通过定义`MyDialogStyle`去除标题栏、设置背景透明度、添加阴影效果以及配置点击外部关闭等功能,从而实现更加美观和符合需求的Dialog效果。代码示例详细,便于开发者快速上手实现。
573 2
|
11月前
|
安全 搜索推荐 Android开发
Android系统SELinux安全机制详解
如此看来,SELinux对于大家来说,就像那位不眠不休,严阵以待的港口管理员,守护我们安卓系统的平安,维护这片海港的和谐生态。SELinux就这样,默默无闻,却卫士如山,给予Android系统一份厚重的安全保障。
378 18
|
前端开发 Java Shell
【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
864 20
【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
|
Dart 前端开发 Android开发
【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
426 4
【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
|
前端开发 Android开发
Android 第三方分享中遇到的问题以及解决方案
                                               本文来自http://blog.csdn.net/liuxian13183/ ,引用必须注明出处! 第三方登录和分享过程中难免遇到各种纠结的问题,下面将我遇到的分享给大家。
994 0
|
5月前
|
移动开发 前端开发 Android开发
【02】建立各项目录和页面标准化产品-vue+vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
【02】建立各项目录和页面标准化产品-vue+vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
957 12
【02】建立各项目录和页面标准化产品-vue+vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡