随着 Google I/O 2015,新的 Android Design Support Library 也出现了。
Android Design Support Library 给开发者带来了一些重要的 Material Design 组件,并且向下兼容到 Android 2.1,Navigation View就是其中之一,可用于方便地创建导航抽屉。
效果如下:
因为需要向下兼容,所以以下内容的 Activity 均继承于AppCompatActivity,使用的主题的父主题均为 AppCompat 的主题。
基本步骤
通过Navigation View创建导航抽屉首先自然需要引入这个支持包:
compile 'com.android.support:design:22.2.0'
布局文件中加入以下代码:
<span class="c"><!-- 需要呈现的内容 --></span>
<span class="nt"><android.support.design.widget.NavigationView</span>
<span class="na">android:layout_width=</span><span class="s">"wrap_content"</span>
<span class="na">android:layout_height=</span><span class="s">"match_parent"</span>
<span class="na">android:layout_gravity=</span><span class="s">"start"</span>
<span class="na">app:headerLayout=</span><span class="s">"@layout/drawer_header"</span>
<span class="na">app:menu=</span><span class="s">"@menu/drawer"</span><span class="nt">/></span>
</android.support.v4.widget.DrawerLayout>
</div>
其中`app:headerLayout`用于指定一个任意的布局,作为导航抽屉的顶部,如效果图中的紫色带 Username 字样部分。
而`app:menu`用于指定导航抽屉的菜单项,具体代码如下:
<div class="codehilite">
<menu xmlns:android=“http://schemas.android.com/apk/res/android">
<span class="nt"><group</span> <span class="na">android:checkableBehavior=</span><span class="s">"single"</span><span class="nt">></span>
<span class="nt"><item</span>
<span class="na">android:id=</span><span class="s">"@+id/nav_home"</span>
<span class="na">android:icon=</span><span class="s">"@drawable/ic_dashboard"</span>
<span class="na">android:title=</span><span class="s">"Home"</span> <span class="nt">/></span>
<span class="nt"><item</span>
<span class="na">android:id=</span><span class="s">"@+id/nav_messages"</span>
<span class="na">android:icon=</span><span class="s">"@drawable/ic_event"</span>
<span class="na">android:title=</span><span class="s">"Messages"</span> <span class="nt">/></span>
<span class="nt"><item</span>
<span class="na">android:id=</span><span class="s">"@+id/nav_friends"</span>
<span class="na">android:icon=</span><span class="s">"@drawable/ic_headset"</span>
<span class="na">android:title=</span><span class="s">"Friends"</span> <span class="nt">/></span>
<span class="nt"><item</span>
<span class="na">android:id=</span><span class="s">"@+id/nav_discussion"</span>
<span class="na">android:icon=</span><span class="s">"@drawable/ic_forum"</span>
<span class="na">android:title=</span><span class="s">"Discussion"</span> <span class="nt">/></span>
<span class="nt"></group></span>
<span class="nt"><item</span> <span class="na">android:title=</span><span class="s">"Sub items"</span><span class="nt">></span>
<span class="nt"><menu></span>
<span class="nt"><item</span>
<span class="na">android:icon=</span><span class="s">"@drawable/ic_dashboard"</span>
<span class="na">android:title=</span><span class="s">"Sub item 1"</span> <span class="nt">/></span>
<span class="nt"><item</span>
<span class="na">android:icon=</span><span class="s">"@drawable/ic_forum"</span>
<span class="na">android:title=</span><span class="s">"Sub item 2"</span> <span class="nt">/></span>
<span class="nt"></menu></span>
<span class="nt"></item></span>
</menu>
</div>
其中通过`<group android:checkableBehavior="single">`设定一组菜单项为至多只有一个可被选中,非常适合用于通过导航抽屉切换呈现的 Fragment,若需要默认选中一个菜单项则只需要给指定 item 加上`android:checked="true"`即可。并且可以通过子菜单的形式显示分割线和子标题。
到此为止布局文件部分就完成了,接下来只需要在Activity中设置监听器即可:
<div class="codehilite">
mDrawerLayout = (DrawerLayout) findViewById(R.id.drawer_layout);
NavigationView navigationView = (NavigationView) findViewById(R.id.nav_view); if (navigationView != null) { navigationView.setNavigationItemSelectedListener(new NavigationView.OnNavigationItemSelectedListener() { @Override public boolean onNavigationItemSelected(MenuItem menuItem) { //切换相应 Fragment 等操作 menuItem.setChecked(true); mDrawerLayout.closeDrawers(); return false; } }); }
</div>
至此导航抽屉就基本完成了,但 Material Design 中建议导航抽屉应该在 status bar 上也显示,而 Android 5.0 及以上版本支持这一特性,因此还需要进一步处理。
# 针对 Android 5.0 及以上版本 {#toc_1}
首先由 window 来绘制 status bar 的背景,否则 status bar 为系统默认背景(原生是黑色背景),而 AppCompat 的主题已带如下属性:
<div class="codehilite">
<item name=“android:windowDrawsSystemBarBackgrounds”>true</item>
</div>
确保 Android 5.0 以上版本使用的主题中并没有把该属性设置成 false 即可。
而通过 window 来绘制 status bar 的背景并不一定是我们需要的最终结果,因为 status bar 的背景在需要呈现的内容上和导航抽屉上不一定是一样的,而通过 window 来绘制则肯定是一样的,因此把 window 绘制的 status bar 的背景设置为透明,在 Android 5.0 以上版本使用的主题中加入如下代码:
<div class="codehilite">
<item name=“android:statusBarColor”>@android:color/transparent</item>
</div>
接下来只需要`DrawerLayout`来接管 status bar 的背景的绘制工作即可,给布局文件中的`DrawerLayout`加入如下属性:
<div class="codehilite">
android:fitsSystemWindows=“true”
</div>
此时导航抽屉已经能够显示在 status bar 上。
至此理论上 Android 2.1 以上版本都能正常显示导航抽屉了,完整代码可参照 Github 上的一个 [Demo][1]。

通过`Navigation View`能够实现上图所示的导航抽屉,但有两处若不设置依旧会占着控件,十分不合理,分别是图标以及 4 处的子菜单标题。所以若需要没有图标的项目或单独加分割线则还得考虑使用自定义布局的形式实现导航抽屉。
但`Navigation View`优点依旧很明显,首先设置的图标经测试会被改变成合适的颜色,无论是选中还是未选中状态,无需任何代码即可达成下图效果,若使用自定义布局的形式可能需要设计未选中和选中两套图标或自己写代码动态改变颜色。

若还是觉得`Navigation View`不够好,又觉得自定义太麻烦,并且只需要兼容到 Android 2.3.3 或以上版本的话可以考虑使用使用 [MaterialDrawer][2] 开源控件,十分强大,能够实现`Navigation View`所有能够实现的效果,并且自由度更高,我通过这个开源控件完成了我毕业设计中所需的两个导航抽屉,具体效果如下图所示:

# 参考资料 {#toc_2}
[Android Design Support Library][3](Android Developers Blog)
[1]: https://github.com/chrisbanes/cheesesquare/
[2]: https://github.com/mikepenz/MaterialDrawer
[3]: http://android-developers.blogspot.fi/2015/05/android-design-support-library.html
💬 评论