MaterialDesign系列文章(十)NavigationView和DrawerLayout的使用

不怕跌倒,所以飞翔

今天我们看看NavigationView这个控件的使用方法(侧滑抽屉的效果)

使用NavigationView实现抽屉效果一定跳不过DrawerLayout这个控件,但是因为不是本文得重点,但是你错了,我也会讲的...

本文的知识点:

  • DrawerLayout的使用
  • NavigationView的使用

DrawerLayout的使用

简单的使用方法

  • 先来说下布局文件吧!
    • 首先这个控件必须有一个子类,只有子类添加了android:layout_gravity="start/end"之后才能保证侧滑
    • 可以左右都有侧滑菜单,但是上下不可添加
      所以布局一般都是这个样子的
<android.support.v4.widget.DrawerLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="com.hejin.materialdesign.activity.NavigationActivity">

    <!--主页面-->
    <android.support.design.widget.CoordinatorLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent">

    </android.support.design.widget.CoordinatorLayout>

    <!--侧边栏的页面-->
    <android.support.design.widget.NavigationView
        android:id="@+id/nav_view"
        android:layout_width="wrap_content"
        android:layout_height="match_parent"
        android:layout_gravity="start"
        android:fitsSystemWindows="true"
        app:headerLayout="@layout/nav_header_navigation_view"
        app:menu="@menu/activity_navigation_view_drawer"/>
</android.support.v4.widget.DrawerLayout>

这里可以先不要去看NavigationView的设置,这里你随便换成一个其他控件都可以,但是layout_gravity这个属性要有,布局基本上也就这么多,讲到代码的时候还要讲一个类,后面再说

关联ToolBar

一般使用DrawerLayout都话连同ToolBar一起使用(但是不妨碍有特例,好比我...)下面是关联的代码:

        ActionBarDrawerToggle toggle = new ActionBarDrawerToggle(this, mDL, mToolbar, R.string.navigation_drawer_open, R.string.navigation_drawer_close) {
            @Override
            public void onDrawerOpened(View drawerView) {
                super.onDrawerOpened(drawerView);
            }

            @Override
            public void onDrawerClosed(View drawerView) {
                super.onDrawerClosed(drawerView);
            }

            @Override
            public void onDrawerSlide(View drawerView, float slideOffset) {
                super.onDrawerSlide(drawerView, slideOffset);
            }
        };
        mDL.addDrawerListener(toggle);
        toggle.syncState();

其实ActionBarDrawerToggle其实就是一个监听,这么理解是不是好多了,这里一般都会重写这三个方法,很好理解,一个打开,一个关闭,一个滑动的时候调用(我觉得这个一般都是加一些动画的,嗯嗯嗯)

一些需要注意的地方

  • 当你真的想关联ToolBar的时候,android:layout_gravity="start/end"这个属性一定要设置成start或者left斗则会报错(这个我真的不知道为什么?希望知道的告知小弟一声).
  • 当你点击左边的侧边栏的时候,可能会存在刷新主页面ToolBar的操作(这里我的处理方法是从新写一遍ToolBar的设置,因为用到的是ToolBar,但是你要是使用ActionBar的话,一定要在开打或者关闭的时候调用getActivity().invalidateOptionsMenu(); 方法去刷新menu,因为menu是一直存在的)
  • 还有几个方法注意下:(我为什么要说这两个方法呢?因为在按返回键的时候,你要判断抽屉是否打开,如果是打开的情况下,要先关闭抽屉的,这样用户体验才好)
    • DrawerLayout.isDrawerOpen(GravityCompat.START);这个方法是判断左边或者右边的抽屉是否打开的方法
    • DrawerLayout.closeDrawer(GravityCompat.START);这个方法是关闭左边或者右边的抽屉的.
      @Override
      public void onBackPressed() {
          DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
          if (drawer.isDrawerOpen(GravityCompat.START)) {
              drawer.closeDrawer(GravityCompat.START);
          } else {
              super.onBackPressed();
          }
      }
    
    这个就是判断返回按钮的时候DrawerLayout是否打开的代码了

