Android EditText实现分割输入内容

所属分类: 软件编程 / Android 阅读数: 42
收藏 0 赞 0 分享

在项目中可能会有许多需要输入手机号码、银行卡号或者身份证号等内容的输入框。如果直接输入的话将会是一堆号码堆在一起,第一是不太美观,第二也容易出错,用户体验不太好。但是若将输入的号码按特定格式进行分割将会大大提高用户体验!

以下是对常用的号码进行简单封装的自定义输入框控件,方便我们在开发过程中使用:

  • 该控件支持xml属性指定,也支持代码指定;
  • 该控件支持类型分别为电话号码(000 0000 0000)、银行卡号(0000 0000 0000 0000 000)和身份证号(000000 0000 0000 0000)。

效果图

自定义EditText

/**
 * @Description 分割输入框
 * @Author 一花一世界
 */
public class ContentWithSpaceEditText extends EditText {

  public static final int TYPE_PHONE = 0;
  public static final int TYPE_CARD = 1;
  public static final int TYPE_IDCARD = 2;
  private int maxLength = 100;
  private int contentType;
  private int start, count, before;
  private String digits;

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

  public ContentWithSpaceEditText(Context context, AttributeSet attrs) {
    super(context, attrs);
    parseAttributeSet(context, attrs);
  }

  public ContentWithSpaceEditText(Context context, AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);
    parseAttributeSet(context, attrs);
  }

  private void parseAttributeSet(Context context, AttributeSet attrs) {
    TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.ContentWithSpaceEditText, 0, 0);
    contentType = ta.getInt(R.styleable.ContentWithSpaceEditText_type, 0);
    ta.recycle();
    initType();
    setSingleLine();
    addTextChangedListener(watcher);
  }

  private void initType() {
    if (contentType == TYPE_PHONE) {
      maxLength = 13;
      digits = "0123456789 ";
      setInputType(InputType.TYPE_CLASS_NUMBER);
    } else if (contentType == TYPE_CARD) {
      maxLength = 23;
      digits = "0123456789 ";
      setInputType(InputType.TYPE_CLASS_NUMBER);
    } else if (contentType == TYPE_IDCARD) {
      maxLength = 21;
      digits = "0123456789xX ";
      setInputType(InputType.TYPE_CLASS_TEXT);
    }
    setFilters(new InputFilter[]{new InputFilter.LengthFilter(maxLength)});
  }

  @Override
  public void setInputType(int type) {
    super.setInputType(type);
    // setKeyListener要在setInputType后面调用,否则无效
    if (!TextUtils.isEmpty(digits)) {
      setKeyListener(DigitsKeyListener.getInstance(digits));
    }
  }

  private TextWatcher watcher = 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) {
      ContentWithSpaceEditText.this.start = start;
      ContentWithSpaceEditText.this.before = before;
      ContentWithSpaceEditText.this.count = count;
    }

    @Override
    public void afterTextChanged(Editable s) {
      if (s == null) {
        return;
      }
      //判断是否是在中间输入,需要重新计算
      boolean isMiddle = (start + count) < (s.length());
      //在末尾输入时,是否需要加入空格
      boolean isNeedSpace = false;
      if (!isMiddle && isSpace(s.length())) {
        isNeedSpace = true;
      }
      if (isMiddle || isNeedSpace || count > 1) {
        String newStr = s.toString();
        newStr = newStr.replace(" ", "");
        StringBuilder sb = new StringBuilder();
        int spaceCount = 0;
        for (int i = 0; i < newStr.length(); i++) {
          sb.append(newStr.substring(i, i + 1));
          //如果当前输入的字符下一位为空格(i+1+1+spaceCount),因为i是从0开始计算的,所以一开始的时候需要先加1
          if (isSpace(i + 2 + spaceCount)) {
            sb.append(" ");
            spaceCount += 1;
          }
        }
        removeTextChangedListener(watcher);
        s.replace(0, s.length(), sb);
        //如果是在末尾的话,或者加入的字符个数大于零的话(输入或者粘贴)
        if (!isMiddle || count > 1) {
          setSelection(s.length() <= maxLength ? s.length() : maxLength);
        } else if (isMiddle) {
          //如果是删除
          if (count == 0) {
            //如果删除时,光标停留在空格的前面,光标则要往前移一位
            if (isSpace(start - before + 1)) {
              setSelection((start - before) > 0 ? start - before : 0);
            } else {
              setSelection((start - before + 1) > s.length() ? s.length() : (start - before + 1));
            }
          }
          //如果是增加
          else {
            if (isSpace(start - before + count)) {
              setSelection((start + count - before + 1) < s.length() ? (start + count - before + 1) : s.length());
            } else {
              setSelection(start + count - before);
            }
          }
        }
        addTextChangedListener(watcher);
      }
    }
  };

  private boolean isSpace(int length) {
    if (contentType == TYPE_PHONE) {
      return isSpacePhone(length);
    } else if (contentType == TYPE_CARD) {
      return isSpaceCard(length);
    } else if (contentType == TYPE_IDCARD) {
      return isSpaceIDCard(length);
    }
    return false;
  }

  private boolean isSpacePhone(int length) {
    return length >= 4 && (length == 4 || (length + 1) % 5 == 0);
  }

  private boolean isSpaceCard(int length) {
    return length % 5 == 0;
  }

  private boolean isSpaceIDCard(int length) {
    return length > 6 && (length == 7 || (length - 2) % 5 == 0);
  }

  public void setContentType(int contentType) {
    this.contentType = contentType;
    initType();
  }

  public String getTextWithoutSpace() {
    return super.getText().toString().replace(" ", "");
  }

  /**
   * @Description 内容校验
   */
  public boolean isContentCheck() {
    String text = getTextWithoutSpace();
    if (contentType == TYPE_PHONE) {
      if (TextUtils.isEmpty(text)) {
        ToastUtil.showText(UIUtils.getString(R.string.phone_number_not_empty));
      } else if (text.length() < 11) {
        ToastUtil.showText(UIUtils.getString(R.string.phone_number_incorrect_length));
      } else {
        return true;
      }
    } else if (contentType == TYPE_CARD) {
      if (TextUtils.isEmpty(text)) {
        ToastUtil.showText(UIUtils.getString(R.string.bank_card_not_empty));
      } else if (text.length() < 16) {
        ToastUtil.showText(UIUtils.getString(R.string.bank_card_incorrect_length));
      } else {
        return true;
      }
    } else if (contentType == TYPE_IDCARD) {
      if (TextUtils.isEmpty(text)) {
        ToastUtil.showText(UIUtils.getString(R.string.identity_number_not_empty));
      } else if (text.length() < 18) {
        ToastUtil.showText(UIUtils.getString(R.string.identity_number_incorrect_length));
      } else {
        return true;
      }
    }
    return false;
  }
}

