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

你传递你需要的任何参数,就像调用一个常规方法一样。请注意,这仅在您从包含 ViewPagerFragmentActivity 的片段中调用方法时才有效。如果你想做相反的事情,从片段到活动,你需要使用接口。

【讨论】:

是的,我已经知道了。但是,如果我尝试使用 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 的方法的主要内容,如果未能解决你的问题,请参考以下文章

从Activity设置子片段的ViewPager页面

Android 从Activity中往ViewPager里的fragment传递数据

ViewPager2+Fragment

是否可以在片段类中设置 ViewPager?

友盟页面统计 - 关于Viewpager中的Fragment的生命周期

何时更新ViewPager的信息?