Android获取底部导航部分机型(华为nova,小米8等)出现了没有底部导航栏,但是却检测到了,手动给视图加高度的UI就出现了问题,下方多了一部分空白,于是查找新的方法。
检测导航栏
先看原来使用的方法,Android想要获取底部虚拟NavigationBar的高度,需要先检测该手机有无底部导航,示例代码如下:
/**
* 检查是否存在虚拟按键栏
*/
@TargetApi(Build.VERSION_CODES.ICE_CREAM_SANDWICH)
private static boolean hasNavBar(Context context) {
Resources res = context.getResources();
int resourceId = res.getIdentifier("config_showNavigationBar", "bool", "android");
if (resourceId != 0) {
boolean hasNav = res.getBoolean(resourceId);
String sNavBarOverride = getNavBarOverride();
if ("1".equals(sNavBarOverride)) {
hasNav = false;
} else if ("0".equals(sNavBarOverride)) {
hasNav = true;
}
return hasNav;
} else {
return !ViewConfiguration.get(context).hasPermanentMenuKey();
}
}
/**
* 判断虚拟按键栏是否重写
*/
private static String getNavBarOverride() {
String sNavBarOverride = null;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
try {
Class c = Class.forName("android.os.SystemProperties");
Method m = c.getDeclaredMethod("get", String.class);
m.setAccessible(true);
sNavBarOverride = (String) m.invoke(null, "qemu.hw.mainkeys");
} catch (Exception e) {
Timber.e(e.toString());
}
}
return sNavBarOverride;
}
获取导航高度
如果返回为true才可以获取底部导航的高度,下边是获取底部导航高度的代码:
/**
* 获取虚拟按键的高度
*/
public static int getNavigationBarHeight(Activity activity) {
int result = 0;
if (hasNavBar(activity)) {
Resources res = activity.getResources();
int resourceId = res.getIdentifier("navigation_bar_height", "dimen", "android");
if (resourceId > 0) {
result = res.getDimensionPixelSize(resourceId);
}
}
Timber.d("NavigationBarHeight = " + result);
return result;
}
但是后边测试过程中出现了问题,有些页面横竖屏切换动态加了这个高度,在部分手机上出现了,没有底部导航,但是检测到了的情况,导致整个布局下方出现了白条,于是开始找如何新的检测底部导航是否存在的方法。
新方法检测底部导航是否存在
private static final String NAVIGATION = "navigationBarBackground";
// 该方法需要在View完全被绘制出来之后调用,否则判断不了
public static boolean isNavigationBarExist(@NonNull Activity activity) {
ViewGroup vp = (ViewGroup) activity.getWindow().getDecorView();
if (vp != null) {
for (int i = 0; i < vp.getChildCount(); i++) {
vp.getChildAt(i).getContext().getPackageName();
if (vp.getChildAt(i).getId() != NO_ID && NAVIGATION.equals(activity.getResources().getResourceEntryName(vp.getChildAt(i).getId()))) {
return true;
}
}
}
return false;
}
这种方式需要在view加载完成后,遍历所有view,来判断底部导航是否显示了,比上边使用getIdentifier获取的方式更准确,可以适配可以检测到有,但实际没有显示的场景。