配置attrs.xml

<?xml version="1.0" encoding="utf-8"?>
<resources>
  <declare-styleable name="ContentWithSpaceEditText">
    <attr name="type" format="enum">
      <enum name="phone" value="0" />
      <enum name="card" value="1" />
      <enum name="IDCard" value="2" />
    </attr>
  </declare-styleable>
</resources>

布局文件中使用

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  xmlns:app="http://schemas.android.com/apk/res-auto"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  android:background="@color/theme_bg"
  android:orientation="vertical">

  <com.wiggins.splitinput.widget.TitleView
    android:id="@+id/titleView"
    android:layout_width="match_parent"
    android:layout_height="wrap_content" />

  <LinearLayout
    android:layout_width="match_parent"
    android:layout_height="@dimen/item_normal"
    android:layout_margin="@dimen/margin_normal"
    android:gravity="center_vertical"
    android:orientation="horizontal">

    <TextView
      android:layout_width="@dimen/btn_width_normal"
      android:layout_height="match_parent"
      android:gravity="center_vertical"
      android:text="@string/phone_number"
      android:textColor="@color/blue"
      android:textSize="@dimen/font_normal" />

    <com.wiggins.splitinput.widget.ContentWithSpaceEditText
      android:id="@+id/edt_phone_input"
      android:layout_width="match_parent"
      android:layout_height="@dimen/item_normal"
      android:background="@color/white"
      android:gravity="center"
      android:hint="@string/please_enter_content"
      android:inputType="number"
      android:textColor="@color/blue"
      android:textColorHint="@color/gray"
      android:textCursorDrawable="@null"
      android:textSize="@dimen/font_normal"
      app:type="phone" />
  </LinearLayout>

  <LinearLayout
    android:layout_width="match_parent"
    android:layout_height="@dimen/item_normal"
    android:layout_margin="@dimen/margin_normal"
    android:gravity="center_vertical"
    android:orientation="horizontal">

    <TextView
      android:layout_width="@dimen/btn_width_normal"
      android:layout_height="match_parent"
      android:gravity="center_vertical"
      android:text="@string/bank_card_number"
      android:textColor="@color/blue"
      android:textSize="@dimen/font_normal" />

    <com.wiggins.splitinput.widget.ContentWithSpaceEditText
      android:id="@+id/edt_bank_card_input"
      android:layout_width="match_parent"
      android:layout_height="@dimen/item_normal"
      android:background="@color/white"
      android:gravity="center"
      android:hint="@string/please_enter_content"
      android:inputType="number"
      android:textColor="@color/blue"
      android:textColorHint="@color/gray"
      android:textCursorDrawable="@null"
      android:textSize="@dimen/font_normal"
      app:type="card" />
  </LinearLayout>

  <LinearLayout
    android:layout_width="match_parent"
    android:layout_height="@dimen/item_normal"
    android:layout_margin="@dimen/margin_normal"
    android:gravity="center_vertical"
    android:orientation="horizontal">

    <TextView
      android:layout_width="@dimen/btn_width_normal"
      android:layout_height="match_parent"
      android:gravity="center_vertical"
      android:text="@string/identity_number"
      android:textColor="@color/blue"
      android:textSize="@dimen/font_normal" />

    <com.wiggins.splitinput.widget.ContentWithSpaceEditText
      android:id="@+id/edt_identity_input"
      android:layout_width="match_parent"
      android:layout_height="@dimen/item_normal"
      android:background="@color/white"
      android:gravity="center"
      android:hint="@string/please_enter_content"
      android:inputType="number"
      android:textColor="@color/blue"
      android:textColorHint="@color/gray"
      android:textCursorDrawable="@null"
      android:textSize="@dimen/font_normal"
      app:type="IDCard" />
  </LinearLayout>
