我知道之前已经问过this/similar question,但是给出的解决方案对我不起作用,所以我再问一次。 我尝试了该答案中给出的解决方案,但仍然在某些设备上调用我的OnKeyListener,尤其是那些具有库存操作系统的设备。当editText中没有字符时,我需要检测按下软键盘的del键。这是我的代码;
EditText et = (EditText) findViewById(R.id.et);
et.setOnKeyListener(new EditText.OnKeyListener() {
@Override
public boolean onKey(View v, int keyCode, KeyEvent event) {
Log.d("OnKeyListener", keyCode + " character(code) to send");
return false;
}
});
答案 0 :(得分:20)
最后通过TextWatcher
实现此功能来解决问题。主要的障碍是,即使EditText
中没有字符,或者至少最终用户认为那里没有字符,我也需要检测退格键。虽然我做了后者,但是第一件事无法实现。以下是详细的解决方案。
首先,我总是在editText
中保留一个空格'。
editText.addTextChangedListener(new TextWatcher() {
@Override
public void onTextChanged(CharSequence cs, int arg1, int arg2, int arg3) {
if(cs.toString().length() == 0)
editText.setText(" ");
}
@Override
public void beforeTextChanged(CharSequence arg0, int arg1, int arg2, int arg3) { }
@Override
public void afterTextChanged(Editable arg0) { }
});
然后我自定义EditText
以通知我每个光标位置的变化。通过覆盖onSelectionChanged
EditText
方法来实现此目的。我的自定义EditText
看起来像这样。
public class SelectionEnabledEditText extends EditText {
public SelectionEnabledEditText(Context context) {
super(context);
}
public SelectionEnabledEditText(Context context, AttributeSet attrs) {
super(context, attrs);
}
public SelectionEnabledEditText(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
}
@Override
protected void onSelectionChanged(int selStart, int selEnd) {
super.onSelectionChanged(selStart, selEnd);
if(onSelectionChangeListener != null)
onSelectionChangeListener.onSelectionChanged(selStart, selEnd);
}
public static interface OnSelectionChangeListener{
public void onSelectionChanged(int selStart, int selEnd);
}
private OnSelectionChangeListener onSelectionChangeListener;
public void setOnSelectionChangeListener(OnSelectionChangeListener onSelectionChangeListener) {
this.onSelectionChangeListener = onSelectionChangeListener;
}
}
最后,在我的活动中,我正在侦听光标位置改变事件并在editText
中重置我的光标位置,如果它在必要的空间charatcer开始,即在第0个索引处,就像这样;
editText.setOnSelectionChangeListener(new SelectionEnabledEditText.OnSelectionChangeListener() {
@Override
public void onSelectionChanged(int selStart, int selEnd) {
if (selEnd == 0) {
if (editText.getText().toString().length() == 0)
editText.setText(" ");
editText.setSelection(1);
}
}
});
希望这对类似情况有所帮助。欢迎提出改进/优化建议。
答案 1 :(得分:11)
文档指出,关键事件只会传播硬件击键,而不是软件。
http://developer.android.com/reference/android/view/View.OnKeyListener.html
设备制造商实际上不鼓励通过关键的听众传播软键盘事件,尽管完全由制造商来尊重或者用相同的术语来实际处理软键盘和硬键盘。
从Android 4.2.2开始,Android系统本身根本不支持软键盘的关键加油事件,因此即使是制造商也无法选择自己的方式。
所以这里唯一的万无一失的选择是实现你自己的IME(软键盘),并自己处理击键。
TextWatcher主要用于替换键侦听器,但是editText.setText(...);还会触发TextWatcher事件,所以如果只对一个键入的键感兴趣,那么TextWatcher可能也不是解决方案。
将TextWatcher与AutocomleteTextView或EditText一起使用时请务必小心。不要在TextWatcher事件中修改AutocompleteTextView / EditText内容中的文本,否则你很可能最终会进入无限事件/监听循环。
答案 2 :(得分:3)
基于OnKeyListener
的{{3}},似乎只为硬件键盘调用了回调。
将硬件键事件调度到此视图时要调用的回调的接口定义。在将键事件提供给视图之前将调用回调。这仅适用于硬件键盘;软件输入法没有义务触发这个监听器。
答案 3 :(得分:0)
EditText有一个属性:android:imeOptions
<androidx.appcompat.widget.AppCompatEditText
android:id="@+id/main_editor"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:imeOptions="actionGo"
/>
我注意到,对于值:actionGo
,actionNone
,normal
,actionSearch
,已调用onKeyListener。
P.S。甚至没有指定该属性就可以工作。
答案 4 :(得分:-3)
尝试这样的事情:
public boolean onKey(View v, int keyCode, KeyEvent event) {
if (keyCode == KeyEvent.KEYCODE_DEL) {
//Log key
Log.d("OnKeyListener", keyCode + " character(code) to send");
}
return false;
}