经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 移动开发 » Android » 查看文章
Android桌面Launcher源码浅析
来源:cnblogs  作者:柳云居士  时间:2024/5/31 17:56:42  对本文有异议

Android启动过程-万字长文(Android14)中介绍了Android系统的启动过程,本篇文章将继续介绍桌面应用Launcher。

一、Launcher介绍

  • Android启动过程-万字长文(Android14)中提到Launcher是Android系统启动后,由SystemServerActivity Manager Service (AMS)加载的第一个应用程序
  • Launcher又被称为桌面程序,负责Android桌面的启动和管理
  • 用户使用的应用程序(App)都是通过Launcher来启动的

二、下载及编译

2.1 下载

  • 使用Git下载Launcher源码:
  1. git clone https://android.googlesource.com/platform/packages/apps/Launcher3
  • 进入项目目录
  1. cd Launcher3
  • 切换到Android14分支
  1. git checkout android14-release

2.2 编译

使用AndroidStudio编译下载好的Launcher3工程

编译过程中遇到问题及解决方案可以参考以下博客:

三、源码解析

3.1 AndroidManifest.xml

在项目根目录的AndroidManifest.xml,定义了Launcher做为桌面程序的属性:

  1. <application>
  2. <activity
  3. android:name="com.android.launcher3.Launcher"
  4. android:launchMode="singleTask">
  5. <intent-filter>
  6. <category android:name="android.intent.category.HOME" />
  7. </intent-filter>
  8. </activity>
  9. </application>
  • android.intent.category.HOME: 告诉系统这是一个启动器(Launcher)应用程序,系统在初始化完成后会通过ActivityTaskManagerServicegetHomeIntent方法获取和启动桌面程序。具体可参见Android启动过程-万字长文(Android14)
  • 开发人员也可以自己开发一个桌面程序(如微软桌面),用户安装完成后,可以在系统设置中修改默认启动的桌面程序

3.2 Launcher.java

Launcher.java是Launcher的启动页面,负责资源初始化和桌面UI创建

3.2.1 onCreate方法

  1. protected void onCreate(Bundle savedInstanceState) {
  2. super.onCreate(savedInstanceState);
  3. // 获取 LauncherAppState 实例和模型
  4. LauncherAppState app = LauncherAppState.getInstance(this);
  5. mModel = app.getModel();
  6. // 初始化不变的设备配置文件
  7. InvariantDeviceProfile idp = app.getInvariantDeviceProfile();
  8. initDeviceProfile(idp);
  9. idp.addOnChangeListener(this);
  10. // 获取共享首选项和图标缓存
  11. mSharedPrefs = LauncherPrefs.getPrefs(this);
  12. mIconCache = app.getIconCache();
  13. // 创建无障碍代理
  14. mAccessibilityDelegate = createAccessibilityDelegate();
  15. // 初始化拖动控制器
  16. initDragController();
  17. // 创建所有应用程序控制器
  18. mAllAppsController = new AllAppsTransitionController(this);
  19. // 创建状态管理器
  20. mStateManager = new StateManager<>(this, NORMAL);
  21. // 创建引导首选项
  22. mOnboardingPrefs = createOnboardingPrefs(mSharedPrefs);
  23. // 设置视图
  24. setupViews();
  25. // 初始化Widget
  26. mAppWidgetManager = new WidgetManagerHelper(this);
  27. mAppWidgetHolder = createAppWidgetHolder();
  28. mAppWidgetHolder.startListening();
  29. // 设置内容视图
  30. setContentView(getRootView());
  31. ComposeInitializer.initCompose(this);
  32. }

