【Android 11】AOSP Settings APP info 界面展示所有应用

2024-02-26 16:36

本文主要是介绍【Android 11】AOSP Settings APP info 界面展示所有应用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

背景

Android 11 的AOSP settings的默认情况中,点击应用和通知,展示全部应用之后里面是筛选过的应用。(APP info界面)
有很多内置的应用以及插件是被过滤掉的不显示的。
但是客户提出想要在右上角菜单栏加一个菜单,可以选择显示或者不显示全部的apps
经过研究之后发现加菜单比较麻烦。所以选择在顶部加一个按钮来实现。

一、APP info界面入口

packages/apps/Settings/src/com/android/settings/applications/RecentAppsPreferenceController.javapublic void displayPreference(PreferenceScreen screen) {super.displayPreference(screen);mDivider = screen.findPreference(KEY_DIVIDER);mRecentAppsPreference = screen.findPreference(getPreferenceKey());final View view = mRecentAppsPreference.findViewById(R.id.app_entities_header);mAppEntitiesController = AppEntitiesHeaderController.newInstance(mContext, view).setHeaderTitleRes(R.string.recent_app_category_title).setHeaderDetailsClickListener((View v) -> {mMetricsFeatureProvider.logClickedPreference(mRecentAppsPreference,getMetricsCategory());new SubSettingLauncher(mContext).setDestination(ManageApplications.class.getName()).setArguments(null /* arguments */).setTitleRes(R.string.application_info_label).setSourceMetricsCategory(getMetricsCategory()).launch();});          }

