美文网首页RecyclerViewiOS知识Android 动画
栗子——ListView+Head滑动显示标题

栗子——ListView+Head滑动显示标题

作者: 淡漠de人生 | 来源:发表于2016-06-08 00:33 被阅读1932次

项目里的效果拎出来做了个栗子,拿来和大家一起分享,一个很简单的栗子,不喜勿喷~

栗子配图.png

栗子惯例,先上GIF

栗子.gif

代码分析

其实核心的地方也是获取ListView的垂直滚动距离,在获取到滚动距离以后,根据垂直滚动距离来设置标题栏的背景透明度。

参考代码:感谢作者~ListView 获取精确的垂直滚动距离,但是有个BUG,待会说,代码中也会标明,不用担心错过BUG~


activity_main.xml

<FrameLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

<ListView
    android:id="@+id/lvTitleFade"
    android:layout_width="match_parent"
    android:layout_height="wrap_content" />

    <RelativeLayout
        android:id="@+id/rlTitle"
        android:layout_width="match_parent"
        android:layout_height="56dp"
        android:background="@color/colorPrimary" >

        <ImageButton
            android:src="@drawable/back"
            android:background="@null"
            android:layout_centerVertical="true"
            android:layout_marginLeft="5dp"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content" />

        <TextView
            android:layout_width="match_parent"
            android:layout_height="56dp"
            android:gravity="center"
            android:text="FJTitleFade"
            android:textColor="#ffffff"
            android:textSize="20sp" />
    </RelativeLayout>
</FrameLayout>

说明:主布局很简单,布局选用FrameLayout,让标题栏在ListView的上方~


MainActivity.java核心代码

private SparseArray recordSp = new SparseArray(0);
private int mCurrentfirstVisibleItem = 0;

//设置标题背景透明
rlTitle.getBackground().setAlpha(0);
//滑动监听,注意implements OnScrollListener
lvTitleFade.setOnScrollListener(this);

@Override
public void onScrollStateChanged(AbsListView view, int scrollState) {
    
}
//滑动事件处理
@Override
public void onScroll(AbsListView view, int firstVisibleItem,
        int visibleItemCount, int totalItemCount) {
    //firstVisibleItem--处于顶部的Item标记
    //visibleItemCount--当前可见item数
    //totalItemCount----总item数
    mCurrentfirstVisibleItem = firstVisibleItem;
    View firstView = view.getChildAt(0);
    if (null != firstView) {
        ItemRecod itemRecord = (ItemRecod) recordSp.get(firstVisibleItem);
        if (null == itemRecord) {
            itemRecord = new ItemRecod();
        }
        itemRecord.height = firstView.getHeight();//获取最顶部Item的高度
        itemRecord.top = firstView.getTop();//获取距离顶部的距离
        recordSp.append(firstVisibleItem, itemRecord);//设置值
    }
    Log.d("dmdrs", "滑动距离:" + getScrollY());
    int ScrollY = getScrollY();
    if (ScrollY >= 0 && ScrollY <= 255) {
        //设置标题栏透明度0~255
        rlTitle.getBackground().setAlpha(ScrollY);
    } else if (ScrollY > 255) {
        //滑动距离大于255就设置为不透明
        rlTitle.getBackground().setAlpha(255);
    }
}

private int getScrollY() {
    int height = 0;
    for (int i = 0; i < mCurrentfirstVisibleItem; i++) {
        ItemRecod itemRecod = (ItemRecod) recordSp.get(i);
        Log.d("dmdrs", "xxx1:" + itemRecod);
        //06-07 21:00:21.601: D/dmdrs(23096): xxx1:
        //          com.dmdrs.titlefade.MainActivity$ItemRecod@529122fc
        //06-07 21:00:21.601: D/dmdrs(23096): xxx2:300
        //06-07 21:00:21.601: D/dmdrs(23096): xxx1:null
        //快速滑动会为空,判断一下,发现的bug
        if(itemRecod != null){
            height += itemRecod.height;
        }
        Log.d("dmdrs", "xxx2:" + height);
    }
    ItemRecod itemRecod = (ItemRecod) recordSp.get(mCurrentfirstVisibleItem);
    if (null == itemRecod) {
        itemRecod = new ItemRecod();
    }
    return height - itemRecod.top;
}

