Android 12 SystemUI下拉状态栏禁止QuickQSPanel展开

1.概述

遇到需求,QuickQSPanel首次下拉后展示快捷功能模块以后就是显示QuickQSPanel,而不展开QSPanel,接下来要从下滑手势下拉出状态栏分析功能实现。也就是直接是展开状态。

2、涉及核心类

frameworks\base\packages\SystemUI\src\com\android\systemui\statusbar\phone

Android12或者Android13frameworks\base\packages\SystemUI\src\com\android\systemui\statusbar\phone\NotificationPanelViewController.java
frameworks/base/packages/SystemUI/res/layout/status_bar_expanded.xmlAndroid10frameworks\base\packages\SystemUI\src\com\android\systemui\statusbar\phone\NotificationPanelViewController.java
frameworks/base/packages/SystemUI/res/layout/status_bar_expanded.xml

3、SystemUI下拉状态栏禁止QuickQSPanel展开的核心功能分析和实现

在系统SystemUI中,它主要负责反馈系统及应用状态并与用户保持大量的交互,systemui中主要核心布局控件为以下部分
在SystemUI中,QSPanel 创建是从 StatusBar#makeStatusBarView 开始的
接下来分析的核心功能和布局如下:
StatusBar:通知消息提示和状态展现
NavigationBar:返回,HOME,Recent
KeyGuard:锁屏模块可以看做单独的应用,提供基本的手机个人隐私保护
Recents:近期应用管理,以堆叠栈的形式展现。
Notification Panel:展示系统或应用通知内容。提供快速系统设置开关。
VolumeUI:来用展示或控制音量的变化:媒体音量、铃声音量与闹钟音量
截屏界面:长按电源键+音量下键后截屏,用以展示截取的屏幕照片/内容
PowerUI:主要处理和Power相关的事件,比如省电模式切换、电池电量变化和开关屏事件等。
RingtonePlayer:铃声播放
StackDivider:控制管理分屏
PipUI:提供对于画中画模式的管理

3.1 status_bar_expanded.xml下拉状态栏布局分析

