Десятичный разделитель запятая (',') с numberDecimal inputType в EditText


133

В inputType numberDecimalin EditTextиспользуется точка '.' как десятичный разделитель. В Европе вместо этого принято использовать запятую. Несмотря на то, что мой языковой стандарт установлен как немецкий, десятичным разделителем по-прежнему является '.'

Есть ли способ использовать запятую в качестве десятичного разделителя?



1
эта ошибка наконец-то исправлена ​​в Android O: issueetracker.google.com/issues/36907764
Ловис,

они говорят, что это исправлено, но я не могу подтвердить, что это исправлено? ты можешь?
Себастиан

5
Я могу подтвердить, что это НЕ исправлено, по крайней мере, на моем Nexus 4 под управлением Android 8.1 (также известный как LineageOS 15.1). Если в Настройках-> Язык установлен французский (Франция), EditText с android: inputType = "numberDecimal" предлагает разделитель ',' (запятая), но по-прежнему отказывается принимать запятую. Предлагаемый '.' (десятичная точка) допускается. С момента первого сообщения об этой ошибке прошло более 9 лет. Это какая-то запись? Ламе.
Пит

Ответы:


105

Обходной путь (пока Google не исправит эту ошибку) - использовать EditTextвместе с android:inputType="numberDecimal"и android:digits="0123456789.,".

Затем добавьте TextChangedListener в EditText со следующим значением afterTextChanged:

public void afterTextChanged(Editable s) {
    double doubleValue = 0;
    if (s != null) {
        try {
            doubleValue = Double.parseDouble(s.toString().replace(',', '.'));
        } catch (NumberFormatException e) {
            //Error
        }
    }
    //Do something with doubleValue
}

1
@Zoombie для отображения запятой (,) на клавиатуре зависит от языка, установленного на вашем устройстве. Если ваш ввод имеет тип numberDecimal и ваш язык - английский (США), он будет отображаться на устройствах Nexus (для справки). Возможно, устройства не Nexus не соблюдают это
hcpl 01

11
Он работает, но имейте в виду, что он пропускает текст типа «24,22,55». Возможно, вам потребуется добавить дополнительную проверку, чтобы исправить это!
димсуз

8
Это все еще путь?
Вилли Ментцель

Еще лучше использовать char localizedSeparator = DecimalFormatSymbols.getInstance (). GetDecimalSeparator (); localizedFloatString = localizedFloatString.replace ('.', localizedSeparator);
southerton

4
Похоже, это просто обмен одной ошибки на другую. Как реализовано выше, это будет работать для локалей, которые используют вместо. за счет обратного, более распространенного во всем мире. Твик @southerton действительно помогает в этом, однако ваши пользователи могут быть удивлены, когда они нажмут. и должны появиться на входе.
Ник

30

Вариант предлагаемых здесь «цифровых» решений:

char separator = DecimalFormatSymbols.getInstance().getDecimalSeparator();
input.setKeyListener(DigitsKeyListener.getInstance("0123456789" + separator));

С учетом разделителя локали.


Это наиболее четкий ответ на исходный вопрос. Спасибо
peter.bartos

Поместите это в onCreate (), это способ, ИМХО.
TechNyquist

6
Мне это нравится, но будьте осторожны ... есть клавиатуры, которые не заботятся о локали пользователя, поэтому пользователи, у которых нет ключа ,на своих клавиатурах. Примеры: клавиатура Samsung (KitKat).
Брайс Габен

2
Это позволит дублировать десятичные разделители. См. Ответ ниже, чтобы справиться с этим: stackoverflow.com/a/45384821/6138589
Эсдрас Лопес

19

Следующая маска валюты кода для EditText (123 125,155 долл. США)

Макет XML

  <EditText
    android:inputType="numberDecimal"
    android:layout_height="wrap_content"
    android:layout_width="200dp"
    android:digits="0123456789.,$" />

Код

EditText testFilter=...
testFilter.addTextChangedListener( new TextWatcher() {
        boolean isEdiging;
        @Override public void onTextChanged(CharSequence s, int start, int before, int count) { }
        @Override public void beforeTextChanged(CharSequence s, int start, int count, int after) { }

        @Override public void afterTextChanged(Editable s) {
            if(isEdiging) return;
            isEdiging = true;

            String str = s.toString().replaceAll( "[^\\d]", "" );
            double s1 = Double.parseDouble(str);

            NumberFormat nf2 = NumberFormat.getInstance(Locale.ENGLISH);
            ((DecimalFormat)nf2).applyPattern("$ ###,###.###");
            s.replace(0, s.length(), nf2.format(s1));

            isEdiging = false;
        }
    });

