1. 程式人生 > >帶有一鍵清空功能的EditText

帶有一鍵清空功能的EditText

介紹

  很常見的一個功能,大部分app在登入介面都會實現這個功能了。因為在掘金上看了一篇類似的文章,所以決定自己實踐一下。
  下圖為實現效果:

一鍵清空

常見實現方法

  • 組合控制元件,EditText + Button
     實現簡單,可以單獨使用。
  • 自定義View,繼承EditText,通過EditText自帶的Drawable來實現。
     佈局複雜度低

繼承EditText來實現一鍵清功能

需要考慮的問題

根據業務場景,有以下幾個問題需要我們在實現中考慮到:
1. 怎麼新增清空按鈕
2. 怎麼處理點選事件
3. 處理清空按鈕的顯示狀態
3.1 有文字時才顯示清空按鈕,沒有文字則掩藏。
3.2 獲取焦點時才顯示清空按鈕,沒有焦點時則隱藏
3.3 EditText的setErrot方法呼叫後,清空按鈕怎麼處理
4. 新增自定義的屬性

實現流程

帶著上面的問題我們開始一步步實現自定義View。

步驟1:繼承EditText,實現構造方法

public class ClearableEditText extends EditText
        implements EditText.OnFocusChangeListener {

    public static final String TAG = "ClearableEditText";

    public ClearableEditText(Context context) {
       this(context, null);
    }

    public
ClearableEditText(Context context, AttributeSet attrs) { this(context, attrs, android.R.attr.editTextStyle); } public ClearableEditText(Context context, AttributeSet attrs, int defStyleAttr) { this(context, attrs, defStyleAttr, 0); } @TargetApi(Build.VERSION_CODES.LOLLIPOP) public
ClearableEditText(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) { super(context, attrs, defStyleAttr, defStyleRes); init(context, attrs, defStyleAttr, defStyleRes); } }

步驟二:新增清空按鈕

TextView中有一個靜態內部類Drawables, 這個類的作用是用來儲存和顯示TextView上下左右的Drawable。通過android:drawable* 來設定的drawable icon就儲存在Drawables中。

通過呼叫 Drawable drawables[] = getCompoundDrawables();方法可以獲取到Drawables中的left, top, right, and bottom的Drawable陣列。而drawables[2]正好就是顯示在TextView右邊的drawable icon,所以這個drawable 正好滿足我們的需求。


    private Drawable mClearDrawable;
    /**
     * Right Drawable 是否可見
     */
    private boolean mIsClearVisible;

    private void init(Context context, AttributeSet attrs, int defStyleAttr, int
            defStyleRes) {

        Drawable drawables[] = getCompoundDrawables();
        mClearDrawable = drawables[2]; // Right Drawable;

        // 第一次隱藏
        setClearDrawableVisible(false);
    }

步驟三:處理點選事件

由於Drawable沒辦法接收處理TouchEvent,所以我們只能通過觸控區域來判斷,當觸控事件的座標在right drawable的範圍內的時候就觸發點選事件。

覆寫onTouchEvent事件,這裡我只判斷了x軸的範圍。那為什麼不加上y軸的判斷呢?個人認為沒什麼必要。

    @Override
    public boolean onTouchEvent(MotionEvent event) {

        // error drawable 不顯示 && clear drawable 顯示 && action up
        if (getError() == null && mIsClearVisible && event.getAction() == MotionEvent.ACTION_UP) {

            float x = event.getX();
            if (x >= getWidth() - getTotalPaddingRight() && x <= getWidth() - getPaddingRight()) {
                Log.d(TAG, "點選清除按鈕!");

                clearText();
            }
        }

        return super.onTouchEvent(event);
    }


步驟四:處理清空按鈕的顯示狀態

有三種情況需要考慮:
1 有文字時才顯示清空按鈕,沒有文字則掩藏。
2 獲取焦點時才顯示清空按鈕,沒有焦點時則隱藏
3 EditText的setErrot方法呼叫後,清空按鈕怎麼處理

