android汽车之家顶部滑动菜单,Material Design 实战 之第二弹——滑动菜单详解实战(DrawerLayout NavigationView)...

本文主要是介绍android汽车之家顶部滑动菜单,Material Design 实战 之第二弹——滑动菜单详解实战(DrawerLayout NavigationView)...,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本模块共有六篇文章,参考郭神的《第一行代码》,对Material Design的学习做一个详细的笔记,大家可以一起交流一下:

文章提要与总结

1. DrawerLayout

控件用处:实现滑动菜单

1.1 首先它是一个布局,在布局中允许放入两个直接子控件,

第一个子控件是主屏幕中显示的内容;

第二个子控件是滑动菜单中显示的内容;

关于第二个子控件有一点需要注意,layout_gravity这个属性是必须指定的:left right start

1.2 添加导航按钮:

1.2.1 首先调用findViewById()方法得到了DrawerLayout的实例;

1.2.2 getSupportActionBar()方法得到了ActionBar的实例;

1.2.3 调用ActionBar的setDisplayHomeAsUpEnabled()让导航按钮显示出来;

1.2.4 调用了setHomeAsUpIndicator()方法来设置一个导航按钮图标;

1.2.5 在onOptionsItemSelected()中对HomeAsUp按钮的点击事件进行处理——调用DrawerLayout的openDrawer()方法将滑动菜单展示出来;

注意openDrawer()方法要求传入一个Gravity参数,

为了保证这里的行为和XML中(DrawerLayout标签下的第二个直接子控件的android:layout_gravity值)定义的一致,

我们传入了GravityCompat.START;

1.2.6 实际上Toolbar最左侧的这个按钮就叫作HomeAsUp按钮,它默认的图标是一个返回的箭头,含义是返回上一个活动;

这里将其换了图标,并将逻辑响应修改了;

HomeAsUp按钮的id永远都是android.R.id.home!!!

2. NavigationView

控件用处:轻松布局华丽炫酷的滑动菜单页面;

2.1 添加了两行依赖关系

compile 'com.android.support:design:24.2.1'

compile 'de.hdodenhof:circleimageview:2.1.0'

2.2

在开始使用NavigationView之前,我们还需要提前准备好两个东西:menu和headerLayout。

2.2.1 menu是用来在NavigationView中显示具体的菜单项的;

为Menu resource file;

标签下的:

android:id属性指定菜单项的id,

android:icon属性指定菜单项的图标,

android:title属性指定菜单项显示的文字。

2.2.2 headerLayout则是用来在NavigationView中显示头部布局的。

为Layout resourcefile;

2.3 使用NavigationView

添加android.support.design.widget.NavigationView标签,

使用app:menu="@menu/nav_menu"

app:headerLayout="@layout/nav_header"

将menu和headerLayout设置完毕

效果图:

44390a0dd203

正文

44390a0dd203

最终成果图

DrawerLayout   /美 [ˈdrɔɚ] /

关于滑动菜单和DrawerLayout,郭神如是说:

44390a0dd203

44390a0dd203

xmlns:android="http://schemas.android.com/apk/res/android"

xmlns:app="http://schemas.android.com/apk/res-auto"

android:id="@+id/drawer_layout"

android:layout_width="match_parent"

android:layout_height="match_parent">

android:layout_width="match_parent"

android:layout_height="match_parent">

android:id="@+id/toolbar"

android:layout_width="match_parent"

android:layout_height="?attr/actionBarSize"

android:background="?attr/colorPrimary"

android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"

app:popupTheme="@style/ThemeOverlay.AppCompat.Light"/>

android:layout_width="match_parent"

android:layout_height="match_parent"

android:layout_gravity="start"

android:text="This is menu"

android:textSize="30sp"

android:background="#FFF"/>

可见这里最外层的控件使用了DrawerLayout,这个控件是由support-v4库提供的。

DrawerLayout中放置了两个直接子控件:

