Android Material Design:常用控件学习笔记

Google I/O 2014 发布了Material Design。希望统一 Android平台设计语言规范。然而再国内的很多产品和设计师并不吃这一套,还是各种仿IOS的UI。作为一个Google粉当然要学会Android Material控件的使用。而且这些控件使用起来非常方便。以下是Android Material常用控件的整理。
请注意:介绍了多个控件、多图预警,流量党珍惜下流量。

介绍的控件:
1、Toolbar和Menu。
2、基于CoordinatorLayout的联动。
3、NavigationView。
4、CardView。
5、TextInputLayout。
6、RecyclerView。
7、TabLayout。
8、SnackBar
9、FloatingActionButton

github地址:https://github.com/AxeChen/MaterialDesignSimple

Android Material Design:常用控件学习笔记_第1张图片
整理

1、Toolbar

ToolBar是Android 5.0推出的一个新的导航控件用于取代之前的ActionBar。

Android Material Design:常用控件学习笔记_第2张图片
Toolbar

1.1、基本使用

导入依赖
compile 'com.android.support:appcompat-v7:25.3.1'
修改主题

我们需要使用Toolbar代替Actionbar,就必须将需要使用Toolbar的Activity的Theme设为NoActionBar。

    

由于示例代码中只有一个Activity,所以我将整个AppTheme继承了Theme.AppCompat.Light.NoActionBar。
如果实际情况只有·部分Activity使用Toolbar,可以单独为这些Activity设置没有ActionBar的主题。

布局文件中定义Toolbar




    

setSupportActionBar

这个方法是 AppCompatActivity 中的方法,所以使用的 Activity 必须继承 AppCompatActivity。

   private Toolbar toolbar;
   private void initToolbar() {
        toolbar = (Toolbar) findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);
    }

通过以上三步,Toolbar就能展示了。

1.2、修改toolbar的属性

通过调用Toolbar提供的方法。

可以直接调用Toolbar的一些方法来改变样式:
我写的示例代码中调用的Toolbar方法。

    private void setToolbarProperty() {
        // 设置正标题
        toolbar.setTitle("正标题");
        // 设置副标题
        toolbar.setSubtitle("副标题");
        // 设置左边按钮图片
        toolbar.setNavigationIcon(R.mipmap.ic_launcher_round);
        // 设置(Log)标题与左边按钮之间图标
        toolbar.setLogo(R.mipmap.ic_launcher);
    }

实际上Toolbar提供修改Toolbar属性的方法非常多,远不止以上提供的方法。

在xml文件里修改

先导入命名空间:

 xmlns:app="http://schemas.android.com/apk/res-auto"
Android Material Design:常用控件学习笔记_第3张图片
在xml文件里修改

属性太多就不整理了。对修改Toolbar的属性,这篇整理得不错,可以直接拿来用:http://www.jianshu.com/p/2c21676033db

通过Actionbar来修改

通过Actionbar的一些方法也能够修改Toolbar的一些样式(因为里面用了代理模式,具体还在研究中)。


Android Material Design:常用控件学习笔记_第4张图片
通过Actionbar来修改

这篇也有提到 : http://www.jianshu.com/p/7b5c99e1cfa3
部分方法的作用:http://blog.csdn.net/andygo_520/article/details/51439688

1.3、Toolbar点击的回调

Toolbar的点击回调主要是左边的按钮点击回调和toolbar上面的Menu点击回调。
对于Menu的点击回调更习惯重写onOptionsItemSelected方法来监听回调,后面的Menu会提到。

   // 点击左侧按钮监听
  toolbar.setNavigationOnClickListener(new View.OnClickListener() {
      @Override
      public void onClick(View v) {
          Toast.makeText(MainActivity.this, "点击了Navigation按钮", Toast.LENGTH_SHORT).show();
      }
  });
   // toolbar的Menu回调
  toolbar.setOnMenuItemClickListener(new Toolbar.OnMenuItemClickListener() {
    @Override
    public boolean onMenuItemClick(MenuItem item) {
        return false;
    }
});

1.4、Toolbar添加Menu

Menu是Toolbar非常重要的部分,可以将一些功能放在Menu上来实现,常见的有发送,分享等等。

