/**
* 输入最大值限制
*/
class InputFilterValue(var context: Context, private var mMax: Int, var showToast: Boolean = true, var toast: String = "最大可输入${mMax}") : InputFilter {
override fun filter(source: CharSequence?, start: Int, end: Int, dest: Spanned?, dstart: Int, dend: Int): CharSequence? {
val plus = dest.toString().plus(source)
if (TextUtils.isEmpty(plus)) return null
return try {
//mother fucker 这里要try catch一下
//场景 : plus为非int类型,会导致numberFormatException
//即使xml文件中设置了digits为0-9,但因为我们在代码中设置了inputFilterValue的过滤器,导致
//原本digits的digitKeyListener过滤器的优先级在inputFilterValue的后面了,所以用户此时还是可以输入除了digits之外的内容的
when {
plus.toInt() > mMax -> {
if (showToast) ToastUtil.showCenter(context, toast)
""
}
else -> null
}
} catch (e: NumberFormatException) {
null
}
}
}
使用
InputFilter[] filters = editText.getFilters();
InputFilter[] newFilters = new InputFilter[filters.length + 1];
System.arraycopy(filters, 0, newFilters, 0, filters.length);
newFilters[filters.length] = new InputFilterValue(...)
editText.setFilters(newFilters);
这里要注意的是,我们需要在自定义的inputFilter中try catch
一下,以为我们如果直接把输入的值toInt
的话是会有问题的。
有些同学可能想了,我xml文件中已经设置了digits
为0-9
了啊,为什么toInt
还会有问题呢,难道此时ediText还可以输入别的值不成?那我设置的digits
岂不是无效了?
这里我们看下editText.setFilters(newFilters)
的源码处理:
public void setFilters(InputFilter[] filters) {
if (filters == null) {
throw new IllegalArgumentException();
}
mFilters = filters;
if (mText instanceof Editable) {
setFilters((Editable) mText, filters);
}
}
private void setFilters(Editable e, InputFilter[] filters) {
if (mEditor != null) {
final boolean undoFilter = mEditor.mUndoInputFilter != null;
final boolean keyFilter = mEditor.mKeyListener instanceof InputFilter;
int num = 0;
if (undoFilter) num++;
if (keyFilter) num++;
if (num > 0) {
InputFilter[] nf = new InputFilter[filters.length + num];
System.arraycopy(filters, 0, nf, 0, filters.length);
num = 0;
if (undoFilter) {
nf[filters.length] = mEditor.mUndoInputFilter;
num++;
}
if (keyFilter) {
nf[filters.length + num] = (InputFilter) mEditor.mKeyListener;
}
e.setFilters(nf);
return;
}
}
e.setFilters(filters);
}
通过源码我们可以发现,我们再重新设置inputFilter
时,系统是会把我们设置的digits
重新copy过来的,但这里有个问题就是,digits
的优先级被降低了,我们再键盘输入的时候就会导致先执行了我们自定义的inputFilter
.这里我们可以看下键盘输入最终对输入内容的处理:
public SpannableStringBuilder replace(final int start, final int end,
CharSequence tb, int tbstart, int tbend) {
checkRange("replace", start, end);
int filtercount = mFilters.length;
for (int i = 0; i < filtercount; i++) {
CharSequence repl = mFilters[i].filter(tb, tbstart, tbend, this, start, end);
if (repl != null) {
tb = repl;
tbstart = 0;
tbend = repl.length();
}
}
....
}
可以看到这里最后调用的就是我们editText
设置的inputFilter
,这儿因为我们上面setInputFilter
导致digits
的优先级变低,所以用户此时输入的内容是没有先受digits
的限制的,而是先受限于我们自定义的inputFilter
,所以我们在上面的自定义的inputFilterValue中需要try catch一下.