Android开发经验谈Android开发Android开发

EditText输入框自动顶上去2种解决方案

2017-10-16  本文已影响286人  JuliusL

1、用一个Space顶上去。
2、用ScrollView解决。

遇到的坑:
1、使用方案1时,部分手机有虚拟键盘,需判断在有虚拟键盘上的手机加一个虚拟键盘的高度。

方案1:用一个Space顶上去。

xml:

<FrameLayout>
     <主内容/>
     <LinearLayout>
         <EditText/>
         </Space>
     </LinearLayout>
</FrameLayout>

java:

private KeyboardChangeListener mKeyboardChangeListener;
private boolean isVisiableForLast;

private void initListener(){
        mKeyboardChangeListener = new KeyboardChangeListener(_mActivity);
        mKeyboardChangeListener.setKeyBoardListener((isShow, keyboardHeight) -> ALog.e(TAG, "isShow = [" + isShow + "], keyboardHeight = [" + keyboardHeight + "]"));

        final View decorView = _mActivity.getWindow().getDecorView();
        //计算出可见屏幕的高度
        //获得屏幕整体的高度
        //获得键盘高度
        globalLayoutListener = () -> {
            Rect rect = new Rect();
            decorView.getWindowVisibleDisplayFrame(rect);
            //计算出可见屏幕的高度
            int displayHight = rect.bottom - rect.top;
            //获得屏幕整体的高度
            int hight = decorView.getHeight();
            //获得键盘高度
            int keyboardHeight = hight - displayHight;
            boolean visible = (double) displayHight / hight < 0.8;
            if (visible != isVisiableForLast) {
                listener.onSoftKeyBoardVisible(visible, keyboardHeight);
            }
            isVisiableForLast = visible;
        };
        //注册布局变化监听
        decorView.getViewTreeObserver().addOnGlobalLayoutListener(globalLayoutListener);
}


IKeyBoardVisibleListener listener = new IKeyBoardVisibleListener() {
        @Override
        public void onSoftKeyBoardVisible(boolean visible, int windowBottom) {
            windowBottom -= ScreenUtils.getStatusBarHeight(InputFromActivity.this);
            if (checkDeviceHasNavigationBar(InputFromActivity.this)) {
                windowBottom -= getBottomKeyboardHeight();
            }
            if (visible) {
                LinearLayout.LayoutParams lp = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, windowBottom);
                spaceInput.setLayoutParams(lp);
            } else {
                LinearLayout.LayoutParams lp = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, 0);
                spaceInput.setLayoutParams(lp);
            }
            Log.d(TAG, "onSoftKeyBoardVisible ---- visible:" + visible + " windowBottom:" + windowBottom);
        }
    };

interface IKeyBoardVisibleListener {
        void onSoftKeyBoardVisible(boolean visible, int windowBottom);
}

//------------------------ 检测虚拟键盘部分----------------------
//获取是否存在NavigationBar
    public static boolean checkDeviceHasNavigationBar(Context context) {
        boolean hasNavigationBar = false;
        Resources rs = context.getResources();
        int id = rs.getIdentifier("config_showNavigationBar", "bool", "android");
        if (id > 0) {
            hasNavigationBar = rs.getBoolean(id);
        }
        try {
            Class systemPropertiesClass = Class.forName("android.os.SystemProperties");
            Method m = systemPropertiesClass.getMethod("get", String.class);
            String navBarOverride = (String) m.invoke(systemPropertiesClass, "qemu.hw.mainkeys");
            if ("1".equals(navBarOverride)) {
                hasNavigationBar = false;
            } else if ("0".equals(navBarOverride)) {
                hasNavigationBar = true;
            }
        } catch (Exception e) {

        }
        return hasNavigationBar;
    }

    //获取虚拟键盘高度
    public int getBottomKeyboardHeight() {
        int screenHeight = getAccurateScreenDpi()[1];
        DisplayMetrics dm = new DisplayMetrics();
        getWindowManager().getDefaultDisplay().getMetrics(dm);
        int heightDifference = screenHeight - dm.heightPixels;
        return heightDifference;
    }

    /**
     * 获取精确的屏幕大小
     */
    public int[] getAccurateScreenDpi() {
        int[] screenWH = new int[2];
        Display display = getWindowManager().getDefaultDisplay();
        DisplayMetrics dm = new DisplayMetrics();
        try {
            Class<?> c = Class.forName("android.view.Display");
            Method method = c.getMethod("getRealMetrics", DisplayMetrics.class);
            method.invoke(display, dm);
            screenWH[0] = dm.widthPixels;
            screenWH[1] = dm.heightPixels;
        } catch (Exception e) {
            e.printStackTrace();
        }
        return screenWH;
    }

方案2:用ScrollView解决。

xml:

<Fragment>
    <ScrollView>
        </主内容>
    </ScrollView>
    <LinearLayout>
         <EditText/>
     </LinearLayout>
</Fragment>

注:采用第二种方案无序写java代码

建议:

方案1最好用在主内容是RecyclerView或ListView的布局,原因是防止和ScrollView发生冲突。

上一篇下一篇

猜你喜欢

热点阅读