Android中EditText的filter过滤(kotlin版)

本篇文章用了两个例子来说明filter的使用,其中注释说明了每一个参数的意义

用于过滤空格和换行的

作用:不能在开头和结尾连续的空格

不能再结尾连续换行

class SpaceAndLineInputFilter : InputFilter {
    override fun filter(
        source: CharSequence?,//即将要输入的字符串
        start: Int,//source的start
        end: Int,//source的end
        dest: Spanned?,//输入框中原来的内容
        dstart: Int,//光标所在位置
        dend: Int//光标终止位置
    ): CharSequence {
        var result = dest.toString()
        var resultInput = source.toString()
        //不能连续空格,只限定在开始和结尾
        if ((" ") == resultInput && result.startsWith(" ") && (dstart == 0 || dstart == 1)) {
            return ""
        }
        if ((" ") == resultInput && result.endsWith(" ") && (dstart == result.length || dstart == result.length - 1)) {
            return ""
        }
        //不能连续换行,只限定在结尾
        if (("\n") == resultInput && result.endsWith("\n") && dstart == result.length) {
            return ""
        }
        return resultInput
    }
}

使用

 et_filter.filters = arrayOf(SpaceAndLineInputFilter())

用于限制输入最大长度的

作用:可以限制最大长度时提示已达到最大长度

可以选择一个汉字是否代表两个字符

/**
 * @Anthor:Tian
 * @Date:2020/10/5
 * @Description: 限制最大字数
 */
class MaxLengthInputFilter(maxLen: Int, isChineseChar: Boolean, message: String) : InputFilter {
    private var maxCount: Int = maxLen //最长的字符数
    private var message: CharSequence? = message //要提示的信息
    private var isChinese2Char = isChineseChar //是否汉字相当于两个字符(在名字中)

    companion object {
        //Unicode编码中文的范围
         const val regEx = "[\\u4e00-\\u9fa5]"
        //表情的范围
         const val regEm = "[\\ud83d-\\ude3d]"
    }


    //过滤规则
    override fun filter(source: CharSequence?, start: Int, end: Int, dest: Spanned?, dstart: Int, dend: Int): CharSequence {
        //如果一个汉字相当于两个字符,则额外加一遍中文个数
        //当前已有的字符数
        val curCount = dest.toString().length + (if (isChinese2Char) getChineseCount(dest.toString()) else 0)
        //输入框的字符数
        val inputCount = source.toString().length + (if (isChinese2Char) getChineseCount(source.toString()) else 0)
        //用于保存可以放下的字符串
        var result: String = ""
        //超出最大字符数
        if (curCount + inputCount > maxCount) {
            var restCount = maxCount - curCount
            var index = 0
            while (restCount > 0) {
                val c = source?.get(index)
                if (isChinese2Char and isChinese(c.toString())
                        or isEmojiCharacter(c.toString())) { //当为汉字或者表情
                    if (restCount >= 2) {
                        restCount -= 2
                        result += c
                    }
                } else {
                    result += c
                    restCount -= 1
                }
                index++
            }
            //当达到最大字符数时提示
            Toast.makeText(BaseApplication.getContext(), message, Toast.LENGTH_SHORT).show()
            return result
        } else {
            return source.toString()
        }
    }

    private fun isChinese(str: String): Boolean {
        return Pattern.matches(regEx, str)
    }

    //判断是否为表情
    private fun isEmojiCharacter(str : String) : Boolean {
        return Pattern.matches(regEm, str);
    }

    // 计算中文字符的个数
    private fun getChineseCount(str: String?): Int {
        var count = 0
        val p = Pattern.compile(regEx)
        val m = p.matcher(str)
        while (m.find()) {
            for (i in 0..m.groupCount()) {
                count += 1
            }
        }
        return count
    }

}

使用 

et_filter.filters = arrayOf(MaxLengthInputFilter(10,true,"已达到最大字数"))

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值