方案一:
方案二:通过setFilter()方法设置过滤器
方案三:通过布局文件中,控件的属性来控制
最近在开发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)
参数简介
可以看出我们可以获取到原本输入的字符串,还有我们即将输入的字符串,然后关于光标所在的位置(可以的出来的),而且我们还可以得到原字符串和即将输入字符串的起始坐标,那么我们就来搞事情吧。废话不多说,直接上代码
/**
* 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);