1.理清概念

我们使用过Dialog和PopupWindow,还有Toast,它们都显示在Activity之上。那么我们首先需要理解的是android中是如何去绘制这些UI的呢?这里我只讲我所理解的,首先看一层次图(盗用网络)

 
首先我们看到左边的Activity层,我们使用到的Activity,其中会有一个PhoneWindow,这个每一个Activity都包含的,并且PhoneActivity去管理一个DecoView根视图,并且这也解释了我们使用的setContentView(int resId)就是将所有的UI组件放入到PhoneWindow中管理。我们也看到DecoView,加载后的资源都放入到这个View中。接下来会看到WindowManager,整个Android的窗口机制是基于一个叫做WindowManager,这个接口可以添加view到屏幕,也可以从屏幕删除view。它面向的对象一端是屏幕,另一端就是View。而我们的android屏幕可以放多个窗口,除了Activity,我们还可以放Dialog,PopupWindow,我研究了一下PopupWIndow,就是通过WindowManager生产的。如果要自定义一个窗口,我们可以不用Dialog和PopupWindow,可以自己来实现它。
 
理解该图,首先思考的是使用的Activity,会有一个PhoneWindow,这个用来管理View。而它的视图是如何被绘制出来的呢,那是WindowManager的事情了,WindowManager就是添加和移除View,从屏幕上。

2.WindowManager介绍

WindowManager的方法很简单,有三个方法,AddView(),removeView(),updateViewLayout();
AddView();    //添加View
removeView();            //移除VIew
updateViewLayout()    //更新View
接下来看一个完整的代码:
  1. View view = LayoutInflater.from(this).inflate(R.layout.dialog,null); //加载View视图,这个就是我们要显示的内容
  2. WindowManager windowManager = (WindowManager) getApplicationContext().getSystemService(Context.WINDOW_SERVICE); //获取WindowManage
  3. WindowManager.LayoutParams layoutParams = new WindowManager.LayoutParams();
  4. //设置LayoutParams的属性
  5. layoutParams.type = WindowManager.LayoutParams.TYPE_APPLICATION_PANEL; //该Type描述的是形成的窗口的层级关系,下面会详细列出它的属性
  6. layoutParams.flags = WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL | //该flags描述的是窗口的模式,是否可以触摸,可以聚焦等
  7. WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE | WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE;
  8. layoutParams.gravity = Gravity.CENTER; //设置窗口的位置
  9. layoutParams.format = PixelFormat.TRANSLUCENT; //不设置这个弹出框的透明遮罩显示为黑色
  10. layoutParams.width = WindowManager.LayoutParams.MATCH_PARENT; //窗口的宽
  11. layoutParams.height = WindowManager.LayoutParams.MATCH_PARENT; //窗口的高
  12. layoutParams.token = ((View)findViewById(R.id.linearlayout)).getWindowToken(); //获取当前Activity中的View中的TOken,来依附Activity,因为设置了该值,纳闷写的这些代码不能出现在onCreate();否则会报错。
  13. windowManager.addView(view,layoutParams);

别忘了在AndroidManifest.xml中添加权限:

  1. <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />

(1)那么接下啦介绍type值

layoutParams.type窗口类型。其中有三种主要类型:
Applicationwindows:
取值在 FIRST_APPLICATION_WINDOW 和 LAST_APPLICATION_WINDOW 之间。
是通常的、顶层的应用程序窗口。必须将 token 设置成 activity 的 token 。
Sub_windows:
取值在 FIRST_SUB_WINDOW 和 LAST_SUB_WINDOW 之间。
与顶层窗口相关联,token 必须设置为它所附着的宿主窗口的 token。
Systemwindows:
取值在 FIRST_SYSTEM_WINDOW 和 LAST_SYSTEM_WINDOW 之间。
用于特定的系统功能。它不能用于应用程序,使用时需要特殊权限。
应用程序窗口。
public static final int FIRST_APPLICATION_WINDOW = 1; 
所有程序窗口的“基地”窗口,其他应用程序窗口都显示在它上面。 
public static final int TYPE_BASE_APPLICATION =1;
//-----------------------------------------------Applicationwindows ------------//
普通应用功能程序窗口。token必须设置为Activity的token,以指出该窗口属谁。
public static final int TYPE_APPLICATION = 2;
用于应用程序启动时所显示的窗口。应用本身不要使用这种类型。
它用于让系统显示些信息,直到应用程序可以开启自己的窗口。 
public static final int TYPE_APPLICATION_STARTING = 3; 
应用程序窗口结束。
public static final int LAST_APPLICATION_WINDOW = 99;
//-----------------------------------------------Applicationwindows ------------//
 
