Android: реализовать бесконечный сколлинг в представлении Recycler

URL моего API: https://zappos.amazon.com/mobileapi/v1/search?term=adidas&page=1
Страница может быть 1,2,3..... N
Каждый предмет, который я надуваю 10 пунктов в моем представлении переработчика с сеткой.
Я реализовал следующий код для обработки бесконечной загрузки элементов в моем представлении переработчика, но проблема заключается в следующем:
после одной прокрутки он продолжает запускать API Вот мой код:

EndlessRecyclerView

public abstract class EndlessRecyclerView extends RecyclerView.OnScrollListener {
    public static String TAG = EndlessRecyclerView.class.getSimpleName();

    private int previousTotal = 0; // The total number of items in the dataset after the last load
    private boolean loading = true; // True if we are still waiting for the last set of data to load.
    private int visibleThreshold = 5; // The minimum amount of items to have below your current scroll position before loading more.
    int firstVisibleItem, visibleItemCount, totalItemCount;

    private int current_page = 1;

    private GridLayoutManager gridLayoutManager;

    public EndlessRecyclerView(GridLayoutManager gridLayoutManager) {
        this.gridLayoutManager = gridLayoutManager;
    }

    @Override
    public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
        super.onScrolled(recyclerView, dx, dy);
        visibleItemCount = recyclerView.getChildCount();
        totalItemCount = gridLayoutManager.getItemCount();
        previousTotal = previousTotal + visibleItemCount;
        if(previousTotal>totalItemCount)
        {
            current_page++;
            onLoadMore(current_page);
            previousTotal = 0;
        }
    }



    public abstract void onLoadMore(int current_page);
}

RecyclerView Snippet внутри основной деятельности

recyclerView.setOnScrollListener(new EndlessRecyclerView(gridLayoutManager) {
                @Override
                public void onLoadMore(int current_page) {
                   //async task that fires API and inflate recycler view
            });

1 ответ

1: Мое решение состоит в том, чтобы переписать SwipeRefreshLayout следующим образом (пример здесь):

@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
    ensureTarget();

    final int action = MotionEventCompat.getActionMasked(ev);

    if (mReturningToStart && action == MotionEvent.ACTION_DOWN) {
        mReturningToStart = false;
    }

    if (!isEnabled() || mReturningToStart || mRefreshing) {
        // Fail fast if we're not in a state where a swipe is possible
        return false;
    }

    switch (action) {
        case MotionEvent.ACTION_DOWN:
            if (canChildScrollUp()) {
            } else {
                setTargetOffsetTopAndBottom(mOriginalOffsetTop - mCircleView.getTop(), true);
                mActivePointerId = MotionEventCompat.getPointerId(ev, 0);
                mIsBeingDragged = false;
                final float initialMotionY = getMotionEventY(ev, mActivePointerId);
                if (initialMotionY == -1) {
                    return false;
                }
                mInitialMotionY = initialMotionY;
            }

        case MotionEvent.ACTION_MOVE:
            if (canChildScrollUp()) {
            } else {
                if (mActivePointerId == INVALID_POINTER) {
                    Log.e(LOG_TAG, "Got ACTION_MOVE event but don't have an active pointer id.");
                    return false;
                }

                final float y = getMotionEventY(ev, mActivePointerId);
                if (y == -1) {
                    return false;
                }
                final float yDiff = y - mInitialMotionY;
                if (yDiff > mTouchSlop && !mIsBeingDragged) {
                    mIsBeingDragged = true;
                    mProgress.setAlpha(STARTING_PROGRESS_ALPHA);
                }
                break;
            }

        case MotionEventCompat.ACTION_POINTER_UP:
            if (canChildScrollUp()) {
            } else {
                onSecondaryPointerUp(ev);
                break;
            }

        case MotionEvent.ACTION_UP:
        case MotionEvent.ACTION_CANCEL:
            if (canChildScrollUp())
            {
                if(mTarget instanceof AbsListView)
                {
                    final AbsListView absListView = (AbsListView) mTarget;

                    int lastPosition=absListView.getLastVisiblePosition();
                    if(lastPosition+1==absListView.getCount())
                    {
                        if(mCanLoadMore)
                        {
                            mCanLoadMore=false;
                            mLoadMoreListener.loadMore();
                        }
                    }
                }else if(mTarget instanceof RecyclerView)
                {
                    final RecyclerView recyclerView=(RecyclerView)mTarget;
                    LinearLayoutManager lLManager=(LinearLayoutManager)recyclerView.getLayoutManager();
                    if((lLManager.findLastVisibleItemPosition()+1)==lLManager.getItemCount())
                    {
                        if(mCanLoadMore)
                        {
                            mCanLoadMore=false;
                            mLoadMoreListener.loadMore();
                        }
                    }
                }
            } else {
                mIsBeingDragged = false;
                mActivePointerId = INVALID_POINTER;
                break;
            }
    }

    return mIsBeingDragged;
}

2: я думаю, что ваше решение может так:

@Override
public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
    super.onScrolled(recyclerView, dx, dy);
    totalItemCount = gridLayoutManager.getItemCount();

    int visibleItemCount=gridLayoutManager.findLastVisibleItemPosition();
    if((visibleItemCount+6)>totalItemCount)
    {
        current_page++;
        onLoadMore(current_page);
    }
}
Другие вопросы по тегам