安卓禁止输入中文的简单实现

之前有过一段时间,安卓输入框中禁止输入中文的实现方式是自定义一个TextWatcher,然后在afterTextChanged中判断是否输入的是中文,如果是中文则再做撤销回滚操作,也就是做删掉输入中文的处理。不过感觉只是为了处理这一个逻辑就要写那么多臃肿的代码,简洁性以及可阅读性不好,后来想到了使用InputFilter过滤的方式,大大简化了代码,并将该方法公布到工具类当中便于其他地方复用。

1、使用方法

public static void filterChinese(TextView v) {
    v.setFilters(new InputFilter[]{new InputFilter() {
        @Override
        public CharSequence filter(CharSequence source, int start, int end, Spanned dest, int dstart, int dend) {
            if (null != source && isChinese(source.toString())) return "";
            return source;
        }
    }});
}

// 完整的判断中文汉字和符号
@SuppressWarnings("unused")
private static boolean isChinese(String strName) {
    char[] ch = strName.toCharArray();
    for (char c : ch) {
        if (isChinese(c)) {
            return true;
        }
    }
    return false;
}

// 根据Unicode编码完美的判断中文汉字和符号
private static boolean isChinese(char c) {
    Character.UnicodeBlock ub = Character.UnicodeBlock.of(c);
    return ub == Character.UnicodeBlock.CJK_UNIFIED_IDEOGRAPHS
            || ub == Character.UnicodeBlock.CJK_COMPATIBILITY_IDEOGRAPHS
            || ub == Character.UnicodeBlock.CJK_UNIFIED_IDEOGRAPHS_EXTENSION_A
            || ub == Character.UnicodeBlock.CJK_UNIFIED_IDEOGRAPHS_EXTENSION_B
            || ub == Character.UnicodeBlock.CJK_SYMBOLS_AND_PUNCTUATION
            || ub == Character.UnicodeBlock.HALFWIDTH_AND_FULLWIDTH_FORMS
            || ub == Character.UnicodeBlock.GENERAL_PUNCTUATION;
}

这里思想很简单,也就是在输入过程中直接过滤掉中文。

2、原理解析

对于想继续知道该过滤原理的同学,这里我直接上一些源码片段。
这里上下TextView的setFilters方法源码:

public void setFilters(InputFilter[] filters) {
    if (filters == null) {
        throw new IllegalArgumentException();
    }

    mFilters = filters;

    if (mText instanceof Editable) {
        setFilters((Editable) mText, filters);
    }
}

可以看出,实际上操作InputFilter的地方应该有两处,一处是TextView本身,还有一处是Editable。

  • 首先看下TextView处理的源码
private void setText(CharSequence text, BufferType type, boolean notifyBefore, int oldlen) {
    ......
    int n = mFilters.length;
    for (int i = 0; i < n; i++) {
        CharSequence out = mFilters[i].filter(text, 0, text.length(), EMPTY_SPANNED, 0, 0);
        if (out != null) {
            text = out;
        }
    }
    ......
}

代码很简单,就是在执行了setText时再执行mFilters数组,从中获取过滤掉之后的内容。

  • 然后再分析下Editable
    Editable实际上只是一个接口,这里可以想到的实际上实现该接口的是SpannableStringBuilder类,参见实现:
public class SpannableStringBuilder implements CharSequence, GetChars, Spannable, Editable, Appendable, GraphicsOperations

SpannableStringBuilder类主要作用是修改部分字体样式,背景颜色等等,这里不做赘述。处理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();
            }
        }
}

可以看出其当执行内容样式替换过程中过滤掉需要过滤的内容。

好了,本篇内容到此结束,如果有用到该方法的地方,拿走不谢。

你可能感兴趣的:(安卓禁止输入中文的简单实现)