//-----------------------------------------------Sub_windows------------//
子窗口。子窗口的Z序和坐标空间都依赖于他们的宿主窗口。
public static final int FIRST_SUB_WINDOW = 1000;
 
面板窗口,显示于宿主窗口上层  PopupWIndow就是使用的该模式。
public static final int TYPE_APPLICATION_PANEL = FIRST_SUB_WINDOW;
 
媒体窗口,例如视频。显示于宿主窗口下层。
public static final int TYPE_APPLICATION_MEDIA = FIRST_SUB_WINDOW+1;
应用程序窗口的子面板。显示于所有面板窗口的上层。(GUI的一般规律,越“子”越靠上)
public static final int TYPE_APPLICATION_SUB_PANEL = FIRST_SUB_WINDOW +2;
对话框。类似于面板窗口,绘制类似于顶层窗口,而不是宿主的子窗口。
public static final int TYPE_APPLICATION_ATTACHED_DIALOG = FIRST_SUB_WINDOW +3;
媒体信息。显示在媒体层和程序窗口之间,需要实现透明(半透明)效果。(例如显示字幕)
public static final int TYPE_APPLICATION_MEDIA_OVERLAY = FIRST_SUB_WINDOW +4;
子窗口结束。( End of types of sub-windows )
public static final int LAST_SUB_WINDOW = 1999;
//-----------------------------------------------Sub_windows------------//

//-----------------------------------------------Systemwindows  系统窗口------------//

系统窗口。非应用程序创建。
public static final int FIRST_SYSTEM_WINDOW = 2000;
状态栏。只能有一个状态栏;它位于屏幕顶端,其他窗口都位于它下方。
public static final int TYPE_STATUS_BAR = FIRST_SYSTEM_WINDOW;
搜索栏。只能有一个搜索栏;它位于屏幕上方。
public static final int TYPE_SEARCH_BAR = FIRST_SYSTEM_WINDOW+1;
电话窗口。它用于电话交互(特别是呼入)。它置于所有应用程序之上,状态栏之下。
public static final int TYPE_PHONE = FIRST_SYSTEM_WINDOW+2;
系统提示。它总是出现在应用程序窗口之上。
public static final int TYPE_SYSTEM_ALERT = FIRST_SYSTEM_WINDOW +3;
锁屏窗口。
public static final int TYPE_KEYGUARD = FIRST_SYSTEM_WINDOW +4;
信息窗口。用于显示toast。
public static final int TYPE_TOAST = FIRST_SYSTEM_WINDOW +5;
系统顶层窗口。显示在其他一切内容之上。此窗口不能获得输入焦点,否则影响锁屏。
public static final int TYPE_SYSTEM_OVERLAY = FIRST_SYSTEM_WINDOW +6;
电话优先,当锁屏时显示。此窗口不能获得输入焦点,否则影响锁屏。
public static final int TYPE_PRIORITY_PHONE = FIRST_SYSTEM_WINDOW +7;
系统对话框。(例如音量调节框)。
public static final int TYPE_SYSTEM_DIALOG = FIRST_SYSTEM_WINDOW +8;
锁屏时显示的对话框。
public static final int TYPE_KEYGUARD_DIALOG = FIRST_SYSTEM_WINDOW +9;
系统内部错误提示,显示于所有内容之上。
public static final int TYPE_SYSTEM_ERROR = FIRST_SYSTEM_WINDOW +10;
内部输入法窗口,显示于普通UI之上。应用程序可重新布局以免被此窗口覆盖。
public static final int TYPE_INPUT_METHOD = FIRST_SYSTEM_WINDOW +11;
内部输入法对话框,显示于当前输入法窗口之上。
public static final int TYPE_INPUT_METHOD_DIALOG= FIRST_SYSTEM_WINDOW +12;
墙纸窗口。
public static final int TYPE_WALLPAPER = FIRST_SYSTEM_WINDOW +13;
状态栏的滑动面板。
public static final int TYPE_STATUS_BAR_PANEL = FIRST_SYSTEM_WINDOW +14;
系统窗口结束。
public static final int LAST_SYSTEM_WINDOW = 2999;
//-----------------------------------------------Systemwindows------------//

