234

下面是我的代码,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
    }
}
4

31 回答 31

213

by selecting an option, I need to update the fragment that is currently visible.

A simple way of doing this is using a trick related to the FragmentPagerAdapter implementation:

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;

Please rethink your variable naming convention, using as the variable name the name of the class is very confusing(so no ViewPager ViewPager, use ViewPager mPager for example).

于 2013-09-04T10:09:51.623 回答
164
    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);
        }
    }
于 2014-01-14T00:36:39.703 回答
119

首先,跟踪所有“活动”片段页面。在这种情况下,您在 ViewPager 使用的 FragmentStatePagerAdapter 中跟踪片段页面。

@Override
public Fragment getItem(int index) {
    Fragment myFragment = MyFragment.newInstance();
    mPageReferenceMap.put(index, myFragment);
    return myFragment;
}

为避免保留对“非活动”片段页面的引用,您需要实现 FragmentStatePagerAdapter 的 destroyItem(...) 方法:

@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);

MyAdapter 的 getFragment(int) 方法如下所示:

public MyFragment getFragment(int key) {
    return mPageReferenceMap.get(key);
}

希望它可以帮助!

于 2013-09-04T09:19:47.153 回答
106

这是我不会为特定片段类的实例变量获取 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);
}
于 2013-09-04T11:21:12.263 回答
41

FragmentStatePagerAdapter 有一个名为 instantiateItem 的公共方法,它根据指定的参数值返回您的片段,该方法有两个参数 ViewGroup (ViewPager) 和 position。

public Object instantiateItem(ViewGroup container, int position);

使用此方法获取指定位置的片段,

Fragment fragment = (Fragment) adaper.instantiateItem(mViewPager, position);
于 2017-03-16T05:57:04.873 回答
15

我知道为时已晚,但我有非常简单的方法,

// 对于 0 位置的片段

((mFragment) viewPager.getAdapter().instantiateItem(viewPager, 0)).yourMethod();
于 2017-02-13T14:14:17.483 回答
15
getSupportFragmentManager().getFragments().get(viewPager.getCurrentItem());

将从上面的行中检索到的实例转换为您要处理的片段。工作得很好。

viewPager

是管理片段的寻呼机实例。

于 2018-08-08T12:21:59.383 回答
11

这里有很多答案并没有真正解决一个基本事实,即确实没有办法以可预测的方式做到这一点,并且在某种程度上不会导致你在未来的某个时候开枪打死自己。

FragmentStatePagerAdapter是唯一知道如何可靠地访问被跟踪的片段的类FragmentManager- 任何尝试猜测片段的 id 或标签的尝试都是不可靠的,长期的。并且在保存/恢复状态时尝试手动跟踪实例可能无法正常工作,因为FragmentStatePagerAdapter在恢复状态时可能不会调用回调。

我唯一能做的就是复制代码FragmentStatePagerAdapter并添加一个返回片段的方法,给定一个位置(mFragments.get(pos))。请注意,此方法假定片段实际上是可用的(即它在某些时候是可见的)。

如果你特别喜欢冒险,你可以使用反射来访问私有mFragments列表的元素,但是我们又回到了原点(列表的名称不能保证保持不变)。

于 2015-06-12T17:27:04.833 回答
7

通过选择一个选项,我需要更新当前可见的片段。

要获得对当前可见片段的引用,假设您有对ViewPageras的引用mPager。然后以下步骤将获得参考currentFragment

  1. PageAdapter adapter = mPager.getAdapter();
  2. int fragmentIndex = mPager.getCurrentItem();
  3. FragmentStatePagerAdapter fspa = (FragmentStatePagerAdapter)adapter;
  4. Fragment currentFragment = fspa.getItem(fragmentIndex);

唯一使用的第 3 行通常是有效的。FragmentStatePagerAdapter是 ViewPager 的有用适配器。

于 2016-07-26T06:41:07.093 回答
7

最好的方法是调用CallingFragmentName fragment = (CallingFragmentName) viewPager .getAdapter() .instantiateItem(viewPager, viewPager.getCurrentItem()); 它将重新实例化您的调用片段,以便它不会抛出空指针异常并调用该片段的任何方法。

于 2018-08-23T10:35:45.717 回答
6

当前片段:

如果您使用片段标签栏模板创建项目,则此方法有效。

Fragment f = mSectionsPagerAdapter.getItem(mViewPager.getCurrentItem());

请注意,这适用于默认的选项卡式活动模板实现。

于 2016-01-22T15:01:20.633 回答
4

