Какой лучший способ ограничить длину текста EditText в Android

Как лучше всего ограничить длину текста EditText в Android?

Есть ли способ сделать это через xml?


person hpique    schedule 19.07.2010    source источник
comment
Я хотел установить максимальное количество символов для моего EditText. Сначала было неочевидно, что ограничение на длину текста - одно и то же. (Просто на заметку для еще одного сбивающего с толку путешественника).   -  person Katedral Pillon    schedule 14.10.2014
comment
Правильный ответ здесь: stackoverflow.com/a/19222238/276949. Этот ответ ограничивает длину И предотвращает постоянное заполнение буфера после достижения предела, что позволяет вашей клавише возврата работать должным образом.   -  person Martin Konecny    schedule 11.11.2015


Ответы (21)


Документация

Пример

android:maxLength="10"
person Austin Hanson    schedule 19.07.2010
comment
Ооо! То же самое случилось со мной. Я смотрел код, а метода setMaxLength нет. - person hpique; 20.07.2010
comment
что здесь делает проверка? Он просто ссылается на эту страницу. - person what is sleep; 24.10.2013
comment
Я думаю, что кто-то подумал, что было бы разумно иметь ссылку на эту страницу. Когда я изначально писал ответ, он ссылался на страницу разработчиков Android, но с тех пор были внесены изменения. Я просмотрю историю и запрошу изменение, чтобы связать его обратно. - person Austin Hanson; 25.10.2013
comment
@Vincy, вы ошибаетесь. Свойство maxLength по-прежнему работает. - person ashishduh; 04.03.2015
comment
Имейте в виду, что android:maxLength эквивалентно InputFilter.LengthFilter, поэтому при программном изменении его фильтра вы также изменили его фильтр XML. - person mr5; 01.08.2015
comment
Людям, которые говорят, что это не работает, имейте в виду, что вызов setFilters остановит android:maxLength работу, потому что он перезаписывает фильтр, установленный XML. Другими словами, если вы устанавливаете какие-либо фильтры программно, вы должны установить их все программно. - person Ian Newson; 19.05.2016
comment
@hpique 9 лет спустя, +1 D'oh! xD - person varun; 09.05.2019

используйте входной фильтр, чтобы ограничить максимальную длину текстового представления.

TextView editEntryView = new TextView(...);
InputFilter[] filterArray = new InputFilter[1];
filterArray[0] = new InputFilter.LengthFilter(8);
editEntryView.setFilters(filterArray);
person jerry    schedule 10.11.2010
comment
Это очень полезно, если кто-то уже сделал InputFilter. Он переопределяет android:maxlength в файле xml, поэтому нам нужно добавить LengthFilter таким образом. - person Seblis; 12.09.2013
comment
Думаю, ваш ответ - лучший ответ, так как он более динамичный, +1 - person Muhannad A.Alhariri; 16.06.2014

Примечание для людей, которые уже используют настраиваемый фильтр ввода и также хотят ограничить максимальную длину:

Когда вы назначаете входные фильтры в коде, все ранее установленные входные фильтры очищаются, включая один, установленный с android:maxLength. Я обнаружил это при попытке использовать настраиваемый фильтр ввода, чтобы предотвратить использование некоторых символов, которые мы не допускаем в поле пароля. После установки этого фильтра с помощью setFilters maxLength больше не соблюдался. Решением было программно установить maxLength и мой настраиваемый фильтр вместе. Что-то вроде этого:

myEditText.setFilters(new InputFilter[] {
        new PasswordCharFilter(), new InputFilter.LengthFilter(20)
});
person goto10    schedule 23.05.2012
comment
Вы можете сначала получить существующие фильтры как InputFilter [] existingFilters = editText.getFilters (); Затем добавьте свой фильтр вместе с существующими фильтрами. - person subair_a; 29.07.2016

У меня была эта проблема, и я считаю, что нам не хватает хорошо объясненного способа сделать это программно без потери уже установленных фильтров.

Установка длины в XML:

Как правильно указано в принятом ответе, если вы хотите определить фиксированную длину EditText, которую вы больше не будете изменять в будущем, просто определите в своем EditText XML:

android:maxLength="10"     

Программная установка длины

Чтобы установить длину программным способом, вам необходимо установить ее с помощью InputFilter. Но если вы создадите новый InputFilter и установите для него EditText, вы потеряете все другие уже определенные фильтры (например, maxLines, inputType и т. Д.), Которые вы могли добавить либо через XML, либо программно.

