Android开发仿QQ聊天滑动listview

主要是对分发事件的处理,代码如下:

public class SweepOnTouchListener implements OnTouchListener {

	private ListView mListView;
	private SweepLayout mDownView;
	private SweepLayout mLastDownView;
	
	private int mLastX, mLastY;
	
	private boolean mSweep = false;
	private boolean mIntercept = false;//是否取消点击事件
	
	private int mMinFlingVelocity;
	private int mMaxFlingVelocity;
	
	private final static int TAN = 2;
	
	private VelocityTracker mVelocityTracker;
	
	public SweepOnTouchListener(ListView listview){
		this.mListView = listview;
		ViewConfiguration vc = ViewConfiguration.get(mListView.getContext());
		mMinFlingVelocity = vc.getScaledMinimumFlingVelocity();
		mMaxFlingVelocity = vc.getScaledMaximumFlingVelocity();
	}
	
	//取消点击事件
	private void cancelTouchEvent(MotionEvent event){
		MotionEvent cancelEvent = MotionEvent.obtain(event);
		cancelEvent.setAction(MotionEvent.ACTION_CANCEL | (event.getActionIndex() << MotionEvent.ACTION_POINTER_INDEX_SHIFT));
		mListView.onTouchEvent(cancelEvent);
		cancelEvent.recycle();
	}
	
	@Override
	public boolean onTouch(View view, MotionEvent event) {
		// TODO Auto-generated method stub
		switch(event.getAction()){
		case MotionEvent.ACTION_DOWN:
			Rect rect = new Rect();
			int mDownX = (int)event.getX();
			int mDownY = (int)event.getY();
			
			//获得item
			for(int i = 0; i < mListView.getChildCount(); i ++){
				View childView = mListView.getChildAt(i);
				childView.getHitRect(rect);
				if(rect.contains(mDownX, mDownY)){
					mDownView = (SweepLayout) childView;
					break;
				}
			}
			
			mVelocityTracker = VelocityTracker.obtain();
			mVelocityTracker.addMovement(event);
			
			//当获得的item与上一个item不一致时,item恢复初始位置,取消其点击事件
			if(mDownView != mLastDownView){
				if(mLastDownView != null && mLastDownView.getScrollX() > 0) {
					mLastDownView.shrik(100);
					mDownView = null;
					mIntercept = true;
				}
			}
			view.onTouchEvent(event);
			break;
		case MotionEvent.ACTION_MOVE:
			if(mDownView == null) return false;
			mVelocityTracker.addMovement(event);
			int deltaX = (int)event.getX() - mLastX;
			int deltaY = (int)event.getY() - mLastY;
			if((deltaX < 0 || mDownView.getScrollX() >0) && Math.abs(deltaX) >= TAN * Math.abs(deltaY)){
				mSweep = true;
				mIntercept = true;
			}
			if(mSweep && mDownView != null){
				int distance = mDownView.getScrollX() - deltaX;
				if(distance < 0) distance = 0;
				if(distance > mDownView.getHolderWidth()) distance = mDownView.getHolderWidth();
				mDownView.scrollTo(distance, 0);
				mListView.requestDisallowInterceptTouchEvent(true);
			}
			if(mIntercept){
				cancelTouchEvent(event);
			}
			break;
		case MotionEvent.ACTION_UP:
			if(mVelocityTracker != null){
				mVelocityTracker.addMovement(event);
				mVelocityTracker.computeCurrentVelocity(1000);
			}
			if(mDownView != null){
				boolean finishScroll = false;
				if(mDownView.getScrollX() > 0.35 * mDownView.getHolderWidth()){
					finishScroll = true;
				}
				if(mVelocityTracker != null){
					float velocityX = mVelocityTracker.getXVelocity();
					float velocityY = mVelocityTracker.getYVelocity();
					//计算滑动速度和方向,如果速度为负值则为左滑,item为滑动位置,否则为初始位置
					if(velocityX < 0 && Math.abs(velocityX) >= mMinFlingVelocity && Math.abs(velocityX) <= mMaxFlingVelocity && Math.abs(velocityX) >= TAN * Math.abs(velocityY)){
						finishScroll = true;
						mIntercept = true;
					}else if(velocityX > 0 && Math.abs(velocityX) >= mMinFlingVelocity && Math.abs(velocityX) <= mMaxFlingVelocity && Math.abs(velocityX) >= TAN * Math.abs(velocityY)){
						finishScroll = false;
						mIntercept = true;
					}
				}
				if(finishScroll){
					mDownView.showSlide(100);
				}else{
					mDownView.shrik(100);
				}
			}
			//如果点击的item为展开的item,并且没有滑动过,则item回到初始位置。
			if(mDownView == mLastDownView){
				if(mDownView != null){
					if(mDownView.getScrollX() > 0 && !mSweep){
						mDownView.shrik(100);
						mIntercept = true;
					}
				}
			}
			if(mIntercept){
				cancelTouchEvent(event);
			}
			mSweep = false;
			mIntercept = false;
			mLastDownView = mDownView;
			
			mVelocityTracker.recycle();
			break;
		}
		mLastX = (int)event.getX();
		mLastY = (int)event.getY();
		return false;
	}
}

源码下载

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值