最近项目需求用到弹窗,但是在弹窗里有EditText,从而引发系统输入法覆盖弹窗的问题,值此记录下。
网上很多关于布局或者ScrollView的解决方案,但是木有起作用。最终从这篇博客里得到灵感,贴下链接
android 解决输入法键盘遮盖布局问题 - kobe8 - 博客园。
不过实践中有个细节不足,此文做下补充。
/**
* @param root 最外层布局,需要调整的布局
* @param scrollToView 被键盘遮挡的scrollToView,滚动root,使scrollToView在root可视区域的底部
*/
private void controlKeyboardLayout(final View root, final View scrollToView) {
root.getViewTreeObserver().addOnGlobalLayoutListener( new OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
Rect rect = new Rect();
//获取root在窗体的可视区域
root.getWindowVisibleDisplayFrame(rect);
//获取root在窗体的不可视区域高度(被其他View遮挡的区域高度)
int rootInvisibleHeight = root.getRootView().getHeight() - rect.bottom;
//若不可视区域高度大于100,则键盘显示
if (rootInvisibleHeight > 100) {
int[] location = new int[2];
//获取scrollToView在窗体的坐标
scrollToView.getLocationInWindow(location);
//计算root滚动高度,使scrollToView在可见区域的底部
int srollHeight = (location[1] + scrollToView.getHeight()) - rect.bottom;
root.scrollTo(0, srollHeight);
} else {
//键盘隐藏
root.scrollTo(0, 0);
}
}
});
}
//获取root在窗体的可视区域
root.getWindowVisibleDisplayFrame(rect);
如果root的宽高是match_parent,那获取的可视区域不包括顶部状态栏和底部的导航栏(有的手机显示上木有导航栏,但是会有导航栏的高度)的。
可以通过一下代码拿导航栏高度
/**
*@paramcontext
*@return
*/
public static intgetNavigationBarHeight(Context context) {
Resources resources = context.getResources();
intresourceId = resources.getIdentifier("navigation_bar_height","dimen","android");
intheight = resources.getDimensionPixelSize(resourceId);
returnheight;
}
所以在系统输入法没有弹出来的时候,会发现
//获取root在窗体的不可视区域高度(被其他View遮挡的区域高度)
int rootInvisibleHeight = root.getRootView().getHeight() - rect.bottom;
拿到的是144而不是100,从而前文的if条件成立,导致多余的Scroll。其实在初始情况下,rootInvisibleHeight的值就是手机底部导航栏的高度。
所以,判断输入法是否弹出的条件是rootInvisibleHeight>NavigationBarHeight,而不是100这个臆想的值。