当前位置: 首页 > news >正文

如何处理Android悬浮弹窗双击返回事件?

目录

1 前言

1.1 准备知识

1.2 问题概述

2 解决方案

3 代码部分

3.1 动态更新窗口焦点

3.2 窗口监听返回事件

3.3 判断焦点是否在窗口内部

3.4 窗口监听焦点移入/移出

4 注意事项

4.1 窗口范围

4.2 空隙处的返回事件处理


1 前言

1.1 准备知识

1)开发环境

  • 2D开发环境:所有界面或窗口都在主界面显示;
  • 3D开发环境:保留原生Android的主界面,在主界面之外绘制各种窗口,配合3D渲染以实现3D效果。

2)焦点:就是Hover点、中央注视点、可与用户交互的点。

3)窗口:就是系统窗口、悬浮弹窗,内部通过addView方法去添加View,本文窗口监听指的就是View监听。

4)事件分发:Android设备一般会使用如下3种,本文采用的第3种setOnHoverListener获取事件。

  • setOnTouchListener(MotionEvent::InputEvent):手机、平板、车载等屏幕可触控的2D设备;
  • setOnKeyListener(KeyEvent::InputEvent):电视、投影仪等屏幕不可触控的2D设备;
  • setOnHoverListener(MotionEvent::InputEvent):AR眼镜等增强现实设备。

5)Hover事件分发:当前View在焦点移出(不再是Hover状态)时,不会立即发送ACTION_HOVER_EXIT退出事件,需要等到下一个View获取到ACTION_HOVER_ENTER状态时才会发送上一个View的ACTION_HOVER_EXIT退出事件。

6)窗口内部View的Hover事件分发过程

  • RootView会先获取到ACTION_HOVER_ENTER事件;
  • 当进入ChildView时,ChildView会先获取到ACTION_HOVER_ENTER事件,然后RootView会获取到ACTION_HOVER_EXIT事件;
  • 当从ChildView退出时,ChildView会先获取到ACTION_HOVER_EXIT事件,然后RootView会获取到ACTION_HOVER_ENTER事件。

1.2 问题概述

        问题描述:在Android悬浮弹窗上双击返回,主界面响应返回事件。

        问题原因:悬浮弹窗设置了flag为窗口不可获取焦点即:WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE。

        问题分析

  • 悬浮弹窗设置flag为窗口不可获取焦点,是为了不影响主界面的焦点响应(Android默认主界面的窗口是获取焦点的);
  • 如果悬浮弹窗设置flag可获取焦点,那么Android的事件分发是无法发送到主界面的,会将事件分发给当前可获取焦点的悬浮弹窗;
  • 如下图,左侧图1为悬浮弹窗,右侧图2为主界面某应用打开一个Activity。图1悬浮弹窗是常驻于图2主界面的左侧,且默认不可获取焦点,但在特殊情况时可获取焦点(如展开键盘、焦点在此悬浮弹窗内部等情况)。

        解决方案:当焦点在悬浮弹窗内部时,设置窗口flag可获取焦点;当焦点不在悬浮弹窗内部时,设置窗口flag不可获取焦点。

2 解决方案

        方案主要分为如下几步:

  1. 窗口默认不可获取焦点;
  2. 窗口监听焦点的移入/移出事件;
  3. 窗口监听到焦点移入,判断窗口是否可获取焦点,否——设置窗口可获取焦点,是——不做任何操作;
  4. 窗口监听到焦点移出,判断焦点是否在窗口内部,否——设置窗口不可获取焦点,是——不做任何操作;

        读者可思考如下2个问题,

1)问题1:为什么在窗口监听到焦点移入后,要再判断窗口是否可获取焦点?

2)问题2:为什么在窗口监听到焦点移出后,要再判断焦点是否在窗口内部?

        相信本文《1.1 准备知识的Hover事件分发部分》可以给你一些灵感。   

     

3 代码部分

