如何用正确的姿势监听Android屏幕旋转
作者:37手游移动客户端团队
背景
关于个人,前段时间由于业务太忙,所以一直没有来得及思考并且沉淀点东西;同时组内一个个都在业务上能有自己的思考和总结,在这样的氛围下,不由自主的驱使周末开始写点东西,希望自己除了日常忙于业务,可以沉淀点东西,加上自己的成长…
关于切入点,最近在做应⽤内悬浮球功能时,需要监听屏幕旋转事件来对悬浮球的位置进⾏调整,发现有些情况下并不能收到系统回调,思考了⼀翻,做了⼀个屏幕旋转的模拟监听,基本上能达到⽬的。
问题
悬浮球在停⽌拖拽后,需要贴边到⼿机屏幕的左右两侧。
在竖屏状态下,x坐标为0即为左边缘,x坐 标为屏幕宽度即为右边缘。
但是在横屏状态下,情况就⽐较复杂了。现在⼤部分Android⼿机都是刘 海屏的设计,在全屏状态下,悬浮球贴边时不能收到刘海下⾯去,不然就点不到了。
所以此时需要算 出刘海的宽度,以此宽度作为悬浮球左边的起始位置,这样悬浮球贴边的时候就不会躲到刘海下⾯ 去。 如下图所示

