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" 部分无效,因为 ListViewOnItemClickListener 永远不会被调用,而且我想不通为什么。

这是我的 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>

在我的MainActivityclass中我有以下准备代码:

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);

MentionsAdaptergetView方法

@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;
    }