Так что это НЕПРАВИЛЬНО:

editText.setFilters(new InputFilter[] {new InputFilter.LengthFilter(maxLength)});

Чтобы избежать потери ранее добавленных фильтров, вам необходимо получить эти фильтры, добавить новый (в данном случае maxLength) и снова установить фильтры на EditText следующим образом:

Java

InputFilter[] editFilters = editText.getFilters();
InputFilter[] newFilters = new InputFilter[editFilters.length + 1];
System.arraycopy(editFilters, 0, newFilters, 0, editFilters.length);
newFilters[editFilters.length] = new InputFilter.LengthFilter(maxLength); 
editText.setFilters(newFilters);

Kotlin, однако, упростил задачу для всех, вам также необходимо добавить фильтр к уже существующим, но вы можете добиться этого с помощью простого:

editText.filters += InputFilter.LengthFilter(maxLength)
person Ricardo    schedule 27.09.2017
comment
+1. Не знал о фильтрах. Ото, после сравнения кода Java и Kotlin мне стало интересно, почему люди продолжают настаивать на Java для разработки под Android xd. - person Xam; 21.01.2021

Для всех, кто интересуется, как этого добиться, вот мой расширенный EditText класс EditTextNumeric.

.setMaxLength(int) - устанавливает максимальное количество цифр

.setMaxValue(int) - ограничение максимального целочисленного значения

.setMin(int) - ограничение минимального целочисленного значения

.getValue() - получить целочисленное значение

import android.content.Context;
import android.text.InputFilter;
import android.text.InputType;
import android.widget.EditText;

public class EditTextNumeric extends EditText {
    protected int max_value = Integer.MAX_VALUE;
    protected int min_value = Integer.MIN_VALUE;

    // constructor
    public EditTextNumeric(Context context) {
        super(context);
        this.setInputType(InputType.TYPE_CLASS_NUMBER);
    }

    // checks whether the limits are set and corrects them if not within limits
    @Override
    protected void onTextChanged(CharSequence text, int start, int before, int after) {
        if (max_value != Integer.MAX_VALUE) {
            try {
                if (Integer.parseInt(this.getText().toString()) > max_value) {
                    // change value and keep cursor position
                    int selection = this.getSelectionStart();
                    this.setText(String.valueOf(max_value));
                    if (selection >= this.getText().toString().length()) {
                        selection = this.getText().toString().length();
                    }
                    this.setSelection(selection);
                }
            } catch (NumberFormatException exception) {
                super.onTextChanged(text, start, before, after);
            }
        }
        if (min_value != Integer.MIN_VALUE) {
            try {
                if (Integer.parseInt(this.getText().toString()) < min_value) {
                    // change value and keep cursor position
                    int selection = this.getSelectionStart();
                    this.setText(String.valueOf(min_value));
                    if (selection >= this.getText().toString().length()) {
                        selection = this.getText().toString().length();
                    }
                    this.setSelection(selection);
                }
            } catch (NumberFormatException exception) {
                super.onTextChanged(text, start, before, after);
            }
        }
        super.onTextChanged(text, start, before, after);
    }

    // set the max number of digits the user can enter
    public void setMaxLength(int length) {
        InputFilter[] FilterArray = new InputFilter[1];
        FilterArray[0] = new InputFilter.LengthFilter(length);
        this.setFilters(FilterArray);
    }

    // set the maximum integer value the user can enter.
    // if exeeded, input value will become equal to the set limit
    public void setMaxValue(int value) {
        max_value = value;
    }
    // set the minimum integer value the user can enter.
    // if entered value is inferior, input value will become equal to the set limit
    public void setMinValue(int value) {
        min_value = value;
    }

    // returns integer value or 0 if errorous value
    public int getValue() {
        try {
            return Integer.parseInt(this.getText().toString());
        } catch (NumberFormatException exception) {
            return 0;
        }
    }
}

Пример использования:

final EditTextNumeric input = new EditTextNumeric(this);
input.setMaxLength(5);
input.setMaxValue(total_pages);
input.setMinValue(1);

Все остальные методы и атрибуты, применимые к EditText, конечно, тоже работают.

person Martynas Januškauskas    schedule 25.01.2012
comment
Было бы лучше, если бы мы добавили это в xml Layout. Я получаю сообщение об ошибке при использовании xml. Вызвано: android.view.InflateException: строка двоичного файла XML № 47: ошибка надувания класса com.passenger.ucabs.utils.EditTextNumeric - person Shihab Uddin; 19.01.2017
comment
@Martynas Получил ту же ошибку, что и Shihab_ возвращает какое-нибудь решение? - person Pranaysharma; 06.03.2017

