面试官:关于Window,你了解多少呢?
看看下面这些问题你都能答上来吗。
如果你遇到这些问题
-
Window是什么?和View的关系?
-
WindowManager是什么?和WMS的关系?
-
怎么添加一个Window?
-
Window怎样可以显示到锁屏界面
-
Window三种类型都存在的情况下,显示层级是怎样。
-
Window就是指PhoneWindow吗?
-
PhoneWindow什么时候被创建的?
-
要实现可以拖动的View该怎么做?
-
Window的添加、删除和更新过程。
-
Activity、PhoneWindow、DecorView、ViewRootImpl 的关系?
-
Window中的token是什么,有什么用?
-
Application中可以直接弹出Dialog吗?
-
关于事件分发,事件到底是先到DecorView还是先到Window的?
Window是什么
窗口。你可以理解为手机上的整个画面,所有的视图都是通过Window呈现的,比如Activity、dialog
都是附加在Window上的。Window类的唯一实现是PhoneWindow
,这个名字就更加好记了吧,手机窗口呗。
那Window
到底在哪里呢?我们看到的View是Window吗?是也不是。
-
如果说的只是
Window概念
的话,那可以说是的,View就是Window的存在形式,Window管理着View。 -
如果说是
Window类
的话,那确实不是View,唯一实现类PhoneWindow管理着当前界面上的View,包括根布局——DecorView,和其他子view的添加删除等等。
不知道你晕没有,我总结下,Window
是个概念性的东西,你看不到他,如果你能感知它的存在,那么就是通过View,所以View是Window的存在形式,有了View,你才感知到View外层有一个皇帝的新衣
——window。
WindowManager是什么?和WMS的关系?
WindowManager
就是用来管理Window的,实现类为WindowManagerImpl
,实际工作会委托给WindowManagerGlobal
类中完成。
而具体的Window操作,WM会通过Binder
告诉WMS,WMS做最后的真正操作Window的工作,会为这个Window分配Surface,并绘制到屏幕上。
怎么添加一个Window?
var windowParams: WindowManager.LayoutParams = WindowManager.LayoutParams()
windowParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
windowParams.type = WindowManager.LayoutParams.TYPE_SYSTEM_DIALOG
var btn = Button(this)
windowManager.addView(btn, windowParams)
简单贴了下代码,加了一个Button。
有的朋友可能会疑惑了,这明明是个Button,是个View啊,咋成了Window?
刚才说过了,View是Window的表现形式,在实际实现中,添加window其实就是添加了一个你看不到的window,并且里面有View才能让你感觉得到这个是一个Window。
所以通过windowManager
添加的View其实就是添加Window的过程。
这其中还有两个比较重要的属性:flags和type
,下面会依次说到。
Window怎样可以显示到锁屏界面
Window的flag可以控制Window
的显示特性,也就是该怎么显示、touch事件处理、与设备的关系、等等。所以这里问的锁屏界面显示也是其中的一种Flag。
// Window不需要获取焦点,也不接受各种输入事件。
public static final int FLAG_NOT_FOCUSABLE = 0x00000008;
// @deprecated Use {@link android.R.attr#showWhenLocked} or
// {@link android.app.Activity#setShowWhenLocked(boolean)} instead to prevent an
// unintentional double life-cycle event.
// 窗口可以在锁屏的 Window 之上显示
public static final int FLAG_SHOW_WHEN_LOCKED = 0x00080000;
Window三种类型都存在的情况下,显示层级是怎样。
Type表示Window的类型,一共三种:
-
应用Window
。对应着一个Activity,Window层级为1~99,在视图最下层。 -
子Window
。不能单独存在,需要附属在特定的父Window之中(如Dialog就是子Window),Window层级为1000~1999。 -
系统Window
。需要声明权限才能创建的Window,比如Toast和系统状态栏,Window层级为2000-2999,处在视图最上层。
可以看到,区别就是有个Window层级(z-ordered),层级高的能覆盖住层级低的,离用户更近。
Window就是指PhoneWindow吗?
如果有人问我这个问题,我肯定心里要大大的疑惑了