软键盘挡住输入框问题的精确解决方案

最近给公司开发App,体验的人员觉得登陆界面总是被软键盘挡住了不爽,自行度娘了一大堆发现都是基于Scrollview的或者是把界面整体拔高。但是我下面有注册按钮我拔高了又太丑了。
所以感觉不爽,在看了郭霖大神发的推送后根据GoogleBugs里解决方案改写了里面的解决方案,改为了一套能够移动部分控件的工具,并且提高了精度。

附上链接:SoftKeyBoardAdapter

package com.example.hele.softkeyboardadapter;

import android.app.Activity;
import android.graphics.Rect;
import android.os.Build;
import android.view.View;
import android.view.ViewTreeObserver;
import android.view.WindowManager;
import android.widget.FrameLayout;

/**
 * Created by Hele on 2017/5/8.
 * 修改自http://mp.weixin.qq.com/s/sWnxIxzNkuTkUaVaPhtqLA
 */
public class SoftHideKeyBoardUtil {

    private static boolean sTranslucentStatus = false;
    private Activity mContext;
    //Root Content View
    private View mChildOfContent;
    private View mViewContainer;
    private int mUsableHeightPrevious;
    private int mLastLocation = -1;
    private int mFrameSize = -1;

    private SoftHideKeyBoardUtil(Activity activity, View container) {
        mContext = activity;
        mViewContainer = container;
        sTranslucentStatus = judgeTranslucentStatus(activity);
        //获取根框
        FrameLayout content = (FrameLayout) activity.findViewById(android.R.id.content);
        //获取ContentView
        mChildOfContent = content.getChildAt(0);
        //ViewTreeObserver:监听界面绘制
        mChildOfContent.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
            @Override
            public void onGlobalLayout() {
                possiblyResizeChildOfContent();
            }
        });
    }

    /**
     * Call this method to prevent your view such as EditText, LoginButton or other
     * from being blocked by a soft keyboard
     *
     * @param activity               : current activity
     * @param view_container_to_move : your view group
     */
    public static void assistActivity(Activity activity, View view_container_to_move) {
        new SoftHideKeyBoardUtil(activity, view_container_to_move);
    }

    private static boolean judgeTranslucentStatus(Activity activity) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            if ((WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS & activity.getWindow().getAttributes().flags)
                    == WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS) {
                return true;
            }
        }
        return false;
    }

    /**
     * Adjust the location of your container
     */
    private void possiblyResizeChildOfContent() {
        if (mLastLocation < 0) {
            mLastLocation = (int) mViewContainer.getY();
            //Use the beginning as default
            mUsableHeightPrevious = computeUsableHeight();
            mFrameSize = mUsableHeightPrevious;
        }
        int usableHeightNow = computeUsableHeight();
        if (usableHeightNow != mUsableHeightPrevious) {
            int heightKeyboard = mFrameSize - usableHeightNow;
            int heightDifference = mUsableHeightPrevious - usableHeightNow;

            float adjustY = mLastLocation;

            //监听键盘变化
            if (heightKeyboard > (mFrameSize / 4) && heightDifference > (mFrameSize / 4)) { 
                //第二个条件是必须的,判断键盘弹起
                //When full screen or translucentStatus is true
                int statusBar = sTranslucentStatus ? DisplayUtil.getStatusBarHeight(mContext) : 0;
                adjustY = mViewContainer.getY() - mViewContainer.getBottom() + usableHeightNow + statusBar;
            } else if (heightKeyboard == 0) {
                //收起键盘
            } else {
                //中英文切换
                //中文切英文 : dif < 0 . 反之, dif > 0
                adjustY = mViewContainer.getY() - heightDifference;
            }
            mViewContainer.setY(adjustY);
            mChildOfContent.requestLayout();
            mUsableHeightPrevious = usableHeightNow;
        }
    }

    /**
     * Compute Visible Height
     * @return
     */
    private int computeUsableHeight() {
        Rect r = new Rect();
        mChildOfContent.getWindowVisibleDisplayFrame(r);
        return (r.bottom - r.top);
    }

}

后来发现有人解释过了所以我就不重新解释一遍了。
附上链接: 解释链接

除了possiblyResizeChildOfContent方法,其他次要内容和Google上面的差不多。主要是添加了渲染模式的处理、中英文的切换。如果采用原来的方式用在内部控件是无法处理中英文切换的,有兴趣的话可以尝试一下。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 177,239评论 25 709
  • 你有才华创意,我有免费摊位。华汇文创空间创意集市招募有才华的你们!❤ 你有没有很多创意的小玩意想跟更多人分享?你是...
    华汇文创空间阅读 9,990评论 0 1
  • 这半辈子颠簸 像一把杂乱的种子 随波逐流的撒了一把在这江南的角落 但我深知只是过客 是发不了芽的 过年本是归家的喜...
    得一生二阅读 2,350评论 0 0
  • 我的妈妈是一名护士,她每天很早出门,但是很晚才回家。爸爸说,妈妈的“家”,是医院。 我问妈妈:“妈妈,你的...
    皓皓_f9d7阅读 3,780评论 0 0
  • [精进每一天]第293天 2017.9.4 我真是幸福得不像话这句话是一本书《人生不设限》里面前言的标题。 作者...
    梁亿阅读 3,297评论 0 0

友情链接更多精彩内容