热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

RecyclerView首个Item滑到顶部渐变

RecyclerView首个Item顶部渐变效果抖音“新鲜”Tab页上有个效果比较有意思,当Item滑到顶部的时候,item上的文字信息会有一个渐变的隐藏效果,如下图这个效果实现起

RecyclerView 首个Item顶部渐变效果

抖音“新鲜”Tab页上有个效果比较有意思,当Item滑到顶部的时候,item上的文字信息会有一个渐变的隐藏效果,如下图

《RecyclerView首个Item滑到顶部渐变》

这个效果实现起来不是特别难,但是有一些细节地方需要我们考虑一下,下面我就把我实现的方式分享一下,先来一张效果图

《RecyclerView首个Item滑到顶部渐变》

实现方式

  • 使用RecyclerView.OnScrollListener监听RecyclerView的滚动
  • 在onScrolled(RecyclerView recyclerView, int dx, int dy)方法中获取第一个Item
  • 获取第一个Item的滑动出屏幕的百分比
  • 根据计算获取当前文字区域的Alpha值

代码如下:

RecyclerView.OnScrollListener mScrollListener = new RecyclerView.OnScrollListener() {
@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
super.onScrollStateChanged(recyclerView, newState);
}
@Override
public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
super.onScrolled(recyclerView, dx, dy);
RecyclerView.LayoutManager manager = recyclerView.getLayoutManager();
//这里我使用的是GridLayoutManager,并且RecyclerView只有两列
if(manager instanceof GridLayoutManager){
GridLayoutManager gridLayoutManager = (GridLayoutManager) manager;
int firstVisiblePos = gridLayoutManager.findFirstVisibleItemPosition();
setViewAplha(gridLayoutManager.findViewByPosition(firstVisiblePos));
setViewAplha(gridLayoutManager.findViewByPosition(firstVisiblePos+1));
int firstCompletelyVisible = gridLayoutManager.findFirstCompletelyVisibleItemPosition();
ItemView view1 = (ItemView) gridLayoutManager.findViewByPosition(firstCompletelyVisible);
ItemView view2 = (ItemView) gridLayoutManager.findViewByPosition(firstCompletelyVisible + 1);
if(view1 != null){
view1.setAlpha(1);
}
if(view2 != null){
view2.setAlpha(1);
}
}
}
};
float beginPercent = 0.2f;
float endValue = 2;
private void setViewAplha(View view){
if (view == null || !(view instanceof ItemView))
return;
float p = UIUtils.px2dip(Math.abs((int) view.getY())) * 1.0f / UIUtils.px2dip(view.getHeight()) * 1.0f;
float curPercent = Float.compare(p - beginPercent, 0.0f) <0 ? 0.0f : p - beginPercent;
curPercent = Float.compare(1, curPercent * endValue) <0 ? 1 : curPercent * endValue;
view.setAlpha(1 - curPercent);
}

这里的ItemView是自定义的每一个内容区域的View,布局比较简单 只有一个TextView 代码如下

private static class ItemView extends LinearLayout{
TextView textView;
Context mContext;
public ItemView(Context context) {
super(context);
mCOntext= context;
init();
}
private void init(){
View root = LayoutInflater.from(mContext).inflate(R.layout.layout_test_item,this);
textView = (TextView) root.findViewById(R.id.item_text);
}
public void setText(String text){
textView.setText(text);
}
public void setAlpha(float alpha){
textView.setAlpha(alpha);
}
}

需要考虑的细节问题

  • Adapter Holder的复用
    我们都知道RecyclerView.Adapter会对Holder进行复用来节约内存的开销,如果假设一屏有十个item,当我们滑动到第十一个Item时会复用第一个Item的Holder,所以在onBindViewHolder方法中我们需要把文字区域的Alpha值设置为不透明,adaper的代码如下:

