Android14 InputManager-焦点窗口的更新

设置焦点时需要

先设置焦点APP

mFo-cusedApp是一个AppWindowToken,在WMS中用来表示当前处于Resume状态的Activity。它是由AMS在开始启动一个Activity时调用WMS的setFocusedApp()函数设置的。

考虑以下应用场景,当用户从Launcher中启动一个Activity之后,在新Activity的窗口显示之前便立刻按下了BACK键。很明显,用户的意图是关闭刚刚启动的Activity,而不是退出Launcher。然而,由于BACK键按得过快,新Activity尚未创建窗口,因此按照之前讨论的焦点窗口的查找条件,Launcher的窗口将会作为焦点窗口而接收BACK键,从而使得Launcher被退出。这与用户的意图是相悖的。为了解决这个问题,WMS要求焦点窗口必须属于mFocusedApp,或者位于mFocu-sedApp的窗口之上。再看添加这个限制之后,上面的应用场景会变成什么样子。当启动新Activity时,尽管其窗口尚未创建,但这个窗口的AppWindowToken已经被添加到mAppTokens列表的顶部,并通过setFocusedApp()设置为mFocusedApp。此时再更新焦点窗口时发现第一个符合焦点条件的窗口属于Launcher,但Launcher的AppWindowToken位于mFocusedApp也就是新Activity之下,因此它不能作为焦点窗口。这样,在新Activity创建窗口之前的一个短暂时间段内,系统处于无焦点窗口的情况。此时到来的BACK键在InputDispatcher中会因找不到目标窗口而触发handleTargetsNotReadyLocked(),从而进入等待重试状态。随后新Activity创建了自己的窗口并添加到WMS里,这时WMS可以成功地将这个窗口作为焦点,并通过IMS.setInputWindows()将其更新到InputDispatcher。这个更新动作会唤醒派发线程立刻对BACK键进行重试,这次重试便找到了新Activity的窗口作为目标,并将事件发送过去,从而正确地体现用户的意图。

    void setLastResumedActivityUncheckLocked(ActivityRecord r, String reason) {boolean focusedAppChanged = false;if (!getTransitionController().isTransientCollect(r)) {focusedAppChanged = r.mDisplayContent.setFocusedApp(r);if (focusedAppChanged) {mWindowManager.updateFocusedWindowLocked(UPDATE_FOCUS_NORMAL,true /*updateInputWindows*/);}}
WindowManagerService.javaboolean updateFocusedWindowLocked(int mode, boolean updateInputWindows) {Trace.traceBegin(TRACE_TAG_WINDOW_MANAGER, "wmUpdateFocus");boolean changed = mRoot.updateFocusedWindowLocked(mode, updateInputWindows);Trace.traceEnd(TRACE_TAG_WINDOW_MANAGER);return changed;}
RootWindowContainer.java
boolean updateFocusedWindowLocked(int mode, boolean updateInputWindows) {mTopFocusedAppByProcess.clear();boolean changed = false;int topFocusedDisplayId = INVALID_DISPLAY;// Go through the children in z-order starting at the top-mostfor (int i = mChildren.size() - 1; i >= 0; --i) {final DisplayContent dc = mChildren.get(i);changed |= dc.updateFocusedWindowLocked(mode, updateInputWindows, topFocusedDisplayId);final WindowState newFocus = dc.mCurrentFocus;if (newFocus != null) {final int pidOfNewFocus = newFocus.mSession.mPid;if (mTopFocusedAppByProcess.get(pidOfNewFocus) == null) {mTopFocusedAppByProcess.put(pidOfNewFocus, newFocus.mActivityRecord);}if (topFocusedDisplayId == INVALID_DISPLAY) {topFocusedDisplayId = dc.getDisplayId();}} else if (topFocusedDisplayId == INVALID_DISPLAY && dc.mFocusedApp != null) {// The top-most display that has a focused app should still be the top focused// display even when the app window is not ready yet (process not attached or// window not added yet).topFocusedDisplayId = dc.getDisplayId();}}if (topFocusedDisplayId == INVALID_DISPLAY) {topFocusedDisplayId = DEFAULT_DISPLAY;}if (mTopFocusedDisplayId != topFocusedDisplayId) {mTopFocusedDisplayId = topFocusedDisplayId;mWmService.mInputManager.setFocusedDisplay(topFocusedDisplayId);mWmService.mPolicy.setTopFocusedDisplay(topFocusedDisplayId);mWmService.mAccessibilityController.setFocusedDisplay(topFocusedDisplayId);ProtoLog.d(WM_DEBUG_FOCUS_LIGHT, "New topFocusedDisplayId=%d", topFocusedDisplayId);}return changed;}
DisplayContent.javaboolean updateFocusedWindowLocked(int mode, boolean updateInputWindows,int topFocusedDisplayId) {// Don't re-assign focus automatically away from a should-keep-focus app window.// `findFocusedWindow` will always grab the transient-launch app since it is "on top" which// would create a mismatch, so just early-out here.if (mCurrentFocus != null && mTransitionController.shouldKeepFocus(mCurrentFocus)// This is only keeping focus, so don't early-out if the focused-app has been// explicitly changed (eg. via setFocusedTask).&& mFocusedApp != null && mCurrentFocus.isDescendantOf(mFocusedApp)&& mCurrentFocus.isVisible() && mCurrentFocus.isFocusable()) {ProtoLog.v(WM_DEBUG_FOCUS, "Current transition prevents automatic focus change");return false;}WindowState newFocus = findFocusedWindowIfNeeded(topFocusedDisplayId);if (mCurrentFocus == newFocus) {return false;}boolean imWindowChanged = false;final WindowState imWindow = mInputMethodWindow;if (imWindow != null) {final WindowState prevTarget = mImeLayeringTarget;final WindowState newTarget = computeImeTarget(true /* updateImeTarget*/);imWindowChanged = prevTarget != newTarget;if (mode != UPDATE_FOCUS_WILL_ASSIGN_LAYERS&& mode != UPDATE_FOCUS_WILL_PLACE_SURFACES) {assignWindowLayers(false /* setLayoutNeeded */);}if (imWindowChanged) {mWmService.mWindowsChanged = true;setLayoutNeeded();newFocus = findFocusedWindowIfNeeded(topFocusedDisplayId);}}ProtoLog.d(WM_DEBUG_FOCUS_LIGHT, "Changing focus from %s to %s displayId=%d Callers=%s",mCurrentFocus, newFocus, getDisplayId(), Debug.getCallers(4));final WindowState oldFocus = mCurrentFocus;mCurrentFocus = newFocus;if (newFocus != null) {mWinAddedSinceNullFocus.clear();mWinRemovedSinceNullFocus.clear();if (newFocus.canReceiveKeys()) {// Displaying a window implicitly causes dispatching to be unpaused.// This is to protect against bugs if someone pauses dispatching but// forgets to resume.newFocus.mToken.paused = false;}}getDisplayPolicy().focusChangedLw(oldFocus, newFocus);mAtmService.mBackNavigationController.onFocusChanged(newFocus);if (imWindowChanged && oldFocus != mInputMethodWindow) {// Focus of the input method window changed. Perform layout if needed.if (mode == UPDATE_FOCUS_PLACING_SURFACES) {performLayout(true /*initial*/,  updateInputWindows);} else if (mode == UPDATE_FOCUS_WILL_PLACE_SURFACES) {// Client will do the layout, but we need to assign layers// for handleNewWindowLocked() below.assignWindowLayers(false /* setLayoutNeeded */);}}if (mode != UPDATE_FOCUS_WILL_ASSIGN_LAYERS) {// If we defer assigning layers, then the caller is responsible for doing this part.getInputMonitor().setInputFocusLw(newFocus, updateInputWindows);}adjustForImeIfNeeded();updateKeepClearAreas();// We may need to schedule some toast windows to be removed. The toasts for an app that// does not have input focus are removed within a timeout to prevent apps to redress// other apps' UI.scheduleToastWindowsTimeoutIfNeededLocked(oldFocus, newFocus);if (mode == UPDATE_FOCUS_PLACING_SURFACES) {pendingLayoutChanges |= FINISH_LAYOUT_REDO_ANIM;}// Notify the accessibility manager for the change so it has the windows before the newly// focused one starts firing events.// TODO(b/151179149) investigate what info accessibility service needs before input can// dispatch focus to clients.if (mWmService.mAccessibilityController.hasCallbacks()) {mWmService.mH.sendMessage(PooledLambda.obtainMessage(this::updateAccessibilityOnWindowFocusChanged,mWmService.mAccessibilityController));}return true;}

updateFocusedWindowLocked

首先寻找焦点窗口findFocusedWindowIfNeeded

然后将焦点窗口设置到input :setInputFocusLw

寻找焦点窗口

DisplayContent.javaWindowState findFocusedWindowIfNeeded(int topFocusedDisplayId) {return (hasOwnFocus() || topFocusedDisplayId == INVALID_DISPLAY)? findFocusedWindow() : null;}/*** Find the focused window of this DisplayContent. The search takes the state of the display* content into account* @return The focused window, null if none was found.*/WindowState findFocusedWindow() {mTmpWindow = null;// mFindFocusedWindow will populate mTmpWindow with the new focused window when found.forAllWindows(mFindFocusedWindow, true /* traverseTopToBottom */);if (mTmpWindow == null) {ProtoLog.v(WM_DEBUG_FOCUS_LIGHT, "findFocusedWindow: No focusable windows, display=%d",getDisplayId());return null;}return mTmpWindow;}
 private final ToBooleanFunction<WindowState> mFindFocusedWindow = w -> {final ActivityRecord focusedApp = mFocusedApp;ProtoLog.v(WM_DEBUG_FOCUS, "Looking for focus: %s, flags=%d, canReceive=%b, reason=%s",w, w.mAttrs.flags, w.canReceiveKeys(),w.canReceiveKeysReason(false /* fromUserTouch */));if (!w.canReceiveKeys()) {return false;}// When switching the app task, we keep the IME window visibility for better// transitioning experiences.// However, in case IME created a child window or the IME selection dialog without// dismissing during the task switching to keep the window focus because IME window has// higher window hierarchy, we don't give it focus if the next IME layering target// doesn't request IME visible.if (w.mIsImWindow && w.isChildWindow() && (mImeLayeringTarget == null|| !mImeLayeringTarget.isRequestedVisible(ime()))) {return false;}if (w.mAttrs.type == TYPE_INPUT_METHOD_DIALOG && mImeLayeringTarget != null&& !mImeLayeringTarget.isRequestedVisible(ime())&& !mImeLayeringTarget.isVisibleRequested()) {return false;}final ActivityRecord activity = w.mActivityRecord;if (focusedApp == null) {ProtoLog.v(WM_DEBUG_FOCUS_LIGHT,"findFocusedWindow: focusedApp=null using new focus @ %s", w);mTmpWindow = w;return true;}if (!focusedApp.windowsAreFocusable()) {// Current focused app windows aren't focusable...ProtoLog.v(WM_DEBUG_FOCUS_LIGHT, "findFocusedWindow: focusedApp windows not"+ " focusable using new focus @ %s", w);mTmpWindow = w;return true;}// Descend through all of the app tokens and find the first that either matches// win.mActivityRecord (return win) or mFocusedApp (return null).if (activity != null && w.mAttrs.type != TYPE_APPLICATION_STARTING) {if (focusedApp.compareTo(activity) > 0) {// App root task below focused app root task. No focus for you!!!ProtoLog.v(WM_DEBUG_FOCUS_LIGHT,"findFocusedWindow: Reached focused app=%s", focusedApp);mTmpWindow = null;return true;}// If the candidate activity is currently being embedded in the focused task, the// activity cannot be focused unless it is on the same TaskFragment as the focusedApp's.TaskFragment parent = activity.getTaskFragment();if (parent != null && parent.isEmbedded()) {if (activity.getTask() == focusedApp.getTask()&& activity.getTaskFragment() != focusedApp.getTaskFragment()) {return false;}}}ProtoLog.v(WM_DEBUG_FOCUS_LIGHT, "findFocusedWindow: Found new focus @ %s", w);mTmpWindow = w;return true;};

该方法中,将依次根据如下条件获得焦点窗口:

  • 如果WindowState不能接收Input事件,则不能作为焦点窗口;
    1. 如果没有前台Activity,则当前WindowState作为焦点窗口返回;
    2. 如果前台Activity是不可获焦状态,则当前WindowState作为焦点窗口返回;
    3. 如果当前WindowState由ActivityRecord管理,且该WindowState不是Staring Window类型,那么当前台Activity在当前WindowState所属Activity之上时,不存在焦点窗口;
    4. 处于focusedApp之下的窗口不能成为焦点窗口
    5. 如果以上条件都不满足,则当前WindowState作为焦点窗口返回;
  • 由此可以得出影响窗口焦点的动作有以下几个:
  • □窗口的增删操作。
  • □窗口次序的调整。
  • □Activity的启动与退出。
  • □DisplayContent的增删操作。
  • 因此当这些动作发生时,WMS都会触发对updateFocusedWindowLocked()的调用,并更新窗口的布局信息到输入系统,以实现按键事件正确派发。
  •     public boolean canReceiveKeys(boolean fromUserTouch) {if (mActivityRecord != null && mTransitionController.shouldKeepFocus(mActivityRecord)) {// During transient launch, the transient-hide windows are not visibleRequested// or on-top but are kept focusable and thus can receive keys.return true;}final boolean canReceiveKeys = isVisibleRequestedOrAdding()&& (mViewVisibility == View.VISIBLE) && !mRemoveOnExit&& ((mAttrs.flags & WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE) == 0)&& (mActivityRecord == null || mActivityRecord.windowsAreFocusable(fromUserTouch))// can it receive touches&& (mActivityRecord == null || mActivityRecord.getTask() == null|| !mActivityRecord.getTask().getRootTask().shouldIgnoreInput());if (!canReceiveKeys) {return false;}// Do not allow untrusted virtual display to receive keys unless user intentionally// touches the display.return fromUserTouch || getDisplayContent().isOnTop()|| getDisplayContent().isTrusted();}

    如果一个WindowState可以接受Input事件,需要同时满足多个条件:

    1. isVisibleRequestedOrAdding方法为true,表示该WindowState可见或处于添加过程中:
    2. mViewVisibility属性为View.VISIBLE,表示客户端View可见;
    3. mRemoveOnExit为false,表示WindowState的退出动画不存在;
    4. mAttrs.flags中不存在FLAG_NOT_FOCUSABLE标记,该标记如果设置,表示该窗口为不可获焦窗口;
    5. mActivityRecord为null或者mActivityRecord可获焦;
    6. cantReceiveTouchInput()方法为false,表示可以接受Touch事件。
    boolean isVisibleRequestedOrAdding() {final ActivityRecord atoken = mActivityRecord;return (mHasSurface || (!mRelayoutCalled && mViewVisibility == View.VISIBLE))&& isVisibleByPolicy() && !isParentWindowHidden()&& (atoken == null || atoken.isVisibleRequested())&& !mAnimatingExit && !mDestroying;}

然后将焦点窗口设置到input :setInputFocusLw

查看Android14 InputManager-InputWindow的更新过程

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

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

相关文章

内存管理——线性内存,进程空间

低2G为进程空间 开始地址结束地址大小属性00xFFFFF1M保留0x1000000x102FFF栈不固定位置、大小0x1030000x143FFF堆不固定位置、大小0x400000主程序文件不固定位置、大小加载dll不固定位置、大小0x7ffdd000TIB位置&#xff0c;大小编译时固定0x7FFFE000系统与用户共享数据块位置…

[newstarctf2023] --RE wp

AndroGenshin: rc4加密表&#xff0c;base64换表&#xff1a; 脚本梭就行 python username b"genshinimpact" base64_table [125, 239, 101, 151, 77, 163, 163, 110, 58, 230, 186, 206, 84, 84, 189, 193, 30, 63, 104, 178, 130, 211,164, 94, 75, 16, 32, 33…

发布 rust 源码包 (crates.io)

rust 编程语言的包 (或者 库, library) 叫做 crate, 也就是软件中的一个组件. 一个完整的软件通常由多个 crate 组成, rust 编译器 (rustc) 一次编译一整个 crate, 不同的 crate 可以同时并行编译. rust 官方有一个集中发布开源包的网站 crates.io. 发布在这上面的 crate 可以…

uniapp微信公众号H5分享

如果项目文件node_modules中没有weixin-js-sdk文件&#xff0c;则直接使用本文章提供的&#xff1b; 如果不生效&#xff0c;则在template.h5.html中引入 <script src"https://res.wx.qq.com/open/js/jweixin-1.6.0.js"></script> 首先引入weixin-js-…

vue3框架组件自动导入unplugin-vue-components

1.安装 npm i unplugin-vue-components -save-dev 2.配置 我这里用的是Vue CLI&#xff0c;所以要在vue.config.js文件中添加配置&#xff0c;官网中有写不同打包工具的配置写法 框架我使用的是Element Plus&#xff0c;使用前去官网查看自己的框架是否支持&#xff0c;主流…

LLM之RAG实战(二十七)| 如何评估RAG系统

有没有想过今天的一些应用程序是如何看起来几乎神奇地智能的&#xff1f;这种魔力很大一部分来自于一种叫做RAG和LLM的东西。把RAG&#xff08;Retrieval Augmented Generation&#xff09;想象成人工智能世界里聪明的书呆子&#xff0c;它会挖掘大量信息&#xff0c;准确地找到…

电脑黑屏什么都不显示怎么办 电脑开机黑屏不显示任何东西的4种解决办法

相信有很多网友都有经历电脑开机黑屏不显示任何东西&#xff0c;找了很多方法都没处理好&#xff0c;其实关于这个的问题&#xff0c;首先还是要了解清楚开机黑屏的原因&#xff0c;才能够对症下药&#xff0c;下面大家可以跟小编一起来看看怎么解决吧 电脑开机黑屏不显示任何…

【无刷电机学习】基础概念及原理介绍(持续更新中...)

目录&#xff08;2024.02.22版&#xff09; 1 定义 2 各种电机优势比较 2.1 有刷与无刷比较 2.2 交流与直流比较 2.3 内转子与外转子比较 2.4 低压BLDC的一些优点 3 基本原理 3.1 单相无刷电机 3.2 三相无刷电机 4 驱动方法 4.1 六步换相控制 4.1.1 基本原理 4…

突发!AI独角兽「竹间智能」被曝停工停产6个月

大家好我是二狗。 今天早上起来刷朋友圈&#xff0c;看到一张截图——AI创企竹间智能&#xff0c;宣称因为公司所处的经营环境艰难&#xff0c;部分部门和岗位将从即日起停工停产6个月。 图源&#xff1a;&#xff08;企服科学&#xff09; 下面是文字版&#xff1a; 由于公司…

Web服务器基础介绍

目录 Web服务器基础介绍 一、HTML是什么&#xff1f; 二、静态网页和动态网页 1、静态网页 2、动态网页 3、动态网页语言 PHP JSP Python Ruby 三、HTTP协议 1、HTTP协议是什么&#xff1f; 2、HTTP请求访问的方法 3、GET与POST比较 GET&#xff1a; POST&…

Linux网络编程(三-UDP协议)

目录 一、UDP概述 二、UDP的首部格式 三、UDP缓冲区 四、基于UDP的应用层协议 五、常见问题 一、UDP概述 UDP(User Datagram Protocol&#xff0c;用户数据协议报)是传输层协议&#xff0c;提供不可靠服务&#xff0c;其特点包括&#xff1a; 无连接&#xff1a;知道对端…

CSP-202309-3-梯度求解

CSP-202309-3-梯度求解 作为一个算法小白&#xff0c;本人第一次接触大模拟的题&#xff0c;本题的算法参考自&#xff1a;【CSP】202309-3 梯度求解 解题思路 1.输入处理 getchar();&#xff1a;从标准输入读取一个字符。这里它的作用可能是用来“吃掉”&#xff08;消耗&a…

Kafka_04_Topic和日志

Kafka_04_Topic和日志 Topic/PartitionTopicPartition 日志存储存储格式日志清理删除压缩 Topic/Partition Topic/Partition: Kafka中消息管理的基础单位 Topic和Partition并不实际存在(仅逻辑上的概念) 如: Topic和Partition关系 // 每个日志文件可对应多个日志分段, 其还可…

缓存篇—缓存击穿

在很多场景下&#xff0c;我们的业务通常会有几个数据会被频繁地访问&#xff0c;比如秒杀活动&#xff0c;这类被频地访问的数据被称为热点数据。 如果缓存中的某个热点数据过期了&#xff0c;此时大量的请求访问了该热点数据&#xff0c;就无法从缓存中读取&#xff0c;直接…

《UE5_C++多人TPS完整教程》学习笔记22 ——《P23 记录加入的玩家(Couting Incoming Players)》

本文为B站系列教学视频 《UE5_C多人TPS完整教程》 —— 《P23 记录加入的玩家&#xff08;Couting Incoming Players&#xff09;》 的学习笔记&#xff0c;该系列教学视频为 Udemy 课程 《Unreal Engine 5 C Multiplayer Shooter》 的中文字幕翻译版&#xff0c;UP主&#xff…

前端面试问题(jwt/布局/vue数组下标/扁平化/菜单树形/url api/新版本)

前端面试问题(jwt/布局/vue数组下标/扁平化/菜单树形/url api/新版本) 1. jwt鉴权逻辑 前端 JWT 鉴权逻辑通常涉及在发起请求时携带 JWT&#xff0c;并在接收到响应后处理可能的授权问题。 1. 用户登录&#xff1a; 用户提供凭证&#xff1a; 用户在登录界面输入用户名和密码…

如何使用Docker部署MongoDB并结合内网穿透实现远程访问本地数据库

文章目录 前言1. 安装Docker2. 使用Docker拉取MongoDB镜像3. 创建并启动MongoDB容器4. 本地连接测试5. 公网远程访问本地MongoDB容器5.1 内网穿透工具安装5.2 创建远程连接公网地址5.3 使用固定TCP地址远程访问 正文开始前给大家推荐个网站&#xff0c;前些天发现了一个巨牛的 …

2024最佳住宅代理IP服务商有哪些?

跨境出海已成为了近几年的最热趋势&#xff0c;大批量的企业开始开拓海外市场&#xff0c;而海外电商领域则是最受欢迎的切入口。新兴的tiktok、Temu&#xff0c;老牌的Amazon、Ebay&#xff0c;热门的Etsy、Mecari等等都是蓝海一片。跨境入门并不难&#xff0c;前期的准备中不…

深入理解文件查看命令:cat、more、less、tail、head

在Linux系统中&#xff0c;有许多命令用于查看文件的内容&#xff0c;其中包括cat、more、less、tail和head。这些命令提供了不同的方式来浏览文本文件&#xff0c;适用于各种查看需求。在本篇博客中&#xff0c;我们将深入介绍这些命令&#xff0c;并通过示例演示它们的用法。…

Spring Boot打war包部署到Tomcat,访问页面404 !!!

水善利万物而不争&#xff0c;处众人之所恶&#xff0c;故几于道&#x1f4a6; 文章目录 Spring Boot打war包部署到Tomcat&#xff0c;访问页面404 &#xff01;&#xff01;&#xff01;解决办法&#xff1a;检查Tomcat版本和Jdk的对应关系&#xff0c;我的Tomcat是6.x&#x…