首先,若是要实现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
在Fragment与Activity产生交互以前,须要先让他们 '链接'到一块儿。android
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
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
这里须要借助 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
咱们在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的数据传入接口方法中。
咱们让 MainActivity 实现刚刚在BFragment中定义的接口Callback
, 而且调用 BFragment的 setCallback(Callback callback),由于MainActivity类实现了Callback
接口,因此参数能够传入MainActivity自己。
在实现Callback
的onBtnClickListener(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();
}
}
}
复制代码
运行,看效果:
正如咱们预想的那样,咱们在点击 Fragment中的 Button,成功使 Activity产生一个Toast,做为响应,而且可以获得传过去的字符串数据。
(完~)