从片段中调用活动方法

时间:2012-09-30 08:12:58

标签: android android-fragments

尝试从片段中调用我的活动中的方法。我希望片段提供方法数据并在方法返回时获取数据。我希望实现类似于静态方法的调用,但不使用静态因为它在活动中产生问题。

片段的新内容所以我需要一个简单的教学解释!

谢谢!

15 个答案:

答案 0 :(得分:728)

从片段到激活:

((YourActivityClassName)getActivity()).yourPublicMethod();

从活动到片段:

FragmentManager fm = getSupportFragmentManager();

//if you added fragment via layout xml
YourFragmentClass fragment = (YourFragmentClass)fm.findFragmentById(R.id.your_fragment_id);
fragment.yourPublicMethod();

如果您在添加片段时通过代码添加了片段并使用了tag字符串,请改为使用findFragmentByTag

YourFragmentClass fragment = (YourFragmentClass)fm.findFragmentByTag("yourTag");

答案 1 :(得分:179)

您可能应该尝试将片段与活动分离,以防您想在其他地方使用它。您可以通过创建活动实现的界面来完成此操作。

所以你要定义一个如下界面:

假设您想为活动提供一个String并让它返回一个Integer:

public interface MyStringListener{
    public Integer computeSomething(String myString);
}

这可以在片段或单独的文件中定义。

然后你会让你的活动实现界面。

public class MyActivity extends FragmentActivity implements MyStringListener{

  @Override
  public Integer computeSomething(String myString){
   /** Do something with the string and return your Integer instead of 0 **/ 
   return 0;
  }

}

然后在你的片段中你将有一个MyStringListener变量,你可以在片段onAttach(Activity activity)方法中设置监听器。

public class MyFragment {

        private MyStringListener listener;

        @Override
        public void onAttach(Context context) {
            super.onAttach(context);
            try {
                listener = (MyStringListener) context;
            } catch (ClassCastException castException) {
                /** The activity does not implement the listener. */
            }
        }

    }

编辑(二○一五年十二月十七日):onAttach(Activity activity) is deprecated, use onAttach(Context context) instead, it works as intended

第一个答案肯定有效,但它将您当前的片段与宿主活动相结合。如果您想在另一个活动中使用它,那么将片段与主机活动分离是一种很好的做法。

答案 2 :(得分:25)

Kotlin 开发者

(activity as YourActivityClassName).methodName()

Java 开发人员

((YourActivityClassName) getActivity()).methodName();

答案 3 :(得分:13)

在我更了解片段如何工作之后更新。每个片段都属于父活动。所以只需使用:

getActivity().whatever

从片段中。这是一个更好的答案,因为你避免了超级演员阵容。如果您无法避免使用此解决方案的演员阵容,请使用以下内容。

============

你要做的就是投射到外部活动

((MainActivity) getActivity()).Method();

创建一个新实例将使android框架混乱,它将无法识别它。 另见:

https://stackoverflow.com/a/12014834/1984636

https://stackoverflow.com/a/2042829/1984636

答案 4 :(得分:9)

虽然我完全喜欢Marco的答案,但我认为你可以使用基于发布/订阅的框架来实现相同的结果是公平的,例如,如果你使用事件总线,你可以执行以下操作

片段:

EventBus.getDefault().post(new DoSomeActionEvent()); 

<强>活动:

 @Subscribe
onSomeActionEventRecieved(DoSomeActionEvent doSomeActionEvent){
//Do something

}

答案 5 :(得分:3)

在kotlin中你可以从下面的片段中调用activity方法:

var mainActivity: MainActivity = activity as MainActivity
        mainActivity.showToast() //Calling show toast method of activity

答案 6 :(得分:3)

我是这样做的:

先制作界面

interface NavigationInterface {
    fun closeActivity()
}

接下来确保活动实现接口并覆盖接口方法

class NotesActivity : AppCompatActivity(), NavigationInterface {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_notes)
        setSupportActionBar(findViewById(R.id.toolbar))
    }

    override fun closeActivity() {
        this.finish()
    }
}

然后确保在片段中创建接口侦听器

private lateinit var navigationInterface: NavigationInterface

override fun onCreateView(
        inflater: LayoutInflater, container: ViewGroup?,
        savedInstanceState: Bundle?
): View? {
    //establish interface communication
    activity?.let {
        instantiateNavigationInterface(it)
    }
    // Inflate the layout for this fragment
    return inflater.inflate(R.layout.fragment_notes_info, container, false)
}

private fun instantiateNavigationInterface(context: FragmentActivity) {
    navigationInterface = context as NavigationInterface
}

然后你可以像这样拨打电话:

