• 9、Android---UI---Material Design


     

    9.1、什么是Material Design

    由谷歌的设计师基于传统优秀设计原则,结合丰富的创意和科学技术所发明的一套全新的界面设计语言

    包含了视觉、运行、互动等效果

    Material Design的出现使得Android首次再UI方面全面超越了IOS

    此时就能解决不同操作系统之间的统一界面

    谷歌从5.0系统开始就将所有内置应用都使用Material Design风格来设计

    主要是面向UI设计人员并不是开发者。

    9.2、Toolbar

    之前的标题栏

    把系统的ActionBar隐藏

    每个活动顶部的标题栏就是ActionBar

    ActionBar由于设计原因只能用于活动的顶部

    不能实现一些Material Design的效果

    因此官方不建议使用ActionBar

    Toolbar的枪法之处在于不仅继承了ActionBar的所有功能

    而且灵活性高

    可以配合其他控件完成一些Material Design的效果

    任何一个项目默认都会显示ActionBar的

       <application
            android:allowBackup="true"
            android:icon="@mipmap/ic_launcher"
            android:label="@string/app_name"
            android:supportsRtl="true"
            android:theme="@style/AppTheme">

    打开res/values/styles.xml文件

    <resources>
    
        <!-- Base application theme. -->
        <style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar">
            <!-- Customize your theme here. -->
            <item name="colorPrimary">@color/colorPrimary</item>
            <item name="colorPrimaryDark">@color/colorPrimaryDark</item>
            <item name="colorAccent">@color/colorAccent</item>
        </style>
    
    </resources>

     定义了一个AppTheme主题

    然后指定它的parent主题

    这里的DarkActionBar是一个深色的Actionbar主题

    之前的实践中自带的ActionBar就是指定了这个主题才出现的

    此时使用Toolbar来代替ActionBar

    因此需要指定一个不带ActionBar的主题

    通常用:

    前者:深色主题,会将界面的主题颜色设计成深色,陪衬的颜色设计成淡色

    后者:淡色主题,会将界面的主题颜色设计成淡色,陪衬的颜色设计成深色

    <?xml version="1.0" encoding="utf-8"?>
    <FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        android:layout_width="match_parent"
        android:layout_height="match_parent">
    
        <android.support.v7.widget.Toolbar
            android:id="@+id/toolbar"
            android:layout_width="match_parent"
            android:background="?attr/colorPrimary"
            android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
            android:layout_height="?attr/actionBarSize"
            app:popupTheme="@style/Theme.AppCompat.Light"
            />
    
    </FrameLayout>

    这里首先引入app的约束

    5.0系统之后才可以使用,之前的使用android:xx

    所以需要引入新的约束

    ToolBer控件由appcompat-v7库提供的

    此时指定id、宽度设置为match_parent

    高科设置为actionBar的高度

    背景色设置为colorPrimary

    再style.xml文件中将程序的主题指定成淡色主题

    此时ToolBar的各种元素就会自动使用深色

    目的是为了和主题颜色进行区分

    这里可以使用android:theme属性,将主题指定为指定的主题

    这里使用app:popupTheme将属性将菜单选项指定成淡色主题,5.0之后新增的

    使用app:popupTheme可以兼容5.0之前的系统

     MainActivity

    public class MainActivity extends AppCompatActivity {
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.action_main);
    
            Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
            setSupportActionBar(toolbar);
        }
    }

    此时的效果

     上述图片中的文字是在:AndroidManifest.xml文件中

    使用android:label进行设置显示的内容

    添菜单:

    <?xml version="1.0" encoding="utf-8"?>
    <menu xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto">
    
        <item android:id="@+id/Back"
            android:title="返回"
            android:icon="@mipmap/ic_launcher"
            app:showAsAction="always"/>
    
        <item android:id="@+id/delete"
            android:title="删除"
            android:icon="@mipmap/ic_launcher"
            app:showAsAction="ifRoom"/>
    
        <item android:id="@+id/setting"
            android:title="设置"
            android:icon="@mipmap/ic_launcher"
            app:showAsAction="never"/>
    </menu>

    使用<item>标签来定义action按钮

    app:showAsAction用于指定按钮的显位置可选值:

    1、always:表示永远现在再Toolbar中,屏幕不够则不显式

    2、ifRoom:表示再屏幕控件足够的情况下再Toolbar中显示,不够的话显示再菜单单中

    3、never:永远显示再菜单选项当中

    Toolbar中的action值会显示图标,菜单中只显示文件

    MainActivity

       @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.Back:
                    Toast.makeText(this,"Back",Toast.LENGTH_LONG).show();
                    break;
                case R.id.delete:
                    Toast.makeText(this,"delete",Toast.LENGTH_LONG).show();
                    break;
                case R.id.setting:
                    Toast.makeText(this,"setting",Toast.LENGTH_LONG).show();
                    break;
                default:
                    break;
            }
    
            return true;
        }

    效果:

    9.3、滑动菜单

    是MaterialDesign中常见效果之一

    9.3.1、DraweLayout

    DrawerLayout是一个布局

    再布局中允许放入两个直接控件

    第一个控件就是主屏幕显示的内容

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

    <?xml version="1.0" encoding="utf-8"?>
    <android.support.v4.widget.DrawerLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:id="@+id/drawe_layout"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <FrameLayout
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <android.support.v7.widget.Toolbar
    android:id="@+id/toolbar"
    android:layout_width="match_parent"
    android:background="?attr/colorPrimary"
    android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
    android:layout_height="?attr/actionBarSize"
    app:popupTheme="@style/Theme.AppCompat.Light"
    />

    </FrameLayout>


    <LinearLayout
    android:id="@+id/left_drawer_layout"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:layout_gravity="start"
    android:background="#FFFFFF">
    <TextView
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:text="This is menu"
    android:textSize="25sp"
    android:textColor="#dd033d"/>
    </LinearLayout>




    </android.support.v4.widget.DrawerLayout>

    DrawerLayout由support-v4库提供

    第一个子控件式FrameLayout,用于再主屏幕显示内容

    第二个子控件是LinearLayout,用于作为滑动显示内容

    第二个控件注意:

    layout_gravity这个属性是必须指定的

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

    指定right再右边,指定start,系统会进行判断,系统语言是从左往右(英语,汉语等)滑动菜单再左边

    系统语言是从右往左的(阿拉伯)滑动菜单就在右边

    然后向左滑动菜单

    或者点击一个菜单之外的区域,滑动菜单就会关闭

    从而回到主界面

    无论隐藏菜单还是滑动菜单都有很流畅的动画过度

    现在只有在屏幕的左侧边缘向右滑动才能显示滑动菜单

    在没有提示的情况下,用户很难知道这个功能是如何实现的

    Material Design建议在Toolbar的最左边介入一个导航按钮

    点击了按钮也将会滑动菜单的内容显示出来

    这样等于提供两种方式给用户

    public class MainActivity extends AppCompatActivity {
    
        private DrawerLayout mdrawerLayout;
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.action_main);
    
            Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
            setSupportActionBar(toolbar );
    
            mdrawerLayout = (DrawerLayout) findViewById(R.id.drawe_layout);
            ActionBar actionBar = getSupportActionBar();
            if (actionBar != null){
                actionBar.setDisplayHomeAsUpEnabled(true);
                actionBar.setHomeAsUpIndicator(R.mipmap.ic_launcher);
            }
        }
    
        @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;
            ....
            }
            return true;
        }
    }

    首先得到了DrawerLayout的实例

    然后调用getSupportActionBar()方法得到ActionBar的实例

    虽然只能个ActionBar具的实现由Toolbar来完成

    然后调用ActionBar的serDisplayHomeAsUpEnabled()方法来让导航按钮显示出来

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

    实际上,Toolbar最左侧的这个按钮叫做HomeAsUp按钮,默认图标是一个箭头

    含义是返回上一个图标

    最后在onOptionsItemSelected()方法中对HomeAsUp按钮点击事件进行处理

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

    在调用DrawerLayout的openDrawer()方法将滑动菜单显示出来

    此时传入一个Gravity参数

    保持和XNL中的数据一直传入GravityCompat.START

    9.3.2、NavigationView

    首先:

    控件由Design Support库提供

    需要引入库

    第一行是Design Supprot库

    第二行是一个开源项目CircleImageView,可以轻松实现图片圆形化的功能

    工程准备:menu、headerLayout

    menu是在NavigationView中显示的具体菜单

    headerLayout用来在NavigationLayout中显示头部布局的

    nav_menu.xml

    <?xml version="1.0" encoding="utf-8"?>
    <menu
        xmlns:android="http://schemas.android.com/apk/res/android">
    
        <group android:checkableBehavior="single">
            <item
                android:title="call"
                android:id="@+id/nav_call"
                android:icon="@mipmap/ic_launcher"/>
            <item
                android:title="friends"
                android:id="@+id/nav_friends"
                android:icon="@mipmap/ic_launcher"/>
            <item
                android:title="location"
                android:id="@+id/nav_location"
                android:icon="@mipmap/ic_launcher"/>
            <item
                android:title="mail"
                android:id="@+id/nav_mail"
                android:icon="@mipmap/ic_launcher"/>
            <item
                android:title="task"
                android:id="@+id/nav_task"
                android:icon="@mipmap/ic_launcher"/>
        </group>
    </menu>

    使用<group>biaoqian 

    将group的checkableBehavior属性指定为single

    标签表示一个组

    single表示所有菜单的选项只能单选

    nav_header.xml

    <?xml version="1.0" encoding="utf-8"?>
    <RelativeLayout
        xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="match_parent"
        android:layout_height="180dp"
        android:padding="10dp"
        android:background="?attr/colorPrimary">
    
        <de.hdodenhof.circleimageview.CircleImageView
            android:layout_width="70dp"
            android:layout_height="70dp"
            android:id="@+id/icon_image"
            android:src="@mipmap/ic_launcher"
            android:layout_centerInParent="true"
            />
        <TextView
            android:layout_alignParentBottom="true"
            android:id="@+id/username"
            android:textSize="14sp"
            android:textColor="#fff"
            android:text="MrChengs"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content" />
    
        <TextView
            android:layout_above="@+id/username"
            android:id="@+id/mail"
            android:textSize="14sp"
            android:textColor="#fff"
            android:text="mrchengs666@163.com"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content" />
    
    </RelativeLayout>

    实现头像、用户名、邮箱的显示

    最外面使用RelativeLayout

    将宽度设置为match_parent

    高度设置为180dp

    这是一个NavifationView适合的高度

    然后指定背景色

    CircleImageView是一个用于圆形化的控件

    用法非常简单

    和ImageView使用是完全一样的

    指定一张图片作为头像

    然后剧中显示

    两个TextView分别用于显示用户名和邮箱地址

    修改主界面的布局代码:

    <?xml version="1.0" encoding="utf-8"?>
    <android.support.v4.widget.DrawerLayout
        xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        android:id="@+id/drawe_layout"
        android:layout_width="match_parent"
        android:layout_height="match_parent">
    
                <FrameLayout
                    android:layout_width="match_parent"
                    android:layout_height="match_parent">
    
                    <android.support.v7.widget.Toolbar
                        android:id="@+id/toolbar"
                        android:layout_width="match_parent"
                        android:background="?attr/colorPrimary"
                        android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
                        android:layout_height="?attr/actionBarSize"
                        app:popupTheme="@style/Theme.AppCompat.Light"
                        />
    
                </FrameLayout>
    
        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:layout_gravity="start">
    
                <android.support.design.widget.NavigationView
                    android:id="@+id/nav_view"
                    android:layout_width="match_parent"
                    android:layout_height="match_parent"
                    app:menu="@menu/nav_menu"
                    app:headerLayout="@layout/nav_header"
                    >
          </android.support.design.widget.NavigationView>
        </LinearLayout>
    
    
    </android.support.v4.widget.DrawerLayout>

    这里使用NavigationView

    这样滑动菜单就变成了NavigationView

    通过app:menu、app:headerLayout属性将menu和headerLayout设置进入

    此时的NavigationView就定义完成

    在MainActivity

    在onCreate()方法中

     NavigationView navigationView = (NavigationView) findViewById(R.id.nav_view);
            navigationView.setCheckedItem(R.id.nav_call);
            navigationView.setNavigationItemSelectedListener(new NavigationView.OnNavigationItemSelectedListener() {
                @Override
                public boolean onNavigationItemSelected(@NonNull MenuItem item) {
                    mdrawerLayout.closeDrawers();
    
                    return true;
                }
            });

    setCheckedItem()将call菜单项设置为默认选中

    setNavigationItemSelectedListener()方法设置一个菜单选中事件的监听器

    当用户点击菜单项中的监听器,就会回调onNavigationItemSelected()方法

    此时掉哟给DrawerLayout的closeDrowers()方法将滑动菜单关闭

    9.4、悬浮按钮和可交互提示

    9.4.1、FloatingActionButton

    是Design Support库提供的一个控件

    可以帮助我们轻松的实现按钮悬浮效果

    <?xml version="1.0" encoding="utf-8"?>
    <android.support.v4.widget.DrawerLayout
        xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        android:id="@+id/drawe_layout"
        android:layout_width="match_parent"
        android:layout_height="match_parent">
    
                <FrameLayout
                    android:layout_width="match_parent"
                    android:layout_height="match_parent">
    
                    <android.support.v7.widget.Toolbar
                        android:id="@+id/toolbar"
                        android:layout_width="match_parent"
                        android:background="?attr/colorPrimary"
                        android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
                        android:layout_height="?attr/actionBarSize"
                        app:popupTheme="@style/Theme.AppCompat.Light"
                        />
    
                    <android.support.design.widget.FloatingActionButton
                        android:layout_width="wrap_content"
                        android:layout_height="wrap_content"
                        android:layout_gravity="bottom|end"
                        android:layout_margin="16dp"
                        android:src="@mipmap/ic_launcher"
                        android:id="@+id/fab"
                        />
                </FrameLayout>
    
        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:layout_gravity="start">
    
                <android.support.design.widget.NavigationView
                    android:id="@+id/nav_view"
                    android:layout_width="match_parent"
                    android:layout_height="match_parent"
                    app:menu="@menu/nav_menu"
                    app:headerLayout="@layout/nav_header"
                    >
          </android.support.design.widget.NavigationView>
    
        </LinearLayout>
    
    </android.support.v4.widget.DrawerLayout>

     

    悬浮球下方会有一点阴影

    FloatingActionButton是在当前页面之上的

    所以会有投影

    使用app:elevation="8dp"指定悬浮高度

    指定一个高度值,值越大投影范围越大,投影效果越淡

    翻译亦然。

    点击事件:

     FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
            fab.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    Toast.makeText(MainActivity.this,"fab",Toast.LENGTH_LONG).show();
                    mdrawerLayout.openDrawer(GravityCompat.START);
                }
            });

     和普通按钮的使用方法一致

     9.4.2、Snackbar

    Snackbar不是Toast的替代品

    两者之间有者不同的应用场景

    Toast用于告诉用户现在发生了什么事情,同时用户只能被动接收这个事情,用户没办法进行选择

    Snackbar允许在提示中加一个按钮,当用户点击的适合可以进行一些额外的操作

     FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
            fab.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    mdrawerLayout.openDrawer(GravityCompat.START);
    
                    Snackbar.make(v,"Data delete ",Snackbar.LENGTH_SHORT)
                            .setAction("Undo", new View.OnClickListener() {
                                @Override
                                public void onClick(View v) {
                                    Toast.makeText(MainActivity.this,"数据恢复",Toast.LENGTH_LONG).show();
                                }
                            }).show();
    
                }
            });

    调用Snackerbar的make()方法来创建一个Snackbar对象

    make()方法:

    第一个参数是一个View,只需要当前页面布局的任意的一个View都可以

    第二个参数是Snackbar中显示的内容

    第三个参数是Snackbar的显示时长

    setAction()方法来设置一个动作

    让Snackbar不仅仅是一个提示,用于和用户进行交互

    在点击事件中里面弹出一个Toast提示

    最后调用show()方法让Snackbar显示出来

     

     9.4.3、CoordinatorLayout

    是一个加强版的FrameLayout

    这个布局Design  Support库提供的 

    普遍情况下与FrameLayout保持一致

    还有一些其他的使用

    CoordinatorLayout可以监听其所有子控件的各种事件

    然后自动帮助我们做出最为合理的响应

    实例:

    Snackbar提示将悬浮窗按钮遮挡住了

    如果能让CoordinatorLayout监听Snackbar的事件

    那么它会自动将内部的FloatingActionButton向上偏移

    从而保证不会被Snackbar遮挡住

    <?xml version="1.0" encoding="utf-8"?>
    <android.support.v4.widget.DrawerLayout
        xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        android:id="@+id/drawe_layout"
        android:layout_width="match_parent"
        android:layout_height="match_parent">
    
        <android.support.design.widget.CoordinatorLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent">
            <android.support.v7.widget.Toolbar
                android:id="@+id/toolbar"
                android:layout_width="match_parent"
                android:background="?attr/colorPrimary"
                android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
                android:layout_height="?attr/actionBarSize"
                app:popupTheme="@style/Theme.AppCompat.Light"
                />
    
    
            <android.support.design.widget.FloatingActionButton
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_gravity="bottom|end"
                android:layout_margin="16dp"
                android:src="@mipmap/ic_launcher"
                android:id="@+id/fab"
                />
        </android.support.design.widget.CoordinatorLayout>
    
        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:layout_gravity="start">
    
                <android.support.design.widget.NavigationView
                    android:id="@+id/nav_view"
                    android:layout_width="match_parent"
                    android:layout_height="match_parent"
                    app:menu="@menu/nav_menu"
                    app:headerLayout="@layout/nav_header"
                    >
    
        </android.support.design.widget.NavigationView>
    
        </LinearLayout>
    
    </android.support.v4.widget.DrawerLayout>

    9.5、卡片式布局

    此时的效果已经 比之前的实现更有显著的效果

    此时页面上还有一块空白区域

    通常用来防止应用的主题内容

    可以使用一些图片来填充这部分区域

    9.5.1、CradView

    CradView用于实现卡片式布局效果的重要控件

    由appcompat-v7提供

    实际上也是FrameLayout,只是提供了圆角和阴影的等效果,看上去会有立体感

    基本使用

    在CradView布局中放置一个TextView,这个TextView就会显示在一张卡片之中了

    需要引入依赖:

    添加了一个Glide库依赖

    是一个超级强大的 图片加载库

    不仅可以用于加载本地图片还能加载网络图片、GIF甚至是本地时评等

    用法非常简单

    首先修改:

    <?xml version="1.0" encoding="utf-8"?>
    <android.support.v4.widget.DrawerLayout
        xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        android:id="@+id/drawe_layout"
        android:layout_width="match_parent"
        android:layout_height="match_parent">
    
        <android.support.design.widget.CoordinatorLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent">
            <android.support.v7.widget.Toolbar
                android:id="@+id/toolbar"
                android:layout_width="match_parent"
                android:background="?attr/colorPrimary"
                android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
                android:layout_height="?attr/actionBarSize"
                app:popupTheme="@style/Theme.AppCompat.Light"
                />
    
    
            //中间的空白页面
            <android.support.v7.widget.RecyclerView
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                android:id="@+id/recycler_view"
                >
            </android.support.v7.widget.RecyclerView>
    
    
    
            <android.support.design.widget.FloatingActionButton
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_gravity="bottom|end"
                android:layout_margin="16dp"
                android:src="@mipmap/ic_launcher"
                android:id="@+id/fab"
                />
        </android.support.design.widget.CoordinatorLayout>
    
        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:layout_gravity="start">
    
                <android.support.design.widget.NavigationView
                    android:id="@+id/nav_view"
                    android:layout_width="match_parent"
                    android:layout_height="match_parent"
                    app:menu="@menu/nav_menu"
                    app:headerLayout="@layout/nav_header"
                    >
    
        </android.support.design.widget.NavigationView>
    
        </LinearLayout>
    
    </android.support.v4.widget.DrawerLayout>

     添加一个RecyclerView

    并且制定一个id

    新建一个显示图片的实体类:

    public class Images {
        private String name;
        private int imageId;
    
        public String getName() {
            return name;
        }
    
        public void setName(String name) {
            this.name = name;
        }
    
        public int getImageId() {
            return imageId;
        }
    
        public void setImageId(int imageId) {
            this.imageId = imageId;
        }
    }

     两个字段

    name表示名字

    id表示对应的图片id

    新建一个布局文件用于显示Images

    <?xml version="1.0" encoding="utf-8"?>
    <android.support.v7.widget.CardView
        xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="5dp"
        app:cardCornerRadius="4dp">
    
        <LinearLayout
            android:orientation="vertical"
            android:layout_width="match_parent"
            android:layout_height="wrap_content">
    
            <ImageView
                android:layout_width="wrap_content"
                android:layout_height="100dp"
                android:id="@+id/images_id"
                android:scaleType="centerCrop"
                />
    
    
            <TextView
                android:id="@+id/images_text"
                android:layout_gravity="center_horizontal"
                android:textSize="16sp"
                android:layout_margin="5dp"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content" />
    
        </LinearLayout>
    </android.support.v7.widget.CardView>

     是哦用CradView来作为最外层布局

    使得RecyclerView中的每个元素都是在卡片当中的

    CardView由于是一个FrameLayout

    因此没有什么方便的定位方式

    这里在进行潜入一个LinearLayout中放置具体的内容

    具体内容:

    1、放置图片的ImageView

    2、TextView用于放置是图片名字

    新建一个适配器:

    public class ImagesAdapter extends  RecyclerView.Adapter<ImagesAdapter.ViewHolder> {
    
        private Context mContext;
        private List<Images> mImagesList;
    
        public ImagesAdapter(List<Images> imagesList){
            mImagesList=imagesList;
        }
    
        @Override
        public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
            if (mContext == null){
                mContext=parent.getContext();
            }
            View view = LayoutInflater.from(mContext).inflate(R.layout.images_item,parent,false);
            return new ViewHolder(view);
        }
    
        @Override
        public void onBindViewHolder(ViewHolder holder, int position) {
            Images images = mImagesList.get(position);
            holder.textView.setText(images.getName());
            Glide.with(mContext).load(images.getImageId()).into(holder.imageView);
        }
    
        @Override
        public int getItemCount() {
            return mImagesList.size();
        }
    
        //内部类
        class ViewHolder extends RecyclerView.ViewHolder{
            CardView cardView;
            ImageView imageView;
            TextView textView;
    
            public ViewHolder(View itemView) {
                super(itemView);
                cardView = (CardView) itemView;
                imageView = (ImageView) itemView.findViewById(R.id.images_id);
                textView = (TextView) itemView.findViewById(R.id.images_text);
            }
        }
    }

    在MainActivity

    package com.example.ccrr.material;
    
    import android.support.annotation.NonNull;
    import android.support.design.widget.FloatingActionButton;
    import android.support.design.widget.NavigationView;
    import android.support.design.widget.Snackbar;
    import android.support.v4.view.GravityCompat;
    import android.support.v4.widget.DrawerLayout;
    import android.support.v7.app.ActionBar;
    import android.support.v7.app.AppCompatActivity;
    import android.os.Bundle;
    import android.support.v7.widget.GridLayoutManager;
    import android.support.v7.widget.RecyclerView;
    import android.support.v7.widget.Toolbar;
    import android.util.Log;
    import android.view.Menu;
    import android.view.MenuItem;
    import android.view.View;
    import android.widget.Toast;
    
    import java.util.ArrayList;
    import java.util.List;
    
    public class MainActivity extends AppCompatActivity {
    
    
        private DrawerLayout mdrawerLayout;
    
    
        private List<Images> imagesList = new ArrayList<>();
        private ImagesAdapter imagesAdapter;
    
        private void init(){
            for (int i=0;i<=10;i++){
                Images images = new Images();
                images.setName("图片" + i);
                images.setImageId(R.drawable.image);
                imagesList.add(images);
            }
        }
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.action_main);
         ...//卡片布局
            init();
            RecyclerView recyclerView = (RecyclerView) findViewById(R.id.recycler_view);
    
            GridLayoutManager layoutManager = new GridLayoutManager(this,2);
            recyclerView.setLayoutManager(layoutManager);
            imagesAdapter = new ImagesAdapter(imagesList);
            recyclerView.setAdapter(imagesAdapter);
    
    }
    ......
    }

    此时有一个很严重的问题:

    之前设置的Toolbar不见了

    其实式被RecylerView遮挡住了

    9.5.2、AppBarLayout

    上述的测试中可以发现RecyclerLayout会把Toolbar遮挡住

     解决方法:

    使用位偏移是唯一的解决方法

    即让RecylerView向下便宜一个Toolbar的高度

    从而不会遮挡住Toolbar

    项目中使用的不是简单的LinearLayout,使用的是CoordinatorLayout,会有更加简单的方法进行操纵

    这里使用Design Support库中的另外一个工具---AppBarLayout

    实际上是一个垂直方向上的LinearLayout

    内部做了很多组件的封装

    并且使用了一些Material Design的设计理念

    <?xml version="1.0" encoding="utf-8"?>
    <android.support.v4.widget.DrawerLayout
        xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        android:id="@+id/drawe_layout"
        android:layout_width="match_parent"
        android:layout_height="match_parent">
    
        <android.support.design.widget.CoordinatorLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent">
    
            <android.support.design.widget.AppBarLayout
                android:layout_width="match_parent"
                android:layout_height="wrap_content">
                <android.support.v7.widget.Toolbar
                    android:id="@+id/toolbar"
                    android:layout_width="match_parent"
                    android:background="?attr/colorPrimary"
                    android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
                    android:layout_height="?attr/actionBarSize"
                    app:popupTheme="@style/Theme.AppCompat.Light"
                    />
            </android.support.design.widget.AppBarLayout>
    
            //中间的空白页面
            <android.support.v7.widget.RecyclerView
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                android:id="@+id/recycler_view"
                app:layout_behavior="@string/appbar_scrolling_view_behavior"
                >
            </android.support.v7.widget.RecyclerView>
    
            <android.support.design.widget.FloatingActionButton
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_gravity="bottom|end"
                android:layout_margin="16dp"
                android:src="@mipmap/ic_launcher"
                android:id="@+id/fab"
                />
        </android.support.design.widget.CoordinatorLayout>
    
        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:layout_gravity="start">
    
                <android.support.design.widget.NavigationView
                    android:id="@+id/nav_view"
                    android:layout_width="match_parent"
                    android:layout_height="match_parent"
                    app:menu="@menu/nav_menu"
                    app:headerLayout="@layout/nav_header"
                    >
    
          </android.support.design.widget.NavigationView>
    
        </LinearLayout>
    
    </android.support.v4.widget.DrawerLayout>

    首先定义一个AppBarLayout,并且将Toolbar放在在其中 

    然后再RecylerLayout中使用app:layout_beehavior属性

    指定了一个布局的行为

    其中appbar_scrolling_view_behavior这个字符串由Design support库支持的

    此时完好的解决了遮挡问题

    但是Material Design的设计里面并没有体现出来

    实际上当RecylerView滚动的时候将事件通知给AppBarLayout

    只是我们还没进行小狐狸

    进一步优化

    实现AppBarLayout的效果:

    当AppBarLayout接收到滚动事件的时候

    它内部的子控件其实是可以指定如何去影响这些事件的

    通过app:layout_scrollFlags属性就能实现

    <?xml version="1.0" encoding="utf-8"?>
    <android.support.v4.widget.DrawerLayout
        xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        android:id="@+id/drawe_layout"
        android:layout_width="match_parent"
        android:layout_height="match_parent">
    
        <android.support.design.widget.CoordinatorLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent">
    
            <android.support.design.widget.AppBarLayout
                android:layout_width="match_parent"
                android:layout_height="wrap_content">
                <android.support.v7.widget.Toolbar
                    android:id="@+id/toolbar"
                    android:layout_width="match_parent"
                    android:background="?attr/colorPrimary"
                    android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
                    android:layout_height="?attr/actionBarSize"
                    app:popupTheme="@style/Theme.AppCompat.Light"
                    app:layout_scrollFlags="scroll|enterAlways|snap"
                    />
            </android.support.design.widget.AppBarLayout>
    
            //中间的空白页面
            <android.support.v7.widget.RecyclerView
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                android:id="@+id/recycler_view"
                app:layout_behavior="@string/appbar_scrolling_view_behavior"
                >
            </android.support.v7.widget.RecyclerView>
    
            <android.support.design.widget.FloatingActionButton
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_gravity="bottom|end"
                android:layout_margin="16dp"
                android:src="@mipmap/ic_launcher"
                android:id="@+id/fab"
                />
        </android.support.design.widget.CoordinatorLayout>
    
        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:layout_gravity="start">
    
                <android.support.design.widget.NavigationView
                    android:id="@+id/nav_view"
                    android:layout_width="match_parent"
                    android:layout_height="match_parent"
                    app:menu="@menu/nav_menu"
                    app:headerLayout="@layout/nav_header"
                    >
    
          </android.support.design.widget.NavigationView>
    
        </LinearLayout>
    
    </android.support.v4.widget.DrawerLayout>

    这里为Toolbbar添加一个app:layout_scrollFlags属性

    并且指定了三个属性值

    scroll:表示当RecylerView向上滚动的时候,Toolbar会跟着一起向上滚动实现隐藏

    enterAlways:表示当RecylerView向下滚动的时候,Toolbar会跟着一起向下滚动的时候重现显示

    snap:表示当当Toolbar还没有完全隐藏的时候或者还显示的时候,会根据当前滚动的距离,自动选择隐藏还是显示

    随着RecylerView向上滚动,Toolbar会随着其一起滚动消失

    随着RecyleView向下滚动,Toolbar又会重新出现

    这事Material Design中的一个重要的设计思想

    因为用户向上滚动的时候,其注意力肯定再RecyleView的内容上

    这个时候如果Toolbar还占据着屏幕控件,就会在一定程度上影响用户的体验

    将Toolbar隐藏可以让阅读体现出现

    9.5、下拉刷新

    下来刷新已经是一种常见的使用方式了

    比如淘宝、快手等软件都可以实现下来进行刷新

     谷歌为了让Android的下拉设计风范有一个统一的标准

    于是再Material Design中规定了一个官方的设计规范

    此时是拿来即用。

    SwipeRefreshLayout就是实现用于下拉刷新的核心类

    由support-v4提供的

    将实现下来刷新的功能控件放置再SwipeRefershLayout中

    就可以迅速让这个控件支持下来刷新

    再xml文件中引入:

      //下拉刷新
            <android.support.v4.widget.SwipeRefreshLayout
                android:id="@+id/swipe_layout"
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                app:layout_behavior="@string/appbar_scrolling_view_behavior"
                >
    
                //中间的空白页面
                <android.support.v7.widget.RecyclerView
                    android:layout_width="match_parent"
                    android:layout_height="match_parent"
                    android:id="@+id/recycler_view"
                    app:layout_behavior="@string/appbar_scrolling_view_behavior"
                    >
                </android.support.v7.widget.RecyclerView>
    
            </android.support.v4.widget.SwipeRefreshLayout>

    再RecylerView的外面嵌套了一层SwipeRefreshLayout

    此时的Recyleriew

    就拥有了下拉自动刷新的功能

    由于RecylerView已经变成了SwipeRefreshLayout的子控件

    因此可以使用app:layout_behavior声明布局行为现在也要移到SwipeRefreshLayout中

    MainActivity

    public class MainActivity extends AppCompatActivity {
        private DrawerLayout mdrawerLayout;
    
        private List<Images> imagesList = new ArrayList<>();
        private ImagesAdapter imagesAdapter;
    
        //下来刷新
        private SwipeRefreshLayout swipeRefreshLayout;
        private void init(){
            for (int i=0;i<=10;i++){
                Images images = new Images();
                images.setName("图片" + i);
                images.setImageId(R.drawable.image);
                imagesList.add(images);
            }
        }
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.action_main);
    
            Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
            setSupportActionBar(toolbar );
    
            mdrawerLayout = (DrawerLayout) findViewById(R.id.drawe_layout);
            ActionBar actionBar = getSupportActionBar();
            if (actionBar != null){
                actionBar.setDisplayHomeAsUpEnabled(true);
                actionBar.setHomeAsUpIndicator(R.mipmap.ic_launcher);
            }
    
            NavigationView navigationView = (NavigationView) findViewById(R.id.nav_view);
            navigationView.setCheckedItem(R.id.nav_call);
            navigationView.setNavigationItemSelectedListener(new NavigationView.OnNavigationItemSelectedListener() {
                @Override
                public boolean onNavigationItemSelected(@NonNull MenuItem item) {
                    mdrawerLayout.closeDrawers();
    
                    return true;
                }
            });
    
            FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
            fab.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    //Toast.makeText(MainActivity.this,"fab",Toast.LENGTH_LONG).show();
                    mdrawerLayout.openDrawer(GravityCompat.START);
    
                    Snackbar.make(v,"Data delete ",Snackbar.LENGTH_SHORT)
                            .setAction("Undo", new View.OnClickListener() {
                                @Override
                                public void onClick(View v) {
                                    Toast.makeText(MainActivity.this,"数据恢复",Toast.LENGTH_LONG).show();
                                }
                            }).show();
    
                }
            });
    
            //卡片布局
            init();
            RecyclerView recyclerView = (RecyclerView) findViewById(R.id.recycler_view);
    
            GridLayoutManager layoutManager = new GridLayoutManager(this,2);
            recyclerView.setLayoutManager(layoutManager);
            imagesAdapter = new ImagesAdapter(imagesList);
            recyclerView.setAdapter(imagesAdapter);
    
    
            //下拉刷新
            swipeRefreshLayout= (SwipeRefreshLayout) findViewById(R.id.swipe_layout);
            swipeRefreshLayout.setColorSchemeResources(R.color.colorPrimary);
            swipeRefreshLayout.setOnRefreshListener(new SwipeRefreshLayout.OnRefreshListener() {
                @Override
                public void onRefresh() {
                    refreshImages();
                }
            });
        }
    
        private  void refreshImages(){
            new Thread(new Runnable() {
                @Override
                public void run() {
                    try{
                        Thread.sleep(1000);
                    }catch (Exception e){
                        e.printStackTrace();
                    }
                    runOnUiThread(new Runnable() {
                        @Override
                        public void run() {
    
                            for (int i=11;i<=25;i++){
                                Images images = new Images();
                                images.setName("图片" + i);
                                images.setImageId(R.drawable.image);
                                imagesList.add(images);
                            }
                            imagesAdapter.notifyDataSetChanged();;
                            swipeRefreshLayout.setRefreshing(false);
                        }
                    });
                }
            }).start();
        }
    
        @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:
                    Toast.makeText(this,"home",Toast.LENGTH_LONG).show();
                    mdrawerLayout.openDrawer(GravityCompat.START);
                    break;
                case R.id.Back:
                    Toast.makeText(this,"Back",Toast.LENGTH_LONG).show();
                    break;
                case R.id.delete:
                    Toast.makeText(this,"delete",Toast.LENGTH_LONG).show();
                    break;
                case R.id.setting:
                    Toast.makeText(this,"setting",Toast.LENGTH_LONG).show();
                    break;
                default:
                    break;
            }
            return true;
        }
    }

    首先通过findViewById()方法拿到SwipeRefreshLayout的实例

    调用setColorSchemeResources()方法来设置下拉进度条的颜色,这里使用主题中的colorPrimary作为进度条的颜色

    在调用setOnRefreshListener()方法设置一个下来刷新器

    当触发下拉操作就会回调这个监听器的onRefresh()方法,在这里处理具体的逻辑

    通常情况下onRefresh()方法中应该是去网络上请求最新的数据

    然后将这些数据展示出来

    此时使用refreshImages()方法进行添加新的数据进行显示

    首先开启开启一个线程,让线程沉睡两秒钟

    因为刷新操作的速度非常快

    沉睡之后

    使用runOnUiThread()方法将线程切换到主线程

    从而看不到刷新的过程

    然后再产生的新的数据

    接着调用ImagesAdapter方法的notifyDataSetChanged()方法通知数据发生了变化

    最后调用SwipeRefreshLayout的setRefreshing()方法传入false,表示刷新事件结束,并且隐藏进度条

    9.7、可折叠式标题栏

    标题栏是使用Toolbar来编写的

    看上去和传统的ActionBar其实没什么区别

    只不过可以响应RecycleView的滚动事件进行隐藏和显示

    而Material Design中并没有限定标题栏必须是什么样子的

    事实上,可以根据自己的喜好随意定制标题栏的样式

    需要借助CollapsingToolbarLayout这个工具

    9.7.1、CoolapsingToolbarLayout

    CollapsingToolbarLayout是一个作用于Toolbar基础之上的布局

    由Design Support库提供的

    CollaspingToolbarLayout可以让Toolbar的效果变得更加丰富

    不仅仅展示一个标题栏,而是能够实现非常华丽的效果

    不过CollaspingLayout是不能独立存在的

    他在设计的时候就被限定只能作为AppBatLayout的直接子布局来使用

    而AppBarLayout又必须是CorrdinatorLayout的子布局

    .................................

  • 相关阅读:
    相交链表 3种方法
    旋转数组 空间复杂度为O(1) 的2 种方法 + 1种空间复杂度O(n)
    各种sort排序总结
    我写过的bug...
    裸BFS题若干
    luence
    mysql 找出外键等约束
    mysql (8.0 或以下)数据 卸载, 安装, 创建用户, 赋权
    navicat 导入sql文件 的正确方法
    [转] git clone 远程分支
  • 原文地址:https://www.cnblogs.com/Mrchengs/p/10726756.html
Copyright © 2020-2023  润新知