Эффект избыточной прокрутки не отображается в моем окне повторного просмотра внутри swiperefreshlayout
У меня есть фрагмент, который содержит SwipeRefreshLayour с RecyclerView внутри.
В этом обзоре не отображаются эффекты переполнения. Я пробовал много вещей, как использовать это:
android:isScrollContainer="true"
android:overScrollMode="always"
Пользовательский цвет через стили не работает
fadingEdge, fadingEdgeLengh, требует FadingEdge...
Swipe to refresh работает, и это показывает это:
Я не понимаю этого:
1 ответ
Я просто решил это, переписав SwipeRefreshLayout и используя его вместо реализации по умолчанию:
import android.content.Context
import android.util.AttributeSet
import android.view.View
import androidx.swiperefreshlayout.widget.SwipeRefreshLayout
class SwipeRefreshLayoutWithRecyclerViewOverscroll : SwipeRefreshLayout {
constructor(context: Context) : super(context)
constructor(context: Context, attrs: AttributeSet) : super(context, attrs)
override fun onStartNestedScroll(child: View, target: View, nestedScrollAxes: Int): Boolean =
(child.canScrollVertically(1) && super.onStartNestedScroll(child, target, nestedScrollAxes))
}
Я использую RecyclerView, который представляет собой линейный вертикальный список и child.canScrollVertically(1)
проверяет, находится ли этот список в самом низу. Если он находится внизу списка, onStartNestedScroll возвращает значение false, что позволяет RecyclerView обрабатывать подтягивающее движение пользователя, чтобы отображался эффект перезаписи.
ПРИМЕЧАНИЕ. Это показывает только нижнюю границу прокрутки, а не верхнюю, потому что я не хочу видеть анимацию подтягивания до обновления вместе с верхней границей прокрутки.
В моей реализации этой функции (избыточная прокрутка работает как сверху, так и снизу), у меня есть такой макет:
<RelativeLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<...<custom view>...SwipeRefreshWrapper
android:id="@+id/swiperefresh_layout"
android:layout_width="match_parent"
android:layout_height="match_parent">
<FrameLayout
android:layout_width="match_parent"
android:layout_height="match_parent">
<RecyclerView
android:id="@+id/video_recyclerview"
android:layout_width="match_parent"
android:layout_height="match_parent"/>
<include layout="@layout/include_video_list_empty_view"/>
<include layout="@layout/include_watchlist_empty_view"/>
</FrameLayout>
<SwipeRefreshWrapper>
И моя реализация SwipeRefreshWrapper выглядит следующим образом:
/**
* This Class ensures that a SwipeRefreshLayout will only refresh if the
* list it wraps cannot scroll up anymore, (that it is at the top). Otherwise
* when the user attempts to scroll up from the middle of the list, the refresh
* will trigger. This is necessary for custom List implementations, (or
* RecyclerViews), only.
*/
public class SwipeRefreshWrapper extends SwipeRefreshLayout {
private ScrollResolver mScrollResolver;
public SwipeRefreshWrapper(Context context) {
super(context);
}
public SwipeRefreshWrapper(Context context, AttributeSet attrs) {
super(context, attrs);
}
public void setScrollResolver(ScrollResolver scrollResolver) {
mScrollResolver = scrollResolver;
}
@Override
public boolean canChildScrollUp() {
if(mScrollResolver != null){
return mScrollResolver.canScrollUp();
} else {
return super.canChildScrollUp();
}
}
public static interface ScrollResolver{
public boolean canScrollUp();
}
}
Затем в своем фрагменте я устанавливаю SwipeRefreshLayout:
/**
* Initializes PullToRefresh handler
*/
private void initPullToRefresh() {
mSwipeRefreshLayout = (SwipeRefreshWrapper) mView.findViewById(R.id.swiperefresh_layout);
mSwipeRefreshLayout.setColorSchemeResources(R.color.colorPrimary, R.color.colorAccent);
mSwipeRefreshLayout.setScrollResolver(new SwipeRefreshWrapper.ScrollResolver() {
@Override
public boolean canScrollUp() {
return mVideoRecycler.canScrollVertically(-1);
}
});
mSwipeRefreshLayout.setOnRefreshListener(...)
}