这个版本号的博客写起来颇费口舌。有些代码自己语言组织能力有限,感觉描写叙述起来非常费劲,前前后后改了五六遍稿子还是不尽人意 ,只是我还是坚持写出来自己当初的思路,假设看得不明确的地方我在文章最后仍然会上传源码,能够直接执行看效果,看过执行的效果后对文中有些别扭的语言预计会能直观的了解。在版本号3.0的尽管实现了随着手指的左右移动listView中的item也随着滚动。可是会出现例如以下的情况:

当左边已经是第一个的时候,会出现例如以下的情况(仍然能够向右移动):

当右边是adapter最后一个item的时候,会出现例如以下的情况(仍然能够向左移动):

正常来说,当左边第一个和右边最后一个应该不能滚动才是,本4.0版本号将解决问题。在解决问题之前先说说相关的知识点:

知识点1):手指在屏幕上移动的时候,会发生 其它相关方法...>onScroll-->onScroll-->onScroll ..-->其它相关方法.这种调用

知识点2):3.0版本号在处理手指滚动的时候用到了onScroll(MotionEvent e1, MotionEvent e2,float distanceX, float distanceY)这种方法,这种方法中有个distanceX參数。这个參数我们知道,当手指想做移动的时候distanceX>0;而手指向右移动的时候distanceX<0;这是由于distanceX的值是由第二个參数决定的:distanceX = 上次掉用onScroll方法的e2.getX()-当前onScroll方法的e2.getX()或者简写成
lastEvent2.getX() - currentEvent2.getX() = distanceX;

在版本号3.0的时候我们直接用distanceX这个变量(确切的说这个变量值的相反数来模拟左右移动滚动的距离),可是这次为了解决上面的问题我们将不再直接用distanceX,而是新定义了两个变量:

变量:preTotalDistanceX :相对于当前onScroll方法调用之前。之前全部onScroll调用所产生的距离之和。比方当前是第n次调用了onScroll。那么preTotalDistanceX=前n-1次调用onScroll中distanceX的累加和;

变量:totalDistanceX:当前调用onScroll方法方法调用中全部distanceX的累加和;比方当前是第n次调用了onScroll方法,那么totalDistanceX就是n个onScrll方法调用中參数里distanceX的累加和。

那么滚动的距离distanceX = totalDistanceX - preTotalDistanceX

用代码表示总距离例如以下:

            public boolean onScroll(MotionEvent e1, MotionEvent e2,float distanceX, float distanceY) {
totalDistanceX += distanceX;
requestLayout();
return true; };

该版本号onLayout的方法就改动例如以下:

		int distanceX = totalDistanceX - preTotalDistanceX;
removeAnvisiableViews(-distanceX);
addRightChildViews(-distanceX);
addLeftChildViews(-distanceX);
layoutChildViews(-distanceX);
preTotalDistanceX = totalDistanceX;

要解决文章开头提出的问题,关键是让distanceX = 0;即pretotalDistanceX=totalDistanceX;所以解决这个问题的关键就是怎样让这两个变量相等。从知识点2能够知道,如果用户的手指先右后左:此时最左边的item是adapter的第一个item,所以手指向右移动的时候不能让listView向右滚动,可是实际上随着手指的向右移动totalDistanceX变量是<0,并沿着平面直角坐标系远离0点而负递增,所以在这里能够初步推断:当totalDistanceX<=0的时候,让totalDistanceX
= 0;这样preoTotalDistanceX=totalDistanceX=0,这样就会让distanceX=0,从而不会让我们的ListView不会向右移动;如果用户的手指先左后右的移动方式:当手指向左移动的时候totalDistanceX>0且不断增大。是递增的。而当手指右移动的时候totalDistanceX是在原来的基础上不断降低的过程,当较少到0的时候继续向右移动的话totalDistanceX变为负数,继续向右移动的话就类似与手指先右后左的现象了;这个是关键的地方。整个左右移动的过程发生诡异的变化类似于物理中的位移。当totalDistanceX=0的时候我们能够断定不能在向右滚动。

所以在onLayout代码中我做了例如以下推断:

