美文网首页AndroidAndroidAndroid 开发收集的一些东西
使用CoordinatorLayout打造各种炫酷的效果

使用CoordinatorLayout打造各种炫酷的效果

作者: 程序员徐公 | 来源:发表于2016-10-19 13:39 被阅读51716次

    使用CoordinatorLayout打造各种炫酷的效果

    自定义Behavior —— 仿知乎,FloatActionButton隐藏与展示

    NestedScrolling 机制深入解析

    一步步带你读懂 CoordinatorLayout 源码

    自定义 Behavior ——仿新浪微博发现页的实现

    CoordinatorLayout简介

    CoordinatorLayout是在 Google IO/15 大会发布的,遵循Material 风格,包含在 support Library中,结合AppbarLayout, CollapsingToolbarLayout等 可 产生各种炫酷的效果

    CoordinatorLayout简介通常用来 干什么

    Google官方地址

    CoordinatorLayout is intended for two primary use cases:

    As a top-level application decor or chrome layout

    As a container for a specific interaction with one or more child views

    简单来说就是

    • 作为最上层的View
    • 作为一个 容器与一个或者多个子View进行交互

    下面我们一起先来看一下我们实现的效果图

    动态图

    结合ToolBar

    结合ViewPager

    ViewPager

    结合ViewPager的视觉特差


    AppBarLayout

    它是继承与LinearLayout的,默认 的 方向 是Vertical

    类型 说明
    int SCROLL_FLAG_ENTER_ALWAYS When entering (scrolling on screen) the view will scroll on any downwards scroll event, regardless of whether the scrolling view is also scrolling.
    int SCROLL_FLAG_ENTER_ALWAYS_COLLAPSED An additional flag for 'enterAlways' which modifies the returning view to only initially scroll back to it's collapsed height.
    int SCROLL_FLAG_EXIT_UNTIL_COLLAPSED When exiting (scrolling off screen) the view will be scrolled until it is 'collapsed'.
    int SCROLL_FLAG_SCROLL The view will be scroll in direct relation to scroll events.
    int SCROLL_FLAG_SNAP Upon a scroll ending, if the view is only partially visible then it will be snapped and scrolled to it's closest edge.
    类型 说明
    int SCROLL_FLAG_ENTER_ALWAYS W((entering) / (scrolling on screen))下拉的时候,这个View也会跟着滑出。
    int SCROLL_FLAG_ENTER_ALWAYS_COLLAPSED 另一种enterAlways,但是只显示折叠后的高度。
    int SCROLL_FLAG_EXIT_UNTIL_COLLAPSED ((exiting) / (scrolling off screen))上拉的时候,这个View会跟着滑动直到折叠。
    int SCROLL_FLAG_SCROLL 这个View将会响应Scroll事件
    int SCROLL_FLAG_SNAP 在Scroll滑动事件结束以前 ,如果这个View部分可见,那么这个View会停在最接近当前View的位置

    我们可以通过两种 方法设置这个Flag

    • 方法一
     setScrollFlags(int) 
    
    • 方法二
     app:layout_scrollFlags="scroll|enterAlways"
    

    注意事项

    AppBarLayout必须作为CoordinatorLayout的直接子View,否则它的大部分功能将不会生效,如layout_scrollFlags等。

    首先我们先来看一下我们 效果图一是怎样实现的

    代码

    <android.support.design.widget.CoordinatorLayout
        android:id="@+id/main_content"
        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.design.widget.AppBarLayout
            android:id="@+id/appbar"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar">
    
            <android.support.v7.widget.Toolbar
                android:id="@+id/toolbar"
                android:layout_width="match_parent"
                android:layout_height="?attr/actionBarSize"
                android:background="?attr/colorPrimary"
                app:layout_scrollFlags="scroll|enterAlways"
                app:popupTheme="@style/ThemeOverlay.AppCompat.Light"/>
    
           .
    
    
        </android.support.design.widget.AppBarLayout>
    
        <android.support.v7.widget.RecyclerView
            android:id="@+id/recyclerView"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            app:layout_behavior="@string/appbar_scrolling_view_behavior"/>
    
        <android.support.design.widget.FloatingActionButton
            android:id="@+id/fab"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="end|bottom"
            android:layout_margin="15dp"
            android:src="@drawable/add_2"/>
    
    </android.support.design.widget.CoordinatorLayout>
    
    

    思路 分析

    从图中我们可以知道 layout_scrollFlags="scroll|enterAlways,
    前面已经说到layout_scrollFlags=scroll的时候,这个View会 跟着 滚动 事件响应,
    layout_scrollFlags=“enterAlways”的时候 这个View会响应下拉事件
    所以呈现出来的结果应该是我们在上拉的时候toolBar 会隐藏,下拉的时候toolBar会出来

    那如果当我们的toolBar 等于 app:layout_scrollFlags="scroll|snap"的时候 ,
    layout_scrollFlags=scroll的时候,这个View会 跟着 滚动 事件响应,
    layout_scrollFlags=“snap”的时候 在Scroll滑动事件结束以前 ,如果这个View部分可见,那么这个View会停在最接近当前View的位置。
    综上呈现的效果如下,代码见ToolBarSampleSnar的布局文件

    结合ViewPager

    布局代码如下

    <android.support.design.widget.CoordinatorLayout
        android:id="@+id/main_content"
        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.design.widget.AppBarLayout
            android:layout_width="match_parent"
            android:layout_height="250dp">
    
    
            <ImageView android:layout_width="match_parent"
                       android:layout_height="200dp"
                       android:background="?attr/colorPrimary"
                       android:scaleType="fitXY"
                       android:src="@drawable/tangyan"
                       app:layout_scrollFlags="scroll|enterAlways"/>
    
            <android.support.design.widget.TabLayout
                android:id="@+id/tabs"
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:layout_alignParentBottom="true"
                android:background="?attr/colorPrimary"
                app:tabIndicatorColor="@color/colorAccent"
                app:tabIndicatorHeight="4dp"
                app:tabSelectedTextColor="#000"
                app:tabTextColor="#fff"/>
    
        </android.support.design.widget.AppBarLayout>
    
    
        <android.support.v4.view.ViewPager
    
            android:id="@+id/viewpager"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            app:layout_behavior="@string/appbar_scrolling_view_behavior"/>
    
        <android.support.design.widget.FloatingActionButton
            android:id="@+id/fab"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="end|bottom"
            android:layout_margin="15dp"
            android:src="@drawable/add_2"/>
    
    </android.support.design.widget.CoordinatorLayout>
    

    思路分析

    其实相对于前 一个例子,只是把 摆放RecyclerView 的位置替换成ViewPager而已,为了有页面导航器的效果,再使用 TabLayout而已,而TabLayout 在我们滑动的时候最终会停靠在 最顶部,是因为我们没有设置其layout_scrollFlags,即TabLayout是静态的

    运行以后,即可看到以下的结果

    ViewPager

    下面我们一起来看一下 TabLayout是怎样结合ViewPager直线 导航器的效果的

    代码注释 里面已经解释地很清楚了 ,这里我就不解释了

    public class ViewPagerSample extends AppCompatActivity {
    
        ViewPager mViewPager;
        List<Fragment> mFragments;
    
        String[] mTitles = new String[]{
                "主页", "微博", "相册"
        };
        private TabLayout mTabLayout;
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_third);
            // 第一步,初始化ViewPager和TabLayout
            mViewPager = (ViewPager) findViewById(R.id.viewpager);
            mTabLayout = (TabLayout) findViewById(R.id.tabs);
            setupViewPager();
        }
    
        private void setupViewPager() {
    
            mFragments = new ArrayList<>();
            for (int i = 0; i < mTitles.length; i++) {
                ListFragment listFragment = ListFragment.newInstance(mTitles[i]);
                mFragments.add(listFragment);
            }
            // 第二步:为ViewPager设置适配器
            BaseFragmentAdapter adapter =
                    new BaseFragmentAdapter(getSupportFragmentManager(), mFragments, mTitles);
    
            mViewPager.setAdapter(adapter);
            //  第三步:将ViewPager与TableLayout 绑定在一起
            mTabLayout.setupWithViewPager(mViewPager);
        }
    
    
    }
    
    

    如果我们想更改Indicator的相关样式,我们可以在布局文件里面使用

    <android.support.design.widget.TabLayout
        android:id="@+id/tabs"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_alignParentBottom="true"
        android:background="?attr/colorPrimary"
        app:tabIndicatorColor="@color/colorAccent"
        app:tabIndicatorHeight="4dp"
        app:tabSelectedTextColor="#000"
        app:tabTextColor="#fff"/>
    
    
    

    如果你不想使用Google 帮我们 封装好的控件的话,你也可以自己自定义一个控件,你可以参考我的这一篇博客仿网易新闻的顶部导航指示器


    在看例子结合ViewPager的视觉特差之前 ,我们需要先了解CollapsingToolbarLayout这个控件

    CollapsingToolbarLayout

    CollapsingToolbarLayout继承与FrameLayout,官网地址,请自备梯子。

    简单来说 ,CollapsingToolbarLayout是工具栏的包装器,它通常作为AppBarLayout的孩子。主要实现以下功能

    • Collapsing title(可以折叠 的 标题 )
    • Content scrim(内容装饰),当我们滑动的位置 到达一定阀值的时候,内容 装饰将会被显示或者隐藏
    • Status bar scrim(状态栏布)
    • Parallax scrolling children,滑动的时候孩子呈现视觉特差效果
    • Pinned position children,固定位置的 孩子

    下面我们一起来看一下几个常量

    常量 解释说明
    int COLLAPSE_MODE_OFF The view will act as normal with no collapsing behavior.(这个 View将会 呈现正常的结果,不会表现出折叠效果)
    int COLLAPSE_MODE_PARALLAX The view will scroll in a parallax fashion. See setParallaxMultiplier(float) to change the multiplier used.(在滑动的时候这个View 会呈现 出 视觉特差效果 )
    int COLLAPSE_MODE_PIN The view will pin in place until it reaches the bottom of the CollapsingToolbarLayout.(当这个View到达 CollapsingToolbarLayout的底部的时候,这个View 将会被放置,即代替整个CollapsingToolbarLayout)

    我们有两种方法可以设置这个常量,

    方法一:在代码中使用这个方法

    setCollapseMode(int collapseMode)
    

    方法 二:在布局文件中使用自定义属性

    app:layout_collapseMode="pin"
    

    到此 ,CollapsingToolbarLayout的一些重要属性已经讲解完毕,下面我们一起来看一下我们是怎样结合ViewPager实现视差效果的


    结合ViewPager的视觉特差

    布局代码

    <?xml version="1.0" encoding="utf-8"?>
    
    <android.support.design.widget.CoordinatorLayout
        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:background="@android:color/background_light"
        android:fitsSystemWindows="true"
    >
    
        <android.support.design.widget.AppBarLayout
            android:id="@+id/main.appbar"
            android:layout_width="match_parent"
            android:layout_height="300dp"
            android:fitsSystemWindows="true"
            android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
        >
    
            <android.support.design.widget.CollapsingToolbarLayout
                android:id="@+id/main.collapsing"
                android:layout_width="match_parent"
                android:layout_height="250dp"
                android:fitsSystemWindows="true"
                app:contentScrim="?attr/colorPrimary"
                app:expandedTitleMarginEnd="64dp"
                app:expandedTitleMarginStart="48dp"
                app:layout_scrollFlags="scroll|exitUntilCollapsed"
            >
    
                <ImageView
                    android:id="@+id/main.backdrop"
                    android:layout_width="match_parent"
                    android:layout_height="match_parent"
                    android:fitsSystemWindows="true"
                    android:scaleType="centerCrop"
                    android:src="@drawable/tangyan"
                    app:layout_collapseMode="parallax"
                />
    
                <android.support.v7.widget.Toolbar
                    android:id="@+id/toolbar"
                    android:layout_width="match_parent"
                    android:layout_height="?attr/actionBarSize"
                    app:layout_collapseMode="pin"
                    app:popupTheme="@style/ThemeOverlay.AppCompat.Light"
                />
            </android.support.design.widget.CollapsingToolbarLayout>
    
            <android.support.design.widget.TabLayout
                android:id="@+id/tabs"
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:layout_alignParentBottom="true"
                android:background="?attr/colorPrimary"
                app:tabIndicatorColor="@color/colorAccent"
                app:tabIndicatorHeight="4dp"
                app:tabSelectedTextColor="#000"
                app:tabTextColor="#fff"/>
        </android.support.design.widget.AppBarLayout>
    
    
        <android.support.v4.view.ViewPager
                android:id="@+id/viewpager"
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                app:layout_behavior="@string/appbar_scrolling_view_behavior">
    
        </android.support.v4.view.ViewPager>
    
    
        <android.support.design.widget.FloatingActionButton
            android:id="@+id/fab"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="end|bottom"
            android:layout_margin="15dp"
            android:src="@drawable/add_2"/>
    
    </android.support.design.widget.CoordinatorLayout>
    
    

    效果图如下

    思路解析

    • 结构图如图片所示,先说明CollapsingToolbarLayout的变化

      CollapsingToolbarLayout里面 包含ImageView 和ToolBar,ImageView的app:layout_collapseMode="parallax",表示视差效果,ToolBar的 app:layout_collapseMode="pin",当这个TooBar到达 CollapsingToolbarLayout的底部的时候,会代替整个CollapsingToolbarLayout显示

    • 接着说明TabLayout的变化

      从前面的描述我们已经知道当 没有指定app:layout_scrollFlags的时候,最终TabLayout会静止,不会随着滑动的 时候消失不见

    拓展

    如果我们仅仅 改变CollapsingToolbarLayout的app:layout_scrollFlags="scroll|exitUntilCollapsed|snap"的时候,其它代码不变,运行以后,我们将可以看到如下效果图


    总结

    这篇博客主要讲解了CoordinatorLayout,AppBarLayout,CollapsingToolbarLayout的一些相关属性。

    • 对于AppBarLayout,我们主要 讲解了这个属性app:layout_scrollFlags,设置不同 的属性我们可以在滚动的时候显示不同 的效果
    • 对于CollapsingToolbarLayout,我们主要讲解了app:layout_collapseMode这个属性,设置不同的值,我们可以让其子View呈现不同的 炫酷效果,如parallax和pin等

    CoordinatorLayout的相关用法还有很多,有兴趣 了解的请自行阅读: 官方文档地址


    题外话

    CoordinatorLayout这个控件真的很强大,使用它可以实现各种炫酷的效果,简化了开发者的许多工作,有能力的话可以去研究一下源码 ,看是怎样实现的?

    参考文章:android-[译]掌握CoordinatorLayout

    源码下载地址:https://github.com/gdutxiaoxu/CoordinatorLayoutExample.git

    最后的最后,卖一下广告,欢迎大家关注我的微信公众号,扫一扫下方二维码或搜索微信号 stormjun,即可关注。 目前专注于 Android 开发,主要分享 Android开发相关知识和一些相关的优秀文章,包括个人总结,职场经验等。


    相关文章

      网友评论

      • 木溪bo:双击66代表看到此篇博客心情
        markRao:波神果然天天学习,到处都是波神的足迹
      • honglei92:手机gif截屏是怎么截得啊
      • 697b90ed256b:你不觉得那个按钮一直在那里好奇怪?
      • 窒息的阿萨谢尔:正好解决了问题,给老哥点赞
      • F野鸽子:为啥都是特差特差的?
      • android_yiluo:我这个主布局是ViewPager,我在ViewPager里面写的ListView,为什么list滑动那个上面的控件不折叠
      • 举头望明月泣:ImageView 也能用 layout_scrollflags????
      • 举头望明月泣:good,认真学习了
      • super甜:看了之后一目了然 老哥有心了
      • 小孩程序员:楼主,我现在刷新或者加载更多,和这个悬浮吸顶,有点冲突。。求解
      • 2e40adfe9a54:学习晚了 ,开卷有益,感恩大牛
      • 9df2a6e99159:小哥,跟viewpager结合那个,能不能列表滑到顶后再下拉才把头部拉下来
      • 正阳Android:学习了,我打算学习了之后自己把理解的也写出来
      • d6116eca5146:我也喜欢唐嫣,哈哈
        程序员徐公:@Boyikia :smiley: 哈哈,一样的审美观
      • 筱南独舞:不错!学习了
      • 路飞4:加什么依赖啊?
      • 98efdc0a730a:你好 ,看了你的文章,非常棒,很想结识你,多向你请教,5.0特性我还不太熟。我的qq 751372658,可以的话加个好友。
      • d05229ae2f0e:你好,我打不开你的源码地址,有没有其他链接
      • Shur:学习了!!!
        程序员徐公:@Shur 一起学习
      • 笑里藏刀我不会AI:当recycleview滑动到头部的时候 会卡一下,怎么可以让他顺滑,不会卡一下的效果???
        HulkHulkHulk:@洛黄沙 大神,能教我下吗?怎么才能不卡住~ 直接顺滑到顶
        笑里藏刀我不会AI:@空白_2d44 解决了。
        HulkHulkHulk:兄弟 解决了吗 我也碰到了你这个问题
      • 13871066fc58:大兄弟,百忙之中能否解答我一个问题. 在ViewPager视觉特差效果中布局 给ToolBar设置 android:layout_height="?attr/actionBarSize" 我换成了warp_content. 为什么就没有效果了,求解答 :yum:
        程序员徐公:@13871066fc58 方便把源码贴 出来吗?
        13871066fc58:@xujun9411 好的 :smile:
        程序员徐公: @13871066fc58 我找个时间试一下,这两天有点忙,到时再答复你
      • 程序员徐公:互相学习
      • 巴图鲁:学习了

      本文标题:使用CoordinatorLayout打造各种炫酷的效果

      本文链接:https://www.haomeiwen.com/subject/hcdxuttx.html