ViewPager2 的 FragmentStateAdapter 中的 notifyItemChanged 方法无法按预期工作

7

我正在使用带有FragmentStateAdapter的ViewPager2,并调用notifyItemChanged(position)方法。但是,如所预期的那样,createFragment方法没有被再次调用。这是故意的行为还是错误?我应该怎么做?

我的代码如下:

    class EEditionPagerAdapter(
    fragmentManager: FragmentManager, lifecycle: Lifecycle, private var date: String
    ) : FragmentStateAdapter(fragmentManager, lifecycle) {

    private var menuList = emptyList<EEditionMenu>()

    override fun getItemCount(): Int = menuList.size

    override fun createFragment(position: Int): Fragment {
        val menu = menuList[position]
        return EEditionListingFragment.newInstance(menu, date)
    }

    fun submitList(list: List<EEditionMenu>) {
        menuList = list
        notifyItemRangeChanged(0, menuList.size)
    }

    fun changeDate(date: String, position: Int){
        this.date = date
        notifyItemChanged(position)
    }
}
3个回答

7

FragmentStateAdapter使用项目ID来确定是否已经创建片段,并重用已经创建的片段。默认情况下,item ID是项目的位置。您可以重写getItemIdcontainsItem来根据位置和日期提供自己的唯一ID,然后在日期更改时它将请求一个新的片段。


4
当您为FragmentStateAdapter调用notifyItemChanged时,它会调用ensureFragment(position),如果已经创建了该片段,则会恢复该片段。
 private void ensureFragment(int position) {
    long itemId = getItemId(position);
    if (!mFragments.containsKey(itemId)) {
        // TODO(133419201): check if a Fragment provided here is a new Fragment
        Fragment newFragment = createFragment(position);
        newFragment.setInitialSavedState(mSavedStates.get(itemId));
        mFragments.put(itemId, newFragment);
    }
}

尝试覆盖onBindViewHolder方法,并将日期设置为片段。
   override fun onBindViewHolder(
    holder: FragmentViewHolder,
    position: Int,
    payloads: MutableList<Any>
) {
    super.onBindViewHolder(holder, position, payloads)
    val fragment: EEditionListingFragment? = fragmentManager.findFragmentByTag("f$position") as EEditionListingFragment?
    fragment.updateDate(date)
}

那么我应该怎么做才能刷新当前页面片段? - Karthick Ramanathan
完成答案。 - kalugin1912
1
很好 - 不过有点奇怪,如果他们真的想要弥合viewpager和recyclerview之间的差距,为什么没有get方法呢... - slott
1
@slott 桥接器正在使用您自己的项目ID。 - StenSoft

0

在FragementStateAdapter.class中放置到mFragment的片段是由

void placeFragmentInViewHolder(@NonNull final FragmentViewHolder holder)

这是一个方法,它是包私有的,我们无法访问。在这个方法中,片段会被添加到 mFragment 管理器中,并带有标签"f" + holder.getItemId(),这与"f" + position相同。

        scheduleViewAttach(fragment, container);
        mFragmentManager.beginTransaction()
                .add(fragment, "f" + holder.getItemId())
                .setMaxLifecycle(fragment, STARTED)
                .commitNow();
        mFragmentMaxLifecycleEnforcer.updateFragmentMaxLifecycle(false);

因此,当显示片段时,我们可以重写onBindViewHolder方法。
@Override
public void onBindViewHolder(@NonNull FragmentViewHolder holder, int position, @NonNull List<Object> payloads) {
    super.onBindViewHolder(holder, position, payloads);
    if (position == 0) {

        String fragmentTag = "f" + holder.getItemId();
        Log.v(TAG,"fragmentTag is : " + fragmentTag);
        MyFragment fragment0 = (MyFragment) this.mFragmentManger.findFragmentByTag(fragmentTag);

        if (fragment0 != null) {
            Log.v(TAG,"onBindViewHolder updating fragment ...");
            // do what ever you want to update the MyFragment 
            fragment0.update(payloads);
        } else {
            Log.v(TAG,"onBindViewHolder called, but fragment0 is null!");
        }
    }
} 

在我的情况下,this.mFragmentManager 是通过构造函数缓存的。
public MyFragmentStateAdapter(@NonNull Fragment fragment) {
        super(fragment); // this calls the fragment.getChildFragmentManager().
        this.mFragmentManger = fragment.getChildFragmentManager();
}

createFragment() 的含义是:

@NonNull
@Override
public Fragment createFragment(int position) {
    if(position == 0){
        if (this.fragment0 == null) {
            this.fragment0 = MyFragment.getInstance();
        }
        return this.fragment0;
    } else {
        Log.v(TAG, "position " + position + " is called");
        return null;
    }
}

最后,在其他地方调用更新函数来触发更新:
ViewPager2.getAdapter().notifyItemChanged(0, null); 
// 0 is the fragment position of the fragment0 and null is payload.

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接