TextWatcher - 光标回到 EditText 的开头

TextWatcher - cursor goes back to the start of EditText

我得到了以下 TextWatcher 定义:

   _textWatcher = new TextWatcher() {
        @Override
        public void beforeTextChanged(CharSequence s, int start, int count, int after) {

        }

        @Override
        public void onTextChanged(CharSequence s, int start, int before, int count) {

        }

        @Override
        public void afterTextChanged(final Editable s) {
            if (!s.toString().equals(_prev)) {
                afterTextHandler.removeCallbacks(work);
                work = new Runnable() {
                    @Override
                    public void run() {                            
                        if (!s.toString().equals(_prevQuoteAmount)) {
                            _prev = s.toString();
                            doSomething();
                        }
                        _et.append("");                            
                    }
                };
                afterTextHandler.postDelayed(work, 1000);
            }
            _et.append("");                
        }
    };

一切都按预期工作(有时 doSomething() 执行,有时不执行,因此它的执行与问题无关),但是在用户在附加了此观察者的 EditText 组件中键入一些文本后,文本的光标返回到 EditText 的开头。

在执行 onTextChanged()(或其处理程序 运行() 执行)后显示以下日志:

12-08 13:10:55.777 1855-1855/com.android.inputmethod.latin I/LatinIME: Starting input. Cursor position = 4,4

为什么刚执行完onTextChanged(),EditText的光标就放在了EditText的开头?

编辑 1: _et.append("") 是一个 hack,应该将光标带到文本的最后一个地方。它在这里不起作用(尽管它在其他场合起作用)。

编辑 2: 请求有关 doSomething 的知识:

    if (isReady()) {
        Sender objSender = new Sender();
        objSender.setParam1(_tvParam1.getText().toString());
        objSender.setParam2(_tvParam2.getText().toString());
        objSender.setParam3(_et.getText().toString());
        Service.startActionSend(getActivity(), quote);
    }

isReady() - 在发出网络请求之前检查条件(_tvParam1、_tvParam2、_et)。 Service.startActionSend() - 发出异步网络请求(使用 IntentService)。

编辑 3: 该问题仅在进入处理程序的 运行() 方法时发生。

我不确定你在 doSomething() 中做了什么,但如果这会以某种方式改变光标的位置,你可以使用以下 (hack) 来带来光标到输入文本的末尾。

mEdittext.setSelection(mEdittext.length());

尝试使用布尔变量来检查状态

desc.addTextChangedListener(new TextWatcher() {
        boolean changed = false;
        @Override
        public void beforeTextChanged(CharSequence charSequence, int i, int i1, int i2) {

        }

        @Override
        public void onTextChanged(CharSequence charSequence, int i, int i1, int i2) {

        }

        @Override
        public void afterTextChanged(Editable editable) {
            if (desc.getText().toString().isEmpty()) {

                changed = false;
            } else {
                if(!changed) {
                    doSomething();
                }
                changed = true;
            }
        }
    });

您可以使用此函数获取要在编辑文本的 setSelection 中使用的光标 还有

private fun getNewCursorPosition(digitCountToRightOfCursor : Int, numberString : String) : Int{
    var position = 0
    var c = digitCountToRightOfCursor
    for (i in numberString.reversed()) {
        if (c == 0)
            break

        if (i.isDigit())
            c --
        position ++


    }
    return numberString.length - position
}

整个样本使用

editText.addTextChangedListener(object : TextWatcher {
        override fun afterTextChanged(p0: Editable?) {
        }

        override fun beforeTextChanged(p0: CharSequence?, start: Int, count: Int, after: Int) {
            beforeText = p0.toString()
        }

        override fun onTextChanged(p0: CharSequence?, start: Int, before: Int, count: Int) {
            
            if (p0 == null)
                return
            // 1. get cursor position : p0 = start + before
            val initialCursorPosition = start + before
            //2. get digit count after cursor position : c0
            val numOfDigitsToRightOfCursor = getNumberOfDigits(beforeText.substring(initialCursorPosition,
                    beforeText.length))
            val newAmount = formatAmount(p0.toString())
            editText.removeTextChangedListener(this)
            editText.setText(newAmount)
            //set new cursor position
            editText.setSelection(getNewCursorPosition(numOfDigitsToRightOfCursor, newAmount))
            editText.addTextChangedListener(this)



        }

    })

你可能也需要这个功能

 private fun getNumberOfDigits(@NonNull text : String) : Int{
    var count = 0
    for (i in text)
        if (i.isDigit())
            count++
    return count
}

参考:FULL Code Description

 <EditText
    android:id="@+id/user_input_first_name"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:focusable="true"
    android:focusableInTouchMode="true"
    android:maxLines="1"
    android:textAllCaps="true"
    android:textColor="@color/black"
    android:textSize="20sp"
    android:textStyle="bold" />

对我来说,当我删除 textAllCaps、focusable 和 focusableInTouchMode 时问题就解决了。