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

recycleview的item的自动居中以及自定义左对齐集代码分析

今天闲的无事写了这个就当储备了 O(∩_∩)O哈哈~这里先说简单的就是item的自动居中这个很简但就是就是加两句话SnapHelpersnapHelpernewLinearSnap

今天闲的无事写了这个 就当储备了  O(∩_∩)O哈哈~

这里先说简单的 就是item的自动居中

这个很简但就是 就是加两句话

// SnapHelper snapHelper = new LinearSnapHelper(); // snapHelper.attachToRecyclerView(my_recycle);//这里放置的是的recycleview

这两句话就搞定了   这里源码的具体分析我这里就不做什么具体描述了简书上就有  
我这里要说的是下一个需求自动左对齐  这里用原生的肯定不行就需要我们写个类继承
LinearSnapHelper 然后对其进行处理

这里我先提下这次比较重要的两个方法
calculateDistanceToFinalSnap和findSnapView我们主要对这里进行了处理

首先
findSnapView

@Override
public int[] calculateDistanceToFinalSnap(@NonNull RecyclerView.LayoutManager layoutManager,
@NonNull View targetView) {
//这里是个数组0返回的是横向的距离 1返回的是纵向的距离 默认的话你将每次都返回到第一个view int[] out = new int[2];
//这里判断你设置的recycleview的布局管理器方向是横向还是纵向 if (layoutManager.canScrollHorizontally()) {
//然后将距离开始的距离存放在内 out[0] = distanceToStart(targetView, getHorizontalHelper(layoutManager));
} else {
out[0] = 0;
}
if (layoutManager.canScrollVertically()) {
out[1] = distanceToStart(targetView, getVerticalHelper(layoutManager));
} else {
out[1] = 0;
}
return out;
}

//这里是获取开始显示的view方法

@Override

public View findSnapView(RecyclerView.LayoutManager layoutManager) {
if (layoutManager instanceof LinearLayoutManager) {
if (layoutManager.canScrollHorizontally()) {
return getStartView(layoutManager, getHorizontalHelper(layoutManager));
} else {
return getStartView(layoutManager, getVerticalHelper(layoutManager));
}
}
return super.findSnapView(layoutManager);
}

这里我们是根据他的滚动方向要进行一些处理 根据我们getstartview进行一些逻辑判断 来处理显示最左边的view是哪个
然后是我们的
calculateDistanceToFinalSnap计算到targetView要移动的距离 

@Override
public int[] calculateDistanceToFinalSnap(@NonNull RecyclerView.LayoutManager layoutManager,
@NonNull View targetView) {
//这里是个数组0返回的是横向的距离 1返回的是纵向的距离 默认的话你将每次都返回到第一个view int[] out = new int[2];
//这里判断你设置的recycleview的布局管理器方向是横向还是纵向 if (layoutManager.canScrollHorizontally()) {
//然后将距离开始的距离存放在内 out[0] = distanceToStart(targetView, getHorizontalHelper(layoutManager));
} else {
out[0] = 0;
}
if (layoutManager.canScrollVertically()) {
out[1] = distanceToStart(targetView, getVerticalHelper(layoutManager));
} else {
out[1] = 0;
}
return out;
}

这里我们是来计算我们的recycleview需要移动的距离 来保持左对齐
好了说了这些还是贴出源码吧

