EditText金额输入(限制输入金额大小和小数点后两位)
标签(空格分隔): Android开发
Android中控制EditText输入内容、长度的方法有三种
-
方案一:
- 通过监听EditText的addTextChangedListener方法
-
方案二:通过setFilter()方法设置过滤器
- 也就是自定义一个类实现InputFilter接口,复写filter这个方法在里面进行相关逻辑
-
方案三:通过布局文件中,控件的属性来控制
- 例如 maxLength、inputType、minLength等
需求
最近在开发app的时候,收到了这样的需求,也就是打赏金额,用户可以自定义金额,并且需要用户输入的金额不能大于500并且限制小数点后两位小数,也就是最多输入499.99元。那么这个只能自己自定义了。可以看到,这个需求以上的三种方式,第三种并不能达到这样的逻辑要求,暂时排除,也就是只有方案一和方案二了.
关于google为啥有这样的三种方式控制EditText相关操作,可以查看这个InputFilter详解、TextWatcher详解 所以接下来可以给出最终的一个实现方案,那就是实现InputFilter接口重写filter这个方法来实现上述需求。
给出方案
在方案代码开始之前,我们需要先了解一下filter方法的各个参数的含义:
CharSequence filter (CharSequence source,
int start,
int end,
Spanned dest,
int dstart,
int dend)
参数简介
- source 新输入的字符串
- start 新输入的字符串起始下标,一般为0
- end 新输入的字符串终点下标,一般为source长度-1
- dest 输入之前文本的内容
- dstart 原内容起始坐标 一般为0
- dend 原内容终点坐标,一般为dest长度-1
可以看出我们可以获取到原本输入的字符串,还有我们即将输入的字符串,然后关于光标所在的位置(可以的出来的),而且我们还可以得到原字符串和即将输入字符串的起始坐标,那么我们就来搞事情吧。废话不多说,直接上代码
/**
* Created by ruolanmingyue on 2017/10/26.
*
* @function 用于过滤输入 防止输入大于500元还有就是限制小数点之后两位
*/
public class EditInputFilter implements InputFilter {
/**
* 最大数字
*/
public static final int MAX_VALUE = 500;
/**
* 小数点后的数字的位数
*/
public static final int POINTER_LENGTH = 2;
private static final String POINTER = ".";
Pattern p;
public EditInputFilter() {
//用于匹配输入的是0-9 . 这几个数字和字符
p = Pattern.compile("([0-9]|\\.)*");
}
/**
* source 新输入的字符串
* start 新输入的字符串起始下标,一般为0
* end 新输入的字符串终点下标,一般为source长度-1
* dest 输入之前文本框内容
* dstart 原内容起始坐标,一般为0
* dend 原内容终点坐标,一般为dest长度-1
*/
@Override
public CharSequence filter(CharSequence source, int start, int end,
Spanned dest, int dstart, int dend) {
String sourceText = source.toString();
String destText = dest.toString();
//验证删除等按键
if (TextUtils.isEmpty(sourceText)) {
if (dstart == 0 && destText.indexOf(POINTER) == 1) {//保证小数点不在第一个位置
return "0";
}
return "";
}
Matcher matcher = p.matcher(source);
//已经输入小数点的情况下,只能输入数字
if (destText.contains(POINTER)) {
if (!matcher.matches()) {
return "";
} else {
if (POINTER.equals(source)) { //只能输入一个小数点
return "";
}
}
//验证小数点精度,保证小数点后只能输入两位
int index = destText.indexOf(POINTER);
int length = destText.trim().length() - index;
if (length > POINTER_LENGTH && dstart > index) {
return "";
}
} else {
//没有输入小数点的情况下,只能输入小数点和数字,但首位不能输入小数点和0
if (!matcher.matches()) {
return "";
} else {
if ((POINTER.equals(source)) && dstart == 0) {//第一个位置输入小数点的情况
return "0.";
} else if ("0".equals(source) && dstart == 0) {
return "";
}
}
}
// dstart
//修复当光标定位到第一位的时候 还能输入其他的 这个是为了修复以下的情况
/**
* <>
* 当如下情况的时候 也就是 已经输入了23.45 这个时候限制是500元
* 那么这个时候如果把光标移动2前面 也就是第0位 在输入一个5 那么这个实际的参与下面的
* 判断的sumText > MAX_VALUE 是23.455 这个是不大于 500的 但是实际情况是523 这个时候
* 已经大于500了 所以之前的是存在bug的 这个要进行修正 也就是拿到的比较数应该是523.45 而不是23.455
* 所以有了下面的分隔 也就是 把23.45 (因为这个时候dstart=0) 分隔成 "" 和23.45 然后把 5放到中间
* 进行拼接 也就是 "" + 5 + 23.45 也就是523.45 然后在进行和500比较
* 还有一个比较明显的就是 23.45 这个时候光标在2和3 之间 那么如果修正之前 是23.455 修正之后 dstart = 1
* 这个时候分隔是 "2" "3.45" 这个时候拼接是253.45 然后和500比较 以此类推
* </>
*/
String first = destText.substring(0,dstart);
String second = destText.substring(dstart,destText.length());
// dend
String sum = first + sourceText + second;
//验证输入金额的大小
double sumText = Double.parseDouble(sum);
//这里得到输入完之后需要计算的金额 如果这个金额大于了事先设定的金额,那么久直接返回 不需要加入输入的字符
if (sumText > MAX_VALUE) {
//
Toast.makeText(MyApp.getContext(), MyApp.getContext().getResources().getString(R.string.appreciate_input), Toast.LENGTH_SHORT).show();
return dest.subSequence(dstart, dend);
}
//如果输入的金额小于事先规定的金额
return dest.subSequence(dstart, dend) + sourceText;
}
}
使用方式
EditText editText = new EditText(getContext());
InputFilter[] filters = {new EditInputFilter()};
editText.setFilters(filters);
参考文档
-
Android实现EditText输入金额
- 但是这个文章写的有bug,发现的bug如下两个
- 1、输入金额之后,当手动改edittext里面的光标的时候,比如刚开始输入的是23.22元,这个时候把光标移动到2和3之间,那么我们还可以输入2223.22元,也就是不符合不能大于500元需求,这个在上面的说明中有详细的说明
- 2、当改动光标在.后面的时候,也是可以说入数字的,也就是不符合输入数字限制两位小数要求
- 但是这个文章写的有bug,发现的bug如下两个