16

Это известная ошибка в Android SDK. Единственный обходной путь - создать собственную виртуальную клавиатуру. Вы можете найти пример реализации здесь .


15
Есть какие-нибудь новости через четыре года?
Антонио Сесто,

Испытали это и в Xamarin.Forms. Язык и региональные параметры: {se-SV}, а цифровая клавиатура показывает бота "," (десятичный разделитель) и "." (разделитель групп тысяч), но после нажатия "," в текстовое поле ничего не вводится и событие не возникает
joacar

Я могу подтвердить, что ошибка все еще существует.
Lensflare

исправлено в предварительной
версии для

6

Ответ Мартинса не сработает, если вы создаете экземпляр EditText программно. Я пошел дальше и изменил включенный DigitsKeyListenerкласс из API 14, чтобы использовать запятую и точку в качестве десятичного разделителя.

Для того, чтобы использовать эту функцию , вызов setKeyListener()на EditText, например ,

// Don't allow for signed input (minus), but allow for decimal points
editText.setKeyListener( new MyDigitsKeyListener( false, true ) );

Однако вам все равно придется использовать уловку Мартина в том, что TextChangedListenerвы заменяете запятые точками

import android.text.InputType;
import android.text.SpannableStringBuilder;
import android.text.Spanned;
import android.text.method.NumberKeyListener;
import android.view.KeyEvent;

class MyDigitsKeyListener extends NumberKeyListener {

