Android ListView 的 OnItemClickListener 无法在不可聚焦的 PopupWindow 上工作
Android ListView's OnItemClickListener not working on non-focusable PopupWindow
在我的 Android 应用程序中,我有一个 EditText
。当聚焦 EditText
时,会显示一个 PopupWindow
,其中包含一个 ListView
。当 PopupWindow
出现在屏幕上时,用户可以继续输入 EditText
,同时 he/she 应该能够单击 ListView
上的项目以关闭PopupWindow
。 (取决于点击了哪个 ListView
项目,除了关闭 PopupWindow
之外还应该发生其他事情,但在这里无关紧要。)
问题是,尽管 "continue typing" 部分有效,但 "click an item" 部分无效,因为 ListView
的 OnItemClickListener
永远不会被调用,而且我想不通为什么。
这是我的 activity_main.xml
:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#ddd"
android:focusableInTouchMode="true"
tools:context="com.example.popuptest3.MainActivity" >
<EditText
android:id="@+id/edtPhone"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:inputType="phone" />
</RelativeLayout>
我的popup_window.xml
:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#fff" >
<ListView
android:id="@+id/lstItems"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_alignParentBottom="true"
android:layout_marginBottom="60dp" />
<Button
android:id="@+id/btnDismiss"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentBottom="true"
android:layout_centerHorizontal="true"
android:text="Dismiss" />
</RelativeLayout>
我的list_item.xml
:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="50dp" >
<TextView
android:id="@+id/txtItem"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:gravity="center_vertical"
android:text="Item Text"
android:textAppearance="?android:attr/textAppearanceMedium" />
</RelativeLayout>
在我的MainActivity
class中我有以下准备代码:
private PopupWindow popup = null;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
prepareControls();
}
private void prepareControls() {
final EditText edtPhone = (EditText)findViewById(R.id.edtPhone);
edtPhone.setOnFocusChangeListener(new OnFocusChangeListener() {
@Override
public void onFocusChange(View v, boolean hasFocus) {
if (hasFocus)
openPopup(edtPhone);
else
closePopup();
}
});
}
也就是我在edtPhone
获得焦点时打开PopupWindow
,在edtPhone
失去焦点时关闭PopupWindow
。然后我有:
private void openPopup(View parent) {
closePopup();
// Inflate the view of the PopupWindow.
LayoutInflater layoutInflater = LayoutInflater.from(this);
View view = layoutInflater.inflate(R.layout.popup_window,
new LinearLayout(this), false);
final Activity activity = this;
// Prepare the ListView.
ListView lstItems = (ListView)view.findViewById(R.id.lstItems);
lstItems.setAdapter(new ListAdapter(this));
lstItems.setOnItemClickListener(new OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view,
int position, long id) {
hideKeyboardAndClearFocus(activity);
}
});
// Prepare the Button.
Button btnDismiss = (Button)view.findViewById(R.id.btnDismiss);
btnDismiss.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
hideKeyboardAndClearFocus(activity);
}
});
// Create and show the PopupWindow.
popup = new PopupWindow(view, 400, 300, false);
popup.showAtLocation(parent, Gravity.LEFT | Gravity.TOP, 0, 60);
}
btnDismiss
按钮不在我的计划之内;我在这里添加它只是为了演示目的。最后我有:
private void closePopup() {
if (popup != null) {
popup.dismiss();
popup = null;
}
}
private static void hideKeyboardAndClearFocus(Activity activity) {
InputMethodManager manager = (InputMethodManager)activity.
getSystemService(Activity.INPUT_METHOD_SERVICE);
manager.hideSoftInputFromWindow(
activity.getCurrentFocus().getWindowToken(), 0);
activity.getWindow().getDecorView().clearFocus();
}
这个我觉得不用解释了,ListAdapter
class 很标准所以就省略了。
现在,当我单击 edtPhone
时,它获得焦点,软键盘出现,PopupWindow
打开。我可以在 edtPhone
中键入内容,当我单击 btnDismiss
时,edtPhone
失去焦点,软键盘消失,PopupWindow
关闭,一切如预期。但是,如果我点击 lstItems
上的某个项目而不是 btnDismiss
,则什么也不会发生。使用 Log
我可以看到 OnItemClickListener
没有被调用。搞笑的是,ListView
可以很好的滚动,就是不能点击
我知道 this post 但该解决方案不适合我的目的。如果我更改此行:
popup = new PopupWindow(view, 400, 300, false);
以下内容:
popup = new PopupWindow(view, 400, 300, true);
PopupWindow
变得可聚焦,ListView
变得可点击,好吧,但我无法继续输入 edtPhone
。事实上,如果activity上有其他控件,它们都变得无法访问,就好像PopupWindow
是"modal"。
我也试过android:focusable="false"
,没有成功。而且我没有进行自动完成,所以 AutoCompleteTextView
不是我需要的。
那么为什么 OnItemClickListener
没有在不可聚焦的 PopupWindow
上调用?我做错了什么吗?
将弹出窗口设置为不可聚焦
popupwindow.setFocusable(false);
popupwindow.setTouchable(true);
popupwindow.setOutsideTouchable(true);
扩展 ListView 并覆盖以下方法
@Override
public boolean hasFocus() {
return true;
}
@Override
public boolean isFocused() {
return true;
}
@Override
public boolean hasWindowFocus() {
return true;
}
这就是 AutoCompleteTextView 实现建议弹出窗口的方式。
如果您需要 select 使用 DPAD 键或其他导航键的列表项,则需要将按键事件从 EditText 传递到 ListView。
我通过在 convertview
上设置一个 OnClickListener
解决了这个问题,这是适配器的 getView
方法的一个参数。然后我调用列表视图的 onItemClickListener 的 onItemClick
方法。
无需使弹出窗口可聚焦。 事实上,只使用 ListPopupWindow 更容易,因为它只在 getView()
适配器方法中,我们正在添加几行代码
ListPopupWindow popupWindow = new ListPopupWindow(mContext);
MentionsAdapter adapter = new MentionsAdapter(mContext, suggestions);
popupWindow.setOnItemClickListener(new android.widget.AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
String username = mSuggestions.get(position);
String mentionCompletion = username.substring(mCurrentQuery.length()).concat(" ");
mEditor.getText().insert(mEditor.getSelectionEnd(), mentionCompletion);
hideSuggestions();
}
});
popupWindow.setAdapter(adapter);
MentionsAdapter
getView
方法
@Override
public View getView(int position, View convertView, ViewGroup parent) {
if(convertView == null){
convertView = LayoutInflater.from(context).inflate(,R.layout.item_user,parent,false);
}
// this is the key point of workaround
convertView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
popupWindow.getListView().getOnItemClickListener()
.onItemClick(listView, v, position, getItemId(position));
}
});
return convertView;
}
在我的 Android 应用程序中,我有一个 EditText
。当聚焦 EditText
时,会显示一个 PopupWindow
,其中包含一个 ListView
。当 PopupWindow
出现在屏幕上时,用户可以继续输入 EditText
,同时 he/she 应该能够单击 ListView
上的项目以关闭PopupWindow
。 (取决于点击了哪个 ListView
项目,除了关闭 PopupWindow
之外还应该发生其他事情,但在这里无关紧要。)
问题是,尽管 "continue typing" 部分有效,但 "click an item" 部分无效,因为 ListView
的 OnItemClickListener
永远不会被调用,而且我想不通为什么。
这是我的 activity_main.xml
:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#ddd"
android:focusableInTouchMode="true"
tools:context="com.example.popuptest3.MainActivity" >
<EditText
android:id="@+id/edtPhone"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:inputType="phone" />
</RelativeLayout>
我的popup_window.xml
:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#fff" >
<ListView
android:id="@+id/lstItems"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_alignParentBottom="true"
android:layout_marginBottom="60dp" />
<Button
android:id="@+id/btnDismiss"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentBottom="true"
android:layout_centerHorizontal="true"
android:text="Dismiss" />
</RelativeLayout>
我的list_item.xml
:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="50dp" >
<TextView
android:id="@+id/txtItem"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:gravity="center_vertical"
android:text="Item Text"
android:textAppearance="?android:attr/textAppearanceMedium" />
</RelativeLayout>
在我的MainActivity
class中我有以下准备代码:
private PopupWindow popup = null;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
prepareControls();
}
private void prepareControls() {
final EditText edtPhone = (EditText)findViewById(R.id.edtPhone);
edtPhone.setOnFocusChangeListener(new OnFocusChangeListener() {
@Override
public void onFocusChange(View v, boolean hasFocus) {
if (hasFocus)
openPopup(edtPhone);
else
closePopup();
}
});
}
也就是我在edtPhone
获得焦点时打开PopupWindow
,在edtPhone
失去焦点时关闭PopupWindow
。然后我有:
private void openPopup(View parent) {
closePopup();
// Inflate the view of the PopupWindow.
LayoutInflater layoutInflater = LayoutInflater.from(this);
View view = layoutInflater.inflate(R.layout.popup_window,
new LinearLayout(this), false);
final Activity activity = this;
// Prepare the ListView.
ListView lstItems = (ListView)view.findViewById(R.id.lstItems);
lstItems.setAdapter(new ListAdapter(this));
lstItems.setOnItemClickListener(new OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view,
int position, long id) {
hideKeyboardAndClearFocus(activity);
}
});
// Prepare the Button.
Button btnDismiss = (Button)view.findViewById(R.id.btnDismiss);
btnDismiss.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
hideKeyboardAndClearFocus(activity);
}
});
// Create and show the PopupWindow.
popup = new PopupWindow(view, 400, 300, false);
popup.showAtLocation(parent, Gravity.LEFT | Gravity.TOP, 0, 60);
}
btnDismiss
按钮不在我的计划之内;我在这里添加它只是为了演示目的。最后我有:
private void closePopup() {
if (popup != null) {
popup.dismiss();
popup = null;
}
}
private static void hideKeyboardAndClearFocus(Activity activity) {
InputMethodManager manager = (InputMethodManager)activity.
getSystemService(Activity.INPUT_METHOD_SERVICE);
manager.hideSoftInputFromWindow(
activity.getCurrentFocus().getWindowToken(), 0);
activity.getWindow().getDecorView().clearFocus();
}
这个我觉得不用解释了,ListAdapter
class 很标准所以就省略了。
现在,当我单击 edtPhone
时,它获得焦点,软键盘出现,PopupWindow
打开。我可以在 edtPhone
中键入内容,当我单击 btnDismiss
时,edtPhone
失去焦点,软键盘消失,PopupWindow
关闭,一切如预期。但是,如果我点击 lstItems
上的某个项目而不是 btnDismiss
,则什么也不会发生。使用 Log
我可以看到 OnItemClickListener
没有被调用。搞笑的是,ListView
可以很好的滚动,就是不能点击
我知道 this post 但该解决方案不适合我的目的。如果我更改此行:
popup = new PopupWindow(view, 400, 300, false);
以下内容:
popup = new PopupWindow(view, 400, 300, true);
PopupWindow
变得可聚焦,ListView
变得可点击,好吧,但我无法继续输入 edtPhone
。事实上,如果activity上有其他控件,它们都变得无法访问,就好像PopupWindow
是"modal"。
我也试过android:focusable="false"
,没有成功。而且我没有进行自动完成,所以 AutoCompleteTextView
不是我需要的。
那么为什么 OnItemClickListener
没有在不可聚焦的 PopupWindow
上调用?我做错了什么吗?
将弹出窗口设置为不可聚焦
popupwindow.setFocusable(false);
popupwindow.setTouchable(true);
popupwindow.setOutsideTouchable(true);
扩展 ListView 并覆盖以下方法
@Override
public boolean hasFocus() {
return true;
}
@Override
public boolean isFocused() {
return true;
}
@Override
public boolean hasWindowFocus() {
return true;
}
这就是 AutoCompleteTextView 实现建议弹出窗口的方式。 如果您需要 select 使用 DPAD 键或其他导航键的列表项,则需要将按键事件从 EditText 传递到 ListView。
我通过在 convertview
上设置一个 OnClickListener
解决了这个问题,这是适配器的 getView
方法的一个参数。然后我调用列表视图的 onItemClickListener 的 onItemClick
方法。
无需使弹出窗口可聚焦。 事实上,只使用 ListPopupWindow 更容易,因为它只在 getView()
适配器方法中,我们正在添加几行代码
ListPopupWindow popupWindow = new ListPopupWindow(mContext);
MentionsAdapter adapter = new MentionsAdapter(mContext, suggestions);
popupWindow.setOnItemClickListener(new android.widget.AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
String username = mSuggestions.get(position);
String mentionCompletion = username.substring(mCurrentQuery.length()).concat(" ");
mEditor.getText().insert(mEditor.getSelectionEnd(), mentionCompletion);
hideSuggestions();
}
});
popupWindow.setAdapter(adapter);
MentionsAdapter
getView
方法
@Override
public View getView(int position, View convertView, ViewGroup parent) {
if(convertView == null){
convertView = LayoutInflater.from(context).inflate(,R.layout.item_user,parent,false);
}
// this is the key point of workaround
convertView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
popupWindow.getListView().getOnItemClickListener()
.onItemClick(listView, v, position, getItemId(position));
}
});
return convertView;
}