【Android】Window和WindowManager

2023-12-02 15:12

本文主要是介绍【Android】Window和WindowManager,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

    • 理解Window和WindowManager
      • Window和WindowManager
      • Window的内部机制
        • Window的添加过程
        • Window的删除过程
        • Window的更新过程
    • Window的创建过程
      • Activity的Window创建过程
      • Dialog的Window创建过程
      • Toast的Window创建过程

理解Window和WindowManager

Window是一个抽象类,它的具体实现是PhoneWindow。WindowManager是外界访问Window的入口,Window的具体实现位于WindowManagerService中,WindowManager和WindowManagerService的交互是一个IPC过程。Android中所有的视图都是通过Window来呈现的,不管是Activity、Dialog还是Toast,它们的视图实际上都是附加在Window上的,因此Window实际是View的直接管理者。

Window和WindowManager

为了分析Window的工作机制,先通过代码了解如何使用WindowManager添加一个Window,下面一段代码将一个Button添加到屏幕坐标为(100, 300)的位置上

mFloatingButton = new Button(this);
mFloatingButton.setText("test button");
mLayoutParams = new WindowManager.LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT, 0, 0, PixelFormat.TRANSPARENT);//0,0 分别是type和flags参数
mLayoutParams.flags = LayoutParams.FLAG_NOT_TOUCH_MODAL | LayoutParams.FLAG_NOT_FOCUSABLE | LayoutParams.FLAG_SHOW_WHEN_LOCKED;
mLayoutParams.type = LayoutParams.TYPE_SYSTEM_ERROR;
mLayoutParams.gravity = Gravity.LEFT | Gravity.TOP;
mLayoutParams.x = 100;
mLayoutParams.y = 300;
mFloatingButton.setOnTouchListener(this);
mWindowManager.addView(mFloatingButton, mLayoutParams);

Flags参数表示Window的属性,以下列举常用的选项:

  • FLAG_NOT_FOCUSABLE:表示Window不需要获取焦点,也不需要接收各种输入事件,此标记会同时启动FLAG_NOT_TOUCH_MODEL,最终事件会传递给下层的具有焦点的Window
  • FLAG_NOT_TOUCH_MODAL:在此模式下,系统会将当前Window区域以外的单击事件传递给底层的Window,当前Window区域以内的单击事件则自己处理。这个标记很重要,一般来说都需要开启此标记,否则其他Window将无法收到单击事件。
  • FLAG_SHOW_WHEN_LOCKED:开启此模式可以让显示在锁屏的界面

Type参数表示Window的类型,Window有三种类型,分别是应用Window子Window系统Window

  • 应用类Window对应着一个Activity。
  • 子Window不能单独存在,它需要附属在特定的父Window之中,比如常见的一些Dialog就是一个子Window。
  • 系统Window是需要声明权限才能创建的Window,比如Toast和系统状态栏这些都是系统Window。

Window是分层的,每个Window都有对应的z-ordered,层级最大的会覆盖在层级小的Window上面,这和HTML中的z-index的概念是完全一致的。在三类Window中,应用Window的层级范围是199,子Window的层级范围是10001999,系统Window的层级范围是2000~2999,这些层级属性范围对应着WindowManager.LayoutParamstype参数。

如果采用TYPE_SYSTEM_ERROR,只需要为type参数指定这个层级即可:

mLayoutParams.type = LayoutParams.TYPE_SYSTEM_ERROR

同时声明权限:<uses-permissionandroid:name="android.permission.SYSTEM_ALERT_WINDOW" />

WindowManager所提供的功能很简单,常用的只有三个方法,即添加View更新View删除View,这三个方法定义在ViewManager中,而WindowManager继承了ViewManager。

WindowManager操作Window的过程更像是在操作Window中的View

Window的内部机制

Window是一个抽象的概念,并不是实际存在的,它是以View的形式存在,每一个Window都对应着一个View和一个ViewRootImpl,Window和View通过ViewRootImpl来建立联系。在实际使用中无法直接访问Window,对Window的访问必须通过WindowManager。

Window的添加过程

Window的添加过程需要通过WindowManageraddView()来实现,WindowManager是一个接口,它的真正实现是WindowManagerImpl类。WindowManager的实现类对于addView()updateView()removeView()方法都是委托给WindowManagerGlobal类。

WindowManagerGlobal的addView()方法分为如下几步:

  1. 检查参数是否合法,如果是子Window那么还需要调整一些布局参数
  2. 创建ViewRootImpl并将View添加到列表中。Window对应的View, ViewRootImpl和待删除的View对象都有对应的列表。
  3. 通过ViewRootImplsetView()来更新界面(View的绘制由ViewRootImpl完成),setView()中通过requestLayout()完成异步刷新请求
  4. 通过WindowSession来添加Window,添加过程的本质是一个IPC过程,其中用到了Binder对象IWindowSession(实现类Session),实际添加是交给WindowManagerService去处理
