我无法在FragmentPagerAdapter中重用片段。使用destroyItem()方法,它正在删除片段,但仍然没有再次调用getItem()..只有2-3个图像所以我使用FragmentPagerAdapter而不是FragmentStatePagerAdapter ..
public class ExamplePagerAdapter extends FragmentPagerAdapter {
ArrayList < String > urls;
int size = 0;
public ExamplePagerAdapter(FragmentManager fm, ArrayList < String > res) {
super(fm);
urls = res;
size = urls.size();
}
@Override
public int getCount() {
if (urls == null) {
return 0;
} else {
return size;
}
}
@Override
public void destroyItem(ViewGroup container, int position, Object object) {
FragmentManager manager = ((Fragment) object).getFragmentManager();
FragmentTransaction trans = manager.beginTransaction();
trans.remove((Fragment) object);
trans.commit();
}
@Override
public Fragment getItem(int position) {
Fragment fragment = new FloorPlanFragment();
Bundle b = new Bundle();
b.putInt("p", position);
b.putString("image", urls.get(position));
Log.i("image", "" + urls.get(position));
fragment.setArguments(b);
return fragment;
}
}
在FragmentActivity中,
pager.setAdapter(new ExamplePagerAdapter(getSupportFragmentManager(), res2));
答案 0 :(得分:282)
KISS答案:
简单使用FragmentStatePagerAdapter代替FragmentPagerAdapter。
我得到了答案。首先我想删除这个问题,因为我犯的是一个非常愚蠢的错误,但这个答案将帮助那些面临同样问题的人而不是FragmentPagerAdapter
,使用{{1} }。
@BlackHatSamurai在评论中提到:
这可行的原因是因为
FragmentStatePagerAdapter
会破坏 作为未使用的碎片。FragmentStatePagerAdapter
没有。
答案 1 :(得分:148)
使用FragmentStatePagerAdapter
没有完全解决我的问题,这是一个类似的问题,其中onCreateView
未在视图寻呼机中为子片段调用。我实际上将FragmentPagerAdapter
嵌套在另一个Fragment
中,因此FragmentManager
在所有这些内容中共享,从而保留了旧片段的实例。修复方法是将getChildFragmentManager
的实例反馈到我的主机片段中FragmentPagerAdapter
的构造函数。有点像...
FragmentPagerAdapter adapter = new FragmentPagerAdapter(getChildFragmentManager());
可以通过片段访问getChildFragmentManager()
方法,这对我有用,因为它为该片段返回一个私有FragmentManager
,专门用于需要嵌套片段的情况。我希望这可以帮助那些可能遇到同样问题的人!!
getChildFragmentManager()
您的最低API版本必须至少为17 (4.2)
,因此这可能会给您带来麻烦。当然,如果您使用支持库v4中的片段,那么您应该没问题。答案 2 :(得分:3)
我做了@kanika和@ Jraco11发布的内容,但我仍然遇到了问题。
所以,经过大量的修改后,我找到了一个适合我的方法,并在下一个代码中添加到我的FragmentPagerAdapter中:
@Override
public int getItemPosition(Object object) {
return POSITION_NONE;
}
根据我读到的内容, getItemPosition 用于通知 ViewPager 是否刷新项目,并且如果可见位置的项目避开,则避免更新改变了。
答案 3 :(得分:2)
有两种不同的情况: 1.)每个寻呼机都有相同的布局: 在这种情况下,如果您扩展自定义适配器会更好 通过PagerAdapter并返回单个布局。
2。)每个寻呼机都有不同的布局: 在这种情况下,如果您扩展自定义适配器会更好 通过FragmentStatePagerAdapter并为每个寻呼机返回不同的碎片。
答案 4 :(得分:1)
覆盖long getItemId (int position)
FragmentPagerAdapter
使用getItem
缓存它创建的片段。即使在调用notifyDataSetChanged()
getItem
之后,我也面临同样的问题。
这实际上是一个功能而不是错误。您需要覆盖getItemId
,以便正确重用您的片段。由于您要移除碎片,您的位置正在发生变化。如文档中所述:
long getItemId (int position)
在给定位置返回项目的唯一标识符。
默认实现返回给定的位置。 如果项目的位置可以更改,则子类应覆盖此方法。
只需为每个片段提供一个唯一的ID即可。
在FragementStatePagerAdapter
中使用POSITION_NONE
或返回int getItemPosition (Object object)
是错误的。你不会得到任何缓存。
答案 5 :(得分:1)
方法getItem()
仅用于创建新项目。创建后,将不会调用此方法。如果需要获取适配器使用的货币项目,请使用以下方法:
pagerAdapter.instantiateItem(viewPager, TAB_POS)
答案 6 :(得分:0)
我发现在tab-layout上设置一个监听器会阻止它被调用,可能是因为它们只有tabLayout.setOnTabSelectedListener
上的一个监听器空间而不是一个监听器数组。