安卓车载多屏互动副屏底部有黑线条NavigationBar分析

背景:

在学习了马哥的wms和多屏互动课程后,大家普遍都可以跟着做出如下图效果的多屏互动:

其实初略来看这个成果已经完成一个多屏互动项目大部分功能,但是其实还是有一些bug的存在,今天我们就来分析一下多屏互动相关的bug补充哈。

副屏Activity底部有黑线条分析

虚拟屏幕下下面的所有Activity在底部都会显示一个黑线条,具体如图展示:

明显第二屏幕的底部黑线看起来很诡异,下面这个黑线条其实是我们不需要的,这里就需要分析一下为啥会有这个黑线条

分析思路:

dumpsys SurfaceFlinger分析

在最后合成部分查看副屏display的图层

明显看到了Activity上面还有一个NavigationBar2这个图层,那么接下来其实就好办了,找到这个NavigationBar2窗口添加地方

看看NavigationBar2这个图层对应窗口是谁创建的

bash 复制代码
Window #0 Window{5819104 u0 NavigationBar2}:
    mDisplayId=2 rootTaskId=1140 mSession=Session{4e45c83 740:u0a10109} mClient=android.os.BinderProxy@e1cd996
    mOwnerUid=10109 showForAllUsers=true package=com.android.systemui appop=NONE
    mAttrs={(0,0)(fillx168) gr=BOTTOM CENTER_VERTICAL sim={adjust=pan} layoutInDisplayCutoutMode=always ty=NAVIGATION_BAR fmt=TRANSLUCENT
      fl=NOT_FOCUSABLE NOT_TOUCH_MODAL WATCH_OUTSIDE_TOUCH SPLIT_TOUCH FLAG_SLIPPERY
      pfl=LAYOUT_SIZE_EXTENDED_BY_CUTOUT COLOR_SPACE_AGNOSTIC USE_BLAST FIT_INSETS_CONTROLLED TRUSTED_OVERLAY
      bhv=DEFAULT

明显看到是package=com.android.systemui创建,那么接下来怎么找对应的NavigationBar2的窗口创建呢?

这里马哥老粉丝学员肯定会想到是不是可以打堆栈呢?

这里确实是可以的,因为一般NavigationBar2属于窗口最后肯定是需要执行下面这一部分WindowManager.addView的代码

具体堆栈就不贴了,属于学员们自己要学会的基本功。

最后其实是新屏幕创建准备好了以后会通知systemui下面代码

这里再看看createNavigationBar

这里核心关键可以看到有一个拦截是否要创建NavigationBar,来看看shouldCreateNavBarAndTaskBar方法

cpp 复制代码
 private boolean shouldCreateNavBarAndTaskBar(Context context, int displayId) {
        final IWindowManager wms = WindowManagerGlobal.getWindowManagerService();
				//主屏幕调用是setting值
        if (displayId == mDisplayTracker.getDefaultDisplayId() &&
                LineageSettings.System.getIntForUser(context.getContentResolver(),
                        LineageSettings.System.FORCE_SHOW_NAVBAR, 0,
                        UserHandle.USER_CURRENT) == 1) {
            return true;
        }
        //非主屏幕就调用wms的hasNavigationBar接口
        try {
            return wms.hasNavigationBar(displayId);
        } catch (RemoteException e) {
            // Cannot get wms, just return false with warning message.
            Log.w(TAG, "Cannot get WindowManager.");
            return false;
        }
    }

下面来看看wms的hasNavigationBar接口:

cpp 复制代码
    public boolean hasNavigationBar(int displayId) {
        synchronized (mGlobalLock) {
            final DisplayContent dc = mRoot.getDisplayContent(displayId);
            if (dc == null) {
                return false;
            }
            return dc.getDisplayPolicy().hasNavigationBar();
        }
    }

这里看看hasNavigationBar方法

就是靠两个标志位来的主要我们来看看mHasNavigationBar,mForceNavbar看了是不满足的

对于副屏主要就是靠mDisplayContent.supportsSystemDecorations()来决定:

可以看到这里的条件还是比较严格的,这里大部分和我们的display创建参数有关系

1、受信任display

2、有设置FLAG_SHOULD_SHOW_SYSTEM_DECORATIONS

(当然也还有其他settings也可以控制,大家可以一一确定,默认一般setting为false)

那么我们dumpsys display确认一下

确实看到创建副屏满足上面条件,到此整个副屏显示一条黑线就清楚,本质就是显示了一个NavigationBar盖在上面,为啥显示NavigationBar是因为副屏创建时候有相关的FLAG让显示及是相关的Trust屏幕

如何修改

原因知道了,那么修改就简单多了,可以直接修改FLAG也好,或者直接暴力修改让所有副屏都不显示NavigationBar也可以,这里其实建议大家可以自己根据副屏的其他特性判断,自己根据情况细分,这里只是为了演示就直接修改成不显示:

修改后看看效果:

明显可以看到副屏已经没有了那个小黑条,即不在副屏创建NavigationBar。

具体详情试看方式:

投屏专题部分:
https://mp.weixin.qq.com/s/IGm6VHMiAOPejC_H3N_SNg

更多framework详细代码和资料参考如下链接

hal+perfetto+surfaceflinger
https://mp.weixin.qq.com/s/LbVLnu1udqExHVKxd74ILg

其他课程七件套专题:

点击这里
https://mp.weixin.qq.com/s/Qv8zjgQ0CkalKmvi8tMGaw

视频试看:
https://www.bilibili.com/video/BV1wc41117L4/

更多framework假威风耗:androidframework007

相关推荐
Jouzzy5 小时前
【Android安全】Ubuntu 16.04安装GDB和GEF
android·ubuntu·gdb
极客先躯5 小时前
java和kotlin 可以同时运行吗
android·java·开发语言·kotlin·同时运行
Good_tea_h8 小时前
Android中的单例模式
android·单例模式
计算机源码社12 小时前
分享一个基于微信小程序的居家养老服务小程序 养老服务预约安卓app uniapp(源码、调试、LW、开题、PPT)
android·微信小程序·uni-app·毕业设计项目·毕业设计源码·计算机课程设计·计算机毕业设计开题
丶白泽13 小时前
重修设计模式-结构型-门面模式
android
晨春计14 小时前
【git】
android·linux·git
标标大人15 小时前
c语言中的局部跳转以及全局跳转
android·c语言·开发语言
竹林海中敲代码16 小时前
Qt安卓开发连接手机调试(红米K60为例)
android·qt·智能手机
木鬼与槐16 小时前
MySQL高阶1783-大满贯数量
android·数据库·mysql
iofomo16 小时前
【Abyss】Android 平台应用级系统调用拦截框架
android·开发工具·移动端