第一个子控件是FrameLayout,用于作为主屏幕中显示的内容,当然里面还有我们刚刚定义的Toolbar。

第二个子控件这里使用了一个TextView,用于作为滑动菜单中显示的内容,其实使用什么都可以,DrawerLayout并没有限制只能使用固定的控件。

但是关于第二个子控件有一点需要注意,layout_gravity这个属性是必须指定的,

因为我们需要告诉DrawerLayout滑动菜单是在屏幕的左边还是右边,

指定left表示滑动菜单在左边;

指定right表示滑动菜单在右边;

这里指定了start,表示会根据系统语言进行判断,如果系统语言是从左往右的,比如英语、汉语,滑动菜单就在左边,如果系统语言是从右往左的,比如阿拉伯语,滑动菜单就在右边。

现在重新运行一下程序,然后在屏幕的左侧边缘向右拖动,就可以让滑动菜单显示出来了,如图:

44390a0dd203

44390a0dd203

44390a0dd203

44390a0dd203

44390a0dd203

这里我们并没有改动多少代码,

首先调用findViewById()方法得到了DrawerLayout的实例,

然后调用getSupportActionBar()方法得到了ActionBar的实例,虽然这个ActionBar的具体实现是由Toolbar来完成的。

接着调用ActionBar的setDisplayHomeAsUpEnabled()方法让导航按钮显示出来,

又调用了setHomeAsUpIndicator()方法来设置一个导航按钮图标。

实际上,Toolbar最左侧的这个按钮就叫作HomeAsUp按钮,它默认的图标是一个返回的箭头,含义是返回上一个活动。很明显,这里我们将它默认的样式(该按钮图标)和作用(改/设置了按钮点击事件)都进行了修改。

接下来在onOptionsItemSelected()方法中对HomeAsUp按钮的点击事件进行处理,

HomeAsUp按钮的id永远都是android.R.id.home;

切记是android.R.id.home,如果写成R.id.home是实现不了功能的!

然后调用DrawerLayout的openDrawer()方法将滑动菜单展示出来;

注意openDrawer()方法要求传入一个Gravity参数,为了保证这里的行为和XML中定义的一致,我们传入了GravityCompat.START;

当前MainActivity全文:

public class MainActivity extends AppCompatActivity {

private DrawerLayout mDrawerLayout;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);

setSupportActionBar(toolbar);

mDrawerLayout = (DrawerLayout) findViewById(R.id.drawer_layout);

ActionBar actionBar = getSupportActionBar();

if(actionBar != null){

actionBar.setDisplayHomeAsUpEnabled(true);//让导航按钮显示出来

actionBar.setHomeAsUpIndicator(R.drawable.ic_menu);//设置一个导航按钮图标

}

}

@Override

public boolean onCreateOptionsMenu(Menu menu) {

getMenuInflater().inflate(R.menu.toolbar,menu);

return true;

}

@Override

public boolean onOptionsItemSelected(MenuItem item) {

switch (item.getItemId()){

case android.R.id.home:

mDrawerLayout.openDrawer(GravityCompat.START);

break;

case R.id.backup:

Toast.makeText(this,"You clicked Backup" , Toast.LENGTH_SHORT).show();

break;

case R.id.delete:

Toast.makeText(this,"You clicked Delete" , Toast.LENGTH_SHORT).show();

break;

case R.id.settings:

Toast.makeText(this,"You clicked Settings" , Toast.LENGTH_SHORT).show();

break;

default:

}

return true;

}

}

运行程序,效果如下:

44390a0dd203

可见在Toolbar的最左边出现了一个导航按钮,用户看到这个按钮就知道这肯定是可以点击的。

现在点击一下这个按钮,滑动菜单界面就会再次展示出来了。

NavigationView

44390a0dd203

44390a0dd203

44390a0dd203

首先这个控件是DesignSupport库中提供的,需要将这个库引入到项目中。

打开app/build.gradle文件,在dependencies闭包中添加依赖:

