记录如何实现两个recycleview同步滚动

前言

两个recycleview同步滚动网上solo的几种方法如下:

作者: Jorge Antonio Díaz-Benito

我相信你了解它的工作原理是相关的,所以我将解释我设计解决方案时所遵循的整个过程。请注意,此示例仅适用于两个RecyclerViews,但使用更多的RecyclerViews就像使用RecyclerViews数组一样简单。

想到的第一个选项是在两个ScrollViews上监听滚动更改,当其中一个滚动时,在另一个上使用scrollBy(int x,int y))。不幸的是,以编程方式滚动也会触发侦听器,因此您将最终处于循环中。

要解决此问题,您需要设置OnItemTouchListener,在触摸RecyclerView时添加正确的ScrollListener,并在滚动停止时将其删除。这几乎完美无缺,但是如果你在一个很长的RecyclerView中快速执行,然后在它完成之前再次滚动它,只会传输第一个滚动。

要解决此问题,您需要确保仅在RecyclerView空闲时添加OnScrollListener。

我们来看看来源:

    public class SelfRemovingOnScrollListener extends RecyclerView.OnScrollListener {

    @Override
    public final void onScrollStateChanged(@NonNull final RecyclerView recyclerView, final int newState) {
        super.onScrollStateChanged(recyclerView, newState);
        if (newState == RecyclerView.SCROLL_STATE_IDLE) {
            recyclerView.removeOnScrollListener(this);
        }
    }
}

这是您需要扩展OnScrollListeners的类。这可确保在需要时将其移除。

然后我有两个监听器,每个RecyclerView一个:

private final RecyclerView.OnScrollListener mLeftOSL = new SelfRemovingOnScrollListener() {
    @Override
    public void onScrolled(@NonNull final RecyclerView recyclerView, final int dx, final int dy) {
        super.onScrolled(recyclerView, dx, dy);
        mRightRecyclerView.scrollBy(dx, dy);
    }
}, mRightOSL = new SelfRemovingOnScrollListener() {

    @Override
    public void onScrolled(@NonNull final RecyclerView recyclerView, final int dx, final int dy) {
        super.onScrolled(recyclerView, dx, dy);
        mLeftRecyclerView.scrollBy(dx, dy);
    }
};

然后在初始化时,您可以设置OnItemTouchListeners。最好为整个视图设置一个监听器,但RecyclerView不支持这个。OnItemTouchListeners无论如何都不会造成问题:

    mLeftRecyclerView.addOnItemTouchListener(new RecyclerView.OnItemTouchListener() {

        private int mLastY;

        @Override
        public boolean onInterceptTouchEvent(@NonNull final RecyclerView rv, @NonNull final
        MotionEvent e) {
            Log.d("debug", "LEFT: onInterceptTouchEvent");

            final Boolean ret = rv.getScrollState() != RecyclerView.SCROLL_STATE_IDLE;
            if (!ret) {
                onTouchEvent(rv, e);
            }
            return Boolean.FALSE;
        }

        @Override
        public void onTouchEvent(@NonNull final RecyclerView rv, @NonNull final MotionEvent e) {
            Log.d("debug", "LEFT: onTouchEvent");

            final int action;
            if ((action = e.getAction()) == MotionEvent.ACTION_DOWN && mRightRecyclerView
                    .getScrollState() == RecyclerView.SCROLL_STATE_IDLE) {
                mLastY = rv.getScrollY();
                rv.addOnScrollListener(mLeftOSL);
            }
            else {
                if (action == MotionEvent.ACTION_UP && rv.getScrollY() == mLastY) {
                    rv.removeOnScrollListener(mLeftOSL);
                }
            }
        }

        @Override
        public void onRequestDisallowInterceptTouchEvent(final boolean disallowIntercept) {
            Log.d("debug", "LEFT: onRequestDisallowInterceptTouchEvent");
        }
    });

    mRightRecyclerView.addOnItemTouchListener(new RecyclerView.OnItemTouchListener() {

        private int mLastY;

        @Override
        public boolean onInterceptTouchEvent(@NonNull final RecyclerView rv, @NonNull final
        MotionEvent e) {
            Log.d("debug", "RIGHT: onInterceptTouchEvent");

            final Boolean ret = rv.getScrollState() != RecyclerView.SCROLL_STATE_IDLE;
            if (!ret) {
                onTouchEvent(rv, e);
            }
            return Boolean.FALSE;
        }

        @Override
        public void onTouchEvent(@NonNull final RecyclerView rv, @NonNull final MotionEvent e) {
            Log.d("debug", "RIGHT: onTouchEvent");

            final int action;
            if ((action = e.getAction()) == MotionEvent.ACTION_DOWN && mLeftRecyclerView
                    .getScrollState
                            () == RecyclerView.SCROLL_STATE_IDLE) {
                mLastY = rv.getScrollY();
                rv.addOnScrollListener(mRightOSL);
            }
            else {
                if (action == MotionEvent.ACTION_UP && rv.getScrollY() == mLastY) {
                    rv.removeOnScrollListener(mRightOSL);
                }
            }
        }

        @Override
        public void onRequestDisallowInterceptTouchEvent(final boolean disallowIntercept) {
            Log.d("debug", "RIGHT: onRequestDisallowInterceptTouchEvent");
        }
    });
}

