乐趣区

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

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

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

NestedScrolling 机制深入解析

一步步带你读懂 CoordinatorLayout 源码

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

ViewPager,ScrollView 嵌套 ViewPager 滑动冲突解决

自定义 behavior – 完美仿 QQ 浏览器首页,美团商家详情页


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 的视觉特差


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 是静态的

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

下面我们一起来看一下 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](http://www.jianshu.com/p/f418…

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

欢迎大家关注我的微信公众号号 stormjun949(徐公码字),即可关注。目前专注于 Android 开发,主要分享 Android 开发相关知识和一些相关的优秀文章,包括个人总结,职场经验等。

退出移动版