44390a0dd203

compile 'com.android.support:design:24.2.1'

compile 'de.hdodenhof:circleimageview:2.1.0'

这里添加了两行依赖关系,

第一行就是DesignSupport库,

第二行是一个开源项目CircleImageView,它可以用来轻松实现图片圆形化的功能,我们待会就会用到它。

!!!

在开始使用NavigationView之前,我们还需要提前准备好两个东西:menu和headerLayout。

menu是用来在NavigationView中显示具体的菜单项的;

headerLayout则是用来在NavigationView中显示头部布局的。

1/4.准备menu

我们先来准备menu,这里我事先找了几张图片来作为按钮的图标,并将它们放在了drawable-xxhdpi目录下。然后右击menu文件夹→New→Menu resource file,创建一个nav_menu.xml文件,并编写如下代码:

android:id="@+id/nav_call"

android:icon="@drawable/nav_call"

android:title="Call"/>

android:id="@+id/nav_friends"

android:icon="@drawable/nav_friends"

android:title="Friends"/>

android:id="@+id/nav_location"

android:icon="@drawable/nav_location"

android:title="Location"/>

android:id="@+id/nav_mail"

android:icon="@drawable/nav_mail"

android:title="Mail"/>

android:id="@+id/nav_task"

android:icon="@drawable/nav_task"

android:title="Tasks"/>

44390a0dd203我们首先在

然后将group的checkableBehavior属性指定为singlegroup表示一个组,

checkableBehavior指定为single表示组中的所有菜单项只能单选;

那么下面我们来看一下这些菜单项吧。这里一共定义了5个item,

分别使用

android:id属性指定菜单项的id,

android:icon属性指定菜单项的图标,

android:title属性指定菜单项显示的文字。

就是这么简单,现在我们已经把menu准备好了。

2/4.准备headerLayout

接下来应该准备headerLayout了,这是一个可以随意定制的布局,不过这里不将它做得太复杂。我们就在headerLayout中放置头像、用户名、邮箱地址这3项内容吧;

说到头像,那我们还需要再准备一张图片,这里找了一张宠物图片,并把它放在了drawable-xxhdpi目录下。

另外这张图片最好是一张正方形图片,因为待会我们会把它圆形化。

然后右击layout文件夹→New→Layout resourcefile,创建一个nav_header.xml文件。

修改其中的代码,如下所示:

android:layout_width="match_parent"

android:layout_height="180dp"

android:padding="10dp"

android:background="?attr/colorPrimary">

android:id="@+id/icon_image"

android:layout_width="70dp"

android:layout_height="70dp"

android:src="@drawable/nav_icon"

android:layout_centerInParent="true"/>

android:id="@+id/mail"

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:layout_alignParentBottom="true"

android:text="8267*****@qq.com"

android:textColor="#FFF"

android:textSize="14sp"/>

android:id="@+id/username"

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:layout_above="@id/mail"

android:text="wanchuangxiaoyun"

android:textColor="#FFF"

android:textSize="14sp"/>

可以看到,布局文件的最外层是一个RelativeLayout,我们将它的

宽度设为match_parent,

高度设为180dp,

这是一个NavigationView比较适合的高度,然后

指定它的背景色为colorPrimary;

在RelativeLayout中我们放置了3个控件,

CircleImageView是一个用于将图片圆形化的控件,它的用法非常简单,基本和ImageView是完全一样的,这里给它指定了一张图片作为头像,然后设置为居中显示。

另外两个TextView分别用于显示用户名和邮箱地址,它们都用到了一些RelativeLayout的定位属性;

3/4.使用NavigationView

现在menu和headerLayout都准备好了,我们终于可以使用NavigationView了。

修改activitymam.xml中的代码,如下所示:

xmlns:android="http://schemas.android.com/apk/res/android"

xmlns:app="http://schemas.android.com/apk/res-auto"

android:id="@+id/drawer_layout"

android:layout_width="match_parent"