添加Menu

首先必须在Activity重写onCreateOptionsMenu方法,添加Menu的操作都在这个方法中执行。

@Override
    public boolean onCreateOptionsMenu(Menu menu) { 
            return super.onCreateOptionsMenu(menu);
    }

添加Menu有两种方式:

  • 通过加载Menu资源文件
  • 代码添加

如果通过资源文件添加,先要在res/menu下添加一个menu的资源文件。



    
    
    

如果是通过代码添加,onCreateOptionsMenu方法中调用menu.add()方法。

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        //使用代码创建menu
        //menu.add(0, 0, 0, "搜索").setShowAsAction(MenuItem.SHOW_AS_ACTION_ALWAYS);
        menu.add(0, 0, 0, "搜索").setIcon(android.support.v7.appcompat.R.drawable.abc_ic_search_api_material).setShowAsAction(MenuItem.SHOW_AS_ACTION_ALWAYS);
        //使用加载xml文件的方式创建
        getMenuInflater().inflate(R.menu.test_menu, menu);
        //添加子菜单
        menu.addSubMenu(0, 1, 0, "submenu").setIcon(R.mipmap.ic_launcher).addSubMenu(0, 2, 0, "submenu1");
        return super.onCreateOptionsMenu(menu);
    }

这里有个重要的属性showAsAction,选项和作用如下:
never:永远不显示的ToolBar上,只放在OverFlow按钮中(最右边三个点的按钮)。只有OverFlow按钮才会弹出。
always :总会展示在Toolbar上。
ifRoom :如Toolbar还有足够的空间,则显示在Toolbar上。否则就放在OverFlow按钮中。
withText,collapseActionView没用过,看下这篇博客的说明吧:http://www.jianshu.com/p/4106e1414d64

3.5、监听Menu点击

首先重写onOptionsItemSelected方法。
根据MenuItem.getItemId()的方法即可找点击的Menu。
如果是加载Menu资源文件,那么ItemId就是给Item设置的id。

 

如果通过代码添加的Menu,那么ItemId就是调用add方法传入的itemid。看下add方法的源码就可以找到。

public MenuItem add(int groupId, int itemId, int order, CharSequence title);

监听点击的代码。

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        switch (item.getItemId()) {
            case 0:
                Toast.makeText(this, "点击搜索", Toast.LENGTH_SHORT).show();
                break;
            case 1:
                Toast.makeText(this, "点击submenu", Toast.LENGTH_SHORT).show();
                break;
            case 2:
                Toast.makeText(this, "submenu1", Toast.LENGTH_SHORT).show();
                break;
            case R.id.action_settings:
                Toast.makeText(this, "点击设置", Toast.LENGTH_SHORT).show();
                break;
            case R.id.action_edit:
                Toast.makeText(this, "点击编辑", Toast.LENGTH_SHORT).show();
                break;
            case R.id.action_share:
                Toast.makeText(this, "点击分享", Toast.LENGTH_SHORT).show();
                break;
        }
        return super.onOptionsItemSelected(item);
    }

附上Fragment中使用Menu的博客:http://www.jcodecraeer.com/a/anzhuokaifa/androidkaifa/2013/0104/777.html

2、基于CoordinatorLayout的联动

CoordinatorLayout是Android Material Design的重要组件,实现协调其他组件实现联动的效果。

2.1、 CollapsingToolbarLayout

效果图:


CollapsingToolbarLayout
添加依赖
 compile 'com.android.support:design:25.3.1'
 compile 'com.android.support:appcompat-v7:25.3.1'
布局结构


    

        

            

            

        
    

    
    

既然是基于CoordinatorLayout,所以控件的父布局就是CoordinatorLayout。
重要的属性(这些没有设置是无法实现效果的):

  • CollapsingToolbarLayout中的属性
    app:layout_scrollFlags="scroll|exitUntilCollapsed“
    决定CollapsingToolbarLayout的滑动方式,这些flag的作用:

