Fragmentが所属するActivityを取得します。
Fragmentに割り当てたタグを取得します。ViewはIDによって管理されますが、FragmentはIDによる管理とタグによる管理が可能です。UIを持たないFragmentはIDを割り当てられないため、タグによって管理します。
FragmentからFragmentManagerを取得します。
FragmentTransactionを取得します。
Fragmentに対する、いわゆる画面遷移には、「トランザクション」という考え方を適用します。ここで取得したFragmentTransactionのメソッドを呼び出しても、即座にそれらを適用するわけではありません。後述する、FragmentTransaction#commit()を呼び出して、初めて適用されます。
トランザクションをバックスタックに追加します。現在のトランザクションをcommit()後に参照したい場合に追加しておきます。任意でこのトランザクションに名前を付けられますが、必要がなければnullを渡します。
トランザクションをコミットします。もし、addToBackStack(String)でトランザクションをバックスタックに追加している場合、このメソッドの戻り値としてバックスタックエントリのIDを」返します。
addToBackStack()でスタックしたトランザクションを1つ戻します。ユーザによるバックキー押下と同様の処理を任意で行えます。
ここからは、Fragmentを使用するアプリのための、設定方法と実装方法について解説します。
Fragmentを画面にレイアウトする方法は、Viewと同様、レイアウトXMLに設定するか、ソースコード上で追加するかを選べます。
レイアウトエディタには、「Fragment」という新たなレイアウトが追加されており、これをドラッグ&ドロップするだけで、レイアウトが行えます。
レイアウトエディタでレイアウトした結果が以下です。
Fragmentをレイアウトするうえで重要なのは、Fragmentの幅と高さ、およびウェイトの設定です。
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="horizontal" android:layout_width="fill_parent" android:layout_height="fill_parent"> <fragment android:id="@+id/fragment1" android:layout_height="match_parent" android:layout_width="0dp" 【1】 android:layout_weight="1" 【2】 android:name="com.example.android.fragment.BookmarkFragment" /> <fragment android:id="@+id/fragment2" android:layout_height="match_parent" android:layout_width="0dp" 【1】 android:layout_weight="3" 【3】 android:name="com.example.android.fragment.WebFragment" /> </LinearLayout>
【1】では、左右に配置されるFragmentのそれぞれの幅を0dpに設定します。
【2】では、左側のリストは幅のウェイトを1に設定します。
【3】では、右側のブラウザは幅のウェイトを3に設定します。このようにすることで、画面サイズによらず、リスト:ブラウザ=1:3に保てます。
レイアウトXMLで指定したFragmentを実装します。ここでは、今回のサンプルの中から、オーバーライドした2つのメソッドを取り上げて解説します。
@Override public View onCreateView(LayoutInflater inflater,ViewGroup container, Bundle savedInstanceState) { List<String> list = new LinkedList<String>(); for (String[] bookmark : BOOKMARKS) { list.add(bookmark[0]); } ArrayAdapter<String> adapter = new ArrayAdapter<String>(getActivity(),android.R.layout.simple_list_item_1, list); ListView listView = new ListView(getActivity()); listView.setAdapter(adapter); return listView; }
上記は、onCreateView()でFragmentに表示するViewを作成・初期化しているコードです。このクラスはListFragmentを継承しているため、メソッドに渡されている引数を一切使用していませんが、Fragmentから継承させた独自Fragmentの場合、別に作成したレイアウトと、LayoutInfraterで紐付ける必要があります。
先にも述べたとおり、ListFragmentの場合はonActivityCreated()で初期化する方が賢明です。
@Override public void onListItemClick(ListView l, View v, int position,long id) { if (position == 0) { getFragmentManager().popBackStack(); } else { String url = BOOKMARKS[position][1]; WebFragment web = new WebFragment(url); FragmentTransaction ft = getFragmentManager().beginTransaction(); ft.replace(R.id.fragment2, web); ft.addToBackStack(null); ft.commit(); } }
上記はリストがクリックされた際に呼び出されるListFragment固有のイベントメソッドです。
クリックされた場所が先頭なら、バックスタックからポップし、そうでなければ、新しいFragmentを作成し、既存のものと置き換え、トランザクションをバックスタックに追加しています。
これだけの処理で、Android 2.xでは実現できなかった画面内の画面遷移が行えるようになります。
Android 3.0から導入されたFragmentsを紹介しました。次回は、この機能がAndroid 1.6以降で使用可能になるAndroid Compatibility Packageを紹介します。
なお、Fragmentは実はもっと奥が深いのですが、今回はここまでとし、スマートフォン向けとタブレット向けが統合されるIce Cream Sandwichが登場してから、Fragmentsとそれがもたらす可能性について再度取り上げる予定です。
ちなみにJavaのSDKではありませんが、今回のようなFragmentsの考え方と違い、画面サイズごとに別のスタイルシートを当てて画面の大きさの違いを解決するという方法があります。Flex SDK 4.5.1の「CSS Media Query」機能です。詳細は、以下の記事をご参照ください。
Copyright © ITmedia, Inc. All Rights Reserved.