public int flags;

//------------------------------------flag-------------//
行为选项/旗标,默认为 none .
下面定义了 flags 的取值:
窗口之后的内容变暗。
public static final int FLAG_DIM_BEHIND = 0x00000002;

窗口之后的内容变模糊。
public static final int FLAG_BLUR_BEHIND = 0x00000004;

不许获得焦点。
不能获得按键输入焦点,所以不能向它发送按键或按钮事件。那些时间将发送给它后面的可以获得焦点的窗口。此选项还会设置FLAG_NOT_TOUCH_MODAL选项。设置此选项,意味着窗口不能与软输入法进行交互,所以它的Z序独立于任何活动的输入法(换句话说,它可以全屏显示,如果需要的话,可覆盖输入法窗口)。要修改这一行为,可参考FLAG_ALT_FOCUSALBE_IM选项。
public static final int FLAG_NOT_FOCUSABLE = 0x00000008;

不接受触摸屏事件。
public static final int FLAG_NOT_TOUCHABLE = 0x00000010;

当窗口可以获得焦点(没有设置 FLAG_NOT_FOCUSALBE 选项)时,仍然将窗口范围之外的点设备事件(鼠标、触摸屏)发送给后面的窗口处理。否则它将独占所有的点设备事件,而不管它们是不是发生在窗口范围内。
public static final int FLAG_NOT_TOUCH_MODAL = 0x00000020;

如果设置了这个标志,当设备休眠时,点击触摸屏,设备将收到这个第一触摸事件。
通常第一触摸事件被系统所消耗,用户不会看到他们点击屏幕有什么反应。
public static final int FLAG_TOUCHABLE_WHEN_WAKING = 0x00000040;

当此窗口为用户可见时,保持设备常开,并保持亮度不变。
public static final int FLAG_KEEP_SCREEN_ON = 0x00000080;

窗口占满整个屏幕,忽略周围的装饰边框(例如状态栏)。此窗口需考虑到装饰边框的内容。
public static final int FLAG_LAYOUT_IN_SCREEN =0x00000100;

允许窗口扩展到屏幕之外。
public static final int FLAG_LAYOUT_NO_LIMITS =0x00000200;

窗口显示时,隐藏所有的屏幕装饰(例如状态条)。使窗口占用整个显示区域。
public static final int FLAG_FULLSCREEN = 0x00000400;

此选项将覆盖FLAG_FULLSCREEN选项,并强制屏幕装饰(如状态条)弹出。
public static final int FLAG_FORCE_NOT_FULLSCREEN =0x00000800;

抖动。指 对半透明的显示方法。又称“点透”。图形处理较差的设备往往用“点透”替代Alpha混合。
public static final int FLAG_DITHER = 0x00001000;

不允许屏幕截图。
public static final int FLAG_SECURE = 0x00002000;

一种特殊模式,布局参数用于指示显示比例。
public static final int FLAG_SCALED = 0x00004000;

当屏幕有可能贴着脸时,这一选项可防止面颊对屏幕造成误操作。
public static final int FLAG_IGNORE_CHEEK_PRESSES = 0x00008000;

