Android 优雅的实现通用格式化编辑

作者:张坤的笔记 时间:2023-02-08 05:24:02 

目录
  • 使用

    • 格式化编辑手机号

    • 格式化编辑身份证号

    • 设置监听

    • 移除格式化的文本

  • 实现原理

    • 项目地址

      格式化编辑的需求一般是从编辑手机号开始的,UI 给出的效果不是11个连续的数字,而是采用3、4、4的形式,每段中间会空一个字符。在技术实现的时候,一般会自定义一个控件 TelEditText 实现功能,随着项目迭代,格式化编辑的需求可能会增加,比如说身份证号、自定义的优惠券码等,这个时候再给每种情况自定义一个控件就没必要了,通过一个控件实现多种格式化编辑需求是更好的方案。

      其实还可以更进一步,格式化编辑的核心逻辑就是给 EditText 添加一个 TextWatcher,通过 TextWatcher 中的文本变化回调来调整 EditText 中的文本,所以自定义 EditText 并不是必须的,对于开发者需要调用的字段和方法,可以通过扩展函数的方式提供。

      使用

      格式化编辑手机号

      布局:


      <androidx.appcompat.widget.AppCompatEditText
      android:id="@+id/etPhone"
      android:layout_width="match_parent"
      android:layout_height="wrap_content"
      android:inputType="number" />

      代码:


      // format is ' '
      etPhone.setFormatRules(3, 4, 4)

      // format is '-'
      etPhone.setFormatRules(3, 4, 4, formatChar = '-')

      Android 优雅的实现通用格式化编辑

      格式化编辑身份证号

      布局:


      <androidx.appcompat.widget.AppCompatEditText
      android:id="@+id/etIDNumber"
      android:layout_width="match_parent"
      android:layout_height="wrap_content"
      android:digits="@string/digits_id_number"/>

      资源:


      <string name="digits_id_number">0123456789xX</string>

      代码:


      etIDNumber.setFormatRules(6, 4, 4, 4)

      Android 优雅的实现通用格式化编辑

      设置监听


      etPhone.setOnFormatEditListener { isComplete, text ->
      if (isComplete) { // 编辑完成
       // 使用 toast 显示移除格式化的文本
       Toast.makeText(this, text, Toast.LENGTH_SHORT).show()
      }
      }

      移除格式化的文本


      etPhone.textWithFormatRemoved

      实现原理

      自定义一个 TextWatcher,定义一个字段 formatChar,值为格式化字符,默认是空格。定义一个字段 formatCharIndexList,值为 EditText 文本中格式化字符所在位置的列表,比如对于格式化编辑手机号, formatCharIndexList 中的值为 [3, 8],既在 EditText 文本中格式化字符的位置应该是3和8。


      var formatChar: Char = ' '

      val formatCharIndexList = ArrayList<Int>()

      EditText 文本发生变化后,如果 EditText 文本的最后一个字符为格式化字符,则删除最后一个字符;然后遍历 EditText 文本中的每一个字符,如果该字符的位置等于格式化字符位置但不是格式化字符,则在该位置插入一个格式化字符,如果该字符的位置不等于格式化字符的位置但又是格式化字符,则删除该格式化字符。

      调用 insertFormatChar 或者 deleteChar 后,afterTextChanged 又会立即回调一次,可能会引起多次添加或删除,导致格式化错误。所以每次 afterTextChanged 回调最多进行一次操作,如果后续还需要操作,放在下一次 afterTextChanged 回调中进行。


      override fun afterTextChanged(s: Editable?) {
      val value = s?.toString() ?: return
      if (value.isEmpty()) return

      if (value.last() == formatChar) {
       deleteChar(s, value.lastIndex)
       return
      }

      value.forEachIndexed { index, c ->
       if (formatCharIndexList.contains(index)) {
        if (c != formatChar) {
         insertFormatChar(s, index)
         return
        }
       } else {
        if (c == formatChar) {
         deleteChar(s, index)
         return
        }
       }
      }
      ...
      }

      项目地址

      format-edit,觉得用起来很爽的,请不要吝啬你的 Star !

      来源:https://juejin.cn/post/6925487327993135117

      标签:Android,格式化,编辑
      0
      投稿

      猜你喜欢

    • C#用Activex实现Web客户端读取RFID功能的代码

      2021-10-28 21:07:36
    • Android开发实现简单计算器功能

      2022-10-17 09:18:16
    • Java 获取网站图片的示例代码

      2022-03-11 14:52:44
    • Java servlet后端开发超详细教程

      2022-11-01 06:13:50
    • Java内存溢出和内存泄露

      2023-01-21 17:48:48
    • Android使用ShareSDK实现应用分享的功能

      2023-10-02 06:37:42
    • 简单聊聊c# 事件

      2022-03-05 06:16:52
    • Java递归实现迷宫游戏

      2023-08-26 07:33:16
    • Android中复制图片的实例代码

      2023-01-07 21:34:26
    • 使用Jenkins来构建SVN+Maven项目的实现

      2023-07-30 12:45:22
    • Spring实战之XML与JavaConfig的混合配置详解

      2023-07-11 18:07:20
    • Android中asset文件夹与raw文件夹的区别深入解析

      2022-11-22 01:40:34
    • SpringBoot如何在普通类加载Spring容器

      2023-10-06 03:56:27
    • 基于C#的音乐播放器主Form实现代码

      2022-07-13 01:21:36
    • Spring security登录过程逻辑详解

      2022-09-04 19:58:20
    • C#中List<T>存放元素的工作机制

      2022-09-29 16:31:23
    • c#方法重写和隐藏的学习示例

      2022-12-24 14:20:12
    • C# 中闭包(Closure)详解

      2022-12-29 20:36:10
    • Android实现EventBus登录界面与传值(粘性事件)

      2023-11-08 00:34:52
    • 从Request.Url中获取根网址的简单操作

      2022-04-18 04:44:40
    • asp之家 软件编程 m.aspxhome.com