EditText點擊外部退出編輯

思路:重寫EdieText父布局的dispatchTouchEvent()方法,判斷EditText正在獲取焦點并且點擊外部時調用EditText的clearFocus()方法,并關閉軟鍵盤。


public class EditTextLinearLayout  extends  LinearLayout {

    public EditTextLinearLayout(Context context,AttributeSet attrs) {

        super(context,attrs);

        setFocusableInTouchMode(true);

    }

    @Override

    public boolean dispatchTouchEvent(MotionEvent ev) {

        if(ev.getAction() == MotionEvent.ACTION_DOWN) {

            View v = ((Activity)getContext()).getCurrentFocus();

            if(isShouldHideInput(v,ev)) {

                InputMethodManager imm = (InputMethodManager)getContext().getSystemService(Context.INPUT_METHOD_SERVICE);

                if(imm !=null) {

                    imm.hideSoftInputFromWindow(v.getWindowToken(),0);

                }

            }

        }

        return super.dispatchTouchEvent(ev);

    }

    private boolean isShouldHideInput(View v,MotionEvent event) {

        if(v !=null&& (vinstanceofEditText)) {

            int[] leftTop = {0,0};

            //獲取輸入框當前的location位置

            v.getLocationInWindow(leftTop);

            intleft = leftTop[0];

            inttop = leftTop[1];

            intbottom = top + v.getHeight();

            intright = left + v.getWidth();

            if(event.getX() > left && event.getX() < right

            && event.getY() > top && event.getY() < bottom) {

                // 點擊的是輸入框區域,保留點擊EditText的事件

                return false;

            }else{

                v.clearFocus();

                return true;

            }

        }

        return false;

    }

Android應用中clearFocus方法調用無效的問題解決

clearFocus 無效?

EditText在focus與非focus的時候,顯示效果是不同的:focus的時候光標是閃的,而且我們通常也會給它設置selector,focus的時候給它加上邊框之類的.

通常當我們觸摸EditText之外的View時,需要清除EditText的焦點.很自然的就會想到EditText.clearFocus(),然而常常并沒有用.(EditText.isFocus()依然是true,光標也依然在跳躍...)

clearFocus的實現

clearFocus的調用棧(重要的部分):

View.clearFocus() ->

View.clearFocusInternal() ->

{

  1. mParent.clearChildFocus(this);// 從該View一直向上遍歷父節點,知道DecorView,作用是將parent(ViewGroup)中存儲的mFocus設置為null,即清除焦點

  2. rootViewRequestFocus();// 調用DecorView的requestFocus()方法,作用是找到視圖中的一個View,并將其設置為焦點

}

根據上面列出的調用棧可以看出,清除focus其實包含2個部分的操作:

清除當前當前View的focus標志,并且清除它的祖先節點中存儲的mFocus信息

調用DecorView的requestFocus()方法,重新尋找一個View,并將其設置為focus

requestFocus()的實現

requestFocus(int)支持FOCUS_UP, FOCUS_DOWN, FOCUS_LEFT, FOCUS_RIGHT 4個參數來表示focus的流向,然而事實上傳入的方向參數并沒有作用.(這個其實比較好理解,以FOCUS_RIGHT來說,是該選擇右子樹種的View,還是繪制在右邊的View呢?)

不管傳怎樣的參數,requestFocus()都是以先序遍歷的方式,找到第一個focusInTouchMode的View,并將其設置為焦點.

設置的方式是:

給當前View focus標志(mPrivateFlags)

調用mParent.requestChildFocus()將自己賦值給其父View的mFocus,然后父View再調用mParent.requestChildFocus()一直到DecorView.

這樣從DecorView開始,只要根據mFocus就可以找到真正focus的View


    @Override

    public View findFocus() {

        if (DBG) {

            System.out.println("Find focus in " + this + ": flags="

            + isFocused() + ", child=" + mFocused);

        }

        if (isFocused()) {

            return this;

        }

        if (mFocused != null) {

            return mFocused.findFocus();

        }

        return null;

    }

注意:按照requestFocus這種尋找策略,那么給定一個起始點,那么尋找到的View將始終相同,也就是說,你多次調用DecorView.requestFocus(),獲得的焦點都是相同的,如果沒有改變視圖層級以及focusable的話.因此當你想讓某個特定的View獲得焦點的話,就應該直接調用它的requestFocus()方法.

tips:對于ViewGroup來說,可以通過descendantFocusability的設置來選擇優先讓parent,還是child獲得焦點.可選值:FOCUS_BEFORE_DESCENDANTS(默認), FOCUS_AFTER_DESCENDANTS, FOCUS_BLOCK_DESCENDANTS.

clearFocus 真的無效嗎?

當然不是,之所以有時候發現EditText.clearFocus()無效,是因為:清除focus之后,還會按照先序遍歷的順序查找一個focusInTouchMode的View,并將其設置為focus,而你的EditText恰好是這第一個符合條件的View.(因此不是沒清除成功,而是清除了之后,又給設置上了!!)

知道了原因之后,解決就很簡單了,找一個在EditText之前的View,將其設置為可獲得焦點的

View.setFocusableInTouchMode(true)

android:focusableInTouchMode="true"

如果不知道怎樣找到一個在EditText之前的View的話,那你可以直接選擇它的parent (xxxLayout),因為ViewGroup默認的策略是: FOCUS_BEFORE_DESCENDANTS

判斷是否focus

isFocused(), 它判斷自己是否擁有焦點

hasFocus(), 它判斷自己或著自己的child是否擁有焦點 常用

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容