【问题分析】关于SF侧Launcher层级的分析【Android15】

2024-08-30 10:04

本文主要是介绍【问题分析】关于SF侧Launcher层级的分析【Android15】,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在这里插入图片描述

一般来说,SF侧的Layer层级和WMS侧WindowContainer侧的层级是一一对应的,但是对Launcher来说,则略有不同,这点之前我在打印SF信息的时候,也有注意过,但是没有去仔细思考过为什么会这样,直到这次分析问题的时候踩了一坑,才发现有必要梳理一下这块逻辑,并做个记录。

1 问题描述

进入超级省电模式(也是一个Launcher),然后随便打开一个App,如Message,然后在Message界面上划,发现无法返回到Home。

2 问题分析

2.1 分析1

最初我分析的方向是错误的,刚拿到这个问题的时候,我复现了一下,先看上层WMS处WindowContainer的信息:

在这里插入图片描述

没问题,Home类型的Task已经在TaskDisplayArea的top了。

再看SF:

在这里插入图片描述

Message仍然是可见的。

因此我就直接认为是SF侧没有把Launcher对应的Task移动到TaskDisplayArea的top,比如Transaction.setLayer这个方法没有被调用之类的,但是继续打印log后才发现,正常情况下也没有为Home类型的Task设置layer的操作,这个分析方向是错的。

2.2 对Home类型Task的层级的特殊处理

正常情况下,当我从任意一个App回到Home后,再看此时winscope的信息:

在这里插入图片描述

发现虽然Launcher是可见的,但是它在SF侧仍然是出于TaskDisplayArea的bottom的,而在WMS侧,它在WindowContainer层级结构中是处于TaskDisplayArea的top的。

画个示意图,按照层级高的在上的形式。

WMS侧的情况为:

在这里插入图片描述

SF侧的情况为:

在这里插入图片描述

WMS侧的情况是符合直觉的,但是SF侧的确是Launcher的Task在底部,为什么会这样呢?

直接看TaskDisplayArea调整Task的层级地方,在TaskDisplayArea.assignRootTaskOrdering:

在这里插入图片描述

其实逻辑很清楚了,首先定义一个局部变量layer,初始化为0,然后每次都是先调用TaskDisplayArea.adjustRootTaskLayer来设置Home类型的Task的层级,所以Home类型的Task在SF侧TaskDisplayArea中就是一直处于bottom的。

最终TaskDisplayArea.adjustRootTaskLayer会调用WindowContainer.assignLayer,这里会调用Transaction.setLayer来完成最终的Layer设置,调用堆栈为:

在这里插入图片描述

可以看到调用的时机是在动画就绪的时候,Transition.onTransactionReady。

最后说一下,想要让Home类型的Task能够在屏幕上被看见,那么就只能在适当的时机隐藏位于Home类型Task之上的其它Task,如果这两个Task都是可见的,那么普通的App Task遮挡住Home类型的Task。

当前问题就是这个原因,继续分析。

2.3 分析2

经过以上分析,可知我们分析的重点不是在于SurfaceControl的层级设置,而是在于SurfaceControl的可见性(show和hide)设置。

先来回顾一下Transition中和可见性相关的重要节点,以从Message回到Launcher为例:

1)、启动Launcher,Launcher相关ActivityRecord变为可见。

2)、在动画就绪,Transition.onTransactionReady的时候,需要将Launcher的相关Layer设置为可见。

3)、然后播放动画,此时参与动画的Message和Launcher的相关Layer都应该是可见的。

4)、动画结束,Message相关的ActivityRecord变为不可见,那么它的Task也会被认为是不可见,进而调用Transaction.hide来隐藏它的SurfaceControl,堆栈为:

在这里插入图片描述

因此我们应该关注动画结束后,Message对应的Task没有去隐藏。

之后就定位到了问题原因,发现整个过程中没有“Finish transition”相关的log打印,说明动画流程没有走完,那么自然也不会将Message对应的Task隐藏。

再结合Launcher那边说它们是通过startRecentsTransition等接口来启动相关Home Activity的,因此很大概率是本次启动是瞬态启动。

