【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

相关文章

利用Python操作Word文档页码的实际应用

《利用Python操作Word文档页码的实际应用》在撰写长篇文档时,经常需要将文档分成多个节,每个节都需要单独的页码,下面:本文主要介绍利用Python操作Word文档页码的相关资料,文中通过代码... 目录需求:文档详情:要求:该程序的功能是:总结需求:一次性处理24个文档的页码。文档详情:1、每个

uni-app小程序项目中实现前端图片压缩实现方式(附详细代码)

《uni-app小程序项目中实现前端图片压缩实现方式(附详细代码)》在uni-app开发中,文件上传和图片处理是很常见的需求,但也经常会遇到各种问题,下面:本文主要介绍uni-app小程序项目中实... 目录方式一:使用<canvas>实现图片压缩(推荐,兼容性好)示例代码(小程序平台):方式二:使用uni

Java中的分布式系统开发基于 Zookeeper 与 Dubbo 的应用案例解析

《Java中的分布式系统开发基于Zookeeper与Dubbo的应用案例解析》本文将通过实际案例,带你走进基于Zookeeper与Dubbo的分布式系统开发,本文通过实例代码给大家介绍的非常详... 目录Java 中的分布式系统开发基于 Zookeeper 与 Dubbo 的应用案例一、分布式系统中的挑战二

Java 缓存框架 Caffeine 应用场景解析

《Java缓存框架Caffeine应用场景解析》文章介绍Caffeine作为高性能Java本地缓存框架,基于W-TinyLFU算法,支持异步加载、灵活过期策略、内存安全机制及统计监控,重点解析其... 目录一、Caffeine 简介1. 框架概述1.1 Caffeine的核心优势二、Caffeine 基础2

使用Node.js和PostgreSQL构建数据库应用

《使用Node.js和PostgreSQL构建数据库应用》PostgreSQL是一个功能强大的开源关系型数据库,而Node.js是构建高效网络应用的理想平台,结合这两个技术,我们可以创建出色的数据驱动... 目录初始化项目与安装依赖建立数据库连接执行CRUD操作查询数据插入数据更新数据删除数据完整示例与最佳

Android实现图片浏览功能的示例详解(附带源码)

《Android实现图片浏览功能的示例详解(附带源码)》在许多应用中,都需要展示图片并支持用户进行浏览,本文主要为大家介绍了如何通过Android实现图片浏览功能,感兴趣的小伙伴可以跟随小编一起学习一... 目录一、项目背景详细介绍二、项目需求详细介绍三、相关技术详细介绍四、实现思路详细介绍五、完整实现代码

在Android中使用WebView在线查看PDF文件的方法示例

《在Android中使用WebView在线查看PDF文件的方法示例》在Android应用开发中,有时我们需要在客户端展示PDF文件,以便用户可以阅读或交互,:本文主要介绍在Android中使用We... 目录简介:1. WebView组件介绍2. 在androidManifest.XML中添加Interne

Python一次性将指定版本所有包上传PyPI镜像解决方案

《Python一次性将指定版本所有包上传PyPI镜像解决方案》本文主要介绍了一个安全、完整、可离线部署的解决方案,用于一次性准备指定Python版本的所有包,然后导出到内网环境,感兴趣的小伙伴可以跟随... 目录为什么需要这个方案完整解决方案1. 项目目录结构2. 创建智能下载脚本3. 创建包清单生成脚本4

PHP应用中处理限流和API节流的最佳实践

《PHP应用中处理限流和API节流的最佳实践》限流和API节流对于确保Web应用程序的可靠性、安全性和可扩展性至关重要,本文将详细介绍PHP应用中处理限流和API节流的最佳实践,下面就来和小编一起学习... 目录限流的重要性在 php 中实施限流的最佳实践使用集中式存储进行状态管理(如 Redis)采用滑动

Three.js构建一个 3D 商品展示空间完整实战项目

《Three.js构建一个3D商品展示空间完整实战项目》Three.js是一个强大的JavaScript库,专用于在Web浏览器中创建3D图形,:本文主要介绍Three.js构建一个3D商品展... 目录引言项目核心技术1. 项目架构与资源组织2. 多模型切换、交互热点绑定3. 移动端适配与帧率优化4. 可