当请求布局时,你的窗口可能出现在状态栏的上面或下面,从而造成遮挡。当设置这一选项后,窗口管理器将确保窗口内容不会被装饰条(状态栏)盖住。
public static final int FLAG_LAYOUT_INSET_DECOR = 0x00010000;

反转FLAG_NOT_FOCUSABLE选项。
如果同时设置了FLAG_NOT_FOCUSABLE选项和本选项,窗口将能够与输入法交互,允许输入法窗口覆盖;
如果FLAG_NOT_FOCUSABLE没有设置而设置了本选项,窗口不能与输入法交互,可以覆盖输入法窗口。
public static final int FLAG_ALT_FOCUSABLE_IM = 0x00020000;

如果你设置了FLAG_NOT_TOUCH_MODAL,那么当触屏事件发生在窗口之外事,可以通过设置此标志接收到一个MotionEvent.ACTION_OUTSIDE事件。注意,你不会收到完整的down/move/up事件,只有第一次down事件时可以收到ACTION_OUTSIDE。
public static final int FLAG_WATCH_OUTSIDE_TOUCH = 0x00040000;

当屏幕锁定时,窗口可以被看到。这使得应用程序窗口优先于锁屏界面。可配合FLAG_KEEP_SCREEN_ON选项点亮屏幕并直接显示在锁屏界面之前。可使用FLAG_DISMISS_KEYGUARD选项直接解除非加锁的锁屏状态。此选项只用于最顶层的全屏幕窗口。
public static final int FLAG_SHOW_WHEN_LOCKED = 0x00080000;

请求系统墙纸显示在你的窗口后面。窗口必须是半透明的。
public static final int FLAG_SHOW_WALLPAPER = 0x00100000;

窗口一旦显示出来,系统将点亮屏幕,正如用户唤醒设备那样。
public static final int FLAG_TURN_SCREEN_ON = 0x00200000;

解除锁屏。只有锁屏界面不是加密的才能解锁。如果锁屏界面是加密的,那么用户解锁之后才能看到此窗口,除非设置了FLAG_SHOW_WHEN_LOCKED选项。
public static final int FLAG_DISMISS_KEYGUARD = 0x00400000;

锁屏界面淡出时,继续运行它的动画。
public static final int FLAG_KEEP_SURFACE_WHILE_ANIMATING =0x10000000;

以原始尺寸显示窗口。用于在兼容模式下运行程序。
public static final int FLAG_COMPATIBLE_WINDOW = 0x20000000;

用于系统对话框。设置此选项的窗口将无条件获得焦点。
public static final int FLAG_SYSTEM_ERROR = 0x40000000;
//------------------------------------flag------------------------------------------------flag-------------//

public int softInputMode;

软输入法模式选项:
以下选项与 softInputMode 有关:
软输入区域是否可见。
public static final int SOFT_INPUT_MASK_STATE = 0x0f;

未指定状态。
public static final int SOFT_INPUT_STATE_UNSPECIFIED = 0;

不要修改软输入法区域的状态。
public static final int SOFT_INPUT_STATE_UNCHANGED = 1;

隐藏输入法区域(当用户进入窗口时)。
public static final int SOFT_INPUT_STATE_HIDDEN = 2;

当窗口获得焦点时,隐藏输入法区域。
public static final int SOFT_INPUT_STATE_ALWAYS_HIDDEN = 3;

显示输入法区域(当用户进入窗口时)。
public static final int SOFT_INPUT_STATE_VISIBLE = 4;

当窗口获得焦点时,显示输入法区域。
public static final int SOFT_INPUT_STATE_ALWAYS_VISIBLE = 5;

窗口应当主动调整,以适应软输入窗口。
public static final int SOFT_INPUT_MASK_ADJUST = 0xf0;

未指定状态,系统将根据窗口内容尝试选择一个输入法样式。
public static final int SOFT_INPUT_ADJUST_UNSPECIFIED = 0x00;

