Який найкращий спосіб обмежити довжину тексту EditText
в Android?
Чи можна це зробити через xml?
Який найкращий спосіб обмежити довжину тексту EditText
в Android?
Чи можна це зробити через xml?
Відповіді:
maxLength
Нерухомість по- , як і раніше працює.
android:maxLength
це еквівалентно InputFilter.LengthFilter
, тому при програмній зміні фільтра ви також змінили його фільтр XML.
setFilters
перестануть android:maxLength
працювати, оскільки він замінює фільтр, встановлений 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)});
TextWatcher
.
Примітка для людей, які вже використовують спеціальний фільтр введення, а також хочуть обмежити максимальну довжину:
При призначенні вхідних фільтрів у коді очищаються всі раніше встановлені вхідні фільтри, включаючи один набір android:maxLength
. Це я з'ясував, намагаючись використовувати спеціальний фільтр введення для запобігання використанню деяких символів, які ми не допускаємо у полі пароля. Після встановлення цього фільтра із setFilters maxLength більше не спостерігався. Рішенням було встановити maxLength і мій спеціальний фільтр разом програмно. Щось на зразок цього:
myEditText.setFilters(new InputFilter[] {
new PasswordCharFilter(), new InputFilter.LengthFilter(20)
});
У мене була ця проблема, і я вважаю, що нам не вистачає добре поясненого способу зробити це програмно, не втрачаючи вже встановлені фільтри.
Встановлення довжини в XML:
Як прийнято відповідати правильно, якщо ви хочете визначити фіксовану довжину для редагування текста, яку ви не будете більше змінювати в майбутньому, просто визначте у своєму редагуванні XML EditText:
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);
Але Котлін полегшив всім, вам також потрібно додати фільтр до вже існуючих, але ви можете досягти цього просто:
editText.filters += InputFilter.LengthFilter(maxLength)
Для всіх, хто цікавиться, як цього досягти, ось мій розширений 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"
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)
Ще один спосіб досягти цього - додавши наступне визначення до файлу 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 символів.
На сайті 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 за допомогою малювання:
Або ви можете обмежити довжину тексту за допомогою / без лічильника:
Залежність:
implementation 'com.google.android.material:material:1.1.0-alpha02'
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
Це спеціальний клас 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 });
}
}
}
це простий спосіб у 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" />
Це добре працює ...
android:maxLength="10"
це прийме лише 10
символи.
Спробуйте це для Java програмно :
myEditText(new InputFilter[] {new InputFilter.LengthFilter(CUSTOM_MAX_LEN)});
myEditText.setFilters(new InputFilter[] {new InputFilter.LengthFilter(CUSTOM_MAX_LEN)});
Я бачив багато хороших рішень, але хотів би дати те, що я вважаю більш повним і зручним для користувача рішенням, яке включає:
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 (старт = 2, кінець = 4) для відновлення статусу початкового коду, результат - просто вставити 2 пробіли як "12 345 678", а не підсвітку початку. Я хотів би, щоб хтось вирішив це.
Котлін:
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)
це простий спосіб у 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 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()]));
}
}