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