项目中有页面根据屏幕高度来计算 View 在屏幕中的位置,结果在 Essential Phone 和小米 MIX 2S 中出现问题,看用户评论在一加 5T 中也有问题,这几个手机都是全面屏,官方方案出来前,各家自己瞎搞。
然后定位问题,发现原来代码计算高度是这样的:
context.getResources().getDisplayMetrics().heightPixels - getStatusBarHeight(context);
而问题就出现在减去状态栏高度这里,过去的手机 heightPixels 减去状态栏高度就是可用高度,但这几个手机,heightPixels 就是可用高度,状态栏高度一减,以为可用的就比实际可用的要小一个状态栏高度。
最初搜索也并未找到可以解决这个问题的方案,作为临时适配,通过 Build 判断手机品牌或型号来适配,但手机越来越多,这终究不能作为最终的解决方案。
- DecorView 的高度是包括状态栏和导航栏的高度,算出来就是手机硬件参数的那个高度。
- RootView 的高度就是自己写的 View 的高度,如果
setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN | View.SYSTEM_UI_FLAG_HIDE_NAVIGATION)
,那么 RootView 高度等于 DecorView 高度,如果没有隐藏导航或状态栏,就是从 DecorView 高度减去对应高度。 - 而
context.getResources().getDisplayMetrics().heightPixels
的注释是The absolute height of the available display size in pixels.
,在过去的手机上它等于 RootView 的高度加状态栏高度。
当然如果设置了 Flag,这几个值的关系会有变化。
int heightPixels = getResources().getDisplayMetrics().heightPixels;
Log.e("OMG", "heightPixels:" + heightPixels);
int statusBarHeight = getResources().getDimensionPixelSize(getResources().getIdentifier("status_bar_height", "dimen", "android"));
Log.e("OMG", "statusBarHeight:" + statusBarHeight);
int navigationBarHeight = getResources().getDimensionPixelSize(getResources().getIdentifier("navigation_bar_height", "dimen", "android"));
Log.e("OMG", "navigationBarHeight:" + navigationBarHeight);
final View decorView = getWindow().getDecorView();
final View rootView = getWindow().findViewById(Window.ID_ANDROID_CONTENT);
rootView.post(new Runnable() {
@Override
public void run() {
// 渲染后求高度
Log.e("OMG", "decorViewHeight:" + decorView.getHeight());
Log.e("OMG", "rootViewHeight:" + rootView.getHeight());
}
});
Nexus 5X 的结果是:
- heightPixels:1794
- statusBarHeight:63
- navigationBarHeight:126
- decorViewHeight:1920
- rootViewHeight:1731
显然 heightPixels = rootViewHeight + statusBarHeight,heightPixels + navigationBarHeight = decorViewHeight
而对于全面屏手机,区别在于获取的 heightPixels = rootViewHeight。