但是在屏幕旋转之后,刘海到了右边,左边就不应该以刘海的宽度作为悬浮球的起点了。 这样的话就需要监听屏幕的旋转了,配合屏幕⽅向的⻆度,就能正确判断。监听屏幕的旋转只需要重 写Activity的onConfiguratuonChanged⽣命周期。
override fun onConfigurationChanged(newConfig: Configuration) {super.onConfigurationChanged(newConfig)Log.i(TAG, "on configuration changed")
}
在AndroidManifest中配置
android:configChanges="orientation|screenSize"
此时发现了⼀个问题,当把Activity的screenOrientation设置成sensorLandscape时,即使屏幕旋转 也收不到这个回调(这个和之前的理解有点不⼀样)。于是将screenOrientation设置成sensor,屏 幕旋转就能正常回调到这⾥,多试⼏次发现,只有在横屏和竖屏之间切换时才能收到回调,如果直接 将横屏倒过来,就是横屏状态不变,⽅向调转,此时也不会收到回调。
解决思路
既然onConfigurationChanged收不到回调,还有另外⼀个办法,就是监听屏幕⽅向度数,代码如下
mOrientationEventListener = object : OrientationEventListener(this) {override fun onOrientationChanged(orientation: Int) {Log.i(TAG, "on orientation changed angle is $orientation")if (orientation > 340 || orientation < 20) {//0} else if (orientation in 71..109) {//90} else if (orientation in 161..199) {//180} else if (orientation in 251..289) {//270}}
}
通过度数来判断刘海是在左边还是在右边,即270度时在左边,90度时在右边。这种⽅式看起来可以 解决问题,但是多旋转⼏次就发现⼜有其他问题。按照正常思维,屏幕的显示⽅向应该和这个度数⼀ 致才对,即屏幕的显示应该是⾃上⽽下的。但是下图就不是这样。

此时度数为90,屏幕却倒⽴着显示的,并没有旋转成正⽴状态,但是按照上⾯的代码,会将90度判定 为正常90度正⽴显示的状态,此时去修改悬浮球的位置就是错误的。
那如果在收到onOrientationChanged这个回调时能判断⼀下屏幕显示的⽅向呢,就是在度数达到90 度范围时,同时判断屏幕的显示⽅向,即两个条件同时满⾜才判定成屏幕旋转了。
⽤下⾯的代码判定屏幕显示⽅向
val windowManager = context.getSystemService(Context.WINDOW_SERVICE) as
WindowManager
val rotation = windowManager.defaultDisplay?.rotation
//rotation为常量0、1、2、3,分别表示屏幕的四个⽅向
通过这样的判断基本上能将屏幕旋转事件监听准确了,onOrientationChanged这个回调很灵敏,⼿ 机屏幕稍微动⼀下就会回调。那我希望模拟正常的屏幕旋转事件来修改悬浮球的位置,总不能很频繁 的刷新吧。这⾥做⼀下控制就好,全部代码如下:
object ScreenOrientationHelper {val ORIENTATION_TYPE_0 = 0val ORIENTATION_TYPE_90 = 90val ORIENTATION_TYPE_180 = 180val ORIENTATION_TYPE_270 = 270private var mOrientationEventListener: OrientationEventListener? = nullprivate var mScreenOrientationChangeListener:ScreenOrientationChangeListener? = nullprivate var currentType = ORIENTATION_TYPE_0fun init(context: Context, listener: ScreenOrientationChangeListener) {mScreenOrientationChangeListener = listenermOrientationEventListener = object :OrientationEventListener(context) {override fun onOrientationChanged(orientation: Int) {if (mScreenOrientationChangeListener == null) {return}if (orientation > 340 || orientation < 20) {//0if (currentType == 0) {return}if (getScreenRotation(context) == Surface.ROTATION_0) {mScreenOrientationChangeListener!!.onChange(ORIENTATION_TYPE_0)currentType = ORIENTATION_TYPE_0}} else if (orientation in 71..109) {//90if (currentType == 90) {return}val angle = getScreenRotation(context)if (angle == Surface.ROTATION_270) {mScreenOrientationChangeListener!!.onChange(ORIENTATION_TYPE_90)currentType = ORIENTATION_TYPE_90}} else if (orientation in 161..199) {//180if (currentType == 180) {return}val angle = getScreenRotation(context)if (angle == Surface.ROTATION_180) {mScreenOrientationChangeListener!!.onChange(ORIENTATION_TYPE_180)currentType = ORIENTATION_TYPE_180}} else if (orientation in 251..289) {//270if (currentType == 270) {return}val angle = getScreenRotation(context)if (angle == Surface.ROTATION_90) {mScreenOrientationChangeListener!!.onChange(ORIENTATION_TYPE_270)currentType = ORIENTATION_TYPE_270}}}}register()}private fun getScreenRotation(context: Context): Int {val windowManager =context.getSystemService(Context.WINDOW_SERVICE) as WindowManagerreturn windowManager.defaultDisplay?.rotation ?: 0}fun register() {if (mOrientationEventListener != null) {mOrientationEventListener!!.enable()}}fun unRegister() {if (mOrientationEventListener != null) {mOrientationEventListener!!.disable()}}interface ScreenOrientationChangeListener {/**** @param orientation*/fun onChange(orientation: Int)}
}
使⽤的话,直接这样:
ScreenOrientationHelper.init(this, object :
ScreenOrientationHelper.ScreenOrientationChangeListener {override fun onChange(orientation: Int) {when(orientation) {ScreenOrientationHelper.ORIENTATION_TYPE_0 -> {}ScreenOrientationHelper.ORIENTATION_TYPE_90 -> {}ScreenOrientationHelper.ORIENTATION_TYPE_180 -> {}ScreenOrientationHelper.ORIENTATION_TYPE_270 -> {}}}
})
通过上⾯的代码发现,在onOrientationChanged回调90度范围内时,判定屏幕显示⽅向是和 Surface.ROTATION_270⽐较的,⽽270范围内时是和Surface.ROTATION_90⽐较的。看得出来⻆度 是顺时针递增的,⽽屏幕⽅向是逆时针计算度数的。
其他问题
在测试过程中,上⾯的⽅案还存在另外⼀个问题,虽然onOrientationChanged这个回调很灵敏,但 是也有度数不变⽽屏幕⽅向旋转的情况发⽣,即保持屏幕⽅向不变,⽽是增加屏幕的坡度(将⼿机⼀ 边贴在桌⾯,慢慢⽴起来),在坡度达到⼀定时,屏幕会发⽣旋转,此时onOrientationChanged是 不会回调的,因为没有变化。这样就收不到屏幕旋转的回调了,但是在实际⽤⼿机的场景中,这种情 况是⽐较少的,可以亲身试试看。
小结
在平时开发中,要区分是哪种状态横屏的场景⽐较少,否则我认为Android会给出准确的回调的。 Android设备碎⽚化严重,除了刘海,在屏幕的下边缘还有虚拟导航栏,在不同的系统设置下,这个 导航栏不显示状态会不⼀样。那么这时候在悬浮球贴边这个需求中就不仅仅要考虑刘海了,还得考虑 导航栏。更有甚者,在旋转过程中,虚拟导航栏会⼀直保持在⼀个⽅向,和刘海叠加。那么要清楚的 算位置,第⼀步就是要监听屏幕的旋转了。
Android 学习笔录
Android 性能优化篇:https://qr18.cn/FVlo89
Android 车载篇:https://qr18.cn/F05ZCM
Android 逆向安全学习笔记:https://qr18.cn/CQ5TcL
Android Framework底层原理篇:https://qr18.cn/AQpN4J
Android 音视频篇:https://qr18.cn/Ei3VPD
Jetpack全家桶篇(内含Compose):https://qr18.cn/A0gajp
Kotlin 篇:https://qr18.cn/CdjtAF
Gradle 篇:https://qr18.cn/DzrmMB
OkHttp 源码解析笔记:https://qr18.cn/Cw0pBD
Flutter 篇:https://qr18.cn/DIvKma
Android 八大知识体:https://qr18.cn/CyxarU
Android 核心笔记:https://qr21.cn/CaZQLo
Android 往年面试题锦:https://qr18.cn/CKV8OZ
2023年最新Android 面试题集:https://qr18.cn/CgxrRy
Android 车载开发岗位面试习题:https://qr18.cn/FTlyCJ
音视频面试题锦:https://qr18.cn/AcV6Ap
相关文章:
如何用正确的姿势监听Android屏幕旋转
作者:37手游移动客户端团队 背景 关于个人,前段时间由于业务太忙,所以一直没有来得及思考并且沉淀点东西;同时组内一个个都在业务上能有自己的思考和总结,在这样的氛围下,不由自主的驱使周末开始写点东西&…...
mysql高级三:sql性能优化+索引优化+慢查询日志
内容介绍 单表索引失效案例 0、思考题:如果把100万数据插入MYSQL ,如何提高插入效率 (1)关闭自动提交,只手动提交一次 (2)删除除主键索引外其他索引 (3)拼写mysql可以执…...
HCIP VLAN--Hybrid接口
一、VLAN的特点 1、一个VLAN就是一个广播域,所以在同一个VLAN内部,计算机可以直接进行二层通信;而不同VLAN内的计算机,无法直接进行二层通信,只能进行三层通信来传递信息,即广播报文被限制在一个VLAN内。 …...
大数据开发面试必问:Hive调优技巧系列二
接上次分享的Hive调优技巧系列一: 数据倾斜、HiveJob优化 第1章 数据倾斜(重点) 绝大部分任务都很快完成,只有一个或者少数几个任务执行的很慢甚至最终执行失败,这样的现象为数据倾斜现象。 一定要和数据过量导致的…...
【C++】STL——list的模拟实现、构造函数、迭代器类的实现、运算符重载、增删查改
文章目录 1.模拟实现list1.1构造函数1.2迭代器类的实现1.3运算符重载1.4增删查改 1.模拟实现list list使用文章 1.1构造函数 析构函数 在定义了一个类模板list时。我们让该类模板包含了一个内部结构体_list_node,用于表示链表的节点。该结构体包含了指向前一个节点…...
vscode 插件::EIDE
最新最全 VSCODE 插件推荐(2023版)_vscode_白墨石-华为云开发者联盟 (csdn.net) 超好用的开发工具-VScode插件EIDE_vscode eide_桃成蹊2.0的博客-CSDN博客 Setup | Embedded IDE For VSCode (em-ide.com)...
Python 网络编程
Python 网络编程 Python 提供了两个级别访问的网络服务: 低级别的网络服务支持基本的 Socket,它提供了标准的 BSD Sockets API,可以访问底层操作系统 Socket 接口的全部方法。高级别的网络服务模块 SocketServer, 它提供了服务器…...
SQL 数据科学:了解和利用联接
推荐:使用 NSDT场景编辑器助你快速搭建可编辑的3D应用场景 什么是 SQL 中的连接? SQL 联接允许您基于公共列合并来自多个数据库表的数据。这样,您就可以将信息合并在一起,并在相关数据集之间创建有意义的连接。 SQL 中的连接类型…...
(统计学习方法|李航)第五章决策树——四五节:决策树的剪枝,CART算法
目录 一,决策数的剪枝 二,CART算法 1.CART生成 (1)回归树的生成 (2)分类树的生成 2.CART剪枝 (1)剪枝,形成一个子树序列 (2)在剪枝得到的子…...
C语言--结构体定义
整型数,浮点数,字符串是分散的数据表示,有时候我们需要很多类型表示一个整体,比如学生信息。 数组是元素类型一样的数据集合,如果是元素类型不同的数据集合,就要用到结构体 结构体一般是个模板,…...
解决Element Plus中Select在El Dialog里层级过低的问题(修改select选项框样式)
Element Plus是Vue.js的一套基于Element UI的组件库,提供了丰富的组件用于构建现代化的Web应用程序。其中,<el-select>是一个常用的下拉选择器组件,但在某些情况下,当<el-select>组件嵌套在<el-dialog>…...
【数据结构】二叉树 链式结构的相关问题
本篇文章来详细介绍一下二叉树链式结构经常使用的相关函数,以及相关的的OJ题。 目录 1.前置说明 2.二叉树的遍历 2.1 前序、中序以及后序遍历 2.2 层次遍历 3.节点个数相关函数实现 3.1 二叉树节点个数 3.2 二叉树叶子节点个数 3.3 二叉树第k层节点个数 3…...
【无标题】云原生在工业互联网的落地及好处!
什么是工业互联网? 工业互联网(Industrial Internet)是新一代信息通信技术与工业经济深度融合的新型基础设施、应用模式和工业生态,通过对人、机、物、系统等的全面连接,构建起覆盖全产业链、全价值链的全新制造和服务…...
人工智能在心电信号分类中的应用
目录 1 引言 2 传统机器学习中的特征提取与选择 3 深度学习中的特征提取与选择...
【Linux 网络】网络层协议之IP协议
IP协议 IP协议所处的位置网络层要解决的问题IP协议格式分片与组装网段划分特殊的IP地址IP地址的数量限制私网IP地址和公网IP地址路由 IP协议所处的位置 IP指网际互连协议,Internet Protocol的缩写,是TCP/IP体系中的网络层协议。 网络层要解决的问题 网络…...
.meta 文件
.meta 文件的作用简单来说是建立 Unity 与资源之间的“桥梁”。 在游戏中引用一个游戏资源,Unity 并不是直接按照文件的路径或者名称,而是使用一个独一无二的 GUID 来指向工程里该资源文件。 这个 GUID 就是存储在 Unity 工程为每一个资源和文件…...
CRITICAL_SECTION 用法
#include <stdio.h> #include <windows.h> typedef RTL_CRITICAL_SECTION CRITICAL_SECTION; CRITICAL_SECTION g_cs; //声明关键段 // 共享资源 char g_cArray[10]; unsigned int g_Count 0; DWORD WINAPI ThreadProc10(LPVOID pParam) { // 进入临界区 …...
汇川运动控制产品故障排查
针对汇川伺服产品(IS600/IS620)的基本检测和一些出现频率较高的故障进行检测判断方法,适用于服务人员在现场排查/判断机器故障时,准确定位问题。 一、简单故障排查 注1:接线错误:1、UVW相序是否正确&#…...
【Groups】50 Matplotlib Visualizations, Python实现,源码可复现
详情请参考博客: Top 50 matplotlib Visualizations 因编译更新问题,本文将稍作更改,以便能够顺利运行。 1 Dendrogram 树状图根据给定的距离度量将相似的点组合在一起,并根据点的相似性将它们组织成树状的链接。 新建文件Dendrogram.py: …...
windows安装kafka配置SASL-PLAIN安全认证
目录 1.Windows安装zookeeper: 1.1下载zookeeper 1.2 解压之后如图二 1.3创建日志文件 1.4复制 “zoo_sample.cfg” 文件 1.5更改 “zoo.cfg” 配置 1.6新建zk_server_jaas.conf 1.7修改zkEnv.cmd 1.8导入相关jar 1.9以上配置就配好啦,接下来启…...
ICLR 2026 | 大模型当裁判也“翻车“?北大清华联合多校提出TrustJudge,让LLM评估更值得信赖
让 GPT-4 给两篇文章打分,A 拿了 4 分、B 拿了 3 分。按常理 A 应该比 B 好吧?但换成成对比较,同一个模型却说 "B 更好"。更离谱的情况也有——A > B > C > A 的"石头剪刀布"循环,连传递性都守不住。…...
AI赋能仿真:借助快马平台让ExtendSim模型学会智能预测与动态调整
今天想和大家分享一个很有意思的实践:如何用AI给传统仿真模型加点"智能"。最近在做一个服务系统的仿真项目,发现顾客等待行为其实很复杂——不同人的耐心程度差异很大,传统仿真很难准确模拟这种动态变化。于是尝试用机器学习来优化…...
面试-并行前缀和优化 Linear Attention
1 什么是前缀和? 定义: 第 k 个元素的状态依赖于第 k-1 个元素; 公式: 前缀和 = 从第 1 个,一直加到当前位置; 例子: 比如有 4 个数: A、B、C、D; 那么前缀和的结果为: S1 = A S2 = A + B S3 = A + B + C S4 = A + B + C + D在 Linear Attention 中有所体现,即,…...
LANCZOS智能压缩+RGB自动转换:Anything to RealCharacters预处理模块详解
LANCZOS智能压缩RGB自动转换:Anything to RealCharacters预处理模块详解 1. 项目概述 Anything to RealCharacters是一款专为RTX 4090显卡设计的2.5D转真人图像转换系统。该系统基于通义千问Qwen-Image-Edit-2511图像编辑模型,集成了专门优化的写实化权…...
XGP-save-extractor:跨平台开源工具守护游戏存档数据安全
XGP-save-extractor:跨平台开源工具守护游戏存档数据安全 【免费下载链接】XGP-save-extractor Python script to extract savefiles out of Xbox Game Pass for PC games 项目地址: https://gitcode.com/gh_mirrors/xg/XGP-save-extractor 在游戏世界中&…...
UnrealPakViewer:虚幻引擎资源分析与Pak文件解析工具指南
UnrealPakViewer:虚幻引擎资源分析与Pak文件解析工具指南 【免费下载链接】UnrealPakViewer 查看 UE4 Pak 文件的图形化工具,支持 UE4 pak/ucas 文件 项目地址: https://gitcode.com/gh_mirrors/un/UnrealPakViewer 作为虚幻引擎开发者࿰…...
5分钟解锁全网视频下载:为什么res-downloader能让你的数字生活更自由?
5分钟解锁全网视频下载:为什么res-downloader能让你的数字生活更自由? 【免费下载链接】res-downloader 视频号、小程序、抖音、快手、小红书、直播流、m3u8、酷狗、QQ音乐等常见网络资源下载! 项目地址: https://gitcode.com/GitHub_Trending/re/res-…...
PDF补丁丁深度解析:高效PDF文档处理与批量优化完整指南
PDF补丁丁深度解析:高效PDF文档处理与批量优化完整指南 【免费下载链接】PDFPatcher PDF补丁丁——PDF工具箱,可以编辑书签、剪裁旋转页面、解除限制、提取或合并文档,探查文档结构,提取图片、转成图片等等 项目地址: https://g…...
终极指南:如何使用Polly.JS实现API版本控制与路径重写
终极指南:如何使用Polly.JS实现API版本控制与路径重写 【免费下载链接】pollyjs Record, Replay, and Stub HTTP Interactions. 项目地址: https://gitcode.com/gh_mirrors/po/pollyjs Polly.JS是一款强大的HTTP交互录制、重放和存根工具,能够帮助…...
RevokeMsgPatcher防撤回工具深度指南:让重要消息不再消失的完整解决方案
RevokeMsgPatcher防撤回工具深度指南:让重要消息不再消失的完整解决方案 【免费下载链接】RevokeMsgPatcher :trollface: A hex editor for WeChat/QQ/TIM - PC版微信/QQ/TIM防撤回补丁(我已经看到了,撤回也没用了) 项目地址: h…...
