android动态添加view滑动删除,Android实现类似QQ的滑动删除效果

本站也搜集了很多滑动删除的开源库,比如 :

有些效果甚至更好,那么为什么还转载这篇文章呢?1.这个纯粹,没有多余的功能,你可以直接了解到这个单一功能的实现细节;2.这有文章解释。以下是原文:

观察QQ的滑动删除效果,可以猜测可以滑动删除的部分主要包含两个部分,

一个是内容区域(用于放置正常显示的view),另一个是操作区域(用于放置删除按钮)。默认情况下,操作区域是不显示的,内容区域的大小是填充整个容

器,操作区域始终位于内容区域的右面。当开始滑动的时候,整个容器中的所有子view都像左滑动,如果操作区域此时是不可见的,设置为可见。

我的实现思路就是自定义一个layout

SwipeLayout继承自FrameLayout。SwipeLayout包含两个子view,第一个子view是内容区域,第二个子view是操作

区域。滑动效果的控制,主要就是通过检测SwipeLayout的touch事件来实现,这里我不想自己去通过监听touch事件来实现滑动效果,那是一

个很繁琐的过程。Android

support库里其实已经提供了一个很好的工具类来帮我们做这件事情ViewDragHelper。如果你看过Android原生的

DrawerLayout的代码,就会发现DrawerLayout的滑动效果也是通过ViewDragHelper类实现的。

下面先介绍一下ViewDragHelper类的使用。

首先需要在容器中创建一个ViewDragHelper类的对象。mDragHelper = ViewDragHelper.create(this, 1.0f, new ViewDragHelper.Callback());

接下来要把容器的事件处理委托给ViewDragHelper对象@Overridepublic boolean onInterceptTouchEvent(MotionEvent event) {

if (mDragHelper.shouldInterceptTouchEvent(event)) {

return true;

}

return super.onInterceptTouchEvent(event);

}

@Overridepublic boolean onTouchEvent(MotionEvent event) {

mDragHelper.processTouchEvent(event);

return true;

}

ViewDragHelper对象来决定

motion event是否是属于拖动过程。如果motion

event属于拖动过程,那么触摸事件就交给ViewDragHelper来处理,ViewDragHelper在处理拖动过程的时候,会调用

ViewDragHelper.Callback对象的一系列方法。

我们可以通过ViewDragHelper.Callback来监听以下几种事件:

1.拖动的状态改变

2.被拖动的view的位置改变

3.被拖动的view被放开的时间和位置

ViewDragHelper.Callback还提供了几个方法用来影响拖动过程。

1.控制view可以拖动的范围

2.确定某个view是否可以拖动

好了,直接看代码分析吧。

在SwipeLayout的inflate事件中,获取到contentView和actionView。@Override

protected void onFinishInflate() {

contentView = getChildAt(0);

actionView = getChildAt(1);

actionView.setVisibility(GONE);

}

在SwipeLayout的measure事件中,设置拖动的距离为actionView的宽度。@Override

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {

super.onMeasure(widthMeasureSpec, heightMeasureSpec);

dragDistance = actionView.getMeasuredWidth();

}

定义DragHelperCallback extends ViewDragHelper.Callback

DragHelperCallback的tryCaptureView方法,用来确定contentView和actionView是可以拖动的@Override

public boolean tryCaptureView(View view, int i) {

return view == contentView || view == actionView;

}

DragHelperCallback的onViewPositionChanged在被拖动的view位置改变的时候调用,如果被拖动的view是contentView,我们需要在这里更新actionView的位置,反之亦然。@Override

public void onViewPositionChanged(View changedView, int left, int top, int dx, int dy) {

draggedX = left;

if (changedView == contentView) {

actionView.offsetLeftAndRight(dx);

} else {

contentView.offsetLeftAndRight(dx);

}

if (actionView.getVisibility() == View.GONE) {

actionView.setVisibility(View.VISIBLE);

}

invalidate();

}

DragHelperCallback

的clampViewPositionHorizontal用来限制view在x轴上拖动,要实现水平拖动效果必须要实现这个方法,我们这里因为仅仅需要

实现水平拖动,所以没有实现clampViewPositionVertical方法。@Override

public int clampViewPositionHorizontal(View child, int left, int dx) {

if (child == contentView) {

final int leftBound = getPaddingLeft();

final int minLeftBound = -leftBound - dragDistance;

final int newLeft = Math.min(Math.max(minLeftBound, left), 0);

return newLeft;

} else {

final int minLeftBound = getPaddingLeft() + contentView.getMeasuredWidth() - dragDistance;

final int maxLeftBound = getPaddingLeft() + contentView.getMeasuredWidth() + getPaddingRight();

final int newLeft = Math.min(Math.max(left, minLeftBound), maxLeftBound);

return newLeft;

}

}

DragHelperCallback的getViewHorizontalDragRange方法用来限制view可以拖动的范围@Override

public int getViewHorizontalDragRange(View child) {

return dragDistance;

}

DragHelperCallback的onViewReleased方法中,根据滑动手势的速度以及滑动的距离来确定是否显示actionView。smoothSlideViewTo方法用来在滑动手势之后实现惯性滑动效果@Override

public void onViewReleased(View releasedChild, float xvel, float yvel) {

super.onViewReleased(releasedChild, xvel, yvel);

boolean settleToOpen = false;

if (xvel > AUTO_OPEN_SPEED_LIMIT) {

settleToOpen = false;

} else if (xvel 

settleToOpen = true;

} else if (draggedX <= -dragDistance / 2) {

settleToOpen = true;

} else if (draggedX > -dragDistance / 2) {

settleToOpen = false;

}

final int settleDestX = settleToOpen ? -dragDistance : 0;

viewDragHelper.smoothSlideViewTo(contentView, settleDestX, 0);

ViewCompat.postInvalidateOnAnimation(SwipeLayout.this);

}

最终的效果:

1429668683124792.gif

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值