【问题标题】:Decimal input edittext in javajava中的十进制输入edittext
【发布时间】:2019-12-18 07:13:37
【问题描述】:

我想要一个编辑文本,其值为 0.00,并且光标必须在左侧。当用户按下 1 时,它应该是 1.00,当用户按下 2 时,它应该是 12.00,当用户按下“。”时,它应该是 12.00。光标应移动到小数点的另一侧。当用户按 3 移动到小数点的另一侧后,它应该是 12.30,按 4 应该是 12.34。同样,在背压上它应该是 12.30 -> 12.00 -> 1.00 -> 0.00.sample

 import android.text.Editable;
import android.text.TextWatcher;
import android.view.KeyEvent;
import android.view.View;
import android.widget.EditText;

import java.util.Locale;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

public class AmountTextWatcher implements TextWatcher {

    private EditText editText;
    private int num_npt;
    private String previous;
    private boolean isBackSpacePressed;

    public AmountTextWatcher(EditText editText, int num_npt) {
        this.editText = editText;
        this.num_npt = num_npt;
    }

    @Override
    public void beforeTextChanged(CharSequence s, int start, int count, int after) {
        /*if (StorageUtils.INTERNAL_TRANSFER_AMOUNT != 0.00)
            previous = String.format(Locale.getDefault(), "%.2f", StorageUtils.INTERNAL_TRANSFER_AMOUNT);
        else*/
        if (after < count) {
            isBackSpacePressed = true;
        } else {
            isBackSpacePressed = false;
        }
        previous = editText.getText().toString();
    }

