从 ViewPager Activity 访问 Fragment 的方法
Posted
技术标签:
【中文标题】从 ViewPager Activity 访问 Fragment 的方法【英文标题】:Access a method of a Fragment from the ViewPager Activity 【发布时间】:2013-07-15 04:18:42 【问题描述】:我有一个带有 setName() 方法的片段,它通过 setText 函数更改 EditText 文本。
通过 ViewPager 从承载该片段的 Activity 调用该方法的最佳方式是什么?
换句话说,我如何通过 ViewPager 从托管该片段的 Activity 访问片段的方法(例如,更改该片段的布局)?
我问这个是因为我尝试了几种方法,但总是出错。
【问题讨论】:
您可以尝试在您的Activity 类中使用setName()
方法创建Fragment 的实例变量。实例化时将变量分配给片段。然后你可以从你的活动中调用fragment.setName()
我有这个问题,但是通过存储的引用调用 fragment.setName() 不是问题。问题是从片段中,getActivity() 返回 null,因此无法访问视图。不过,我可以从活动中访问视图,并且片段可以在配置时将信息放入属于片段的视图中。但后来,getActivity() 返回 null,这可能是 viewPager 控制上下文导致 getActivity 实际上无效的原因吗?那么需要从 Viewpager 请求视图吗?我在问。
【参考方案1】:
您可以在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);
你传递你需要的任何参数,就像调用一个常规方法一样。请注意,这仅在您从包含 ViewPager
或 FragmentActivity
的片段中调用方法时才有效。如果你想做相反的事情,从片段到活动,你需要使用接口。
【讨论】:
是的,我已经知道了。但是,如果我尝试使用 Frag1 fragment = (Frag1) mAppSectionsAdapter.getItem(0); 之类的东西来实例化它怎么办?那么我应该如何在没有空指针异常的情况下使用该方法? 就像 fragment.getView() 总是为空! 您想从您的适配器获取参考?我不在电脑前,但如果有问题,我可以稍后发布代码。 对我有任何帮助请...***.com/questions/39907625/… 您的答案在视图寻呼机中始终对我有用,谢谢你,只为你我现在为这个问题加注了星标......谢谢你:)【参考方案2】:我知道这有点晚了,但我遇到了同样的问题,如果你已经解决了它也许会对其他人有所帮助。
我发现 ViewPager 的第一个问题是几乎不可能获得对片段的引用。片段是在 getItem() 中动态创建的,因此您无法设置 ID,并且它们会由 swicher 自动重新标记,因此您也无法通过标记找到它。有一些方法可以做到这一点,但它们都是解决方法。 (Update data in ListFragment as part of ViewPager)
我解决它的方法本质上是使用双重回调。 Fragment A 有一个由 Main Activity 实现的接口,Main Activity 有一个由 Fragment B 实现的接口。 Fragment A 中的按钮叮当声调用 Main Activity 中的回调函数,然后调用 Fragment B 中的回调。请看下面的代码。我希望我发布了所有内容,它会有所帮助。顺便说一句,我只用 ViewPager 试过这个,但我认为它适用于任何类型的 Fragment 通信。
主要活动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_
android:layout_
tools:context=".MainActivity" >
<android.support.v4.view.PagerTitleStrip
android:id="@+id/pager_title_strip"
android:layout_
android:layout_
android:layout_gravity="top"
android:background="#33b5e5"
android:paddingBottom="4dp"
android:paddingTop="4dp"
android:textColor="#fff" />
</android.support.v4.view.ViewPager>
片段 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;
片段A xml:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_
android:layout_ >
<TextView
android:id="@+id/textView1"
android:layout_
android:layout_
android:layout_alignParentLeft="true"
android:layout_alignParentTop="true"
android:text="This is Fragment A" />
<Button
android:id="@+id/btnCallB"
android:layout_
android:layout_
android:layout_alignParentLeft="true"
android:layout_below="@+id/textView1"
android:text="Call Fragment B" />
</RelativeLayout>
片段 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_
android:layout_ >
<TextView
android:id="@+id/textView1"
android:layout_
android:layout_
android:layout_alignParentLeft="true"
android:layout_alignParentTop="true"
android:text="This is Fragment B" />
</RelativeLayout>
【讨论】:
旋转时不起作用。因为当您旋转活动并重新创建片段时,这也会重新创建您的 SectionsPagerAdapter 但它从不调用 getItem(Android 会在不调用 getItem 的情况下重新创建片段),这意味着 passOnToB 保持为空。 我编辑了我的代码,这样它就可以在屏幕旋转后工作。诀窍是不仅在 getItem() 中设置回调,而且还在寻呼机适配器中包含一个自定义方法,该方法将在片段中搜索并在找到正确片段时手动设置回调。现在您只需要检查是否设置了回调,如果没有在 passOn 方法中设置它。我希望这有点清楚。 我有什么想法吗? ***.com/questions/39907625/…【参考方案3】:片段
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();
【讨论】:
【参考方案4】:最好的方法是调用
CallingFragmentName fragment = (CallingFragmentName) viewPager
.getAdapter()
.instantiateItem(viewPager, viewPager.getCurrentItem());
它会重新实例化你的调用Fragment,这样就不会抛出空指针异常。
【讨论】:
这应该是公认的答案。非常感谢。你是救生员。 对我很有帮助,谢谢。 你节省了我的时间。谢谢。完美的解决方案以上是关于从 ViewPager Activity 访问 Fragment 的方法的主要内容,如果未能解决你的问题,请参考以下文章
Android 从Activity中往ViewPager里的fragment传递数据