Viewpage+TabLayout+Fragment 刷新问题详解

今天在写首页的时候遇到一个问题,采用的是Viewpage+TabLayout+Fragment的布局,然后发现后台添加数据的时候或者更改tablayout的数据位置的,viewpage的数据显示不对应,请教大佬终于找到了解决方案

  ViewPager与TabLayout的初始化:

private void initData(){
      mTabFragmentAdapter = new HomeFragmentPagerAdapter(getChildFragmentManager());
        mViewPager.setAdapter(mTabFragmentAdapter);
        mViewPager.setOffscreenPageLimit(0);
        mViewPager.addOnPageChangeListener(
        new TabLayout.TabLayoutOnPageChangeListener(mTabLayout));
        mTabLayout.setupWithViewPager(mViewPager);
        mViewPager.setIsScroll(false, false);
}

2.刷新数据FragmentPagerAdapter的notifyDataSetChanged()

Viewpage+TabLayout+Fragment 刷新问题详解_第1张图片

3,FragmentPagerAdapter主要设置了

public class HomeFragmentPagerAdapter extends FragmentPagerAdapter {
    private List mTitles = new ArrayList<>();

    public void setDataAndType(List titles) {
        mTitles.clear();
        mTitles.addAll(titles);
    }

    @SuppressWarnings("deprecation")
    public HomeFragmentPagerAdapter(FragmentManager fm) {
        super(fm);
    }

    @NotNull
    @Override
    public Fragment getItem(int position) {
        int id = mTitles.get(position).getId();
        Log.d("test","id = "+id);
        return HomeGoodsFragment.newInstance(id);
    }

    @Override
    public int getCount() {
        return this.mTitles == null ? 0 : this.mTitles.size();
    }

    @Nullable
    @Override
    public CharSequence getPageTitle(int position) {
        return mTitles.get(position).getName();
    }

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

    @Override
    public void restoreState(Parcelable state, ClassLoader loader) {
//        super.restoreState(state, loader);
    }

    @Override
    public int getItemPosition(@NotNull Object object) {
        Log.d("test","刷新数据");
        ((HomeGoodsFragment) object).refresh();
        return POSITION_NONE;
    }


    @Override
    public long getItemId(int position) {
        return mTitles.get(position).getId();
    }
}

我们发现每次创建fragment时候,FragmentManager都会通过findFragmentByTag去缓存中查找,是否存在指定tagName的Fragment,有就复用,没有就调用getItem()创建新的。而这里的tagName就是通过getItemId()方法生成的。

原来FragmentPagerAdapter里在根据getItemId(int position)来判断当前position里Fragment是否存在,如果存在,则不会创建亦不会更新,那么要让FragmentPagerAdapter的更新生效,那在getItemId(int)里根据数据返回一个唯一的数据ID,当FragmentPagerAdapter更新时,数据ID改变了,那么Fragment就会调用getItem(int)去获取新Fragment,达到更新效果
好了,我们的解决方案就是通过重新getItemId()方法,返回唯一的id。
 

 

mTabItemIds我是使用的每个tab的id集合。然后返回不同tabId的hashcode作为唯一id。
同时还要重写getItemPosition()方法,通知刷新位置变化了。

Viewpage+TabLayout+Fragment 刷新问题详解_第2张图片

你可能感兴趣的:(android,java)