Android设置界面层级为最上层实现
Android设置界面层级为最上层实现
文章目录
- Android设置界面层级为最上层实现
- 一、前言
- 二、Android设置界面层级为最上层实现
- 1、主要代码
- 2、后遗症
- 三、其他
- 1、Android设置界面层级为最上层小结
- 2、悬浮框的主要代码
- 悬浮框 注意事项
- (1)权限限制
- (2)悬浮框窗口类型
- (3)悬浮框窗口管理
- Activity添加悬浮框后关闭Activity的情况,没有Activity,没有Service,这个应用是存活的,为啥呢?
- 关键窗口类型对比:
- (4)悬浮框的正常使用
- (5)查看当前顶层界面信息
- (6)查看当前所有窗口
- 3、Android对话框的使用总结
- 4、Android 几个简单的自定义对话框介绍
一、前言
Android设置界面层级为最上层怎么实现?
正常情况设置悬浮框就可以了,悬浮框默认是最上层的。
但是Java还有个地方的代码可以设置界面再任何界面之上;
具体实现逻辑没有去分析,这个只是其他人这样分析实现了,我顺便记录一下。
真正用到的场景估计比较少,有些系统应用的唯一悬浮框界面估计会需要用到;
最后也介绍到了一些悬浮框相关的小众知识,有兴趣的可以看看。
二、Android设置界面层级为最上层实现
1、主要代码
frameworks\base\services\core\java\com\android\server\wm\Task.java
@Overridepublic boolean isAlwaysOnTop() {boolean isinstallCts = SystemProperties.getBoolean("persist.skg.isinstall.cts", false);if (!isinstallCts) {// debug settings return trueString pk = getBasePackageName();if (pk.equals("com.debug.dialog") || pk.equals("com.android.settings") ) {return true;}}return !isForceHidden() && super.isAlwaysOnTop();}
修改上面代码后就可以实现top的包名应用会出现在最上层;
即使悬浮框和对话框都是显示在它的下层。
如果两个应用同时是top,后面拉起的一个会显示在最上层。
实现原理这里不进行分析了,因为我没研究过!
2、后遗症
按照上面的修改会有点问题:
如果只是悬浮框显示在最上层,并且可以移动,交互上是没啥问题的;
如果最上层的是Activity 界面,按Home和后台任务无效,未添加堆栈前返回按键无法关闭界面;
因为Home之后的Launcher界面的优先级也比它低;
后台任务的界面优先级也比它低。
所以只有是悬浮框的应用或特定的系统界面可以采用这种方式设置方式设置界面层级最高级;
其他情况慎用。
三、其他
1、Android设置界面层级为最上层小结
本文是强制设置某个应用的界面为最顶层。一般不建议这样修改。
简单的理解每个界面显示系统都会判断它的级别,随后排序显示显示。
普通悬浮于应用的场景,使用系统悬浮框就可以了。
大致优先级:top>悬浮框>对话框>Activity.
如果是锁屏界面呢?在top上面还是下面?
试了一下,锁屏界面是在top应用界面上面的。锁屏界面的层级不清楚怎么样的。
2、悬浮框的主要代码
// 获取 WindowManager 服务WindowManager windowManager = (WindowManager) getSystemService(WINDOW_SERVICE);// 设置 WindowManager 布局参数WindowManager.LayoutParams params = new WindowManager.LayoutParams(WindowManager.LayoutParams.WRAP_CONTENT,WindowManager.LayoutParams.WRAP_CONTENT,WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY, //悬浮框类别WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE,android.graphics.PixelFormat.TRANSLUCENT);//floatingView 是某个layout对象,不能是普通的view,比如TextViewwindowManager.addView(floatingView, params);//移除它
windowManager.removeView(floatingLayout)
floatingView 示例:
//正确示例:
LinearLayout floatingView = (LinearLayout) View.inflate(this, R.layout.activity_main, null);//错误示例(传入这个floatingView会报错):
LinearLayout floatingView = findViewById(R.id.activity_demo);
悬浮框 注意事项
(1)权限限制
从 Android 6.0 (API 23) 开始,必须通过 Settings.ACTION_MANAGE_OVERLAY_PERMISSION
显式请求悬浮窗权限。
部分厂商定制系统可能进一步限制此权限。
(2)悬浮框窗口类型
-
Android 8.0 (API 26) 及以上版本必须使用
TYPE_APPLICATION_OVERLAY
类型。 -
旧版本使用
TYPE_PHONE
或TYPE_SYSTEM_ALERT
。 -
TYPE_SYSTEM_ALERT 在新版本也能使用,但是必须是系统签名权限应用。
-
系统系统应用不常用窗口类别:
这些类型用于系统组件,普通应用通常无法使用: TYPE_STATUS_BAR 描述:状态栏窗口。 TYPE_NAVIGATION_BAR 描述:导航栏窗口。 TYPE_INPUT_METHOD 描述:输入法窗口。 TYPE_KEYGUARD 描述:锁屏界面窗口。 TYPE_ACCESSIBILITY_OVERLAY 描述:无障碍服务覆盖窗口。 TYPE_DREAM 描述:屏保窗口(Doze 模式)。 TYPE_VOICE_INTERACTION 描述:语音交互窗口(如 Google Assistant)。
-
窗口类型的层级关系(从低到高):
应用界面 < TYPE_APPLICATION_OVERLAY < TYPE_SYSTEM_ALERT < 状态栏 < 输入法 < 系统错误窗口
上面的窗口类别了解一下就可以了,有些窗口类别是需要再特定条件下才能使用的。
(3)悬浮框窗口管理
WindowManager 添加的悬浮框窗口,后面的窗口不属于应用了?
如果是demo应用的Activity添加的悬浮框,Activity界面退出后,悬浮框窗口还是会显示在系统界面上;
那么这个时候demo应用还存活吗?
使用命令:“ps -ef | grep XXX包名” 查看了一下,应用进程是在的;
即使使用后台进程列表,去除该应用,发现应用进程还在,悬浮窗口还是存在;
这种情况应该是系统清除进程过程判断当前应用有关联的窗口,所以没有kill应用进程;
使用命令 “ am force-stop XXX包名”,kill进程后,悬浮框确实关闭了。
所以说悬浮框窗口显示的前提必须要原本拉起悬浮窗口的应用必须存活。
Activity添加悬浮框后关闭Activity的情况,没有Activity,没有Service,这个应用是存活的,为啥呢?
这个和系统进程策略有关,有些系统列表的窗口就是可以不依赖Activity存在。
为啥没有Activity也能存活,具体是哪个类哪个逻辑的处理,我也不清楚!
关键窗口类型对比:
窗口类型 | 所属层级 | 是否允许后台存在(无 Activity/Service) |
---|---|---|
TYPE_APPLICATION | 应用层级 | 不允许:依赖 Activity 存在 |
TYPE_APPLICATION_OVERLAY | 应用覆盖层级 | 允许:可在后台独立存在(需权限) |
TYPE_SYSTEM_ALERT | 系统层级 | 允许:系统级窗口,可后台存在 |
TYPE_APPLICATION 类别的窗口,和对话框是一样类型的,不太算悬浮框了。
(4)悬浮框的正常使用
正常情况下,悬浮框的使用都是在Service拉起的,并且可以在Service实时监听数据变化更新界面显示。
(5)查看当前顶层界面信息
//普通应用正常运行,查看当前顶层界面信息
console:/ # dumpsys window | grep mFocmFocusedApp=ActivityRecord{3609e53 u0 com.example.an15/.MainActivity t66}mFocusedWindow=Window{4e5b6fa u0 com.example.an15/com.example.an15.MainActivity}
console:/ #
console:/ #
//Activity退出,悬浮框界面显示的时候
console:/ # dumpsys window | grep mFocmFocusedApp=ActivityRecord{85383ad u0 com.android.launcher3/.uioverrides.QuickstepLauncher t63}mFocusedWindow=Window{c1dff8 u0 com.android.launcher3/com.android.launcher3.uioverrides.QuickstepLauncher}
可以看到显示悬浮框界面的时候,当前顶层界面是悬浮框背后的界面,
如果是桌面应用,就是Launcher界面了。
(6)查看当前所有窗口
查看当前所有窗口命令:dumpsys window windows | grep “Window{”
这个命令能看到:当前顶层窗口,悬浮窗口,系统其他窗口信息等。
示例如下:
//*************显示悬浮窗,并且未关闭demo应用的Activity前
console:/ #
console:/ # dumpsys window windows | grep "Window{" Window #0 Window{6c3345b u0 ScreenDecorOverlayBottom}:...//添加的悬浮窗口信息Window #5 Window{f3d263 u0 Sys2038:com.example.an15/com.example.an15.MainActivity}:Window #6 Window{e3472f4 u0 ShellDropTarget}:Window #7 Window{d2f67df u0 InputMethod}://当前的顶层界面信息,demo应用Window #8 Window{365bc0c u0 com.example.an15/com.example.an15.MainActivity}://其他界面信息,比如Launcher 、后台未关闭的窗口Window #9 Window{8b073fa u0 com.android.launcher3/com.android.launcher3.uioverrides.QuickstepLauncher}:Window #10 Window{c4d9262 u0 com.android.systemui.wallpapers.ImageWallpaper}:
...
console:/ # //***********显示悬浮窗,并且已关闭demo应用的Activity后,显示的是Launcher桌面
console:/ #
console:/ # dumpsys window windows | grep "Window{" Window #0 Window{6c3345b u0 ScreenDecorOverlayBottom}://添加的悬浮窗口信息Window #5 Window{f3d263 u0 Sys2038:com.example.an15/com.example.an15.MainActivity}:Window #6 Window{e3472f4 u0 ShellDropTarget}:Window #7 Window{d2f67df u0 InputMethod}://当前的顶层界面信息,Launcher应用Window #8 Window{8b073fa u0 com.android.launcher3/com.android.launcher3.uioverrides.QuickstepLauncher}://其他信息界面Window #9 Window{c4d9262 u0 com.android.systemui.wallpapers.ImageWallpaper}:imeLayeringTarget in display# 0 Window{8b073fa u0 com.android.launcher3/com.android.launcher3.uioverrides.QuickstepLauncher}imeInputTarget in display# 0 Window{8b073fa u0 com.android.launcher3/com.android.launcher3.uioverrides.QuickstepLauncher}
console:/ #
从window窗口信息,可以大致看到,window的次序信息和界面显示的是大致一致的;
前面先是显示悬浮窗口,后面显示应用窗口;
应用的悬浮窗口信息比较明显,包含了包名,其他的悬浮窗口,是一些系统的悬浮窗口信息;
悬浮窗口后面的信息,就到普通应用的Activity界面了,
第一个非悬浮窗口界面就是当前的顶层界面,后面的界面就是后台存在的界面了;
应用界面打开后,按Home返回,所有后台应用界面都会显示在上面;
不排除系统判断当前应用进程太多的情况,会杀死一些应用界面。
以前看这个信息也没注意这么多,现在看看dumpsys这些window信息对界面分析还是比较有用的。
3、Android对话框的使用总结
(一)警告对话框 AlertDialog: 一个可以有0到3个按钮, 一个单选框或复选框的列表的对话框. 警告对话框可以创建大多数的交互界面, 是推荐的类型.
(二)进度对话框 ProgressDialog: 显示一个进度环或者一个进度条. 由于它是AlertDialog的扩展, 所以它也支持按钮.
(三)日期选择对话框 DatePickerDialog: 让用户选择一个日期.
(四)时间选择对话框 TimePickerDialog: 让用户选择一个时间
原文链接:https://blog.csdn.net/wenzhi20102321/article/details/52818351
4、Android 几个简单的自定义对话框介绍
Android 对话框直接使用肯定是不好看的,如果需要添加取消和确定按钮,肯定是需要进行自定义修改的。
简单介绍一个简单的可扩展的对话框,并且支持不同类型样式,加了点小动画。
https://blog.csdn.net/wenzhi20102321/article/details/139040977
相关文章:
Android设置界面层级为最上层实现
Android设置界面层级为最上层实现 文章目录 Android设置界面层级为最上层实现一、前言二、Android设置界面层级为最上层实现1、主要代码2、后遗症 三、其他1、Android设置界面层级为最上层小结2、悬浮框的主要代码悬浮框 注意事项(1)权限限制(…...
云原生微服务架构演进之路:理念、挑战与实践
📝个人主页🌹:慌ZHANG-CSDN博客 🌹🌹期待您的关注 🌹🌹 一、引言:架构的演进是业务进化的技术反射 在软件行业的发展过程中,架构变迁总是伴随着技术浪潮与业务复杂度的升…...
Go语言使用阿里云模版短信服务
在当今的互联网项目中,短信验证码、通知等功能已成为标配。本文将详细介绍如何使用Go语言集成阿里云短信服务(DYSMSAPI)实现短信发送功能。 一、准备工作 在开始之前,您需要完成以下准备工作: 注册阿里云账号并实名认证开通短信服务(SMS)申…...

Leetcode 3231. 要删除的递增子序列的最小数量
1.题目基本信息 1.1.题目描述 给定一个整数数组 nums,你可以执行任意次下面的操作: 从数组删除一个 严格递增 的 子序列。 您的任务是找到使数组为 空 所需的 最小 操作数。 1.2.题目地址 https://leetcode.cn/problems/minimum-number-of-increas…...

4.2.5 Spark SQL 分区自动推断
在本节实战中,我们学习了Spark SQL的分区自动推断功能,这是一种提升查询性能的有效手段。通过创建具有不同分区的目录结构,并在这些目录中放置JSON文件,我们模拟了一个分区表的环境。使用Spark SQL读取这些数据时,Spar…...
基于昇腾MindSpeed训练加速库玩转智谱GLM-4-0414模型
智谱GLM-4-0414模型提供32B和9B两种参数规模,涵盖基础、推理和沉思等多种模型类型,均基于 MIT 许可协议开放。其中,推理模型 GLM-Z1-32B-0414 性能卓越,与 DeepSeek-R1 等领先模型相当,实测推理速度达每秒200个Tokens。…...

【图像处理入门】2. Python中OpenCV与Matplotlib的图像操作指南
一、环境准备 import cv2 import numpy as np import matplotlib.pyplot as plt# 配置中文字体显示(可选) plt.rcParams[font.sans-serif] [SimHei] plt.rcParams[axes.unicode_minus] False二、图像的基本操作 1. 图像读取、显示与保存 使用OpenCV…...

Spring Boot微服务架构(九):设计哲学是什么?
一、Spring Boot设计哲学是什么? Spring Boot 的设计哲学可以概括为 “约定优于配置” 和 “开箱即用”,其核心目标是极大地简化基于 Spring 框架的生产级应用的初始搭建和开发过程,让开发者能够快速启动并运行项目…...
GRCh38版本染色体位置转换GRCh37(hg19)
目录 方法 1:使用 Ensembl REST API(推荐,适用于少量位点查询)方法 2:使用 UCSC API方法 3:使用 NCBI API 并转换坐标(需要额外步骤)方法 4:使用本地数据库(最…...

TC/BC/OC P2P/E2E有啥区别?-PTP协议基础概念介绍
前言 时间同步网络中的每个节点,都被称为时钟,PTP协议定义了三种基本时钟节点。本文将介绍这三种类型的时钟,以及gPTP在同步机制上与其他机制的区别 本系列文章将由浅入深的带你了解gPTP,欢迎关注 时钟类型 在PTP中我们将各节…...
解决微信小程序中 Flex 布局下 margin-right 不生效的问题
解决微信小程序中 Flex 布局下 margin-right 不生效的问题 在做微信小程序开发时,遇到了一个棘手的布局问题:在 flex 布局下,给元素设置的 margin-right 不生效,被“吞噬”了。这个问题导致了横向滚动列表的右边距失效࿰…...

Kafka数据怎么保障不丢失
在分布式消息系统中,数据不丢失是核心可靠性需求之一。Apache Kafka 通过生产者配置、副本机制、持久化策略、消费者偏移量管理等多层机制保障数据可靠性。以下从不同维度解析 Kafka 数据不丢失的核心策略,并附示意图辅助理解。 一、生产者端:…...
使用HTTPS进行传输加密
说明 日期:2025年5月30日 与以纯文本形式发送和接收消息的标准 HTTP 不同,HTTPS 使用SSL/TLS等协议对服务器进行身份验证、加密通信内容和检测篡改。 这样可以防止欺骗、中间人攻击和窃听等攻击。 证书很重要,如果用户主动信任了伪造证书&…...

AI书签管理工具开发全记录(八):Ai创建书签功能实现
文章目录 AI书签管理工具开发全记录(八):AI智能创建书签功能深度解析前言 📝1. AI功能设计思路 🧠1.1 传统书签创建的痛点1.2 AI解决方案设计 2. 后端API实现 ⚙️2.1 新增url相关工具方法2.1 创建后端api2.2 创建crea…...

X-plore v4.43.05 强大的安卓文件管理器-MOD解锁高级版 手机平板/电视TV通用
X-plore v4.43.05 强大的安卓文件管理器-MOD解锁高级版 手机平板/电视TV通用 应用简介: X-plore 是一款强大的安卓端文件管理器,它可以在电视或者手机上管理大量媒体文件、应用程序。…...

使用多Agent进行海报生成的技术方案及评估套件-P2P、paper2poster
最近字节、滑铁卢大学相关团队同时放出了他们使用Agent进行海报生成的技术方案,P2P和Paper2Poster,传统方案如类似ppt生成等思路,基本上采用固定的模版,提取相关的关键元素进行模版填充,因此,海报生成的质量…...

Redis--缓存工具封装
经过前面的学习,发现缓存中的问题,无论是缓存穿透,缓存雪崩,还是缓存击穿,这些问题的解决方案业务代码逻辑都很复杂,我们也不应该每次都来重写这些逻辑,我们可以将其封装成工具。而在封装的时候…...

python:在 PyMOL 中如何查看和使用内置示例文件?
参阅:开源版PyMol安装保姆级教程 百度网盘下载 提取码:csub pip show pymol 简介: PyMOL是一个Python增强的分子图形工具。它擅长蛋白质、小分子、密度、表面和轨迹的3D可视化。它还包括分子编辑、射线追踪和动画。 可视化示例:打开 PyM…...

SpringCloud——Docker
1.命令解读 docker run -d 解释:创建并运行一个容器,-d则是让容器以后台进程运行 --name mysql 解释: 给容器起个名字叫mysql -p 3306:3306 解释:-p 宿主机端口:容器内端口,设置端口映射 注意: 1、…...

机器学习:欠拟合、过拟合、正则化
本文目录: 一、欠拟合二、过拟合三、拟合问题原因及解决办法四、正则化:尽量减少高次幂特征的影响(一)L1正则化(二)L2正则化(三)L1正则化与L2正则化的对比 五、正好拟合代码…...

运用集合知识做斗地主案例
方法中可变参数 一种特殊形参,定义在方法,构造器的形参列表里,格式:数据类型...参数名称; 可变参数的特点和好处 特点:可以不传数据给它;可以传一个或者同时传多个数据给它;也可以…...

《HelloGitHub》第 110 期
兴趣是最好的老师,HelloGitHub 让你对开源感兴趣! 简介 HelloGitHub 分享 GitHub 上有趣、入门级的开源项目。 github.com/521xueweihan/HelloGitHub 这里有实战项目、入门教程、黑科技、开源书籍、大厂开源项目等,涵盖多种编程语言 Python、…...

使用 Shell 脚本实现 Spring Boot 项目自动化部署到 Docker(Ubuntu 服务器)
使用 Shell 脚本实现 Spring Boot 项目自动化部署到 Docker(Ubuntu 服务器) 在日常项目开发中,我们经常会将 Spring Boot 项目打包并部署到服务器上的 Docker 环境中。为了提升效率、减少重复操作,我们可以通过 Shell 脚本实现自动…...

day023-网络基础与OSI七层模型
文章目录 1. 网络基础知识点1.1 网络中的单位1.2 查看实时网速:iftop1.3 交换机、路由器 2. 路由表2.1 查看路由表的命令2.2 路由追踪命令 3. 通用网站网络架构4. 局域网上网原理-NAT5. 虚拟机上网原理6. 虚拟机的网络模式6.1 NAT模式6.2 桥接模式6.3 仅主机模式 7.…...

SpringAI系列4: Tool Calling 工具调用 【感觉这版本有bug】
前言:在最近发布的 Spring AI 1.0.0.M6 版本中,其中一个重大变化是 Function Calling 被废弃,被 Tool Calling 取代。Tool Calling工具调用(也称为函数调用)是AI应用中的常见模式,允许模型通过一组API或工具…...

机器人--里程计
教程 轮式里程计视频讲解 里程计分类 ros--odometry 什么是里程计 里程计是一种利用从移动传感器获得的数据来估计物体位置随时间的变化而改变的方法。该方法被用在许多机器人系统来估计机器人相对于初始位置移动的距离。 注意:里程计是一套算法,不…...

设计模式——原型设计模式(创建型)
摘要 本文详细介绍了原型设计模式,这是一种创建型设计模式,通过复制现有对象(原型)来创建新对象,避免使用new关键字,可提高性能并简化对象创建逻辑。文章阐述了其优点,如提高性能、动态扩展和简…...
react库:class-variance-authority
文章目录 前言一、cva 的核心作用二、代码逐层解析参数详解基础样式(第一个参数):variant:定义颜色/风格变体(如 default、destructive)。size:定义尺寸变体(如 sm、lg)。…...

通过mqtt 点灯
1 解析mqtt 传过来的json 用cjson 解析。 2 类似mvc的结构,调用具体的动作函数 定义设备处理结构体:使用结构体数组映射设备名称与处理函数,实现可扩展的指令分发分离设备逻辑:为每个设备(如 LED、Motor࿰…...
随笔笔记记录5.28
1.setOptMode -opt_leakage_to_dynamic_ratio 调整漏电与动态功耗的优化权重( 1.0 表示仅优化漏电)。 需指定-opt_power_effort(none | low | high),同时使用 2.set_ccopt_property max_source_to_sink_net_length …...