当输入法显示时,允许窗口重新计算尺寸,使内容不被输入法所覆盖。
不可与SOFT_INPUT_ADJUSP_PAN混合使用,如果两个都没有设置,系统将根据窗口内容自动设置一个选项。
public static final int SOFT_INPUT_ADJUST_RESIZE = 0x10;

输入法显示时平移窗口。它不需要处理尺寸变化,框架能够移动窗口以确保输入焦点可见。
不可与SOFT_INPUT_ADJUST_RESIZE混合使用;如果两个都没设置,系统将根据窗口内容自动设置一个选项。
public static final int SOFT_INPUT_ADJUST_PAN = 0x20;

当用户转至此窗口时,由系统自动设置,所以你不要设置它。
当窗口显示之后该标志自动清除。
public static final int SOFT_INPUT_IS_FORWARD_NAVIGATION = 0x100;

public int gravity;

gravity 属性。什么是gravity属性呢?简单地说,就是窗口如何停靠。

9. public float horizontalMargin;
水平边距,容器与widget之间的距离,占容器宽度的百分率。

10. public float verticalMargin;
纵向边距。

11. public int format;
期望的位图格式。默认为不透明。参考android.graphics.PixelFormat。

12. public int windowAnimations;
窗口所使用的动画设置。它必须是一个系统资源而不是应用程序资源,因为窗口管理器不能访问应用程序。

13. public float alpha = 1.0f;
整个窗口的半透明值,1.0表示不透明,0.0表示全透明。

14. public float dimAmount = 1.0f;
当FLAG_DIM_BEHIND设置后生效。该变量指示后面的窗口变暗的程度。
1.0表示完全不透明,0.0表示没有变暗。

15. public float screenBrightness = -1.0f;
用来覆盖用户设置的屏幕亮度。表示应用用户设置的屏幕亮度。
从0到1调整亮度从暗到最亮发生变化。

16. public IBinder token = null;(改属性标识需要去依附的那个View,当弹框不View进行依附后,View被销毁,如果没有对弹框进行销毁,就会报错。)
窗口的标示符。( Identifier for this window. This will usually be filled in for you. )

17. public String packageName = null;
此窗口所在的包名。

18. public int screenOrientation =ActivityInfo.SCREEN_ORIENTATION_UNSPECIFIED;
屏幕方向,参见android.content.pm.ActivityInfo#screenOrientation。

//-------------------------------------------------------------------

 
属性大家估计是跳跃式看到,甚至都没有看完,哈哈,本人也是没有看完,但是扫了一下,知道大概。那么简单的说一下:
layoutParams.type:窗口类型,它是用来定义窗口显示在哪个地方,因为屏幕有很多层,有很多视图,具体显示在哪里,就用type来定义了。比如显示PopupWindow,就使用TYPE_APPLICATION_PANEL ,解释:面板窗口,显示于宿主窗口上层。它的值是在1000~1999。
在这个当中的定义的值,都类似定义在视图之上的子视图。并且它一般在应用中显示。Systemwindows
类型的,是可以显示在android机的桌面上的(并不需要依附应用中的Activity).
 
layoutParams.flags  定义窗口的一些特性,比如说窗口是否聚焦,是否可以触摸灯
其他的具体可以看上面的信息。

这是它的基本属性的详细描述,我们用到的就是上面我写的代码那几个属性。
属性介绍完了,那么就来一个例子,显示我们的成果。
 
