从ViewPager Activity访问Fragment的方法

时间:2013-07-16 19:55:12

标签: android android-fragments nullpointerexception android-viewpager android-fragmentactivity

我有一个带有方法setName()的Fragment,它通过setText函数更改EditText文本。

通过ViewPager从托管该片段的活动调用该方法的最佳方法是什么?

换句话说,如何通过ViewPager从托管该片段的Activity访问Fragment的方法(例如,更改该片段的布局)?

我问这个是因为我尝试了几种方法,但总是有错误。

4 个答案:

答案 0 :(得分:7)

我知道这有点晚了,但我遇到了同样的问题,如果你已经解决了,也许会有所帮助。

我在ViewPager中发现的第一个问题是几乎不可能获得对片段的引用。片段是在getItem()中动态创建的,因此您无法设置ID并且它们会被swicher自动重新调整,因此您也无法通过标记找到它。有一些方法可以做到这一点,但它们都是变通方法。 (Update data in ListFragment as part of ViewPager

我解决它的方式基本上是使用双回调。片段A具有由主活动实现的接口,主活动具有由片段B实现的接口。 Fragment A中的按钮clink调用Main Activity中的回调函数,然后调用Fragment B中的回调。查看下面的代码。我希望我发布了一切,这将有所帮助。顺便说一句,我只用ViewPager尝试过这个,但我认为它适用于任何类型的片段通信。

Main Avtivity java:

import android.os.Bundle;
import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentActivity;
import android.support.v4.app.FragmentManager;
import android.support.v4.app.FragmentPagerAdapter;
import android.support.v4.view.ViewPager;

public class MainActivity extends FragmentActivity implements FragmentA.Caller {

    SectionsPagerAdapter mSectionsPagerAdapter;
    ViewPager mViewPager;
    PassCallToB passOnToB = null;
    FragmentManager myManager = null;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mSectionsPagerAdapter = new SectionsPagerAdapter(getSupportFragmentManager());
        mViewPager = (ViewPager) findViewById(R.id.pager);
        mViewPager.setAdapter(mSectionsPagerAdapter);
    }

    public class SectionsPagerAdapter extends FragmentPagerAdapter {

        public SectionsPagerAdapter(FragmentManager fm) {
            super(fm);
            MyManager = fm;
        }

        @Override
        public Fragment getItem(int position) {
            Fragment fragment = null;
            if(position == 0) {
                fragment = new FragmentA();
            } else if (position == 1) {
                fragment = new FragmentB();
                passOnToB = (PassCallToB)fragment;
            }
            return fragment;
        }

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

        @Override
        public CharSequence getPageTitle(int position) {
            switch (position) {
            case 0:
                return "Frag A";
            case 1:
                return "Frag B";
            }
            return null;
        }

        public void setCallback() {
            List<Fragment> frags = myManager.getFragments();
            for(Fragment fragment : frags) {
                if(fragment instanceof FragmentB){
                    passOnToB = (PassCallToB)fragment;
                }
            }
        }
    }

    public interface PassCallToB {
        public void passItOn();
    }

    @Override
    public void CallB() {
        if(passOnToB instanceof Fragment)
            passOnToB.passItOn();
        else {
            mSectionsPagerAdapter.setCallback();
            passOnToB.passItOn();
        }
    }
}

主要活动xml:

<android.support.v4.view.ViewPager xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/pager"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity" >

    <android.support.v4.view.PagerTitleStrip
        android:id="@+id/pager_title_strip"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_gravity="top"
        android:background="#33b5e5"
        android:paddingBottom="4dp"
        android:paddingTop="4dp"
        android:textColor="#fff" />

</android.support.v4.view.ViewPager>

Fragment A java:

import android.app.Activity;
import android.os.Bundle;
import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentActivity;
import android.view.LayoutInflater;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewGroup;
import android.widget.Button;

public class FragmentA extends Fragment {

    Button btnCallB = null;

    Caller listener = null;

    public FragmentA() {
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle inState) {
        View rootView = inflater.inflate(R.layout.fragment_a, container, false);

        btnCallB = (Button)rootView.findViewById(R.id.btnCallB);
        btnCallB.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View view) {
                listener.CallB();
            }

        });

        return rootView;
    }

    public interface Caller {
        public void CallB();
    }

    @Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        if (activity instanceof FragmentActivity) {
          listener = (Caller) activity;
        } else {
          throw new ClassCastException(activity.toString() + " must implemenet listener");
        }
    }

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

Fragment A xml:

<RelativeLayout 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" >


    <TextView
        android:id="@+id/textView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentLeft="true"
        android:layout_alignParentTop="true"
        android:text="This is Fragment A" />

    <Button
        android:id="@+id/btnCallB"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentLeft="true"
        android:layout_below="@+id/textView1"
        android:text="Call Fragment B" />

</RelativeLayout>

Fragment B Java:

import android.os.Bundle;
import android.support.v4.app.Fragment;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Toast;

public class FragmentB extends Fragment implements MainActivity.PassCallToB {

    public FragmentB() {
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle inState) {
        View rootView = inflater.inflate(R.layout.fragment_b, container, false);
        return rootView;
    }

    @Override
    public void passItOn() {
        Toast.makeText(getActivity(), "Hello from B", Toast.LENGTH_SHORT).show();

    }
}

片段B xml:

<RelativeLayout 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" >

    <TextView
        android:id="@+id/textView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentLeft="true"
        android:layout_alignParentTop="true"
        android:text="This is Fragment B" />

</RelativeLayout>

答案 1 :(得分:3)

您可以访问ViewPager所拥有的片段中的公共方法。您需要(1)在创建它时存储对Fragment的引用,并将其添加到将支持寻呼机适配器的列表中,或者(2)您需要从寻呼机适配器获取对该片段的引用本身。例如:

Fragment fragmentA = null; //instance variable

fragmenA = new Fragment(); //whereever you instantiate your fragment

如果你的方法是

public void setName(String args){
    //do something
}

您要做的就是从ViewPager

所持的片段的引用中调用该方法
fragmentA.setName(args);

你传递所需的任何参数,就像调用常规方法一样。请注意,仅当您从包含ViewPagerFragmentActivity的片段中调用方法时,此方法才有效。如果你想反过来,片段到活动,你需要使用一个inerface。

答案 2 :(得分:2)

片段

private static FragmentName instance;

public static synchronized FragmentName getInstance()
{
    return instance;
}

@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
instance=this;
....
}

public void methodName()
{...}

活动

FragmentName.getInstance().methodName();

答案 3 :(得分:2)

执行此操作的最佳方法,只需调用 CallingFragmentName片段=(CallingFragmentName)viewPager                         .getAdapter()                         .instantiateItem(viewPager,viewPager.getCurrentItem()); 它将重新实例化您的调用Fragment,以便它不会引发空指针异常。