EditText輸入金額型別

kylinarm發表於2017-09-05

editText預設的屬性裡面是沒有金額型別的,所以要實現這個功能我們就必須自己動手豐衣足食。

一.EditText只允許輸入數字、小數點。

首先要知道金額有兩部分構成,整數部分和小數部分,要實現只輸入數字和小數點很簡單。自己查能很容易查到。

把type設定成InputType.TYPE_NUMBER_FLAG_DECIMAL|InputType.TYPE_CLASS_NUMBER
就行

那這樣做有個缺點,小數部分可以有很多位。而我們知道金額型別的小數部分只能有兩位,所以這個方法不合適。

二.設定字元過濾

網上有很多文章都是這樣寫。

mEdit.setFilters(new InputFilter[]{new InputFilter() {
    @Override
    public CharSequence filter(CharSequence source, int start, int end, Spanned dest, int dstart, int dend) {
        if(source.equals(".") && dest.toString().length() == 0){
            return "0.";
        }
        if(dest.toString().contains(".")){
            int index = dest.toString().indexOf(".");
            int mlength = dest.toString().substring(index).length();
            if(mlength == 3){
                return "";
            }
        }
        return null;
    }
}});

這樣的做法能實現是能實現,但是這樣的寫法沒有考慮到很多種會出現的情況,也就是說只是這樣寫的話,在某些情況下的使用者體驗很不好。

三.設定監聽addTextChangedListener

上面的第二種方法你可以在裡邊加自己的演算法來處理特殊情況,但是我不太喜歡這個方法,直到我看到一個哥們用addTextChangedListener來實現。我才發現,用addTextChangedListener來做金額型別的輸入挺合適的。雖然也是要自己寫演算法去解決特殊情況下的問題,但是用起來比第二種方法舒服。

我先貼程式碼再做解釋。

public class EditTextUtils {
    /**
     *  設定edittext只能輸入小數點後兩位
     */
    public static void afterDotTwo(final EditText editText) {
        editText.addTextChangedListener(new TextWatcher() {
            @Override
            public void onTextChanged(CharSequence s, int start, int before, int count) {
                // 限制最多能輸入9位整數
                if (s.toString().contains(".")) {
                    if (s.toString().indexOf(".") > 9) {
                        s = s.toString().subSequence(0,9) + s.toString().substring(s.toString().indexOf("."));
                        editText.setText(s);
                        editText.setSelection(9);
                    }
                }else {
                    if (s.toString().length() > 9){
                        s = s.toString().subSequence(0,9);
                        editText.setText(s);
                        editText.setSelection(9);
                    }
                }
                // 判斷小數點後只能輸入兩位
                if (s.toString().contains(".")) {
                    if (s.length() - 1 - s.toString().indexOf(".") > 2) {
                        s = s.toString().subSequence(0,
                                s.toString().indexOf(".") + 3);
                        editText.setText(s);
                        editText.setSelection(s.length());
                    }
                }
                //如果第一個數字為0,第二個不為點,就不允許輸入
                if (s.toString().startsWith("0") && s.toString().trim().length() > 1) {
                    if (!s.toString().substring(1, 2).equals(".")) {
                        editText.setText(s.subSequence(0, 1));
                        editText.setSelection(1);
                        return;
                    }
                }
            }
            @Override
            public void beforeTextChanged(CharSequence s, int start, int count, int after) {

            }
            @Override
            public void afterTextChanged(Editable s) {
                if (editText.getText().toString().trim() != null && !editText.getText().toString().trim().equals("")) {
                    if (editText.getText().toString().trim().substring(0, 1).equals(".")) {
                        editText.setText("0" + editText.getText().toString().trim());
                        editText.setSelection(1);
                    }
                }
            }
        });
    }

}

