写了有两年多的Android代码,软键盘问题一直是一个很恶心的问题,在style.xml文件中对Activity设置不同的Theme、在AndroidManifest.xml中对Activity设置windowsSoftInputMode、是否有Toolbar等等都会对软键盘的设置造成影响。正好手里的项目很多页面(既有Activity页面的,也有Fragment页面的)有大量的EditText,这两天发现在Android4.x.x系统有bug,这次做一个记录,以供大家参考,手里只有Android4.4.2、Android4.4.4、Android5.1、Android6.0、Android7.1、Android8.0的手机,实测这些手机中没有问题。
1、对于在Activity中的处理还比较容易,处理的方法也比较多,这里说一种可行的方法:将所有EditText(比如除了Toolbar以外的其他所有控件)统统放到一个ScrollView中,AndroidManifest.xml中不需要对Activity进行软键盘设置,如果设置了android:windowSoftInputMode="adjustPan",请去掉 (否则Toolbar会被一起弹出屏幕),布局xml中也不需要进行其他设置,这样点击EditText的时候,Toolbar会保持不动,ScrollView会自行滚动以适应布局变化。
2、ViewPager+Fragment组合下,且Activity顶部包含Toolbar,底部有一个LinearLayout(里面是n个Fragment的切换按钮),见下方示意图:
先说说我踩过的坑:父Activity的布局结构从上到下依次是Toolbar,ViewPager,LinearLayout。对于Fragment的xml布局,我依旧是采用和Activity中一样,根布局采用ScrollView,结果发现软键盘弹起来的时候,父Activity底部的LinearLayout也被弹起来了,虽然Fragment的输入框没有被遮住,但是看着非常不爽。然后各种百度,说来说去都是说在AndroidManifest中给Activity添加属性android:windowSoftInputMode="adjustPan",可是这样虽然能解决Activity底部LinearLayout不弹起来,但是顶部的Toolbar却又被整体移出屏幕了啊!无奈之下采取了下面这个方法:Fragment的布局依旧采用ScrollView作为根布局,Activity在代码中对根布局进行布局变化监听(软键盘的弹起和收回都会触发这个方法),如果键盘弹起来,先记录当前底部LinearLayout的高度,然后将高度动态设置为0;如果键盘收起来,则动态设置底部LinearLayout的高度恢复为之前记录的高度,代码如下:
findViewById(R.id.ll_root).addOnLayoutChangeListener(new View.OnLayoutChangeListener() {
@Override
public void onLayoutChange(View v, int left, int top, int right, int bottom, int oldLeft, int oldTop, int oldRight, int oldBottom) {
LogUtils.d("bottom:" + bottom + ",oldBottom:" + oldBottom + ",menuHeight:" + menuHeight);
if (bottom - oldBottom < -1) {
//软键盘弹上去了,先记录下底部控件的高度,再动态设置高度为0
menuHeight = llMenu.getHeight();
LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, 0);
llMenu.setLayoutParams(params);
} else if (bottom - oldBottom > 1) {
if (isFirst) {
isFirst = false;
} else {
//软键盘弹下去了,动态设置高度,恢复原先控件高度
LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, menuHeight);
llMenu.setLayoutParams(params);
}
}
}
});
说一下各个变量的意思:ll_root是Activity根布局的id;llMenu是Activity底部切换Fragment的一组按钮所在的LinearLayout,menuHeight则是这个LinearLayout的高度;isFirst是标记是否是刚进入页面,因为在页面还在渲染的时候也会触发这个方法,而且是bottom
然而,下午忽然发现在Android4.x.x系统的手机上,依旧会弹起Activity底部的llMenu,打印Log一看,原来在Android4.x.x系统的手机上,监听Activity的根布局变化时,软键盘的弹起与隐藏不会对bottom的值造成改变,导致bottom与oldBottom始终相等,也就不会进入if判断。既然根布局变化监听不行,索性直接监听llMenu好了,然后把findViewById(R.id.ll_root)换成了llMenu,结果发现能正常进入if判断而且判断的也是正确的,可是在 onLayoutChange()方法中不执行llMenu.setLayoutParams(params);语句。最终的解决方案是:在llMenu外再嵌套一个LinearLayout(下方代码里面的menuRoot),对外层的LinearLayout的位置进行监听(软键盘的弹起和收回会改变bottom的值),然后在onLayoutChange()方法中动态改变llMenu的高度。
// ①注意此处不采用直接对根布局的变化监听,因为在Android5.0以下的系统,监听根布局变化时,软键盘的弹起和隐藏,并不会对bottom造成改变
// 导致bottom和oldBottom始终相等;
// ②也不能直接对llMenu进行监听,在onLayoutChange()方法中llMenu.setLayoutParams(params)并没有效果;
// ③采取的方法是在llMenu外层包裹一个menuRoot,对menuRoot进行监听,至此解决问题
findViewById(R.id.menuRoot).addOnLayoutChangeListener(new View.OnLayoutChangeListener() {
@Override
public void onLayoutChange(View v, int left, int top, int right, int bottom, int oldLeft, int oldTop, int oldRight, int oldBottom) {
if (bottom - oldBottom < -1) {
LogUtils.d("ViewPosition", "键盘显示,bottom:" + bottom + ",oldBottom:" + oldBottom + ",llMenu.getBottom():" + llMenu.getBottom() + ",menuHeight:" + menuHeight);
//软键盘弹上去了,先记录下底部控件的高度,再动态设置高度为0
menuHeight = llMenu.getHeight();
LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, 0);
llMenu.setLayoutParams(params);
} else if (bottom - oldBottom > 1) {
LogUtils.d("ViewPosition", "键盘隐藏,bottom:" + bottom + ",oldBottom:" + oldBottom + ",llMenu.getBottom():" + llMenu.getBottom() + ",menuHeight:" + menuHeight);
if (isFirst) {
isFirst = false;
} else {
//软键盘弹下去了,动态设置高度,恢复原先控件高度
LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, menuHeight);
llMenu.setLayoutParams(params);
}
}
}
});
至此解决了问题,工作太忙没时间写demo,如有问题欢迎评论区交流。