查看源码会报红点不进去的话
- 先学些下系统架构,用的android.jar其实在Framework层,查看系统架构官网
- 如果就想看下源码,去androidxref根据类名啥的搜源码查看
- 或者下载这个仓库里提供的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()过程
-
检查参数是否合法,如果是子view就调整一些布局参数
创建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>();
- 通过ViewRootImpl将view添加到window,再通过windowSession(Binder对象)完成window的添加【这是一个IPC调用的过程】windowSession会在调用WindowManageService(WMS)添加view。
window删除
通过WindowmanageGlobal查找mViews数组,通过
viewRootImpl.die(){
doDie(){
dispatchDetachedFromWindow()
}
}
viewRootImpl.dispatchDetachedFromWindow()过程
- 垃圾回收,比如清除数据、消息、回调
- 通过windowSession完成删除、和添加一样,windwoSession再调用WindowManageService删除
- 调用view.dispatchDetachedFromWindow()
- 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() //入口
acitvityClientRecord中获取待启动的activity的组件信息
创建activity = Instrumentation.newActivity() //Instumentation实用类加载器创建对象
调用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里面的逻辑
- 如果没有DecorView----->创建。 //decorview,activity的顶层view
installDecor()--->generateDecor()- 初始化DecorView的布局
PhoneWindow.generateLayout()- 将view通过inflater添加到DecorView的mContentParent中
- 回调activity.onContentChanged() 通知activity试图变化
- 调用activity.onResume()使Activity显示
Dialog的window创建过程
- PolicyManager.makeNewWindow()
- 初始化 DecorView,并且把Dialog的view添加到DecorView
- 调用Dialog.show()时,显示。 //这里是通过WindowManager把decorView添加到window中的
Toast的window创建过程
因为Toast有定时取消的功能,用了Handler来实现,所以toast内部有两类IPC过程,
- Toast 访问 NotificationManagerService // getService()
- NotificationManagerService 回调Toast的TN接口,用来显示、隐藏。NMS运行在系统的进程里,TN是一个运行在Binder线程池的对象,NMS调用TN,TN调用使用handler把显示隐藏时间消息切换到当前线程。
IPC:进程间通信那就是下一篇的内容了