我使用了以下内容:

 int index = vpPager.getCurrentItem();
 MyPagerAdapter adapter = ((MyPagerAdapter)vpPager.getAdapter());
 MyFragment suraVersesFragment = (MyFragment)adapter.getRegisteredFragment(index);
于 2015-02-07T07:40:53.363 回答
3

当我们使用viewPager时,在activity中访问fragment实例的一个好方法是instantiateItem(viewpager,index)。//index- 你想要实例的片段的索引。

例如,我正在访问 1 个索引的片段实例-

Fragment fragment = (Fragment) viewPageradapter.instantiateItem(viewPager, 1);

if (fragment != null && fragment instanceof MyFragment) {      
 ((MyFragment) fragment).callYourFunction();

}
于 2018-03-08T09:27:45.670 回答
3

在我之前的实现中,我存储了一个子片段列表,以便以后能够访问它们,但结果证明这是一个错误的实现,导致巨大的内存泄漏。

我最终使用instantiateItem(...)方法来获取当前片段:

val currentFragment = adapter?.instantiateItem(viewPager, viewPager.currentItem)

或者要获得任何其他 Fragment 的位置:

val position = 0
val myFirstFragment: MyFragment? = (adapter?.instantiateItem(viewPager, position) as? MyFragment)

文档

为给定位置创建页面。适配器负责将视图添加到此处给出的容器中,尽管它只必须确保在它从 finishUpdate(ViewGroup) 返回时完成此操作。

于 2019-11-05T08:28:20.070 回答
2

在阅读所有评论和答案后,我将解释这个问题的最佳解决方案。最好的选择是@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();
}

PS要小心你把代码放在哪里,如果你insatiateItem在系统实际创建它之前调用savedInstanceState你的子片段将是空的

public void onCreate(@Nullable Bundle savedInstanceState){
      super(savedInstanceState)
}

会使您的应用程序崩溃。

祝你好运

于 2017-01-05T20:17:42.673 回答
2

你可以定义PagerAdapter这样的,然后你就可以得到任何Fragmentin 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());
于 2017-12-13T09:16:38.197 回答
2

FragmentStatePagerAdapter有一个名为mCurrentPrimaryItemtype的私有实例变量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;
}
于 2017-05-05T10:57:07.663 回答
1

我有同样的问题并使用此代码解决了它。

MyFragment fragment = (MyFragment) thisActivity.getFragmentManager().findFragmentById(R.id.container);

只需将名称 MyFragment 替换为您的片段名称并添加您的片段容器的 id。

于 2015-07-07T17:32:40.077 回答
1

如官方文档中所述,每个 Fragment 添加自己的菜单项并直接处理 onOptionsItemSelected() 可以更好地服务于所讨论的场景。最好避免未记录的技巧。

于 2016-06-30T23:07:10.877 回答
1

根据他对@chahat jain 的回答:

“当我们使用 viewPager 时,在 activity 中访问片段实例的一个好方法是 instantiateItem(viewpager,index)。//index- 您想要实例的片段的索引。”

如果你想在kotlin中这样做

val fragment =  mv_viewpager.adapter!!.instantiateItem(mv_viewpager, 0) as Fragment
                if ( fragment is YourFragmentFragment)
                 {
                    //DO somthign 
                 }

0到 0 的片段实例

//================================================= =========================//////#################### ########使用示例################################////== ==================================================== ======================//

这是一个完整的例子,可以让你迷失方向

这是我在 .xml 文件中的 veiewPager

   ...
    <android.support.v4.view.ViewPager
                android:id="@+id/mv_viewpager"
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                android:layout_margin="5dp"/>
    ...

还有我插入标签的家庭活动

...
import kotlinx.android.synthetic.main.movie_tab.*

class HomeActivity : AppCompatActivity() {

    lateinit var  adapter:HomeTabPagerAdapter

    override fun onCreate(savedInstanceState: Bundle?) {
       ...
    }



    override fun onCreateOptionsMenu(menu: Menu) :Boolean{
            ...

        mSearchView.setOnQueryTextListener(object : SearchView.OnQueryTextListener {
          ...

            override fun onQueryTextChange(newText: String): Boolean {

                if (mv_viewpager.currentItem  ==0)
                {
                    val fragment =  mv_viewpager.adapter!!.instantiateItem(mv_viewpager, 0) as Fragment
                    if ( fragment is ListMoviesFragment)
                        fragment.onQueryTextChange(newText)
                }
                else
                {
                    val fragment =  mv_viewpager.adapter!!.instantiateItem(mv_viewpager, 1) as Fragment
                    if ( fragment is ListShowFragment)
                        fragment.onQueryTextChange(newText)
                }
                return true
            }
        })
        return super.onCreateOptionsMenu(menu)
    }
        ...

}
于 2018-06-21T18:30:12.253 回答
1