    @Override
    public void onTextChanged(CharSequence s, int start, int before, int count) {
        editText.removeTextChangedListener(this);


        if (isBackSpacePressed) {
            String old_val = previous;

            int point = old_val.indexOf(".");
            if (point == -1) point = old_val.length();

            int selectionStart = editText.getSelectionStart() + 1;
            int selectionEnd = editText.getSelectionEnd();

            String new_val = "";
            if (selectionStart < selectionEnd) {
                if (selectionEnd <= point) {
                    new_val = old_val.substring(0, selectionStart) + old_val.substring(selectionEnd);
                } else if (selectionStart > point) {
                    new_val = old_val.substring(0, selectionStart) + "0" + old_val.substring(selectionEnd);
                    if (selectionEnd - selectionStart == num_npt) new_val += "0";
                } else if (selectionStart <= point && selectionEnd > point) {
                    new_val = old_val.substring(0, selectionStart) + ".";
                    for (int z = 0; z < selectionEnd - point - 1; z++) new_val += "0";
                    new_val += old_val.substring(selectionEnd);
                }

                if (new_val.indexOf(".") == 0) new_val = "0" + new_val;

            } else if (selectionStart == 1 && point == 1) {
                new_val = "0" + old_val.substring(1);
                selectionStart = 0;
            } else if (selectionStart > 0 && selectionStart <= point) {
                new_val = old_val.substring(0, selectionStart - 1) + old_val.substring(selectionStart);
                selectionStart--;
            } else if (selectionStart == point + 1) {
                new_val = old_val;
                selectionStart = point;
            } else if (selectionStart > point + 1) {
                new_val = old_val.substring(0, selectionStart - 1) + old_val.substring(selectionStart) + "0";
                selectionStart--;
            } else {
                new_val = old_val;
            }

            if (new_val.equals("")) new_val = "0";
            editText.setText(new_val);
            editText.setSelection(selectionStart);
            editText.addTextChangedListener(this);
        } else {

            int point = previous.indexOf(".");
            if (point == -1) point = previous.length();

            int selectionStart = editText.getSelectionStart() - 1;
            int selectionEnd = editText.getSelectionEnd() - 1;

            String c = s.subSequence(start, start + count).toString();
            if (c.equals("-") || c.equals(" ")) {
                editText.setText(previous);
                editText.setSelection(selectionStart);
                editText.addTextChangedListener(this);
                return;
            }
            String new_val = "";

            if (selectionStart > -1 && selectionStart < editText.getText().length() - 1) {
                if (c != null && c.length() > 0) {
                    if (c.equals(",")) c = ".";
                    if (Character.isDigit(c.charAt(0)) || c.equals(".")) {
                        if (selectionStart < selectionEnd) {
                            if (selectionEnd <= point) {
                                new_val = previous.substring(0, selectionStart) + c + previous.substring(selectionEnd);
                            } else if (selectionStart > point) {
                                new_val = previous.substring(0, selectionStart) + c + previous.substring(selectionEnd);
                                for (int x = 0; x < selectionEnd - selectionStart - 1; x++)
                                    new_val += "0";
                            } else if (selectionStart <= point && selectionEnd > point && !c.equals(".")) {
                                new_val = previous.substring(0, selectionStart) + c + ".";
                                for (int y = 0; y < selectionEnd - point - 1; y++)
                                    new_val += "0";
                                new_val += previous.substring(selectionEnd);
                            }
                        } else if ((num_npt == 0 && previous.equals("0")) || (previous.substring(0, 2).equals("0.") && selectionStart == 0)) {
                            if (c.equals(".")) {
                                new_val = previous;
                                editText.setSelection(0);
                            } else
                                new_val = c + previous.substring(1);
                        } else if (selectionStart > point && selectionStart <= point + num_npt) {
                            if (c.equals(".")) {
                                if (editText.getSelectionStart() > previous.indexOf("."))
                                    new_val = previous.substring(0, selectionStart) + previous.substring(selectionStart);
                                else
                                    new_val = previous.substring(0, selectionStart - 1) + c + previous.substring(selectionStart);
                                editText.setSelection(selectionStart - 1);
                            } else
                                new_val = previous.substring(0, selectionStart) + c + previous.substring(selectionStart + 1);
                        }
                        // this is the code where our edittext lags :(
                        else {
                            if (c.equals("."))
                                if (editText.getSelectionStart() - 1 == 0) {
                                    if (c.equals("."))
                                        new_val = previous.substring(0, 1) + previous.substring(selectionEnd + 1);
                                    else
                                        new_val = previous.substring(0, 1) + c + previous.substring(selectionEnd + 2);
                                } else {
                                    if (selectionStart > 0 && selectionStart < previous.indexOf("."))
                                        new_val = previous.substring(0, selectionStart + 1) + previous.substring(selectionEnd + 1);
                                    else
                                        new_val = previous.substring(0, selectionStart) + c + previous.substring(selectionEnd + 1);
                                }
                            else
                                new_val = previous.substring(0, selectionStart) + c + previous.substring(selectionEnd);
                            if (new_val.substring(new_val.length() - 1).equals("."))
                                new_val = new_val.substring(0, new_val.length() - 1);
                        }
                        // *********//

                        if (new_val.substring(0, 1).equals("0") && !new_val.substring(0, 2).equals("0.")) {
                            new_val = new_val.substring(1);
                            selectionStart--;
                        }

                        Pattern pattern = Pattern.compile("^[0-9]+(\\.[0-9]{0," + num_npt + "})?$");
                        Matcher matcher = pattern.matcher(new_val);
                        boolean doNothing = false;
                        if (matcher.matches()) {
                            editText.setText(new_val);
                            if (!c.equals("."))
                                selectionStart++;
                            else if (c.equals(".") && selectionStart == previous.indexOf("."))
                                selectionStart++;
                            editText.setSelection(selectionStart);
                        } else if (c.equals(".")) {
                            selectionStart++;
                            editText.setSelection(selectionStart);
                        }
                        editText.addTextChangedListener(this);
                    }
                } else {
                    editText.setText(previous);
                    editText.setSelection(selectionStart);
                    editText.addTextChangedListener(this);
                }
            } else {
                int length = editText.getText().length();
                if (selectionStart < 0)
                    selectionStart = 0;
                else if (selectionStart > length || selectionEnd < length)
                    selectionStart = length - 1;
                editText.setText(previous);
                editText.setSelection(selectionStart);
                editText.addTextChangedListener(this);
            }
        }
    }

    @Override
    public void afterTextChanged(Editable s) {

    }
}

