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

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开发,学习心得)