內容目錄
建立專案
建立一個有 EmptyActivity 的專案,其名稱為 FragmentLab01,Minimun SDK為 API 21
處理相依性
這個例子並不需要新增任何的 Dependencies
建立第一個 Fragment,並加入 Activity
建立 FirstFragment
建立一個名為 FirstFragment 的 BlankFragment,修改其相關的檔案內容。
FirstFragment.kt
class FirstFragment : Fragment() { override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? { return inflater.inflate(R.layout.fragment_first, container, false) } }
fragment_first.xml
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" android:gravity="center_horizontal" tools:context=".FirstFragment"> <TextView android:layout_width="wrap_content" android:layout_height="match_parent" android:id="@+id/tv_first" android:textSize="20sp" android:text="First text"/> </LinearLayout>
將 FirstFragment 加入 MainActivity
我們需要修改 activity_main.xml,建立一個 FragmentContainerView 來放我們的 Fragment,我們將這個 FragmentContainerView 命名為 first_fragment_container_view。
activity_main.xml
<?xml version="1.0" encoding="utf-8"?> <LinearLayout 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" android:orientation="vertical" android:gravity="center_horizontal" android:padding="8dp" tools:context=".MainActivity" > <androidx.fragment.app.FragmentContainerView android:id="@+id/first_fragment_container_view" android:name="app.kirin.android.fragmentlab01.FirstFragment" android:layout_width="match_parent" android:layout_height="match_parent"/> </LinearLayout>
執行結果
此時執行 app 可以看到顯示 First text
建立第二個 Fragment,並加入 Activity 中
建立 SecondFragment
建立一個名為 SecondFragment 的 BlankFragment,修改其相關的檔案內容。
SecondFragment.kt
lass SecondFragment : Fragment() { override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? { return inflater.inflate(R.layout.fragment_second, container, false) } }
fragment_second.xml
我們把背景設為灰色(#EEEEEE),來和 FirstFragment 區別開來。
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" android:gravity="center_horizontal" android:background="#EEEEEE" tools:context=".SecondFragment"> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:id="@+id/tv_second" android:textSize="20sp" android:text="Second text"/> <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:id="@+id/btn_second" android:textSize="20sp" android:text="Second" /> </LinearLayout>
將 SecondFragment 加入 MainActivity
我們需要修改 activity_main.xml,加入另一個 FragmentContainerView 來放我們的 Fragment,我們將這個 FragmentContainerView 命名為 second_fragment_container_view。
然後調整兩個 FragmentContainerView 的 layout_height 及 layout_weight,使其能上下均分畫面。
activity_main.xml
<?xml version="1.0" encoding="utf-8"?> <LinearLayout 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" android:orientation="vertical" android:gravity="center_horizontal" android:padding="8dp" tools:context=".MainActivity" > <androidx.fragment.app.FragmentContainerView android:id="@+id/first_fragment_container_view" android:name="app.kirin.android.fragmentlab01.FirstFragment" android:layout_width="match_parent" android:layout_height="0dp" android:layout_weight="1"/> <androidx.fragment.app.FragmentContainerView android:id="@+id/second_fragment_container_view" android:name="app.kirin.android.fragmentlab01.SecondFragment" android:layout_width="match_parent" android:layout_height="0dp" android:layout_weight="1"/> </LinearLayout>
執行結果
此時執行 app 可以看到上下排列的 2 個 Fragment。
讓 2 個 Fragment 產生互動
我們要讓 SecondFragment 的 Button 能影響 FirstFragment 的 TextView。
在 MainActivity 中設定 Button 行為
MainActivity.kt
class MainActivity : AppCompatActivity() { override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_main) } override fun onResume() { super.onResume() val btnSecond = findViewById<Button>(R.id.btn_second) btnSecond.setOnClickListener { val tvFirst= findViewById<TextView>(R.id.tv_first) tvFirst.setText("Action by Second Button") } } }
備註
如果要在 MainActivity 控制 Fragment 中的元件,不能在 onCreate() 中處理,必須要在 onResume() 中處理,才不會出現元件的 null-pointer 錯誤;但是如果把 FragmentContainerView 改為 fragment,就可以在 onCreate 中處理,原因未明,有同再研究。
執行結果
按下 SecondFragment 的按鈕,可以改變 FirstFragment 的文字。
建立第三個 Fragment,在執行時期替換第一個 Fragment
建立 ThirdFragment
建立一個名為 ThirdFragment 的 BlankFragment,修改其相關的檔案內容。
ThirdFragment.kt
class ThirdFragment : Fragment() { override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? { return inflater.inflate(R.layout.fragment_third, container, false) } }
fragment_thrid.kt
<?xml version="1.0" encoding="utf-8"?> <FrameLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context=".ThirdFragment"> <TextView android:layout_width="match_parent" android:layout_height="match_parent" android:textSize="24sp" android:text="Third"/> </FrameLayout>
在 MainActivity 中設定替換 Fragment 的行為
將原來改變 TextView 的動作改為替換 FirstFragment 為 ThirdFragment。
class MainActivity : AppCompatActivity() { override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_main) } override fun onResume() { super.onResume() val btnSecond = findViewById<Button>(R.id.btn_second) btnSecond.setOnClickListener { val fragmentManager = supportFragmentManager val transaction = fragmentManager.beginTransaction() transaction.replace(R.id.first_fragment_container_view, ThirdFragment()) transaction.commit() } } }
執行結果
按下 SecondFragment 裏的 Button,就會發現 FirstFragment 己替換為 ThirdFragment。
參考資料
developers Guide: Fragment
developers Reference: androidx.fragment.FragmentTransaction
舊的版本是:android.app.FragmentTransaction (Deprecated)
Comments