Какой лучший способ ограничить длину текста EditText
в Android?
Есть ли способ сделать это через xml?
используйте входной фильтр, чтобы ограничить максимальную длину текстового вида.
TextView editEntryView = new TextView(...);
InputFilter[] filterArray = new InputFilter[1];
filterArray[0] = new InputFilter.LengthFilter(8);
editEntryView.setFilters(filterArray);
InputFilter
. Он переопределяет android:maxlength
в XML-файле, поэтому нам нужно добавить LengthFilter
таким образом.
EditText editText = new EditText(this);
int maxLength = 3;
editText.setFilters(new InputFilter[] {new InputFilter.LengthFilter(maxLength)});
Заметка для людей, которые уже используют настраиваемый входной фильтр, а также хотят ограничить максимальную длину:
При назначении входных фильтров в коде все ранее установленные фильтры ввода очищаются, в том числе один набор с android:maxLength
. Я нашел это при попытке использовать настраиваемый фильтр ввода, чтобы предотвратить использование некоторых символов, которые мы не разрешаем в поле пароля. После установки этого фильтра с помощью setFilters maxLength больше не наблюдалось. Решение заключалось в том, чтобы установить maxLength и мой пользовательский фильтр вместе программно. Что-то вроде этого:
myEditText.setFilters(new InputFilter[] {
new PasswordCharFilter(), new InputFilter.LengthFilter(20)
});
TextView tv = new TextView(this);
tv.setFilters(new InputFilter[]{ new InputFilter.LengthFilter(250) });
Для кого-то еще интересно, как этого добиться, вот мой расширенный 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
, тоже работают.
Из-за наблюдения 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 });
}
}
//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()});
Xml
android:maxLength="10"
Джава:
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)
У меня была эта проблема, и я считаю, что нам не хватает хорошо объясненного способа сделать это программно, не теряя уже установленные фильтры.
Установка длины в XML:
Поскольку правильно принятый ответ правильно, если вы хотите определить фиксированную длину в EditText, который вы не измените дальше в будущем, просто определите в XML-редакторе EditText:
android:maxLength="10"
Настройка длины программно
Чтобы установить длину программно, вам необходимо установить InputFilter
.
Проблема для Java: после того, как вы установите ее, все остальные фильтры исчезнут (например, maxLines, inputType и т.д.). Чтобы избежать потери предыдущих фильтров, вам необходимо получить эти предыдущие применяемые фильтры, добавить maxLength и установить фильтры обратно в EditText следующим образом:
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(10); //the desired length
editText.setFilters(newFilters);
Kotlin, однако, облегчил для всех, вам также нужно добавить фильтр к уже существующим, но вы можете добиться этого простым:
editText.filters += InputFilter.LengthFilter(maxLength)
Еще один способ, которым вы можете достичь этого, - добавить в 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 символов.
Это отлично работает...
android:maxLength="10"
это будет принимать только символы 10
.
это простой способ в 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" />
Это настраиваемый класс 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 });
}
}
}
Из 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 с помощью drawable:
Или вы можете ограничить длину текста с/без счетчика:
Зависимость:
implementation 'com.google.android.material:material:1.1.0-alpha02'
Вы можете использовать android:maxLength="10"
в EditText (здесь предел составляет до 10 символов)
Я видел много хороших решений, но я хотел бы дать то, что я думаю, как более полное и удобное для пользователя решение, которое включает в себя:
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 ', а не изначальное выделение. Я бы хотел, чтобы кто-то решил это.
попробуй этот код
android:maxLength="10"