<?xml version="1.0" encoding="utf-8"?>
<!--
**
** Copyright 2006, The Android Open Source Project
**
** Licensed under the Apache License, Version 2.0 (the "License");
** you may not use this file except in compliance with the License.
** You may obtain a copy of the License at
**
**     http://www.apache.org/licenses/LICENSE-2.0
**
** Unless required by applicable law or agreed to in writing, software
** distributed under the License is distributed on an "AS IS" BASIS,
** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
** See the License for the specific language governing permissions and
** limitations under the License.
*/
--><com.android.systemui.statusbar.phone.NotificationPanelViewxmlns:android="http://schemas.android.com/apk/res/android"xmlns:systemui="http://schemas.android.com/apk/res-auto"android:id="@+id/notification_panel"android:layout_width="match_parent"android:layout_height="match_parent"android:background="@android:color/transparent"><FrameLayoutandroid:id="@+id/big_clock_container"android:layout_width="match_parent"android:layout_height="match_parent"android:visibility="gone" /><ViewStubandroid:id="@+id/keyguard_qs_user_switch_stub"android:layout="@layout/keyguard_qs_user_switch"android:layout_height="match_parent"android:layout_width="match_parent" /><includelayout="@layout/keyguard_bottom_area"android:visibility="gone" /><ViewStubandroid:id="@+id/keyguard_user_switcher_stub"android:layout="@layout/keyguard_user_switcher"android:layout_height="match_parent"android:layout_width="match_parent" /><include layout="@layout/status_bar_expanded_plugin_frame"/><include layout="@layout/dock_info_bottom_area_overlay" /><com.android.keyguard.LockIconViewandroid:id="@+id/lock_icon_view"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_gravity="center"><!-- Background protection --><ImageViewandroid:id="@+id/lock_icon_bg"android:layout_width="match_parent"android:layout_height="match_parent"android:background="@drawable/fingerprint_bg"android:visibility="invisible"/><ImageViewandroid:id="@+id/lock_icon"android:layout_width="match_parent"android:layout_height="match_parent"android:padding="@dimen/lock_icon_padding"android:layout_gravity="center"android:scaleType="centerCrop"/></com.android.keyguard.LockIconView><com.android.systemui.statusbar.phone.NotificationsQuickSettingsContainerandroid:layout_width="match_parent"android:layout_height="match_parent"android:layout_gravity="@integer/notification_panel_layout_gravity"android:id="@+id/notification_container_parent"android:clipToPadding="false"android:clipChildren="false"><includelayout="@layout/keyguard_status_view"android:visibility="gone"/><include layout="@layout/dock_info_overlay" /><FrameLayoutandroid:id="@+id/qs_frame"android:layout="@layout/qs_panel"android:layout_width="@dimen/qs_panel_width"android:layout_height="0dp"android:clipToPadding="false"android:clipChildren="false"systemui:viewType="com.android.systemui.plugins.qs.QS"systemui:layout_constraintStart_toStartOf="parent"systemui:layout_constraintEnd_toEndOf="parent"systemui:layout_constraintTop_toTopOf="parent"systemui:layout_constraintBottom_toBottomOf="parent"/><androidx.constraintlayout.widget.Guidelineandroid:id="@+id/qs_edge_guideline"android:layout_width="wrap_content"android:layout_height="wrap_content"systemui:layout_constraintGuide_percent="0.5"android:orientation="vertical"/><com.android.systemui.statusbar.notification.stack.NotificationStackScrollLayoutandroid:id="@+id/notification_stack_scroller"android:layout_marginTop="@dimen/notification_panel_margin_top"android:layout_width="@dimen/notification_panel_width"android:layout_height="match_parent"android:layout_marginBottom="@dimen/close_handle_underlap"systemui:layout_constraintStart_toStartOf="parent"systemui:layout_constraintEnd_toEndOf="parent"/><include layout="@layout/ambient_indication"android:id="@+id/ambient_indication_container" /><include layout="@layout/photo_preview_overlay" /><includelayout="@layout/keyguard_status_bar"android:visibility="invisible" /><Buttonandroid:id="@+id/report_rejected_touch"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_marginTop="@dimen/status_bar_header_height_keyguard"android:text="@string/report_rejected_touch"android:visibility="gone" /><com.android.systemui.statusbar.phone.TapAgainViewandroid:id="@+id/shade_falsing_tap_again"android:layout_width="wrap_content"android:layout_height="wrap_content"systemui:layout_constraintLeft_toLeftOf="parent"systemui:layout_constraintRight_toRightOf="parent"systemui:layout_constraintBottom_toBottomOf="parent"android:layout_marginBottom="20dp"android:paddingHorizontal="16dp"android:minHeight="44dp"android:elevation="4dp"android:background="@drawable/rounded_bg_full"android:gravity="center"android:text="@string/tap_again"android:visibility="gone"/></com.android.systemui.statusbar.phone.NotificationsQuickSettingsContainer><FrameLayoutandroid:id="@+id/preview_container"android:layout_width="match_parent"android:layout_height="match_parent"></FrameLayout>
</com.android.systemui.statusbar.phone.NotificationPanelView>

在实现SystemUI下拉状态栏禁止QuickQSPanel展开的核心功能中,通过上述的分析得知,
从SystemUI中状态栏下拉的布局status_bar_expanded.xml的布局文件可以看到主要的下拉事件
是在NotificationPanelViewController.java中负责的,所以具体的下滑事件是在NotificationPanelViewController.java中处理的
所以接下来分析NotificationPanelViewController.java有关下拉状态栏的下滑展开事件

3.2 NotificationPanelViewController.java的下拉状态栏下滑事件分析

