鍍金池/ 教程/ Android/ 創(chuàng)建抽屜式導航(navigation drawer)
檢測常用的手勢
優(yōu)化layout的層級
用戶輸入
管理應用的內存
聯系人信息
開發(fā)輔助程序
Android多媒體
添加語音功能
顯示位置地址
提供向下與橫向導航
支持游戲控制器
訪問可穿戴數據層
處理多點觸控手勢
全屏沉浸式應用
為多線程創(chuàng)建管理器
數據保存
Intent的發(fā)送
更新Notification
優(yōu)化下載以高效地訪問網絡
打印
打包可穿戴應用
接收從其他App傳送來的數據
發(fā)送與接收消息
建立靈活動態(tài)的UI
處理鍵盤輸入
Building a Work Policy Controller
建立測試環(huán)境
創(chuàng)建表盤
分享文件
顯示Notification進度
實現自適應UI流(Flows)
使用設備管理策略增強安全性
使用能感知版本的組件
執(zhí)行網絡操作
建立文件分享
添加移動
更新你的Security Provider來對抗SSL漏洞利用
支持鍵盤導航
創(chuàng)建和監(jiān)視地理圍欄
發(fā)送并同步數據
使用BigView樣式
無線連接設備
提供向上導航與歷史導航
最小化定期更新造成的影響
實現向下的導航
支持不同的屏幕大小
Android 可穿戴應用
添加動畫
顯示聯系人頭像
使用OpenGL ES顯示圖像
處理輸入法可見性
分享文件
保持設備喚醒
淡化系統(tǒng)Bar
使用NFC分享文件
保存到Preference
Android聯系人信息與位置信息
創(chuàng)建標準的網絡請求
使用Drawables
管理Bitmap的內存使用
管理Activity的生命周期
按需加載視圖
傳輸資源
為可穿戴設備創(chuàng)建自定義UI
在一個線程中執(zhí)行一段特定的代碼
性能優(yōu)化
隱藏導航欄
創(chuàng)建目錄瀏覽器
為多種大小的屏幕進行規(guī)劃
View間漸變
使用觸摸手勢
高效加載大圖
使用CursorLoader在后臺加載數據
創(chuàng)建抽屜式導航(navigation drawer)
管理音頻焦點
創(chuàng)建后臺服務
創(chuàng)建功能測試
創(chuàng)建使用Material Design的應用
停止與重啟Activity
添加一個簡便的分享功能
啟動Activity時保留導航
TV應用清單
創(chuàng)建向后兼容的UI
?# 優(yōu)化自定義View
創(chuàng)建單元測試
在UI上顯示Bitmap
建立OpenGL ES的環(huán)境
構建表盤服務
JNI Tips
建立搜索界面
實現自定義View的繪制
使用HTTPS與SSL
按需操控BroadcastReceiver
分享簡單的數據
繪制形狀
Android位置信息
創(chuàng)建并運行可穿戴應用
執(zhí)行 Sync Adpater
獲取最后可知位置
創(chuàng)建 Android 項目
實現高效的導航
退出全屏的Activity
創(chuàng)建Card
兼容音頻輸出設備
同步數據單元
傳輸數據時避免消耗大量電量
保存到文件
緩存Bitmap
提供配置 Activity
調度重復的鬧鐘
實現輔助功能
重復的下載是冗余的
隱藏狀態(tài)欄
實現自定義的網絡請求
規(guī)劃界面和他們之間的關系
使用Sync Adapter傳輸數據
TV應用內搜索
響應觸摸事件
使用Google Cloud Messaging(已廢棄)
控制相機
Android網絡連接與云服務
請求分享一個文件
處理TV硬件
響應UI可見性的變化
使用網絡服務發(fā)現
指定輸入法類型
優(yōu)化電池壽命
創(chuàng)建TV應用
獲取聯系人列表
拖拽與縮放
啟動與停止線程池中的線程
創(chuàng)建 Sync Adpater
使用 WiFi P2P 服務發(fā)現
開始使用Material Design
代理至新的APIs
使用include標簽重用layouts
使得View可交互
高效顯示Bitmap
創(chuàng)建企業(yè)級應用
Fragments之間的交互
創(chuàng)建與執(zhí)行測試用例
綜合:設計我們的樣例 App
繪制表盤
建立簡單的用戶界面
自定義動畫
開發(fā)輔助服務
避免出現程序無響應ANR(Keeping Your App Responsive)
使用ViewPager實現屏幕滑動
設計高效的導航
Android分享操作(Building Apps with Content Sharing)
提供向后的導航
保持向下兼容
創(chuàng)建TV播放應用
縮放View
使用 WiFi 建立 P2P 連接
Android后臺任務
連接到網絡
為 Notification 添加頁面
使TV應用是可被搜索的
添加Action Bar
使用Material的主題
啟動另一個Activity
顯示正在播放卡片
適配不同的系統(tǒng)版本
輕松錄制視頻
創(chuàng)建可穿戴的應用
創(chuàng)建自定義的布局
重新創(chuàng)建Activity
使用CursorLoader執(zhí)行查詢任務
使用舊的APIs實現新API的效果
使用備份API
安全要點
Android入門基礎:從這里開始
保存并搜索數據
根據網絡連接類型來調整下載模式
使用Tabs創(chuàng)建Swipe視圖
SMP(Symmetric Multi-Processor) Primer for Android
解析 XML 數據
使用 Volley 傳輸網絡數據
建立ActionBar
Android交互設計
使用Intent修改聯系人信息
增加搜索功能
輕松拍攝照片
定義形狀
測試你的Activity
在 Notifcation 中接收語音輸入
與其他應用的交互
管理系統(tǒng)UI
追蹤手勢移動
Android界面設計
執(zhí)行 Android 程序
顯示確認界面
創(chuàng)建Lists與Cards
打印HTML文檔
創(chuàng)建TV應用
為多屏幕設計
定義Shadows與Clipping視圖
使用Fragment建立動態(tài)UI
接收Activity返回的結果
布局變更動畫
定位常見的問題
自定義ActionBar的風格
定義Layouts
發(fā)送簡單的網絡請求
啟動與銷毀Activity
與UI線程通信
非UI線程處理Bitmap
創(chuàng)建TV布局
提升Layout的性能
報告任務執(zhí)行狀態(tài)
判斷并監(jiān)測網絡連接狀態(tài)
兼容不同的設備
處理按鍵動作
優(yōu)化性能和電池使用時間
給其他App發(fā)送簡單的數據
Implementing App Restrictions
向后臺服務發(fā)送任務請求
展示Card翻轉動畫
管理ViewGroup中的觸摸事件
兼容不同的屏幕密度
通過藍牙進行調試
為可穿戴設備創(chuàng)建Notification
控制音量與音頻播放
獲取聯系人詳情
在表盤上顯示信息
提供向上的導航
滾動手勢動畫
幫助用戶在TV上找到內容
創(chuàng)建TV導航
為索引指定App內容
ActionBar的覆蓋疊加
Android Wear 上的位置檢測
保護安全與隱私的最佳策略
Ensuring Compatibility with Managed Profiles
解決云同步的保存沖突
獲取位置更新
創(chuàng)建List
測試程序
管理網絡的使用情況
為App內容開啟深度鏈接
推薦TV內容
建立一個Notification
管理音頻播放
設計表盤
拍照
處理控制器輸入動作
判斷并監(jiān)測設備的底座狀態(tài)與類型
處理查詢的結果
保存到數據庫
支持多個游戲控制器
創(chuàng)建 Stub Content Provider
使得ListView滑動順暢
處理數據層的事件
創(chuàng)建TV應用的第一步
使得你的App內容可被Google搜索
將 Notification 放成一疊
創(chuàng)建 Stub 授權器
暫停與恢復Activity
管理設備的喚醒狀態(tài)
Android圖像與動畫
打印照片
云同步
創(chuàng)建TV直播應用
為Notification賦加可穿戴特性
提供一個Card視圖
建立請求隊列(RequestQueue)
適配不同的語言
創(chuàng)建詳情頁
測試UI組件
接收其他設備的文件
創(chuàng)建自定義View
建立第一個App
創(chuàng)建2D Picker
監(jiān)測電池的電量與充電狀態(tài)
打印自定義文檔
抽象出新的APIs
通知提示用戶
獲取文件信息
運用投影與相機視角
在IntentService中執(zhí)行后臺任務
多線程操作
創(chuàng)建一個Fragment
添加Action按鈕
在不同的 Android 系統(tǒng)版本支持控制器
維護兼容性
發(fā)送文件給其他設備
創(chuàng)建TV游戲應用
創(chuàng)建自定義的View類
代碼性能優(yōu)化建議
Intent過濾
適配不同的屏幕