    /**
     * The characters that are used.
     *
     * @see KeyEvent#getMatch
     * @see #getAcceptedChars
     */
    private static final char[][] CHARACTERS = new char[][] {
        new char[] { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9' },
        new char[] { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '-' },
        new char[] { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '.', ',' },
        new char[] { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '-', '.', ',' },
    };

    private char[] mAccepted;
    private boolean mSign;
    private boolean mDecimal;

    private static final int SIGN = 1;
    private static final int DECIMAL = 2;

    private static MyDigitsKeyListener[] sInstance = new MyDigitsKeyListener[4];

    @Override
    protected char[] getAcceptedChars() {
        return mAccepted;
    }

    /**
     * Allocates a DigitsKeyListener that accepts the digits 0 through 9.
     */
    public MyDigitsKeyListener() {
        this(false, false);
    }

    /**
     * Allocates a DigitsKeyListener that accepts the digits 0 through 9,
     * plus the minus sign (only at the beginning) and/or decimal point
     * (only one per field) if specified.
     */
    public MyDigitsKeyListener(boolean sign, boolean decimal) {
        mSign = sign;
        mDecimal = decimal;

        int kind = (sign ? SIGN : 0) | (decimal ? DECIMAL : 0);
        mAccepted = CHARACTERS[kind];
    }

    /**
     * Returns a DigitsKeyListener that accepts the digits 0 through 9.
     */
    public static MyDigitsKeyListener getInstance() {
        return getInstance(false, false);
    }

    /**
     * Returns a DigitsKeyListener that accepts the digits 0 through 9,
     * plus the minus sign (only at the beginning) and/or decimal point
     * (only one per field) if specified.
     */
    public static MyDigitsKeyListener getInstance(boolean sign, boolean decimal) {
        int kind = (sign ? SIGN : 0) | (decimal ? DECIMAL : 0);

        if (sInstance[kind] != null)
            return sInstance[kind];

        sInstance[kind] = new MyDigitsKeyListener(sign, decimal);
        return sInstance[kind];
    }

    /**
     * Returns a DigitsKeyListener that accepts only the characters
     * that appear in the specified String.  Note that not all characters
     * may be available on every keyboard.
     */
    public static MyDigitsKeyListener getInstance(String accepted) {
        // TODO: do we need a cache of these to avoid allocating?

        MyDigitsKeyListener dim = new MyDigitsKeyListener();

        dim.mAccepted = new char[accepted.length()];
        accepted.getChars(0, accepted.length(), dim.mAccepted, 0);

        return dim;
    }

    public int getInputType() {
        int contentType = InputType.TYPE_CLASS_NUMBER;
        if (mSign) {
            contentType |= InputType.TYPE_NUMBER_FLAG_SIGNED;
        }
        if (mDecimal) {
            contentType |= InputType.TYPE_NUMBER_FLAG_DECIMAL;
        }
        return contentType;
    }

    @Override
    public CharSequence filter(CharSequence source, int start, int end,
                               Spanned dest, int dstart, int dend) {
        CharSequence out = super.filter(source, start, end, dest, dstart, dend);

        if (mSign == false && mDecimal == false) {
            return out;
        }

        if (out != null) {
            source = out;
            start = 0;
            end = out.length();
        }

        int sign = -1;
        int decimal = -1;
        int dlen = dest.length();

        /*
         * Find out if the existing text has '-' or '.' characters.
         */

        for (int i = 0; i < dstart; i++) {
            char c = dest.charAt(i);

            if (c == '-') {
                sign = i;
            } else if (c == '.' || c == ',') {
                decimal = i;
            }
        }
        for (int i = dend; i < dlen; i++) {
            char c = dest.charAt(i);

            if (c == '-') {
                return "";    // Nothing can be inserted in front of a '-'.
            } else if (c == '.' ||  c == ',') {
                decimal = i;
            }
        }

        /*
         * If it does, we must strip them out from the source.
         * In addition, '-' must be the very first character,
         * and nothing can be inserted before an existing '-'.
         * Go in reverse order so the offsets are stable.
         */

        SpannableStringBuilder stripped = null;

        for (int i = end - 1; i >= start; i--) {
            char c = source.charAt(i);
            boolean strip = false;

            if (c == '-') {
                if (i != start || dstart != 0) {
                    strip = true;
                } else if (sign >= 0) {
                    strip = true;
                } else {
                    sign = i;
                }
            } else if (c == '.' || c == ',') {
                if (decimal >= 0) {
                    strip = true;
                } else {
                    decimal = i;
                }
            }

            if (strip) {
                if (end == start + 1) {
                    return "";  // Only one character, and it was stripped.
                }

                if (stripped == null) {
                    stripped = new SpannableStringBuilder(source, start, end);
                }

                stripped.delete(i - start, i + 1 - start);
            }
        }

        if (stripped != null) {
            return stripped;
        } else if (out != null) {
            return out;
        } else {
            return null;
        }
    }
}

из документа: KeyListener следует использовать только в тех случаях, когда приложение имеет собственную экранную клавиатуру и также хочет обрабатывать жесткие события клавиатуры, чтобы соответствовать ей. developer.android.com/reference/android/text/method/…
Loda

6

вы можете использовать следующее для разных языков

private void localeDecimalInput(final EditText editText){

    DecimalFormat decFormat = (DecimalFormat) DecimalFormat.getInstance(Locale.getDefault());
    DecimalFormatSymbols symbols=decFormat.getDecimalFormatSymbols();
    final String defaultSeperator=Character.toString(symbols.getDecimalSeparator());

    editText.addTextChangedListener(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) {

        }

        @Override
        public void afterTextChanged(Editable editable) {
            if(editable.toString().contains(defaultSeperator))
                editText.setKeyListener(DigitsKeyListener.getInstance("0123456789"));
            else
                editText.setKeyListener(DigitsKeyListener.getInstance("0123456789" + defaultSeperator));
        }
    });
}

Это лучшее решение для меня, но есть проблема с некоторыми телефонами, например Samsung, которые не показывают кома «,» на клавиатуре. Поэтому я изменил это, чтобы разрешить и запятую, и точку, но затем заменил в соответствии с
локалью

5

Вы можете использовать следующий обходной путь, чтобы также включить запятую в качестве допустимого ввода: -

Через XML:

<EditText
    android:inputType="number"
    android:digits="0123456789.," />

Программный:

EditText input = new EditText(THE_CONTEXT);
input.setKeyListener(DigitsKeyListener.getInstance("0123456789.,"));

Таким образом, система Android покажет клавиатуру цифр и разрешит ввод запятой. Надеюсь, это ответит на вопрос :)


С этим решением, когда вы нажимаете ",", но текст редактирования отображается "."
Мара Хименес,

2

Для решений Mono (Droid):

decimal decimalValue = decimal.Parse(input.Text.Replace(",", ".") , CultureInfo.InvariantCulture);

1

Вы можете сделать следующее:

DecimalFormatSymbols d = DecimalFormatSymbols.getInstance(Locale.getDefault());
input.setFilters(new InputFilter[] { new DecimalDigitsInputFilter(5, 2) });
input.setKeyListener(DigitsKeyListener.getInstance("0123456789" + d.getDecimalSeparator()));

