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