Android学习笔记13-Acitivty与Fragment通讯

1,概述

首先,若是要实现Activity与其余的Fragment之间实现通讯,那么这个Fragment须要拥有一个独立的Layout文件,以便重用,而且以代码添加的方式出如今Activity中。
这么作的好处在于咱们能够把相关的业务逻辑写在这个Fragment中,减小它所依附的Activity中的代码。php

官方文档中,建议咱们在实现Fragment与其余Activity/Fragment通讯的时候使用ViewModel,在ViewModel存储咱们的模型数据,经过在咱们想要创建联系的Fragment与Activity中共享这个ViewModel 实现数据通讯(其中一个Fragment/Activity更新了ViewModel,其余关联了此ViewModel的Fragment/Activity也会变化)。html

不过咱们今天不说经过ViewModel的方式,感兴趣的同窗能够经过官方文档看看。咱们来讲说经过Interface来实现Activity与其余的Fragment之间的通讯。这种方式与使用ViewModel比起来,须要咱们作更多的工做。java

2,准备工做:将Fragment添加到Activity

在Fragment与Activity产生交互以前,须要先让他们 '链接'到一块儿。android

(1) 新建一个项目,主活动是 MainActivity,咱们将它的布局改为以下 :

activity_main.xmlapp

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity"
    android:id="@+id/fragment_container">

</FrameLayout>
复制代码

主活动布局中只有一个FrameLayout,并为它指定一个id 。ide

(2) 建立一个Fragment

public class BFragment extends Fragment {

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

   @Nullable
   @Override
   public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
       View rootView = inflater.inflate(R.layout.fragment_b_layout, container, false);
       Button sendMsgBtn = rootView.findViewById(R.id.send_msg_btn);
       sendMsgBtn.setOnClickListener(new View.OnClickListener() {
           @Override
           public void onClick(View v) {
               //发送消息给Activity
           }
       });
       return rootView;
   }
}
复制代码

并为这个Fragment建立一个布局文件 fragment_b_layout.xml布局

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent">

   <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="BUTTON" android:id="@+id/send_msg_btn"/>
</RelativeLayout>
复制代码

Fragment布局中只有一个Button,咱们将实现经过点击Button,将Fragment中的数据传到Activity中。this

(3),将Fragment显示到Activity上

这里须要借助 FragmentManager来管理Fragment,而且将Fragment显示在Activity上。spa

public class MainActivity extends AppCompatActivity{

   @Override
   protected void onCreate(Bundle savedInstanceState) {
       super.onCreate(savedInstanceState);
       setContentView(R.layout.activity_main);
       FragmentManager fm = getSupportFragmentManager();
       FragmentTransaction transaction = fm.beginTransaction();

       Fragment fragment = fm.findFragmentById(R.id.fragment_container);
       if (fragment == null) {
           fragment = new BFragment();
           transaction.add(R.id.fragment_container, fragment);
           transaction.commit();
       }
   } 
}
复制代码

OK,准备作完了,如今运行一下,页面应该是这样的 :3d

20190509235554-image.png

3,实现通讯

(1),定义接口

咱们在BFragment中,定义一个接口,

public class BFragment extends Fragment {

    ...
    Callback mCallback;

    public interface Callback{
        public void onBtnClickListener(String str);
    }

    public void setCallback(Callback callback) {
        mCallback = callback;
    }
    
    ....
 }
复制代码

接口中只有一个方法,而且有一个String 类型的参数。

为button,设置一个点击事件: 当点击按钮时,调用接口的方法:

@Override
public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        
        ...
        sendMsgBtn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (mCallback != null ) {
                    mCallback.onBtnClickListener("我来自fragment B");
                }
            }
        });
        
        ...
    }
复制代码

咱们将想要传递给Activity的数据传入接口方法中。

(2),实现接口

咱们让 MainActivity 实现刚刚在BFragment中定义的接口Callback, 而且调用 BFragment的 setCallback(Callback callback),由于MainActivity类实现了Callback接口,因此参数能够传入MainActivity自己。
在实现CallbackonBtnClickListener(String str)方法中,咱们弹出含有信息的 toast.

public class MainActivity extends AppCompatActivity implements BFragment.Callback{
	...
  
  @Override
  public void onBtnClickListener(String str) {
      Toast.makeText(this, str, Toast.LENGTH_SHORT).show();
  }
  
  ...
  
  @Override
  protected void onCreate(Bundle savedInstanceState) {
      super.onCreate(savedInstanceState);
      ...//省略刚刚写过的代码
      if (fragment == null) {
          fragment = new BFragment();
          ((BFragment) fragment).setCallback(this);
          transaction.add(R.id.fragment_container, fragment);
          transaction.commit();
      }
  }

  
}
复制代码

运行,看效果:

20190510001608-image.png

正如咱们预想的那样,咱们在点击 Fragment中的 Button,成功使 Activity产生一个Toast,做为响应,而且可以获得传过去的字符串数据。

(完~)

相关文章
相关标签/搜索