首先定义一个视图popwindow.xml文件
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout
  3. xmlns:android="http://schemas.android.com/apk/res/android"
  4. android:layout_width="match_parent"
  5. android:layout_height="match_parent"
  6. android:orientation="vertical"
  7. android:gravity="center"
  8. android:background="#66000000">
  9. <RelativeLayout
  10. android:id="@+id/popupwindow"
  11. android:layout_width="200dp"
  12. android:layout_height="200dp"
  13. android:background="#ffffff">
  14. <TextView
  15. android:id="@+id/title_tv"
  16. android:layout_width="wrap_content"
  17. android:layout_height="wrap_content"
  18. android:text="标题"
  19. android:textSize="22dp"
  20. android:textColor="#649DF2"
  21. android:layout_centerHorizontal="true"
  22. android:layout_marginTop="16dp"/>
  23. <View
  24. android:id="@+id/line"
  25. android:layout_width="match_parent"
  26. android:layout_height="1dp"
  27. android:background="#95BEF9"
  28. android:layout_below="@+id/title_tv"
  29. android:layout_marginTop="8dp"/>
  30. <TextView
  31. android:id="@+id/content_tv"
  32. android:layout_width="wrap_content"
  33. android:layout_height="wrap_content"
  34. android:text="内容"
  35. android:textSize="22dp"
  36. android:layout_centerHorizontal="true"
  37. android:layout_below="@+id/line"
  38. android:layout_marginTop="24dp"/>
  39. <LinearLayout
  40. android:layout_width="match_parent"
  41. android:layout_height="wrap_content"
  42. android:orientation="horizontal"
  43. android:layout_below="@+id/content_tv"
  44. android:layout_marginTop="34dp"
  45. android:gravity="center">
  46. <Button
  47. android:id="@+id/cancel_bt"
  48. android:layout_width="wrap_content"
  49. android:layout_height="wrap_content"
  50. android:text="取消"/>
  51. <Button
  52. android:id="@+id/ok_bt"
  53. android:layout_width="wrap_content"
  54. android:layout_height="wrap_content"
  55. android:text="确定"
  56. android:background="#649DF2"/>
  57. </LinearLayout>
  58. </RelativeLayout>
  59. </LinearLayout>

接下来去写WIndowUtil

  1. public class WindowUtils {
  2. private static View mView;
  3. private static Context mContext;
  4. private static WindowManager windowManager;
  5. public static Boolean isShown = false;
  6. public static void showPopupWindow(final Context context,int resId){
  7. if (isShown){
  8. return;
  9. }
  10. isShown = true;
  11. mContext = context.getApplicationContext();
  12. windowManager = (WindowManager) mContext.getSystemService(Context.WINDOW_SERVICE);
  13. mView = setUpView(mContext,resId);
  14. final WindowManager.LayoutParams layoutParams = new WindowManager.LayoutParams();
  15. layoutParams.type = WindowManager.LayoutParams.TYPE_SYSTEM_ALERT;
  16. int flags = WindowManager.LayoutParams.FLAG_ALT_FOCUSABLE_IM;
  17. // | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE;
  18. // 如果设置了WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE,弹出的View收不到Back键的事件
  19. layoutParams.flags = flags;
  20. // 不设置这个弹出框的透明遮罩显示为黑色
  21. layoutParams.format = PixelFormat.TRANSLUCENT;
  22. // FLAG_NOT_TOUCH_MODAL不阻塞事件传递到后面的窗口
  23. // 设置 FLAG_NOT_FOCUSABLE 悬浮窗口较小时,后面的应用图标由不可长按变为可长按
  24. // 不设置这个flag的话,home页的划屏会有问题
  25. layoutParams.width = WindowManager.LayoutParams.MATCH_PARENT;
  26. layoutParams.height = WindowManager.LayoutParams.MATCH_PARENT;
  27. layoutParams.gravity = Gravity.CENTER;
  28. windowManager.addView(mView,layoutParams);
  29. }
  30. /**
  31. * 隐藏弹框
  32. */
  33. public static void hidePopUpWindow(){
  34. if (isShown && mView != null){
  35. windowManager.removeViewImmediate(mView);
  36. isShown = false;
  37. }
  38. }
  39. private static View setUpView(Context context,int resId){
  40. View view = LayoutInflater.from(context).inflate(resId,null);
  41. Button positiveBtn = (Button) view.findViewById(R.id.ok_bt);
  42. positiveBtn.setOnClickListener(new View.OnClickListener() {
  43. @Override
  44. public void onClick(View v) {
  45. //隐藏弹框
  46. System.out.println("点击了确定按钮");
  47. hidePopUpWindow();
  48. }
  49. });
  50. Button negativeBtn = (Button) view.findViewById(R.id.cancel_bt);
  51. negativeBtn.setOnClickListener(new View.OnClickListener() {
  52. @Override
  53. public void onClick(View v) {
  54. System.out.println("点击了取消按钮");
  55. hidePopUpWindow();
  56. }
  57. });
  58. final View popwindow = view.findViewById(R.id.popupwindow);
  59. //点击弹框外面的,可以取消
  60. view.setOnTouchListener(new View.OnTouchListener() {
  61. @Override
  62. public boolean onTouch(View v, MotionEvent event) {
  63. int x = (int) event.getX();
  64. int y = (int) event.getY();
  65. Rect rect = new Rect();
  66. popwindow.getLocalVisibleRect(rect);
  67. if (!rect.contains(x,y)){
  68. hidePopUpWindow();
  69. }
  70. return false;
  71. }
  72. });
  73. //点击back键可消除
  74. view.setOnKeyListener(new View.OnKeyListener() {
  75. @Override
  76. public boolean onKey(View v, int keyCode, KeyEvent event) {
  77. switch (keyCode){
  78. case KeyEvent.KEYCODE_BACK:
  79. hidePopUpWindow();
  80. return true;
  81. default:
  82. return false;
  83. }
  84. }
  85. });
  86. return view;
  87. }
  88. }