А затем вы можете использовать входной фильтр:

    public class DecimalDigitsInputFilter implements InputFilter {

Pattern mPattern;

public DecimalDigitsInputFilter(int digitsBeforeZero, int digitsAfterZero) {
    DecimalFormatSymbols d = new DecimalFormatSymbols(Locale.getDefault());
    String s = "\\" + d.getDecimalSeparator();
    mPattern = Pattern.compile("[0-9]{0," + (digitsBeforeZero - 1) + "}+((" + s + "[0-9]{0," + (digitsAfterZero - 1) + "})?)||(" + s + ")?");
}

@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;
}

}


между тысячей и сотней может быть пробел, этот шаблон откажется от форматированного ввода
Эрик Чжао

1

IMHO лучший подход к этой проблеме - просто использовать InputFilter. Хорошая суть здесь DecimalDigitsInputFilter . Тогда вы можете просто:

editText.setInputType(TYPE_NUMBER_FLAG_DECIMAL | TYPE_NUMBER_FLAG_SIGNED | TYPE_CLASS_NUMBER)
editText.setKeyListener(DigitsKeyListener.getInstance("0123456789,.-"))
editText.setFilters(new InputFilter[] {new DecimalDigitsInputFilter(5,2)});

Это сработало как шарм, спасибо! (после стольких неверных решений выше ... :() Но у меня есть вопрос: как я могу добиться, чтобы на экране отображалась запятая (","), а не точка ("."), потому что в Венгрии мы используем запятую в качестве десятичного разделителя .
Абигейл La'Fay

1
android: digits = "0123456789" можно добавить в EditText. Более того, вместо того, чтобы возвращать null в DecimalDigitsInputFilter, вы можете вернуть source.replace (".", ",") В соответствии с ответом stackoverflow.com/a/40020731/1510222, нет способа скрыть точку на стандартной клавиатуре
Аркадиуш Цеслински

1

чтобы локализовать использование ввода:

char sep = DecimalFormatSymbols.getInstance().getDecimalSeparator();

а затем добавьте:

textEdit.setKeyListener(DigitsKeyListener.getInstance("0123456789" + sep));

чем не забудьте заменить "," на "." поэтому Float или Double могут анализировать его без ошибок.


1
Это решение позволяет вводить несколько запятых
Leo Droidcoder

1

Во всех остальных сообщениях здесь были серьезные дыры, поэтому вот решение, которое:

  • Использовать запятые или точки в зависимости от региона, вы не сможете вводить противоположный.
  • Если EditText начинается с некоторого значения, он при необходимости заменяет правильный разделитель.

В XML:

<EditText
    ...
    android:inputType="numberDecimal" 
    ... />

Переменная класса:

private boolean isDecimalSeparatorComma = false;

В onCreate найдите разделитель, используемый в текущей локали:

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
    NumberFormat nf = NumberFormat.getInstance();
    if (nf instanceof DecimalFormat) {
        DecimalFormatSymbols sym = ((DecimalFormat) nf).getDecimalFormatSymbols();
        char decSeparator = sym.getDecimalSeparator();
        isDecimalSeparatorComma = Character.toString(decSeparator).equals(",");
    }
}

Также onCreate, используйте это, чтобы обновить его, если вы загружаете текущее значение:

// Replace editText with commas or periods as needed for viewing
String editTextValue = getEditTextValue(); // load your current value
if (editTextValue.contains(".") && isDecimalSeparatorComma) {
    editTextValue = editTextValue.replaceAll("\\.",",");
} else if (editTextValue.contains(",") && !isDecimalSeparatorComma) {
    editTextValue = editTextValue.replaceAll(",",".");
}
setEditTextValue(editTextValue); // override your current value

Также onCreate, добавьте слушателей

editText.addTextChangedListener(editTextWatcher);

if (isDecimalSeparatorComma) {
    editText.setKeyListener(DigitsKeyListener.getInstance("0123456789,"));
} else {
    editText.setKeyListener(DigitsKeyListener.getInstance("0123456789."));
}

editTextWatcher