RecyclerView.Adapter adapter = new RecyclerView.Adapter() {
@Override
public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
ItemView itemView = new ItemView(parent.getContext());
ViewHolder holder = new ViewHolder(itemView);
return holder;
}
@Override
public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
ViewHolder holder1 = (ViewHolder) holder;
holder1.itemView.setText("我是Item"+position);
//需要把item设置不透明,否则可能会因为复用导致item刚显示就是看不见的
holder1.itemView.setAlpha(1);
}
@Override
public int getItemCount() {
return 40;
}
class ViewHolder extends RecyclerView.ViewHolder{
ItemView itemView;
public ViewHolder(View itemView) {
super(itemView);
this.itemView = (ItemView) itemView;
}
}
};

  • 快速滑动的问题
    RecyclerView 滑动时会有如下三种状态

/**
* The RecyclerView is not currently scrolling.
* @see #getScrollState()
*/
public static final int SCROLL_STATE_IDLE = 0;
/**
* The RecyclerView is currently being dragged by outside input such as user touch input.
* @see #getScrollState()
*/
public static final int SCROLL_STATE_DRAGGING = 1;
/**
* The RecyclerView is currently animating to a final position while not under
* outside control.
* @see #getScrollState()
*/
public static final int SCROLL_STATE_SETTLING = 2;

当我们快速滑动时可能会由于滑动速度过快,导致在下次回调时可能已经是下一个item元素,例如,在当前onScrolled方法回调时,findFirstVisibleItemPosition可能是item4,然后我们计算出文字区域的Alpha是0.6,因为滑动过快,下次回调onScrolled方法时,findFirstVisibleItemPosition可能就变成了item6,Alpha值是0.8,因为我们缺少了item4的完整状态,就导致了当快速滑动停止时,可能item6是正确的透明度,但是我们希望item4是完全不透明的,但是实际却是0.6
为了解决这个问题,我尝试了几种方法,像监听滑动状态啊、判断onScrolled中的dy参数啊,后来我发现我有点给想的麻烦了,其实不管滑动速度多快,每个item在滑动到第一个完全可展示的位置时,是一定会被我们知道的,比如虽然我们不能拿到item4从完全透明到完全不透明的所有状态,但是当item6是第一个item时,item4一定是第一可以完全展示的item(我的RecyclerView有两列),所以我们就可以通过这个点来下手,通过LayoutManager的findFirstCompletelyVisibleItemPosition来找到第一个完全可见的Item位置,具体体现就是刚才RecyclerView.OnScrollListener 里面的代码

int firstCompletelyVisible = gridLayoutManager.findFirstCompletelyVisibleItemPosition();
ItemView view1 = (ItemView) gridLayoutManager.findViewByPosition(firstCompletelyVisible);
ItemView view2 = (ItemView) gridLayoutManager.findViewByPosition(firstCompletelyVisible + 1);
if(view1 != null){
view1.setAlpha(1);
}
if(view2 != null){
view2.setAlpha(1);
}

  • Alpha值的计算
    这里还有一个要考虑的问题,就是Alpha的取值问题,仔细看抖音中的效果,并不是item刚开始滑动就开始改变透明度,而是大概Item滑出屏幕1/5左右开始,这里就需要我们做一点点小小的计算,当然我的计算方法可能有点low,计算方法就是刚才的setViewAplha里面所做的,在这在粘贴一下

float beginPercent = 0.2f;
float endValue = 2;
private void setViewAplha(View view){
if (view == null || !(view instanceof ItemView))
return;
float p = UIUtils.px2dip(Math.abs((int) view.getY())) * 1.0f / UIUtils.px2dip(view.getHeight()) * 1.0f;
float curPercent = Float.compare(p - beginPercent, 0.0f) <0 ? 0.0f : p - beginPercent;
curPercent = Float.compare(1, curPercent * endValue) <0 ? 1 : curPercent * endValue;
view.setAlpha(1 - curPercent);
}

这个beginPercent变量表示我想要在Item滑出屏幕多少时,才开始改变Alpha值,这里我定义成了0.2也就是1/5,然后在第一次计算curPercent时

