Android EditText实现分割输入内容
发布时间 - 2026-01-11 00:53:19 点击率:次在项目中可能会有许多需要输入手机号码、银行卡号或者身份证号等内容的输入框。如果直接输入的话将会是一堆号码堆在一起,第一是不太美观,第二也容易出错,用户体验不太好。但是若将输入的号码按特定格式进行分割将会大大提高用户体验!

以下是对常用的号码进行简单封装的自定义输入框控件,方便我们在开发过程中使用:
- 该控件支持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
# EditText
# 分割
# Android实现自定义带删除功能的EditText实例
# Android编程实现实时监听EditText文本输入的方法
# Android中搜索图标和文字居中的EditText实例
# Android中自定义的dialog中的EditText无法弹出输入法解决方案
# Android 设置Edittext获取焦点并弹出软键盘
# Android之EditText控制禁止输入空格和回车
# Android EditText限制输入字符的方法总结
# 是在
# 输入框
# 将会
# 自定义
# 卡号
# 不太
# 是从
# 要在
# 有许多
# 分别为
# 等内容
# 停留在
# 大家多多
# 则要
# 过程中
# 直接输入
# 判断是否
# 传送门
# 不太好
# 第一是
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
如何用狗爹虚拟主机快速搭建网站?
Laravel如何使用Blade组件和插槽?(Component代码示例)
Laravel如何使用缓存系统提升性能_Laravel缓存驱动和应用优化方案
Laravel如何操作JSON类型的数据库字段?(Eloquent示例)
Android中Textview和图片同行显示(文字超出用省略号,图片自动靠右边)
Laravel如何使用Telescope进行调试?(安装和使用教程)
微信小程序 require机制详解及实例代码
活动邀请函制作网站有哪些,活动邀请函文案?
香港服务器部署网站为何提示未备案?
如何彻底删除建站之星生成的Banner?
Laravel怎么在Blade中安全地输出原始HTML内容
Laravel怎么做缓存_Laravel Cache系统提升应用速度的策略与技巧
怎么制作网站设计模板图片,有电商商品详情页面的免费模板素材网站推荐吗?
HTML透明颜色代码在Angular里怎么设置_Angular透明颜色使用指南【详解】
Swift中switch语句区间和元组模式匹配
Laravel如何使用.env文件管理环境变量?(最佳实践)
企业在线网站设计制作流程,想建设一个属于自己的企业网站,该如何去做?
html5如何实现懒加载图片_ intersectionobserver api用法【教程】
google浏览器怎么清理缓存_谷歌浏览器清除缓存加速详细步骤
Laravel如何实现模型的全局作用域?(Global Scope示例)
laravel怎么通过契约(Contracts)编程_laravel契约(Contracts)编程方法
详解免费开源的.NET多类型文件解压缩组件SharpZipLib(.NET组件介绍之七)
Android使用GridView实现日历的简单功能
如何在万网自助建站中设置域名及备案?
php485函数参数是什么意思_php485各参数详细说明【介绍】
Laravel如何处理异常和错误?(Handler示例)
专业型网站制作公司有哪些,我设计专业的,谁给推荐几个设计师兼职类的网站?
HTML5建模怎么导出为FBX格式_FBX格式兼容性及导出步骤【指南】
Laravel怎么实现一对多关联查询_Laravel Eloquent模型关系定义与预加载【实战】
html文件怎么打开证书错误_https协议的html打开提示不安全【指南】
长沙做网站要多少钱,长沙国安网络怎么样?
深圳网站制作公司好吗,在深圳找工作哪个网站最好啊?
如何在VPS电脑上快速搭建网站?
JavaScript如何实现倒计时_时间函数如何精确控制
Laravel如何实现多语言支持_Laravel本地化与国际化(i18n)配置教程
Python正则表达式进阶教程_复杂匹配与分组替换解析
如何在Windows环境下新建FTP站点并设置权限?
Laravel DB事务怎么使用_Laravel数据库事务回滚操作
Laravel如何处理CORS跨域请求?(配置示例)
PHP正则匹配日期和时间(时间戳转换)的实例代码
郑州企业网站制作公司,郑州招聘网站有哪些?
Laravel Eloquent:优雅地将关联模型字段扁平化到主模型中
,网页ppt怎么弄成自己的ppt?
JavaScript如何实现继承_有哪些常用方法
如何用景安虚拟主机手机版绑定域名建站?
Laravel如何使用Laravel Vite编译前端_Laravel10以上版本前端静态资源管理【教程】
JS实现鼠标移上去显示图片或微信二维码
MySQL查询结果复制到新表的方法(更新、插入)
Laravel如何实现多对多模型关联?(Eloquent教程)
lovemo网页版地址 lovemo官网手机登录

