首先,若是要實現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,做爲響應,而且可以獲得傳過去的字符串數據。
(完~)