TextWatcher editTextWatcher = 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) { }

    @Override
    public void afterTextChanged(Editable s) {
        String editTextValue = s.toString();

        // Count up the number of commas and periods
        Pattern pattern = Pattern.compile("[,.]");
        Matcher matcher = pattern.matcher(editTextValue);
        int count = 0;
        while (matcher.find()) {
            count++;
        }

        // Don't let it put more than one comma or period
        if (count > 1) {
            s.delete(s.length()-1, s.length());
        } else {
            // If there is a comma or period at the end the value hasn't changed so don't update
            if (!editTextValue.endsWith(",") && !editTextValue.endsWith(".")) {
                doSomething()
            }
        }
    }
};

Пример doSomething (), преобразование в стандартный период для обработки данных

private void doSomething() {
    try {
        String editTextStr = editText.getText().toString();
        if (isDecimalSeparatorComma) {
            editTextStr = editTextStr.replaceAll(",",".");
        }
        float editTextFloatValue = editTextStr.isEmpty() ?
                0.0f :
                Float.valueOf(editTextStr);

        ... use editTextFloatValue
    } catch (NumberFormatException e) {
        Log.e(TAG, "Error converting String to Double");
    }
}

0

Android имеет встроенный модуль форматирования чисел.

Вы можете добавить это в свой, EditTextчтобы разрешить десятичные дроби и запятые: android:inputType="numberDecimal"иandroid:digits="0123456789.,"

Затем где-нибудь в вашем коде, когда пользователь нажимает кнопку «Сохранить» или после ввода текста (используйте прослушиватель).

// Format the number to the appropriate double
try { 
    Number formatted = NumberFormat.getInstance().parse(editText.getText().toString());
    cost = formatted.doubleValue();
} catch (ParseException e) {
    System.out.println("Error parsing cost string " + editText.getText().toString());
    cost = 0.0;
}

0

Я решил заменить запятую на точку только при редактировании. Вот мой хитрый и относительно простой обходной путь:

    editText.setOnFocusChangeListener(new View.OnFocusChangeListener() {
        @Override
        public void onFocusChange(View v, boolean hasFocus) {
            EditText editText = (EditText) v; 
            String text = editText.getText().toString();
            if (hasFocus) {
                editText.setText(text.replace(",", "."));
            } else {
                if (!text.isEmpty()) {
                    Double doubleValue = Double.valueOf(text.replace(",", "."));
                    editText.setText(someDecimalFormatter.format(doubleValue));
                }
            }
        }
    });

someDecimalFormatter будет использовать запятую или точку в зависимости от локали


0

Я не знаю, почему ваши ответы такие сложные. Если в SDK есть ошибка, вы должны ее исправить или обойти.

Я выбрал второй способ решения этой проблемы. Если вы отформатируете свою строку как, Locale.ENGLISHа затем поместите ее в EditText(даже как пустую строку). Пример:

String.format(Locale.ENGLISH,"%.6f", yourFloatNumber);

Преследуя это решение, ваш результат совместим с показанной клавиатурой. Тогда числа с плавающей запятой и двойные числа работают типично для языков программирования с точкой вместо запятой.


0

Мое решение:

  • По основному виду деятельности:

    char separator =DecimalFormatSymbols.getInstance().getDecimalSeparator(); textViewPitchDeadZone.setKeyListener(DigitsKeyListener.getInstance("0123456789" + separator));

  • В xml файле: android:imeOptions="flagNoFullscreen" android:inputType="numberDecimal"

и я взял двойное значение в editText как String.


0

Я могу подтвердить, что предложенные исправления не работают с IME Samsung (по крайней мере, на S6 и S9) и, возможно, LG. Они по-прежнему показывают точку в качестве десятичного разделителя независимо от языкового стандарта. Переход на IME Google исправляет это, но вряд ли вариант для большинства разработчиков.

Это также не было исправлено в Oreo для этих клавиатур, поскольку это исправление, которое Samsung и / или LG должны сделать, а затем подтолкнуть даже к своим древним телефонам.

Вместо этого я раздвоил проект цифровой клавиатуры и добавил режим, в котором он ведет себя как вилка IME : . Подробнее см. В примере проекта. Это сработало для меня очень хорошо и похоже на многие поддельные IME для ввода PIN-кода, которые вы видите в банковских приложениях.

Образец снимка экрана приложения


0

Прошло более 8 лет, и я удивлен, эта проблема еще не устранена ...
Я боролся с этой простой проблемой, так как ответ @Martin, получивший наибольшее количество голосов, позволяет вводить несколько разделителей, т.е. пользователь может ввести «12 ,,, ,,, 12,1,, 21,2, "
Кроме того, вторая проблема заключается в том, что на некоторых устройствах запятая не отображается на цифровой клавиатуре (или требует многократного нажатия кнопки с точкой)

