玩转AppBarLayout,更酷炫的顶部栏

作者: huachao1001 | 来源:发表于2016-06-02 14:05 被阅读76248次

    我的CSDN博客同步发布:玩转AppBarLayout,更酷炫的顶部栏

    上一篇文章[《CoordinateLayout的使用如此简单 》]上一篇文章《CoordinateLayout的使用如此简单 》CoordinateLayout的使用做了讲解,今天我们再讲解常常与其一起使用的几个ViewAppBarLayoutCollapsingToolbarLayout以及Toolbar。一下子出现3个陌生的View,是不是觉得很慌张~,很多人都写了这几个布局的使用,但是他们却没有有针对性的单独讲解每个View的作用以及如何使用,我看的很多文章都是一上来就把AppBarLayoutCollapsingToolbarLayout以及Toolbar写到一个布局里面去,然后一个一个布局属性去说,一下子感觉好混乱,本文是从Toolbar开始说起,最终让你把这3个View彻底掌握下来!

    其实,这三个View都是针对我们以往常用的ActionBar的,就是针对我们的App的顶部的Bar玩各种花样~我们往下看,看看他们把我们的App的"顶部栏"玩出个什么花样!

    1 Toolbar

    Toobar主要是用来替换ActionBar的,换句话说,ActionBar能做的,Toolbar都能做。如果你对ActionBar的使用比较熟悉,你会发现Toolbar使用起来非常简单。ok,既然是替换,当然用Toolbar的时候就得先去把ActionBar给隐藏掉啦~

    隐藏ActionBar的方法有很多,可以通过代码的方式隐藏,也可以通过配置文件的方式,我们主要是通过配置文件的方式来隐藏。在我们的styles.xml文件中的AppTheme标签中加入如下两行:

    <item name="windowActionBar">false</item>
    <item name="windowNoTitle">true</item>
    

    当然了,你也可以新建一个<style>标签,将上面两行代码加入,并且将这个新建的标签作为<application>theme。还可以选择通过将AppTheme的parent设置为Theme.AppCompat.Light.NoActionBar的方式。方法很多,可以自己随便选。

    接下来就是将Toolbar放入到布局文件(没啥好解释的):

      <android.support.v7.widget.Toolbar
            android:id="@+id/toolbar"
            android:layout_width="match_parent"
            android:background="?attr/colorPrimary"
            android:layout_height="?android:attr/actionBarSize"  />
    

    最后将Toobar作为“ActionBar”来用

     Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
            toolbar.setTitle("这里是Title");
            toolbar.setSubtitle("这里是子标题");
            toolbar.setLogo(R.drawable.icon); 
            setSupportActionBar(toolbar);
    
    

    可以对Toolbar设置Logo、标题、子标题等等还有很多其他的设置,自己去慢慢玩,这里不提啦。当然了,也可以在布局文件中设置这些,在布局文件设置就不写啦,hongyang大神有篇博客写的挺好的《 Android 5.x Theme 与 ToolBar 实战 》可以去参考一下。

    看看效果:

    Toolbar

    如果Toolbar仅仅是用来对以往的ActionBar做一次替换,那也太没创意啦!完全没必要去替换了,因为它们表现出来的都是一样的,而且并没有让我们觉得用起来比ActionBar方便。那为啥要替换呢,总应该有他的理由吧:ActionBar是固定在顶部,并不能移动,我觉得这是最大的不好,而我们的ToolBar可以让我们随便摆放,就就可以带来很多灵活性和效果啦!

    正如你所看的这样,Toolbar根本就不够看的,一点都不复杂。接下来我们继续学习在Toolbar上面再套一层父View,让Toolbar更有互动性。

    2 AppBarLayout

    AppBarLayout继承自LinearLayout,布局方向为垂直方向。所以你可以把它当成垂直布局的LinearLayout来使用。AppBarLayout是在LinearLayou上加了一些材料设计的概念,它可以让你定制当某个可滚动View的滚动手势发生变化时,其内部的子View实现何种动作。

    请注意:上面提到的某个可滚动View,可以理解为某个ScrollView。怎么理解上面的话呢?就是说,当某个ScrollView发生滚动时,你可以定制你的“顶部栏”应该执行哪些动作(如跟着一起滚动、保持不动等等)。那某个可移动的View到底是哪个可移动的View呢?这是由你自己指定的!如何指定,我们后面说。

    2.1 AppBarLayout子View的动作

    内部的子View通过在布局中加app:layout_scrollFlags设置执行的动作,那么app:layout_scrollFlags可以设置哪些动作呢?分别如下:

    (1) scroll:值设为scroll的View会跟随滚动事件一起发生移动。

    什么意思呢?简单的说,就是当指定的ScrollView发生滚动时,该View也跟随一起滚动,就好像这个View也是属于这个ScrollView一样。

    一张gif足以说明:

    scroll

    对应的布局文件

    <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:layout_height="?android:attr/actionBarSize"
                android:background="?attr/colorPrimary"
                app:layout_scrollFlags="scroll" />
    </android.support.design.widget.AppBarLayout>
    

    (2) enterAlways:值设为enterAlways的View,当ScrollView往下滚动时,该View会直接往下滚动。而不用考虑ScrollView是否在滚动。

    看个动画片(Y(o)Y)(ToolBar高度设为:?android:attr/actionBarSize,app:layout_scrollFlags="scroll|enterAlways"):

    scroll|enterAlways

    (3) exitUntilCollapsed:值设为exitUntilCollapsed的View,当这个View要往上逐渐“消逝”时,会一直往上滑动,直到剩下的的高度达到它的最小高度后,再响应ScrollView的内部滑动事件。

    怎么理解呢?简单解释:在ScrollView往上滑动时,首先是View把滑动事件“夺走”,由View去执行滑动,直到滑动最小高度后,把这个滑动事件“还”回去,让ScrollView内部去上滑。看个gif感受一下(图中将高度设的比较大:200dp,并将最小高度设置为?android:attr/actionBarSize,app:layout_scrollFlags="scroll|exitUntilCollapsed"):

    scroll|exitUntilCollapsed

    (4) enterAlwaysCollapsed:是enterAlways的附加选项,一般跟enterAlways一起使用,它是指,View在往下“出现”的时候,首先是enterAlways效果,当View的高度达到最小高度时,View就暂时不去往下滚动,直到ScrollView滑动到顶部不再滑动时,View再继续往下滑动,直到滑到View的顶部结束。

    来个gif感受一下(图中将高度设的比较大:200dp,并将最小高度设置为?android:attr/actionBarSize,app:layout_scrollFlags="scroll|enerAlways|enterAlwaysCollapsed"):

    scroll|enerAlways|enterAlwaysCollapsed

    2.2 将AppBarLayout与ScrollView关联起来

    前面说了一直反复说“当ScrollView发生滚动时”,那么怎么将AppBarLayout与ScrollView关联起来呢?我们注意到,AppBarLayout与ScrollView之间动作“相互依赖”,这不就是我们上一篇《CoordinateLayout的使用如此简单 》所学的内容吗?把ScrollView和AppBarLayout作为CoordinateLayout的子View,然后编写一个Behavior,在这个Behavior里面判断当前的操作是应该让ScrollView时刻保持在AppBarLayout之下(即只要改变AppBarLayout的位置就可以一起滑动),还是应该让ScrollView内部滚动而不让AppBarLayout位置发生变化等等这些需求,都是可以在Behavior里面处理的。你可以去针对你的ScrollView编写Behavior。然而,我们看到我们的AppBarLayout事先的功能比较复杂,如果我们自己去定义这样的效果,代码非常复杂,还要考虑很多方面,好在Android帮我们写好啦,我们直接用就是了,这个ScrollView就是NestedScrollView,请注意,它并没有继承ScrollView,它继承的是FrameLayout,但是它实现的效果把它可以看成是ScrollView。

    把NestedScrollView放入到我们的layout文件里面就可以啦~~,很方便

     <android.support.v4.widget.NestedScrollView
    
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            app:layout_behavior="@string/appbar_scrolling_view_behavior">
    
           <!--将你的内容放在这里-->
    
        </android.support.v4.widget.NestedScrollView>
    
    

    有没有注意到有个属性:app:layout_behavior="@string/appbar_scrolling_view_behavior",它就是指定Behavior的,appbar_scrolling_view_behavior对应的类的名称是:android.support.design.widget.AppBarLayout$ScrollingViewBehavior感兴趣的可以去分析源码。

    好了,我们现在会用AppBarLayout啦~是不是发现用起来so easy!接下来我们把剩下CollapsingToolbarLayout的给"消化"掉!

    3 CollapsingToolbarLayout

    CollapsingToolbarLayout是用来对Toolbar进行再次包装的ViewGroup,主要是用于实现折叠(其实就是看起来像伸缩~)的App Bar效果。它需要放在AppBarLayout布局里面,并且作为AppBarLayout的直接子ViewCollapsingToolbarLayout主要包括几个功能(参照了官方网站上内容,略加自己的理解进行解释):

    (1) 折叠Title(Collapsing title):当布局内容全部显示出来时,title是最大的,但是随着View逐步移出屏幕顶部,title变得越来越小。你可以通过调用setTitle函数来设置title。

    (2)内容纱布(Content scrim):根据滚动的位置是否到达一个阀值,来决定是否对View“盖上纱布”。可以通过setContentScrim(Drawable)来设置纱布的图片.

    (3)状态栏纱布(Status bar scrim):根据滚动位置是否到达一个阀值决定是否对状态栏“盖上纱布”,你可以通过setStatusBarScrim(Drawable)来设置纱布图片,但是只能在LOLLIPOP设备上面有作用。

    (4)视差滚动子View(Parallax scrolling children):子View可以选择在当前的布局当时是否以“视差”的方式来跟随滚动。(PS:其实就是让这个View的滚动的速度比其他正常滚动的View速度稍微慢一点)。将布局参数app:layout_collapseMode设为parallax

    (5)将子View位置固定(Pinned position children):子View可以选择是否在全局空间上固定位置,这对于Toolbar来说非常有用,因为当布局在移动时,可以将Toolbar固定位置而不受移动的影响。 将app:layout_collapseMode设为pin

    了解这些概念后,我们来看看布局吧~

    <?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.support.design.widget.AppBarLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar">
    
            <android.support.design.widget.CollapsingToolbarLayout
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                app:expandedTitleMarginEnd="64dp"
                app:expandedTitleMarginStart="48dp"
                app:layout_scrollFlags="scroll|exitUntilCollapsed">
    
                <ImageView
                    android:id="@+id/main.backdrop"
                    android:layout_width="wrap_content"
                    android:layout_height="300dp"
                    android:scaleType="centerCrop"
                    android:src="@drawable/material_img"
                    app:layout_collapseMode="parallax" />
    
                <android.support.v7.widget.Toolbar
                    android:id="@+id/toolbar"
                    android:layout_width="match_parent"
                    android:layout_height="?android:attr/actionBarSize"
                    app:layout_collapseMode="pin"  />
    
            </android.support.design.widget.CollapsingToolbarLayout>
        </android.support.design.widget.AppBarLayout>
    
        <android.support.v4.widget.NestedScrollView
    
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:paddingTop="50dp"
            app:layout_behavior="@string/appbar_scrolling_view_behavior">
    
            <TextView
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:text="@string/my_txt"
                android:textSize="20sp" />
    
        </android.support.v4.widget.NestedScrollView>
    
    </android.support.design.widget.CoordinatorLayout>
    
    
    
    

    上面的都看得懂吧,每个陌生的属性都是讲过的哦,忘记了的话回头看,稍微解释一下,图片被设置为有视差的滑动,Toolbar设置为固定不动,另外,CollapsingToolbarLayout会对title进行放大和缩小,我们看看效果吧~

    CollapsingToolbarLayout效果

    如果你希望拖动过程中状态栏是透明的,可以在CollapsingToolbarLayout中加 app:statusBarScrim="@android:color/transparent",并且在onCreate中调用getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS)将状态栏设置为透明就好啦~

    献上源码,请笑纳:http://download.csdn.net/detail/huachao1001/9538934

    相关文章

      网友评论

      • 小生来咯:大哥,那个title的字体太大了,怎么控制俩个title的大小
      • yi923:第一个fragment有bug
      • Vivi成长吧:写得太好了!
      • 薛定谔的_猫:写的真好,感谢作者,谢谢!
      • Shmily鱼:写的很棒,学习了! 但是我有个疑问,想请教下: 在 2.1 AppBarLayout子View的动作中,举例的四种情况描述中,关于AppBarLayout与ScrollView的滚动关系, 我理解是AppBarLayout的滚动依赖ScrollView的滚动,也就是说,在依赖关系中,ScrollView是Dependency,而AppBarLayout是child.而在2.2的开头描述中,与此段布局代码中: <android.support.v4.widget.NestedScrollView

        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        app:layout_behavior="@string/appbar_scrolling_view_behavior">

        <!--将你的内容放在这里-->

        </android.support.v4.widget.NestedScrollView>
        却是ScrollView依赖AppBarLayout而滚动.感觉有些矛盾了.还是2.1之前的依赖关系我没有理解清楚?
      • Wing_Li:非常棒
      • dabf8e02a0f4:讲解简单细致,图文并茂,小白受教了:clap:
      • 伟大的小炮殿下:有个问题,在上篇文章里是btn跟着tempView移动,所以dependancy是tempView,然后这边应该是title跟着ScrollView移动,那为什么是ScrollView加了behavior,在我看来,是title依赖ScrollView才是,楼主或者哪位大神能麻烦解答下疑惑么
      • 7e625c8993c4:为什么 app:layout_scrollFlags="scroll" 这个属性不起作用
      • 疯狂搬砖者:希望博主以后代码上传到gitHub上面更方便查阅
      • 一顆糖与半杯水:学习了,666的
      • 庞哈哈哈12138:正好需要,多谢
      • 冰凌孤星:好文. 把折叠布局撸的很清晰明了
      • android_cyw:楼主,下载你的源码,发现NestedScrollView向上滑动时有点卡顿啊,会卡在某个位置,并不能一下子滑动到toolbar的高度
      • YZune:大神你好,我想问一下最后那张效果图中,toolbar收缩之后以图片为背景是怎么做到的呀?我做出来的效果是收缩之后显示为主题色。谢谢。
      • 一切从简_e156:这个写的还是很通俗易懂的,很不错
      • Lonie233:有没有办法让appbarlayout 里的view 部分滚动? 比如appbarlyout里有两个view A,B, 滚动的时候, A 置顶,B 滚动上去?
        Lonie233:@泽毛 貌似是不行的,我进去看源码了,是把他们当成一个整体view来处理的,只能做相当于垂直方向的平移操作,
        // As soon as a view doesn't have the scroll flag, we end the range calculation.
        // This is because views below can not scroll under a fixed view.
        泽毛:可以试一下,在 AppBarLayout 里面,按 B A 的顺序排列,然后只给 B 设置 app:layout_scrollFlags="scroll",A不用设置。
      • 3d2c340bbed7:写的通俗易懂,很好理解,谢谢分享
      • 猪上村树:太赞了。。。
      • tigerkint:我有个问题,coordinatelayout里面可不可以嵌套listview,嵌套listview时能不能在向上滑动listview时先把上面的布局给压缩了,向下滑动listview再显示呢?我把listview写在nestscrollview里面只会显示一行。
        思婷:滑动coordinatelayout中的listview时,拦截coordinatelayout的触发事件
        阿西吧喽:如果是列表的话可以用recycleview替代listview,默认就有联动属性
        泽毛:如果是 Android 5.0 以后的,可以试一下把 ListView 放在 CoordinatorLayout 的根布局下面,然后给 ListView 加上 android:nestedScrollingEnabled="true"
      • 281072019671:写的很棒,通俗易懂。这几个view之间的关系都理解了。但我有一个问题,那就是如何做到默认情况下就是收起的效果。目前是一打开图片就是展开的,有什么办法能够让一打开就是收起的。我尝试了各种flag结合都没能实现这个效果。
      • Ning1994:写的通俗易懂,很好理解,作者很棒,赞一个啊
      • Oterman:不错不错 讲得很好
      • 6c94521b4765:讲解的不错,继续消化。谢谢分享!
      • Android绝世小菜鸟:大佬,转了。。
      • 青蛙要fly:比如在你最后的那个例子中,CollapsingToolbarLayout包含了ImageView和ToolBarLayout。然后enterAlwaysCollapsed和exitUntilCollapsed所谓可以设置minHeight中,这个minHeight应该设置在哪个中,然后我刚设置在CollapsingToolbarLayout中,好像也没看出有什么变化
      • 2ed47d88ae96:学习了,图文并茂 :clap:
      • 69edf3facddb:谢谢你的博客,让我受益匪浅,加油!
      • b0e59a682e63:CollapsingToolbarLayout下的子view使用视差滚动的模式的话,常常搭配layout_collapseParallaxMultiplier属性来设置视差乘数。视差滚动其实是对view设置了一个位移动画,视差乘数设置该子view的位移动画相较于其他view移动速度倍率。文中对视差滚动属性的ps描述有点问题:
        “PS:其实就是让这个View的滚动的速度比其他正常滚动的View速度稍微慢一点”
        实际上当视差乘数设置的比较大时,这个视差的效果是移动速度相较于其他view快很多,视差乘数决定了位移动画的相对速度。
      • Fritz_Xu:写的很好.我看了下你的demod和AppBarLayout里面的ScrollingViewBehavior代码,AppBarLayout作为dependency,但是我们平时都是在NestedScrollView里面滑动的,在AppBarLayout里面的代码貌似没有找到Touch事件的获取.AppBarLayout到底是怎么影响到NestedScrollView的,博主大大可以简单说下吗?
        b0e59a682e63:@武械 你可以关注下behavior和coordinatorLayout,搞清楚这两者之间的关系,你的问题就迎刃而解。
        实际上AppBarLayout内部自己实现了一个behavior,当CoordinatorLayout下的NestedScrollView触发移动事件后,CoordinatorLayout会通知其直接子view的behavior。所以滑动事件的获取是父容器通知到view设置的behavior上的。
        以上,你就会发现,必须使用CoordinatorLayout作为父容器,然后使用behavior作为中间人才能将CoordinatorLayout下的可滑动直接子view的滑动事件传递到其他子view,从而实现view联动。
      • danieldai:蛮棒的
      • d45390a4c552:滑动下方文字,没有滑动效果哦
      • 8d540151b72d:写的很棒
      • 肚子好饿:楼主,最后一个效果图,文字上方的空白处,这是Bug吗?
        肚子好饿:原来有个padingtop....是我眼拙了...
      • 3f96bcd1651f:讲的实在是太好了,图文并茂,大赞。网上其他的文章写得很乱,连作者们其实都没搞懂那个srollFlags各自的作用。即使看了官方文档也没搞懂。真是太感谢你了。
        huachao1001: @3f96bcd1651f 感谢您的打赏,能让大家有收获就好
      • 7d0a14cf1595:为什么会出现就是感觉卡顿的感觉!
        ChangQin: @没有梦想的boy 我感觉是图片太大了,放个小图就一点不卡了
      • dbe5f5be8b51:清晰明了
      • xiasuhuei321:这篇文不错啊~
      • 落魄的安卓开发:好!表达的灰常清晰
      • Jsonzhang:非常感谢,写的非常好,学习了
      • 22747556ac69: 博主 请教个问题 怎样能手动控制AppBarLayout的折叠状态 我现在有个bug 就是如果折叠是收起的状态 刷新列表 recycleview刷新后折叠的部分还是折叠状态 我想刷新列表后手动将它显示
      • 大凡:mark
      • 小胡闹:博主你好,我想问一下,如你最后一张gif,在toolbar最大化的时候,如何将状态栏透明?就是说,显示图片的背景,而不是蓝色;
        恁月:@小胡闹 http://www.imooc.com/video/6099 这里有篇单独讲StatusBar的文章,不知道跟作者说的效果会不会有冲突,可以参考一下
        小胡闹:@huachao1001 这样的话,在4.4中是有显示问题的
        huachao1001:@小胡闹 可以使用fitsSystemWindows属性并且将状态栏设置为透明
      • 05310ca6a6e0:谢谢分享,图文并茂,很好!
        huachao1001:@bigFlyingBird 感谢支持~

      本文标题:玩转AppBarLayout,更酷炫的顶部栏

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