![](/img/trans.png)
[英]FragmentStateAdapter for ViewPager2 notifyItemChanged not working as expected
[英]NotifyDataSetChanged not working in FragmentStateAdapter with Viewpager2
我正在使用带有 FragmentStateAdapter 的 viewpager2 创建一个文件选择页面。 在其中一个页面中,我显示了所有已安装的存储设备,我想在这些设备上点击以查看内容,但我希望内容位于另一个片段上。
我想在 viewpager 的最后一页中用另一个片段替换片段。 但是使用下面的代码,页面只是刷新并且没有创建新的片段,因为createFragment
没有被调用。
我的寻呼机适配器
public static class PagerAdapter extends FragmentStateAdapter {
private Fragment mFragmentAtPos0;
private final FragmentManager mFragmentManager;
private final FirstPageListener listener = new FirstPageListener();
public final class FirstPageListener implements
FirstPageFragmentListener {
public void onSwitchToNextFragment() {
mFragmentManager.beginTransaction().remove(mFragmentAtPos0)
.commit();
if (mFragmentAtPos0 instanceof FilesandFolder_Others_MainPage){
mFragmentAtPos0 = new FileExplorer(listener);
}else{ // Instance of NextFragment
mFragmentAtPos0 = new FilesandFolder_Others_MainPage(listener);
}
notifyDataSetChanged();
}
public void onSwitchToNextFragment(Bundle bundle) {
// mFragmentManager.beginTransaction().remove(mFragmentAtPos0)
// .commit();
if (mFragmentAtPos0 instanceof FilesandFolder_Others_MainPage){
mFragmentAtPos0 = new FileExplorer(listener);
mFragmentAtPos0.setArguments(bundle);
}else{ // Instance of NextFragment
mFragmentAtPos0 = new FilesandFolder_Others_MainPage(listener);
mFragmentAtPos0.setArguments(bundle);
}
notifyDataSetChanged();
// notifyItemChanged(getItemPosition(mFragmentAtPos0));
}
}
// @Override
// public void onBindViewHolder(@NonNull FragmentViewHolder holder, int position, @NonNull List<Object> payloads) {
// super.onBindViewHolder(holder, position, payloads);
// if (position == getItemPosition(mFragmentAtPos0)) {
//
// Fragment f = mFragmentManager.findFragmentByTag("f" + holder.getItemId());
// if (f != null) {
// mFragmentManager.beginTransaction().replace(holder.getItemId(), )
// }
// }
// }
private int getItemPosition(Fragment mFragmentAtPos0) {
for (int i = 0; i < getItemCount(); i++){
if (createFragment(i).equals(mFragmentAtPos0)){
return i;
}
}
return -1;
}
public PagerAdapter(FragmentActivity fm) {
super(fm);
mFragmentManager = fm.getSupportFragmentManager();
mFragmentAtPos0 = new FilesandFolder_Others_MainPage(listener);
}
@NonNull
@Override
public Fragment createFragment(int position) {
switch (position){
case 0:
return new AppSelectionFragment();
case 1:
return new Photos();
case 2:
return new VideoGalleryFragment();
case 3:
return new Gallery();
default:
return mFragmentAtPos0;
}
}
@Override
public int getItemCount() {
return 5;
}
}
这是我想替换为另一个 Fragment 的 Fragment 的构造函数:
private static FileSelection.PagerAdapter.FirstPageListener pageFragmentListener;
public FilesandFolder_Others_MainPage() {
}
public FilesandFolder_Others_MainPage(FileSelection.PagerAdapter.FirstPageListener firstPageFragmentListener) {
pageFragmentListener = firstPageFragmentListener;
}
在点击视图时,应该调用以下代码:-
Bundle bundle = new Bundle();
bundle.putString("PATH", volumes.get(position).path);
pageFragmentListener.onSwitchToNextFragment(bundle);
我已经搜索了整个 stackoverflow,但从来没有看到 viewpager 2 和 FragmentStateAdapter 的正确答案,因为每个人似乎都在使用 Viewpager
我试图实现这个:- ViewPager2 notifyItemChanged 的 FragmentStateAdapter 没有按预期工作
但无法理解如何编辑我的代码。 请帮助提前谢谢
有同样的问题,这就是我发现的:
FragmentStateAdapter
具有onBindViewHolder(holder, position, payloads)
方法,您可以覆盖该方法(与最终的onBindViewHolder(holder, position)
不同)。 此方法在您调用notify
方法时调用,对于notifyItemChanged()
、 notifyItemRangeChanged()
方法,您可以访问片段并手动更新它。
这是访问片段的代码:
public void onBindViewHolder(@NonNull FragmentViewHolder holder, int position, @NonNull List<Object> payloads) {
String tag = "f" + holder.getItemId();
Fragment fragment = fragmentManager.findFragmentByTag(tag);
if (fragment != null) {
//manual update fragment
} else {
// fragment might be null, if it`s call of notifyDatasetChanged()
// which is updates whole list, not specific fragment
super.onBindViewHolder(holder, position, payloads);
}
}
FragmentStateAdapter
由"f" + holder.getItemId()
标记他的片段,这就是它起作用的原因。
您也可以使用DiffUtil
,这是更好的方法。 这是解决这个问题的好例子,最后还有一些链接可以更好地理解ViewPager2
。
我今天遇到了同样的问题,并想在这里分享我的解决方案。 我的解决方案不是拦截onBindViewHolder()
,而是简单地替换旧片段:
OldFragment.instance.containerId?.let {
replace(it, NewFragment.instance)
}
更具体地说:
我的应用程序有一个带有两个片段的 ViewPager2。 我们称它们为MainFragment
和DetailFragment
。 在MainFragment
中单击按钮时,我想用SecondMainFragment
替换它。 反之亦然:单击SecondMainFragment
中的按钮将用MainFragment
替换自身。
class CustomFragmentStateAdapter(
private val fragmentActivity: FragmentActivity
) : FragmentStateAdapter(fragmentActivity) {
var displaySecondMainFragment: Boolean = false
set(value) {
if (field != value) {
switchFragments(replaceWithSecondMainFragment = value)
}
field = value
}
private fun switchFragments(replaceWithSecondMainFragment: Boolean) {
fragmentActivity.supportFragmentManager.commit {
if (replaceWithSecondMainFragment) {
MainFragment.instance.containerId?.let {
replace(it, SecondMainFragment.instance)
}
} else {
SecondMainFragment.instance.containerId?.let {
replace(it, MainFragment.instance)
}
}
}
}
// Standard adapter overrides
override fun getItemCount(): Int = 2
override fun createFragment(position: Int): Fragment = when (position) {
0 -> if (displaySecondMainFragment) SecondMainFragment.instance else MainFragment.instance
else -> DetailFragment.instance
}
}
使用此扩展属性来获取片段所在容器的 id:
inline val Fragment.containerId: Int?
get() = (view?.parent as? ViewGroup?)?.id
最后我只需要做这样的事情:
button.setOnClickListener {
customFragmentStateAdapter.displaySecondMainFragment = true // Or false for the other way
}
你可以覆盖这两种方法
override fun getItemId(position: Int): Long {
// generate new id
return getItem(position).hashCode().toLong()
}
override fun containsItem(itemId: Long): Boolean {
// false if item is changed
return dataList.find { it.hashCode().toLong() == itemId } != null
}
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.