在app开发中有很多项目使用底部tab+ ViewPager + fragment 的框架,那么这个时候如果app安装在底部带有虚拟键的设备上的话,会产生设备底部的虚拟键遮挡app底部tab的情况,这个时候对app的外观和功能的使用都产生了很大的影响,下边我们对此情况进行适配。
1、首先我们进行工具类的封装,主要思路是addOnGlobalLayoutListener全局监听视图的变化(onGlobalLayoutListener是viewTreeObserver的内部类,当视图变化时onGlobalLayoutListener可以监听到),那么当视图的高度发生变化时,就对这个视图重新布局,使视图不被遮挡。
-
import android.content.Context;
-
import android.content.res.Resources;
-
import android.graphics.Rect;
-
import android.view.View;
-
import android.view.ViewGroup;
-
import android.view.ViewTreeObserver;
-
import java.lang.reflect.Method;
-
public class NavigationBarUtil {
-
public static void initActivity(View content) {
-
new NavigationBarUtil(content);
-
}
-
private View mObserved;//被监听的视图
-
private int usableHeightView;//视图变化前的可用高度
-
private ViewGroup.LayoutParams layoutParams;
-
private NavigationBarUtil(View content) {
-
mObserved = content;
-
//给View添加全局的布局监听器监听视图的变化
-
mObserved.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
-
public void onGlobalLayout() {
-
resetViewHeight();
-
}
-
});
-
layoutParams = mObserved.getLayoutParams();
-
}
-
/**
-
* 重置视图的高度,使不被底部虚拟键遮挡
-
*/
-
private void resetViewHeight() {
-
int usableHeightViewNow = CalculateAvailableHeight();
-
//比较布局变化前后的View的可用高度
-
if (usableHeightViewNow != usableHeightView) {
-
//如果两次高度不一致
-
//将当前的View的可用高度设置成View的实际高度
-
layoutParams.height = usableHeightViewNow;
-
mObserved.requestLayout();//请求重新布局
-
usableHeightView = usableHeightViewNow;
-
}
-
}
-
/**
-
* 计算试图高度
-
* @return
-
*/
-
private int CalculateAvailableHeight() {
-
Rect r = new Rect();
-
mObserved.getWindowVisibleDisplayFrame(r);
-
return (r.bottom - r.top);//如果不是沉浸状态栏,需要减去顶部高度
-
// return (r.bottom );//如果是沉浸状态栏
-
}
-
/**
-
* 判断底部是否有虚拟键
-
* @param context
-
* @return
-
*/
-
public static boolean hasNavigationBar(Context context) {
-
boolean hasNavigationBar = false;
-
Resources rs = context.getResources();
-
int id = rs.getIdentifier("config_showNavigationBar", "bool", "android");
-
if (id > 0) {
-
hasNavigationBar = rs.getBoolean(id);
-
}
-
try {
-
Class systemPropertiesClass = Class.forName("android.os.SystemProperties");
-
Method m = systemPropertiesClass.getMethod("get", String.class);
-
String navBarOverride = (String) m.invoke(systemPropertiesClass, "qemu.hw.mainkeys");
-
if ("1".equals(navBarOverride)) {
-
hasNavigationBar = false;
-
} else if ("0".equals(navBarOverride)) {
-
hasNavigationBar = true;
-
}
-
} catch (Exception e) {
-
}
-
return hasNavigationBar;
-
}
-
}
2、在需要适配的activity 的 onCreate方法中的 super.onCreate(savedInstanceState)之后 调用
-
if(NavigationBarUtil.hasNavigationBar(this)){
-
NavigationBarUtil.initActivity(findViewById(android.R.id.content));
-
}
(推荐封装base 这样就可以直接继承base就可以)。
最后的效果就是app底部tab 在设备的虚拟键之上。