这比公认的答案更具前瞻性:

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
    /* ------------------------------------------------------------------------------------------ */
}
于 2016-05-09T15:07:28.657 回答
0

在我的活动中,我有:

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);
}
于 2015-05-04T08:16:43.853 回答
0

您可以在 Fragment 中实现 BroadcastReceiver 并从任何地方发送 Intent。片段的接收者可以监听特定的动作并调用实例的方法。

一个警告是确保 View 组件已经实例化并且(对于某些操作,例如滚动列表,ListView 必须已经被渲染)。

于 2018-02-22T22:33:05.330 回答
0

您可以将片段数组声明为寄存器片段

class DashboardPagerAdapter(fm: FragmentManager?) : FragmentStatePagerAdapter(fm!!) {
    // CURRENT FRAGMENT
    val registeredFragments = SparseArray<Fragment>()

    override fun instantiateItem(container: ViewGroup, position: Int): Any {
        val fragment = super.instantiateItem(container, position) as Fragment
        registeredFragments.put(position, fragment)
        return fragment
    }

    override fun getItem(position: Int): Fragment {
        return when (position) {
            0 -> HomeFragment.newInstance()
            1 -> ConverterDashboardFragment.newInstance()
            2 -> CartFragment.newInstance()
            3 -> CustomerSupportFragment.newInstance()
            4 -> ProfileFragment.newInstance()
            else -> ProfileFragment.newInstance()
        }
    }

    override fun getCount(): Int {
        return 5
    }

}

然后您可以将其用作

adapter?.let {
    val cartFragment = it.registeredFragments[2] as CartFragment?
    cartFragment?.myCartApi(true)
}
于 2020-01-22T09:24:31.680 回答
0

如果您的寻呼机位于 Fragment 内,请使用以下命令:

private fun getPagerCurrentFragment(): Fragment? {
    return childFragmentManager.findFragmentByTag("android:switcher:${R.id.myViewPagerId}:${myViewPager.currentItem}")
}

R.id.myViewPagerId您的 ViewPager 在 xml 布局中的 id 在哪里。

于 2018-03-08T14:01:37.500 回答
0

只需从寻呼机获取当前项目,然后将您的适配器询问到该位置的片段。

 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;
    }
于 2017-01-11T10:53:11.347 回答
0

这是最简单的黑客:

fun getCurrentFragment(): Fragment? {
    return if (count == 0) null
    else instantiateItem(view_pager, view_pager.currentItem) as? Fragment
}

(科特林代码)

只需调用instantiateItem(viewPager, viewPager.getCurrentItem()并将其转换为Fragment. 您的项目已经被实例化了。为确保您可以添加检查getCount.

FragmentPagerAdapter和一起使用FragmentStatePagerAdapter

于 2018-03-03T16:54:31.553 回答
0

我尝试了以下方法:

 int index = mViewPager.getCurrentItem();
 List<Fragment> fragments = getSupportFragmentManager().getFragments();
 View rootView = fragments.get(index).getView();
于 2020-07-19T16:39:37.727 回答
0

要获取当前片段 - 在 ViewPager 中的public void onPageSelected(final int position)处获取位置,然后

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

于 2017-02-27T06:15:07.320 回答
0

从 FragmentPagerAdapter 覆盖 setPrimaryItem:对象是可见片段:

 @Override
        public void setPrimaryItem(ViewGroup container, int position, Object object) {
            if (mCurrentFragment != object) {
                mCurrentFragment = (LeggiCapitoloFragment) object;
            }
            super.setPrimaryItem(container, position, object);
        }
于 2017-01-05T19:39:54.103 回答
-1

因为 FragmentStateAdapter 会添加新的 Fragment 到 FragmentManager 会引用 FragmentManager 中的 Fragment 而不是在屏幕旋转时创建新的,所以你只需要进入 FragmentManager 对性能有好处

public MyFragment getActiveFragment() {
        List<Fragment> fragments = getSupportFragmentManager().getFragments();
        for (Fragment fragment : fragments){
            if(fragment instanceof  MyFragment){
                return (MyFragment) fragment;
            }
        }
        return null;
    }

于 2021-05-26T10:42:09.453 回答