我先把我工具類中的其它方法遮蔽調,只留這個方法。你會看到演算法的程式碼量不算稍微有點雜。這是我參考那哥們寫的再加上自己碰到的特殊需求改進的。當然每個人都應該根據自己的需求去寫不同的演算法。
但是至少先要掌握兩點:addTextChangedListener的瞭解和Java String字串的一些基本操作。


我這裡做了4中情況的判斷:

1. 限制整數只能輸入多位

我這邊的需求是整數最多為9位數,所以我先判斷是否有小數點,有的話就獲取前面9位加上小數點以及其後所有。

              if (s.toString().contains(".")) {
                    if (s.toString().indexOf(".") > 9) {
                        s = s.toString().subSequence(0,9) + s.toString().substring(s.toString().indexOf("."));
                        editText.setText(s);
                        editText.setSelection(9);
                    }
                }else {
                    if (s.toString().length() > 9){
                        s = s.toString().subSequence(0,9);
                        editText.setText(s);
                        editText.setSelection(9);
                    }
                }
2. 限制小數點後只能輸兩位

這個判斷是判斷金額的重點,本來金額小數點後就只能有兩位,當然有些人的需求可能是隻能有一位,這不要緊,把數字改下就行。首先判斷是否有小數點,如果有,判斷是否小數點後大於兩位,如果大於就只取前兩位。


                if (s.toString().contains(".")) {
                    if (s.length() - 1 - s.toString().indexOf(".") > 2) {
                        s = s.toString().subSequence(0,
                                s.toString().indexOf(".") + 3);
                        editText.setText(s);
                        editText.setSelection(s.length());
                    }
                }
3.第一個數為0的情況

我這裡寫的判斷不是很好,之後可以改進。這個特殊情況是這樣的,假如你輸入的第一位是0,你想想,你能輸入0233,05這種數字嗎,就是您能,這樣展示給使用者的效果也很不友好,所以要判斷如果第一位數是0,第二位不是小數點的話,就輸出0。

                if (s.toString().startsWith("0") && s.toString().trim().length() > 1) {
                    if (!s.toString().substring(1, 2).equals(".")) {
                        editText.setText(s.subSequence(0, 1));
                        editText.setSelection(1);
                        return;
                    }
                }

上面也說了這步的演算法有兩個問題,第一,如果在0.26的情況下,我把游標點在0後面,輸入一個不為0的數字假設5,得到的結果是0而不是5.26。第二,假如還是0.26的情況下,我把游標點在小數點後面,刪除小數點,得到的不是26而是0。所以這步的演算法有些問題。那為什麼不現在改呢,不好意思,最近太忙,寫文章都很趕。

4.第一個是小數點的情況

假如你的數是5.26,這時候你把游標放到5後面,刪除5,如果你不做處理,那展示出來的就是.26,這樣的展示就很不友好,所以我們要想辦法要在第一位為小數點的情況下補0。

            public void afterTextChanged(Editable s) {
                if (editText.getText().toString().trim() != null && !editText.getText().toString().trim().equals("")) {
                    if (editText.getText().toString().trim().substring(0, 1).equals(".")) {
                        editText.setText("0" + editText.getText().toString().trim());
                        editText.setSelection(1);
                    }
                }
            }

這裡寫在afterTextChanged裡面,所以讓大家先要了解addTextChangedListener再來看演算法,寫在afterTextChanged裡面是因為我要在“之後”做操作。演算法也很簡單,判斷有字串並且第一位為小數點的情況下,新增個0在前面。

因為這個演算法是很趕的情況下寫的,很多地方也許可以合併,可能也有一些特殊的情況沒有考慮到,最好不要直接拿來用,我只是舉個例子說明addTextChangedListener能很好的實現這個功能,具體要怎麼實現還需要根據個人自己的需求和特殊情況去寫演算法。

四.最方便的方法

沒錯,要實現這個功能的最方便方法就是……跪求谷歌在下一個版本新增一個MONEY型別的TYPE,這樣就是一行程式碼的事情了,呵呵。


相關文章