Window添加悬浮窗解析

  • WindowManager获取
    window作为一种视图抽象承载者,唯一的实现类是PhoneWindow,PhoneWindow中包含一个视图结构DecorView(FrameLayout 包含Title布局) 该View便是我们会addView时的视图根布局 包含结构如下图:

    添加Window时代码如下:
    WindowManager windowManager = (WindowManager) activity.getSystemService(Context.WINDOW_SERVICE);WindowManager windowManager = (WindowManager) getApplicationContext().getSystemService(Context.WINDOW_SERVICE);

上图中第一种获取的WindowManager是当前Activity窗口管理类WindowManagerImp,这个窗口管理实现类会有多个每个Activirty都会包含一个该实例,Activity销毁 该窗口管理器包含的视图也将销毁 。第二种是通过获取全局应用级别的WindowManagerImpl实现类 ,该窗口管理器包含的窗口个和Application的生命一样长
下面是Activty获取WindowManager的实现:

Activity.class@Overridepublic Object getSystemService(@ServiceName @NonNull String name) {if (getBaseContext() == null) {throw new IllegalStateException("System services not available to Activities before onCreate()");}if (WINDOW_SERVICE.equals(name)) {  return mWindowManager; // 返回当前Acticvty的成员变量mWindowManager} else if (SEARCH_SERVICE.equals(name)) {ensureSearchManager();return mSearchManager;}return super.getSystemService(name);}

该mWindowManager是在ActivityThread采用反射加载Activity之后调用Activity.attach()方法时,将当前Activity的PhoneWindow与WindowManager关联

final void attach(Context context, ActivityThread aThread,Instrumentation instr, IBinder token, int ident,Application application, Intent intent, ActivityInfo info,CharSequence title, Activity parent, String id,NonConfigurationInstances lastNonConfigurationInstances,Configuration config, String referrer, IVoiceInteractor voiceInteractor,Window window, ActivityConfigCallback activityConfigCallback) {attachBaseContext(context);...mWindow.setWindowManager((WindowManager)context.getSystemService(Context.WINDOW_SERVICE),   //关联WindowManagerImpl   mToken, mComponent.flattenToString(),(info.flags & ActivityInfo.FLAG_HARDWARE_ACCELERATED) != 0);if (mParent != null) {mWindow.setContainer(mParent.getWindow());}mWindowManager = mWindow.getWindowManager();// 将当前WindowManagerImpl赋值mCurrentConfig = config;mWindow.setColorMode(info.colorMode);setAutofillCompatibilityEnabled(application.isAutofillCompatibilityEnabled());enableAutofillCompatibilityIfNeeded();}

继续看关联WindowManager代码:

   public void setWindowManager(WindowManager wm, IBinder appToken, String appName,boolean hardwareAccelerated) {mAppToken = appToken;mAppName = appName;mHardwareAccelerated = hardwareAccelerated|| SystemProperties.getBoolean(PROPERTY_HARDWARE_UI, false);if (wm == null) {wm = (WindowManager)mContext.getSystemService(Context.WINDOW_SERVICE); // 获取WindowManagerService}mWindowManager = ((WindowManagerImpl)wm).createLocalWindowManager(this);  // 创建一个WindowManagerImpl}

对于使用Activity上下文获取的WindowManager是专门管理当前Activity得窗口实例 ,当前界面销毁时 ,该管理器管理的界面也都要回收销毁, 这个也就是Dialog和Popwindow这种子窗口的实现方式
而第二种方式是通过获取Aplication的上下文从而调用ContextWrap的getSystemService,因为Context的实现采用桥接设计模式将所有ContextWrap的实现交给ContextImpl来执行 创建application的代码:

ActivityThread.class  - > makeApplication
ContextImpl appContext = ContextImpl.createAppContext(mActivityThread, this);app = mActivityThread.mInstrumentation.newApplication(cl, appClass, appContext);

看出application的方法就是ContextImpl实现

ContextImpl.class@Overridepublic Object getSystemService(String name) {return SystemServiceRegistry.getSystemService(this, name);}

SystemServiceRegistry这个类是用来管理所有的系统Service的获取类,内部包含一个静态代码块,实例化系统服务的客户端调用的代理

SystemServiceRegistry.classstatic{...registerService(Context.WINDOW_SERVICE, WindowManager.class,new CachedServiceFetcher<WindowManager>() {@Overridepublic WindowManager createService(ContextImpl ctx) {return new WindowManagerImpl(ctx);}});
...
}

因此第二种获取WindowManager才是作为与Applicatuon的全局唯一的WindowManagerImpl

  • Window与WindowManagerService交互
    每WinowManagerImpl实例内部对View的操作也是通过获取WindowManagerGlobal这个类的单例来实现实现,WindowManagerGlobal内部将ViewRootImpl、view 、 windowmanager.params、刚才销毁的View存储起来:
WindowManagerGlobal.class private final ArrayList<View> mViews = new ArrayList<View>();private final ArrayList<ViewRootImpl> mRoots = new ArrayList<ViewRootImpl>();private final ArrayList<WindowManager.LayoutParams> mParams =new ArrayList<WindowManager.LayoutParams>();private final ArraySet<View> mDyingViews = new ArraySet<View>();

WindowManagerGlobal(后面统称wmg)通过Binder获取wms,然后wms同样通过Binder通信获取 IWindowSession.stub 的实现类Session ,Session类的注释:

Session与windowManager协同管理每个app的窗口交互,每个app拥有一个Session和一个全局的WindowManagerGlobal ,
在wmg的setView以及update等View的操作又统统交给ViewRootImpl来实现,ViewRoorImpl在操作View展示以及跟新时调用wmg的静态方法获取IwindowSession.Stub实例,ViewRootImpl通过IWindowSession与WMS进行交互,IWindowSession定义在IWindowSession.aidl文件中,而ViewRootImpl的内部类 class W extends IWindow.Stub 作为在WMS操作后利用Binder通信的响应类,主要用来通知ViewRootImpl来进行View的调整等信息。

  • Window类型
  1. WindowManager.LayoutParams.TYPE_TOAST
    Toast类型的窗口 由NMS管理 Android4.4之前该类型的窗口系统为其添加两个flags WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE | WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE 表明这种类型窗口无法获取焦点与交互行为 4.4 之后该类型的Toast取消了以上两个flag 而且这种类型窗口不需要SYSTEM_ALERT_WINDOW权限的申明 直到6.0 google也没有解决这个类似bug的漏洞,但是7.1 google 添加了一个限制 就是每一个Uid只能弹出一个该类型的窗口 不能重叠弹出 (由于国内rom定制需要具体分析 )8.0 时google为了解决这些悬浮窗问题不用户用户使用TYPE_TOAST 、TYPE_PHONE、TYPE_SYSTEM_ALERT 如果设置为该类型会直接奔溃 必须修改为 TYPE_APPLICATION_OVERLAY

  2. WindowManager.LayoutParams.TYPE_PHONE(窗口类型和手机来电窗口级别一致)WindowManager.LayoutParams.TYPE_SYSTEM_ALERT(该窗口类型和与低电量系统提示窗口级别一致 )
    在6.0以前只需在Manifest中注册SYSTEM_ALERT_WINDOW权限即可,6.0以后可以通过Setting.canDrawOverLays来判断该权限授予情况 则需要条状到设置页面手动授权,6.0以前国内厂商可能也需要动态去授权改权限

  3. TYPE_APPLICATION_OVERLAY
    该窗口类型是Android8.0以后新增 为了统一之前版本对于用户随意使用悬浮窗导致混乱的情况 该窗口类型需要通过Setting.canDrawOverLays判断是否授权 如果未授权 通过一下代码跳转到允许悬浮窗权限界面
    Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION, Uri.parse(“package:” + this.getPackageName()));
    startActivityForResult(intent, OVERLAY_PERMISSION_REQ_CODE);

Window添加悬浮窗解析相关推荐

  1. Java安卓如何添加悬浮窗_Android悬浮窗的实现

    Android悬浮窗的实现 *本篇文章已授权微信公众号 guolin_blog (郭霖)独家发布 0. 前言 现在很多应用都使用到悬浮窗,例如微信在视频的时候,点击Home键,视频小窗口仍然会在屏幕上 ...

  2. Java安卓如何添加悬浮窗_Android桌面悬浮窗效果实现

    360手机卫士我相信大家都知道,好多人手机上都会装这一款软件,那么我们对它的一个桌面悬浮窗效果想必都不会陌生.请看下图: 首先是一个小的悬浮窗显示的是当前使用了百分之多少的内存,点击一下小悬浮窗,就会 ...

  3. android添加悬浮窗,在Android中使用WindowManager实现悬浮窗口

    最近在做一个项目的过程中,客户提出要一个能悬浮于所有应用的界面.下图右侧这个界面就是需要实现的悬浮于所有应用之上的视图. 下面就介绍一下如何通过WindowManager来实现这个效果. 通过Wind ...

  4. Android悬浮窗原理解析(Window)[源码]

    悬浮窗,在大多数应用中还是很少见的,目前我们接触到的悬浮窗,差不多都是一些系统级的应用软件,例如:360安全卫士,腾讯手机管家等:在某些服务行业如金融,餐饮等,也会在应用中添加悬浮窗,例如:美团的偷红 ...

  5. 项目中的一个技术方案替换历程(surfaceview+fragment 变成悬浮窗window)

    背景: 项目中UI层有SurfaceView,其渲染展示的是摄像机等采集画面,但是测试提了一个问题单,如果在当前页面中跳出到其他页面,会crash,经过log分析,是由于surfaceview 在失去 ...

  6. Android窗口Window的创建(悬浮窗)

    创建悬浮窗以及基于无障碍服务的窗口 关于悬浮窗的创建 启动悬浮窗 关于前台服务 启动服务 无障碍窗口 知识点 参考链接 关于悬浮窗的创建 首先需要获取WindowManager WindowManag ...

  7. 电脑桌面悬浮窗记事本_安卓悬浮窗便签记事本怎么添加?求安卓手机悬浮窗便签...

    便签记事本对于我而言,真的是太重要了,如果在日常工作中没有敬业签悬浮窗便签记事本软件的话,我很难想象每天要处理的繁重工作该怎么解决,尤其是在外出办公时,少了这一款悬浮窗便签记事本的话,我的工作就会乱如 ...

  8. android 微信浮窗实现_Android实现类似qq微信消息悬浮窗通知功能

    实现方法:(需要开启悬浮窗通知权限.允许应用在其他应用上显示) 一.利用headsup 悬挂式Notification,他是5.0中新增的,也就是API中的Headsup的Notification,可 ...

  9. Android 悬浮窗功能的实现

    前言 我们大多数在两种情况下可以看到悬浮窗,一个是视频通话时的悬浮窗,另一个是360卫士的悬浮球,实现此功能的方式比较多,这里以视频通话悬浮窗中的需求为例.编码实现使用Kotlin.Java版本留言邮 ...

最新文章

  1. bs架构与cs架构的区别_Oracle vs Mysql--架构、sql查询执行流程及SQL解析顺序区别说明...
  2. np实现sigmoid_使用numpy实现一个深度学习框架
  3. xmanager 开启X11转发失败问题解决
  4. Struts2 访问web元素
  5. Angular jasmine单元测试框架里使用it函数定义single spec
  6. ACM中java快速入门
  7. cordova 更改app版本_【ios马甲包cps联运】App上架难 马甲包不知道该怎么做?
  8. CSS3单词及属性大全
  9. android content provider线程安全,Android ContentProvider的线程安全(二)
  10. centos7部署两个mysql_基于Canal和Kafka实现MySQL的Binlog近实时同步
  11. 自动控制原理第3章——时域分析法(思维导图)
  12. 图片无损放大器有什么软件推荐?这个不要错过
  13. C++ 应用程序性能优化 -- 内存池
  14. 程序员的九阳神功,学会逆天改命!
  15. Springboot物理地址映射和Nginx静态资源代理实现前端上传并访问服务器图片
  16. python shapely
  17. Teams登录报错最全的解决方法ERROR CAA20002 caa70004
  18. 蓝桥杯-基础练习-特殊回文数
  19. 服务器本地视频怎么网站播放视频教程,远程服务器的视频怎么在本地播放
  20. python中不等于号_python的不等于号是什么

热门文章

  1. sata3G 540MB/460M
  2. 【一文读懂】Java代理模式
  3. windows 8 应用商店揭秘(推荐)
  4. ios c语言与windows,微软Win10直接运行iOS应用解读
  5. Bilibili视频下载器
  6. Flutter - 这么炫酷的App你见过吗?
  7. 用C语言将16个数存放到40H中,2012年微型计算机原理与接口技术自考题模拟(1)...
  8. Elsevier期刊LaTeX投稿过程中注意的问题
  9. 一键部署tinyproxy脚本
  10. 2‘13寸墨水屏时钟diy教程