view.findViewById<Button>(R.id.button_second).setOnClickListener {
    navigationInterface.closeActivity()
}

答案 7 :(得分:2)

要通过片段访问Activity中声明的函数,请使用界面,如marco的答案所示。

要通过您的活动访问Fragment中声明的功能,如果您没有标签或ID,则可以使用此功能

private void setupViewPager(ViewPager viewPager) {
    //fragmentOne,fragmentTwo and fragmentThree are all global variables
    fragmentOne= new FragmentOne();
    fragmentTwo= new FragmentTwo();
    fragmentThree = new FragmentThree();

    viewPagerAdapteradapter = new ViewPagerAdapter(getSupportFragmentManager());
    viewPagerAdapteradapter.addFragment(fragmentOne, "Frag1");
    viewPagerAdapteradapter.addFragment(fragmentTwo, "Frag2");
    viewPagerAdapteradapter.addFragment(fragmentThree, "Frag3");

    //viewPager has to be instantiated when you create the activity:
    //ViewPager viewPager = (ViewPager)findViewById(R.id.pager);
    //setupViewPager(viewPager);
    //Where R.id.pager is the id of the viewPager defined in your activity's xml page.

    viewPager.setAdapter(viewPagerAdapteradapter);


    //frag1 and frag2 are also global variables
    frag1 = (FragmentOne)viewPagerAdapteradapter.mFragmentList.get(0);
    frag2 = (FragmentTwo)viewPagerAdapteradapter.mFragmentList.get(1);;


    //You can use the variable fragmentOne or frag1 to access functions declared in FragmentOne


}

这是ViewpagerAdapterClass

    class ViewPagerAdapter extends FragmentPagerAdapter {
    public final List<Fragment> mFragmentList = new ArrayList<>();
    private final List<String> mFragmentTitleList = new ArrayList<>();

    public ViewPagerAdapter(FragmentManager manager) {
        super(manager);
    }

    @Override
    public Fragment getItem(int position) {
        return mFragmentList.get(position);
    }

    @Override
    public int getCount() {
        return mFragmentList.size();
    }

    public void addFragment(Fragment fragment, String title) {
        mFragmentList.add(fragment);
        mFragmentTitleList.add(title);
    }

    @Override
    public CharSequence getPageTitle(int position) {
        return mFragmentTitleList.get(position);
    }
}

这个答案适用于像我这样的新手。祝你有个美好的一天。

答案 8 :(得分:1)

这是From Fragment class ...

((KidsStoryDashboard)getActivity()).values(title_txt,bannerImgUrl);

此活动类代码......

 public void values(String title_txts, String bannerImgUrl) {
    if (!title_txts.isEmpty()) {

//Do something to set text 
    }
    imageLoader.displayImage(bannerImgUrl, htab_header_image, doption);
}

答案 9 :(得分:1)

我已经尝试了该线程中显示的所有方法,但没有一个对我有用,请尝试此方法。它对我有用。

((MainActivity) getContext().getApplicationContext()).Method();

答案 10 :(得分:1)

感谢@BIJAY_JHA和@Manaus。我使用Kotlin版本来调用我的signIn()方法,该方法位于Activity中,并且是从Fragment调用的。我在Android中使用导航架构,因此Listener接口模式不在Fragment中:

 (activity as MainActivity).signIn() 

答案 11 :(得分:0)

我一直在寻找最好的方法,因为并非我们想要调用的每个方法都位于具有相同Activity Parent的Fragment中。

在你的片段中

public void methodExemple(View view){

        // your code here

        Toast.makeText(view.getContext(), "Clicked clicked",Toast.LENGTH_LONG).show();
    }

在您的活动中

new ExempleFragment().methodExemple(context); 

答案 12 :(得分:0)

((YourActivityName)getActivity()).functionName();

示例:((SessionActivity)getActivity()).changeFragment();

注意:班级名称应为公开

答案 13 :(得分:0)

((your_activity) getActivity).method_name()

your_activity是您的活动的名称,method_name()是您要调用的方法的名称。

答案 14 :(得分:0)

对于Kotlin,请尝试

class DataForm : Fragment() {
    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        Tasks(this).getData()
    }

    fun getResponse(response: String) {
        // code
    }
}

class Tasks(private val context: Any) {
    fun getData() {

        val getContext = (context as DataForm).activity
        val getFragment = (context as DataForm)

        val responseListener = Response.Listener<String> { response ->
            getFragment.getResponse(response)
        }

        val errorListener = Response.ErrorListener { error ->
            error.printStackTrace();
        }

        val stringRequest = StringRequest(Request.Method.GET, url, responseListener, errorListener)
        Volley.newRequestQueue(getContext).add(stringRequest)
    }
}