美文网首页面试面试Android技术点
Android 7.0经验谈:Dialog不显示之迷

Android 7.0经验谈:Dialog不显示之迷

作者: goeasyway | 来源:发表于2016-09-11 11:10 被阅读6729次

    最近遇到个怪问题,在其他手机上都能正常的显示Dialog,但是在Android 7.0的手机上只能显示Dialog的半透明背景,无法显示Dialog的内容。

    用图给大家展示一个遇到的现象,正常时应该是这个样子的:


    而我们遇到的情况如下,更坏的是如果我们设置Dialog不能点击空白处取消(dismiss),那么这个半透明背景一直覆盖在Activity上面,只有通过杀进程重新运行应用才能去掉它。

    因为项目工程比较大,排查这个问题也花费了很多时间。我最开始时在Android 7.0写一个Demo显示Dialog发现是正常的,然后再在自己的框架上显示Dialog发现也是正常的,说明不是框架的问题。

    在测试了一些怀疑地方没有效果后,只有用土办法挨个排查了,两个方向一个是不断的注释各个模块看是否正常显示,第二个是从正常开始加上各个模块看那个模块代码加上了出问题。最后发现罪魁祸首是下面这段代码:

            Resources resources = context.getResources();
            Configuration config = resources.getConfiguration();
    
            //解决修改系统字体大小时,应用页面布局、字体等显示或者排版混乱问题
            config.setToDefaults();
    

    在Application的onCreate方法会执行这段代码去设置资源的Configuration。再分析setToDefaults的源码,主要是进行了如下的初始化设置:

        /**
         * Set this object to the system defaults.
         */
        public void setToDefaults() {
            fontScale = 1;
            mcc = mnc = 0;
            locale = null;
            userSetLocale = false;
            touchscreen = TOUCHSCREEN_UNDEFINED;
            keyboard = KEYBOARD_UNDEFINED;
            keyboardHidden = KEYBOARDHIDDEN_UNDEFINED;
            hardKeyboardHidden = HARDKEYBOARDHIDDEN_UNDEFINED;
            navigation = NAVIGATION_UNDEFINED;
            navigationHidden = NAVIGATIONHIDDEN_UNDEFINED;
            orientation = ORIENTATION_UNDEFINED;
            screenLayout = SCREENLAYOUT_UNDEFINED;
            uiMode = UI_MODE_TYPE_UNDEFINED;
            screenWidthDp = compatScreenWidthDp = SCREEN_WIDTH_DP_UNDEFINED;
            screenHeightDp = compatScreenHeightDp = SCREEN_HEIGHT_DP_UNDEFINED;
            smallestScreenWidthDp = compatSmallestScreenWidthDp = SMALLEST_SCREEN_WIDTH_DP_UNDEFINED;
            densityDpi = DENSITY_DPI_UNDEFINED;
            seq = 0;
        }
    

    其实看了一下很容易看出screenWidthDp和screenHeightDp比较可疑,测试了一下也确实是这两个值被设置成0后(SCREEN_WIDTH_DP_UNDEFINED = 0)Dialog的内容布局就无法显示了。

    那么,为什么同样的代码在Android 7.0之前的系统是好的呢?

    我们知道Dialog本质上也是一个PhoneWindow对像,显示时被加到WindowManager,由WMS负责显示。从DDMS的HierarchyView工具我们也可以看出,Dialog是一个PhoneWindow,且位于Activity的PhoneWindow之上,并且它的DecorView并不是全屏的。

    在Android 7.0 DecorView被独立成一个类DecorView.java,之前的版本是
    PhoneWindow的内部类,每次DecorView初始化时会进行一个更新:

        private void updateAvailableWidth() {
            Resources res = getResources();
            mAvailableWidth = TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP,
                    res.getConfiguration().screenWidthDp, res.getDisplayMetrics());
        }
    

    在这个方法里我们遇到了之前设置为0的res.getConfiguration().screenWidthDp,把DecorView的mAvailableWidth设置为0,自然无法看到Dialog的Layout内容(宽度为0)。而7.0之前的版本并没有这项设置。

    疑问

    Activity的界面也是一个PhoneWindow,为什么screenWidthDp为0后Activity的Layout还是可以正常展示呢?如下图所示,Activity的DecorView的宽并不是受screenWidthDp的影响。

    从DecorView.java的onMeasure方法中我们看到,是否使用mAvailableWidth(updateAvailableWidth方法中赋值)还要看TypedValue的类型,可以判断Dialog和Activity应该是在这个类型上有所区别,使得这个设置对Activity没有什么影响。

    相关文章

      网友评论

      • 3674d0183b56:关注此坑
      • Android_Happy22:怎么解决这个问题啊 在哪里设置啊 大神
      • de1c1dcd69ea:关注此坑
      • 0f7d096211a6:我遇到的问题是,7.0上面,同一个activity,第一次跳进去,dialog走了show,hide方法,退出activity,dialog dismiss,然后再进这个activity,dialog走了show和hide,然后整个页面不能点击了,如果不用hide,用dimiss就是可以的,你上面提到的config.setToDefaults();也用到了,感觉没影响
        Android_Happy22:在哪里设置config.setToDefaults()啊
        smile三七:我也遇到你这个问题,发现是dialog的原因,改成dismiss就可以了,不知道为啥会这样:grin:
      • f5b4358f4896:楼主能否分享下解决办法 :grin:
        f5b4358f4896:@goeasyway 额,我去掉 ”configuration.setToDefaults();“,然后直接 ”resources.updateConfiguration();“ 这样没问题了。 :joy:
        f5b4358f4896:@goeasyway 能简单的说一下原因吗?我这边使用的是 “resources.updateConfiguration” 来切换APP语音,设置 ”fontScale = 1“ 后,dialog 还是不能显示
        goeasyway:@liushang 直接设置“fontScale = 1;”就好了,不要设置其他不相关的属性。
      • 捡淑:关注此坑

      本文标题:Android 7.0经验谈:Dialog不显示之迷

      本文链接:https://www.haomeiwen.com/subject/jgwjettx.html