3.2.2 setupViews方法

  1. protected void setupViews() {
  2. // 创建根视图
  3. inflateRootView(R.layout.launcher);
  4. // 获取拖动层和焦点处理器
  5. mDragLayer = findViewById(R.id.drag_layer);
  6. mFocusHandler = mDragLayer.getFocusIndicatorHelper();
  7. // 获取工作区、总览面板和Hotseat
  8. mWorkspace = mDragLayer.findViewById(R.id.workspace);
  9. mWorkspace.initParentViews(mDragLayer);
  10. mOverviewPanel = findViewById(R.id.overview_panel);
  11. mHotseat = findViewById(R.id.hotseat);
  12. // 将工作区设置为Hotseat
  13. mHotseat.setWorkspace(mWorkspace);
  14. // 设置拖动层
  15. mDragLayer.setup(mDragController, mWorkspace);
  16. // 设置工作区
  17. mWorkspace.setup(mDragController);
  18. // 在工作区绑定之前,确保我们将壁纸偏移锁定到默认状态,否则在RTL中我们将更新错误的偏移量
  19. mWorkspace.lockWallpaperToDefaultPage();
  20. mWorkspace.bindAndInitFirstWorkspaceScreen();
  21. mDragController.addDragListener(mWorkspace);
  22. // 获取搜索/删除/卸载栏
  23. mDropTargetBar = mDragLayer.findViewById(R.id.drop_target_bar);
  24. // 设置应用程序视图
  25. mAppsView = findViewById(R.id.apps_view);
  26. mAppsView.setAllAppsTransitionController(mAllAppsController);
  27. // 设置拖动控制器(拖动目标必须按优先级的相反顺序添加)
  28. mDropTargetBar.setup(mDragController);
  29. mAllAppsController.setupViews(mScrimView, mAppsView);
  30. // 如果启用了点分页,则设置工作区的分页指示器
  31. if (SHOW_DOT_PAGINATION.get()) {
  32. mWorkspace.getPageIndicator().setShouldAutoHide(true);
  33. mWorkspace.getPageIndicator().setPaintColor(
  34. Themes.getAttrBoolean(this, R.attr.isWorkspaceDarkText)
  35. ? Color.BLACK
  36. : Color.WHITE);
  37. }
  38. }
  • Workspace:工作区,也是我们常说的桌面区域,包括搜索框,桌面,壁纸
  • AppsView:应用程序列表
  • Widget:小组件

三、Workspace、AppsView和Widget示例

3.1 Workspace(工作区)

  • 结构说明

3.2 AppsView(应用程序视图)

3.3 Widget(小组件)

四、点击App图标的事件响应

4.1 触发ItemClickHandler的onClick方法

  • ItemClickHandler负责处理桌面应用图标的点击事件。
  • 桌面图标的点击事件最终会触发ItemClickHandleronClick方法
  • onClick方法最终会触发startAppShortcutOrInfoActivity方法
  1. /**
  2. * Class for handling clicks on workspace and all-apps items
  3. */
  4. public class ItemClickHandler {
  5. private static void onClick(View v) {
  6. startAppShortcutOrInfoActivity(v, (AppInfo) tag, launcher);
  7. }
  8. // 通知launcher启动Activity
  9. private static void startAppShortcutOrInfoActivity(View v, ItemInfo item, Launcher launcher) {
  10. launcher.startActivitySafely(v, intent, item);
  11. }
  12. }

4.2 Launcher通知系统启动App

Launcher.java的startActivitySafely方法中调用ActivityContext.java的startActivitySafely方法

  1. public RunnableList startActivitySafely(View v, Intent intent, ItemInfo item) {
  2. RunnableList result = super.startActivitySafely(v, intent, item);
  3. }

ActivityContext.java的startActivitySafely方法中调用了

  1. public interface ActivityContext {
  2. default RunnableList startActivitySafely(
  3. View v, Intent intent, @Nullable ItemInfo item) {
  4. if (isShortcut) {
  5. // Shortcuts need some special checks due to legacy reasons.
  6. startShortcutIntentSafely(intent, optsBundle, item);
  7. }
  8. }
  9. default void startShortcutIntentSafely(Intent intent, Bundle optsBundle, ItemInfo info) {
  10. if (info.itemType == LauncherSettings.Favorites.ITEM_TYPE_DEEP_SHORTCUT) {
  11. // 通过快捷方式启动
  12. startShortcut(packageName, id, intent.getSourceBounds(), optsBundle, info.user);
  13. } else {
  14. // 普通方式启动,应用程序走这个分支
  15. ((Context) this).startActivity(intent, optsBundle);
  16. }
  17. }
  18. }

Android 应用快捷方式(Shortcut)官方文档

最终通过frameworks/base/core/java/android/app/Activity.java 源码地址中的startActivity方法启动了对应的应用程序。

原文链接:https://www.cnblogs.com/anywherego/p/18225051

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号