3.1 动态更新窗口焦点

        核心API:

  • WindowManager.updateViewLayout
  • WindowManager.LayoutParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
    private fun initLiveDataBus() {LiveDataBus.get().with(Constants.NOTIFICATION_EVENT_BUS_FOCUSABLE, Boolean::class.java).observeForever { focusable: Boolean ->Log.d(TAG, "onChanged: $focusable")updateNotificationParams(focusable)}}private fun updateNotificationParams(focusable: Boolean) {initLayoutParams(focusable)mUiHandler.post {synchronized(this) {if (mIsBarWindowAdded) {try {mWindowManager.updateViewLayout(mNotificationBar, mLayoutParams)} catch (e: Exception) {e.printStackTrace()}}}}}private fun initLayoutParams(focusable: Boolean) {mLayoutParams = WindowManager.LayoutParams().apply {type = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAYval density = mContext.resources.displayMetrics.densitywidth = (640 * density).toInt()height = (640 * density).toInt()flags =WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL or WindowManager.LayoutParams.FLAG_LAYOUT_IN_SCREEN or WindowManager.LayoutParams.FLAG_LAYOUT_INSET_DECOR or WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH or WindowManager.LayoutParams.FLAG_TOUCHABLE_WHEN_WAKING or WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITSif (!focusable) {flags = flags or WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE}format = PixelFormat.RGBA_8888 // 去除默认时有的黑色背景,设置为全透明gravity = Gravity.TOP or Gravity.STARTtitle = SYSUI_NOTIFICATIONx = -(640 * density).toInt()y = 0}}

3.2 窗口监听返回事件

        窗口设置可获得焦点后,内部View会获取到事件分发的事件,在此View中重写dispatchKeyEvent方法,监听keyCode == KeyEvent.KEYCODE_BACK事件,就可对返回事件进行处理。

    override fun dispatchKeyEvent(event: KeyEvent): Boolean {if (event.keyCode == KeyEvent.KEYCODE_BACK) {Log.i(TAG, "dispatchKeyEvent: KEYCODE_BACK")// 窗口设置可获得焦点后,内部View会获取到事件分发的事件,并可对返回事件进行处理}return super.dispatchKeyEvent(event)}

3.3 判断焦点是否在窗口内部

        通过View相对于屏幕位置X/Y、以及View宽高,共同确定View的边界。

    mRootView.post {val locationXY = IntArray(2)mRootView.getLocationOnScreen(locationXY)val locationX = locationXY[0]val locationY = locationXY[1]val measuredWidth = mRootView.measuredWidthval measuredHeight = mRootView.measuredHeight}/*** 焦点:就是Hover点、中央注视点、可与用户交互的点。** @param locationX View相对于屏幕位置X* @param locationY View相对于屏幕位置Y* @param measuredWidth View宽* @param measuredHeight View高* @param rawX 焦点相对于屏幕位置X* @param rawY 焦点相对于屏幕位置Y** @return 焦点是否未在View内部*/private fun isViewNotFocus(locationX: Int,locationY: Int,measuredWidth: Int,measuredHeight: Int,rawX: Float,rawY: Float) =if (rawX <= locationX || rawX >= locationX + measuredWidth || rawY <= locationY || rawY >= locationY + measuredHeight) {// 焦点不在View内部Log.i(TAG, "isViewNotFocus: 焦点不在View内部")true} else {// 焦点在View内部Log.i(TAG, "isViewNotFocus: 焦点在View内部")false}

3.4 窗口监听焦点移入/移出

  • 窗口监听到焦点移入,判断窗口是否可获取焦点,否——设置窗口可获取焦点,是——不做任何操作;
  • 窗口监听到焦点移出,判断焦点是否在窗口内部,否——设置窗口不可获取焦点,是——不做任何操作;
  • 最后,通过发送NOTIFICATION_EVENT_BUS_FOCUSABLE事件,进而设置窗口的是否可获取焦点。
    // 注:Focus移出时需要包含边界。mRootView.setOnHoverListener { v, event ->when (event.action) {MotionEvent.ACTION_HOVER_ENTER -> {Log.i(TAG,"OnHoverListener: 进入, action =  ${event.action},motionX = ${event.rawX},motionY = ${event.rawY}")LiveDataBus.get().with(NOTIFICATION_EVENT_BUS_FOCUSABLE).value?.let {if (!(it as Boolean)) {Log.i(TAG, "OnHoverListener: 进入, focus-true-again")LiveDataBus.get().with(NOTIFICATION_EVENT_BUS_FOCUSABLE).value =true}} ?: let {Log.i(TAG, "OnHoverListener: 进入, focus-true-init")LiveDataBus.get().with(NOTIFICATION_EVENT_BUS_FOCUSABLE).value = true}}MotionEvent.ACTION_HOVER_MOVE -> {}MotionEvent.ACTION_HOVER_EXIT -> {Log.i(TAG,"OnHoverListener: 退出, action =  ${event.action},motionX = ${event.rawX},motionY = ${event.rawY}")if (isViewNotFocus(locationX,locationY,measuredWidth,measuredHeight,event.rawX,event.rawY)) {Log.i(TAG, "OnHoverListener: 退出, focus-false")LiveDataBus.get().with(NOTIFICATION_EVENT_BUS_FOCUSABLE).value = false}}}false}

4 注意事项

4.1 窗口范围

        在判断焦点是否在窗口内部时,需要确认窗口范围,如果窗口内部的View有设置Padding或Margin,应该将其去掉。

        如:本文的窗口大小是640*640,但View大小是540*580,所以计算时需要去掉相应Padding或Margin,重写isViewNotFocus()方法如下:

    private fun isViewNotFocus(locationX: Int,locationY: Int,measuredWidth: Int,measuredHeight: Int,rawX: Float,rawY: Float): Boolean {val density = context.resources.displayMetrics.densityreturn rawX <= locationX + 50 * density || rawX >= locationX + measuredWidth - 100 * density || rawY <= locationY + 15 * density || rawY >= locationY + measuredHeight - 60 * density}

4.2 空隙处的返回事件处理

        1)从窗口移出到空隙处

        通过本文1.1准备知识的第5部分《Hover事件分发》,我们知道,从窗口移出但还未有下一个View获取焦点时,此时窗口还是会接收到返回事件。

        2)从View移出到空隙处

        从当前View移出但还未有下一个View获取焦点时,此时当前View还是会接收到返回事件。

那么,如何处理这种空隙处的返回事件呢?

       核心:从系统层拦截此种情况下的返回事件 。

  1. 渲染层:提供接口,返回焦点移入移出时当前layer的名称,是否有碰撞窗口等信息;
  2. 系统层:当没有碰撞窗口时,从系统层拦截掉返回事件的分发;
  3. 应用层:监听焦点移入移出,改变窗口focus属性,并处理返回事件;

解决方案:

        当空隙处有返回事件产生时,系统层通过渲染层的接口,获取到当前焦点所在位置的layer名称,如果layer名称为空则断定为空隙处,直接做拦截处理,不再往应用层分发。

注:每个窗口、Activity在其Window中,都有设置其title属性,layer名称就是此title属性的值。


目录

1 前言

1.1 准备知识

1.2 问题概述

2 解决方案

3 代码部分

3.1 动态更新窗口焦点

3.2 窗口监听返回事件

3.3 判断焦点是否在窗口内部

3.4 窗口监听焦点移入/移出

4 注意事项

4.1 窗口范围

4.2 空隙处的返回事件处理

相关文章:

如何处理Android悬浮弹窗双击返回事件?

目录 1 前言 1.1 准备知识 1.2 问题概述 2 解决方案 3 代码部分 3.1 动态更新窗口焦点 3.2 窗口监听返回事件 3.3 判断焦点是否在窗口内部 3.4 窗口监听焦点移入/移出 4 注意事项 4.1 窗口范围 4.2 空隙处的返回事件处理 1 前言 1.1 准备知识 1&#xff09;开发环…...

高可用篇_A Docker容器化技术_II Docker环境搭建和常见命令

原创作者&#xff1a;田超凡&#xff08;程序员田宝宝&#xff09; 版权所有&#xff0c;引用请注明原作者&#xff0c;严禁复制转载 Docker安装 Docker 要求 CentOS7 系统的内核版本在 3.10以上 &#xff0c;查看本页面的前提条件来验证你的CentOS 版本是否支持 Docker 。 …...

Vue.js+SpringBoot开发食品生产管理系统

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 加工厂管理模块2.2 客户管理模块2.3 食品管理模块2.4 生产销售订单管理模块2.5 系统管理模块2.6 其他管理模块 三、系统展示四、核心代码4.1 查询食品4.2 查询加工厂4.3 新增生产订单4.4 新增销售订单4.5 查询客户 五、…...

Python面试笔记

Python面试笔记 PythonQ. Python中可变数据类型与不可变数据类型&#xff0c;浅拷贝与深拷贝详解Q. 解释什么是lambda函数&#xff1f;它有什么好处&#xff1f;Q. 什么是装饰器&#xff1f;Q. 什么是Python的垃圾回收机制&#xff1f;Q. Python内置函数dir的用法&#xff1f;Q…...

springboot 查看和修改内置 tomcat 版本

解析Spring Boot父级依赖 去到项目的根pom文件中&#xff0c;找到parent依赖&#xff1a; <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>${springboot.version}…...

003——移植鸿蒙

目录 一、顶层Make分析 二、添加一个新的单板 2.1 Kconfig 2.2 Makefile 2.2.1 顶层Makefile 2.2.2 platform下的Makefile 2.2.3 platform下的bsp.mk文件 2.3 编译与调试 2.4 解决链接错误 三、内核启动流程的学习 3.1 韦东山老师总结的启动四步 3.2 启动文件分析…...

罗马数字转整数-力扣通过自己编译器编译

学会将力扣题目用自己自带的编译软件编译---纯自己想的本题解法 字符 数值 I 1 V 5 X 10 L 50 C 100 D 500 M 1000 例如&#xff0c; 罗马数字 2 写做 II &#xff0c;即为两…...

深入解析JVM加载机制

一、背景 Java代码被编译器变成生成Class字节码&#xff0c;但字节码仅是一个特殊的二进制文件&#xff0c;无法直接使用。因此&#xff0c;都需要放到JVM系统中执行&#xff0c;将Class字节码文件放入到JVM的过程&#xff0c;简称类加载。 二、整体流程 三、阶段逻辑分析 3…...

python redis中blpop和lpop的区别

python redis中lpop()方法是获取并删除左边第一个对象。 def lpop(self,name: str,count: Optional[int] None,) -> Union[Awaitable[Union[str, List, None]], Union[str, List, None]]:"""Removes and returns the first elements of the list name.By de…...

第四百一十回

文章目录 1. 概念介绍2. 方法与细节2.1 获取方法2.2 使用细节 3. 示例代码4. 内容总结 我们在上一章回中介绍了"如何获取当前系统语言"相关的内容&#xff0c;本章回中将介绍如何获取时间戳.闲话休提&#xff0c;让我们一起Talk Flutter吧。 1. 概念介绍 我们在本章…...

程序员的README——编写可维护的代码(一)

用户行为不可预测&#xff0c;网络不可靠&#xff0c;事情总会出错。生产环境下的软件必须一直保持可用状态。 编写可维护的代码有助于你应对不可预见的情况&#xff0c;可维护的代码有内置的保护、诊断和控制。 切记通过安全和有弹性的编码实践进行防御式编程来保护你的系统&a…...

数据库管理-第160期 Oracle Vector DB AI-11(20240312)

数据库管理160期 2024-03-12 数据库管理-第160期 Oracle Vector DB & AI-11&#xff08;20240312&#xff09;1 向量的函数操作to_vector()将vector转换为标准值vector_norm()vector_dimension_count()vector_dimension_format() 2 将向量转换为字符串或CLOBvector_seriali…...

(C++进阶)boost库笔记

目录 1、boost::function 1.1 概述 1.2 boost包装器和C11包装器对比 1.2、代码示例 1、boost::function 1.1 概述 boost::function 是 Boost 库中提供的一个通用函数对象包装器&#xff0c;它可以存储指向任何可调用对象的指针&#xff0c;并且可以在任何时候通过 operat…...

MapReduce面试重点

文章目录 1. 简述MapReduce整个流程2. join原理 1. 简述MapReduce整个流程 数据划分(Input Splitting)&#xff1a;开始时&#xff0c;输入数据被分割成逻辑上的小块&#xff0c;每个块被称为Input Split。 映射(Map)&#xff1a;每个Input Split 由一个或多个Map任务处理&…...

C语言简单题(7)从主函数中输入10个等长字符串,用一个函数对他们排序,然后在主函数输出这10个已排好序的字符串

从主函数中输入10个等长字符串&#xff0c;用一个函数对他们排序&#xff0c;然后在主函数输出这10个已排好序的字符串 /* 从主函数中输入10个等长字符串&#xff0c;用一个函数对他们排序&#xff0c;然后在主函数输出这10个已排好序的字符串 */ #include<stdio.h> …...

光伏科普|太阳能光伏发电应用场景有哪些?

太阳能光伏发电的应用领域其实非常广泛&#xff0c;很多人会不相信&#xff0c;但在我们的日常生活中随处可见太阳能光伏产业&#xff0c;本文将详细介绍其应用场景有哪些。 一、工业领域厂房 太阳能光伏发电作为一种清洁、可再生的能源&#xff0c;安装在工业领域厂房&#…...

Go 构建高效的二叉搜索树联系簿

引言 树是一种重要的数据结构&#xff0c;而二叉搜索树&#xff08;BST&#xff09;则是树的一种常见形式。在本文中&#xff0c;我们将学习如何构建一个高效的二叉搜索树联系簿&#xff0c;以便快速插入、搜索和删除联系人信息。 介绍二叉搜索树 二叉搜索树是一种有序的二叉…...

基于YOLOv8/YOLOv7/YOLOv6/YOLOv5的交通信号灯识别系统(深度学习+UI界面+训练数据集+Python代码)

摘要&#xff1a;本研究详细介绍了一种采用深度学习技术的交通信号灯识别系统&#xff0c;该系统集成了最新的YOLOv8算法&#xff0c;并与YOLOv7、YOLOv6、YOLOv5等早期算法进行了性能评估对比。该系统能够在各种媒介——包括图像、视频文件、实时视频流及批量文件中——准确地…...

以太坊开发学习-solidity(三)函数类型

目录 函数类型 函数类型 solidity官方文档里把函数归到数值类型 函数类型是一种表示函数的类型。可以将一个函数赋值给另一个函数类型的变量&#xff0c; 也可以将一个函数作为参数进行传递&#xff0c;还能在函数调用中返回函数类型变量。 函数类型有两类&#xff1a;- 内部&…...

教你把公司吃干抹净、榨干带走

大家好&#xff1a; 衷心希望各位点赞。 您的问题请留在评论区&#xff0c;我会及时回答 正文 打工人一定要做到够自私&#xff0c;把公司的一切为我所用&#xff0c;你要知道闷头打工是没有出路的。聪明的人会以最快的速度榨干带走公司的一切资源、人脉、技能&#xff0c;为…...

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站&#xff0c;会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后&#xff0c;网站没有变化的情况。 不熟悉siteground主机的新手&#xff0c;遇到这个问题&#xff0c;就很抓狂&#xff0c;明明是哪都没操作错误&#x…...

RestClient

什么是RestClient RestClient 是 Elasticsearch 官方提供的 Java 低级 REST 客户端&#xff0c;它允许HTTP与Elasticsearch 集群通信&#xff0c;而无需处理 JSON 序列化/反序列化等底层细节。它是 Elasticsearch Java API 客户端的基础。 RestClient 主要特点 轻量级&#xff…...

python/java环境配置

环境变量放一起 python&#xff1a; 1.首先下载Python Python下载地址&#xff1a;Download Python | Python.org downloads ---windows -- 64 2.安装Python 下面两个&#xff0c;然后自定义&#xff0c;全选 可以把前4个选上 3.环境配置 1&#xff09;搜高级系统设置 2…...

汽车生产虚拟实训中的技能提升与生产优化​

在制造业蓬勃发展的大背景下&#xff0c;虚拟教学实训宛如一颗璀璨的新星&#xff0c;正发挥着不可或缺且日益凸显的关键作用&#xff0c;源源不断地为企业的稳健前行与创新发展注入磅礴强大的动力。就以汽车制造企业这一极具代表性的行业主体为例&#xff0c;汽车生产线上各类…...

高危文件识别的常用算法:原理、应用与企业场景

高危文件识别的常用算法&#xff1a;原理、应用与企业场景 高危文件识别旨在检测可能导致安全威胁的文件&#xff0c;如包含恶意代码、敏感数据或欺诈内容的文档&#xff0c;在企业协同办公环境中&#xff08;如Teams、Google Workspace&#xff09;尤为重要。结合大模型技术&…...

TRS收益互换:跨境资本流动的金融创新工具与系统化解决方案

一、TRS收益互换的本质与业务逻辑 &#xff08;一&#xff09;概念解析 TRS&#xff08;Total Return Swap&#xff09;收益互换是一种金融衍生工具&#xff0c;指交易双方约定在未来一定期限内&#xff0c;基于特定资产或指数的表现进行现金流交换的协议。其核心特征包括&am…...

Android Bitmap治理全解析:从加载优化到泄漏防控的全生命周期管理

引言 Bitmap&#xff08;位图&#xff09;是Android应用内存占用的“头号杀手”。一张1080P&#xff08;1920x1080&#xff09;的图片以ARGB_8888格式加载时&#xff0c;内存占用高达8MB&#xff08;192010804字节&#xff09;。据统计&#xff0c;超过60%的应用OOM崩溃与Bitm…...

pikachu靶场通关笔记22-1 SQL注入05-1-insert注入(报错法)

目录 一、SQL注入 二、insert注入 三、报错型注入 四、updatexml函数 五、源码审计 六、insert渗透实战 1、渗透准备 2、获取数据库名database 3、获取表名table 4、获取列名column 5、获取字段 本系列为通过《pikachu靶场通关笔记》的SQL注入关卡(共10关&#xff0…...

python执行测试用例,allure报乱码且未成功生成报告

allure执行测试用例时显示乱码&#xff1a;‘allure’ &#xfffd;&#xfffd;&#xfffd;&#xfffd;&#xfffd;ڲ&#xfffd;&#xfffd;&#xfffd;&#xfffd;ⲿ&#xfffd;&#xfffd;&#xfffd;Ҳ&#xfffd;&#xfffd;&#xfffd;ǿ&#xfffd;&am…...

深度学习习题2

1.如果增加神经网络的宽度&#xff0c;精确度会增加到一个特定阈值后&#xff0c;便开始降低。造成这一现象的可能原因是什么&#xff1f; A、即使增加卷积核的数量&#xff0c;只有少部分的核会被用作预测 B、当卷积核数量增加时&#xff0c;神经网络的预测能力会降低 C、当卷…...