創(chuàng)建抽屜式導航(navigation drawer)

編寫:Lin-H - 原文: http://developer.android.com/training/implementing-navigation/nav-drawer.html

Navigation drawer是一個在屏幕左側邊緣顯示導航選項的面板。大部分時候是隱藏的,當用戶從屏幕左側劃屏,或在top level模式的app中點擊action bar中的app圖標時,才會顯示。

這節(jié)課敘述如何使用Support Library中的DrawerLayout API,來實現navigation drawer。

Navigation Drawer 設計:在你決定在你的app中使用Navigation Drawer之前,你應該先理解在Navigation Drawer design guide中定義的使用情況和設計準則。

創(chuàng)建一個Drawer Layout

要添加一個navigation drawer,在你的用戶界面layout中聲明一個用作root view(根視圖)的DrawerLayout對象。在DrawerLayout中為屏幕添加一個包含主要內容的view(當drawer隱藏時的主layout),和其他一些包含navigation drawer內容的view。

例如,下面的layout使用了有兩個子視圖(child view)的DrawerLayout:一個FrameLayout用來包含主要內容(在運行時被Fragment填入),和一個navigation drawer使用的ListView。

<android.support.v4.widget.DrawerLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/drawer_layout"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <!-- 包含主要內容的 view -->
    <FrameLayout
        android:id="@+id/content_frame"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />
    <!-- navigation drawer(抽屜式導航) -->
    <ListView android:id="@+id/left_drawer"
        android:layout_width="240dp"
        android:layout_height="match_parent"
        android:layout_gravity="start"
        android:choiceMode="singleChoice"
        android:divider="@android:color/transparent"
        android:dividerHeight="0dp"
        android:background="#111"/>
