【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

相关文章

Android中Dialog的使用详解

《Android中Dialog的使用详解》Dialog(对话框)是Android中常用的UI组件,用于临时显示重要信息或获取用户输入,本文给大家介绍Android中Dialog的使用,感兴趣的朋友一起... 目录android中Dialog的使用详解1. 基本Dialog类型1.1 AlertDialog(

Python中随机休眠技术原理与应用详解

《Python中随机休眠技术原理与应用详解》在编程中,让程序暂停执行特定时间是常见需求,当需要引入不确定性时,随机休眠就成为关键技巧,下面我们就来看看Python中随机休眠技术的具体实现与应用吧... 目录引言一、实现原理与基础方法1.1 核心函数解析1.2 基础实现模板1.3 整数版实现二、典型应用场景2

Python Dash框架在数据可视化仪表板中的应用与实践记录

《PythonDash框架在数据可视化仪表板中的应用与实践记录》Python的PlotlyDash库提供了一种简便且强大的方式来构建和展示互动式数据仪表板,本篇文章将深入探讨如何使用Dash设计一... 目录python Dash框架在数据可视化仪表板中的应用与实践1. 什么是Plotly Dash?1.1

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda

Python实现将MySQL中所有表的数据都导出为CSV文件并压缩

《Python实现将MySQL中所有表的数据都导出为CSV文件并压缩》这篇文章主要为大家详细介绍了如何使用Python将MySQL数据库中所有表的数据都导出为CSV文件到一个目录,并压缩为zip文件到... python将mysql数据库中所有表的数据都导出为CSV文件到一个目录,并压缩为zip文件到另一个

利用Go语言开发文件操作工具轻松处理所有文件

《利用Go语言开发文件操作工具轻松处理所有文件》在后端开发中,文件操作是一个非常常见但又容易出错的场景,本文小编要向大家介绍一个强大的Go语言文件操作工具库,它能帮你轻松处理各种文件操作场景... 目录为什么需要这个工具?核心功能详解1. 文件/目录存javascript在性检查2. 批量创建目录3. 文件

Android自定义Scrollbar的两种实现方式

《Android自定义Scrollbar的两种实现方式》本文介绍两种实现自定义滚动条的方法,分别通过ItemDecoration方案和独立View方案实现滚动条定制化,文章通过代码示例讲解的非常详细,... 目录方案一:ItemDecoration实现(推荐用于RecyclerView)实现原理完整代码实现

Java中&和&&以及|和||的区别、应用场景和代码示例

《Java中&和&&以及|和||的区别、应用场景和代码示例》:本文主要介绍Java中的逻辑运算符&、&&、|和||的区别,包括它们在布尔和整数类型上的应用,文中通过代码介绍的非常详细,需要的朋友可... 目录前言1. & 和 &&代码示例2. | 和 ||代码示例3. 为什么要使用 & 和 | 而不是总是使

Python循环缓冲区的应用详解

《Python循环缓冲区的应用详解》循环缓冲区是一个线性缓冲区,逻辑上被视为一个循环的结构,本文主要为大家介绍了Python中循环缓冲区的相关应用,有兴趣的小伙伴可以了解一下... 目录什么是循环缓冲区循环缓冲区的结构python中的循环缓冲区实现运行循环缓冲区循环缓冲区的优势应用案例Python中的实现库

SpringBoot整合MybatisPlus的基本应用指南

《SpringBoot整合MybatisPlus的基本应用指南》MyBatis-Plus,简称MP,是一个MyBatis的增强工具,在MyBatis的基础上只做增强不做改变,下面小编就来和大家介绍一下... 目录一、MyBATisPlus简介二、SpringBoot整合MybatisPlus1、创建数据库和