Qual é a melhor maneira de limitar o tamanho do texto do EditText no Android

702

Qual é a melhor maneira de limitar o tamanho do texto de um EditTextno Android?

Existe uma maneira de fazer isso via xml?

hpique
fonte
1
Eu queria definir o número máximo de caracteres para o meu EditText. A princípio, não era óbvio que o limite de tamanho do texto era a mesma coisa. (Apenas uma nota para outro viajante confuso).
Katedral Pillon
A resposta correta está aqui : stackoverflow.com/a/19222238/276949 . Essa resposta limita o comprimento E evita que um buffer seja preenchido continuamente após atingir o limite, permitindo, assim, que a tecla backspace funcione corretamente.
Martin Konecny

Respostas:

1381

Documentação

Exemplo

android:maxLength="10"
Austin Hanson
fonte
26
D'oh! A mesma coisa aconteceu comigo. Eu estava olhando o código e não há um método setMaxLength.
Hpique
1
o que o Cheque aqui faz? É apenas um link para esta página.
que é sono
5
@Vincy, você está incorreto. A maxLengthpropriedade ainda funciona.
ashishduh
6
Esteja ciente de que android:maxLengthé equivalente a InputFilter.LengthFilter, portanto, ao alterar programaticamente o filtro, você também modificou o filtro XML.
Mr5
20
Para as pessoas que dizem que não funciona, saiba que a chamada setFiltersdeixará de android:maxLengthfuncionar, porque substitui o filtro definido pelo XML. Em outras palavras, se você definir qualquer filtro programaticamente, deverá defini-los todos programaticamente.
19616 Ian Newson
339

use um filtro de entrada para limitar o comprimento máximo de uma exibição de texto.

TextView editEntryView = new TextView(...);
InputFilter[] filterArray = new InputFilter[1];
filterArray[0] = new InputFilter.LengthFilter(8);
editEntryView.setFilters(filterArray);
jerry
fonte
5
Isso é muito útil se alguém já fez algum InputFilter. Ele substitui android:maxlengthno arquivo xml, por isso precisamos adicionar LengthFilterdessa maneira.
Seblis 12/09
Eu acho que sua resposta é a melhor resposta, uma vez que é mais dinâmico, +1
Muhannad A.Alhariri
197
EditText editText = new EditText(this);
int maxLength = 3;    
editText.setFilters(new InputFilter[] {new InputFilter.LengthFilter(maxLength)});
Emran Hamza
fonte
Isso é semelhante à maneira como o Android faz com o xml.
Nicolas Tyler
4
Como posso definir o comprimento mínimo?
Akhila Madari
@AkhilaMadari, provavelmente com TextWatcher.
CoolMind 10/10/19
65

Uma observação para as pessoas que já estão usando um filtro de entrada personalizado e também desejam limitar o comprimento máximo:

Quando você atribui filtros de entrada no código, todos os filtros de entrada definidos anteriormente são limpos, incluindo um conjunto com android:maxLength. Descobri isso ao tentar usar um filtro de entrada personalizado para impedir o uso de alguns caracteres que não permitimos em um campo de senha. Depois de definir esse filtro com setFilters, o maxLength não era mais observado. A solução foi definir maxLength e meu filtro personalizado juntos por meio de programação. Algo assim:

myEditText.setFilters(new InputFilter[] {
        new PasswordCharFilter(), new InputFilter.LengthFilter(20)
});
goto10
fonte
5
Você pode recuperar os filtros existentes primeiro como, InputFilter [] existingFilters = editText.getFilters (); Em seguida, adicione o filtro junto com este filtros existentes
subair_a
39
TextView tv = new TextView(this);
tv.setFilters(new InputFilter[]{ new InputFilter.LengthFilter(250) });
ramo2712
fonte
7
Claro, apenas para android
VAdaihiep 06/04
27

Eu tive esse problema e considero que estamos perdendo uma maneira bem explicada de fazer isso programaticamente sem perder os filtros já configurados.

Definindo o comprimento em XML:

Como a resposta aceita afirma corretamente, se você deseja definir um comprimento fixo para um EditText que não será alterado mais no futuro, basta definir no XML do EditText:

android:maxLength="10" 

Definir o comprimento programaticamente

Para definir o comprimento programaticamente, você precisará defini-lo através de um InputFilter. Mas se você criar um novo InputFilter e configurá-lo como EditText, perderá todos os outros filtros já definidos (por exemplo, maxLines, inputType, etc) que você pode ter adicionado por meio de XML ou programaticamente.

Então, isso está errado :

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

Para evitar a perda de filtros adicionados anteriormente, você precisa obtê-los, adicione o novo (maxLength nesse caso) e defina os filtros novamente da EditTextseguinte maneira:

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);

No entanto, o Kotlin tornou mais fácil para todos, você também precisa adicionar o filtro aos já existentes, mas pode conseguir isso com um simples:

editText.filters += InputFilter.LengthFilter(maxLength)
Ricardo
fonte
23

Para qualquer um que esteja se perguntando como conseguir isso, aqui está a minha EditTextaula estendida EditTextNumeric.

.setMaxLength(int) - define o número máximo de dígitos

.setMaxValue(int) - limite o valor inteiro máximo

.setMin(int) - limite do valor inteiro mínimo

.getValue() - obtém valor inteiro

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

Exemplo de uso:

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

