自定义EditText实现清除文本内容按钮

在使用EditText的时候,发现想要添加一个清除文本的按钮还是听麻烦的,于是诞生出想自定一个控件来实现这样的效果。
好了,先来看看实现的效果吧。

effect.gif

通过效果图,我们可以看到我们的自定义EditText实现了以下的公呢个:

  • 当有内容输入的时候,右边会出现一个红色的删除图片。点击图片就会清除文本的内容;
  • 当文本框中没有内容的时候,右边的xx会消失;
  • 当我们输入不合法的数据时,文本框会震动会提示输入错误。
  • 当输入合法的数据时,文本空不会有提示。
    基本上实现的功能就这些了把。下面先讲讲怎么使用。

1. 直接添加依赖

因为我将这个控件已经打包放到Bintray上了,所以可以通过添加依赖的方式添加到自己的项目中。具体操作如下:

  1. 在项目的==build.gradle==添加依赖
 maven { url 'https://hut.bintray.com/Maven' }
  1. 在要使用的module的build.gradle上添加依赖:
compile 'reoger.hut.com.mylibrary:mylibrary:1.0.1'

然后就可以在项目中进行引用了。
因为是自定义的控件,所以是放在xml文件中进行引用。示例如下:

 <reoger.hut.com.mylibrary.CheckEditText
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:drawableRight="@drawable/x"
        app:errorMsg="请输入数字"
        app:openShake="true"
        app:matchType="@string/checkPhoneNumber"/>

如果你的xml文件中没有定义app的话,需要在根部局添加如下的代码,以便导入我的自定义属性:

xmlns:app="http://schemas.android.com/apk/res-auto"

下面对上面自定义的属性进行说明:

  • matchType 属性定义的是匹配的规则(正则表达式),我找了三个比较常用的匹配规则,包括检验是否为手机号码、电子邮箱和RUL地址,如果这个属性不定义的话,下面两个属性都将没有作用
  • errorMsg 属性定义的是当输入错误时,提示的错误信息。
  • openShake 属性是当输入信息不合法时,是否震荡提示,true表示开启震荡提示
  • drawableRight 属性定义的是右边图片的样式。

最后在使用的时候,提供CheckEditText.dataLegality 这个变量来表示数据是否合法。为true时表示合法,false表示不合法。

2.下载library包,添加依赖

首先贴上我的github地址,下载myLibrary包,导入到项目中,就可以直接使用了。使用方法同方法1.

3. 直接贴代码:

简单起见,我直接贴上主要的代码了,代码中有部分注释,相信理解起来很简单:

package reoger.hut.com.mylibrary;

import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.drawable.Drawable;
import android.text.Editable;
import android.text.TextUtils;
import android.text.TextWatcher;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.view.animation.Animation;
import android.view.animation.CycleInterpolator;
import android.view.animation.TranslateAnimation;
import android.widget.EditText;

import java.util.regex.Matcher;
import java.util.regex.Pattern;

/**
 * Created by 24540 on 2017/3/8.
 *
 */

public class CheckEditText extends EditText implements TextWatcher, View.OnFocusChangeListener {

    //数据是否合法
    public static boolean dataLegality = false;

    //text的文本内容
    private String msg = "";
    private Context mContext;
    /**
     * 是否获取焦点,默认没有焦点
     */
    private boolean mIsFoucse = false;

    //匹配的正则表达式
    private String mMatchString = "";

    //提示的错误信息
    private String mEeorMsg = "";

    /**
     * 左右两侧图片资源
     */
    private Drawable left, right;

    /**
     * 手指抬起时的X坐标
     */
    private int xUp = 0;

    //是否打开震动提示错误
    private boolean isOpenShake = false;


    public CheckEditText(Context context) {
        super(context, null);
    }

    public CheckEditText(Context context, AttributeSet attrs) {
        super(context, attrs);
        mContext = context;
        initWedgits();
        initData(attrs);
        setOnFocusChangeListener(this);
        addTextChangedListener(this);
    }