enterAlways: 一旦向上滚动这个view就可见。
enterAlwaysCollapsed: 当你定义了一个minHeight,那么view将在到达这个最小高度的时候开始显示,并且从这个时候开始慢慢展开,当滚动到顶部的时候展开完。
exitUntilCollapsed: 当你定义了一个minHeight,这个view将在滚动到达这个最小高度的时候消失。

当然这些flag最好是实际使用看效果。

  • NestedScrollView中的属性
    app:layout_behavior="@string/appbar_scrolling_view_behavior”
    这个属性用于通知AppBarLayout 。View(这里是NestedScrollView
    )何时发生了滚动事件。
1.2、向上滑动隐藏Toolbar

效果图:


向上滑动隐藏Toolbar
添加依赖
 compile 'com.android.support:design:25.3.1'
 compile 'com.android.support:appcompat-v7:25.3.1'
布局结构


    
        

    
 
    


这里同样需要设置这两个属性:
app:layout_scrollFlags="scroll|enterAlways"
app:layout_behavior="@string/appbar_scrolling_view_behavior"

1.3、其他的一些问题

通过以上的两个例子可以总结出基于CoordinatorLayout联动的两点规律:
1、父布局肯定是CoordinatorLayout
2、一定会设置app:layout_scrollFlagsapp:layout_behavior两个属性。
滑动的View:滑动的view官方建议使用NestedScrollView或者RecyclerView。ListView在5.0以下就没有效果了。
高阶的使用:高阶使用当然是自定义behavior了。http://blog.csdn.net/gdutxiaoxu/article/details/53453958

3、NavigationView

NavigationView用来实现侧滑导航的布局。
效果图:


Android Material Design:常用控件学习笔记_第5张图片
NavigationView

如果使用Android studio新建Android项目,在选择Activity时会有Navigation Drawer Activity。选择它会生成一个带有侧滑导航栏的Activity。


Android Material Design:常用控件学习笔记_第6张图片
新建Navigation Drawer Activity
添加依赖
    compile 'com.android.support:appcompat-v7:25.3.1'
    compile 'com.android.support:design:25.3.1'
布局结构



    

    

NavigationView中,主要的两个属性:

 app:headerLayout="@layout/nav_header_main"
 app:menu="@menu/activity_main_drawer" 

如下图所示,NavigationView分为两部分,headerView 和 menu:


Android Material Design:常用控件学习笔记_第7张图片
QQ截图20170827155506.png

headview:就是普通的布局。




    

    

    


menu:需要在res/menu下定义:




    
        
        
        
        
    
    
        
            
            
        
    

3.1 修改样式

  • 用代码修改 : NavigationView提供了很多setXXX()的方法。
  • xml文件中修改:
    Android Material Design:常用控件学习笔记_第8张图片
    xml文件中修改

    记得添加命名空间

3.1、其他的问题

menu图片为灰色问题
在使用的时候,会发现所有的图片多会展示成灰色,不管你的图片是否为彩色的图片。
如下方法解决:

NavigationView navigationView = (NavigationView) findViewById(R.id.navigation_view);  
navigationView.setItemIconTintList(null);  

4、CardView

效果图:


Android Material Design:常用控件学习笔记_第9张图片
CardView

CardView并不属于design包,所以使用CardView必须先添加依赖:

    compile 'com.android.support:cardview-v7:25.3.1'

CardView是一个继承FrameLayout的View。用于布局的圆角、阴影等效果。基本使用只要知道CardView一些常用属性即可。

属性 作用
cardElevation 阴影的大小
cardElevation 阴影最大高度
cardElevation 卡片的背景色
cardCornerRadius 卡片的圆角大小
contentPadding 卡片内容于边距的间隔
contentPaddingBottom 卡片内容与底部的边距
contentPaddingTop 卡片内容与顶部的边距
contentPaddingLeft 卡片内容与左边的边距
contentPaddingRight 卡片内容与右边的边距
contentPaddingStart 卡片内容于边距的间隔起始
contentPaddingEnd 卡片内容于边距的间隔终止
cardUseCompatPadding 设置内边距,V21+的版本和之前的版本仍旧具有一样的计算方式
cardPreventConrerOverlap 在V20和之前的版本中添加内边距,这个属性为了防止内容和边角的重叠

