在同一個窗口中,可以同時有多個AlertDialog,但是他們之間相互覆蓋,最后show()的那個在最上面,Why?
代碼如下:
- new AlertDialog.Builder(SpinnerTestActivity.this).setTitle("One").setMessage("one")
- .setNegativeButton("button", new DialogInterface.OnClickListener(){
- public void onClick(DialogInterface dialoginterface, int i){
-
- }
- }).show();
-
- new AlertDialog.Builder(SpinnerTestActivity.this).setTitle("TWO").setMessage("Two")
- .setNegativeButton("button", new DialogInterface.OnClickListener(){
- public void onClick(DialogInterface dialoginterface, int j){
- }
- }).show();
答 : 每一個AlertDialog都會新建一個Window(該Window不添加到原窗口中),在該新Window中,創建這個AlertDialog,然 后取出該窗口的decorView,并將其加入到原窗口中,此時原窗口中存在兩個decorView,此時新來的decorView就覆蓋掉原來的 decorView,顯示在UI 上,因此,最后加入的decorView顯示在最上面
分析:查看源碼,在Dialog.class中
- public void show() {
- if (mShowing) {
- if (mDecor != null) {
- mDecor.setVisibility(View.VISIBLE);
- }
- return;
- }
-
- if (!mCreated) {
- dispatchOnCreate(null);
- }
-
- onStart();
- mDecor = mWindow.getDecorView();
- WindowManager.LayoutParams l = mWindow.getAttributes();
- if ((l.softInputMode
- & WindowManager.LayoutParams.SOFT_INPUT_IS_FORWARD_NAVIGATION) == 0) {
- WindowManager.LayoutParams nl = new WindowManager.LayoutParams();
- nl.copyFrom(l);
- nl.softInputMode |=
- WindowManager.LayoutParams.SOFT_INPUT_IS_FORWARD_NAVIGATION;
- l = nl;
- }
-
- try {
- mWindowManager.addView(mDecor, l);
- mShowing = true;
-
- sendShowMessage();
- } finally {
- }
- }
其中 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