@Override
protected void onLayout(boolean changed, int left, int top, int right,
int bottom) { if (listAdapter == null) {
return;
}
/**确保党左边是第一个的时候不再滚动*/
if(totalDistanceX<=0) {
totalDistanceX = 0;
} int distanceX = totalDistanceX - preTotalDistanceX;
removeAnvisiableViews(-distanceX);
addRightChildViews(-distanceX);
addLeftChildViews(-distanceX);
layoutChildViews(-distanceX);
preTotalDistanceX = totalDistanceX; }

执行一把发现上面的设想是成立的。到此为止,博客开头的第一个问题得到了解决。

以下将解决第二个问题:当adapter里面最后一个item加入到viewGroup里面而且全然显示的时候,禁止listView继续向左滚动。

先看最有一个item没有显示全然的情况,例如以下图:

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvY2h1bnFpdXdlaQ==/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="">

解决第二个问题的关键仍然是怎样让distanceX= totalDistanceX-pretotalDistanceX=0。也就是怎样让totalDistanceX=preototalDistanceX;依据图示我们非常easy得出例如以下结论:要让最后一个item显示完。在之前滚动距离总和的基础上再让listView滚动X大小的距离能够了,用上面的等式表示为totalDistanceX=X+pretotalDistanceX,所以当totalDistanceX>X+pretotalDistanceX的时候我们让 totalDistanceX
= X+preototalDistanceX;在程序用我是用scrollXMax来表示X+preototalDistanceX的值(此变量名起的有点垃圾);在代码用为:

//scrollXMax = X + pretotalDistanceX
if(totalDistanceX>scrollXMax) {
totalDistanceX = scrollXMax;
} ......
preototalDistanceX = totalDistanceX;

这样当最后一个item滚动了X距离的情况下调用layout绘制到listView之后,手指在移动的情况下if(totalDistanceX>scrollXMax仍然成立)。这样计算的distanceX=0.可是话有说回来了,X+preDistanceX这段代码加入到哪儿呢?结合之前的几篇博客,不难分析出应该在addRightChildViews方法里面加入,代码例如以下:

private void addRightChildViews(int distanceX) {
// 2.让屏幕尽可能的显示Item。注意刚開始的时候是没有
View rightChildView = getChildAt(getChildCount() - 1); // 获取此childView右边框距离parentView左边框的距离
int rightEdge = rightChildView != null ? rightChildView.getRight() : 0;
while (rightEdge + distanceX < getWidth()
&& rightIndex < listAdapter.getCount()) {
View child = listAdapter.getView(rightIndex, null, null);
child = measureChild(child);
addViewInLayout(child, -1, child.getLayoutParams(), true);
rightEdge += child.getMeasuredWidth(); //推断最后一个item
if (rightIndex == listAdapter.getCount() - 1) {
scrollXMax = rightEdge +preTotalDistanceX- getWidth();
} rightIndex++;
}
}

到此位置。上面的的两个问题都得到圆满解决。自己在整这个版本号的时候走了不少弯路,捣鼓了半天,到最后完成还是有种小小的成就感;何为编程。说白了就是在遵循特性编程语言规则的情况下,用该编程语言描写叙述或者表达程序猿思路的过程。

在4.0版本号的实现中。自己在纸上又是写又是画的,思路清晰了然后就非常自然而然的用编程语言把这个思路表达出来。这就是编程吧!啰嗦完成。当然该版本号还没有完好完成,比方说不能点击就是这个,将在下一个版本号解决剩余的问题。时间同意的话直接写完。此处为项目源代码,欢迎批评指正

