您的位置:首页 > 移动开发 > Android开发

EditText金额输入(限制输入金额大小和小数点后两位)

2017-11-25 12:23 447 查看

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){
//用于修复能输入多位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、当改动光标在.后面的时候,也是可以说入数字的,也就是不符合输入数字限制两位小数要求
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  android filter