XML

android:maxLength="10"

Java:

InputFilter[] editFilters = editText.getFilters();
InputFilter[] newFilters = new InputFilter[editFilters.length + 1];
System.arraycopy(editFilters, 0, newFilters, 0, editFilters.length);
newFilters[editFilters.length] = new InputFilter.LengthFilter(maxLength);
editText.setFilters(newFilters);

Котлин:

editText.filters += InputFilter.LengthFilter(maxLength)
person João Carlos    schedule 24.08.2018

Из-за наблюдения goto10 я собрал следующий код для защиты от потери других фильтров с установкой максимальной длины:

/**
 * This sets the maximum length in characters of an EditText view. Since the
 * max length must be done with a filter, this method gets the current
 * filters. If there is already a length filter in the view, it will replace
 * it, otherwise, it will add the max length filter preserving the other
 * 
 * @param view
 * @param length
 */
public static void setMaxLength(EditText view, int length) {
    InputFilter curFilters[];
    InputFilter.LengthFilter lengthFilter;
    int idx;

    lengthFilter = new InputFilter.LengthFilter(length);

    curFilters = view.getFilters();
    if (curFilters != null) {
        for (idx = 0; idx < curFilters.length; idx++) {
            if (curFilters[idx] instanceof InputFilter.LengthFilter) {
                curFilters[idx] = lengthFilter;
                return;
            }
        }

        // since the length filter was not part of the list, but
        // there are filters, then add the length filter
        InputFilter newFilters[] = new InputFilter[curFilters.length + 1];
        System.arraycopy(curFilters, 0, newFilters, 0, curFilters.length);
        newFilters[curFilters.length] = lengthFilter;
        view.setFilters(newFilters);
    } else {
        view.setFilters(new InputFilter[] { lengthFilter });
    }
}
person Tim Gallagher    schedule 26.05.2012
comment
Возможно, вы захотите немного обновить код. Выделенный массив должен иметь размер curFilters.length + 1, и после создания newFilters вы не устанавливали его для вновь выделенного массива. InputFilter newFilters [] = новый InputFilter [curFilters.length + 1]; System.arraycopy (curFilters, 0, newFilters, 0, curFilters.length); this.setFilters (newFilters); - person JavaCoderEx; 17.07.2012

Другой способ добиться этого - добавить в XML-файл следующее определение:

<EditText
    android:id="@+id/input"
    android:layout_width="0dp"
    android:layout_height="wrap_content"
    android:inputType="number"
    android:maxLength="6"
    android:hint="@string/hint_gov"
    android:layout_weight="1"/>

Это ограничит максимальную длину виджета EditText до 6 символов.

person user2586194    schedule 16.07.2013

XML

android:maxLength="10"

Программно:

int maxLength = 10;
InputFilter[] filters = new InputFilter[1];
filters[0] = new InputFilter.LengthFilter(maxLength);
yourEditText.setFilters(filters);

Примечание: внутри EditText и TextView анализируют значение android:maxLength в XML и используют InputFilter.LengthFilter() для его применения.

См .: TextView.java # L1564

person nhoxbypass    schedule 18.05.2019

Из material.io вы можете использовать TextInputEditText в сочетании с TextInputLayout:

<com.google.android.material.textfield.TextInputLayout
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    app:counterEnabled="true"
    app:counterMaxLength="1000"
    app:passwordToggleEnabled="false">

    <com.google.android.material.textfield.TextInputEditText
        android:id="@+id/edit_text"
        android:hint="@string/description"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:maxLength="1000"
        android:gravity="top|start"
        android:inputType="textMultiLine|textNoSuggestions"/>

</com.google.android.material.textfield.TextInputLayout>

Вы можете настроить пароль EditText с возможностью рисования:

пример пароля

Или вы можете ограничить длину текста с помощью счетчика / без счетчика:

counter example

Зависимость:

implementation 'com.google.android.material:material:1.1.0-alpha02'
person RonTLV    schedule 02.01.2019

это простой способ в xml:

android:maxLength="@{length}"

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

