1. 程式人生 > >Android基礎之使用Fragment適應不同螢幕和解析度(分享)

Android基礎之使用Fragment適應不同螢幕和解析度(分享)

最近事情很忙,一個新專案趕著出來,但是很多功能都要重新做,一直在編寫程式碼、Debug。今天因為一個新程式要使用Fragment來做,雖然以前也使用過Fragment,不過沒有仔細研究,今天順道寫篇文章記錄一下Fragment的使用。這文章主要參考了Android官網的介紹。

Fragment是Android3.0後增加的新控制元件,有點類似於Activity元件,也是用來承載各種View元素。Google增加這個玩意的目的是為了平板電腦裡面可以複用部分顯示的View,只要寫好一個View,可以同時在手機和平板等不同尺寸的裝置上使用。而且這個轉換過程系統幫你搞定了。下面我們分類說說Fragment的使用。

1、為何使用Fragment
下面是Android官網為了說明Fragment作用的例子:一個新聞應用可以在螢幕左側使用一個fragment來展示一個文章的列表,然後在螢幕右側使用另一個fragment來展示一篇文章--2個fragment並排顯示在相同的一個activity中,並且每一個fragment擁有它自己的一套生命週期回撥方法,並且處理它們自己的使用者輸入事件。 因此, 取代使用一個activity來選擇一篇文章而另一個activity來閱讀文章的方式,使用者可以在同一個activity中選擇一篇文章並且閱讀, 如圖所示:

當執行在一個特別大的螢幕時(例如平板電腦),應用可以在Activity A中嵌入2個fragment。但是如果在一個正常尺寸的螢幕(例如手機)上,沒有足夠的空間同時供2個fragment用, 因此, Activity A會僅包含文章列表的fragment, 而當用戶選擇一篇文章時, 它會啟動ActivityB,它包含閱讀文章的fragment.因此, 應用可以同時支援上圖中的2種設計模式。

fragment是一個為了介面檢視可以重用的元件,因為fragment定義了它自己的佈局, 以及通過使用它自己的生命週期回撥方法定義了它自己的行為,你可以將fragment包含到多個activity中. 這點特別重要, 因為這允許你將你的使用者體驗適配到不同的螢幕尺寸.舉個例子,你可能會僅當在螢幕尺寸足夠大時,在一個activity中包含多個fragment,並且,當不屬於這種情況時,會啟動另一個單獨的,使用不同fragment的activity。

2、建立Fragment
要建立一個Fragment, 必須建立一個 Fragment 的子類 (或者繼承自一個已存在的它的子類)。Fragment類的程式碼看起來很像 Activity 。它包含了和activity類似的回撥方法, 例如onCreate()、 onStart()、onPause()以及 onStop()。事實上, 如果你準備將一個現成的Android應用轉換到使用fragment,可能只需簡單的將程式碼從你的activity的回撥方法分別移動到你的fragment的回撥方法即可。
下面我們看看Fragment的子類有哪些:

•DialogFragment 顯示一個浮動的對話方塊。用這個類來建立一個對話方塊,是使用在Activity類的對話方塊工具方法之外的一個好的選擇, 因為你可以將一個fragment對話方塊合併到activity管理的fragment back stack中,允許使用者返回到一個之前曾被摒棄的fragment。

•ListFragment 顯示一個由一個adapter(例如 SimpleCursorAdapter)管理的專案的列表, 類似於ListActivity。 它提供一些方法來管理一個list view, 例如onListItemClick()回撥來處理點選事件。

•PreferenceFragment 顯示一個 Preference物件的層次結構的列表, 類似於PreferenceActivity。 這在為你的應用建立一個"設定"activity時有用處。

3、Fragment生命週期
下面是Fragment的生命週期圖,摘自Android官網。

繼承了Fragment相關類之後,我們需要重寫幾個回撥函式,實現相關功能,下面面3個函式是我們一般都會重寫的回撥方法:

複製程式碼 程式碼如下:
//Edited by mythou
 public class MainFragment extends Fragment
{
    //建立Fragment
    public void onCreate()
    {}

    //返回View給Activity使用
    public View onCreateView()
    {}

    public void onPause()
    {}

}


•onCreate() 當建立fragment時, 系統呼叫該方法. 在實現程式碼中,應當初始化想要在fragment中保持的必要元件, 當fragment被暫停或者停止後可以恢復。

•onCreateView() fragment第一次繪製它的使用者介面的時候, 系統會呼叫此方法. 為了繪製fragment的UI,此方法必須返回一個View, 這個view是你的fragment佈局的根view. 如果fragment不提供UI, 可以返回null。