</LinearLayout>

项目地址:传送门

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

更多精彩内容其他人还在看

Android控件系列之CheckBox使用介绍

CheckBox和Button一样,也是一种古老的控件,它的优点在于,不用用户去填写具体的信息,只需轻轻点击,缺点在于只有“是”和“否”两种情况,但我们往往利用它的这个特性,来获取用户的一些信息
收藏 0 赞 0 分享

Android控件系列之EditText使用方法

EditText是接受用户输入信息的最重要控件。通过前面课程的学习,您可能会猜到可以利用EditText.getText()获取它的文本,但真正的项目中,可能没那么简单,需要更多的限制,如文本长度限制,是否数字限制等等
收藏 0 赞 0 分享

Android控件系列之TextView使用介绍

TextView类似一般UI中的Label,TextBlock等控件,只是为了单纯的显示一行或多行文本,本文介绍了Android中文本控件TextView的用法和常用属性的用法
收藏 0 赞 0 分享

asynctask的用法详解

Android UI操作并不是线程安全的并且这些操作必须在UI线程中执行,本文将为您介绍asynctask的用法
收藏 0 赞 0 分享

Android开发 旋转屏幕导致Activity重建解决方法

Android开发文档上专门有一小节解释这个问题。简单来说,Activity是负责与用户交互的最主要机制,接下来为您详细介绍
收藏 0 赞 0 分享

Notification与NotificationManager详细介绍

在Android系统中,发一个状态栏通知还是很方便的。下面我们就来看一下,怎么发送状态栏通知,状态栏通知又有哪些参数可以设置
收藏 0 赞 0 分享

android LinearLayout和RelativeLayout组合实现精确布局方法介绍

用android LinearLayout和RelativeLayout实现精确布局此方法适合很适合新人看
收藏 0 赞 0 分享

android listview优化几种写法详细介绍

这篇文章只是总结下getView里面优化视图的几种写法,需要的朋友可以参考下
收藏 0 赞 0 分享

Android应用开发SharedPreferences存储数据的使用方法

SharedPreferences是Android中最容易理解的数据存储技术,实际上SharedPreferences处理的就是一个key-value(键值对)SharedPreferences常用来存储一些轻量级的数据
收藏 0 赞 0 分享

Android之PreferenceActivity应用详解

为了引入这个概率 首先从需求说起 即:现有某Activity专门用于手机属性设置 那么应该如何做呢
收藏 0 赞 0 分享
查看更多