public static void setMaxLengthOfEditText(EditText editText, int length) {
    InputFilter[] filters = editText.getFilters();
    List arrayList = new ArrayList();
    int i2 = 0;
    if (filters != null && filters.length > 0) {
        int filtersSize = filters.length;
        int i3 = 0;
        while (i2 < filtersSize) {
            Object obj = filters[i2];
            if (obj instanceof LengthFilter) {
                arrayList.add(new LengthFilter(length));
                i3 = 1;
            } else {
                arrayList.add(obj);
            }
            i2++;
        }
        i2 = i3;
    }
    if (i2 == 0) {
        arrayList.add(new LengthFilter(length));
    }
    if (!arrayList.isEmpty()) {
        editText.setFilters((InputFilter[]) arrayList.toArray(new InputFilter[arrayList.size()]));
    }
}
person Mohamed Fadel Buffon    schedule 13.03.2020

Это настраиваемый класс EditText, который позволяет фильтру длины жить вместе с другими фильтрами. Благодаря ответу Тима Галлахера (ниже)

import android.content.Context;
import android.text.InputFilter;
import android.util.AttributeSet;
import android.widget.EditText;


public class EditTextMultiFiltering extends EditText{

    public EditTextMultiFiltering(Context context) {
        super(context);
    }

    public EditTextMultiFiltering(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public EditTextMultiFiltering(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }

    public void setMaxLength(int length) {
        InputFilter curFilters[];
        InputFilter.LengthFilter lengthFilter;
        int idx;

        lengthFilter = new InputFilter.LengthFilter(length);

        curFilters = this.getFilters();
        if (curFilters != null) {
            for (idx = 0; idx < curFilters.length; idx++) {
                if (curFilters[idx] instanceof InputFilter.LengthFilter) {
                    curFilters[idx] = lengthFilter;
                    return;
                }
            }

            // since the length filter was not part of the list, but
            // there are filters, then add the length filter
            InputFilter newFilters[] = new InputFilter[curFilters.length + 1];
            System.arraycopy(curFilters, 0, newFilters, 0, curFilters.length);
            newFilters[curFilters.length] = lengthFilter;
            this.setFilters(newFilters);
        } else {
            this.setFilters(new InputFilter[] { lengthFilter });
        }
    }
}
person Gal Rom    schedule 18.08.2015

это простой способ в xml:

android:maxLength="4"

если вам нужно установить 4 символа в текстовом редакторе xml, используйте это

<EditText
    android:id="@+id/edtUserCode"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:maxLength="4"
    android:hint="Enter user code" />
person Navadip Patel    schedule 05.09.2017

Это отлично работает ...

android:maxLength="10"

это будет принимать только 10 символов.

person Sanjay Kumaar    schedule 05.09.2017

Попробуйте это программно для Java:

myEditText(new InputFilter[] {new InputFilter.LengthFilter(CUSTOM_MAX_LEN)});
person reza_khalafi    schedule 20.07.2019
comment
Вы забыли вызвать метод, поэтому он должен выглядеть так: myEditText.setFilters(new InputFilter[] {new InputFilter.LengthFilter(CUSTOM_MAX_LEN)}); - person StevenTB; 13.09.2019

Kotlin однострочный

etxt_userinput.filters = arrayOf<InputFilter>(InputFilter.LengthFilter(100))

где 100 - максимальная длина

person Gastón Saillén    schedule 14.07.2020
comment
Это ужасное решение, потому что с ним вы потеряете все ранее установленные фильтры. - person Ricardo; 10.08.2020
comment
OP спросил, как добавить один фильтр, который является максимальной длиной редактируемого текста, ответ не означает наличие нескольких фильтров. - person Gastón Saillén; 10.08.2020
comment
Он спросил, как добавить фильтр maxLength, а не maxLength как отдельный фильтр. Очевидно, что ваше решение работает для единственного случая настройки только этого фильтра, что в конечном итоге бесполезно, если вы в основном уничтожаете все остальные фильтры, которые может иметь текст редактирования. - person Ricardo; 11.08.2020
comment
Точно, как вы сказали, это могло быть, но в этом случае требовалось добавить фильтр, а не оптимизировать его для нескольких фильтров. - person Gastón Saillén; 11.08.2020
comment
Итак, вы хотите сказать: есть отличное решение, но я не дам его вам, потому что ваш вопрос слишком конкретен? - person Ricardo; 12.08.2020
comment
Лол никогда не говорил, что у меня есть отличное решение, но не откажусь от него - person Gastón Saillén; 03.05.2021

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

1, ограничить длину.
2, если ввести больше, дайте обратный вызов, чтобы вызвать тост.
3, курсор может быть в середине или в конце.
4, пользователь может ввести, вставив строку.
5, Всегда отбрасывайте ввод переполнения и сохраняйте происхождение.

public class LimitTextWatcher implements TextWatcher {