另请注意,在我的特定情况下,RecyclerViews不是第一个接收触摸事件的人,所以我需要拦截它。如果不是这种情况,您可以(应该)将onInterceptTouchEvent(…)中的代码合并到onTouchEvent(…)中。

最后,如果您的用户尝试同时滚动两个RecyclerViews,这将导致崩溃。这里可能的最佳努力质量解决方案是android:splitMotionEvents="false"在包含RecyclerViews的直接父级中进行设置。

您可以在此处查看此代码的示例。

作者: Koen Van Looveren

我想我找到了一个非常容易和简短的答案。

正如JorgeAntonioDíaz-Benito所说:“第一个想到的选择是在ScrollViews上监听滚动更改,当其中一个滚动时,在另一个上使用scrollBy(int x,int y)。不幸的是,以编程方式滚动将也会触发听众,所以你最终会陷入一个循环。“

所以你需要解决这个问题。如果您只是跟踪谁在滚动它们将不会循环。

public class SelfScrolListener extends RecyclerView.OnScrollListener {
    @Override
    public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
        super.onScrollStateChanged(recyclerView, newState);
        if (newState == RecyclerView.SCROLL_STATE_IDLE) {
            viewIsScrolling = -1;
        }
    }
}

这是您自定义的OnScrollListener,用于检查scrollState是否为IDLE。这是真的 - >没有人滚动。所以`int viewIsScolling = -1

现在您需要检测是否可以滚动。这是代码:

int viewIsScrolling = 1;
boolean firstIsTouched = false;
boolean secondIsTouched = false;
SelfScrolListener firstOSL= new SelfScrolListener() {
    @Override
    public void onScrolled(@NonNull final RecyclerView recyclerView, final int dx, final int dy) {
        super.onScrolled(recyclerView, dx, dy);

        if (firstIsTouched) {
             if (viewIsScrolling == -1) {
                 viewIsScrolling = 0;
             }
             if (viewIsScrolling == 0) {
                 secondRecyclerView.scrollBy(dx, dy);
             }
        }
    }
};
SelfScrolListener secondOSL= new SelfScrolListener() {
    @Override
    public void onScrolled(@NonNull final RecyclerView recyclerView, final int dx, final int dy) {
        super.onScrolled(recyclerView, dx, dy);
        if(secondIsTouched){
             if (viewIsScrolling == -1) {
                 viewIsScrolling = 1;
             }
             if (viewIsScrolling == 1) {
                 firstRecyclerView.scrollBy(dx, dy);
             }
        }
    }
};
firstRecyclerView.setOnTouchListener(new View.OnTouchListener() {
    @Override
    public boolean onTouch(View v, MotionEvent event) {
        firstIsTouched= true;
        return false;
    }
});
secondRecyclerView.setOnTouchListener(new View.OnTouchListener() {
    @Override
    public boolean onTouch(View v, MotionEvent event) {
        secondIsTouched= true;
        return false;
    }
});
firstRecyclerView.addOnScrollListener(firstOSL);
secondRecyclerView.addOnScrollListener(secondOSL);

viewIsScrolling =一个全局int并在开头设置为-1; 没有人滚动的状态。您可以根据需要添加尽可能多的回收视图。

作者:Aace

这是我的解决方案。代码越少越好……

lvDetail和lvDetail2是您希望保持同步的RecyclerView。

    final RecyclerView.OnScrollListener[] scrollListeners = new RecyclerView.OnScrollListener[2];
    scrollListeners[0] = new RecyclerView.OnScrollListener( )
    {
        @Override
        public void onScrolled(RecyclerView recyclerView, int dx, int dy)
        {
            super.onScrolled(recyclerView, dx, dy);
            lvDetail2.removeOnScrollListener(scrollListeners[1]);
            lvDetail2.scrollBy(dx, dy);
            lvDetail2.addOnScrollListener(scrollListeners[1]);
        }
    };
    scrollListeners[1] = new RecyclerView.OnScrollListener( )
    {
        @Override
        public void onScrolled(RecyclerView recyclerView, int dx, int dy)
        {
            super.onScrolled(recyclerView, dx, dy);
            lvDetail.removeOnScrollListener(scrollListeners[0]);
            lvDetail.scrollBy(dx, dy);
            lvDetail.addOnScrollListener(scrollListeners[0]);
        }
    };
    lvDetail.addOnScrollListener(scrollListeners[0]);
    lvDetail2.addOnScrollListener(scrollListeners[1]);

总结

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout 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="@color/main_bj"
    android:orientation="vertical">


    <View
        android:layout_width="match_parent"
        android:layout_height="@dimen/px_1"
        android:background="@color/c_line"></View>

    <RelativeLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:background="@color/white">


        <RelativeLayout
            android:id="@+id/rl_table_title"
            android:layout_width="@dimen/dp_80"
            android:layout_height="@dimen/dp_35">

            <TextView
                android:id="@+id/tv_table_title"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_centerInParent="true"
                android:text="小区号"
                android:textColor="@color/c_33"
                android:textSize="@dimen/sp_12" />

            <View
                android:layout_width="match_parent"
                android:layout_height="@dimen/px_1"
                android:layout_alignParentBottom="true"
                android:background="@color/c_line"></View>

            <View
                android:layout_width="@dimen/px_1"
                android:layout_height="match_parent"
                android:layout_alignParentRight="true"
                android:background="@color/c_line"></View>

        </RelativeLayout>

        <com.kanghanbin.mytable.MyHorizontalScrollView
            android:id="@+id/mhsc_row"
            android:layout_width="match_parent"
            android:layout_height="@dimen/dp_35"
            android:overScrollMode="never"
            android:scrollbars="none"
            android:layout_toRightOf="@+id/rl_table_title">


            <LinearLayout
                android:id="@+id/ll_row_content"
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                android:orientation="horizontal">

            </LinearLayout>

        </com.kanghanbin.mytable.MyHorizontalScrollView>


        <android.support.v7.widget.RecyclerView
            android:scrollbars="none"
            android:id="@+id/lv_column"
            android:layout_width="@dimen/dp_80"
            android:layout_height="match_parent"
            android:layout_below="@+id/rl_table_title"
            android:divider="@null"></android.support.v7.widget.RecyclerView>


            <com.kanghanbin.mytable.MyHorizontalScrollView
                android:id="@+id/mhsc_content"
                android:layout_width="match_parent"
                android:overScrollMode="never"
                android:layout_below="@+id/rl_table_title"
                android:layout_toRightOf="@+id/lv_column"
                android:layout_height="match_parent">

                <android.support.v7.widget.RecyclerView
                    android:scrollbars="none"
                    android:id="@+id/lv_content"
                    android:layout_width="match_parent"
                    android:layout_height="match_parent"
                    android:divider="@null"></android.support.v7.widget.RecyclerView>

            </com.kanghanbin.mytable.MyHorizontalScrollView>

    </RelativeLayout>

</LinearLayout>


刚开始用第一种方法,但是由于我的RecyclerView外面套了个HorizontalScrollView,所以在事件分发上会有问题,所以我用的是第三种方法实现了最终效果。

  • 6
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
实现左右联动的方式有很多种,其中一种常用的方法是利用 `RecyclerView` 的滑动监听来实现。 首先,需要在布局文件中定义两个 `RecyclerView`,分别表示左侧和右侧的列表,如下所示: ```xml <LinearLayout android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="horizontal"> <androidx.recyclerview.widget.RecyclerView android:id="@+id/left_rv" android:layout_width="0dp" android:layout_height="match_parent" android:layout_weight="1"/> <androidx.recyclerview.widget.RecyclerView android:id="@+id/right_rv" android:layout_width="0dp" android:layout_height="match_parent" android:layout_weight="1"/> </LinearLayout> ``` 接下来,在代码中分别获取这两个 `RecyclerView`,并设置它们的布局管理器和适配器。 ```java RecyclerView leftRv = findViewById(R.id.left_rv); RecyclerView rightRv = findViewById(R.id.right_rv); // 设置布局管理器 leftRv.setLayoutManager(new LinearLayoutManager(this)); rightRv.setLayoutManager(new LinearLayoutManager(this)); // 设置适配器 leftRv.setAdapter(leftAdapter); rightRv.setAdapter(rightAdapter); ``` 接着,在左侧列表的滑动监听中获取当前可见的第一个 item,然后将右侧列表滚动到对应的位置。具体实现如下: ```java leftRv.addOnScrollListener(new RecyclerView.OnScrollListener() { @Override public void onScrolled(@NonNull RecyclerView recyclerView, int dx, int dy) { super.onScrolled(recyclerView, dx, dy); int firstVisibleItemPosition = ((LinearLayoutManager) leftRv.getLayoutManager()).findFirstVisibleItemPosition(); rightRv.scrollToPosition(firstVisibleItemPosition); } }); ``` 这样,当左侧列表滑动时,右侧列表也会跟着滑动,实现了左右联动的效果。需要注意的是,这种实现方式只适用于两个列表的 item 数量相同的情况,如果两个列表的 item 数量不同,则需要进行一些额外的处理。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值