float p = UIUtils.px2dip(Math.abs((int) view.getY())) * 1.0f / UIUtils.px2dip(view.getHeight()) * 1.0f;
float curPercent = Float.compare(p - beginPercent, 0.0f) <0 ? 0.0f : p - beginPercent;

可见如果当p<0.2时,我的curPercent会等于0,后续在做什么计算都不会改变文字区域的Alpha值
现在已经可以设置从item滑动到多少才开始出现渐变的效果了,那么下一步我们就应该定义当item滑动到什么位置时文字区域完全透明也就是Alpha值等于0,这时候endValue的作用就来了,我们可以通过给curPercent乘一个值来让它达到指定位置时可以变成1,然后1-curPercent=0

curPercent = Float.compare(1, curPercent * endValue) <0 ? 1 : curPercent * endValue;

这里我的endValue=2,可以在item滑出屏幕70%时,文字区域完全透明,我的计算方法如下表格

curPercent与0.2的差值(滑动从1/5开始计算)
0.0&#8211;
0.1&#8211;
0.2&#8211;
0.30.1
0.40.2
0.50.3
0.60.4
0.70.5
0.80.6
0.90.7

所以如果我想让Item滑动到70%时完全透明,我只要设置endValue是2既可,因为此时curPercent是0.7-0.2=0.5 然后乘2刚好等于1,之前一直都会小于1,这样的话,我们可以随意设定滑动渐变从哪开始到哪结束,比如我现在希望item滑出屏幕30%开始 同样在item滑出屏幕70%完全透明那么根据下面这个表格

curPercent与0.3的差值(滑动从30%开始计算)
0.0&#8211;
0.1&#8211;
0.2&#8211;
0.3&#8211;
0.40.1
0.50.2
0.60.3
0.70.4
0.80.5
0.90.6

我们可以计算出 endValue=3( (0.7-0.3)* endValue =1.2 )即可保证在滑动到70%时 完全的透明

ok 到这我们就可以完美表现出这种渐变的效果了,这是我第一次写简书,写的还有点小激动呢。欢迎所有读过的人 批评指正~


推荐阅读
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • Spring特性实现接口多类的动态调用详解
    本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • 自动轮播,反转播放的ViewPagerAdapter的使用方法和效果展示
    本文介绍了如何使用自动轮播、反转播放的ViewPagerAdapter,并展示了其效果。该ViewPagerAdapter支持无限循环、触摸暂停、切换缩放等功能。同时提供了使用GIF.gif的示例和github地址。通过LoopFragmentPagerAdapter类的getActualCount、getActualItem和getActualPagerTitle方法可以实现自定义的循环效果和标题展示。 ... [详细]
  • 如何自行分析定位SAP BSP错误
    The“BSPtag”Imentionedintheblogtitlemeansforexamplethetagchtmlb:configCelleratorbelowwhichi ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • JavaSE笔试题-接口、抽象类、多态等问题解答
    本文解答了JavaSE笔试题中关于接口、抽象类、多态等问题。包括Math类的取整数方法、接口是否可继承、抽象类是否可实现接口、抽象类是否可继承具体类、抽象类中是否可以有静态main方法等问题。同时介绍了面向对象的特征,以及Java中实现多态的机制。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • 前景:当UI一个查询条件为多项选择,或录入多个条件的时候,比如查询所有名称里面包含以下动态条件,需要模糊查询里面每一项时比如是这样一个数组条件:newstring[]{兴业银行, ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • 本文介绍了iOS数据库Sqlite的SQL语句分类和常见约束关键字。SQL语句分为DDL、DML和DQL三种类型,其中DDL语句用于定义、删除和修改数据表,关键字包括create、drop和alter。常见约束关键字包括if not exists、if exists、primary key、autoincrement、not null和default。此外,还介绍了常见的数据库数据类型,包括integer、text和real。 ... [详细]
  • 先看官方文档TheJavaTutorialshavebeenwrittenforJDK8.Examplesandpracticesdescribedinthispagedontta ... [详细]
author-avatar
Amyjionydp
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有