package com.example.admin.juzhongietm;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.LinearSnapHelper;
import android.support.v7.widget.OrientationHelper;
import android.support.v7.widget.RecyclerView;
import android.util.Log;
import android.view.View;
/** * Created by admin on 2017/1/26. */ public class StartSnapHelper extends LinearSnapHelper {
private OrientationHelper mVerticalHelper, mHorizontalHelper;
public StartSnapHelper() {
}
//这个是放入的被设置的recycleview'的 @Override
public void attachToRecyclerView(@Nullable RecyclerView recyclerView)
throws IllegalStateException {
super.attachToRecyclerView(recyclerView);
}
/** * 计算到targetView要移动的距离 这个也是关键 这个方法是在我们判断处理之后(也就是findSnapView方法) 所以distanceToStart的方法中直接计算此时recycleview的离最屏幕开始处的距离减去内边距就是需要移动的距离 */ @Override
public int[] calculateDistanceToFinalSnap(@NonNull RecyclerView.LayoutManager layoutManager,
@NonNull View targetView) {
//这里是个数组0返回的是横向的距离 1返回的是纵向的距离 默认的话你将每次都返回到第一个view int[] out = new int[2];
//这里判断你设置的recycleview的布局管理器方向是横向还是纵向 if (layoutManager.canScrollHorizontally()) {
//然后将距离开始的距离存放在内 out[0] = distanceToStart(targetView, getHorizontalHelper(layoutManager));
} else {
out[0] = 0;
}
if (layoutManager.canScrollVertically()) {
out[1] = distanceToStart(targetView, getVerticalHelper(layoutManager));
} else {
out[1] = 0;
}
return out;
}
//这里是获取开始显示的view方法 @Override
public View findSnapView(RecyclerView.LayoutManager layoutManager) {
if (layoutManager instanceof LinearLayoutManager) {
if (layoutManager.canScrollHorizontally()) {
return getStartView(layoutManager, getHorizontalHelper(layoutManager));
} else {
return getStartView(layoutManager, getVerticalHelper(layoutManager));
}
}
return super.findSnapView(layoutManager);
}
//这里是 计算距离开始的距离 由于calculateDistanceToFinalSnap的方法是在findSnapView之后所以这里计算的是recycleview的离最屏幕开始处的距离减去内边距就是需要移动的距离 private int distanceToStart(View targetView, OrientationHelper helper) {
Log.i("StartSnapHelper", "disanceToStart: "+helper.getDecoratedStart(targetView)+"========="+helper.getStartAfterPadding());
return helper.getDecoratedStart(targetView) - helper.getStartAfterPadding();
}
//这里是我们自定义的右对齐的显示规则 也就是这里就是我实现自动最左对齐的关键 当然如果有其他的奇葩右对齐 -- (什么奇葩客户没见过)实现原理一样的 private View getStartView(RecyclerView.LayoutManager layoutManager,
OrientationHelper helper) {
//这里返回null表示不做任何操作保持当前的位置 if (layoutManager instanceof LinearLayoutManager) {
Log.i("StartSnapHelper", "getStartView: "+1);
int firstChild = ((LinearLayoutManager) layoutManager).findFirstVisibleItemPosition();
boolean isLastItem = ((LinearLayoutManager) layoutManager)
.findLastCompletelyVisibleItemPosition()
== layoutManager.getItemCount() - 1;
//当发现显示了最后一个item的时候不执行任何操作 if (firstChild == RecyclerView.NO_POSITION || isLastItem) {
Log.i("StartSnapHelper", "getStartView: "+2);
return null;
}
View child = layoutManager.findViewByPosition(firstChild);
Log.i("StartSnapHelper11", "getStartView: "+helper.getDecoratedEnd(child)+ "------------"+helper.getDecoratedMeasurement(child));
//当发现当前第一个view最后到结束位置到屏幕距离大于item的一半的时候自动复位 或者用getDecoratedStart()判断条件反过来而已 if (helper.getDecoratedEnd(child) >= helper.getDecoratedMeasurement(child) / 2
&& helper.getDecoratedEnd(child) > 0) {
Log.i("StartSnapHelper", "getStartView: "+3);
return child;
} else {
//如果发现最后停留的位置小于item自身的一半 则再次判断是不是最后一个view 如果是将不执行任何操作 if (((LinearLayoutManager) layoutManager).findLastCompletelyVisibleItemPosition()
== layoutManager.getItemCount() - 1) {
Log.i("StartSnapHelper", "getStartView: "+4);
return null;
} else {
//如果不是返回下一个view Log.i("StartSnapHelper", "getStartView: "+5);
return layoutManager.findViewByPosition(firstChild + 1);
}
}
}
return super.findSnapView(layoutManager);
}
private OrientationHelper getVerticalHelper(RecyclerView.LayoutManager layoutManager) {
if (mVerticalHelper == null) {
mVerticalHelper = OrientationHelper.createVerticalHelper(layoutManager);
}
return mVerticalHelper;
}
private OrientationHelper getHorizontalHelper(RecyclerView.LayoutManager layoutManager) {
if (mHorizontalHelper == null) {
mHorizontalHelper = OrientationHelper.createHorizontalHelper(layoutManager);
}
return mHorizontalHelper;
}
}