</android.support.v4.widget.DrawerLayout>

這個layout展示了一些layout的重要特點:

  • 主內容view(上面的FrameLayout),在DrawerLayout必須是第一個子視圖,因為XML的順序代表著Z軸(垂直于手機屏幕)的順序,并且drawer必須在內容的前端。

  • 主內容view被設置為匹配父視圖的寬和高,因為當navigation drawer隱藏時,主內容表示整個UI部分。

  • drawer視圖(ListView)必須使用android:layout_gravity屬性指定它的horizontal gravity。為了支持從右邊閱讀的語言(right-to-left(RTL) language),指定它的值為"start"而不是"left"(當layout是RTL時drawer在右邊顯示)。

  • drawer視圖以dp為單位指定它的寬和高來匹配父視圖。drawer的寬度不能大于320dp,這樣用戶總能看到部分主內容。

初始化Drawer List

在你的activity中,首先要做的事就是要初始化drawer的item列表。這要根據你的app內容來處理,但是一個navigation drawer通常由一個ListView組成,所以列表應該通過一個Adapter(例如ArrayAdapterSimpleCursorAdapter)填入。

例如,如何使用一個字符串數組(string array)來初始化導航列表(navigation list):

public class MainActivity extends Activity {
    private String[] mPlanetTitles;
    private DrawerLayout mDrawerLayout;
    private ListView mDrawerList;
    ...

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        mPlanetTitles = getResources().getStringArray(R.array.planets_array);
        mDrawerLayout = (DrawerLayout) findViewById(R.id.drawer_layout);
        mDrawerList = (ListView) findViewById(R.id.left_drawer);

        // 為list view設置adapter
        mDrawerList.setAdapter(new ArrayAdapter<String>(this,
                R.layout.drawer_list_item, mPlanetTitles));
        // 為list設置click listener
        mDrawerList.setOnItemClickListener(new DrawerItemClickListener());

