使用ViewPager从Activity向Fragment进行通信

5

我有一个Activity和两个Fragment用于包含一个tablayout和一个viewpager。现在,我可以通过实现Google的指南回调接口从Fragment向Activity进行通信。但是如何实现另一种方式从Activity到Fragment的通信呢?如果Activity中发生了某些事情(外部事件),我想要更新Fragment。我成功获取了Frag1 fragment。

MyFragmentPagerAdapter a = (MyFragmentPagerAdapter) viewPager.getAdapter();
Frag1 frag = (Frag1) a.getItem(0);

但是当我在frag上调用公共方法时,会收到一个IllegalStateException:Fragment not attached to Activity,这可能是因为getItem(0)返回Frag1的新实例,而此时尚未附加...是否有人可以提供整个Viewpager -> Activity到Fragment通信的清晰解决方案?以下是一些代码:

在Activity中:

@Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        ViewPager viewPager = (ViewPager) findViewById(R.id.viewpager);
        if (viewPager != null) {
            viewPager.setAdapter(new MyFragmentPagerAdapter(getSupportFragmentManager()));
        }

        TabLayout tabLayout = (TabLayout) findViewById(R.id.sliding_tabs);
        if (tabLayout != null) {
            tabLayout.setupWithViewPager(viewPager);
        }
}

活动布局:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:orientation="vertical"
    tools:context="com.MainActivity">

    <android.support.design.widget.TabLayout
        android:id="@+id/sliding_tabs"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"/>

    <android.support.v4.view.ViewPager
        android:id="@+id/viewpager"
        android:layout_width="match_parent"
        android:layout_height="0px"
        android:layout_weight="1"/>
</LinearLayout>

MyFragmentPagerAdapter:

public class MyFragmentPagerAdapter extends FragmentPagerAdapter {

    final int PAGE_COUNT = 2;
    private String tabTitles[] = new String[] { "tab1", "tab2" };

    public MyFragmentPagerAdapter(FragmentManager fm) {
        super(fm);
    }

    @Override
    public Fragment getItem(int position) {
        switch (position) {
            case 0:
                return Frag1.newInstance(position + 1);
            case 1:
                return Frag2.newInstance(position + 1);
            default:
                return null;
        }
    }

    @Override
    public int getCount() {
        return PAGE_COUNT;
    }

    @Override
    public CharSequence getPageTitle(int position) {
        return tabTitles[position];
    }
}

片段1:

public class Frag1 extends Fragment {
    public static final String ARG_PAGE = "ARG_PAGE";

    private int mPage;

    private onFrag1InteractionListener mListener;

    public Frag1() {
        // Required empty public constructor
    }

    /**
     * Use this factory method to create a new instance of
     * this fragment using the provided parameters.
     *
     * @return A new instance of fragment Frag1.
     */
    public static Frag1 newInstance(int page) {
        Frag1 fragment = new Frag1();
        Bundle args = new Bundle();
        args.putInt(ARG_PAGE, page);
        fragment.setArguments(args);
        return fragment;
    }

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        if (getArguments() != null) {
            mPage = getArguments().getInt(ARG_PAGE);
        }
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.fragment_1, container, false);
        return view;
    }


    @Override
    public void onAttach(Context context) {
        super.onAttach(context);
        if (context instanceof onFrag1InteractionListener) {
            mListener = (onFrag1InteractionListener) context;
        } else {
            throw new RuntimeException(context.toString()
                    + " must implement onFrag1InteractionListener");
        }
    }

    @Override
    public void onDetach() {
        super.onDetach();
        mListener = null;
    }

    public interface onFrag1InteractionListener {
        // TODO: Update argument type and name
        void onFrag1Interaction(Action action);
    }

扩展AppCompatActivity而不是Activity,然后检查发生了什么? - Nitesh
外部事件是什么意思?比如说,如果你想在选项卡被滑动时更新片段。就像这样。 - Nitesh
你想根据父活动中的某些事件向片段发送一些数据。我说得对吗?@Zuop - Swaminathan V
当主活动发生某些事件时,您是否想要刷新片段?我的问题是否正确?@Zuop - Swaminathan V
我只是猜测,你能否注释掉newInstance()方法并使用默认构造函数获取片段引用并尝试一次? - Raghavendra
显示剩余9条评论
1个回答

4

您需要设置一些事件监听逻辑,其中片段向活动注册其所需的事件。 片段应在创建时进行注册并在销毁时取消注册。

当事件发生时,活动会遍历已注册的侦听器列表,并通知它们该事件。

我在此答案中提供了一个详细的示例。

您可以使用事件总线库来简化活动和片段之间的某些连接。


我按照您回答中的链接中的说明操作,问题得到了解决...尽管我觉得从片段(Fragment)->活动(Activity)实现接口进行通信,然后又需要使用另一个接口来从活动(Activity)->片段(Fragment)进行通信,这很奇怪。 - Zuop
我的示例在活动上没有实现接口,但这样做仍然是单元测试和重用的良好实践。Google 的许多关于导航抽屉之类的示例都是这样做的。 - kris larson
哦,好的。我可能误解了你的回答,但现在它可以工作了。 尽管你在链接中说:“定义一个监听器接口。我通常会在活动内部定义一个内部接口”。 :) - Zuop
与其拥有一个独立的、独立的接口定义,我将接口定义放在活动内部,类似于内部类。这是有道理的,因为活动具有将接收事件回调的侦听器实现的集合。因此,活动“定义”接口,片段“实现”接口,例如extends Fragment implements MyActivity.EventListener。希望这使一切都更清晰明了。 - kris larson

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接