•onPause() 使用者將要離開fragment時,系統呼叫這個方法作為第一個指示(然而它不總是意味著fragment將被銷燬)。 在當前使用者會話結束之前,通常應當在這裡提交任何應該持久化的變化(因為使用者有可能不會返回)。
除了這些,對比上面的生命週期,還可以看到其他的回撥方法。我們可以根據需要重寫相關方法。

4、建立Fragment
fragment通常用來作為一個activity的使用者介面的一部分,並將它的layout提供給activity。為了給一個fragment提供一 個layout,你必須實現 onCreateView()回撥方法, 當到了fragment繪製它自己的layout的時候,Android系統呼叫它。你的此方法的實現程式碼必須返回一個你的fragment的 layout的根view。

另外,如果你的fragment是ListFragment的子類,它的預設實現是返回從onCreateView()返回一個ListView,所以一般情況下不必實現它。

從onCreateView()返回的View, 也可以從一個layout的xml資原始檔中讀取並生成。為了幫助你這麼做, onCreateView() 提供了一個LayoutInflater 物件。

下面看個例子,從XML載入檢視View,跟我們一般的View裡面動態載入解析XML生成View一樣。

複製程式碼 程式碼如下:
//Edited by mythou
public static class ExampleFragment extends Fragment {
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        // Inflate the layout for this fragment
        return inflater.inflate(R.layout.example_fragment, container, false);
    }
}

傳入onCreateView()的container引數是你的fragmentlayout將被插入的父ViewGroup(來自activity的layout)  savedInstanceState 引數是一個Bundle, 如果fragment是被恢復的,它提供關於fragment的之前的例項的資料,inflate() 方法有3個引數:

•RSID:想要載入的layout的resource ID。

•container:載入的layout的父ViewGroup. 傳入container是很重要的, 目的是為了讓系統接受所要載入的layout的根view的layout引數, 由它將掛靠的父view指定。

•布林量:指示在載入期間, 展開的layout是否應當附著到ViewGroup(第二個引數)  (在這個例子中, 指定了false, 因為系統已經把展開的layout插入到container –傳入true會在最後的layout中建立一個多餘的view group)
 

5、把Fragment新增到Activity
通常地, fragment為宿主activity提供UI的一部分, 被作為activity的整個viewhierarchy的一部分被嵌入, 有2種方法你可以新增一個fragment到activity layout:

在activity的layout檔案中宣告fragment:
在這種情況下,你可以像為View一樣, 為fragment指定layout屬性.例子是一個有2個fragment的activity的layout:

複製程式碼 程式碼如下:
//Edited by mythou
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="horizontal"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <fragment android:name="com.example.news.ArticleListFragment"
            android:id="@+id/list"
            android:layout_weight="1"
            android:layout_width="0dp"
            android:layout_height="match_parent" />
    <fragment android:name="com.example.news.ArticleReaderFragment"
            android:id="@+id/viewer"
            android:layout_weight="2"
            android:layout_width="0dp"
            android:layout_height="match_parent" />
</LinearLayout>

<fragment> 中的 android:name屬性指定了在layout中例項化的Fragment類。當系統建立這個activity layout時,它例項化每一個在layout中指定的fragment,並呼叫每一個上的onCreateView()方法,來獲取每一個 fragment的layout。系統將從fragment返回的 View直接插入到<fragment>元素所在的地方。注意:每一個fragment都需要一個唯一的標識,如果activity重啟,系統可以用來恢復fragment(並且你也可以用來捕獲fragment來處理事務,例如移除它) 。

有3種方法來為一個fragment提供一個標識:
•為android:id 屬性提供一個唯一ID。
•為android:tag 屬性提供一個唯一字串。
•如果以上2個你都沒有提供, 系統使用容器view的ID。

撰寫程式碼將fragment新增到一個已存在的ViewGroup.
當activity執行的任何時候, 都可以將fragment新增到activity layout。只需簡單的指定一個需要放置fragment的ViewGroup。為了在你的 activity中操作fragment事務(例如新增,移除,或代替一個fragment),必須使用來自FragmentTransaction 的API。可以按如下方法,從你的Activity取得一個 FragmentTransaction 的例項:

複製程式碼 程式碼如下:
FragmentManager fragmentManager =getFragmentManager();
FragmentTransaction fragmentTransaction =fragmentManager.beginTransaction();

然後你可以使用 add() 方法新增一個fragment, 指定要新增的fragment和要插入的view。
複製程式碼 程式碼如下:
fragment = newExampleFragment();
fragmentTransaction.add(R.id.fragment_container,fragment);
fragmentTransaction.commit();

add()的第一個引數是fragment要放入的ViewGroup, 由resource ID指定,第二個引數是需要新增的fragment。一旦用FragmentTransaction做了改變,為了使改變生效,必須呼叫commit()。