Получение текущего экземпляра фрагмента в viewpager
Ниже мой код имеет 3 Fragment classes
, каждый из которых вставлен с каждой из трех вкладок на ViewPager
. У меня есть опция меню. Как показано в onOptionsItemSelected()
, выбирая опцию, мне нужно обновить фрагмент, который в настоящее время виден. Чтобы обновить, я должен вызвать метод, который находится в классе фрагмента. Может кто-нибудь предложить, как вызвать этот метод?
public class MainActivity extends ActionBarActivity {
ViewPager ViewPager;
TabsAdapter TabsAdapter;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ViewPager = new ViewPager(this);
ViewPager.setId(R.id.pager);
setContentView(ViewPager);
final ActionBar bar = getSupportActionBar();
bar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);
//Attaching the Tabs to the fragment classes and setting the tab title.
TabsAdapter = new TabsAdapter(this, ViewPager);
TabsAdapter.addTab(bar.newTab().setText("FragmentClass1"),
FragmentClass1.class, null);
TabsAdapter.addTab(bar.newTab().setText("FragmentClass2"),
FragmentClass2.class, null);
TabsAdapter.addTab(bar.newTab().setText("FragmentClass3"),
FragmentClass3.class, null);
if (savedInstanceState != null) {
bar.setSelectedNavigationItem(savedInstanceState.getInt("tab", 0));
}
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
switch (item.getItemId()) {
case R.id.addText:
**// Here I need to call the method which exists in the currently visible Fragment class**
return true;
}
return super.onOptionsItemSelected(item);
}
@Override
protected void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
outState.putInt("tab", getSupportActionBar().getSelectedNavigationIndex());
}
public static class TabsAdapter extends FragmentPagerAdapter
implements ActionBar.TabListener, ViewPager.OnPageChangeListener {
private final Context mContext;
private final ActionBar mActionBar;
private final ViewPager mViewPager;
private final ArrayList<TabInfo> mTabs = new ArrayList<TabInfo>();
static final class TabInfo {
private final Class<?> clss;
private final Bundle args;
TabInfo(Class<?> _class, Bundle _args) {
clss = _class;
args = _args;
}
}
public TabsAdapter(ActionBarActivity activity, ViewPager pager) {
super(activity.getSupportFragmentManager());
mContext = activity;
mActionBar = activity.getSupportActionBar();
mViewPager = pager;
mViewPager.setAdapter(this);
mViewPager.setOnPageChangeListener(this);
}
public void addTab(ActionBar.Tab tab, Class<?> clss, Bundle args) {
TabInfo info = new TabInfo(clss, args);
tab.setTag(info);
tab.setTabListener(this);
mTabs.add(info);
mActionBar.addTab(tab);
notifyDataSetChanged();
}
@Override
public void onPageScrollStateChanged(int state) {
// TODO Auto-generated method stub
}
@Override
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
// TODO Auto-generated method stub
}
@Override
public void onPageSelected(int position) {
// TODO Auto-generated method stub
mActionBar.setSelectedNavigationItem(position);
}
@Override
public void onTabReselected(Tab tab, FragmentTransaction ft) {
// TODO Auto-generated method stub
}
@Override
public void onTabSelected(Tab tab, FragmentTransaction ft) {
Object tag = tab.getTag();
for (int i=0; i<mTabs.size(); i++) {
if (mTabs.get(i) == tag) {
mViewPager.setCurrentItem(i);
}
}
tabPosition = tab.getPosition();
}
@Override
public void onTabUnselected(Tab tab, FragmentTransaction ft) {
// TODO Auto-generated method stub
}
@Override
public Fragment getItem(int position) {
TabInfo info = mTabs.get(position);
return Fragment.instantiate(mContext, info.clss.getName(), info.args);
}
@Override
public int getCount() {
return mTabs.size();
}
}
}
Предположим ниже, что класс фрагмента с методом updateList()
Я хочу вызвать:
public class FragmentClass1{
ArrayList<String> originalData;
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
View fragmentView = inflater.inflate(R.layout.frag1, container, false);
originalData = getOriginalDataFromDB();
return fragmentView;
}
public void updateList(String text)
{
originalData.add(text);
//Here I could do other UI part that need to added
}
}
Ответы
Ответ 1
выбрав параметр, мне нужно обновить фрагмент, который в настоящее время видны.
Простой способ сделать это - использовать трюк, связанный с реализацией FragmentPagerAdapter
:
case R.id.addText:
Fragment page = getSupportFragmentManager().findFragmentByTag("android:switcher:" + R.id.pager + ":" + ViewPager.getCurrentItem());
// based on the current position you can then cast the page to the correct
// class and call the method:
if (ViewPager.getCurrentItem() == 0 && page != null) {
((FragmentClass1)page).updateList("new item");
}
return true;
Пожалуйста, переосмыслите свое соглашение об именах переменных, используя в качестве имени переменной имя класса очень сбивает с толку (поэтому no ViewPager ViewPager
, например, используйте ViewPager mPager
).
Ответ 2
public class MyPagerAdapter extends FragmentPagerAdapter {
private Fragment mCurrentFragment;
public Fragment getCurrentFragment() {
return mCurrentFragment;
}
//...
@Override
public void setPrimaryItem(ViewGroup container, int position, Object object) {
if (getCurrentFragment() != object) {
mCurrentFragment = ((Fragment) object);
}
super.setPrimaryItem(container, position, object);
}
}
Ответ 3
Прежде всего, отслеживайте все "активные" фрагменты. В этом случае вы отслеживаете страницы фрагментов в FragmentStatePagerAdapter, который используется ViewPager.
@Override
public Fragment getItem(int index) {
Fragment myFragment = MyFragment.newInstance();
mPageReferenceMap.put(index, myFragment);
return myFragment;
}
Чтобы избежать ссылки на "неактивные" фрагментные страницы, вам необходимо реализовать метод destroyItem (...) FragmentStatePagerAdapter:
@Override
public void destroyItem (ViewGroup container, int position, Object object) {
super.destroyItem(container, position, object);
mPageReferenceMap.remove(position);
}
и когда вам нужно получить доступ к текущей видимой странице, вы вызываете:
int index = mViewPager.getCurrentItem();
MyAdapter adapter = ((MyAdapter)mViewPager.getAdapter());
MyFragment fragment = adapter.getFragment(index);
Где метод getFragment (int) MyAdapter выглядит следующим образом:
public MyFragment getFragment(int key) {
return mPageReferenceMap.get(key);
}
Надеюсь, это поможет!
Ответ 4
Это единственный способ, которым я не получаю NullPointerException для переменных экземпляра этих классов фрагментов. Это может быть полезно для других, которые придерживаются одного и того же. В onOptionsItemSelected() я закодировал следующий способ:
if(viewPager.getCurrentItem() == 0) {
FragmentClass1 frag1 = (FragmentClass1)viewPager.getAdapter().instantiateItem(viewPager, viewPager.getCurrentItem());
frag1.updateList(text);
}else if(viewPager.getCurrentItem() == 1) {
FragmentClass2 frag2 = (FragRecentApps)viewPager.getAdapter().instantiateItem(viewPager, viewPager.getCurrentItem());
frag2.updateList(text);
}
Ответ 5
FragmentStatePagerAdapter имеет общедоступный метод с именем instantiateItem, который возвращает ваш кадр на основе заданных значений параметров, этот метод имеет два параметра ViewGroup (ViewPager) и позицию.
public Object instantiateItem(ViewGroup container, int position);
Используется этот метод для получения указанного фрагмента позиции,
Fragment fragment = (Fragment) adaper.instantiateItem(mViewPager, position);
Этот код поможет мне, проверьте его.
Ответ 6
Здесь много ответов, которые на самом деле не затрагивают основополагающий факт, что на самом деле НЕТ ПУТЕМ, чтобы сделать это предсказуемо, и таким образом, чтобы вы не стреляли в ногу в какой-то момент в будущем.
FragmentStatePagerAdapter
- единственный класс, который знает, как надежно получить доступ к фрагментам, которые отслеживаются с помощью FragmentManager
- любая попытка попытаться угадать идентификатор фрагмента или тег не является надежным, долгосрочным. И попытки отследить экземпляры вручную, скорее всего, не сработают, когда состояние будет сохранено/восстановлено, потому что FragmentStatePagerAdapter
может не вызывать обратные вызовы при восстановлении состояния.
О том, что я смог сделать работу, является копирование кода для FragmentStatePagerAdapter
и добавление метода, возвращающего фрагмент, с учетом позиции (mFragments.get(pos)
). Обратите внимание, что этот метод предполагает, что фрагмент фактически доступен (т.е. Он был видимым в какой-то момент).
Если вы особенно предприимчивы, вы можете использовать отражение для доступа к элементам частного mFragments
списка, но затем мы вернемся к квадрату (имя списка не будет гарантировано оставаться неизменным).
Ответ 7
Текущий фрагмент:
Fragment f = mSectionsPagerAdapter.getItem(mViewPager.getCurrentItem());
Обратите внимание, что это работает с реализацией шаблона действия с вкладками по умолчанию.
Ответ 8
Я использовал следующее:
int index = vpPager.getCurrentItem();
MyPagerAdapter adapter = ((MyPagerAdapter)vpPager.getAdapter());
MyFragment suraVersesFragment = (MyFragment)adapter.getRegisteredFragment(index);
Ответ 9
выбрав параметр, мне нужно обновить фрагмент, который в настоящее время виден.
Чтобы получить ссылку на текущий видимый фрагмент, предположим, что у вас есть ссылка на ViewPager
как mPager
. Затем последующие шаги получат ссылку на currentFragment
:
-
PageAdapter adapter = mPager.getAdapter();
-
int fragmentIndex = mPager.getCurrentItem();
-
FragmentStatePagerAdapter fspa = (FragmentStatePagerAdapter)adapter;
-
Fragment currentFragment = fspa.getItem(fragmentIndex);
Обычно используется только литая строка 3. FragmentStatePagerAdapter
- полезный адаптер для ViewPager.
Ответ 10
Я знаю его слишком поздно, но у меня есть очень простые способы сделать это,
//для фрагмента с 0 положением
((mFragment) viewPager.getAdapter().instantiateItem(viewPager, 0)).yourMethod();
Ответ 11
У меня была такая же проблема и я решил использовать этот код.
MyFragment fragment = (MyFragment) thisActivity.getFragmentManager().findFragmentById(R.id.container);
Просто замените имя MyFragment на имя вашего фрагмента и добавьте идентификатор вашего контейнера фрагментов.
Ответ 12
Это более надежное будущее, чем принятый ответ:
public class MyFragmentPagerAdapter extends FragmentPagerAdapter {
/* ------------------------------------------------------------------------------------------ */
// region Private attributes :
private Context _context;
private FragmentManager _fragmentManager;
private Map<Integer, String> _fragmentsTags = new HashMap<>();
// endregion
/* ------------------------------------------------------------------------------------------ */
/* ------------------------------------------------------------------------------------------ */
// region Constructor :
public MyFragmentPagerAdapter(Context context, FragmentManager fragmentManager) {
super(fragmentManager);
_context = context;
_fragmentManager = fragmentManager;
}
// endregion
/* ------------------------------------------------------------------------------------------ */
/* ------------------------------------------------------------------------------------------ */
// region FragmentPagerAdapter methods :
@Override
public int getCount() { return 2; }
@Override
public Fragment getItem(int position) {
if(_fragmentsTags.containsKey(position)) {
return _fragmentManager.findFragmentByTag(_fragmentsTags.get(position));
}
else {
switch (position) {
case 0 : { return Fragment.instantiate(_context, Tab1Fragment.class.getName()); }
case 1 : { return Fragment.instantiate(_context, Tab2Fragment.class.getName()); }
}
}
return null;
}
@Override
public Object instantiateItem(ViewGroup container, int position) {
// Instantiate the fragment and get its tag :
Fragment result = (Fragment) super.instantiateItem(container, position);
_fragmentsTags.put(position, result.getTag());
return result;
}
// endregion
/* ------------------------------------------------------------------------------------------ */
}
Ответ 13
Сценарий, о котором идет речь, лучше обслуживается каждым фрагментом, добавляя его собственные пункты меню и непосредственно обрабатывая onOptionsItemSelected(), как описано в официальной документации. Лучше избегать недокументированных трюков.
Ответ 14
Просто получите текущий элемент из пейджера, а затем попросите адаптер к фрагменту этой позиции.
int currentItem = viewPager.getCurrentItem();
Fragment item = mPagerAdapter.getItem(currentItem);
if (null != item && item.isVisible()) {
//do whatever want to do with fragment after doing type checking
return;
}
Ответ 15
FragmentStatePagerAdapter
имеет переменную частного экземпляра под названием mCurrentPrimaryItem
типа Fragment
. Можно только задаться вопросом, почему разработчики Android не предоставили его с помощью геттера. Эта переменная создается в методе setPrimaryItem()
. Таким образом, переопределите этот метод таким образом, чтобы вы могли получить ссылку на эту переменную. Я просто закончил с объявлением моего собственного mCurrentPrimaryItem
и копированием содержимого setPrimaryItem()
в мое переопределение.
В вашей реализации FragmentStatePagerAdapter
:
private Fragment mCurrentPrimaryItem = null;
@Override
public void setPrimaryItem(ViewGroup container, int position, Object object) {
Fragment fragment = (Fragment)object;
if (fragment != mCurrentPrimaryItem) {
if (mCurrentPrimaryItem != null) {
mCurrentPrimaryItem.setMenuVisibility(false);
mCurrentPrimaryItem.setUserVisibleHint(false);
}
if (fragment != null) {
fragment.setMenuVisibility(true);
fragment.setUserVisibleHint(true);
}
mCurrentPrimaryItem = fragment;
}
}
public TasksListFragment getCurrentFragment() {
return (YourFragment) mCurrentPrimaryItem;
}
Ответ 16
В моей деятельности у меня есть:
int currentPage = 0;//start at the first tab
private SparseArray<Fragment> fragments;//list off fragments
viewPager.setOnPageChangeListener(new OnPageChangeListener() {
@Override
public void onPageSelected(int pos) {
currentPage = pos;//update current page
}
@Override
public void onPageScrolled(int arg0, float arg1, int arg2) {}
@Override
public void onPageScrollStateChanged(int arg0) {}
});
@Override
public void onAttachFragment(Fragment fragment) {
super.onAttachFragment(fragment);
if(fragment instanceof Fragment1)
fragments.put(0, fragment);
if(fragment instanceof Fragment2)
fragments.put(2, fragment);
if(fragment instanceof Fragment3)
fragments.put(3, fragment);
if(fragment instanceof Fragment4)
fragments.put(4, fragment);
}
Then I have the following method for getting the current fragment
public Fragment getCurrentFragment() {
return fragments.get(currentPage);
}
Ответ 17
final ViewPager.OnPageChangeListener onPageChangeListener =
new ViewPager.OnPageChangeListener() {
@Override public void onPageSelected(int position) {
switch (position) {
case 0:
FragmentClass1 frag1 =
(FragmentClass1) viewPager.getAdapter()
.instantiateItem(viewPager, viewPager.getCurrentItem());
frag1.updateList("Custom text");
break;
case 1:
break;
case 2:
break;
}
}
@Override
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
}
@Override public void onPageScrollStateChanged(int state) {
}
};
viewPager.addOnPageChangeListener(onPageChangeListener);
viewPager.setAdapter(adapter);
viewPager.post(new Runnable() {
@Override public void run() {
onPageChangeListener.onPageSelected(viewPager.getCurrentItem());
}
});
Иногда фрагмент может отправлять нулевой объект, поэтому вы можете запустить новый Runnable для решения проблемы при первой загрузке.
Спасибо, Рик
Ответ 18
Переопределить setPrimaryItem из вашего FragmentPagerAdapter: объект является видимым фрагментом:
@Override
public void setPrimaryItem(ViewGroup container, int position, Object object) {
if (mCurrentFragment != object) {
mCurrentFragment = (LeggiCapitoloFragment) object;
}
super.setPrimaryItem(container, position, object);
}
Ответ 19
Прочитав все комментарии и ответы, я собираюсь объяснить оптимальное решение этой проблемы. Лучшим вариантом является решение @rik, поэтому мое улучшение основывается на его.
Вместо того, чтобы задавать каждый FragmentClass как
if(FragmentClass1){
...
if(FragmentClass2){
...
}
Создайте свой собственный интерфейс и создайте его дочерние фрагменты, что-то вроде
public interface MyChildFragment {
void updateView(int position);
}
Затем вы можете сделать что-то подобное, чтобы инициировать и обновлять свои внутренние фрагменты.
Fragment childFragment = (Fragment) mViewPagerDetailsAdapter.instantiateItem(mViewPager,mViewPager.getCurrentItem());
if (childFragment != null) {
((MyChildFragment) childFragment).updateView();
}
P.S. Будьте осторожны, когда вы поместите этот код, если вы вызываете insatiateItem до того, как система действительно создаст его, savedInstanceState
вашего дочернего фрагмента будет пустым для него
public void onCreate(@Nullable Bundle savedInstanceState){
super(savedInstanceState)
}
Сбой приложения.
Удачи.
Ответ 20
Получить текущий фрагмент - получить позицию в ViewPager в public void onPageSelected (конечная позиция int), а затем
public PlaceholderFragment getFragmentByPosition(Integer pos){
for(Fragment f:getChildFragmentManager().getFragments()){
if(f.getId()==R.viewpager && f.getArguments().getInt("SECTNUM") - 1 == pos) {
return (PlaceholderFragment) f;
}
}
return null;
}
SECTNUM - аргумент позиции, назначенный в public static PlaceholderFragment newInstance (int sectionNumber); фрагмента
getChildFragmentManager() или getFragmentManager() - зависит от того, как созданные SectionsPagerAdapter
Ответ 21
Вы можете определить PagerAdapter
как это, тогда вы сможете получить любой Fragment
в ViewPager
.
private class PagerAdapter extends FragmentPagerAdapter {
private final List<Fragment> mFragmentList = new ArrayList<>();
public PagerAdapter(FragmentManager fm) {
super(fm);
}
@Override
public Fragment getItem(int position) {
return mFragmentList.get(position);
}
@Override
public int getCount() {
return mFragmentList.size();
}
public void addFragment(Fragment fragment) {
mFragmentList.add(fragment);
}
}
Чтобы получить текущий Fragment
Fragment currentFragment = mPagerAdapter.getItem(mViewPager.getCurrentItem());