Activity嵌套多个Fragment且包含Toolbar和Tabhost时对软键盘的一种处理方法

写了有两年多的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嵌套多个Fragment且包含Toolbar和Tabhost时对软键盘的一种处理方法_第1张图片

先说说我踩过的坑:父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,如有问题欢迎评论区交流。

 

你可能感兴趣的:(Android软键盘)