Window的删除过程

和添加过程一样,都是先通过WindowManagerImpl后,再进一步通过WindowManagerGlobal来实现的↓

调用removeView(),其中先找到待删除的View索引,然后调用removeViewLocked()->ViewRootImpl,有同步删除和异步删除,在异步删除中,就会发送一个信息,放进刚刚ViewRootImpl中的待删除View的列表。

真正删除View的逻辑在dispatchDetachedFromWindow()方法的内部实现。主要做四件事:

  1. 垃圾回收的工作,比如清除数据和消息,移除回调。
  2. 通过Session的remove方法删除Window,mWindowSession.remove(mWindow),这同样是一个IP C过程,最终会调用WindowManagerService的removeWindow方法
  3. 调用View的dispatchDetachedFromWindow方法,在内部调用View的onDetachedFromWindow()以及onDetachedFromWindowInternal()。
  4. 调用WindowManagerGlobal的doRemoveView方法刷新数据,包括mRoots、mParams以及mDyingViews,需要将当前Window所关联的这三类对象从列表中删除。
Window的更新过程

调用WindowManagerGlobal 的updateViewLayout():

首先需要更新View的LayoutParams并替换掉老的LayoutParams,接着再更新ViewRootImpl中的LayoutParams,这一步是通过ViewRootImpl的setLayoutParams()方法来实现的。在ViewRootImpl中会通过scheduleTrversals方法来对View重新布局,包括测量、布局、重绘三个过程。除了View本身的重绘以外,ViewRootImpl还会通过WindowSession来更新Window的视图,这个过程最终是由WindowManagerService的relayoutWindow()来具体实现的,同样是一个IPC过程。

Window的创建过程

Activity的Window创建过程

1、Activity的启动过程很复杂,最终会由ActivityThread中的performLaunchActivity()来完成整个启动过程,在这个方法内部会通过类加载器创建Activity的实例对象,并调用其attach()方法为其关联运行过程中所依赖的一系列上下文环境变量。

在attach()中,系统会创建所属的Window对象并设置回调接口。

Window对象的创建是通过PolicyManagermakeNewWindow() 实现

2、Activity实现了Window的Callback接口,当Window接收到外界的状态变化时就会调用Activity的方法,例如onAttachedToWindow()onDetachedFromWindow()dispatchTouchEvent()等。

3、Activity的Window是由PolicyManager来创建的 - > 真正实现是Policy类,它会新建一个PhoneWindow对象,Activity的setContentView()的实现是由PhoneWindow来实现的/

PhoneWindowsetContentView()方法大致遵循如下几个步骤:

  1. 如果没有DecorView(FrameLayout,顶级View,包含内容和标题栏),那么就创建它,通过gernerateLayout()加载具体的布局文件。
  2. 将View添加到DecorView的mContentParent中,
  3. 回调Activity的onCreateChanged()方法通知Activity视图已经发生改变

Window更多表示的是一种抽象的功能集合…

Dialog的Window创建过程

Dialog的Window的创建过程和Activity类似,有如下步骤:

  1. 创建Window:Diolog中Window的创建同样是通过PolicyManager的makeNewWindow()方法来完成的,创建后的对象实际上就是PhoneWindow。
  2. 初始化DecorView并将Dialog的视图添加到DecorView中,与Activity类似。
  3. 调用Dialog的show(),将DecorView添加到Window中并显示, 也与Activity类似。
  4. 普通的Dialog有一个特殊之处,就是必须采用Activity的Context,如果采用Application的Context,那么就会报错 -> 应用token只有Activity拥有,所以这里只需要Activity作为Context来显示对话框即可。

系统Window比较特殊,不需要token,系统Window的层级范围type: 2000~2999,可以指定Dialog的Window类型为系统Window。

dialog.getWindow().setType(LayoutParams.TYPE_SYSTEM_ERROR);
//要声明权限

Toast的Window创建过程

  • 在Toast的内部有两类IPC过程,第一类是Toast访问NotificationManagerService,第二类是NotificationManagerService回调Toast里的TN接口。
  • Toast属于系统Window,它内部的视图由两种方式指定:一种是系统默认的演示,另一种是通过setView方法来指定一个自定义的View
  • Toast具有定时取消功能,所以系统采用了Handler。
  • Toast的显示show()和隐藏cancel()是IPC过程,都需要NotificationManagerService(NMS)来实现,在Toast和NMS进行IPC过程时,NMS会跨进程回调Toast中的TN类中的方法,TN类是一个Binder类,运行在Binder线程池中,所以需要通过Handler将其切换到当前发送Toast请求所在的线程,因为使用了Handler,所以Toast无法在没有Looper的线程中弹出。