简单的横向ListView实现(version 4.0)的更多相关文章

  1. 简单的横向ListView实现(version 3.0)

    版本号2仅仅是简单的实现了当手指按下的时候listView的Item向左移动一定的距离,并没有随着手指的左右移动而左右滚动.在这个版本号3.0中将会实现随着手指的移动而滚动的目标:当手指向左移动的时候 ...

  2. gitlab 源码安装=》rpm安装横向迁移(version 9.0)

    准备: 下载版本地址: https://packages.gitlab.com/gitlab/gitlab-ce 迁移环境: 源码安装的gitlab9.0.13 目标迁移至9.0.13 RPM安装的环 ...

  3. Java简单双向链表实现 @version 1.0

    package com.list; /** * 数据结构和算法Java表示 双向链表 * * @version 1.0 * @author 小明 * */ public class MyDoublel ...

  4. Java单链表简单实现* @version 1.0

    package com.list; /** * 数据结构与算法Java表示 * @version 1.0 * @author 小明 * */ public class MyLinkedList { p ...

  5. ExpandableListView简单应用及listview模拟ExpandableListView

    首先我们还是来看一些案例,还是拿搜狐新闻客户端,因为我天天上下班没事爱看这个东东,上班又没时间看新闻,上下班路途之余浏览下新闻打发时间嘛.           看这个效果挺棒吧,其实实现起来也不难,我 ...

  6. 张高兴的 UWP 开发笔记:横向 ListView

    ListView 默认的排列方向是纵向 ( Orientation="Vertical" ) ,但如果我们需要横向显示的 ListView 怎么办? Blend for Visua ...

  7. eclipse 和 android studio 编译时报 Unsupported major.minor version 52.0错解决办法

    version 52.0 是java8的环境.当gradle tools 升级到2.2.1时候,可能编译时候会报该错误. 很多网友说更改java version,但是很多时候无效.下面是我遇到时候的解 ...

  8. (class file version 53.0), Java Runtime versions up to 52.0错误的解决方法

    遇到这个错误是在Apache Tomcat上部署应用程序的时候遇到的,具体的错误描述是: java.lang.UnsupportedClassVersionError: HelloWorld has ...

  9. Unsupported major.minor version 52.0解决办法

    一.错误现象:当改变了jdk版本时,在编译java时,会遇到Unsupported major.minor version错误.jdk版本和stanford parser对应关系 JDK版本和Java ...

随机推荐

  1. 1.windows(64位)下使用curl命令

    转自:https://www.cnblogs.com/xing901022/p/4652624.html Curl命令可以通过命令行的方式,执行Http请求.在Elasticsearch中有使用的场景 ...

  2. Git 操作笔记

    ¥先放个地址:这个视频超详细 https://www.bilibili.com/video/av10475153 分了2部分: 1是GitHub的功能介绍:2是pc端+doc命令的操作+发布个人网站 ...

  3. Installation from source on Windows 7 with Visual C++2012

    在这部分说明里,你将会学习到在配备有Visual C++的Windows平台下从源码安装ViSP.下面的这些安装步骤已经在32位Windows系统,CMake3.1和Visual Studio 201 ...

  4. 使用UltraEdit配置多行注释和取消多行注释

    UltraEdit功能强大,使用方便,成为软件开发者必备的文档和代码编辑工具.有很多人也直接用它来写代码,如C/Java,脚本如:Perl/Tcl/JavaScript 等. 如果用来写代码,有一个不 ...

  5. 洛谷 P2690 接苹果

    P2690 接苹果 题目背景 USACO 题目描述 很少有人知道奶牛爱吃苹果.农夫约翰的农场上有两棵苹果树(编号为1和2), 每一棵树上都长满了苹果.奶牛贝茜无法摘下树上的苹果,所以她只能等待苹果 从 ...

  6. HDU 2886 Lou 1 Zhuang

    思维好重要.. 对于n+m == k , 当n == m || abs(n-m) == 1 时n*m取得最大值. 设 s = x*(l-x),s = lx-x^2.其导函数为 s' = -1/2x + ...

  7. js---16原型链

    var p = {name:"sss"}; var c2 = Object.create(p,{age:32,salar:"eee"});//c2就继承了p的属 ...

  8. Vue项目自动转换 px 为 rem,高保真还原设计图

    技术栈 vue-cli:使用脚手架工具创建项目. postcss-pxtorem:转换px为rem的插件. 自动设置根节点html的font-size 因为rem单位是相对于根节点的字体大小的,所以通 ...

  9. Android实现QQ分享及注意事项

    一.获取APPID和帮助文档 在前面我介绍了关于Android中微信分享的文章< Android实现微信分享及注意事项>这一篇文章来看看关于QQ分享. 可以参看新手引导和接入说明:http ...

  10. thuwc9102划水记

    thuwc9102划水记 Day -2 时隔两个月之后终于回一次家,心情非常愉悦,开始浪. Day 0 晚上回到学校,然而机房里并没多少人,大佬们明天才回来.╮(╯▽╰)╭ Day 1 中午饭菜挺好吃 ...