上面我应该是给了足够多的注释了 这里要注意的是我们这么处理是因为 由于calculateDistanceToFinalSnap的方法是在findSnapView之后

所以我们思路基本上就是根据当前item的第一的移动位置和本身的长度进行对比 进行一系列的判断 来确定的最靠左的是哪个item 然后我们的计算移动

距离的calculateDistanceToFinalSnap方法就可以根据我们findSnapView返回的view计算出实现左对其的距离了

O(∩_∩)O哈哈~今天就说这么多 希望能对你有所帮助


推荐阅读
  • 如何自行分析定位SAP BSP错误
    The“BSPtag”Imentionedintheblogtitlemeansforexamplethetagchtmlb:configCelleratorbelowwhichi ... [详细]
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • 自动轮播,反转播放的ViewPagerAdapter的使用方法和效果展示
    本文介绍了如何使用自动轮播、反转播放的ViewPagerAdapter,并展示了其效果。该ViewPagerAdapter支持无限循环、触摸暂停、切换缩放等功能。同时提供了使用GIF.gif的示例和github地址。通过LoopFragmentPagerAdapter类的getActualCount、getActualItem和getActualPagerTitle方法可以实现自定义的循环效果和标题展示。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • 关键词:Golang, Cookie, 跟踪位置, net/http/cookiejar, package main, golang.org/x/net/publicsuffix, io/ioutil, log, net/http, net/http/cookiejar ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • Android开发实现的计时器功能示例
    本文分享了Android开发实现的计时器功能示例,包括效果图、布局和按钮的使用。通过使用Chronometer控件,可以实现计时器功能。该示例适用于Android平台,供开发者参考。 ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • JavaSE笔试题-接口、抽象类、多态等问题解答
    本文解答了JavaSE笔试题中关于接口、抽象类、多态等问题。包括Math类的取整数方法、接口是否可继承、抽象类是否可实现接口、抽象类是否可继承具体类、抽象类中是否可以有静态main方法等问题。同时介绍了面向对象的特征,以及Java中实现多态的机制。 ... [详细]
  • 闭包一直是Java社区中争论不断的话题,很多语言都支持闭包这个语言特性,闭包定义了一个依赖于外部环境的自由变量的函数,这个函数能够访问外部环境的变量。本文以JavaScript的一个闭包为例,介绍了闭包的定义和特性。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • Go Cobra命令行工具入门教程
    本文介绍了Go语言实现的命令行工具Cobra的基本概念、安装方法和入门实践。Cobra被广泛应用于各种项目中,如Kubernetes、Hugo和Github CLI等。通过使用Cobra,我们可以快速创建命令行工具,适用于写测试脚本和各种服务的Admin CLI。文章还通过一个简单的demo演示了Cobra的使用方法。 ... [详细]
  • 本文讨论了在openwrt-17.01版本中,mt7628设备上初始化启动时eth0的mac地址总是随机生成的问题。每次随机生成的eth0的mac地址都会写到/sys/class/net/eth0/address目录下,而openwrt-17.01原版的SDK会根据随机生成的eth0的mac地址再生成eth0.1、eth0.2等,生成后的mac地址会保存在/etc/config/network下。 ... [详细]
  • 本文介绍了如何使用Express App提供静态文件,同时提到了一些不需要使用的文件,如package.json和/.ssh/known_hosts,并解释了为什么app.get('*')无法捕获所有请求以及为什么app.use(express.static(__dirname))可能会提供不需要的文件。 ... [详细]
author-avatar
晨曦--雨荷
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有