im类项目的聊天界面中需要在键盘上显示一个输入控制框,所以需要获取到软键盘的高度,这里就需要使用到android中的getWindowVisibleDisplayFrame()方法。收集整理了一些getWindowVisibleDisplayFrame()的相关资料,这里记录一下备忘。
getWindowVisibleDisplayFrame()是View类下的一个方法,用来获取当前窗口可视区域的大小。该方法原型为:
public void getWindowVisibleDisplayFrame(Rect outRect);
outRect中保存了可视区域的范围,如left, top, right, bottom。
该方法使用注意事项
- 调用该方法的view对象必须在有效的window中,比如activity,fragment或者dialog的layout中。类似new TextView(context).getWindowVisibleDisplayFrame(rect)无法得到正确的结果。
- 该方法必须在view已经attach到window时调用才能得到期望的正确结果。比如我们可以在Activity、Fragment和Dialog的onWindowFocusChanged()方法中执行,在view的onAttachedToWindow()中可能无法获得正确结果
- outRect所表示的只是窗体可见范围,其会受到系统状态栏,虚拟键盘和导航栏的影响,状态栏主要影响outRect的top值,虚拟键盘和导航栏会影响outRect的bottom值
getWindowVisibleDisplayFrame()的应用场景
由于Android系统并没有提供api来获取系统状态栏,软键盘和导航栏的高度,所以我们通常使用getWindowVisibleDisplayFrame()来得到这些模块的值。对系统状态栏高度,获取一个非全屏,且窗口的LayoutParams的height设置为WindowManager.LayoutParams.MATCH_PARENT的窗口可视区域大小,其top值就是状态栏的高度。对系统软键盘,获取一个高度是MATCH_PARENT的窗口在软键盘显示和隐藏两种不同状态下的可视区域大小,将bottom值相减就可以得到软键盘的高度。对系统虚拟按键栏,获取一个高度是MATCH_PARENT的窗口在虚拟按键显示和隐藏两种不同状态下的可视区域大小,将bottom值相减就可以得到虚拟按键的高度。
下面提供获取状态栏,虚拟键盘和导航栏的几种方法:
public static int getStatusBarHeight(Context context) {
int statusbarheight = 0;
try {
Class<?> c = Class.forName("com.android.internal.R$dimen");
Object o = c.newInstance();
Field field = c.getField("status_bar_height");
int x = (Integer) field.get(o);
statusbarheight = context.getResources().getDimensionPixelSize(x);
} catch (Exception e) {
e.printStackTrace();
}
return statusbarheight;
}
public static int getStatusBarHeight(View view) {
Rect outRect = new Rect();
view.getWindowVisibleDisplayFrame(outRect);
statusbarheight = outRect.top;
return outRect.top;
}
/**
* 获取软键盘高度
*/
view.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
Rect rect = new Rect();
int windowVisibleBottomWithoutKeyboard;
@Override
public void onGlobalLayout() {
rect.setEmpty();
view.getWindowVisibleDisplayFrame(rect);
int detal = windowVisibleBottomWithoutKeyboard - rect.bottom;
if (detal > screenHeight / 3) {
keyboardHeight = detal;
} else {
windowVisibleBottomWithoutKeyboard = rect.bottom;
}
}
});
public static int getNavBarHeight(Context context){
if (isMeizu()) {
return getSmartBarHeight(context);
}
if (checkDeviceHasNavigationBar(context)) {
Resources resources = context.getResources();
int resourceId = resources.getIdentifier("navigation_bar_height", "dimen", "android");
if (resourceId > 0) {
return resources.getDimensionPixelSize(resourceId);
}
}
return 0;
}
public static boolean checkDeviceHasNavigationBar(Context context) {
boolean hasNavigationBar = false;
Resources rs = context.getResources();
int id = rs.getIdentifier("config_showNavigationBar", "bool", "android");
if (id > 0) {
hasNavigationBar = rs.getBoolean(id);
}
String navBarOverride = getSystemProperty("qemu.hw.mainkeys", "");
if ("1".equals(navBarOverride)) {
hasNavigationBar = false;
} else if ("0".equals(navBarOverride)) {
hasNavigationBar = true;
}
return hasNavigationBar;
}
public static boolean isMeizu() {
/* 获取魅族系统操作版本标识*/
String meizuFlymeOSFlag = getSystemProperty("ro.build.display.id", "");
if (TextUtils.isEmpty(meizuFlymeOSFlag)) {
return false;
} else if (meizuFlymeOSFlag.contains("flyme") || meizuFlymeOSFlag.toLowerCase().contains("flyme")) {
return true;
} else {
return false;
}
}
public static int getSmartBarHeight(Context context) {
try {
Class c = Class.forName("com.android.internal.R$dimen");
Object obj = c.newInstance();
Field field = c.getField("mz_action_button_min_height");
int height = Integer.parseInt(field.get(obj).toString());
return context.getResources().getDimensionPixelSize(height);
} catch (Exception e) {
e.printStackTrace();
}
return 0;