对于非系统应用来说,mToastQueue最多能同时存在50个ToastRecord(应用的mToastQueue队列最多只能存在50个),这样做是为了防止DOS(Denial of Service,拒绝服务)。因为如果某个应用弹出太多的Toast会导致其他应用没有机会弹出Toast。

  • Toast的显示是通过ToastRecord的callback来完成,callback -> Tn对象的远程Binder,需要跨进程,会运行在Toast的应用的Binder线程池中。

  • 并且会发送一个延时消息,时长取决于Toast的持续时长,用来调用cancelToastLocked来隐藏Toast,并且从队列中移除,然后继续显示队列中的Toast

Toast的显示和隐藏实际上通过Toast的TN类中的两个Runnable --> handleShow()handleHide() 方法,用于将Toast的视图从Window添加或者移除。

这篇关于【Android】Window和WindowManager的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/445809

相关文章

Android数据库Room的实际使用过程总结

《Android数据库Room的实际使用过程总结》这篇文章主要给大家介绍了关于Android数据库Room的实际使用过程,详细介绍了如何创建实体类、数据访问对象(DAO)和数据库抽象类,需要的朋友可以... 目录前言一、Room的基本使用1.项目配置2.创建实体类(Entity)3.创建数据访问对象(DAO

Android WebView的加载超时处理方案

《AndroidWebView的加载超时处理方案》在Android开发中,WebView是一个常用的组件,用于在应用中嵌入网页,然而,当网络状况不佳或页面加载过慢时,用户可能会遇到加载超时的问题,本... 目录引言一、WebView加载超时的原因二、加载超时处理方案1. 使用Handler和Timer进行超

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

Android平台播放RTSP流的几种方案探究(VLC VS ExoPlayer VS SmartPlayer)

技术背景 好多开发者需要遴选Android平台RTSP直播播放器的时候,不知道如何选的好,本文针对常用的方案,做个大概的说明: 1. 使用VLC for Android VLC Media Player(VLC多媒体播放器),最初命名为VideoLAN客户端,是VideoLAN品牌产品,是VideoLAN计划的多媒体播放器。它支持众多音频与视频解码器及文件格式,并支持DVD影音光盘,VCD影

android-opencv-jni

//------------------start opencv--------------------@Override public void onResume(){ super.onResume(); //通过OpenCV引擎服务加载并初始化OpenCV类库,所谓OpenCV引擎服务即是 //OpenCV_2.4.3.2_Manager_2.4_*.apk程序包,存

从状态管理到性能优化:全面解析 Android Compose

文章目录 引言一、Android Compose基本概念1.1 什么是Android Compose?1.2 Compose的优势1.3 如何在项目中使用Compose 二、Compose中的状态管理2.1 状态管理的重要性2.2 Compose中的状态和数据流2.3 使用State和MutableState处理状态2.4 通过ViewModel进行状态管理 三、Compose中的列表和滚动

Android 10.0 mtk平板camera2横屏预览旋转90度横屏拍照图片旋转90度功能实现

1.前言 在10.0的系统rom定制化开发中,在进行一些平板等默认横屏的设备开发的过程中,需要在进入camera2的 时候,默认预览图像也是需要横屏显示的,在上一篇已经实现了横屏预览功能,然后发现横屏预览后,拍照保存的图片 依然是竖屏的,所以说同样需要将图片也保存为横屏图标了,所以就需要看下mtk的camera2的相关横屏保存图片功能, 如何实现实现横屏保存图片功能 如图所示: 2.mtk

android应用中res目录说明

Android应用的res目录是一个特殊的项目,该项目里存放了Android应用所用的全部资源,包括图片、字符串、颜色、尺寸、样式等,类似于web开发中的public目录,js、css、image、style。。。。 Android按照约定,将不同的资源放在不同的文件夹中,这样可以方便的让AAPT(即Android Asset Packaging Tool , 在SDK的build-tools目

Android fill_parent、match_parent、wrap_content三者的作用及区别

这三个属性都是用来适应视图的水平或者垂直大小,以视图的内容或尺寸为基础的布局,比精确的指定视图的范围更加方便。 1、fill_parent 设置一个视图的布局为fill_parent将强制性的使视图扩展至它父元素的大小 2、match_parent 和fill_parent一样,从字面上的意思match_parent更贴切一些,于是从2.2开始,两个属性都可以使用,但2.3版本以后的建议使

Android Environment 获取的路径问题

1. 以获取 /System 路径为例 /*** Return root of the "system" partition holding the core Android OS.* Always present and mounted read-only.*/public static @NonNull File getRootDirectory() {return DIR_ANDR