如何在RecyclerView中保留ViewPager状态?

时间:2015-12-14 19:05:00

标签: android android-viewpager android-recyclerview

我的RecyclerView包含许多ViewPagers。每个ViewPager中的页面数是常量(2)。我使用PagerAdapter。

当我滚动我的RecyclerView时,ViewPagers会丢失其状态并重置为第1页。我想知道如何保留每个ViewPager的最后一个选定页面。

例如: 1)我在第一个ViewPager中滚动到第2页 2)我将RecyclerView滚动到底部,我的第一个ViewPager离开了屏幕。 3)我滚动到ViewPager的顶部,因此我的第一个ViewPager再次加载。 4)我的第一个ViewPager被设置为第一页而不是我最后滚动到的第2页。

RecyclerView中有很多ViewPagers,我想保留所有状态,无论是否因为RecyclerView而重新加载。

如何解决这个逻辑?

2 个答案:

答案 0 :(得分:5)

好吧,我为此写了一点暴力解决方案。

我的PagerAdapter

public class ViewPagerAdapter extends PagerAdapter {

    @Override
    public int getCount() {
        return 2;
    }

    @Override
    public Object instantiateItem(ViewGroup container, int position) {
        FrameLayout layout = new FrameLayout(container.getContext());
        FrameLayout.LayoutParams layoutParams = new FrameLayout.LayoutParams(
                FrameLayout.LayoutParams.MATCH_PARENT, FrameLayout.LayoutParams.MATCH_PARENT);
        layout.setLayoutParams(layoutParams);

        layout.setBackgroundColor(position % 2 == 0 ? Color.RED : Color.BLUE);
        container.addView(layout);
        return layout;
    }

    @Override
    public void destroyItem(ViewGroup container, int position, Object object) {
        super.destroyItem(container, position, object);
    }

    @Override
    public boolean isViewFromObject(View view, Object object) {
        return view == object;
    }
}

(所以我们视觉上#34;状态"它的颜色不同 - 它在第一个位置是红色,在第二个位置是蓝色)

RecyclerViewAdapter看起来像:

public class RecyclerViewAdapter extends RecyclerView.Adapter<RecyclerView.ViewHolder> {

    public HashMap<Integer, Integer> viewPageStates = new HashMap<>();

    public RecyclerViewAdapter() {}

    @Override
    public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup viewGroup, int viewType) {
        return new ViewHolder(new ViewWithViewPager(viewGroup.getContext()));
    }

    @Override
    public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
        ((ViewWithViewPager) holder.itemView).refreshState(position, viewPageStates.containsKey(position)? viewPageStates.get(position) : 0);
    }

    @Override
    public int getItemCount() {
        return 42;
    }

    @Override
    public void onViewRecycled(RecyclerView.ViewHolder holder) {
        ViewWithViewPager recycledViewPager = ((ViewWithViewPager)holder.itemView);
        viewPageStates.put(recycledViewPager.viewPosition, recycledViewPager.viewPager.getCurrentItem());
        super.onViewRecycled(holder);
    }

    public class ViewHolder extends RecyclerView.ViewHolder {

        public ViewHolder(View itemView) {
            super(itemView);
        }
    }
}   

即。每当我们绑定RecyclerView&#39; ViewHolder时,我们都会传递viewpager的当前状态(来自viewPageStates)及其在RecyclerView中的顺序数据集。 一旦它被回收 - 我们会阅读其当前页面状态并将其保存到HashMap viewPageStates

最后一部分 - ViewWithViewPager

public class ViewWithViewPager extends FrameLayout {

    ViewPager viewPager;
    int viewPosition = 0;

    public ViewWithViewPager(Context context) {
        super(context);
        LayoutInflater.from(context).inflate(R.layout.item_layout, this);
        viewPager = (ViewPager) findViewById(R.id.viewPager);
        viewPager.setAdapter(new ViewPagerAdapter());
    }

    public void refreshState(int position, int selectedPage) {
        viewPosition = position;
        viewPager.setCurrentItem(selectedPage);
    }
}

即。在refreshState中,我们将ViewPager设置为正确的页面

答案 1 :(得分:0)

OP解决方案。

  • 在你的内部覆盖onViewRecycled(ViewHolder viewHolder){} RecyclerView.Adapter
  • 使用HashMap存储ViewPager的位置 RecyclerView和相应的当前所选页面 ViewPager。
  • 在onViewRecycled()方法内输入上述值 创建了HashMap。
  • 在onBindViewHolder()方法中加载保存的状态/页面 ViewPager从HashMap使用位置并设置保存 使用setCurrentItem()方法调用ViewPager的状态/页面。

感谢Konstantin和atabouraya。