如果直接给CardView添加android:foreground="?attr/selectableItemBackground"就会加上水波纹点击反馈。

** 遇到的问题:**如果你在RecyclerView中使用CardView,如果你用的context是getApplicationContext(),可能会出现展示出夜间模式的效果。我当时使用getApplicationContext()是为了防止内存溢出。具体原因暂时还不清楚。
出现问题的代码。

    public RecommendAdapter(Context context, List parkBeanList) {
        this.context = context.getApplicationContext();
        this.parkBeanList = parkBeanList;
    }

解决方案:不使用 context.getApplicationContext();

  public RecommendAdapter(Context context, List parkBeanList) {
        //this.context = context.getApplicationContext();
        this.context = context;
        this.parkBeanList = parkBeanList;
    }

5、TextInputLayout

5.1 基本使用

使用TextInputLayout控件的效果,需要将EditText作为TextInputLayout的子控件使用,而且只能有一个EditText。

添加依赖
  compile 'com.android.support:design:25.3.1'
布局结构
    
        
    

效果图:


Android Material Design:常用控件学习笔记_第10张图片
TextInputLayout
5.2 错误提示

效果图:


验证输入的字符

TextInputLayout提供了非常方便的错误提示:

  • 当需要错误提示时,调用TextInputLayout的setError("错误提示文字")
  • 如果要清空错误提示时,调用TextInputLayout的setError(null)

以效果图中的用户名为例:
实现Editext的addTextChangedListener方法,监听输入字符。然后根据字符判断输入的字符串是否合理。

    private void initListener() {
        edName.addTextChangedListener(new TextWatcher() {
            @Override
            public void beforeTextChanged(CharSequence s, int start, int count, int after) {
            }
            @Override
            public void onTextChanged(CharSequence s, int start, int before, int count) {
            }
            @Override
            public void afterTextChanged(Editable s) {
                checkName(s.toString());
            }
        });
    }

    private void checkName(String s) {
        if (s.length() < 10) {
            nameInput.setError("名字长度太短");
        } else {
            nameInput.setError(null);
        }
    }

6、RecyclerView

RecyclerView是Android程序员必学控件,如果现在还不会RecyclerView就赶紧学起来!
RecyclerView可以实现GirdView,ListView和瀑布流的效果。可以设置滑动的方向。

添加依赖
    compile 'com.android.support:appcompat-v7:25.3.1'
6.1、LayoutManager

决定RecyclerView以一种怎样的形式展示由LayoutManager决定。
设置GridLayoutManager,实现GridView的效果。
设置LinearLayoutManager,实现ListView的效果。
设置StaggeredGridLayoutManager,实现瀑布流的的效果。


Android Material Design:常用控件学习笔记_第11张图片
LayoutManager

6.2 RecyclerView.Adapter、 RecyclerView.ViewHolder

实现数据的展示需要用到Adapter和ViewHolder

继承RecyclerView.Adapter的Adapter

必须重写的三个方法:
onCreateViewHolder: 创建ViewHolder并返回,负责为给Item创建视图。
onBindViewHolder: 负责将数据绑定到Item的视图上。
getItemCount: 返回总数据条数。

    private class MyAdapter extends RecyclerView.Adapter {
        @Override
        public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
            return new ItemViewHolder(LayoutInflater.from(RecyclerListViewActivity.this).inflate(R.layout.item_list, parent, false));
        }
        @Override
        public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
            bindItemViewHolder((ItemViewHolder) holder, position);
        }
        @Override
        public int getItemCount() {
            return items.size();
        }
    }
继承RecyclerView.ViewHolder的ViewHolder

和使用Listview的Viewholder差不多。里面是一些初始化控件的操作。

public class ItemViewHolder extends RecyclerView.ViewHolder {

        private ImageView imageView;
        private TextView tvTitle;
        private TextView tvDescription;

        public ItemViewHolder(View itemView) {
            super(itemView);
            imageView = (ImageView) itemView.findViewById(R.id.ivImg);
            tvTitle = (TextView) itemView.findViewById(R.id.tvTitle);
            tvDescription = (TextView) itemView.findViewById(R.id.tvDescription);
        }
}

6.3 实现ListView、Gridview的效果