【问题讨论】:

  • 这段代码可以随心所欲地工作。你面临的困难是什么?
  • 这不是好的解决方案。它在写作时滞后。错过很多数字,用户体验很差
  • @Md 你试过这个代码吗?
  • 有没有人找到一些优化的答案?

标签: java android decimal currency-formatting


【解决方案1】:

尝试在 Java 中使用 DecimalFormat 类。它可能会以更好的方式帮助您。

String pattern = "##.##";
DecimalFormat decimalFormat = new DecimalFormat(pattern);
String output = decimalFormat.format( YOUR NUMBER HERE );

【讨论】:

  • 如果您使用 DecimalFormat,请避免使用 textwatcher
  • thanku @arnab 你能把代码sn-p发给我吗,我可以试试。我已经尝试了互联网上所有可用的东西,但无法实现这一目标
【解决方案2】:

您可以在onTextChanged 中使用String.format("%.2f", d) 进行格式化

更新 1:

etText.setFilters(new InputFilter[]{new DecimalDigitsInputFilter(5, 2)});
class DecimalDigitsInputFilter implements InputFilter {
   private Pattern mPattern;
   DecimalDigitsInputFilter(int digitsBeforeZero, int digitsAfterZero) {
      mPattern = Pattern.compile("[0-9]{0," + (digitsBeforeZero - 1) + "}+((\\.[0-9]{0," + (digitsAfterZero - 1) + "})?)||(\\.)?");
   }
   @Override
   public CharSequence filter(CharSequence source, int start, int end, Spanned dest, int dstart, int dend) {
      Matcher matcher = mPattern.matcher(dest);
      if (!matcher.matches())
      return "";
      return null;
   }
}

【讨论】:

  • 我已经尝试过了,它会舍入双精度值。我想要的是在小数点两边写数字
  • 感谢@Niraj 的回答。它只限制小数点前后的位数,但不处理小数点点击的后压情况和移位情况,它只是限制小数点前后的位数
  • 我觉得我的问题只能通过textwatcher来解决。没有其他解决办法
【解决方案3】:

xml

<EditText
    android:id="@+id/et_des_input"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:inputType="numberDecimal"
    android:gravity="right"
    android:text="0.00"
    android:digits=".0123456789"/>

设置 TextWacher

EditText et_des_input = (EditText) findViewById(R.id.et_des_input);
et_des_input.addTextChangedListener(new AmountTextWatcher(et_des_input, 2));

TextWacher 类与您发布的相同。它正在按您的要求工作。

See the output

【讨论】:

  • 我知道这个解决方案有效,但问题是当你在三星 s7 上测试它时它会滞后。它不像其他edittext那样写。有没有更好的解决方案。
【解决方案4】:

我已经为两位小数做了这个解决方案。请大家检查一下。这适用于小数点后两位。

公共类 TxLabzTextWatcher 实现 TextWatcher {

EditText editText;
String tempText = "";
StringBuilder stringBuilder;
Context ctx;
boolean isBackSpacePressed = false;
int digitsAfteZero;
int count;
boolean isFirstDigitBeforeDecimalZero = true;


public TxLabzTextWatcher(Context context, EditText editText, int digitsAfterZero) {

    this.editText = editText;
    tempText = editText.getText().toString();
    stringBuilder = new StringBuilder();
    this.ctx = context;
    this.digitsAfteZero = digitsAfterZero;
    count = 0;

}

@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
    if (after < count) {
        isBackSpacePressed = true;
    } else {
        isBackSpacePressed = false;
    }

}

