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
}
<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 时问题就解决了。
我得到了以下 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
}
<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 时问题就解决了。