Todos os outros métodos e atributos que se aplicam a EditText, é claro, também funcionam.

Martynas Januškauskas
fonte
1
Seria melhor se adicionarmos isso no layout xml. Estou recebendo erro com este quando utilizar xml Causada por: android.view.InflateException: linha do arquivo XML binário # 47: Erro de classe inflar com.passenger.ucabs.utils.EditTextNumeric
Shihab Uddin
@ Martynas Tem o mesmo erro como Shihab_returns qualquer solução?
Pranaysharma 6/03
17

Devido à observação do goto10, montei o seguinte código para proteger contra perda de outros filtros ao definir o comprimento máximo:

/**
 * 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 });
    }
}
Tim Gallagher
fonte
3
Você pode atualizar um pouco o código. A matriz alocada precisa ter curFilters.length + 1 em tamanho e depois de criar novosFiltros, você não definiu "this" para a matriz recém-alocada. InputFilter newFilters [] = novo InputFilter [curFilters.length + 1]; System.arraycopy (curFilters, 0, newFilters, 0, curFilters.length); this.setFilters (newFilters);
JavaCoderEx
15
//Set Length filter. Restricting to 10 characters only
editText.setFilters(new InputFilter[]{new InputFilter.LengthFilter(MAX_LENGTH)});

//Allowing only upper case characters
editText.setFilters(new InputFilter[]{new InputFilter.AllCaps()});

//Attaching multiple filters
editText.setFilters(new InputFilter[]{new InputFilter.LengthFilter(MAX_LENGTH), new InputFilter.AllCaps()});
Kishore Reddy
fonte
12

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);

Kotlin:

editText.filters += InputFilter.LengthFilter(maxLength)
João Carlos
fonte
8

Outra maneira de conseguir isso é adicionando a seguinte definição ao arquivo 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"/>

Isso limitará o comprimento máximo do EditTextwidget a 6 caracteres.

user2586194
fonte
6

No material.io , você pode usar TextInputEditTextcombinado com 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>

Você pode configurar uma senha EditText com drawable:

exemplo de senha

Ou você pode limitar o tamanho do texto com / sem um contador:

exemplo contrário

Dependência:

implementation 'com.google.android.material:material:1.1.0-alpha02'
RonTLV
fonte
6

XML

android:maxLength="10"

Programaticamente:

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

Nota: internamente, EditText e TextView analisam o valor android:maxLengthem XML e usam InputFilter.LengthFilter()para aplicá-lo.

Consulte: TextView.java # L1564

nhoxbypass
fonte
2

Essa é uma classe EditText personalizada que permite que o filtro Comprimento seja exibido junto com outros filtros. Obrigado à resposta de Tim Gallagher (abaixo)

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 });
        }
    }
}
Gal Rom
fonte
2

maneira simples em xml:

android:maxLength="4"

se você precisar definir 4 caracteres em xml edit-text, use este

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

Isso funciona bem ...

android:maxLength="10"

isso aceitará apenas 10caracteres.

Sanjay Kumaar
fonte
2

Tente isso para Java programaticamente :

myEditText(new InputFilter[] {new InputFilter.LengthFilter(CUSTOM_MAX_LEN)});
reza_khalafi
fonte
1
Você esqueceu de chamar o método para que ele se pareça com: myEditText.setFilters(new InputFilter[] {new InputFilter.LengthFilter(CUSTOM_MAX_LEN)});
StevenTB 13/09/1919
1

Eu já tinha visto muitas soluções boas, mas gostaria de dar uma solução mais completa e fácil de usar, que inclui:

1, limite de comprimento.
2, Se inserir mais, dê um retorno de chamada para acionar sua torrada.
3, o cursor pode estar no meio ou na cauda.
4, o usuário pode inserir colando uma string.
5, sempre descarte a entrada de excesso e mantenha a origem.

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();
        }
    }
}));

O que falhei em fazer é, se o usuário destacar uma parte da entrada atual e tentar colar uma sequência muito longa, não sei como restaurar o destaque.

Como comprimento máximo definido como 10, o usuário inseriu '12345678' e marque '345' como destaque e tente colar uma sequência de '0000' que excederá a limitação.

Quando tento usar edit_text.setSelection (start = 2, end = 4) para restaurar o status de origem, o resultado é: basta inserir 2 espaços como '12 345 678 ', não o destaque da origem. Eu gostaria que alguém resolvesse isso.

hyyou2010
fonte
1

Você pode usar android:maxLength="10"o EditText (aqui o limite é de até 10 caracteres)

Jocelin
fonte
0

Kotlin:

edit_text.filters += InputFilter.LengthFilter(10)

ZTE Blade A520tem efeito estranho. Quando você digita mais de 10 símbolos (por exemplo, 15), EditTextmostra os 10 primeiros, mas os outros 5 não são visíveis nem acessíveis. Mas quando você exclui símbolos com Backspace, ele primeiro exclui os 5 símbolos corretos e remove os 10 restantes. Para superar esse comportamento, use uma solução :

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

ou isto:

android:inputType="textNoSuggestions"

ou isso, se você quiser sugestões:

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)
CoolMind
fonte
0

maneira simples em xml:

android:maxLength="@{length}"

para defini-lo programaticamente, você pode usar a seguinte função

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 length = filters.length;
        int i3 = 0;
        while (i2 < length) {
            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()]));
    }
}
Mohamed Fadel Buffon
fonte