Вот мое обходное решение, которое решает упомянутые проблемы и позволяет пользователю вводить "." и ',', но в EditText он увидит единственный десятичный разделитель, соответствующий текущей локали:

editText.apply { addTextChangedListener(DoubleTextChangedListener(this)) }

И наблюдатель текста:

  open class DoubleTextChangedListener(private val et: EditText) : TextWatcher {

    init {
        et.inputType = InputType.TYPE_CLASS_NUMBER or InputType.TYPE_NUMBER_FLAG_DECIMAL
        et.keyListener = DigitsKeyListener.getInstance("0123456789.,")
    }

    private val separator = DecimalFormatSymbols.getInstance().decimalSeparator

    override fun beforeTextChanged(s: CharSequence?, start: Int, count: Int, after: Int) {
        //empty
    }

    @CallSuper
    override fun onTextChanged(s: CharSequence, start: Int, before: Int, count: Int) {
        et.run {
            removeTextChangedListener(this@DoubleTextChangedListener)
            val formatted = toLocalizedDecimal(s.toString(), separator)
            setText(formatted)
            setSelection(formatted.length)
            addTextChangedListener(this@DoubleTextChangedListener)
        }
    }

    override fun afterTextChanged(s: Editable?) {
        // empty
    }

    /**
     * Formats input to a decimal. Leaves the only separator (or none), which matches [separator].
     * Examples:
     * 1. [s]="12.12", [separator]=',' -> result= "12,12"
     * 2. [s]="12.12", [separator]='.' -> result= "12.12"
     * 4. [s]="12,12", [separator]='.' -> result= "12.12"
     * 5. [s]="12,12,,..,,,,,34..,", [separator]=',' -> result= "12,1234"
     * 6. [s]="12.12,,..,,,,,34..,", [separator]='.' -> result= "12.1234"
     * 7. [s]="5" -> result= "5"
     */
    private fun toLocalizedDecimal(s: String, separator: Char): String {
        val cleared = s.replace(",", ".")
        val splitted = cleared.split('.').filter { it.isNotBlank() }
        return when (splitted.size) {
            0 -> s
            1 -> cleared.replace('.', separator).replaceAfter(separator, "")
            2 -> splitted.joinToString(separator.toString())
            else -> splitted[0]
                    .plus(separator)
                    .plus(splitted.subList(1, splitted.size - 1).joinToString(""))
        }
    }
}

0

Простое решение, сделайте настраиваемый элемент управления. (это сделано в Xamarin android, но должно легко переноситься на java)

public class EditTextDecimalNumber:EditText
{
    readonly string _numberFormatDecimalSeparator;

    public EditTextDecimalNumber(Context context, IAttributeSet attrs) : base(context, attrs)
    {
        InputType = InputTypes.NumberFlagDecimal;
        TextChanged += EditTextDecimalNumber_TextChanged;
        _numberFormatDecimalSeparator = System.Threading.Thread.CurrentThread.CurrentUICulture.NumberFormat.NumberDecimalSeparator;

        KeyListener = DigitsKeyListener.GetInstance($"0123456789{_numberFormatDecimalSeparator}");
    }

    private void EditTextDecimalNumber_TextChanged(object sender, TextChangedEventArgs e)
    {
        int noOfOccurence = this.Text.Count(x => x.ToString() == _numberFormatDecimalSeparator);
        if (noOfOccurence >=2)
        {
            int lastIndexOf = this.Text.LastIndexOf(_numberFormatDecimalSeparator,StringComparison.CurrentCulture);
            if (lastIndexOf!=-1)
            {
                this.Text = this.Text.Substring(0, lastIndexOf);
                this.SetSelection(this.Text.Length);
            }

        }
    }
}

0

Вы можете использовать inputType="phone", однако в этом случае вам придется иметь дело с несколькими ,или .присутствующими, поэтому потребуется дополнительная проверка.


-1

Я думаю, что это решение менее сложное, чем другие, написанные здесь:

<EditText
    android:inputType="numberDecimal"
    android:digits="0123456789," />

Таким образом, когда вы нажмете "." в мягкой клавиатуре ничего не происходит; разрешены только цифры и запятая.


4
если вы сделаете это, вы просто сломаете все локали, которые используют '.' вместо.
Ник
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.