覆盖getText的EditText

3

我想重写EditText的getText()方法。

我接收到这样的字符串:"12,345,678"

我的目的只是去掉逗号并返回Editable,但当我使用我的代码时,出现了一个错误。

public class AmountEditText extends EditText {
    @Override
    public Editable getText() {
        Editable s = super.getText();
        if(s!=null && s.length()>0) {
            if (s.toString().contains(",")) {
                return new SpannableStringBuilder(s.toString().replace(",", ""));
            }
        }
        return s;
    }
    private TextWatcher watcher = new TextWatcher() {



        @Override
        public void beforeTextChanged(CharSequence s, int start, int count, int after) {

        }

        @Override
        public void onTextChanged(CharSequence s, int start, int before, int count) {
            int position = getSelectionStart();
            int nbCommaBefore;
            int nbCommaAfter;
            String str = s.toString();
            String finalStr;
            String formattedStr;
            nbCommaBefore  = str.length() - str.replace(",", "").length();
            boolean containsDot = false;
            if (str.contains(".")) {
                containsDot = true;
                formattedStr = str.split("\\.")[0];

            } else {
                formattedStr = str;
            }
            if (!s.toString().isEmpty()) {
                removeTextChangedListener(watcher);
                formattedStr = formattedStr.replace(",", "");
                formattedStr = formattedStr.replaceAll("(\\d)(?=(\\d{3})+$)", "$1,");
                if (containsDot) {
                    if (str.split("\\.").length != 1) {
                        finalStr = formattedStr + "." + str.split("\\.")[1].replace(",", "");
                    } else {
                        finalStr = formattedStr + ".";
                    }
                } else {
                    finalStr = formattedStr;
                }
                nbCommaAfter  = finalStr.length() - finalStr.replace(",", "").length();
                setText(finalStr);
                if (position == str.length()){
                    setSelection(finalStr.length());
                }
                else if (position == 0)
                {
                    setSelection(0);
                }
                else if (nbCommaBefore < nbCommaAfter){
                    setSelection(position + 1);
                }

                else if (nbCommaAfter < nbCommaBefore){
                    setSelection(position - 1);
                }
                else{
                    setSelection(position);
                }
                addTextChangedListener(watcher);
            }
        }

        @Override
        public void afterTextChanged(Editable s) {
        }

    };

    public AmountEditText(Context context) {
        this(context, null);
    }

    public AmountEditText(Context context, AttributeSet attrs) {
        super(context, attrs);
        addTextChangedListener(watcher);
    }

    public AmountEditText(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        addTextChangedListener(watcher);
    }
}

E/MessageQueue-JNI: 在处理消息队列回调时出现异常: handleReceiveCallback E/MessageQueue-JNI: java.lang.IndexOutOfBoundsException: setSpan (0 ... 5) 超出了长度为4的范围 at android.text.SpannableStringBuilder.checkRange(SpannableStringBuilder.java:1265) at android.text.SpannableStringBuilder.setSpan(SpannableStringBuilder.java:684) at android.text.SpannableStringBuilder.setSpan(SpannableStringBuilder.java:677) at android.widget.SpellChecker$SpellParser.setRangeSpan(SpellChecker.java:532) at android.widget.SpellChecker$SpellParser.parse(SpellChecker.java:515) at android.widget.SpellChecker.spellCheck(SpellChecker.java:242) at android.widget.Editor.updateSpellCheckSpans(Editor.java:679) at android.widget.Editor.sendOnTextChanged(Editor.java:1249) at android.widget.TextView.sendOnTextChanged(TextView.java:8191) at android.widget.TextView.setText(TextView.java:4483) at android.widget.TextView.setText(TextView.java:4337) at android.widget.EditText.setText(EditText.java:89) at android.widget.TextView.setText(TextView.java:4312) at org.newtonproject.newpay.widgetlib.AmountEditText$1.onTextChanged(AmountEditText.java:74)

需要澄清的是,错误并不是来自于我的onTextChanged方法,没有getText() override时一切正常。

编辑: 用户可以输入数字,我将添加一些逗号以格式化数字。但是,当我覆盖getText()时,我想删除这些逗号,这样我就不必每次过滤getText()的返回结果了。


第74行是哪一行? - Pier Giorgio Misley
感谢您的解释 :) - Pier Giorgio Misley
错误不是来自于我的onTextChanged方法。崩溃日志显示它是来自这个方法的。你应该逐步调试这个方法并注意任何使用基于字符串长度的索引的地方,因为那通常是出错的地方。 - Markus Kauppinen
@PierGiorgioMisley 我编辑了我的帖子以明确我的目的。 - Itoun
“将添加一些逗号以格式化数字。但是当我重写getText()时,我想删除这些逗号,这样我就不必每次过滤getText()的返回值。” - 你不能在输入输出时应用DecimalFormat吗?这似乎会更简单和更健壮。 - PPartisan
显示剩余4条评论
4个回答

2

好的,我调试了一下,发现问题出在那行代码上。

if (position == str.length()){
    setSelection(finalStr.length());
}

lenght() 方法的参数基于0,如果超出了范围就会产生错误

将你的代码更改为以下内容即可正常运行

if (position == str.length()){
   setSelection(finalStr.length() - 1);
}