        ...
    }
}

這段代碼也調用了setOnItemClickListener()來接收navigation drawer列表的點擊事件。下一節(jié)會說明如何實現這個接口,并且當用戶選擇一個item時如何改變內容視圖(content view)。

處理導航的點擊事件

當用戶選擇drawer列表中的item,系統(tǒng)會調用在setOnItemClickListener()中所設置的OnItemClickListeneronItemClick()。

onItemClick()方法中做什么,取決于你如何實現你的app結構(app structure)。在下面的例子中,每選擇一個列表中的item,就插入一個不同的Fragment到主內容視圖中(FrameLayout元素通過R.id.content_frame ID辨識):

private class DrawerItemClickListener implements ListView.OnItemClickListener {
    @Override
    public void onItemClick(AdapterView parent, View view, int position, long id) {
        selectItem(position);
    }
}

/** 在主內容視圖中交換fragment */
private void selectItem(int position) {
    // 創(chuàng)建一個新的fragment并且根據行星的位置來顯示
    Fragment fragment = new PlanetFragment();
    Bundle args = new Bundle();
    args.putInt(PlanetFragment.ARG_PLANET_NUMBER, position);
    fragment.setArguments(args);

    // 通過替換已存在的fragment來插入新的fragment
    FragmentManager fragmentManager = getFragmentManager();
    fragmentManager.beginTransaction()
                   .replace(R.id.content_frame, fragment)
                   .commit();

    // 高亮被選擇的item, 更新標題, 并關閉drawer
    mDrawerList.setItemChecked(position, true);
    setTitle(mPlanetTitles[position]);
    mDrawerLayout.closeDrawer(mDrawerList);
}

@Override
public void setTitle(CharSequence title) {
    mTitle = title;
    getActionBar().setTitle(mTitle);
}

監(jiān)聽打開和關閉事件

要監(jiān)聽drawer的打開和關閉事件,在你的DrawerLayout中調用setDrawerListener(),并傳入一個DrawerLayout.DrawerListener的實現。這個接口提供drawer事件的回調例如onDrawerOpened()onDrawerClosed()。

但是,如果你的activity包含有action bar可以不用實現DrawerLayout.DrawerListener,你可以繼承ActionBarDrawerToggle來替代。ActionBarDrawerToggle實現了DrawerLayout.DrawerListener,所以你仍然可以重寫這些回調。這么做也能使action bar圖標和 navigation drawer的交互操作變得更容易(在下節(jié)詳述)。

Navigation Drawer design guide中所述,當drawer可見時,你應該修改action bar的內容,比如改變標題和移除與主文字內容相關的action item。下面的代碼向你說明如何通過ActionBarDrawerToggle類的實例,重寫DrawerLayout.DrawerListener的回調方法來實現這個目的:

public class MainActivity extends Activity {
    private DrawerLayout mDrawerLayout;
    private ActionBarDrawerToggle mDrawerToggle;
    private CharSequence mDrawerTitle;
    private CharSequence mTitle;
    ...

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ...

        mTitle = mDrawerTitle = getTitle();
        mDrawerLayout = (DrawerLayout) findViewById(R.id.drawer_layout);
        mDrawerToggle = new ActionBarDrawerToggle(this, mDrawerLayout,
                R.drawable.ic_drawer, R.string.drawer_open, R.string.drawer_close) {

            /** 當drawer處于完全關閉的狀態(tài)時調用 */
            public void onDrawerClosed(View view) {
                super.onDrawerClosed(view);
                getActionBar().setTitle(mTitle);
                invalidateOptionsMenu(); // 創(chuàng)建對onPrepareOptionsMenu()的調用
            }

            /** 當drawer處于完全打開的狀態(tài)時調用 */
            public void onDrawerOpened(View drawerView) {
                super.onDrawerOpened(drawerView);
                getActionBar().setTitle(mDrawerTitle);
                invalidateOptionsMenu(); // 創(chuàng)建對onPrepareOptionsMenu()的調用
            }
        };

