大龙的博客

常用链接

统计

最新评论

由AlertDialog想到的PhoneWindow, Dialog的覆盖问题

在同一个窗口中,可以同时有多个AlertDialog,但是他们之间相互覆盖,最后show()的那个在最上面,Why?

代码如下:

  1. new AlertDialog.Builder(SpinnerTestActivity.this).setTitle("One").setMessage("one")  
  2.                       .setNegativeButton("button", new DialogInterface.OnClickListener(){  
  3.                         public void onClick(DialogInterface dialoginterface, int i){  
  4.               
  5.                         }  
  6.                 }).show();  
  7.                   
  8. new AlertDialog.Builder(SpinnerTestActivity.this).setTitle("TWO").setMessage("Two")  
  9.                 .setNegativeButton("button", new DialogInterface.OnClickListener(){  
  10.                     public void onClick(DialogInterface dialoginterface, int j){  
  11.                     }  
  12.                 }).show();  


答 : 每一个AlertDialog都会新建一个Window(该Window不添加到原窗口中),在该新Window中,创建这个AlertDialog,然 后取出该窗口的decorView,并将其加入到原窗口中,此时原窗口中存在两个decorView,此时新来的decorView就覆盖掉原来的 decorView,显示在UI 上,因此,最后加入的decorView显示在最上面
分析:查看源码,在Dialog.class中
  1. public void show() {  
  2.       if (mShowing) {  
  3.           if (mDecor != null) {  
  4.               mDecor.setVisibility(View.VISIBLE);  
  5.           }  
  6.           return;  
  7.       }  
  8.   
  9.       if (!mCreated) {  
  10.           dispatchOnCreate(null);  
  11.       }  
  12.   
  13.       onStart();  
  14.       mDecor = mWindow.getDecorView();  
  15.       WindowManager.LayoutParams l = mWindow.getAttributes();  
  16.       if ((l.softInputMode  
  17.               & WindowManager.LayoutParams.SOFT_INPUT_IS_FORWARD_NAVIGATION) == 0) {  
  18.           WindowManager.LayoutParams nl = new WindowManager.LayoutParams();  
  19.           nl.copyFrom(l);  
  20.           nl.softInputMode |=  
  21.                   WindowManager.LayoutParams.SOFT_INPUT_IS_FORWARD_NAVIGATION;  
  22.           l = nl;  
  23.       }  
  24.   
  25.       try {  
  26.           mWindowManager.addView(mDecor, l);  
  27.           mShowing = true;  
  28.     
  29.           sendShowMessage();  
  30.       } finally {  
  31.       }  
  32.   }  

其中 mWindow 是在 构造函数 public Dialog(Context context, int theme) 中初始化 :

Window w = PolicyManager.makeNewWindow(mContext); 

 mWindow = w;

由此,在上面的show()代码中,将从mWindow中的decorView加入到 原窗口中,覆盖了原窗口中的decorView的显示

本程序中,最后的显示的窗口中,具有3个decorView(两个存放Dialog, 一个存放原布局):mWindowManager - > mWindowManager - > mViews 



========================================================================================================================


在Hierarchy Viewer中,原先的窗口显示的布局是:


其布局无关精要,主要有一个TextView,一个Spinner,一个 Button

show了AlertDialog之后,其布局为:

不难看出,原decorView被覆盖了,显示的是第二个Dialog

posted on 2013-01-13 04:03 大龙 阅读(477) 评论(0)  编辑 收藏 引用


只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   知识库   博问   管理