只需要在Activity调用
WindowUtils.showPopupWindow(WindowTest.this,R.layout.popupwindow);
权限不要忘了。

总结:

理解了WindowManager后,再去理解Dialog和PopupWindow后,会容易很多,那PopupWindow来说,我们就当它为一个窗口,使用了TYPE_APPLICATION_PANEL 模式的窗口,并且它依赖于当前Activity,当前Activity会传入Token。这个窗口就和上面我们实现的例子很像了(PopupWindow内部内部细节很多)。当我们首先把它理解为一个窗口,然后在这个窗口中实现自己的操作,我们看源码的时候,思路就会打开。

Android WindowManager和WindowManager.LayoutParams的使用以及实现悬浮窗口的方法的更多相关文章

  1. WindowManager和WindowManager.LayoutParams的使用以及实现悬浮窗口的方法

    写Android程序的时候一般用WindowManager就是去获得屏幕的宽和高,来布局一些小的东西.基本上没有怎么看他的其他的接口. 这两天想写一个简单的类似于Toast的东西,自定义布局,突然发现 ...

  2. Android视频悬浮窗口实现

    前言 本文例子实现了点击显示悬浮窗口,同时窗口可播放视频,拖动位置,点击关闭及返回APP页面,通过例子来讲述悬浮窗口实现原理及细节处理,效果图如下所示: 原理 WindowManager对View视图 ...

  3. Android之悬浮窗口实现(WindowManager)

    工作中遇到一些项目需要把窗体显示在最上层,像来电弹窗显示电话号码等信息.拦截短信信息显示给用户或者游戏中实现声音的调节,我们想这些数据放在最上层,activity就满足不了我们的需求了,有些开发者使用 ...

  4. 使用WindowManager添加View——悬浮窗口的基本原理

    Android系统中的“窗口”类型虽然很多,但只有两大类是经常使用的:一是由系统进程管理的,称之为“系统窗口”:第二个就是由应用程序产生的,用于显示UI界面的“应用窗口”.如果大家熟悉WindowMa ...

  5. Android悬浮窗口的实现

    效果图:(悬浮框可拖动) 在项目开发中有一个需求:弹出悬浮窗后,响应悬浮窗的事件再弹出对话框,但是对话框怎么也不显示.也就是说在弹出悬浮框的同时,不能再弹出对话框,可能的原因: 1.悬浮框的焦点在最前 ...

  6. Android中悬浮窗口

    调用WindowManager,并设置WindowManager.LayoutParams的相关属性,通过WindowManager的addView方法创建View,这样产生出来的View根据Wind ...

  7. Android中悬浮窗口的实现原理和示例代码

    用了我一个周末的时间,个中愤懑就不说了,就这个问题,我翻遍全球网络没有一篇像样的资料,现在将实现原理简单叙述如下: 调用WindowManager,并设置WindowManager.LayoutPar ...

  8. Android 悬浮窗口

    Android 悬浮窗口 一.创建悬浮窗口步骤    1.实现一个ViewGroup类,作为悬浮窗口的界面类,以便在里面重写onInterceptTouchEvent和onTouchEvent方法,实 ...

  9. Android 类似360悬浮窗口实现源码

    当我们在手机上安装360安全卫士时,手机屏幕上时刻都会出现一个小浮动窗口,点击该浮动窗口可跳转到安全卫士的操作界面,而且该浮动窗口不受其他activity的覆盖影响仍然可见(多米音乐也有相关的和主界面 ...

随机推荐

  1. mac系统下安装、启动、停止mongodb

    mongodb是非关系型数据库,mysquel是关系型数据库,前者没有数据表这个说法,后者有 一. 下载nodejs,安装,一直到 node -v显示版本号,表示安装成功. 二. 本文主要讲解,安装包 ...

  2. selenium使用Xpath+CSS+JavaScript+jQuery的定位方法(治疗selenium各种定位不到,点击不了的并发症)

    跟你说,你总是靠那个firebug,chrome的F12啥的右击复制xpath绝对总有一天踩着地雷炸的你死活定位不到,这个时候就需要自己学会动手写xpath,人脑总比电脑聪明,开始把xpath语法给我 ...

  3. swift-UITextfield控件的基本属性设置

    //1.初始化UITextField let userName=UITextField(frame: CGRectMake(0, 100, 100, 100)); //2.将文本框userName添加 ...

  4. 八进制、十进制、操作符(day04)

    把二进制表示的数字从右向左每三个数位分成 一组,每组用一个0到7之间的数字替换. 这个替换结果叫做数字的八进制表示方式 (八进制) 可以直接在程序里用八进制方式表示数字, 这种数字必须以0做开头 可以 ...

  5. 洛谷P1316 丢瓶盖【二分+递推】

    陶陶是个贪玩的孩子,他在地上丢了A个瓶盖,为了简化问题,我们可以当作这A个瓶盖丢在一条直线上,现在他想从这些瓶盖里找出B个,使得距离最近的2个距离最大,他想知道,最大可以到多少呢? 输入输出格式 输入 ...

  6. [luogu4799 CEOI2015 Day2] 世界冰球锦标赛(折半搜索)

    传送门 Solution 折半搜索裸题,注意\(long long\) Code #include <cmath> #include <cstdio> #include < ...

  7. JS移动客户端--触屏滑动事件及js手机拖拽效果

    移动端触屏滑动的效果其实就是图片轮播,在PC的页面上很好实现,绑定click和mouseover等事件来完成.但是在移动设备上,要实现这种轮播的效果,就需要用到核心的touch事件.处理touch事件 ...

  8. Silverlight之我见——制作星星闪烁动画

    圣诞节来了,无聊,做点东西纪念一下. 原理很简单,生成1000个圆,从随机数来布置它们的位置,通过动画来处理它们的透明度,动画时长也是随机生成. 1.创建图形数组并设置背景透明,渐变笔触,大小等,而后 ...

  9. openc下cv::Mat和IplImage的相互转换

    opencv2.0的类CV::Mat和opencv1.0的IplImage之间烦人转换: cv::Mat matimg = cv::imread ("girl.jpg"); Ipl ...

  10. 【codeforces 798A】Mike and palindrome

    [题目链接]:http://codeforces.com/contest/798/problem/A [题意] 让你严格改变一个字符,使得改变后的字符串为一个回文串; 让你输出可不可能; [题解] 直 ...