android:layout_height="match_parent">

android:layout_width="match_parent"

android:layout_height="match_parent">

android:id="@+id/toolbar"

android:layout_width="match_parent"

android:layout_height="?attr/actionBarSize"

android:background="?attr/colorPrimary"

android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"

app:popupTheme="@style/ThemeOverlay.AppCompat.Light"/>

android:id="@+id/nav_view"

android:layout_width="match_parent"

android:layout_height="match_parent"

android:layout_gravity="start"

app:menu="@menu/nav_menu"

app:headerLayout="@layout/nav_header">

可见这里将之前的TextView换成了NavigationView,这样滑动菜单中显示的内容也就变成NavigationView了。

这里又通过app:menu和app:headerLayout将我们刚才准备好的menu和headerLayout设置了进去,

android:layout_gravity="start"则是跟上面的textview一个道理了,用于指明滑动菜单的滑动位置,

这样NavigationView就定义完成了。

接下来还要去处理菜单项的点击事件。修改MainActivity中的代码:

44390a0dd203

代码还是比较简单的,

这里首先获取到了NavigauonView的实例,

然后调用它的setCheckedItem()方法将Call菜单项设置为默认选中。

接着调用了setNavigationItemSelectedListener()方法来设置一个菜单项选中事件的监听器,当用户点击了任意菜单项时,就会回调到onNavigationItemSelected()方法中。

我们可以在这个方法中写相应的逻辑处理,不过这里并没有附加任何逻辑,只是调用了DrawerLayout的closeDrawers()方法将滑动菜单关闭,这也是合情合理的做法。

下面是当前MainActivity.java的全文:

public class MainActivity extends AppCompatActivity {

private DrawerLayout mDrawerLayout;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);

setSupportActionBar(toolbar);

mDrawerLayout = (DrawerLayout) findViewById(R.id.drawer_layout);

NavigationView navView = (NavigationView) findViewById(R.id.nav_view);

ActionBar actionBar = getSupportActionBar();

if(actionBar != null){

actionBar.setDisplayHomeAsUpEnabled(true);//让导航按钮显示出来

actionBar.setHomeAsUpIndicator(R.drawable.ic_menu);//设置一个导航按钮图标

}

navView.setCheckedItem(R.id.nav_call);//将Call菜单项设置为默认选中

navView.setNavigationItemSelectedListener(new NavigationView.OnNavigationItemSelectedListener(){

@Override

public boolean onNavigationItemSelected(@NonNull MenuItem item) {

mDrawerLayout.closeDrawers();//关闭滑动菜单

return true;

}

});

}

@Override

public boolean onCreateOptionsMenu(Menu menu) {

getMenuInflater().inflate(R.menu.toolbar,menu);

return true;

}

@Override

public boolean onOptionsItemSelected(MenuItem item) {

switch (item.getItemId()){

case R.id.home:

mDrawerLayout.openDrawer(GravityCompat.START);

break;

case R.id.backup:

Toast.makeText(this,"You clicked Backup" , Toast.LENGTH_SHORT).show();

break;

case R.id.delete:

Toast.makeText(this,"You clicked Delete" , Toast.LENGTH_SHORT).show();

break;

case R.id.settings:

Toast.makeText(this,"You clicked Settings" , Toast.LENGTH_SHORT).show();

break;

default:

}

return true;

}

}

现在可以重新运行一下程序了,点击一下Toolbar左侧的导航按钮,效果如图所示:

44390a0dd203

44390a0dd203

怎么样?这样的滑动菜单页面,你无论如何也不能说它丑了吧?MaterialDesign的魅力就在

这里,它真的是一种非常美观的设计理念,只要你按照它的各种规范和建议来设计界面,最终做

出来的程序就是特别好看的。——郭霖大神

这篇关于android汽车之家顶部滑动菜单,Material Design 实战 之第二弹——滑动菜单详解实战(DrawerLayout NavigationView)...的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/205422

相关文章