@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
    editText.removeTextChangedListener(this);

    if (stringBuilder.toString().isEmpty()) {
        stringBuilder.append("0.00");
    }

    if (!isBackSpacePressed) {
        editText.removeTextChangedListener(this);
        stringBuilder.setLength(0);
        stringBuilder.append(s);
        char ch = s.charAt(start + before);

        if (!isClickDot(ch)) {
            if (stringBuilder.length() > 0 && stringBuilder.indexOf(".") - 1 == editText.getSelectionStart() && stringBuilder.charAt(stringBuilder.indexOf(".") - 1) == '0' && ch != '0' && isFirstDigitBeforeDecimalZero) {
                stringBuilder.deleteCharAt(stringBuilder.indexOf(".") - 1);
                editText.setText(stringBuilder.toString());
                editText.setSelection(editText.getText().length() - 3);
                isFirstDigitBeforeDecimalZero = false;
            } else if (ch == '0' && stringBuilder.indexOf(".") - 1 == editText.getSelectionStart()) {
                stringBuilder.deleteCharAt(stringBuilder.indexOf(".") - 1);
                editText.setText(stringBuilder.toString());
                editText.setSelection(editText.getText().length() - 3);
                isFirstDigitBeforeDecimalZero = false;
            }
        }


        // if i press dot then move to the other side of
        if (isClickDot(ch) && StringUtils.countMatches(stringBuilder.toString(), ".") >= 1) {
            stringBuilder.deleteCharAt(s.toString().indexOf('.'));
            editText.setText(stringBuilder.toString());
            editText.setSelection(s.toString().indexOf('.') + 1);
        }

        // replace first digit after decimal
        if (stringBuilder.indexOf(".") + 2 == editText.getSelectionStart() && !isClickDot(ch)) {
            int currIndex = stringBuilder.indexOf(".") + 1;
            stringBuilder.deleteCharAt(currIndex + 1);
            editText.setText(stringBuilder.toString());
            editText.setSelection(currIndex + 1);
            count++;
        }

        // replace last digit after decimal
        if (stringBuilder.indexOf(".") + 3 == editText.getSelectionStart() && !isClickDot(ch)) {
            int currIndex = stringBuilder.indexOf(".") + 2;
            stringBuilder.deleteCharAt(currIndex + 1);
            editText.setText(stringBuilder.toString());
            editText.setSelection(currIndex + 1);
            count++;
        }

        // if string builder after decimal have more than 2 length then delete it
        if (editText.getSelectionEnd() == stringBuilder.length() && stringBuilder.subSequence(stringBuilder.indexOf(".") + 1, stringBuilder.length()).length() > digitsAfteZero) {
            stringBuilder.deleteCharAt(editText.getSelectionEnd() - 1);
            editText.setText(stringBuilder.toString());
            editText.setSelection(stringBuilder.length());
        }


        editText.addTextChangedListener(this);
    } else {
        editText.removeTextChangedListener(this);

        if (s.toString().length() == 3 && s.toString().contains(".") && s.toString().substring(0, s.toString().indexOf('.')).length() == 0) {
            // place zero befor decimal
            String zeroBeforeDecimal = "0" + s.toString();
            stringBuilder.setLength(0);
            stringBuilder.append(zeroBeforeDecimal);
            editText.setText(stringBuilder.toString());
            isFirstDigitBeforeDecimalZero = true;
        }

        // if cursor is moved to other side of decimal now move it to left side of decimal
        if (stringBuilder.toString().contains(".") && editText.getSelectionStart() == stringBuilder.indexOf(".")) {
            stringBuilder.setLength(0);
            stringBuilder.append(s);
            stringBuilder.insert(stringBuilder.length() - 2, ".");
            editText.setText(stringBuilder.toString());
            editText.setSelection(stringBuilder.indexOf("."));
        }

        // if cursor is after one digit after decimal
        if (stringBuilder.indexOf(".") + 1 == editText.getSelectionStart()) {
            int currIndex = stringBuilder.indexOf(".") + 1;
            stringBuilder.deleteCharAt(currIndex);
            stringBuilder.insert(currIndex, "0");
            editText.setText(stringBuilder.toString());
            editText.setSelection(currIndex - 1);
        }

        // if cursor is on last digit then replace it with zero
        if (stringBuilder.indexOf(".") + 2 == editText.getSelectionStart()) {
            int currIndex = stringBuilder.indexOf(".") + 2;
            stringBuilder.deleteCharAt(currIndex);
            stringBuilder.insert(currIndex, "0");
            editText.setText(stringBuilder.toString());
            editText.setSelection(currIndex);
        }

        editText.addTextChangedListener(this);
    }
}

@Override
public void afterTextChanged(Editable arg0) {
}

boolean isClickDot(char dot) {
    if (dot == '.')
        return true;
    return false;
}

}

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2011-06-18
    • 2013-07-22
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2015-12-13
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多