这里通过.setHeaderDetailsClickListener((View v)注册了"显示剩下的XX个应用"的按钮的监听事件。
这个按钮在代码里就是HeaderDetails点击之后就可以跳转到APP info界面
原本我想在这里多加一个同样的按钮,发现不是很好加。
在这段代码的下面还有一个onCountComplete函数,里面会计算app的数量然后返回给复数资源

R.plurals.see_all_apps_title
protected void onCountComplete(int num) {                mAppEntitiesController.setHeaderDetails(                                            mContext.getResources().getQuantityString(R.plurals.see_all_apps_title ,                                num, num));                  mAppEntitiesController.apply();            
}

二、APP info界面的标题部分

xml文件:packages/apps/Settings/res/layout/manage_applications_apps.xml
代码文件:packages/apps/Settings/src/com/android/settings/applications/manageapplications/ManageApplications.java
稍微梳理一下,前面的入口代码中的.launch()方法会进入到ManageApplications.java
然后会根据.setArguments(null /* arguments */)传的参数的内容,进行很多不同的处理

  String className = args != null ? args.getString(EXTRA_CLASSNAME) : null;if (className == null) {className = intent.getComponent().getClassName();}if (className.equals(StorageUseActivity.class.getName())) {if (args != null && args.containsKey(EXTRA_VOLUME_UUID)) {mVolumeUuid = args.getString(EXTRA_VOLUME_UUID);mStorageType = args.getInt(EXTRA_STORAGE_TYPE, STORAGE_TYPE_DEFAULT);mListType = LIST_TYPE_STORAGE;} else {// No volume selected, display a normal list, sorted by size.mListType = LIST_TYPE_MAIN;}mSortOrder = R.id.sort_order_size;} else if (className.equals(UsageAccessSettingsActivity.class.getName())) {.........

我们传的是空参所以是

else {if (screenTitle == -1) {screenTitle = R.string.application_info_label;}mListType = LIST_TYPE_MAIN;}

仅仅是把标题设置成了R.string.application_info_label而已

三、APP info界面的数据来源

最值得注意的是onCreateView中的代码

    mRecyclerView = mListContainer.findViewById(R.id.apps_list);mRecyclerView.setItemAnimator(null);mRecyclerView.setLayoutManager(new LinearLayoutManager(getContext(), RecyclerView.VERTICAL, false /* reverseLayout */));mRecyclerView.setAdapter(mApplications);

这里的 mRecyclerView 就是应用列表,setAdapter设置了一个适配器,应用列表的所有数据都来自于适配器。
因此筛选显示的应用列表的逻辑也和这个Adapter有关系。

创造过滤器:mFilter = appFilterRegistry.get(appFilterRegistry.getDefaultFilterType(mListType));
创造适配器:mApplications = new ApplicationsAdapter(mApplicationsState, this, mFilter,savedInstanceState);
设置适配器:mRecyclerView.setAdapter(mApplications);

ApplicationsAdapter的具体内容在这个java文件的下面的部分

四、APP info界面的加载函数

rebuild()函数:是用来重新加载刷新这个界面的

if (!mManageApplications.mShowSystem) {if (LIST_TYPES_WITH_INSTANT.contains(mManageApplications.mListType)) {filterObj = new CompoundFilter(filterObj,ApplicationsState.FILTER_DOWNLOADED_AND_LAUNCHER_AND_INSTANT);} else {filterObj = new CompoundFilter(filterObj,ApplicationsState.FILTER_DOWNLOADED_AND_LAUNCHER);}
}

rebuild()函数中有这么一段代码,通过判断mManageApplications.mShowSystem的值,来选择是否要加上某些过滤器。这就是过滤掉一些不必要的app的代码所在。

五、处理方法

  1. 首先在顶部加一个Button
--- a/packages/apps/Settings/res/layout/manage_applications_apps.xml
+++ b/packages/apps/Settings/res/layout/manage_applications_apps.xml
@@ -32,11 +32,22 @@android:layout_height="match_parent"android:visibility="gone">+            <Button
+                android:id="@+id/show_all_apps"
+                android:layout_width="wrap_content"
+                android:layout_height="wrap_content"
+                android:text="@string/show_all_apps_button"
+                android:layout_gravity="center_horizontal"
+                android:layout_marginStart="16dp"
+                android:layout_marginTop="16dp"
+                />
+<androidx.recyclerview.widget.RecyclerViewandroid:id="@+id/apps_list"android:layout_width="match_parent"android:layout_height="match_parent"android:clipToPadding="false"
+                android:layout_marginTop="80dp"android:scrollbars="none"settings:fastScrollEnabled="true"settings:fastScrollHorizontalThumbDrawable="@drawable/thumb_drawable"

这里加了一个Button同时给app列表一个上边距,防止和按钮重叠。

  1. 实现按钮逻辑
--- a/packages/apps/Settings/src/com/android/settings/applications/manageapplications/ManageApplications.java
+++ b/packages/apps/Settings/src/com/android/settings/applications/manageapplications/ManageApplications.java
@@ -134,6 +134,7 @@ import java.util.Arrays;import java.util.Collections;import java.util.Comparator;import java.util.Set;
+import android.widget.Button;@@ -446,6 +447,28 @@ public class ManageApplications extends InstrumentedFragmentmResetAppsHelper.onRestoreInstanceState(savedInstanceState);+            Button showAllAppsButton = mRootView.findViewById(R.id.show_all_apps);
+            if (showAllAppsButton != null) {
+                if (mApplications.getShowSystem()) {
+                    showAllAppsButton.setText(R.string.show_installed_apps_button);
+                } else {
+                    showAllAppsButton.setText(R.string.show_all_apps_button);
+                }
+                showAllAppsButton.setOnClickListener(new View.OnClickListener() {
+                    @Override
+                    public void onClick(View v) {
+                        boolean newShowSystemValue = !mApplications.getShowSystem();
+                        mApplications.setShowSystem(newShowSystemValue);
+                        if (newShowSystemValue) {
+                            showAllAppsButton.setText(R.string.show_installed_apps_button);
+                        } else {
+                            showAllAppsButton.setText(R.string.show_all_apps_button);
+                        }
+                        mApplications.rebuild();
+                    }
+                });
+            }
+return mRootView;}@@ -1243,6 +1266,19 @@
+        public void setShowSystem(boolean isShowSystem) {
+            if (mManageApplications != null) {
+                mManageApplications.mShowSystem = isShowSystem;
+            }
+        }
+
+        public boolean getShowSystem() {
+            if (mManageApplications != null) {
+                return mManageApplications.mShowSystem;
+            }
+            return false;   // default to false
+        }
+

首先在ApplicationsAdapter类里面加入了两个公共方法用来设置mShowSystem 的值
然后在ManageApplications类的onCreateView方法里面设置监听动态改变相应的值就好

这篇关于【Android 11】AOSP Settings APP info 界面展示所有应用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/749483

相关文章

分布式锁在Spring Boot应用中的实现过程

《分布式锁在SpringBoot应用中的实现过程》文章介绍在SpringBoot中通过自定义Lock注解、LockAspect切面和RedisLockUtils工具类实现分布式锁,确保多实例并发操作... 目录Lock注解LockASPect切面RedisLockUtils工具类总结在现代微服务架构中,分布

Python标准库之数据压缩和存档的应用详解

《Python标准库之数据压缩和存档的应用详解》在数据处理与存储领域,压缩和存档是提升效率的关键技术,Python标准库提供了一套完整的工具链,下面小编就来和大家简单介绍一下吧... 目录一、核心模块架构与设计哲学二、关键模块深度解析1.tarfile:专业级归档工具2.zipfile:跨平台归档首选3.

使用IDEA部署Docker应用指南分享

《使用IDEA部署Docker应用指南分享》本文介绍了使用IDEA部署Docker应用的四步流程:创建Dockerfile、配置IDEADocker连接、设置运行调试环境、构建运行镜像,并强调需准备本... 目录一、创建 dockerfile 配置文件二、配置 IDEA 的 Docker 连接三、配置 Do

Android Paging 分页加载库使用实践

《AndroidPaging分页加载库使用实践》AndroidPaging库是Jetpack组件的一部分,它提供了一套完整的解决方案来处理大型数据集的分页加载,本文将深入探讨Paging库... 目录前言一、Paging 库概述二、Paging 3 核心组件1. PagingSource2. Pager3.

深入浅出SpringBoot WebSocket构建实时应用全面指南

《深入浅出SpringBootWebSocket构建实时应用全面指南》WebSocket是一种在单个TCP连接上进行全双工通信的协议,这篇文章主要为大家详细介绍了SpringBoot如何集成WebS... 目录前言为什么需要 WebSocketWebSocket 是什么Spring Boot 如何简化 We

Java Stream流之GroupBy的用法及应用场景

《JavaStream流之GroupBy的用法及应用场景》本教程将详细介绍如何在Java中使用Stream流的groupby方法,包括基本用法和一些常见的实际应用场景,感兴趣的朋友一起看看吧... 目录Java Stream流之GroupBy的用法1. 前言2. 基础概念什么是 GroupBy?Stream

python中列表应用和扩展性实用详解

《python中列表应用和扩展性实用详解》文章介绍了Python列表的核心特性:有序数据集合,用[]定义,元素类型可不同,支持迭代、循环、切片,可执行增删改查、排序、推导式及嵌套操作,是常用的数据处理... 目录1、列表定义2、格式3、列表是可迭代对象4、列表的常见操作总结1、列表定义是处理一组有序项目的

C#中的Converter的具体应用

《C#中的Converter的具体应用》C#中的Converter提供了一种灵活的类型转换机制,本文详细介绍了Converter的基本概念、使用场景,具有一定的参考价值,感兴趣的可以了解一下... 目录Converter的基本概念1. Converter委托2. 使用场景布尔型转换示例示例1:简单的字符串到

Spring Boot Actuator应用监控与管理的详细步骤

《SpringBootActuator应用监控与管理的详细步骤》SpringBootActuator是SpringBoot的监控工具,提供健康检查、性能指标、日志管理等核心功能,支持自定义和扩展端... 目录一、 Spring Boot Actuator 概述二、 集成 Spring Boot Actuat

PyTorch中的词嵌入层(nn.Embedding)详解与实战应用示例

《PyTorch中的词嵌入层(nn.Embedding)详解与实战应用示例》词嵌入解决NLP维度灾难,捕捉语义关系,PyTorch的nn.Embedding模块提供灵活实现,支持参数配置、预训练及变长... 目录一、词嵌入(Word Embedding)简介为什么需要词嵌入?二、PyTorch中的nn.Em