Window

作者: 简安理思 | 来源:发表于2022-02-28 17:57 被阅读0次

查看源码会报红点不进去的话

  1. 先学些下系统架构,用的android.jar其实在Framework层,查看系统架构官网
  2. 如果就想看下源码,去androidxref根据类名啥的搜源码查看
  3. 或者下载这个仓库里提供的Android.jar,替换你本地的,可以解决一大部分代码跳不进去的问题

window

除了在实现悬浮窗的时候具体使用到了,其他时候都是作为中间件实现activity,dialog,toast的view显示在屏幕上,window是view的直接管理者,也是事件分发的一部分。

WindowManager

WindowManager是外界访问Window的入口,Window的具体实现是WindowManagerService里,通过WindowManager可以创建Window,添加、更新、删除view。

Window内部机制

window添加

通过Windowmanager.addView() //实际是实现类WindowManagerImpl调用WindowManageGlobal.addview()

WindowManageGlobal.addview()过程

  1. 检查参数是否合法,如果是子view就调整一些布局参数


    第一步
  2. 创建viewRootImp并且把view添加到列表

mViews.add(view); //view列表,对应window里面的view
mRoots.add(root); //view的viewRootImpl对象
mParams.add(wparams); //view对应的LayoutParam
  • WindowManageGlobal里还有一个列表mDyingViews,是用来装被移除,但是移除过程还没有走完的view
ArraySet<View> mDyingViews = new ArraySet<View>();
  1. 通过ViewRootImpl将view添加到window,再通过windowSession(Binder对象)完成window的添加【这是一个IPC调用的过程】windowSession会在调用WindowManageService(WMS)添加view。

window删除

通过WindowmanageGlobal查找mViews数组,通过

viewRootImpl.die(){
  doDie(){
     dispatchDetachedFromWindow()
  }
}

viewRootImpl.dispatchDetachedFromWindow()过程

  1. 垃圾回收,比如清除数据、消息、回调
  2. 通过windowSession完成删除、和添加一样,windwoSession再调用WindowManageService删除
  3. 调用view.dispatchDetachedFromWindow()
  4. WindowManageGlobal刷新数据

window更新

windowManagerGlobal.updateViewLayout(view,layoutParam);

创建悬浮窗

 private fun createFloatWindow() {
        val button = Button(this)
        button.setBackgroundColor(Color.BLUE)
        val mWindowParam = WindowManager.LayoutParams(
            LayoutParams.WRAP_CONTENT,
            LayoutParams.WRAP_CONTENT,
        0,0,PixelFormat.TRANSPARENT)

        mWindowParam.flags = LayoutParams.FLAG_NOT_TOUCH_MODAL or LayoutParams.FLAG_NOT_FOCUSABLE
        mWindowParam.gravity = Gravity.CENTER or Gravity.TOP
        mWindowParam.x = 100
        mWindowParam.y = 300
        windowManager.addView(button,mWindowParam)
    }
  • 要显示在系统window上,要申请权限SYSTEM_ALERT_WINDOW
  • layoutparam.flag设置window的属性,比如是否可点击可获取焦点等可设置多个。官网FLAG详情说明
  • layoutparam.tag设置window的类型
类型 说明 举例 层级范围
应用window 对应着一个activity - 对应着一个activity
子window 不能单独存在,要附属在特定父window中 dialog 1000-1999
系统window 声明权限才能创建 系统状态栏,toast 2000-2999

activity的window创建过程

ActivityThread.performLaunchActivity() //入口
  1. acitvityClientRecord中获取待启动的activity的组件信息

  2. 创建activity = Instrumentation.newActivity() //Instumentation实用类加载器创建对象

  3. 调用activity.attach()这里通过PolicyManager.makeNewWindow() // 创建window对象(PhoneWindow)并添加window回调接口。

  • window Callback接口中有很多事件回调方法,activity实现了该接口,比如onAttachedToWindow(),dispatchTouchEvent()等

上面三步初始化之后,windowManager还没有识别DecorView。
当ActivityThread.handleResumeActivity()方法中调用

Activity.onResume()方法,然后调用

Activity.makeVisible()方法完成DecorView的添加和显示。

4.activity的view通过setContentView()方法设置,调用了window具体实现类PhoneWindow.setContentView()

setContentView里面的逻辑

  1. 如果没有DecorView----->创建。 //decorview,activity的顶层view
    installDecor()--->generateDecor()
  2. 初始化DecorView的布局
    PhoneWindow.generateLayout()
  3. 将view通过inflater添加到DecorView的mContentParent中
  4. 回调activity.onContentChanged() 通知activity试图变化
  5. 调用activity.onResume()使Activity显示

Dialog的window创建过程

  1. PolicyManager.makeNewWindow()
  2. 初始化 DecorView,并且把Dialog的view添加到DecorView
  3. 调用Dialog.show()时,显示。 //这里是通过WindowManager把decorView添加到window中的

Toast的window创建过程

因为Toast有定时取消的功能,用了Handler来实现,所以toast内部有两类IPC过程,

  1. Toast 访问 NotificationManagerService // getService()
  2. NotificationManagerService 回调Toast的TN接口,用来显示、隐藏。NMS运行在系统的进程里,TN是一个运行在Binder线程池的对象,NMS调用TN,TN调用使用handler把显示隐藏时间消息切换到当前线程。
toast.show()方法

IPC:进程间通信那就是下一篇的内容了

相关文章

网友评论

      本文标题:Window

      本文链接:https://www.haomeiwen.com/subject/vzhyfrtx.html