如果需要,完整的代码在此处(我使用了AppCompatEditText,但是它们是相同的):
public class AmountEditText extends android.support.v7.widget.AppCompatEditText {
    @Override
    public Editable getText() {
        Editable s = super.getText();
        if(s!=null && s.length()>0) {
            if (s.toString().contains(",")) {
                return new SpannableStringBuilder(s.toString().replace(",", ""));
            }
        }
        return s;
    }
    private TextWatcher watcher = new TextWatcher() {



        @Override
        public void beforeTextChanged(CharSequence s, int start, int count, int after) {

        }

        @Override
        public void onTextChanged(CharSequence s, int start, int before, int count) {
            int position = getSelectionStart();
            int nbCommaBefore;
            int nbCommaAfter;
            String str = s.toString();
            String finalStr;
            String formattedStr;
            nbCommaBefore  = str.length() - str.replace(",", "").length();
            boolean containsDot = false;
            if (str.contains(".")) {
                containsDot = true;
                formattedStr = str.split("\\.")[0];

            } else {
                formattedStr = str;
            }
            if (!s.toString().isEmpty()) {
                removeTextChangedListener(watcher);
                formattedStr = formattedStr.replace(",", "");
                formattedStr = formattedStr.replaceAll("(\\d)(?=(\\d{3})+$)", "$1,");
                if (containsDot) {
                    if (str.split("\\.").length != 1) {
                        finalStr = formattedStr + "." + str.split("\\.")[1].replace(",", "");
                    } else {
                        finalStr = formattedStr + ".";
                    }
                } else {
                    finalStr = formattedStr;
                }
                nbCommaAfter  = finalStr.length() - finalStr.replace(",", "").length();
                setText(finalStr);
                if (position == str.length()){
                    setSelection(finalStr.length() - 1);
                }
                else if (position == 0)
                {
                    setSelection(0);
                }
                else if (nbCommaBefore < nbCommaAfter){
                    setSelection(position + 1);
                }

                else if (nbCommaAfter < nbCommaBefore){
                    setSelection(position - 1);
                }
                else{
                    setSelection(position);
                }
                addTextChangedListener(watcher);
            }
        }

        @Override
        public void afterTextChanged(Editable s) {
        }

    };

    public AmountEditText(Context context) {
        this(context, null);
    }

    public AmountEditText(Context context, AttributeSet attrs) {
        super(context, attrs);
        addTextChangedListener(watcher);
    }

    public AmountEditText(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        addTextChangedListener(watcher);
    }
}

最初的回答。如果有帮助,请告诉我!

1
谢谢你的帮助,虽然它有所帮助,但最终我会像另一个用户建议的那样实现一个全新的方法。谢谢! - Itoun

1
在您的情况下,您无法同时覆盖 getText()、调整大小和使用 TextWatcher。查看下面的Android源代码,您将明白原因。

SpannableStringBuilder.java

public void setSpan(Object what, int start, int end, int flags) {
    setSpan(true, what, start, end, flags, true/*enforceParagraph*/);
}

private void setSpan(boolean send, Object what, int start, int end, int flags,
            boolean enforceParagraph) {
    checkRange("setSpan", start, end);
}

private void checkRange(final String operation, int start, int end) {
    ...
    int len = length();
    if (start > len || end > len) {
        throw new IndexOutOfBoundsException(operation + " " +
                    region(start, end) + " ends beyond length " + len); // here is you exception
    }
}

public int length() {
    return mText.length - mGapLength;
}

SpellChecker.java

private void setRangeSpan(Editable editable, int start, int end) {
    ...
    editable.setSpan(mRange, start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
}

public void parse(int start, int end) {
    ...
    if (parseEnd > start) {
        setRangeSpan((Editable) mTextView.getText(), start, parseEnd); // I think the error happened from here, they use your getText() function here and receive shorter string, but the start, parseEnd still stick with original string
        parse();
    }
}

解决方案
您可以简单地定义一个新的函数,如getBeautifulText()


0

试试这个,它会帮助你的。

editText.addTextChangedListener(new TextWatcher() {
            boolean isEdiging;

            public void onTextChanged(CharSequence s, int start, int before, int count) {
            }

            public void beforeTextChanged(CharSequence s, int start, int count, int after) {
            }

            public void afterTextChanged(Editable s) {
                if (isEdiging) return;
                isEdiging = true;
                String str = s.toString().replaceAll("[^\\d]", "");
                double s1 = 0;
                try {
                    s1 = Double.parseDouble(str);
                } catch (NumberFormatException e) {
                    e.printStackTrace();
                }
                NumberFormat nf2 = NumberFormat.getInstance(Locale.ENGLISH);
                ((DecimalFormat) nf2).applyPattern("###,###.###");
                s.replace(0, s.length(), nf2.format(s1));

                if (s.toString().equals("0")) {
                    editText.setText("");
                }
                isEdiging = false;
            }
        });

我会澄清我想从这个EditText类中得到什么。 - Itoun

0
根据您的问题要求:
用户可以输入数字,我将添加一些逗号以便格式化数字。但是当我覆盖getText()时,我想删除这些逗号。
我相信您可以使用一个更简单的解决方案,涉及DecimalFormat:
class Formatter {
    private final DecimalFormat f = new DecimalFormat(",###");
    private final DecimalFormat o = new DecimalFormat("#");

    String withCommas(String in) {
        try {
            return withCommas(Long.parseLong(in));
        } catch (NumberFormatException e) {
            e.printStackTrace();
            return withCommas(Long.MIN_VALUE);
        }
    }

    String withCommas(long in) {
        return f.format(in);
    }

    Number stripCommas(String in) {
        try {
            return f.parse(in);
        } catch (ParseException e) {
            return Long.MIN_VALUE;
        }
    }

    String stripCommasAsString(String in) {
        return o.format(stripCommas(in));
    }
}

这将会给出:

final long num = 12345678L;
final Formatter f = new Formatter();

assertEquals("12,345,678", f.withCommas("12345678"));
assertEquals("12,345,678", f.withCommas(num));
assertEquals(num, f.stripCommas("12,345,678");
assertEquals("12345678", f.stripCommasAsString("12,345,678"));

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接