Android——Fragment中键盘遮挡问题的解决方案

在Mainfest中设置Activity的windowSoftInputMode属性,主要有以下几种:
【A】stateUnspecified:软键盘的状态并没有指定,系统将选择一个合适的状态或依赖于主题的设置
【B】stateUnchanged:当这个activity出现时,软键盘将一直保持在上一个activity里的状态,无论是隐藏还是显示
【C】stateHidden:用户选择activity时,软键盘总是被隐藏
【D】stateAlwaysHidden:当该Activity主窗口获取焦点时,软键盘也总是被隐藏的
【E】stateVisible:软键盘通常是可见的
【F】stateAlwaysVisible:用户选择activity时,软键盘总是显示的状态
【G】adjustUnspecified:默认设置,通常由系统自行决定是隐藏还是显示
【H】adjustResize:该Activity总是调整屏幕的大小以便留出软键盘的空间
【I】adjustPan:当前窗口的内容将自动移动以便当前焦点从不被键盘覆盖和用户能总是看到输入内容的部分
【J】adjustNothing:当前窗口的内容将自动移动以便当前焦点从不被键盘覆盖和用户能总是看到输入内容的部分

参数含义以及显示的效果图引用自:
http://www.jianshu.com/p/dddcaac97cdc

adjustPan
Activity 窗口(DecorView) 大小不变。当获取到焦点的 EditText 位于屏幕下方,软键盘弹出会遮挡到 EditText 时,整个 DecorView 会往上移动,至于上移多少并不确定。
一般是上移至使 EditText 刚好不被软键盘遮挡住为止。这个属性用的比较多,常常配合 ScrollView 来使用。
Android——Fragment中键盘遮挡问题的解决方案_第1张图片

adjustResize
这种模式会自动调整大小。通过 hierarchy View 观察,decorView 大小本身并不会改变,但是我们的内容区 contentView (id = android.R.content) 会相应的缩小,为键盘的显示挪出空间。contentView 的下面为空白区域,软键盘就是覆盖在这个区域。
注意:adjustResize 只是调整 contentView 的大小,所以还是有可能覆盖掉 EditText。
adjustResize 最有用的是,这种模式可以轻松的获取到软键盘的高度,软键盘弹出后 contentView 高度变化的差值即为软键盘高度。
另外:这种模式可能会有个问题,当键盘消失时屏幕会出现一闪感觉有点难受。原因是键盘弹出时,键盘位置显示的是 windowBackground ,如果 windowBackground 为黑色而 Activity 背景为白色,当键盘消失时就会有闪动。解决办法是在 Activity 主题上添加一个 android:windowBackground 属性修改 windows 背景。
Android——Fragment中键盘遮挡问题的解决方案_第2张图片

adjustNothing
这种模式 Activity窗口不会做任何调整,contentView 大小也不会改变。

解决键盘弹出控件被遮挡的解决办法:(该方法来自网上,原始项目找不到了,快照为)
这里写图片描述
1、manifest中设置activity的windowSoftInputMode属性为adjustNothing.

 ...
            android:windowSoftInputMode="adjustNothing">
            ...
     

2、通过一个弹出的PopWindow监听键盘是否弹出以及获得键盘的高度,在PopWindow高度发生变化时回调键盘状态变更接口,在接口中把键盘高度信息(键盘高度=当前屏幕的高度-PopWindow的高度)透露出去

//在KeyboardPopWindow.class extends PopWindow.class中
public KeyboardPopWindow(Activity activity) {
        super(activity);
        this.activity = activity;
        LayoutInflater inflator = (LayoutInflater) activity.getSystemService(Activity.LAYOUT_INFLATER_SERVICE);
        this.popupView = inflator.inflate(R.layout.popupwindow, null, false);
        setContentView(popupView);
        setSoftInputMode(LayoutParams.SOFT_INPUT_ADJUST_RESIZE |         LayoutParams.SOFT_INPUT_STATE_ALWAYS_VISIBLE);//为了当键盘变化时调整PopWindow的大小。    
        setInputMethodMode(PopupWindow.INPUT_METHOD_NEEDED);
        parentView = activity.findViewById(android.R.id.content);
        setWidth(0);
        setHeight(LayoutParams.MATCH_PARENT);
        popupView.getViewTreeObserver().addOnGlobalLayoutListener(new OnGlobalLayoutListener() {
            @Override
            public void onGlobalLayout() {
                if (popupView != null) {
                    handleOnGlobalLayout();
                }
            }
        });
    }

private void handleOnGlobalLayout() {
        Point screenSize = new Point();
        activity.getWindowManager().getDefaultDisplay().getSize(screenSize);
        Rect rect = new Rect();
        popupView.getWindowVisibleDisplayFrame(rect);
        int orientation = getScreenOrientation();

        //键盘高度=屏幕高度-popWindow的高度(需要设置   showAtLocation(parentView, Gravity.NO_GRAVITY, 0, 0);)
        int keyboardHeight = screenSize.y - rect.bottom;
        if (keyboardHeight == 0) {
            notifyKeyboardHeightChanged(0, orientation);
        } else if (orientation == Configuration.ORIENTATION_PORTRAIT) {
            this.keyboardPortraitHeight = keyboardHeight;
            notifyKeyboardHeightChanged(keyboardPortraitHeight, orientation);
        } else {
            this.keyboardLandscapeHeight = keyboardHeight;
            notifyKeyboardHeightChanged(keyboardLandscapeHeight, orientation);
        }
    }

private void notifyKeyboardHeightChanged(int height, int orientation) {
        if (observer != null) {
            observer.onKeyboardHeightChanged(height, orientation);
        }
    }

3、在键盘状态变更接口中处理键盘状态变更事件
/* @param height 键盘的高度
* @orientation 屏幕的朝向
*/

@Override
    public void onKeyboardHeightChanged(int height, int orientation) {
        String or = orientation == Configuration.ORIENTATION_PORTRAIT ? "portrait" : "landscape";
        Rect r = new Rect();

        // 需要动态调整的View
        View fixView = findViewById(R.id.layout_edit_text);

       //View在屏幕中的位置
        fixView.getGlobalVisibleRect(r);

        FrameLayout.LayoutParams params = (FrameLayout.LayoutParams) fixView.getLayoutParams();
       //计算需要的偏移量
       int offset = height - (getWindowManager().getDefaultDisplay().getHeight() - r.bottom);
        if (height == 0)
            params.bottomMargin = 0;
        else
            params.bottomMargin = offset;
        //通过设置View 的bottomMargin改变其位置
        fixView.setLayoutParams(params);
    }

你可能感兴趣的:(android)