同步滚动多个RecyclerViews

ger*_*ler 15 android android-recyclerview

我有一个ViewPager每页显示一个片段.此片段包含RecyclerView中的项目列表.项目列表始终具有相同的大小,并且项目的视图也具有相同的高度.当滚动其中一个RecyclerViews时,我希望其他RecyclerViews同时滚动相同的距离.如何同步RecyclerViews的滚动?

Aac*_*ace 20

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

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]);
Run Code Online (Sandbox Code Playgroud)

  • 有时这两者会不同步(因为我们依赖“scrollBy”)。然后怎样呢? (2认同)
  • 如果你扔掉其中一辆房车,它们就会失去同步。 (2认同)

Jor*_*ito 16

我相信你了解它的工作原理是相关的,所以我将解释我设计解决方案时所遵循的整个过程.请注意,此示例仅适用于两个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);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

这是您需要扩展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);
    }
};
Run Code Online (Sandbox Code Playgroud)

然后在初始化时,您可以设置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");
        }
    });
}
Run Code Online (Sandbox Code Playgroud)

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

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

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


Koe*_*ren 5

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

正如 Jorge Antonio Dí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;
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

这是您的自定义 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);
Run Code Online (Sandbox Code Playgroud)

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


Har*_*shi 0

在寻呼机适配器中创建变量,该变量将负责保存scrollY其页面的当前位置,并且每当您getItem(position)调用更新位置时,请 ListView查看https://github.com/ksoichiro/Android-ObservableScrollView/blob/master/samples/src/ main/java/com/github/ksoichiro/android/observablescrollview/samples/ViewPagerTabActivity.javaCacheFragmentStatePagerAdapter