基本上能用到的内容我都讲了一遍,我基本上就用到这么多,等到以后要是用到了别的话在进行补充!

NavigationView的使用

一般用在侧滑面板上,也可以用到其他的地方,自己想吧!样子是什么样子的呢?就是上面一块放头像区域,下面是一堆条目布局,其实要是你自己想要去实现的话也是可以的,所以说这个控件没有太多好讲的

简单的使用方法:

    <android.support.design.widget.NavigationView
        android:id="@+id/nav_view"
        android:layout_width="wrap_content"
        android:layout_height="match_parent"
        android:layout_gravity="start"
        android:fitsSystemWindows="true"
        app:headerLayout="@layout/nav_header_navigation_view"
        app:menu="@menu/activity_navigation_view_drawer"/>

这里要说的就以下这两个问题:

  • app:headerLayout 这个是外部关联头像那部分的布局,其实就是一个正常的布局而已,但是我看官网的Demo上面的高度用的是@dimen/nav_header_height但是我点不进去(这个应该是一个固定高度),但是你要是使用wrap_content的话也是可以的
  • app:menu 这个主要是关联底部条目的,这个menu文件的写法,可以参考ToolBar那篇文章,里面对这个介绍的很详细,这里呢,只贴一下代码吧!
<?xml version="1.0" encoding="utf-8"?>
<menu xmlns:android="http://schemas.android.com/apk/res/android">
  <group android:checkableBehavior="single">
      <item
          android:id="@+id/nav_camera"
          android:icon="@drawable/ic_menu_camera"
          android:title="Import"/>
      <item
          android:id="@+id/nav_gallery"
          android:icon="@drawable/ic_menu_gallery"
          android:title="Gallery"/>
      <item
          android:id="@+id/nav_slideshow"
          android:icon="@drawable/ic_menu_slideshow"
          android:title="Slideshow"/>
      <item
          android:id="@+id/nav_manage"
          android:icon="@drawable/ic_menu_manage"
          android:title="Tools"/>
  </group>

  <item android:title="Communicate">
      <menu>
          <item
              android:id="@+id/nav_share"
              android:icon="@drawable/ic_menu_share"
              android:title="Share"/>
          <item
              android:id="@+id/nav_send"
              android:icon="@drawable/ic_menu_send"
              android:title="Send"/>
      </menu>
  </item>
</menu>

一些常见的问题:

1.头像的点击事件:

这个点击事件没有具体的API去直接实现,所以我们只有通过另一种方法去实现:

解决方法:动态添加head,这样也确保了整个head中的控件都能实现点击事件

        View drawview = navigationView.inflateHeaderView(R.layout.nav_header_main);
        ImageVIewhead_iv = (ImageVIew) drawview.findViewById(R.id.head_iv);
        head_iv.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Log.d("Log:","头部被点击了!");
            }

2.底部menu的点击事件:

这个相对于头像的点击事件就简单多了

直接调用navigationView.setNavigationItemSelectedListener(@Nullable OnNavigationItemSelectedListener listener);这个API就可以了,在回调中区分Id就可以了(这里别忘了关闭DrawerLayout).

@Override
    public boolean onNavigationItemSelected(MenuItem item) {
        // Handle navigation view item clicks here.
        int id = item.getItemId();

        if (id == R.id.nav_camera) {
            // Handle the camera action
        } else if (id == R.id.nav_gallery) {

        } else if (id == R.id.nav_slideshow) {

        } else if (id == R.id.nav_manage) {

        } else if (id == R.id.nav_share) {

        } else if (id == R.id.nav_send) {

        }
        DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
        drawer.closeDrawer(GravityCompat.START);
        return true;
    }

这一系列文章的地址,希望对大家有帮助

项目地址

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,033评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,725评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,473评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,846评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,848评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,691评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,053评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,700评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,856评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,676评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,787评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,430评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,034评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,990评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,218评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,174评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,526评论 2 343

推荐阅读更多精彩内容