點選EditText之外隐藏鍵盤的實作方式
重寫事件分發dispatchTouchEvent,注意不要在onTouchEvent中操作,因為onTouchEvent并非任何情況下都會被調用。通過計算EditText在布局中的位置,進行鍵盤的顯示和隐藏處理
/**
* 點選區域在輸入框之外都隐藏掉鍵盤
* @param ev
* @return
*/
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
if (ev.getAction() == MotionEvent.ACTION_DOWN) {
View v = getCurrentFocus();
if (isShouldHideInput(v, ev)) {
hideSoftKeyboard(v);
}
return super.dispatchTouchEvent(ev);
}
// 必不可少,否則所有的元件都不會有TouchEvent了
if (getWindow().superDispatchTouchEvent(ev)) {
return true;
}
return onTouchEvent(ev);
}
public boolean isShouldHideInput(View v, MotionEvent event) {
if (v != null && (v instanceof EditText)) {
int[] leftTop = { 0, 0 };
//擷取輸入框目前的location位置
v.getLocationInWindow(leftTop);
int left = leftTop[0];
int top = leftTop[1];
int bottom = top + v.getHeight();
int right = left + v.getWidth();
if (event.getX() > left && event.getX() < right
&& event.getY() > top && event.getY() < bottom) {
// 點選的是輸入框區域,保留點選EditText的事件
return false;
} else {
return true;
}
}
return false;
}
監聽鍵盤彈起和隐藏的方式
自定義一個view,因為項目中用的LinearLayout,是以以這個布局為例,将其作為根布局,通過布局的高度判斷鍵盤的隐藏和顯示,通過接口将結果回調出去
需要在AndroidManifest中配置鍵盤屬性
android:windowSoftInputMode="adjustUnspecified|stateHidden"
android:windowSoftInputMode="adjustResize|stateHidden"
經過測試上邊兩種配置都可以實作,但是下邊這種不行,具體原因,你可以去看看 adjustUnspecified adjustResize adjustPan的差別
android:windowSoftInputMode="adjustPan|stateHidden"
package com.anjuke.library.uicomponent.view;
import android.content.Context;
import android.util.AttributeSet;
import android.widget.LinearLayout;
/**
* Author: renzhenming
* Time: 2018/8/11 16:58
* Email: [email protected]
* Version:12.3
* Description: 用于監聽鍵盤的隐藏和出現
*/
public class AjkAdjustSizeLinearLayout extends LinearLayout {
public AjkAdjustSizeLinearLayout(Context context) {
super(context);
}
public AjkAdjustSizeLinearLayout(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
}
public AjkAdjustSizeLinearLayout(Context context, AttributeSet attrs) {
super(context, attrs);
}
private int mChangeSize = 200;
/**
* 鍵盤彈起時,布局的高度因為受到擠壓,會變小,是以新的高度h減去舊的高度oldh會得到一個負數
* 這個負數的絕對值等于鍵盤的高度,而且基本可以确定的是鍵盤的高度一定是大于200的,是以滿足
* (oldw != 0 && h - oldh < -mChangeSize)就可以當做鍵盤彈起
*
* 鍵盤收起時,布局高度恢複到最初,新的高度h減去oldh得到一個正數,這個數值正好就是鍵盤的高度
* 是以滿足(oldw != 0 && h - oldh > mChangeSize)時,可以看做是鍵盤收起
* @param w
* @param h
* @param oldw
* @param oldh
*/
@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
super.onSizeChanged(w, h, oldw, oldh);
if (oldw == 0 || oldh == 0)
return;
if (boardListener != null) {
if (oldw != 0 && h - oldh < -mChangeSize) {
boardListener.keyBoardVisible(Math.abs(h - oldh));
}
if (oldw != 0 && h - oldh > mChangeSize) {
boardListener.keyBoardInvisible(Math.abs(h - oldh));
}
}
}
public interface SoftKeyBoardListener {
void keyBoardVisible(int move);
void keyBoardInvisible(int move);
}
SoftKeyBoardListener boardListener;
public void setSoftKeyBoardListener(SoftKeyBoardListener boardListener) {
this.boardListener = boardListener;
}
}