前面已经提到过设置不同LayoutManager会展示不同的效果。
实现Listview效果:

 recyclerView.setLayoutManager(new LinearLayoutManager(this, LinearLayoutManager.VERTICAL, false));

注意:LinearLayoutManager.VERTICAL为垂直滑动,如果设置成LinearLayoutManager.HORIZONTAL则会水平滑动。
效果图:

ListView的效果

实现GridView效果:

recyclerView.setLayoutManager(new GridLayoutManager(this, 3, LinearLayoutManager.VERTICAL, false));
Android Material Design:常用控件学习笔记_第12张图片
GridView的效果

至于瀑布流,由于平时使用比较少,这里就不做详细的解释了。

7、TabLayout

效果图:


Android Material Design:常用控件学习笔记_第13张图片
基本的使用
添加依赖
  compile 'com.android.support:design:26.0.0-alpha1'

7.1、如何添加Tab。

Android Material Design:常用控件学习笔记_第14张图片
添加Tab

从上图可看出,TabLayout提供了addTab()的方法来添加Tab。初始化Tab之后可以调用setXXX()来决定是展示文字、图片或者自定义布局。

7.2、TabMode

TabLayout.MODE_SCROLLABLE:当元素过多时会超出父布局,并可以滑动Tab,Tab的宽度为Tab的实际宽度。
TabLayout.MODE_FIXED:无论界面由多少元素都会充满父布局。并且平均分配Tab的宽度。

7.3、监听Tab的选择状态

tabLayoutOne.addOnTabSelectedListener(new TabLayout.OnTabSelectedListener() {
            @Override
            public void onTabSelected(TabLayout.Tab tab) {
            }
            @Override
            public void onTabUnselected(TabLayout.Tab tab) {
            }
            @Override
            public void onTabReselected(TabLayout.Tab tab) {
            }
        });

7.4、修改TabLayou的样式

  • 在布局文件中设置
    在使用自定义属性时,记得添加命名空间。

    Android Material Design:常用控件学习笔记_第15张图片
    布局中设置属性

  • 在代码中设置
    调用Tablayout的setXXX()的方法。

  • 在styles中修改
    继承Widget.Design.TabLayout修改TabLayout的样式。
    继承TextAppearance.Design.Tab修改Tab的样式。



修改完毕之后记得将styles设置到TabLayout控件上。

7.5、与Viewpager的联动

效果图:

Android Material Design:常用控件学习笔记_第16张图片
与viewpager的联动

Viewpager+fragment+导航栏的这种UI结构。已经应用得非常广泛了,几乎是随处可见。TabLayout提供了和Viewpager联动方法,实现简单联动的方法也比较简单。
注:这里代码只考虑了Tab展示Text的情况。如果Tab展示的是图片或者自定义View,会稍微麻烦点。

  viewPager.setAdapter(new PagerAdapter(getSupportFragmentManager()));
  tabLayout.setupWithViewPager(viewPager);

viewPager设置完PagerAdapter之后,调用 tabLayout.setupWithViewPager(viewPager);即可。
PagerAdapter重写getPageTitle方法,根据Viewpager页面的position来展示不同Tab的Text。

 @Override
  public CharSequence getPageTitle(int position) {
            return "第" + (position + 1) + "页";
   }

不过这里还是埋了坑,有时候会出现Tab无法展示的情况(写过的都懂)。
这个链接可以解决这个问题:http://blog.csdn.net/sundy_tu/article/details/52682246

8、SnackBar

添加依赖
 compile 'com.android.support:design:26.0.0-alpha1'

效果图:


SnackBar

SnackBar和Toast的用法差不多,都是通过show方法展示。和Toast不同之处是能和用户交互,同时提供了一些展示和消失的回调,修改SnackBar的文字颜色也比较简单。

8.1 SnackBar的基本使用

  • 仅仅只提示文字:
    弹出提示文字之后一段时间后会消失。
Snackbar.make(button, "第一次使用SnackBar", Snackbar.LENGTH_SHORT).show();
  • 提供一个可以点击的按钮:
    添加一个可以点击的按钮。