    public CheckEditText(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs);

    }

    @Override
    public void beforeTextChanged(CharSequence s, int start, int count, int after) {
    }


    @Override
    public void afterTextChanged(Editable s) {
    }

    @Override
    public void onTextChanged(CharSequence text, int start, int lengthBefore, int lengthAfter) {
        super.onTextChanged(text, start, lengthBefore, lengthAfter);
        msg = text.toString();
        if (mIsFoucse) {
            if (TextUtils.isEmpty(text)) {
                // 如果为空,则不显示删除图标
                setCompoundDrawablesWithIntrinsicBounds(left, null, null, null);
            } else {
                // 如果非空,则要显示删除图标
                if (null == right) {
                    right = getCompoundDrawables()[2];
                }
                setCompoundDrawablesWithIntrinsicBounds(left, null, right, null);
            }
        }

    }


    @Override
    public void onFocusChange(View v, boolean hasFocus) {
        this.mIsFoucse = hasFocus;
        if (!mIsFoucse) {
            if (!"".equals(mMatchString)) {
                Pattern p = Pattern
                        .compile(mMatchString);
                Matcher m = p.matcher(msg);
                dataLegality = m.matches();
                if (!dataLegality) {
                    if (!"".equals(mEeorMsg)) {
                        setError(mEeorMsg);
                        if (isOpenShake)
                            setAnimation(shakeAnimation(5));
                    }

                }
            }
        }
    }


    /**
     * 初始化各组件
     */
    private void initWedgits() {
        try {
            left = getCompoundDrawables()[0];
            right = getCompoundDrawables()[2];

            if (length() < 1)
                setCompoundDrawablesWithIntrinsicBounds(left, null, null, null);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }


    private void initData(AttributeSet attrs) {
        TypedArray mTypedArray = mContext.obtainStyledAttributes(attrs, R.styleable.CheckEditText);
        mMatchString = mTypedArray.getString(R.styleable.CheckEditText_matchType);
        mEeorMsg = mTypedArray.getString(R.styleable.CheckEditText_errorMsg);
        isOpenShake = mTypedArray.getBoolean(R.styleable.CheckEditText_openShake, false);
    }


    /**
     * 晃动动画
     *
     * @param counts 1秒钟晃动多少下
     * @return
     */
    public static Animation shakeAnimation(int counts) {
        Animation translateAnimation = new TranslateAnimation(0, 10, 0, 0);
        //设置一个循环加速器,使用传入的次数就会出现摆动的效果。
        translateAnimation.setInterpolator(new CycleInterpolator(counts));
        translateAnimation.setDuration(400);
        return translateAnimation;
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        switch (event.getAction()) {
            case MotionEvent.ACTION_UP:
                // 获取点击时手指抬起的X坐标
                xUp = (int) event.getX();
                // 当点击的坐标到当前输入框右侧的距离小于等于getCompoundPaddingRight()的距离时,则认为是点击了删除图标
                // getCompoundPaddingRight()的说明:Returns the right padding of the view, plus space for the right Drawable if any.
                if ((getWidth() - xUp) <= getCompoundPaddingRight()) {
                    if (!TextUtils.isEmpty(getText().toString())) {
                        setText("");
                    }
                }
                int yUp = (int) event.getY();
                int width = getMeasuredWidth();
                int height = getMeasuredHeight();
                Log.d("TAG",xUp+"::"+yUp);
                Log.d("TAG","width"+width +"::" +height);


                break;
            default:
                break;
        }
        return super.onTouchEvent(event);
    }
}

然后在values文件加下新建attrs文件,代码如下:

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <declare-styleable name="CheckEditText">
        <attr name="matchType" format="string" />
        <attr name="errorMsg" format="string" />
        <attr name="openShake" format="boolean"/>
    </declare-styleable>
</resources>

在strings添加如下的代码:

<resources>
    <string name="checkPhoneNumber">^((13[0-9])|(15[^4,\\D])|(18[0,5-9]))\\d{8}$</string>
    <string name="checkEmail">^([a-zA-Z0-9]*[-_]?[a-zA-Z0-9]+)*@([a-zA-Z0-9]*[-_]?[a-zA-Z0-9]+)+[\\.][A-Za-z]{2,3}([\\.][A-Za-z]{2})?$</string>
    <string name="checkURL">((http|ftp|https)://)(([a-zA-Z0-9\\._-]+\\.[a-zA-Z]{2,6})|([0-9]{1,3}\\.[0-9]{1,3}\\.[0-9]{1,3}\\.[0-9]{1,3}))(:[0-9]{1,4})*(/[a-zA-Z0-9\\&amp;%_\\./-~-]*)?</string>
</resources>

如此一来,也就完全的实现了自定义的功效。使用方法还是同方法1。

(ps,这个自定义的控件还是有一个小问题,因为采用的是 当控件失去焦点时去判断里面的内容是否合法,当没有其他控件能让这个控件去失去焦点的时候,就无法进行判断是否合法的逻辑。目前找到的方式是在父布局中分发点击事件,但是如此就无法完全达到将此控件独立出来简单使用的目的、如果有什么好方法,还请多多指教!)。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,100评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,308评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,718评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,275评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,376评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,454评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,464评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,248评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,686评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,974评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,150评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,817评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,484评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,140评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,374评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,012评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,041评论 2 351

推荐阅读更多精彩内容