        // 設置drawer觸發(fā)器為DrawerListener
        mDrawerLayout.setDrawerListener(mDrawerToggle);
    }

    /* 當invalidateOptionsMenu()調用時調用 */
    @Override
    public boolean onPrepareOptionsMenu(Menu menu) {
        // 如果nav drawer是打開的, 隱藏與內容視圖相關聯的action items
        boolean drawerOpen = mDrawerLayout.isDrawerOpen(mDrawerList);
        menu.findItem(R.id.action_websearch).setVisible(!drawerOpen);
        return super.onPrepareOptionsMenu(menu);
    }
}

下一節(jié)會描述ActionBarDrawerToggle的構造參數,和處理與action bar圖標交互所需的其他步驟。

使用App圖標來打開和關閉

用戶可以在屏幕左側使用劃屏手勢來打開和關閉navigation drawer,但是如果你使用action bar,你也應該允許用戶通過點擊app圖標來打開或關閉。并且app圖標也應該使用一個特殊的圖標來指明navigation drawer的存在。你可以通過使用上一節(jié)所說的ActionBarDrawerToggle來實現所有的這些操作。

要使ActionBarDrawerToggle起作用,通過它的構造函數創(chuàng)建一個實例,需要用到以下參數:

  • Activity用來容納drawer。

  • DrawerLayout。

  • 一個drawable資源用作drawer指示器。 標準的navigation drawer可以在Download the Action Bar Icon Pack獲的

  • 一個字符串資源描述"打開抽屜"操作(便于訪問)

  • 一個字符串資源描述"關閉抽屜"操作(便于訪問)

那么,不論你是否創(chuàng)建了用作drawer監(jiān)聽器的ActionBarDrawerToggle的子類,你都需要在activity生命周期中的某些地方根據你的ActionBarDrawerToggle來調用。

public class MainActivity extends Activity {
    private DrawerLayout mDrawerLayout;
    private ActionBarDrawerToggle mDrawerToggle;
    ...

    public void onCreate(Bundle savedInstanceState) {
        ...

        mDrawerLayout = (DrawerLayout) findViewById(R.id.drawer_layout);
        mDrawerToggle = new ActionBarDrawerToggle(
                this,                  /* 承載 Activity */
                mDrawerLayout,         /* DrawerLayout 對象 */
                R.drawable.ic_drawer,  /* nav drawer 圖標用來替換'Up'符號 */
                R.string.drawer_open,  /* "打開 drawer" 描述 */
                R.string.drawer_close  /* "關閉 drawer" 描述 */
                ) {

            /** 當drawer處于完全關閉的狀態(tài)時調用 */
            public void onDrawerClosed(View view) {
                super.onDrawerClosed(view);
                getActionBar().setTitle(mTitle);
            }

            /** 當drawer處于完全打開的狀態(tài)時調用 */
            public void onDrawerOpened(View drawerView) {
                super.onDrawerOpened(drawerView);
                getActionBar().setTitle(mDrawerTitle);
            }
        };

        // 設置drawer觸發(fā)器為DrawerListener
        mDrawerLayout.setDrawerListener(mDrawerToggle);

        getActionBar().setDisplayHomeAsUpEnabled(true);
        getActionBar().setHomeButtonEnabled(true);
    }

    @Override
    protected void onPostCreate(Bundle savedInstanceState) {
        super.onPostCreate(savedInstanceState);
        // 在onRestoreInstanceState發(fā)生后,同步觸發(fā)器狀態(tài).
        mDrawerToggle.syncState();
    }

    @Override
    public void onConfigurationChanged(Configuration newConfig) {
        super.onConfigurationChanged(newConfig);
        mDrawerToggle.onConfigurationChanged(newConfig);
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // 將事件傳遞給ActionBarDrawerToggle, 如果返回true,表示app 圖標點擊事件已經被處理
        if (mDrawerToggle.onOptionsItemSelected(item)) {
          return true;
        }
        // 處理你的其他action bar items...

        return super.onOptionsItemSelected(item);
    }

    ...
}

一個完整的navigation drawer例子,可以在原文頁面頂端的sample下載