Snackbar.make(button, "第一次使用SnackBar", Snackbar.LENGTH_SHORT).setAction("确定", new View.OnClickListener() {
                    @Override
                    public void onClick(View v) {

                    }
                }).show();
  • 必须和用户交互之后才消失:
    Snackbar.make()设置时间的参数改为:Snackbar.LENGTH_INDEFINITE。如果不设置这个参数,Snackbar都会在一段时间后消失。
 Snackbar.make(button, "第一次使用SnackBar", Snackbar.LENGTH_INDEFINITE).setAction("确定", new View.OnClickListener() {
                    @Override
                    public void onClick(View v) {

                    }
                }).show();
  • 添加状态回调:
    监听Snackbar的展示和消失。
Snackbar.make(button, "第一次使用SnackBar", Snackbar.LENGTH_INDEFINITE).setAction("确定", new View.OnClickListener() {
                    @Override
                    public void onClick(View v) {

                    }
                }).addCallback(new Snackbar.Callback() {
                    @Override
                    public void onShown(Snackbar sb) {
                        super.onShown(sb);
                        Toast.makeText(MainActivity.this, "弹出了", Toast.LENGTH_SHORT).show();
                    }

                    @Override
                    public void onDismissed(Snackbar transientBottomBar, int event) {
                        super.onDismissed(transientBottomBar, event);
                        Toast.makeText(MainActivity.this, "消失了", Toast.LENGTH_SHORT).show();
                    }
                }).show();

8.2、修改Snackbar的样式

Snackbar只提供了一个修改按钮文字的方法:snackbar.setActionTextColor(Color.GREEN)
如需修改提示文字,背景的颜色等。可以通过snackbar.getView()获取到Snackbar对应的View进行修改。获取到Snackbar的View之后通过findById就能获取其他的一些控件。(Snackbar对应的布局文件在源码中可以找到。)
注意:这种方式还是存在风险。如果哪天Snackbar更新,Snackbar开发者修改了这些控件的Id,那就无法获取了。

 Snackbar snackbar = Snackbar.make(button, "第一次使用SnackBar", Snackbar.LENGTH_INDEFINITE).setAction("确定", new View.OnClickListener() {
                    @Override
                    public void onClick(View v) {

                    }
                });
                snackbar.setActionTextColor(Color.GREEN);

  //修改背景
  snackbar.getView().setBackgroundResource(R.color.colorPrimaryDark);
  /* snackbar 并没有提供修改提示文字颜色的方法。不过可以通过找到
      snackbar的布局design_layout_snackbar_include通过布局可以找到textview的id。
     再通过snackbar.getView().findViewById(R.id.snackbar_text);
   */
  TextView textView = (TextView) snackbar.getView().findViewById(R.id.snackbar_text);
  textView.setTextColor(getResources().getColor(R.color.colorAccent));
  snackbar.show();

9、FloatingActionButton

FloatingActionButton:界面浮动的标签,一般用于页面关键功能入口。FloatingActionButton非常简单,知道了解FloatingActionButton的一些属性和点击回调即可。
效果图:

Android Material Design:常用控件学习笔记_第17张图片
FloatingActionButton

常用属性:
先加入命名空间!

属性 作用
fabSize 定义FloatingActionButton的大小。auto(大) mini(小) normal(中)
elevation 普通状态下的阴影深度
pressedTranslationZ 按下时的阴影深度
backgroundTint 默认展示的背景颜色
rippleColor 按下时的颜色(5.0以后为水波纹的颜色)
layout_anchor 定位其他控件,和其他控件边界相交
layout_anchorGravity 和layout_anchor属性联用,在其他控件的相对位置
useCompatPadding 设置内边距
borderWidth 边框宽度(使用的比较少,效果也不好看)

点击回调:

 FloatingActionButton fabOne = (FloatingActionButton) findViewById(R.id.fabOne);
 fabOne.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Snackbar.make(v, "正在打开地图", Snackbar.LENGTH_SHORT).show();
            }
  });

最后

感谢各位程序员老爷看完这个又臭又长的博客!
源码地址:https://github.com/AxeChen/MaterialDesignSimple (欢迎star)

你可能感兴趣的:(Android Material Design:常用控件学习笔记)