将适当的键盘支持添加到RecyclerView

15
在我之前的问题中,我问了一个问题(并且按照这篇博客文章的方法自己回答了),如何在RecyclerView上正确处理关键输入。现在我意识到,如果我按住箭头键,比如向下键,向下滚动会停止,RecyclerView会失去焦点,可能是因为滚动速度比所有子View的生成速度更快。有没有什么解决方法或更好的做法来正确处理RecyclerView上的硬件键盘输入?更新:我发布了一个基本示例在这里,现在它可以完美地工作,不再失去焦点了。

2
你有一个最小可复现的例子吗?昨天我刚刚为RecyclerView添加了键盘支持,就像你所做的那样。在我的情况下,它是一个全屏的RecyclerView,所以我没有任何焦点问题。如果我们有一个具体的问题实现,人们将更容易帮助你解决问题。 - CommonsWare
@CommonsWare 我会尽快创建我的示例。我认为您可以尝试在RecyclerView下添加一个可聚焦的视图(例如按钮),并在适配器中使用大量数据集,比如100项。您应该会看到焦点的行为很奇怪。 - Vektor88
明天我会更深入地研究这个问题。我看到了其他奇怪的结果(例如,模糊效果)。我认为也许我们需要等待第一次处理完成后再进行第二次RecyclerView更新,而不是每个按键事件都更新一次。话虽如此,在垂直滚动列表下方放置一个可聚焦视图本身就不是一个好的键盘用户体验。您将在编写Android TV应用程序(或其Google TV前身,Fire TV等)的相关报道中看到有关这类事情的讨论。 - CommonsWare
@CommonsWare 我可能已经自己解决了这个问题,我从头开始编码,并得到了这个工作示例 https://github.com/zevektor/KeyboardRecyclerView - Vektor88
1个回答

5

我成功地实现了一个抽象的适配器类,能够跟踪所选项目而不失去项目焦点。可以在这里找到一个示例项目,适配器类的具体实现如下:

import android.content.Context;
import android.support.v7.widget.RecyclerView;
import android.view.KeyEvent;
import android.view.View;


/**
 * Created by vektor on 31/05/16.
 */
public abstract class InputTrackingRecyclerViewAdapter<VH extends RecyclerView.ViewHolder> extends RecyclerView.Adapter<VH>{

    private Context mContext;
    private int mSelectedItem = 0;
    private RecyclerView mRecyclerView;

    public InputTrackingRecyclerViewAdapter(Context context){
        mContext = context;
    }

    @Override
    public void onAttachedToRecyclerView(final RecyclerView recyclerView) {
        super.onAttachedToRecyclerView(recyclerView);

        mRecyclerView = recyclerView;
        // Handle key up and key down and attempt to move selection
        recyclerView.setOnKeyListener(new View.OnKeyListener() {
            @Override
            public boolean onKey(View v, int keyCode, KeyEvent event) {
                RecyclerView.LayoutManager lm = recyclerView.getLayoutManager();

                // Return false if scrolled to the bounds and allow focus to move off the list
                if (event.getAction() == KeyEvent.ACTION_DOWN) {
                    if (isConfirmButton(event)) {
                        if ((event.getFlags() & KeyEvent.FLAG_LONG_PRESS) == KeyEvent.FLAG_LONG_PRESS) {
                            mRecyclerView.findViewHolderForAdapterPosition(mSelectedItem).itemView.performLongClick();
                        } else {
                            event.startTracking();
                        }
                        return true;
                    }
                    else {
                        if (keyCode == KeyEvent.KEYCODE_DPAD_DOWN) {
                            return tryMoveSelection(lm, 1);
                        } else if (keyCode == KeyEvent.KEYCODE_DPAD_UP) {
                            return tryMoveSelection(lm, -1);
                        }
                    }
                }
                else if(event.getAction() == KeyEvent.ACTION_UP && isConfirmButton(event)
                        && ((event.getFlags() & KeyEvent.FLAG_LONG_PRESS) != KeyEvent.FLAG_LONG_PRESS)){
                    mRecyclerView.findViewHolderForAdapterPosition(mSelectedItem).itemView.performClick();
                    return true;
                }
                return false;
            }
        });
    }

    private boolean tryMoveSelection(RecyclerView.LayoutManager lm, int direction) {
        int nextSelectItem = mSelectedItem + direction;

        // If still within valid bounds, move the selection, notify to redraw, and scroll
        if (nextSelectItem >= 0 && nextSelectItem < getItemCount()) {
            notifyItemChanged(mSelectedItem);
            mSelectedItem = nextSelectItem;
            notifyItemChanged(mSelectedItem);
            //lm.scrollToPosition(mSelectedItem);
            mRecyclerView.smoothScrollToPosition(mSelectedItem);
            return true;
        }

        return false;
    }

    public Context getContext(){ return mContext; }

    public int getSelectedItem() { return mSelectedItem; }
    public void setSelectedItem(int selectedItem) { mSelectedItem = selectedItem; }

    public RecyclerView getRecyclerView() { return mRecyclerView; }


    @Override
    public void onBindViewHolder(VH holder, int position) {
        onBindViewHolder(holder, position);
    }

    public static boolean isConfirmButton(KeyEvent event){
        switch (event.getKeyCode()){
            case KeyEvent.KEYCODE_ENTER:
            case KeyEvent.KEYCODE_DPAD_CENTER:
            case KeyEvent.KEYCODE_BUTTON_A:
                return true;
            default:
                return false;
        }
    }


}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接