NotificationPanelViewController是一个以面板形式展示通知内容的主要视图控制器。它可以显示系统通知,
例如来自操作系统的警告、提醒或更新通知,以及来自其他应用程序的通知,如社交媒体应用或
即时消息应用发送的消息通知。通过notificationPanelViewController,用户可以方便地查看和管理收到的通知。

    public class NotificationPanelViewController extends PanelViewController { {@Overrideprotected TouchHandler createTouchHandler() {return new TouchHandler() {@Overridepublic boolean onInterceptTouchEvent(MotionEvent event) {if (mBlockTouches || mQsFullyExpanded && mQs.disallowPanelTouches()) {return false;}initDownStates(event);// Do not let touches go to shade or QS if the bouncer is visible,// but still let user swipe down to expand the panel, dismissing the bouncer.if (mStatusBar.isBouncerShowing()) {return true;}if (mBar.panelEnabled() && mHeadsUpTouchHelper.onInterceptTouchEvent(event)) {mMetricsLogger.count(COUNTER_PANEL_OPEN, 1);mMetricsLogger.count(COUNTER_PANEL_OPEN_PEEK, 1);return true;}if (!shouldQuickSettingsIntercept(mDownX, mDownY, 0)&& mPulseExpansionHandler.onInterceptTouchEvent(event)) {return true;}if (!isFullyCollapsed() && onQsIntercept(event)) {return true;}return super.onInterceptTouchEvent(event);}

在实现SystemUI下拉状态栏禁止QuickQSPanel展开的核心功能中,通过上述的分析得知,
在NotificationPanelViewController中的onInterceptTouchEvent(MotionEvent event) 手势下滑上滑事件的
处理,而具体处理下拉状态栏下滑事件的是在onQsIntercept(MotionEvent event)中继续处理的
接下来看下onQsIntercept(MotionEvent event)处理手势事件

 private boolean onQsIntercept(MotionEvent event) {int pointerIndex = event.findPointerIndex(mTrackingPointer);if (pointerIndex < 0) {pointerIndex = 0;mTrackingPointer = event.getPointerId(pointerIndex);}final float x = event.getX(pointerIndex);final float y = event.getY(pointerIndex);switch (event.getActionMasked()) {case MotionEvent.ACTION_DOWN:mIntercepting = true;mInitialTouchY = y;mInitialTouchX = x;initVelocityTracker();trackMovement(event);if (shouldQuickSettingsIntercept(mInitialTouchX, mInitialTouchY, 0)) {getParent().requestDisallowInterceptTouchEvent(true);}if (mQsExpansionAnimator != null) {onQsExpansionStarted();mInitialHeightOnTouch = mQsExpansionHeight;mQsTracking = true;mIntercepting = false;mNotificationStackScroller.cancelLongPress();}break;case MotionEvent.ACTION_POINTER_UP:final int upPointer = event.getPointerId(event.getActionIndex());if (mTrackingPointer == upPointer) {// gesture is ongoing, find a new pointer to trackfinal int newIndex = event.getPointerId(0) != upPointer ? 0 : 1;mTrackingPointer = event.getPointerId(newIndex);mInitialTouchX = event.getX(newIndex);mInitialTouchY = event.getY(newIndex);}break;case MotionEvent.ACTION_MOVE:final float h = y - mInitialTouchY;trackMovement(event);if (mQsTracking) {// Already tracking because onOverscrolled was called. We need to update here// so we don't stop for a frame until the next touch event gets handled in// onTouchEvent.setQsExpansion(h + mInitialHeightOnTouch);trackMovement(event);mIntercepting = false;return true;}if (Math.abs(h) > mTouchSlop && Math.abs(h) > Math.abs(x - mInitialTouchX)&& shouldQuickSettingsIntercept(mInitialTouchX, mInitialTouchY, h)) {mQsTracking = true;onQsExpansionStarted();notifyExpandingFinished();mInitialHeightOnTouch = mQsExpansionHeight;mInitialTouchY = y;mInitialTouchX = x;mIntercepting = false;mNotificationStackScroller.cancelLongPress();return true;}break;case MotionEvent.ACTION_CANCEL:case MotionEvent.ACTION_UP:trackMovement(event);if (mQsTracking) {flingQsWithCurrentVelocity(y,event.getActionMasked() == MotionEvent.ACTION_CANCEL);mQsTracking = false;}mIntercepting = false;break;}return false;}

在实现SystemUI下拉状态栏禁止QuickQSPanel展开的核心功能中,通过上述的分析得知,
在NotificationPanelViewController中onQsIntercept(MotionEvent event)的是具体处理下拉事件的方法
在case MotionEvent.ACTION_MOVE:中可以根据手势滑动的距离来设置下拉状态栏中的
QSPanel展开和收缩,通过setQsExpansion(h + mInitialHeightOnTouch);来根据手势滑动的
距离来设置QSPanel是否需要展开,所以具体看下setQsExpansion的相关代码

 private void setQsExpansion(float height) {// add core startif (mQs != null) return;//add core endheight = Math.min(Math.max(height, mQsMinExpansionHeight), mQsMaxExpansionHeight);mQsFullyExpanded = height == mQsMaxExpansionHeight && mQsMaxExpansionHeight != 0;if (height > mQsMinExpansionHeight && !mQsExpanded && !mStackScrollerOverscrolling) {setQsExpanded(true);} else if (height <= mQsMinExpansionHeight && mQsExpanded) {setQsExpanded(false);}mQsExpansionHeight = height;updateQsExpansion();requestScrollerTopPaddingUpdate(false /* animate */);updateHeaderKeyguardAlpha();if (mBarState == StatusBarState.SHADE_LOCKED|| mBarState == StatusBarState.KEYGUARD) {updateKeyguardBottomAreaAlpha();updateBigClockAlpha();}if (mBarState == StatusBarState.SHADE && mQsExpanded&& !mStackScrollerOverscrolling && mQsScrimEnabled) {mQsNavbarScrim.setAlpha(getQsExpansionFraction());}if (mAccessibilityManager.isEnabled()) {setAccessibilityPaneTitle(determineAccessibilityPaneTitle());}if (!mFalsingManager.isUnlockingDisabled() && mQsFullyExpanded&& mFalsingManager.shouldEnforceBouncer()) {mStatusBar.executeRunnableDismissingKeyguard(null, null /* cancelAction */,false /* dismissShade */, true /* afterKeyguardGone */, false /* deferred */);}if (mExpansionListener != null) {mExpansionListener.onQsExpansionChanged(mQsMaxExpansionHeight != 0? mQsExpansionHeight / mQsMaxExpansionHeight : 0);}if (DEBUG) {invalidate();}}

4、修改

在实现SystemUI下拉状态栏禁止QuickQSPanel展开的核心功能中,通过上述的分析得知,
在NotificationPanelViewController.java类中setQsExpansion(float height) 方法来设置下拉状态栏
QSPanel高度的方法中,根据高度判断是否需要展开收缩QSPanel,所以通过增加判断

if (mQs != null)return;

来去掉下滑展开QSPanel,就可以做到只显示QuickQSPanel,而
不用展开QSPanel,实现单次下拉状态栏,定制UI 就可以在QuickQSPanel中定制就可以了

 private void setQsExpansion(float height) {// mQs  不等于空直接复用if (mQs != null) return;// mQs 等于空新建height = Math.min(Math.max(height, mQsMinExpansionHeight), mQsMaxExpansionHeight);mQsFullyExpanded = height == mQsMaxExpansionHeight && mQsMaxExpansionHeight != 0;if (height > mQsMinExpansionHeight && !mQsExpanded && !mStackScrollerOverscrolling&& !mDozing) {setQsExpanded(true);} else if (height <= mQsMinExpansionHeight && mQsExpanded) {setQsExpanded(false);}mQsExpansionHeight = height;updateQsExpansion();requestScrollerTopPaddingUpdate(false /* animate */);updateHeaderKeyguardAlpha();if (mBarState == StatusBarState.SHADE_LOCKED || mBarState == KEYGUARD) {updateKeyguardBottomAreaAlpha();positionClockAndNotifications();updateBigClockAlpha();}if (mAccessibilityManager.isEnabled()) {mView.setAccessibilityPaneTitle(determineAccessibilityPaneTitle());}if (!mFalsingManager.isUnlockingDisabled() && mQsFullyExpanded&& mFalsingCollector.shouldEnforceBouncer()) {mStatusBar.executeRunnableDismissingKeyguard(null, null /* cancelAction */,false /* dismissShade */, true /* afterKeyguardGone */, false /* deferred */);}for (int i = 0; i < mExpansionListeners.size(); i++) {mExpansionListeners.get(i).onQsExpansionChanged(mQsMaxExpansionHeight != 0 ? mQsExpansionHeight / mQsMaxExpansionHeight : 0);}if (DEBUG) {mView.invalidate();}}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/53993.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

PHP一键约课高效健身智能健身管理系统小程序源码

一键约课&#xff0c;高效健身 —— 智能健身管理系统让健康触手可及 &#x1f3cb;️‍♀️ 告别繁琐&#xff0c;一键开启健身之旅 你还在为每次去健身房前的繁琐预约流程而烦恼吗&#xff1f;现在有了“一键约课高效健身智能健身管理系统”&#xff0c;所有问题都迎刃而解…

智能体-AI-Agent-简介

文章目录 一&#xff0c;什么是AI Agent二&#xff0c;扣子个人空间团队空间探索区 一&#xff0c;什么是AI Agent AI智能体并没有什么特别&#xff0c;本质上就是一个帮助你解决工作和学习中的一个工具。 很多自媒体把智能体描述的天花乱坠&#xff0c;那不过是他们畅想的智…

Spring Security认证与授权

1 Spring Security介绍 Spring Security是一个能够为基于Spring的企业应用系统提供声明式的安全访问控制解决方案的安全框架。由于它是Spring生态系统中的一员&#xff0c;因此它伴随着整个Spring生态系统不断修正、升级&#xff0c;在spring boot项目中加入springsecurity更是…

Vue的学习(三)

目录 一、for循环中key的作用 1‌.提高性能‌&#xff1a; ‌2.优化用户体验‌&#xff1a; ‌3.辅助Vue进行列表渲染‌&#xff1a; 4‌.方便可复用组件的使用‌&#xff1a; 二、methods及computed及wacth的区别 三、过滤器 1.Vue 2 过滤器简介 定义过滤器 使用过滤…

用 Swift 写 Android App ?来了解下 Skip 原生级跨平台框架

最近在找资料的时候&#xff0c;机缘巧合发现了一个有趣的商业跨平台框架 Skip &#xff0c;刚好看到了它发布 1.0 正式版&#xff0c;主要作用是将 Swift 开发引入到 Android 领域&#xff0c;这样 App 就可以共享 Swift 的业务逻辑&#xff0c;在 SwiftUI 中完成 Android App…

Python | Leetcode Python题解之第395题至少有K个重复字符的最长子串

题目&#xff1a; 题解&#xff1a; class Solution:def longestSubstring(self, s1: str, k: int) -> int:if k 1: return len(s1)n len(s1)res 0for c in range(1, len(set(s1)) 1):# 滑窗中字母种类个数恰好为 cfreq Counter()l cnt tcnt 0 for r, ch in enu…

代码随想录训练营Day3 | 链表理论基础 | 203.移除链表元素 | 707.设计链表 | 206.反转链表

今天任务&#xff1a;学习链表理论基础 链表的类型 链表的存储方式 链表的定义…

开发一款通过蓝牙连接控制水电表的微信小程序

增强软硬件交互 为了更好的解决师生生活中的实际问题&#xff0c;开发蓝牙小程序加强了和校区硬件的交互。 比如通过蓝牙连接控制水电表&#xff0c;减少实体卡片的使用。添加人脸活体检测功能&#xff0c;提高本人认证效率&#xff0c;减少师生等待时间。 蓝牙水电控展示 蓝…

HashMap常用方法及底层原理

目录 一、什么是HashMap二、HashMap的链表与红黑树1、数据结构2、链表转为红黑树3、红黑树退化为链表 三、存储&#xff08;put&#xff09;操作四、读取&#xff08;get&#xff09;操作五、扩容&#xff08;resize&#xff09;操作六、HashMap的线程安全与顺序1、线程安全2、…

【LeetCode每日一题】2024年9月第二周(上)

2024.9.9 中等 难度评分 1333 链接&#xff1a;2181. 合并零之间的节点 &#xff08;1&#xff09;题目描述&#xff1a; &#xff08;2&#xff09;示例 &#xff08;3&#xff09;分析 整体来说&#xff0c;描述还算清晰的题目&#xff0c;找到0节点所框定的区域&#xff0c…

Pandas读取某列、某行数据——loc、iloc区别

loc&#xff1a;通过行、列的名称或标签来索引 iloc&#xff1a;通过行、列的索引位置来寻找数据 首先&#xff0c;我们先创建一个DataFrame生成数据 import pandas as pddata {a:[1,2,3,4,5],b:[6,7,8,9,10],c:[11,12,13,14,15] } data pd.DataFrame(data) print(data) 运行…

工具、环境等其他小问题归纳

此篇文章内容会不定期更新&#xff0c;仅作为学习过程中的笔记记录 一、查询Windows 10环境下python版本与安装路径 若电脑成功安装了python环境&#xff0c;不小心忘了版本。 I、查询版本 1、cmd窗口快捷查询 Win R 输入cmd 进入窗口&#xff1b; 直接输入 python --version …

[数据集][目标检测]血细胞检测数据集VOC+YOLO格式2757张4类别

数据集格式&#xff1a;Pascal VOC格式YOLO格式(不包含分割路径的txt文件&#xff0c;仅仅包含jpg图片以及对应的VOC格式xml文件和yolo格式txt文件) 图片数量(jpg文件个数)&#xff1a;2757 标注数量(xml文件个数)&#xff1a;2757 标注数量(txt文件个数)&#xff1a;2757 标注…

关于武汉芯景科技有限公司的IIC电平转换芯片XJ9517开发指南(兼容PCF9517)

一、芯片引脚介绍 1.芯片引脚 2.引脚描述 二、系统结构图 三、功能描述 1.电平转换 2.芯片使能/失能 EN 引脚为高电平有效&#xff0c;内部上拉至 VCC&#xff08;B&#xff09;&#xff0c;允许用户选择中继器何时有效。这可用于在上电时隔离行为不良的从机&#xff0c;直到…

4052A/4052B/4052C/4052D/4052E/4052F/4052G /4052H信号/频谱分析仪

4052A/4052B/4052C/4052D/4052E/4052F/4052G /4052H信号/频谱分析仪 苏州新利通 Ceyear 4052具备出色的测试动态范围、相位噪声、幅度精度和测试速度&#xff0c;具备频谱分析、I/Q分析、实时频谱分析、瞬态分析、矢量信号分析、脉冲分析、音频分析等丰富的测试功能。 Ceyear…

OpenAI发布o1预览模型:推理能力更强可达理科博士生水准

近日OpenAI宣布推出了新一代 AI 模型系列 OpenAI o1&#xff0c;按照官方技术博客说法&#xff0c;o1 在推理能力上代表了人工智能最强的水平。 那究竟是怎么一回事呢&#xff1f; OpenAI CEO Sam Altman 表示&#xff1a;o1 系列的推出代表了 AI 能力的新起点&#xff0c;能…

240909-ChuanhuChatGPT集成Ollama的环境配置

A. 最终效果 B. 需求文件 requirements.txt (至少需要安装这个&#xff0c;具体参见官网)requirements_advanced.txt &#xff08;如果安装了Ollama&#xff0c;并且可以进行对话&#xff0c;可以不需要安装&#xff0c;具体参见官网&#xff09;requirements_succcess.txt&am…

gin配置swagger文档

一、基本准备工作 1、安装依赖包 go get -u github.com/swaggo/swag/cmd/swag go get -u github.com/swaggo/gin-swagger go get -u github.com/swaggo/files2、在根目录上配置swagger的路由文件 //2.初始化路由router : initialize.Routers()// 配置swaggerdocs.SwaggerInfo…

微服务杂谈

几个概念 还是第一次听说Spring Cloud Alibaba &#xff0c;真是孤陋寡闻了&#xff0c;以前只知道 SpringCloud 是为了搭建微服务的&#xff0c;spring boot 则是快速创建一个项目&#xff0c;也可以是一个微服务 。那么SpringCloud 和 Spring boot 有什么区别呢&#xff1f;S…

Unity for Android使用蓝牙低功耗Bluetooth LE

Unity2021.3.35f1 插件&#xff1a;Bluetooth LE for iOS and Android v2.3.unitypackage 1、将插件资源包导入unity中 2.修改插件中的AndroidManifest文件 <?xml version"1.0" encoding"utf-8"?> <manifest xmlns:android"http://schem…