Как я могу форматировать EditText
для соответствия формату "dd/mm/yyyy
" так же, как мы можем форматировать, используя TextWatcher
, чтобы замаскировать пользовательский ввод, чтобы он выглядел как "0.05 €". Я не говорю об ограничении символов или подтверждении даты, просто маскируя предыдущий формат.
Как замаскировать EditText, чтобы показать формат даты dd/mm/yyyy
Ответ 1
Я написал этот TextWatcher
для проекта, надеюсь, кому-то это будет полезно. Обратите внимание, что он не проверяет дату, введенную пользователем, и вы должны обращаться с этим при изменении фокуса, так как пользователь может не завершить ввод даты.
Обновление 25/06 Сделано это вики, чтобы узнать, достигли ли мы лучшего финального кода.
Обновление 07/06 Наконец, я добавил некоторую валидацию самому наблюдателю. Он будет делать следующие недопустимые даты:
- Если месяц больше 12, это будет 12 (декабрь)
- Если дата больше той, которая была выбрана для выбранного месяца, сделайте ее максимальной для этого месяца.
- Если год не находится в диапазоне
1900-2100
, измените его на диапазон
Эта проверка соответствует моим потребностям, но некоторые из вас могут немного изменить ее, диапазоны легко меняются, и вы можете подключить эти проверки к сообщению Toast
, например, чтобы уведомить пользователя о том, что мы изменили его/ее дата, поскольку она была недействительной.
В этом коде я предполагаю, что мы ссылаемся на наш EditText
, называемый date
, который имеет этот TextWatcher
, прикрепленный к нему, это можно сделать примерно так:
EditText date;
date = (EditText)findViewById(R.id.whichdate);
date.addTextChangedListener(tw);
TextWatcher tw = new TextWatcher() {
private String current = "";
private String ddmmyyyy = "DDMMYYYY";
private Calendar cal = Calendar.getInstance();
Когда пользователь меняет текст EditText
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
if (!s.toString().equals(current)) {
String clean = s.toString().replaceAll("[^\\d.]|\\.", "");
String cleanC = current.replaceAll("[^\\d.]|\\.", "");
int cl = clean.length();
int sel = cl;
for (int i = 2; i <= cl && i < 6; i += 2) {
sel++;
}
//Fix for pressing delete next to a forward slash
if (clean.equals(cleanC)) sel--;
if (clean.length() < 8){
clean = clean + ddmmyyyy.substring(clean.length());
}else{
//This part makes sure that when we finish entering numbers
//the date is correct, fixing it otherwise
int day = Integer.parseInt(clean.substring(0,2));
int mon = Integer.parseInt(clean.substring(2,4));
int year = Integer.parseInt(clean.substring(4,8));
mon = mon < 1 ? 1 : mon > 12 ? 12 : mon;
cal.set(Calendar.MONTH, mon-1);
year = (year<1900)?1900:(year>2100)?2100:year;
cal.set(Calendar.YEAR, year);
// ^ first set year for the line below to work correctly
//with leap years - otherwise, date e.g. 29/02/2012
//would be automatically corrected to 28/02/2012
day = (day > cal.getActualMaximum(Calendar.DATE))? cal.getActualMaximum(Calendar.DATE):day;
clean = String.format("%02d%02d%02d",day, mon, year);
}
clean = String.format("%s/%s/%s", clean.substring(0, 2),
clean.substring(2, 4),
clean.substring(4, 8));
sel = sel < 0 ? 0 : sel;
current = clean;
date.setText(current);
date.setSelection(sel < current.length() ? sel : current.length());
}
}
Мы также реализуем две другие функции, поскольку мы должны
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {}
@Override
public void afterTextChanged(Editable s) {}
};
Это создает следующий эффект, при котором удаление или вставка символов откроет или скроет маску dd/mm/yyyy
. Его легко модифицировать, чтобы соответствовать другим форматным маскам, поскольку я попытался оставить код как можно более простым.
Ответ 2
Нынешний ответ очень хорош и помог мне помочь моему собственному решению. Есть несколько причин, по которым я решил опубликовать собственное решение, хотя этот вопрос уже имеет правильный ответ:
- Я работаю в Котлине, а не на Java. Люди, которые оказываются с одинаковой проблемой, должны будут перевести текущее решение.
- Я хотел написать ответ, который был более разборчивым, чтобы люди могли легче адаптировать его к своим собственным проблемам.
- Как было предложено dengue8830, я инкапсулировал решение этой проблемы в класс, поэтому любой может использовать, даже не беспокоясь о реализации.
Чтобы использовать его, просто сделайте что-нибудь вроде:
- DateInputMask (mEditText).listen()
И решение показано ниже:
class DateInputMask(val input : EditText) {
fun listen() {
input.addTextChangedListener(mDateEntryWatcher)
}
private val mDateEntryWatcher = object : TextWatcher {
var edited = false
val dividerCharacter = "/"
override fun onTextChanged(s: CharSequence, start: Int, before: Int, count: Int) {
if (edited) {
edited = false
return
}
var working = getEditText()
working = manageDateDivider(working, 2, start, before)
working = manageDateDivider(working, 5, start, before)
edited = true
input.setText(working)
input.setSelection(input.text.length)
}
private fun manageDateDivider(working: String, position : Int, start: Int, before: Int) : String{
if (working.length == position) {
return if (before <= position && start < position)
working + dividerCharacter
else
working.dropLast(1)
}
return working
}
private fun getEditText() : String {
return if (input.text.length >= 10)
input.text.toString().substring(0,10)
else
input.text.toString()
}
override fun afterTextChanged(s: Editable) {}
override fun beforeTextChanged(s: CharSequence, start: Int, count: Int, after: Int) {}
}
}
Ответ 3
более чистый способ использования кода Juan Cortés помещается в класс:
public class DateInputMask implements TextWatcher {
private String current = "";
private String ddmmyyyy = "DDMMYYYY";
private Calendar cal = Calendar.getInstance();
private EditText input;
public DateInputMask(EditText input) {
this.input = input;
this.input.addTextChangedListener(this);
}
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
if (s.toString().equals(current)) {
return;
}
String clean = s.toString().replaceAll("[^\\d.]|\\.", "");
String cleanC = current.replaceAll("[^\\d.]|\\.", "");
int cl = clean.length();
int sel = cl;
for (int i = 2; i <= cl && i < 6; i += 2) {
sel++;
}
//Fix for pressing delete next to a forward slash
if (clean.equals(cleanC)) sel--;
if (clean.length() < 8){
clean = clean + ddmmyyyy.substring(clean.length());
}else{
//This part makes sure that when we finish entering numbers
//the date is correct, fixing it otherwise
int day = Integer.parseInt(clean.substring(0,2));
int mon = Integer.parseInt(clean.substring(2,4));
int year = Integer.parseInt(clean.substring(4,8));
mon = mon < 1 ? 1 : mon > 12 ? 12 : mon;
cal.set(Calendar.MONTH, mon-1);
year = (year<1900)?1900:(year>2100)?2100:year;
cal.set(Calendar.YEAR, year);
// ^ first set year for the line below to work correctly
//with leap years - otherwise, date e.g. 29/02/2012
//would be automatically corrected to 28/02/2012
day = (day > cal.getActualMaximum(Calendar.DATE))? cal.getActualMaximum(Calendar.DATE):day;
clean = String.format("%02d%02d%02d",day, mon, year);
}
clean = String.format("%s/%s/%s", clean.substring(0, 2),
clean.substring(2, 4),
clean.substring(4, 8));
sel = sel < 0 ? 0 : sel;
current = clean;
input.setText(current);
input.setSelection(sel < current.length() ? sel : current.length());
}
@Override
public void afterTextChanged(Editable s) {
}
}
то вы можете его повторно использовать
new DateInputMask(myEditTextInstance);
Ответ 4
Вики Хуана Кортеса работает как шарм fooobar.com/questions/170380/...
Вот моя версия Kotlin
fun setBirthdayEditText() {
birthdayEditText.addTextChangedListener(object : TextWatcher {
private var current = ""
private val ddmmyyyy = "DDMMYYYY"
private val cal = Calendar.getInstance()
override fun onTextChanged(p0: CharSequence?, p1: Int, p2: Int, p3: Int) {
if (p0.toString() != current) {
var clean = p0.toString().replace("[^\\d.]|\\.".toRegex(), "")
val cleanC = current.replace("[^\\d.]|\\.", "")
val cl = clean.length
var sel = cl
var i = 2
while (i <= cl && i < 6) {
sel++
i += 2
}
//Fix for pressing delete next to a forward slash
if (clean == cleanC) sel--
if (clean.length < 8) {
clean = clean + ddmmyyyy.substring(clean.length)
} else {
//This part makes sure that when we finish entering numbers
//the date is correct, fixing it otherwise
var day = Integer.parseInt(clean.substring(0, 2))
var mon = Integer.parseInt(clean.substring(2, 4))
var year = Integer.parseInt(clean.substring(4, 8))
mon = if (mon < 1) 1 else if (mon > 12) 12 else mon
cal.set(Calendar.MONTH, mon - 1)
year = if (year < 1900) 1900 else if (year > 2100) 2100 else year
cal.set(Calendar.YEAR, year)
// ^ first set year for the line below to work correctly
//with leap years - otherwise, date e.g. 29/02/2012
//would be automatically corrected to 28/02/2012
day = if (day > cal.getActualMaximum(Calendar.DATE)) cal.getActualMaximum(Calendar.DATE) else day
clean = String.format("%02d%02d%02d", day, mon, year)
}
clean = String.format("%s/%s/%s", clean.substring(0, 2),
clean.substring(2, 4),
clean.substring(4, 8))
sel = if (sel < 0) 0 else sel
current = clean
birthdayEditText.setText(current)
birthdayEditText.setSelection(if (sel < current.count()) sel else current.count())
}
}
override fun beforeTextChanged(p0: CharSequence?, p1: Int, p2: Int, p3: Int) {
}
override fun afterTextChanged(p0: Editable) {
}
})
}
Ответ 5
Попробуйте использовать библиотеку, которая решает эту проблему, так как маскировка недоступна из коробки. Есть много угловых случаев (например, добавление/удаление символов в середине уже замаскированного текста), и для правильной обработки этого у вас будет много кода (и ошибок).
Вот несколько доступных библиотек:
https://github.com/egslava/edittext-mask
https://github.com/dimitar-zabaznoski/MaskedEditText
https://github.com/pinball83/Masked-Edittext
https://github.com/RedMadRobot/input-mask-android
https://github.com/santalu/mask-edittext
** Помните, что на момент написания этих библиотек не обошлось без проблем, поэтому вы сами выбираете, какая из них вам больше подходит, и тестируйте код.
Ответ 6
Котлин версия без проверки
editText.addTextChangedListener(object : TextWatcher{
var sb : StringBuilder = StringBuilder("")
var _ignore = false
override fun afterTextChanged(s: Editable?) {}
override fun beforeTextChanged(s: CharSequence?, start: Int, count: Int, after: Int) {}
override fun onTextChanged(s: CharSequence?, start: Int, before: Int, count: Int) {
if(_ignore){
_ignore = false
return
}
sb.clear()
sb.append(if(s!!.length > 10){ s.subSequence(0,10) }else{ s })
if(sb.lastIndex == 2){
if(sb[2] != '/'){
sb.insert(2,"/")
}
} else if(sb.lastIndex == 5){
if(sb[5] != '/'){
sb.insert(5,"/")
}
}
_ignore = true
editText.setText(sb.toString())
editText.setSelection(sb.length)
}
})
Ответ 7
добавить android:inputType="date"
в ваш EditText
Ответ 8
В этом ответе не применяется полная маска для оставшихся нетипизированных цифр. Однако это связано и является решением, которое мне нужно. Он работает так же, как работает PhoneNumberFormattingTextWatcher
.
По мере ввода добавляет косые черты, чтобы отделить дату, отформатированную как mm/dd/yyyy
. Он не делает никакой проверки - просто форматирование.
Не требуется ссылка EditText
.
Просто установите слушателя, и он работает.
myEditText.addTextChangedListener(new DateTextWatcher());
import android.text.Editable;
import android.text.TextWatcher;
import java.util.Locale;
/**
* Adds slashes to a date so that it matches mm/dd/yyyy.
*
* Created by Mark Miller on 12/4/17.
*/
public class DateTextWatcher implements TextWatcher {
public static final int MAX_FORMAT_LENGTH = 8;
public static final int MIN_FORMAT_LENGTH = 3;
private String updatedText;
private boolean editing;
@Override
public void beforeTextChanged(CharSequence charSequence, int start, int before, int count) {
}
@Override
public void onTextChanged(CharSequence text, int start, int before, int count) {
if (text.toString().equals(updatedText) || editing) return;
String digitsOnly = text.toString().replaceAll("\\D", "");
int digitLen = digitsOnly.length();
if (digitLen < MIN_FORMAT_LENGTH || digitLen > MAX_FORMAT_LENGTH) {
updatedText = digitsOnly;
return;
}
if (digitLen <= 4) {
String month = digitsOnly.substring(0, 2);
String day = digitsOnly.substring(2);
updatedText = String.format(Locale.US, "%s/%s", month, day);
}
else {
String month = digitsOnly.substring(0, 2);
String day = digitsOnly.substring(2, 4);
String year = digitsOnly.substring(4);
updatedText = String.format(Locale.US, "%s/%s/%s", month, day, year);
}
}
@Override
public void afterTextChanged(Editable editable) {
if (editing) return;
editing = true;
editable.clear();
editable.insert(0, updatedText);
editing = false;
}
}
Ответ 9
Вы можете использовать приведенный ниже код, и он добавляет все проверки для даты, которая также будет действительной. Как дни не может быть больше 31; месяц не может быть больше 12 и т.д.
class DateMask : TextWatcher {
private var updatedText: String? = null
private var editing: Boolean = false
companion object {
private const val MAX_LENGTH = 8
private const val MIN_LENGTH = 2
}
override fun beforeTextChanged(charSequence: CharSequence, start: Int, before: Int, count: Int) {
}
override fun onTextChanged(text: CharSequence, start: Int, before: Int, count: Int) {
if (text.toString() == updatedText || editing) return
var digits = text.toString().replace("\\D".toRegex(), "")
val length = digits.length
if (length <= MIN_LENGTH) {
digits = validateMonth(digits)
updatedText = digits
return
}
if (length > MAX_LENGTH) {
digits = digits.substring(0, MAX_LENGTH)
}
updatedText = if (length <= 4) {
digits = validateDay(digits.substring(0, 2), digits.substring(2))
val month = digits.substring(0, 2)
val day = digits.substring(2)
String.format(Locale.US, "%s/%s", month, day)
} else {
digits = digits.substring(0, 2) + digits.substring(2, 4) + validateYear(digits.substring(4))
val month = digits.substring(0, 2)
val day = digits.substring(2, 4)
val year = digits.substring(4)
String.format(Locale.US, "%s/%s/%s", month, day, year)
}
}
private fun validateDay(month: String, day: String): String {
val arr31 = intArrayOf(1, 3, 5, 7, 8, 10, 12)
val arr30 = intArrayOf(4, 6, 9, 11)
val arrFeb = intArrayOf(2)
if (day.length == 1 &&
((day.toInt() > 3 && month.toInt() !in arrFeb)
|| (day.toInt() > 2 && month.toInt() in arrFeb))) {
return month
}
return when (month.toInt()) {
in arr31 -> validateDay(month, arr31, day, 31)
in arr30 -> validateDay(month, arr30, day, 30)
in arrFeb -> validateDay(month, arrFeb, day, 29)
else -> "$month$day"
}
}
private fun validateDay(month: String, arr: IntArray, day: String, maxDay: Int): String {
if (month.toInt() in arr) {
if (day.toInt() > maxDay) {
return "$month${day.substring(0, 1)}"
}
}
return "$month$day"
}
private fun validateYear(year: String): String {
if (year.length == 1 && (year.toInt() in 3..9 || year.toInt() == 0)) {
return ""
}
if (year.length == 2 && year.toInt() !in 19..20) {
return year.substring(0, 1)
}
return year
}
private fun validateMonth(month: String): String {
if (month.length == 1 && month.toInt() in 2..9) {
return "0$month"
}
if (month.length == 2 && month.toInt() > 12) {
return month.substring(0, 1)
}
return month
}
override fun afterTextChanged(editable: Editable) {
if (editing) return
editing = true
editable.clear()
editable.insert(0, updatedText)
editing = false
}
}
В вашем fragment
или Activity
вы можете использовать это DateMask
так:
mEditText?.addTextChangedListener(dateMask)