page.title=Creating Swipe Views with Tabs page.tags=viewpager,horizontal,paging,swipe view,tabs trainingnavtop=true @jd:body <div id="tb-wrapper"> <div id="tb"> <h2>This lesson teaches you to</h2> <ol> <li><a href="#horizontal-paging">Implement Swipe Views</a></li> <li><a href="#tabs">Add Tabs to the Action Bar</a></li> <li><a href="#swipe-tabs">Change Tabs with Swipe Views</a></li> <li><a href="#PagerTitleStrip">Use a Title Strip Instead of Tabs</a></li> </ol> <h2>You should also read</h2> <ul> <li><a href="{@docRoot}training/design-navigation/descendant-lateral.html">Providing Descendant and Lateral Navigation</a></li> <li><a href="{@docRoot}design/building-blocks/tabs.html">Android Design: Tabs</a></li> <li><a href="{@docRoot}design/patterns/swipe-views.html">Android Design: Swipe Views</a></li> </ul> <h2>Try it out</h2> <div class="download-box"> <a href="http://developer.android.com/shareables/training/EffectiveNavigation.zip" class="button">Download the sample app</a> <p class="filename">EffectiveNavigation.zip</p> </div> </div> </div> <p>Swipe views provide lateral navigation between sibling screens such as tabs with a horizontal finger gesture (a pattern sometimes known as horizontal paging). This lesson teaches you how to create a tab layout with swipe views for switching between tabs, or how to show a title strip instead of tabs.</p> <div class="note design"> <p><strong>Swipe View Design</strong></p> <p>Before implementing these features, you should understand the concepts and recommendations as described in <a href="{@docRoot}training/design-navigation/descendant-lateral.html">Designing Effective Navigation</a> and the <a href="{@docRoot}design/patterns/swipe-views.html">Swipe Views</a> design guide.</p> </div> <h2 id="horizontal-paging">Implement Swipe Views</h2> <p>You can create swipe views in your app using the {@link android.support.v4.view.ViewPager} widget, available in the <a href="{@docRoot}tools/support-library/index.html">Support Library</a>. The {@link android.support.v4.view.ViewPager} is a layout widget in which each child view is a separate page (a separate tab) in the layout.</p> <p>To set up your layout with {@link android.support.v4.view.ViewPager}, add a {@code <ViewPager>} element to your XML layout. For example, if each page in the swipe view should consume the entire layout, then your layout looks like this:</p> <pre> <?xml version="1.0" encoding="utf-8"?> <android.support.v4.view.ViewPager xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/pager" android:layout_width="match_parent" android:layout_height="match_parent" /> </pre> <p>To insert child views that represent each page, you need to hook this layout to a {@link android.support.v4.view.PagerAdapter}. There are two kinds of adapter you can use:</p> <dl> <dt>{@link android.support.v4.app.FragmentPagerAdapter}</dt> <dd>This is best when navigating between sibling screens representing a fixed, small number of pages.</dd> <dt>{@link android.support.v4.app.FragmentStatePagerAdapter}</dt> <dd>This is best for paging across a collection of objects for which the number of pages is undetermined. It destroys fragments as the user navigates to other pages, minimizing memory usage.</dd> </dl> <p>For example, here's how you might use {@link android.support.v4.app.FragmentStatePagerAdapter} to swipe across a collection of {@link android.app.Fragment} objects:</p> <pre> public class CollectionDemoActivity extends FragmentActivity { // When requested, this adapter returns a DemoObjectFragment, // representing an object in the collection. DemoCollectionPagerAdapter mDemoCollectionPagerAdapter; ViewPager mViewPager; public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_collection_demo); // ViewPager and its adapters use support library // fragments, so use getSupportFragmentManager. mDemoCollectionPagerAdapter = new DemoCollectionPagerAdapter( getSupportFragmentManager()); mViewPager = (ViewPager) findViewById(R.id.pager); mViewPager.setAdapter(mDemoCollectionPagerAdapter); } } // Since this is an object collection, use a FragmentStatePagerAdapter, // and NOT a FragmentPagerAdapter. public class DemoCollectionPagerAdapter extends FragmentStatePagerAdapter { public DemoCollectionPagerAdapter(FragmentManager fm) { super(fm); } {@literal @}Override public Fragment getItem(int i) { Fragment fragment = new DemoObjectFragment(); Bundle args = new Bundle(); // Our object is just an integer :-P args.putInt(DemoObjectFragment.ARG_OBJECT, i + 1); fragment.setArguments(args); return fragment; } {@literal @}Override public int getCount() { return 100; } {@literal @}Override public CharSequence getPageTitle(int position) { return "OBJECT " + (position + 1); } } // Instances of this class are fragments representing a single // object in our collection. public static class DemoObjectFragment extends Fragment { public static final String ARG_OBJECT = "object"; {@literal @}Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { // The last two arguments ensure LayoutParams are inflated // properly. View rootView = inflater.inflate( R.layout.fragment_collection_object, container, false); Bundle args = getArguments(); ((TextView) rootView.findViewById(android.R.id.text1)).setText( Integer.toString(args.getInt(ARG_OBJECT))); return rootView; } } </pre> <p>This example shows only the code necessary to create the swipe views. The following sections show how you can add tabs to help facilitate navigation between pages.</p> <h2 id="tabs">Add Tabs to the Action Bar</h2> <p>Action bar <a href="{@docRoot}design/building-blocks/tabs.html">tabs</a> offer users a familiar interface for navigating between and identifying sibling screens in your app.</p> <p>To create tabs using {@link android.app.ActionBar}, you need to enable {@link android.app.ActionBar#NAVIGATION_MODE_TABS}, then create several instances of {@link android.app.ActionBar.Tab} and supply an implementation of the {@link android.app.ActionBar.TabListener} interface for each one. For example, in your activity's {@link android.app.Activity#onCreate onCreate()} method, you can use code similar to this:</p> <pre> {@literal @}Override public void onCreate(Bundle savedInstanceState) { final ActionBar actionBar = getActionBar(); ... // Specify that tabs should be displayed in the action bar. actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS); // Create a tab listener that is called when the user changes tabs. ActionBar.TabListener tabListener = new ActionBar.TabListener() { public void onTabSelected(ActionBar.Tab tab, FragmentTransaction ft) { // show the given tab } public void onTabUnselected(ActionBar.Tab tab, FragmentTransaction ft) { // hide the given tab } public void onTabReselected(ActionBar.Tab tab, FragmentTransaction ft) { // probably ignore this event } }; // Add 3 tabs, specifying the tab's text and TabListener for (int i = 0; i < 3; i++) { actionBar.addTab( actionBar.newTab() .setText("Tab " + (i + 1)) .setTabListener(tabListener)); } } </pre> <p>How you handle the {@link android.app.ActionBar.TabListener} callbacks to change tabs depends on how you've constructed your content. But if you're using fragments for each tab with {@link android.support.v4.view.ViewPager} as shown above, the following section shows how to switch between pages when the user selects a tab and also update the selected tab when the user swipes between pages.</p> <h2 id="swipe-tabs">Change Tabs with Swipe Views</h2> <p>To switch between pages in a {@link android.support.v4.view.ViewPager} when the user selects a tab, implement your {@link android.app.ActionBar.TabListener} to select the appropriate page by calling {@link android.support.v4.view.ViewPager#setCurrentItem setCurrentItem()} on your {@link android.support.v4.view.ViewPager}:</p> <pre> {@literal @}Override public void onCreate(Bundle savedInstanceState) { ... // Create a tab listener that is called when the user changes tabs. ActionBar.TabListener tabListener = new ActionBar.TabListener() { public void onTabSelected(ActionBar.Tab tab, FragmentTransaction ft) { // When the tab is selected, switch to the // corresponding page in the ViewPager. mViewPager.setCurrentItem(tab.getPosition()); } ... }; } </pre> <p>Likewise, you should select the corresponding tab when the user swipes between pages with a touch gesture. You can set up this behavior by implementing the {@link android.support.v4.view.ViewPager.OnPageChangeListener} interface to change the current tab each time the page changes. For example:</p> <pre> {@literal @}Override public void onCreate(Bundle savedInstanceState) { ... mViewPager = (ViewPager) findViewById(R.id.pager); mViewPager.setOnPageChangeListener( new ViewPager.SimpleOnPageChangeListener() { {@literal @}Override public void onPageSelected(int position) { // When swiping between pages, select the // corresponding tab. getActionBar().setSelectedNavigationItem(position); } }); ... } </pre> <h2 id="PagerTitleStrip">Use a Title Strip Instead of Tabs</h2> <p>If you don't want to include action bar tabs and prefer to provide <a href="{@docRoot}design/building-blocks/tabs.html#scrollable">scrollable tabs</a> for a shorter visual profile, you can use {@link android.support.v4.view.PagerTitleStrip} with your swipe views.</p> <p>Below is an example layout XML file for an activity whose entire contents are a {@link android.support.v4.view.ViewPager} and a top-aligned {@link android.support.v4.view.PagerTitleStrip} inside it. Individual pages (provided by the adapter) occupy the remaining space inside the {@link android.support.v4.view.ViewPager}.</p> <pre> <android.support.v4.view.ViewPager xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/pager" android:layout_width="match_parent" android:layout_height="match_parent"> <android.support.v4.view.PagerTitleStrip android:id="@+id/pager_title_strip" android:layout_width="match_parent" android:layout_height="wrap_content" android:layout_gravity="top" android:background="#33b5e5" android:textColor="#fff" android:paddingTop="4dp" android:paddingBottom="4dp" /> </android.support.v4.view.ViewPager> </pre>