Spring IoC 容器的使用详解(最新整理)

《SpringIoC容器的使用详解(最新整理)》文章介绍了Spring框架中的应用分层思想与IoC容器原理,通过分层解耦业务逻辑、数据访问等模块,IoC容器利用@Component注解管理Bean... 目录1. 应用分层2. IoC 的介绍3. IoC 容器的使用3.1. bean 的存储3.2. 方法注

MySQL 删除数据详解(最新整理)

《MySQL删除数据详解(最新整理)》:本文主要介绍MySQL删除数据的相关知识,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录一、前言二、mysql 中的三种删除方式1.DELETE语句✅ 基本语法: 示例:2.TRUNCATE语句✅ 基本语

Python内置函数之classmethod函数使用详解

《Python内置函数之classmethod函数使用详解》:本文主要介绍Python内置函数之classmethod函数使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 类方法定义与基本语法2. 类方法 vs 实例方法 vs 静态方法3. 核心特性与用法(1编程客

Python函数作用域示例详解

《Python函数作用域示例详解》本文介绍了Python中的LEGB作用域规则,详细解析了变量查找的四个层级,通过具体代码示例,展示了各层级的变量访问规则和特性,对python函数作用域相关知识感兴趣... 目录一、LEGB 规则二、作用域实例2.1 局部作用域(Local)2.2 闭包作用域(Enclos

Python实现对阿里云OSS对象存储的操作详解

《Python实现对阿里云OSS对象存储的操作详解》这篇文章主要为大家详细介绍了Python实现对阿里云OSS对象存储的操作相关知识,包括连接,上传,下载,列举等功能,感兴趣的小伙伴可以了解下... 目录一、直接使用代码二、详细使用1. 环境准备2. 初始化配置3. bucket配置创建4. 文件上传到os

Java内存分配与JVM参数详解(推荐)

《Java内存分配与JVM参数详解(推荐)》本文详解JVM内存结构与参数调整,涵盖堆分代、元空间、GC选择及优化策略,帮助开发者提升性能、避免内存泄漏,本文给大家介绍Java内存分配与JVM参数详解,... 目录引言JVM内存结构JVM参数概述堆内存分配年轻代与老年代调整堆内存大小调整年轻代与老年代比例元空

从原理到实战深入理解Java 断言assert

《从原理到实战深入理解Java断言assert》本文深入解析Java断言机制,涵盖语法、工作原理、启用方式及与异常的区别,推荐用于开发阶段的条件检查与状态验证,并强调生产环境应使用参数验证工具类替代... 目录深入理解 Java 断言(assert):从原理到实战引言:为什么需要断言?一、断言基础1.1 语

Python中注释使用方法举例详解

《Python中注释使用方法举例详解》在Python编程语言中注释是必不可少的一部分,它有助于提高代码的可读性和维护性,:本文主要介绍Python中注释使用方法的相关资料,需要的朋友可以参考下... 目录一、前言二、什么是注释?示例:三、单行注释语法:以 China编程# 开头,后面的内容为注释内容示例:示例:四

mysql表操作与查询功能详解

《mysql表操作与查询功能详解》本文系统讲解MySQL表操作与查询,涵盖创建、修改、复制表语法,基本查询结构及WHERE、GROUPBY等子句,本文结合实例代码给大家介绍的非常详细,感兴趣的朋友跟随... 目录01.表的操作1.1表操作概览1.2创建表1.3修改表1.4复制表02.基本查询操作2.1 SE

MySQL中的锁机制详解之全局锁,表级锁,行级锁

《MySQL中的锁机制详解之全局锁,表级锁,行级锁》MySQL锁机制通过全局、表级、行级锁控制并发,保障数据一致性与隔离性,全局锁适用于全库备份,表级锁适合读多写少场景,行级锁(InnoDB)实现高并... 目录一、锁机制基础:从并发问题到锁分类1.1 并发访问的三大问题1.2 锁的核心作用1.3 锁粒度分