為了解決1和2的兩個問題,我們需要為EditText監聽文字輸入的狀態和獲取失去焦點的狀態,因此我們需要實現onFocusChange和addTextChangedListener。

    @Override
    public void onFocusChange(View v, boolean hasFocus) {

        if (getError() == null) {
            if (hasFocus) {
                if (getText().length() > 0) {
                    setClearDrawableVisible(true);
                }
            } else {
                setClearDrawableVisible(false);
            }
        }
    }
        // 新增TextChangedListener
        addTextChangedListener(new TextWatcher() {
            @Override
            public void beforeTextChanged(CharSequence s, int start, int count, int after) {

            }

            @Override
            public void onTextChanged(CharSequence s, int start, int before, int count) {
                Log.d(TAG, "onTextChanged " + s);

                setClearDrawableVisible(s.length() > 0);
            }

            @Override
            public void afterTextChanged(Editable s) {

            }
        });

setClearDrawableVisible的具體實現:
我們可以通過setCompoundDrawables來設定TextView的left、top、right、bottom drawable。傳遞null表示不需要顯示。

    /**
     * 設定Right Drawable是否可見
     *
     * @param isVisible true for visible , false for invisible
     */
    public void setClearDrawableVisible(boolean isVisible) {

        setCompoundDrawables(getCompoundDrawables()[0], getCompoundDrawables()[1],
                isVisible ? mClearDrawable : null, getCompoundDrawables()[3]);

        mIsClearVisible = isVisible;
    }

最後考慮第三種情況,EditText的setErrot方法可以給出校驗提示,從下圖中大致可以看出這個提示由兩部分組成,一個icon和一個popup window,而這個icon正好佔據了我們上面提到的right drawable的位置也就是我們的清除按鈕需要用到的位置。那麼error icon是不是也用了right drawable來實現的呢?
setError.jpg

在TextView中有這樣一個方法applyErrorDrawableIfNeeded,從方法名就可以猜到它就是用來設定error drawable的,最終通過mShowing[Drawables.RIGHT] = mDrawableError; 把error drawable 放到了Drawables.RIGHT中。哦,對了。。這一切都是基於LTR的Layout方向。

            // then, if needed, assign the Error drawable to the correct location
            if (mDrawableError != null) {
                switch(layoutDirection) {
                    case LAYOUT_DIRECTION_RTL:
                        mDrawableSaved = DRAWABLE_LEFT;

                        mDrawableTemp = mShowing[Drawables.LEFT];
                        mDrawableSizeTemp = mDrawableSizeLeft;
                        mDrawableHeightTemp = mDrawableHeightLeft;

                        mShowing[Drawables.LEFT] = mDrawableError;
                        mDrawableSizeLeft = mDrawableSizeError;
                        mDrawableHeightLeft = mDrawableHeightError;
                        break;
                    case LAYOUT_DIRECTION_LTR:
                    default:
                        mDrawableSaved = DRAWABLE_RIGHT;

                        mDrawableTemp = mShowing[Drawables.RIGHT];
                        mDrawableSizeTemp = mDrawableSizeRight;
                        mDrawableHeightTemp = mDrawableHeightRight;

                        mShowing[Drawables.RIGHT] = mDrawableError;
                        mDrawableSizeRight = mDrawableSizeError;
                        mDrawableHeightRight = mDrawableHeightError;
                        break;
                }
            }

那麼我們就可以處理第三種情況。覆寫setError方法。想要知道error drawable是否顯示,可以通過“getError() == null”來判斷,為ture則表示不顯示,false表示已顯示。

    @Override
    public void setError(CharSequence error, Drawable icon) {
        if (error != null) {
            setClearDrawableVisible(true);
        }
        super.setError(error, icon);
    }

為什麼是覆寫setError方法?我們可以通過TextView.sendAfterTextChanged來找到答案。當EditText中重新輸入文字的後,sendAfterTextChanged會被呼叫,在sendAfterTextChanged方法中會呼叫hideErrorIfUnchanged,而 hideErrorIfUnchanged則是直接呼叫了setError(null, null)。通過setError(null, null)方法隱藏 error drawable。

步驟五:新增自定義屬性

attrs.xml中申明style

    <declare-styleable name="ClearableEditText">

        <attr name="right_drawable_color" format="color|reference" />

    </declare-styleable>

在init函式中獲取自定義屬性並做相關處理。

        final Resources.Theme theme = context.getTheme();

        TypedArray a = theme.obtainStyledAttributes(attrs, R.styleable.ClearableEditText,
                defStyleAttr, defStyleRes);

        int rightDrawableColor = a.getColor(R.styleable.ClearableEditText_right_drawable_color,
                Color.BLACK);

        a.recycle();

        // 給mRightDrawable上色
        DrawableCompat.setTint(mClearDrawable, rightDrawableColor);

例項效果

一鍵清空

原始碼

參考