class ItemRecod {
    int height = 0;
    int top = 0;
}

说明:
①设置标题背景透明为完全透明
②设置监听ListView的滑动事件
③在onScroll里来获取当前Item的参数,设置到SparseArray中,然后调用getScrollY方法来计算滑动距离并返回参数,然后根据距离来设置rlTitle的透明度。


总结:就上面这点了,代码不很多,但实现了想要的效果。在撸代码的过程中也发现了个小BUG,在快速滑动ListView的时候itemRecod会出现为null的情况。这个BUG出现是因为滑动过快,导致值还没有设置进去,就在取值,所以出现了null的情况,导致空指针错误。目前解决办法是加了个if判断null解决了这个小BUG,如果有好的解决方式请留言交流,谢谢~~


未经本人允许禁止转载,违者必究


个人博客:WWW.FJ917.COM
简书:www.jianshu.com/u/3d2770e6e489

欢迎加入QQ交流群657206000点我加入
QQ交流群:657206000

相关文章

网友评论

  • 3e2f50d36f66:这个如果用recycleview去实现这种效果的话,怎么做呢
    淡漠de人生:recycleview的话可以参考我的这篇http://www.jianshu.com/p/079fc7396d7f把里面的滑动显示title运用起来就可以了
  • fendo:赞一个
    淡漠de人生:@fendo 谢谢!😄
  • 爱喝营养快线:可以看一下nestscrollview源码,然后有个方法可以准确获取距离,自己仿照写个接口就可以了,楼主写麻烦了
    淡漠de人生:@爱喝营养快线 实现效果千千万~换中方式而已~为了多学习,多学种套路(写法)而已~呵呵~~~
  • 柴柴777:最近项目忙,我百度试了好多种,弄了好久才成功😂以后还得常刷简书啊
    淡漠de人生: @学编程的女生么 嗯嗯,所以我也整点干货,呵呵😊
    柴柴777:@淡漠de人生 嗯嗯😁干货很多
    淡漠de人生:@学编程的女生么 恩,简书是个好地方,有很多干货,可以经常来看看,说不准什么时候就看到想要的了。
  • Luh的星河:写的不错
  • Luh的星河:用toobar title不是可以实现么
    淡漠de人生: @过期的旅行 恩恩,谢谢关注,栗子系列会不断更新,不会吃亏的,😃
    Luh的星河: @淡漠de人生 第一次用简书果断关注楼主😊
    淡漠de人生: @过期的旅行 谢谢,写法多种多样,看你怎么写咯!
  • null_null_:我记得有个框架可以直接实现
    MeloDev:@HELLO丶GUY 什么框架,可以分享一下吗
    淡漠de人生: @HELLO丶GUY 实现方式很多,你也可以写出来给大家参考哦!
  • 浅笑_JIE:好吧 又是你 嘿嘿 用 CoordinatorLayout 和 CollapsingToolbarLayout 和 RecyclerView
    浅笑_JIE:@淡漠de人生 局限性? RecyclerView 支持 了 GridView、ListView、瀑布流 列表了 要是不喜欢 用NestedScrollView代替 要加下拉刷新用 SwipeRefreshLayout
    淡漠de人生:@浅笑_JIE 而且列表只能用RecyclerView,有局限性!
    淡漠de人生:@浅笑_JIE 我试过,我用你说的写了另外的一个效果,有时间会出文章,谢谢,不过这篇我没去用你说的这种方式去实现!
  • 菲利柯斯:如果下面是碎片,碎片里面放着listview,该怎么弄这个效果
    淡漠de人生:@菲利柯斯 下面指的是item部分么,换成fragment?头部是固定的还是可以滑动的?如果是头部可以滑动的话用ScrollView嵌套就可以了,ScrollView可以获取垂直滚动距离。如果不是这样请描述的清楚一点~

本文标题:栗子——ListView+Head滑动显示标题

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