瞬态启动一个重要的特点就是,从一个界面进入Recents,并且离开Recents后,一个完成的Transition才算完成,如果只是进入Recents,那么Transition只走到了Transition.onTransactionReady,只有从Recents界面离开(选择Recents界面的一个应用进入,或者点击Recents界面的空白区域回到Home),动画才会开始播放,并且最终走到finishTransition阶段,也就是说需要Launcher那边的动画开始播放并且播放完成后Transition才会结束,因此需要Launcher那边继续排查。

这篇关于【问题分析】关于SF侧Launcher层级的分析【Android15】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1120510

相关文章

MyBatis Plus 中 update_time 字段自动填充失效的原因分析及解决方案(最新整理)

《MyBatisPlus中update_time字段自动填充失效的原因分析及解决方案(最新整理)》在使用MyBatisPlus时,通常我们会在数据库表中设置create_time和update... 目录前言一、问题现象二、原因分析三、总结:常见原因与解决方法对照表四、推荐写法前言在使用 MyBATis

Python主动抛出异常的各种用法和场景分析

《Python主动抛出异常的各种用法和场景分析》在Python中,我们不仅可以捕获和处理异常,还可以主动抛出异常,也就是以类的方式自定义错误的类型和提示信息,这在编程中非常有用,下面我将详细解释主动抛... 目录一、为什么要主动抛出异常?二、基本语法:raise关键字基本示例三、raise的多种用法1. 抛

MySQL 设置AUTO_INCREMENT 无效的问题解决

《MySQL设置AUTO_INCREMENT无效的问题解决》本文主要介绍了MySQL设置AUTO_INCREMENT无效的问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参... 目录快速设置mysql的auto_increment参数一、修改 AUTO_INCREMENT 的值。

关于跨域无效的问题及解决(java后端方案)

《关于跨域无效的问题及解决(java后端方案)》:本文主要介绍关于跨域无效的问题及解决(java后端方案),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录通用后端跨域方法1、@CrossOrigin 注解2、springboot2.0 实现WebMvcConfig

Go语言中泄漏缓冲区的问题解决

《Go语言中泄漏缓冲区的问题解决》缓冲区是一种常见的数据结构,常被用于在不同的并发单元之间传递数据,然而,若缓冲区使用不当,就可能引发泄漏缓冲区问题,本文就来介绍一下问题的解决,感兴趣的可以了解一下... 目录引言泄漏缓冲区的基本概念代码示例:泄漏缓冲区的产生项目场景:Web 服务器中的请求缓冲场景描述代码

Java死锁问题解决方案及示例详解

《Java死锁问题解决方案及示例详解》死锁是指两个或多个线程因争夺资源而相互等待,导致所有线程都无法继续执行的一种状态,本文给大家详细介绍了Java死锁问题解决方案详解及实践样例,需要的朋友可以参考下... 目录1、简述死锁的四个必要条件:2、死锁示例代码3、如何检测死锁?3.1 使用 jstack3.2

解决JSONField、JsonProperty不生效的问题

《解决JSONField、JsonProperty不生效的问题》:本文主要介绍解决JSONField、JsonProperty不生效的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录jsONField、JsonProperty不生效javascript问题排查总结JSONField

github打不开的问题分析及解决

《github打不开的问题分析及解决》:本文主要介绍github打不开的问题分析及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、找到github.com域名解析的ip地址二、找到github.global.ssl.fastly.net网址解析的ip地址三

MySQL版本问题导致项目无法启动问题的解决方案

《MySQL版本问题导致项目无法启动问题的解决方案》本文记录了一次因MySQL版本不一致导致项目启动失败的经历,详细解析了连接错误的原因,并提供了两种解决方案:调整连接字符串禁用SSL或统一MySQL... 目录本地项目启动报错报错原因:解决方案第一个:第二种:容器启动mysql的坑两种修改时区的方法:本地

springboot加载不到nacos配置中心的配置问题处理

《springboot加载不到nacos配置中心的配置问题处理》:本文主要介绍springboot加载不到nacos配置中心的配置问题处理,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录springboot加载不到nacos配置中心的配置两种可能Spring Boot 版本Nacos