解决RcyclerView嵌套EditText时,产生的焦点冲突,导致RecyclerView自动滑动问题

解决方法一:

在RecyclerView中设置android:descendantFocusability属性

  android:descendantFocusability="blocksDescendants"

属性介绍

1、afterDescendants:子控件优先获得焦点,当没有任何一个子控件获取到焦点时(即子控件不需要获取焦点时),父控件才会获取到焦点;
2、beforeDescendants :父控件会优先于子控件而获取到焦点;
3、blockDescendants:父控件会阻塞子控件获取焦点,即父控件会覆盖子类控件而直接获得焦点(只有父控件获得焦点);

解决方法二:

自定义layoutManager

/**
 * 为了解决recycle人View上添加的headView 中的EditText等控件获取了焦点导致RecyclerView 莫名滚动
 */
public class FocusLinearLayoutManager extends LinearLayoutManager {

    public FocusLinearLayoutManager(Context context) {
        super(context);
    }

    public FocusLinearLayoutManager(Context context, int orientation, boolean reverseLayout) {
        super(context, orientation, reverseLayout);
    }

    public FocusLinearLayoutManager(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
        super(context, attrs, defStyleAttr, defStyleRes);
    }


    /**
     *   public boolean requestChildRectangleOnScreen (View child, Rect rectangle, boolean immediate)
     * <p>
     *   当组里的某个子视图需要被定位在屏幕的某个矩形范围时,调用此方法。重载此方法的ViewGroup可确认以下几点:
     * <p>
     *   * 子项目将是组里的直系子项
     *   * 矩形将在子项目的坐标体系中
     *   重载此方法的ViewGroup应该支持以下几点:
     *   * 若矩形已经是可见的,则没有东西会改变
     *   * 为使矩形区域全部可见,视图将可以被滚动显示
     *   参数
     *   child        发出请求的子视图
     *   rectangle    子项目坐标系内的矩形,即此子项目希望在屏幕上的定位
     *   immediate   设为true,则禁止动画和平滑移动滚动条
     * <p>
     *   返回值
     *   进行了滚动操作的这个组(group),是否处理此操作。
     *
     * @param parent
     * @param child
     * @param rect
     * @param immediate
     * @return
     */
    @Override
    public boolean requestChildRectangleOnScreen(RecyclerView parent, View child, Rect rect, boolean immediate) {

        //这里的child 是整个HeadView 而不是某个具体的editText
        LogUtils.e("requestChildRectangleOnScreen()====> chlild==", child.getId() + "parent==" + parent.getId());
        return false;
    }

    @Override
    public boolean requestChildRectangleOnScreen(RecyclerView parent, View child, Rect rect, boolean immediate, boolean focusedChildVisible) {

        //这里的child 是整个HeadView 而不是某个具体的editText
        LogUtils.e("requestChildRectangleOnScreen( focusedChildVisible=)====> chlild==", child.getId() + "parent==" + parent.getId());
        return false;
    }
}

参考:

https://www.jianshu.com/p/827fd1195cc6
https://blog.csdn.net/okg0111/article/details/79886421

  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
你可以在RecyclerView的Adapter中为每个EditText设置FocusChange监听器,然后在监听器中处理焦点变化的逻辑。例如,当EditText失去焦点,可以调用其clearFocus()方法来让其失去焦点。同,你需要在RecyclerView的Item布局中添加一个可获取焦点的控件,例如一个Button,当用户点击这个Button,你可以调用RecyclerView的requestFocus()方法来让RecyclerView获取焦点,这样所有的EditText就会失去焦点了。 以下是一个简单的示例代码: ```java public class MyAdapter extends RecyclerView.Adapter<MyAdapter.ViewHolder> { private List<String> mData; public MyAdapter(List<String> data) { mData = data; } @NonNull @Override public ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.item_layout, parent, false); return new ViewHolder(view); } @Override public void onBindViewHolder(@NonNull ViewHolder holder, int position) { holder.mEditText.setText(mData.get(position)); holder.mEditText.setOnFocusChangeListener(new View.OnFocusChangeListener() { @Override public void onFocusChange(View v, boolean hasFocus) { if (!hasFocus) { holder.mEditText.clearFocus(); } } }); holder.mButton.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { holder.itemView.requestFocus(); } }); } @Override public int getItemCount() { return mData.size(); } public static class ViewHolder extends RecyclerView.ViewHolder { public EditText mEditText; public Button mButton; public ViewHolder(@NonNull View itemView) { super(itemView); mEditText = itemView.findViewById(R.id.edit_text); mButton = itemView.findViewById(R.id.button); } } } ``` 在上述代码中,我们为每个EditText设置了FocusChange监听器,并在监听器中处理了焦点变化的逻辑。同,在Item布局中添加了一个Button,并为其设置了点击事件,当用户点击这个Button,我们调用RecyclerView的requestFocus()方法来让RecyclerView获取焦点,从而让所有的EditText失去焦点
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值