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的区别,以及如何使用系统属性和包管理服务来控制应用的启动。

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

相关文章
|
1月前
|
JSON Java Android开发
探索安卓开发之旅:打造你的第一个天气应用
【10月更文挑战第30天】在这个数字时代,掌握移动应用开发技能无疑是进入IT行业的敲门砖。本文将引导你开启安卓开发的奇妙之旅,通过构建一个简易的天气应用来实践你的编程技能。无论你是初学者还是有一定经验的开发者,这篇文章都将成为你宝贵的学习资源。我们将一步步地深入到安卓开发的世界中,从搭建开发环境到实现核心功能,每个环节都充满了发现和创造的乐趣。让我们开始吧,一起在代码的海洋中航行!
|
1月前
|
算法 JavaScript Android开发
|
1月前
|
存储 搜索推荐 Java
打造个性化安卓应用:从设计到实现
【10月更文挑战第30天】在数字化时代,拥有一个个性化的安卓应用不仅能够提升用户体验,还能加强品牌识别度。本文将引导您了解如何从零开始设计和实现一个安卓应用,涵盖用户界面设计、功能开发和性能优化等关键环节。我们将以一个简单的记事本应用为例,展示如何通过Android Studio工具和Java语言实现基本功能,同时确保应用流畅运行。无论您是初学者还是希望提升现有技能的开发者,这篇文章都将为您提供宝贵的见解和实用的技巧。
|
1月前
|
安全 搜索推荐 Android开发
揭秘安卓与iOS系统的差异:技术深度对比
【10月更文挑战第27天】 本文深入探讨了安卓(Android)与iOS两大移动操作系统的技术特点和用户体验差异。通过对比两者的系统架构、应用生态、用户界面、安全性等方面,揭示了为何这两种系统能够在市场中各占一席之地,并为用户提供不同的选择。文章旨在为读者提供一个全面的视角,理解两种系统的优势与局限,从而更好地根据自己的需求做出选择。
79 2
|
1月前
|
搜索推荐 开发工具 Android开发
打造个性化Android应用:从设计到实现的旅程
【10月更文挑战第26天】在这个数字时代,拥有一个能够脱颖而出的移动应用是成功的关键。本文将引导您了解如何从概念化阶段出发,通过设计、开发直至发布,一步步构建一个既美观又实用的Android应用。我们将探讨用户体验(UX)设计的重要性,介绍Android开发的核心组件,并通过实际案例展示如何克服开发中的挑战。无论您是初学者还是有经验的开发者,这篇文章都将为您提供宝贵的见解和实用的技巧,帮助您在竞争激烈的应用市场中脱颖而出。
|
1月前
|
算法 Java 数据库
Android 应用的主线程在什么情况下会被阻塞?
【10月更文挑战第20天】为了避免主线程阻塞,我们需要合理地设计和优化应用的代码。将耗时操作移到后台线程执行,使用异步任务、线程池等技术来提高应用的并发处理能力。同时,要注意避免出现死循环、不合理的锁使用等问题。通过这些措施,可以确保主线程能够高效地运行,提供流畅的用户体验。
45 2
|
1月前
|
安全 搜索推荐 程序员
深入探索Android系统的碎片化问题及其解决方案
在移动操作系统的世界中,Android以其开放性和灵活性赢得了广泛的市场份额。然而,这种开放性也带来了一个众所周知的问题——系统碎片化。本文旨在探讨Android系统碎片化的现状、成因以及可能的解决方案,为开发者和用户提供一种全新的视角来理解这一现象。通过分析不同版本的Android系统分布、硬件多样性以及更新机制的影响,我们提出了一系列针对性的策略,旨在减少碎片化带来的影响,提升用户体验。
|
1月前
|
安全 Android开发 iOS开发
深入探索iOS与Android系统的差异性及优化策略
在当今数字化时代,移动操作系统的竞争尤为激烈,其中iOS和Android作为市场上的两大巨头,各自拥有庞大的用户基础和独特的技术特点。本文旨在通过对比分析iOS与Android的核心差异,探讨各自的优势与局限,并提出针对性的优化策略,以期为用户提供更优质的使用体验和为开发者提供有价值的参考。
|
1月前
|
安全 Android开发 iOS开发
安卓系统与iOS系统的比较####
【10月更文挑战第26天】 本文将深入探讨安卓(Android)和iOS这两大主流移动操作系统的各自特点、优势与不足。通过对比分析,帮助读者更好地理解两者在用户体验、应用生态、系统安全等方面的差异,从而为消费者在选择智能手机时提供参考依据。无论你是技术爱好者还是普通用户,这篇文章都将为你揭示两大系统背后的故事和技术细节。 ####
47 0