FragmentStatePagerAdapter, обрабатывающий позицию getItem()
Я попробовал это, но неправильные списки привязаны к представлению переработчика. Я тоже пробовал SparseArray, даже это не работает. getItem() вызывается дважды, когда я запускаю Mainactivity. Как мне справиться с возвращенной позицией? Я попытался вернуть viewPager currentItem, даже это не работает.
Основная деятельность
public class PagerAdapter extends FragmentStatePagerAdapter {
int mNumOfTabs;
public PagerAdapter(android.support.v4.app.FragmentManager fragmentManager, int tabCount)
{
super(fragmentManager);
this.mNumOfTabs = tabCount;
}
@Override
public Fragment getItem(int position) {
return Categories_Fragment.newInstance(position);
}
@Override
public int getCount() {
return mNumOfTabs;
}
}
Categories_Fragment
public class Categories_Fragment extends Fragment
{
private RecyclerView lvMessages;
private int fromPage;
public static Categories_Fragment newInstance(int num) {
Categories_Fragment f = new Categories_Fragment();
// Supply num input as an argument.
Bundle args = new Bundle();
args.putInt("from", num);
f.setArguments(args);
return f;
}
@Override
public void onCreate(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onCreate(savedInstanceState);
fromPage = getArguments().getInt("from");
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
// TODO Auto-generated method stub
View rootView = inflater.inflate(R.layout.categories_filter, container, false);
lvMessages = (RecyclerView)rootView.findViewById(R.id.lv_categories_filter);
LinearLayoutManager x = new LinearLayoutManager(getActivity());
x.setOrientation(LinearLayoutManager.VERTICAL);
lvMessages.setLayoutManager(x);
return rootView;
}
@Override
public void onActivityCreated(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onActivityCreated(savedInstanceState);
// bind different data to the RecyclerView based on fromPage value.
}
}
4 ответа
Вот что может быть кикером...
@Override
public void setUserVisibleHint(boolean isVisible) {
super.setUserVisibleHint(isVisible);
if (isVisible) {
// bind to stuff using value of fromPage from here
}
}
public void setUserVisibleHint (boolean isVisibleToUser): установить подсказку системе о том, является ли пользовательский интерфейс этого фрагмента в данный момент видимым для пользователя. Этот совет по умолчанию имеет значение true и является постоянным для сохранения и восстановления состояния экземпляра фрагмента.
В чем может быть проблема (если я правильно понимаю ваш вопрос) в том, что пейджер проходит по своей "последней известной" позиции, что не обязательно означает позицию фрагмента, который в данный момент отображается на экране устройства.
ОБНОВЛЕНИЕ: вот полный пример с ViewPager, обрабатывающим тот же фрагмент, но с различными данными, переданными ему:
public class ViewPagerAdapter extends FragmentStatePagerAdapter {
private JSONArray itemsArray;
private int fragmentId;
public ViewPagerAdapter(JSONArray itemsArray, int fragmentId, FragmentManager manager) {
super(manager);
this.itemsArray = itemsArray;
this.fragmentId = fragmentId;
}
@Override
public Fragment getItem(int position) {
JSONObject itemObject = new Listing(itemsArray, position).getObject();
Bundle bundle = new Bundle();
bundle.putString("item_object", itemObject.toString());
bundle.putInt("item_position", position);
bundle.putInt("total_items", getCount());
bundle.putInt("fragment_Id", fragmentId);
Fragment fragment = new ListingFragment();
fragment.setArguments(bundle);
return fragment;
}
@Override
public int getCount() {
return itemsArray.length();
}
}
Вот фрагмент листингов (за вычетом неважных битов):
public class ListingFragment extends Fragment {
private Listing listing;
private String itemObject;
private int itemPosition;
private int totalItems;
private int fragmentId;
public ListingFragment() { }
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Bundle bundle = getArguments();
itemObject = bundle.getString("item_object");
itemPosition = bundle.getInt("item_position");
totalItems = bundle.getInt("total_items");
fragmentId = bundle.getInt("fragment_Id");
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
View view = ...
// nothing special to see here, just the usual stuff
return view;
}
@Override
public void setUserVisibleHint(boolean isVisible) {
super.setUserVisibleHint(isVisible);
if (isVisible) {
// send fragmentID to wherever you want to use it
// such as your recyclerview/adapter from here
}
}
}
Надеюсь, пример пролил некоторый свет, я знаю, что он работает для меня в любом случае.
С помощью FragmentStatePagerAdapter
,HashMap<Integer,Demo_Fragment>
public class ViewPagerAdapter extends FragmentStatePagerAdapter {
List<String> mList;
public ViewPagerAdapter(FragmentManager fm, List<String> list) {
super(fm);
mList = list;
}
@Override
public CharSequence getPageTitle(int position) {
String title = mList.get(position);
return title;
}
public int getCount() {
return mList.size();
}
public Fragment getItem(int position) {
Demo_Fragment fragment = new Demo_Fragment();
Bundle bundle = new Bundle();
bundle.putString("text",mList.get(position));
fragment.setArguments(bundle);
fragment.setTags(position);
mPageReferenceMap.put(position, fragment);
return fragment;
}
public void destroyItem(ViewGroup container, int position, Object object) {
super.destroyItem(container, position, object);
mPageReferenceMap.remove(position);
}
public int getItemPosition(Object item) {
Demo_Fragment fragment = (Demo_Fragment) item;
int position = fragment.getTags();
if (position >= 0) {
return position;
} else {
return POSITION_NONE;
}
}
}
Для получения фрагмента
public Demo_Fragment getFragment(int key) {
return mPageReferenceMap.get(key);
}
Для получения текущего фрагмента в addOnPageChangeListener ViewPager
mViewPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {
@Override
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
}
@Override
public void onPageSelected(int position) {
final Demo_Fragment demo_Fragment = getFragment(mViewPager.getCurrentItem());
final String str = mArrString.get(mViewPager.getCurrentItem());
demo_Fragment.updateView(str);
showDialog();
mRunnable = new Runnable() {
@Override
public void run() {
randomGenerator = new Random();
randomInt = randomGenerator.nextInt(10000);
demo_Fragment.updateView(str +"\n" +randomInt+"-Updated Value");
hideDialog();
}
};
mHandler.postDelayed(mRunnable, 2000);
}
@Override
public void onPageScrollStateChanged(int state) {
}
});
Рабочий пример - Нажмите здесь
Подозрительно, что вы создали класс с именем PagerAdapter, который является абстрактным классом Android. Переименуйте его в уникальное имя, например, MyPagerAdapter.
Причина, по которой метод getItem() вызывается дважды, заключается в значении по умолчанию, назначенном для offScreenPageLimit в ViewPager.
Вы можете установить offScreenPageLimit в 1 или 0, используя метод setOffscreenPageLimit (int limit) класса ViewPager, и это решит вашу проблему вызова getItem() дважды.
Следовательно, если getItem() вызывается один раз, вы получите правильное значение позиции во фрагменте, и ваша проблема будет решена.
проверить ссылку для определения и описания метода
http://developer.android.com/reference/android/support/v4/view/ViewPager.html