面试面试Android技术点

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

2016-09-11  本文已影响6729人  goeasyway

最近遇到个怪问题,在其他手机上都能正常的显示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没有什么影响。

上一篇下一篇

猜你喜欢

热点阅读