最近在工做中,遇到须要自定义Listview Item的左右滑动效果,ListView Item自己有点击事件,滑动过程当中item须要显示不一样于其余的背景颜色,滑开以后,背景颜色保持而且又有item内的点击事件这样:java
说明这个问题呢,须要先说明一下这个效果的实现逻辑:app
一、根据ACTION_DOWN事件的按下位置,获取ListView被按下位置的item,并设置按下背景颜色ide
二、根据ACTION_MOVE时间判断滑动距离,若是滑动达到左右滑动阈值,那么根据左右滑动距离,将View进行左右位置偏移,若是未达到阈值则不处理该事件。this
三、用户可点击整个item,触发跳转事件spa
四、View位置移动后,用户点击Delete按钮,删除该条itemrest
五、若是View已经被用户移开,那么若是再次触发ACTION_DOWN,被移开的View自动恢复到原来位置。code
主要逻辑如上,效果实现不难,重写ListView的onTouch事件就行了啊,可是在自测过程当中,发现一个大问题,就是在快速滑动ListView的时候,onTouch方法有事会丢失ACTION_DOWN事件。可是后面的Move事件和up事件历来没有丢失过。若是都丢失还好,可是MOVE和UP没有丢失,就致使关键判断丢失,又加上ListView item的复用机制,致使整个ListView状态出错,搞地整我的都很差了。事件
觉得是我本身写的方法哪里返回值有错,就将全部本身的写的东西注释掉,而后在onTouchEvent里面打印DOWN的状态,发现就是没有我代码,DOWN事件同样在快速滑动的时候接收不到。不是本身代码的问题就好。而后Google了好几下,发现问这个问题的很多,可是能解决个人问题方案没有。就在我觉得这是一个不可解决的问题的时候,我惊恐的发现,QQ的item就不存在这个问题,每次划开,再快的速度滑动,均可以正常关闭,我擦擦,不能输啊。get
因而开始看源码,细节就很少说,结论以下:源码
该DOWN事件在ListView 的dispatchTouchEvent的时候是能够接收到的,可是在onTouchEvent方法里面就接收不到了。因此咱们就去看看ListView的dispatchTouchEvent方法,ListView的dispatchTouchEvent是在ViewGroup里面,只看通过我分析关键的部分:
@Override public boolean dispatchTouchEvent(MotionEvent ev) { if (mInputEventConsistencyVerifier != null) { mInputEventConsistencyVerifier.onTouchEvent(ev, 1); } boolean handled = false; if (onFilterTouchEventForSecurity(ev)) { final int action = ev.getAction(); final int actionMasked = action & MotionEvent.ACTION_MASK; // Handle an initial down. if (actionMasked == MotionEvent.ACTION_DOWN) { // Throw away all previous state when starting a new touch gesture. // The framework may have dropped the up or cancel event for the previous gesture // due to an app switch, ANR, or some other state change. cancelAndClearTouchTargets(ev); resetTouchState(); } // Check for interception. final boolean intercepted; if (actionMasked == MotionEvent.ACTION_DOWN || mFirstTouchTarget != null) { final boolean disallowIntercept = (mGroupFlags & FLAG_DISALLOW_INTERCEPT) != 0; if (!disallowIntercept) { intercepted = onInterceptTouchEvent(ev); ev.setAction(action); // restore action in case it was changed } else { intercepted = false; } } else { // There are no touch targets and this action is not an initial down // so this view group continues to intercept touches. intercepted = true; } . . . }
if (actionMasked == MotionEvent.ACTION_DOWN || mFirstTouchTarget != null) { final boolean disallowIntercept = (mGroupFlags & FLAG_DISALLOW_INTERCEPT) != 0; if (!disallowIntercept) { intercepted = onInterceptTouchEvent(ev); ev.setAction(action); // restore action in case it was changed } else { intercepted = false; }
关键就在这里,咱们能够看到有一句:// restore action in case it was changed,由于快速滑动,当DOWN事件传递到onInterceptTouchEvent方法去判断是否是该拦截的时候,MOVE事件来了,而后被restore了。
因而在onInterceptTouchEvent方法里面打印了一下是否有丢失DOWN事件,发现是:木有丢失!!!好哒,问题就在这里了。了解问题在这里,这个问题就容易解决了。重写onInterceptTouchEvent,判断DOWN事件的位置是否是须要拦截,返回正确的TRUE or FALSE值就OK了喽。