    public interface IF_callback{
        void callback(int left);
    }

    public IF_callback if_callback;

    EditText editText;
    int maxLength;

    int cursorPositionLast;
    String textLast;
    boolean bypass;

    public LimitTextWatcher(EditText editText, int maxLength, IF_callback if_callback) {

        this.editText = editText;
        this.maxLength = maxLength;
        this.if_callback = if_callback;
    }

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

        if (bypass) {

            bypass = false;

        } else {

            StringBuilder stringBuilder = new StringBuilder();
            stringBuilder.append(s);
            textLast = stringBuilder.toString();

            this.cursorPositionLast = editText.getSelectionStart();
        }
    }

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

    }

    @Override
    public void afterTextChanged(Editable s) {
        if (s.toString().length() > maxLength) {

            int left = maxLength - s.toString().length();

            bypass = true;
            s.clear();

            bypass = true;
            s.append(textLast);

            editText.setSelection(this.cursorPositionLast);

            if (if_callback != null) {
                if_callback.callback(left);
            }
        }

    }

}


edit_text.addTextChangedListener(new LimitTextWatcher(edit_text, MAX_LENGTH, new LimitTextWatcher.IF_callback() {
    @Override
    public void callback(int left) {
        if(left <= 0) {
            Toast.makeText(MainActivity.this, "input is full", Toast.LENGTH_SHORT).show();
        }
    }
}));

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

Например, максимальная длина установлена ​​на 10, пользователь ввел «12345678» и пометил «345» как выделение и попытайтесь вставить строку «0000», которая превысит ограничение.

Когда я пытаюсь использовать edit_text.setSelection (start = 2, end = 4) для восстановления статуса источника, в результате он просто вставляет 2 пробела как «12 345 678», а не выделение источника. Я бы хотел, чтобы кто-нибудь это решил.

person hyyou2010    schedule 25.10.2017

Вы можете использовать android:maxLength="10" в EditText. (Здесь ограничение составляет до 10 символов).

person Jocelin    schedule 29.04.2018

Котлин:

edit_text.filters += InputFilter.LengthFilter(10)

ZTE Blade A520 имеет странный эффект. Когда вы вводите более 10 символов (например, 15), EditText показывает первые 10, но остальные 5 не видны и недоступны. Но когда вы удаляете символы с помощью Backspace, он сначала удаляет правые 5 символов, а затем удаляет оставшиеся 10. Чтобы преодолеть такое поведение, используйте решение:

android:inputType="textNoSuggestions|textVisiblePassword"
android:maxLength="10"

или это:

android:inputType="textNoSuggestions"

или это, если вы хотите получить предложения:

private class EditTextWatcher(private val view: EditText) : TextWatcher {
    private var position = 0
    private var oldText = ""

    override fun afterTextChanged(s: Editable?) = Unit

    override fun beforeTextChanged(s: CharSequence?, start: Int, count: Int, after: Int) {
        oldText = s?.toString() ?: ""
        position = view.selectionStart
    }

    override fun onTextChanged(s: CharSequence?, start: Int, before: Int, count: Int) {
        val newText = s?.toString() ?: ""
        if (newText.length > 10) {
            with(view) {
                setText(oldText)
                position = if (start > 0 && count > 2) {
                    // Text paste in nonempty field.
                    start
                } else {
                    if (position in 1..10 + 1) {
                        // Symbol paste in the beginning or middle of the field.
                        position - 1
                    } else {
                        if (start > 0) {
                            // Adding symbol to the end of the field.
                            start - 1
                        } else {
                            // Text paste in the empty field.
                            0
                        }
                    }
                }
                setSelection(position)
            }
        }
    }
}

// Usage:
editTextWatcher = EditTextWatcher(view.edit_text)
view.edit_text.addTextChangedListener(editTextWatcher)
person CoolMind    schedule 14.10.2019

person    schedule
comment
Это похоже на то, как Android делает это с xml. - person Nicolas Tyler; 01.08.2017
comment
Как установить минимальную длину? - person Akhila Madari; 01.11.2018
comment
@AkhilaMadari, наверное, с TextWatcher. - person CoolMind; 10.10.2019

person    schedule
comment
Конечно, только для android - person VAdaihiep; 06.04.2016