android侧拉菜单只能手动划出,Android 中通过ViewDragHelper实现ListView的Item的侧拉划出效果...

先来看看,今天要实现的自定义控件效果图:

c47ab6f51082ac20a3304aee84b09ee0.gif

关于ViewDragHelper的使用,大家可以先看这篇文章ViewDragHelper的使用介绍

实现该自定义控件的大体步骤如下:

1.ViewDragHelper使用的3部曲,初始化ViewDragHelper,传递触摸事件,实现ViewDragHelper.Callback抽象类.

2.需要创建2个直接的子View,分别是前景View和背景View,代表ListView每一项Item的布局的组成,如下所示:

未划出时显示的FrontView:

896341fffb5199b18ab3343202c0f8e5.png

划出后的右边显示BackView:

57b34427a22e28148541e4e5b059de7d.png

以上2部分就是该自定义控件要包含的2个直接子View.

3.需要获取FrontView的宽高,宽度其实就是屏幕的宽度,高度就是ListView每一项Item的高度;还需获取BackView的宽度,因为这个宽度就是侧滑的最大范围.

4.需要确定FrontView和BackView的初始位置,在onLayout方法中确定,即默认情况下是只显示FrontView的.这个实现起来也很简单,FrontView的left=0,BackView的left=FrontView的right即可.

5.需要同步FrontView和BackView的滑动,即滑动FrontView的时候BackView也需要跟着划出,同样滑动BackView的时候也需要FrontView跟着滑动.

6.需要解决侧拉划出的效果是否有动画效果.平滑滑动的动画可以通过ViewDragHelper轻松实现.

好了,直接上自定义的SwipeLayout源码:

/**

* Created by mChenys on 2015/12/26.

*/

public class SwipeLayout extends FrameLayout {

private ViewDragHelper.Callback mCallback;

private ViewDragHelper mDragHelper;

private View mBackView; //item的侧边布局

private View mFrontView;//当前显示的item布局

private int mWidth; //屏幕的宽度,mFrontView的宽度

private int mHeight; //mFrontView的高度

private int mRange;//mFrontView侧拉时向左移动的最大距离,即mBackView的宽度

public SwipeLayout(Context context) {

this(context, null);

}

public SwipeLayout(Context context, AttributeSet attrs) {

this(context, attrs, 0);

}

public SwipeLayout(Context context, AttributeSet attrs, int defStyleAttr) {

super(context, attrs, defStyleAttr);

init();

}

//1.初始ViewDragHelper

private void init() {

mCallback = new ViewDragHelper.Callback() {

//3.在回调方法中处理触摸事件

@Override

public boolean tryCaptureView(View child, int pointerId) {

return true; //允许所有子控件的滑动

}

//设定滑动的边界值

@Override

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

if (child == mFrontView) {

//前景View的滑动范围是(0~ -mRange)

if (left > 0) {

left = 0;

} else if (left < -mRange) {

left = -mRange;

}

}

if (child == mBackView) {

//背景View的滑动范围是(mWidth - mRange ~ mWidth)

if (left > mWidth) {

left = mWidth;

} else if (left < (mWidth - mRange)) {

left = mWidth - mRange;

}

}

//返回修正过的建议值

return left;

}

//监听View的滑动位置的改变,同步前景View和背景View的滑动事件

@Override

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

if (changedView == mFrontView) {

//当滑动前景View时,也需要滑动背景View

mBackView.offsetLeftAndRight(dx);

} else if (changedView == mBackView) {

//当滑动背景View时,也需要滑动前景View

mFrontView.offsetLeftAndRight(dx);

}

// 兼容老版本

invalidate();

}

//处理释放后的开启和关闭动作

@Override

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

if (xvel < 0) {

//有向左滑动的速度,则打开

open();

} else if (xvel == 0 && mFrontView.getLeft() < -mRange / 2.0f) {

//前景View向左滑动的left小于背景View宽度一半的负值时,打开

open();

} else {

//其他情况为关闭

close();

}

}

};

mDragHelper = ViewDragHelper.create(this, mCallback);

}

//2.传递触摸事件

@Override

public boolean onInterceptTouchEvent(MotionEvent ev) {

return mDragHelper.shouldInterceptTouchEvent(ev);

}

@Override

public boolean onTouchEvent(MotionEvent event) {

try {

mDragHelper.processTouchEvent(event);

} catch (Exception e) {

e.printStackTrace();

}

return true;

}

//获取子控件的引用

@Override

protected void onFinishInflate() {

super.onFinishInflate();

mBackView = getChildAt(0); //获取背景View,即展示数据的Item的右边隐藏的侧滑布局

mFrontView = getChildAt(1);//获取前景View,即展示数据的Item

}

//获取子控件的相关宽高信息

@Override

protected void onSizeChanged(int w, int h, int oldw, int oldh) {

super.onSizeChanged(w, h, oldw, oldh);

mWidth = mFrontView.getMeasuredWidth();

mHeight = mFrontView.getMeasuredHeight();

mRange = mBackView.getMeasuredWidth();

}

//确定子控件的初始位置

@Override

protected void onLayout(boolean changed, int left, int top, int right, int bottom) {

super.onLayout(changed, left, top, right, bottom);

layoutChildView(false);

}

/**

* 放置子控件的位置

*

* @param isOpen 是否是打开前景View,true打开,false关闭

*/

private void layoutChildView(boolean isOpen) {

//计算前景View的位置,将坐标信息封装到矩形中

Rect fontRect = computerFontViewRect(isOpen);

//摆放前景View

mFrontView.layout(fontRect.left, fontRect.top, fontRect.right, fontRect.bottom);

//摆放背景View,left坐标是前景View的right坐标

int left = fontRect.right;

mBackView.layout(left, 0, left + mRange, mHeight);

//由于上面是后摆放背景View,所以会覆盖前景View,因此需要通过下面的方式将前景View显示在前面

bringChildToFront(mFrontView);

}

/**

* 计算前景View的坐标

*

* @param isOpen 是否是打开前景View

* @return

*/

private Rect computerFontViewRect(boolean isOpen) {

int left = isOpen ? -mRange : 0;

return new Rect(left, 0, left + mWidth, mHeight);

}

/**

* 打开侧边栏mBackView,默认平滑打开

*/

public void open() {

open(true);

}

/**

* 打开侧边栏mBackView

*

* @param isSmooth 是否平滑打开

*/

public void open(boolean isSmooth) {

if (isSmooth) {

if (mDragHelper.smoothSlideViewTo(mFrontView, -mRange, 0)) {

//动画在继续

ViewCompat.postInvalidateOnAnimation(this);

}

} else {

layoutChildView(true);

}

}

/**

* 关闭侧边栏mBackView,默认平滑关闭

*/

public void close() {

close(true);

}

/**

* 关闭侧边栏mBackView

*

* @param isSmooth 是否平滑关闭

*/

public void close(boolean isSmooth) {

if (isSmooth) {

if (mDragHelper.smoothSlideViewTo(mBackView, mWidth, 0)) {

//动画在继续

ViewCompat.postInvalidateOnAnimation(this);

}

} else {

layoutChildView(false);

}

}

@Override

public void computeScroll() {

super.computeScroll();

if (mDragHelper.continueSettling(true)) {

//动画还在继续

ViewCompat.postInvalidateOnAnimation(this);

}

}

}

如何使用呢?

使用该控件,必须要让其有2个直接的子控件,如下布局所示:

xmlns:android="http://schemas.android.com/apk/res/android"

android:id="@+id/sl"

android:layout_width="match_parent"

android:layout_height="60dp"

android:minHeight="60dp"

android:background="#44000000" >

android:layout_width="wrap_content"

android:layout_height="match_parent"

android:orientation="horizontal" >

android:id="@+id/tv_call"

android:layout_width="60dp"

android:layout_height="match_parent"

android:background="#666666"

android:gravity="center"

android:text="Edit"

android:textColor="#ffffff" />

android:id="@+id/tv_del"

android:layout_width="60dp"

android:layout_height="match_parent"

android:background="#ff0000"

android:gravity="center"

android:text="Delete"

android:textColor="#ffffff" />

android:layout_width="match_parent"

android:layout_height="match_parent"

android:background="#44ffffff"

android:gravity="center_vertical"

android:orientation="horizontal" >

android:id="@+id/iv_image"

android:layout_width="40dp"

android:layout_height="40dp"

android:layout_marginLeft="15dp"

android:src="@drawable/head_1" />

android:id="@+id/tv_name"

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:layout_marginLeft="15dp"

android:text="Name" />

就是这么简单,跑起来就可以用了.不过这个只是定义出了SwipeLayout控件,如果要集成到ListView中,还需要做进一步的处理.

例如实现如下效果:

f9b1a5ea32908c45bdf046826d45d17e.gif

需要考虑2点:

1.在自定义SwipeLayout控件内需要处理3种状态,打开,关闭,拖拽.

2.需要添加一个侧滑监听接口,用于对外暴露当前SwipeLayout的打开,关闭,拖拽,将要打开,将要关闭这5种情况.接口定义如下所示:

/**

* 侧拉SwipeLayout的监听

* Created by mChenys on 2015/12/26.

*/

public interface SwipeViewListener {

//关闭

void onClose(SwipeLayout mSwipeLayout);

//打开

void onOpen(SwipeLayout mSwipeLayout);

//正在侧拉

void onDraging(SwipeLayout mSwipeLayout);

//开始要去关闭

void onStartClose(SwipeLayout mSwipeLayout);

//开始要去开启

void onStartOpen(SwipeLayout mSwipeLayout);

}

SwipeLayout的3种状态,用enum表示即定义接收获取SwipeViewListener监听器的方法1

//以下是定义SwipeLayout的打开,关闭,滑动的3种状态

public enum Status {

CLOSE, OPEN, DRAGING;

}

//默认关闭

private Status mStatus = Status.CLOSE;

//滑动的监听器

private SwipeViewListener mSwipeViewListener;

//设置监听器

public void setSwipeViewListener(SwipeViewListener swipeViewListener) {

mSwipeViewListener = swipeViewListener;

}

在onViewPositionChanged方法内添加多一个方法,用于处理拖拽的监听.

/**

* 处理滑动,打开,关闭的3种情况

* 在onViewPositionChanged 调用

*/

private void dispatchSwipeEvent() {

if (mSwipeViewListener != null) {

mSwipeViewListener.onDraging(this);

}

//记录上一次的状态

Status preStatus = mStatus;

//获取当前的状态

mStatus = getCurrStatus();

if (preStatus != mStatus && null != mSwipeViewListener) {

//说明有状态发生变化

if (mStatus == Status.CLOSE) {

//关闭

mSwipeViewListener.onClose(this);

} else if (mStatus == Status.OPEN) {

//打开

mSwipeViewListener.onOpen(this);

} else if (mStatus == Status.DRAGING) {

//这里有2中情况,要么要打开,要么要关闭

if (preStatus == Status.CLOSE) {

//如果之前是关闭的,那么就是要打开

mSwipeViewListener.onStartOpen(this);

} else if (preStatus == Status.OPEN) {

//如果之前是打开,那么就是要关闭

mSwipeViewListener.onStartClose(this);

}

}

}

}

/**

* 获取当前的状态

*

* @return

*/

private Status getCurrStatus() {

int left = mFrontView.getLeft();

if (left == 0) {

return Status.CLOSE;

} else if (left == -mRange) {

return Status.OPEN;

}

return Status.DRAGING;

}

最后来看看MainActivity的测试:

public class MainActivity extends AppCompatActivity {

private List mData = new ArrayList<>();//数据集合

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

//获取数据,注意:Arrays.asList返回的并不是一个java.util.ArrayList,而是一个Arrays类的内部类,该List实现是不能进行增删操作的

//因此必须再包装一下

mData = new ArrayList<>(Arrays.asList(Constant.NAME));

ListView listView = new ListView(this);

listView.setAdapter(mAdapter);

setContentView(listView);

}

//自定义适配器

private BaseAdapter mAdapter = new BaseAdapter() {

//标记当前打开的SwipeLayout的集合

private List mOpenItem = new ArrayList<>();

@Override

public int getCount() {

return mData.size();

}

@Override

public String getItem(int position) {

return mData.get(position);

}

@Override

public long getItemId(int position) {

return position;

}

@Override

public View getView(final int position, View convertView, ViewGroup parent) {

ViewHolder holder = null;

if (null == convertView) {

holder = new ViewHolder();

convertView = View.inflate(MainActivity.this, R.layout.item_list, null);

holder.mSwipeLayout = (SwipeLayout) convertView;

holder.tvName = (TextView) convertView.findViewById(R.id.tv_name);

holder.tvDel = (TextView) convertView.findViewById(R.id.tv_del);

holder.tvEdit = (TextView) convertView.findViewById(R.id.tv_edit);

convertView.setTag(holder);

} else {

holder = (ViewHolder) convertView.getTag();

}

//设置侧拉监听

holder.mSwipeLayout.setSwipeViewListener(getSwipeViewListener());

holder.tvName.setText(getItem(position));

holder.tvDel.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

//删除

mData.remove(position);

mAdapter.notifyDataSetChanged();

}

});

holder.tvEdit.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

ToastUtils.showToast(MainActivity.this,"编辑");

}

});

return convertView;

}

class ViewHolder {

TextView tvName, tvDel, tvEdit;

SwipeLayout mSwipeLayout;

}

//获取滑动监听器

private SwipeViewListener getSwipeViewListener() {

return new SwipeViewListener() {

@Override

public void onClose(SwipeLayout mSwipeLayout) {

//关闭是移除

mOpenItem.remove(mSwipeLayout);

ToastUtils.showToast(MainActivity.this, "关闭");

}

@Override

public void onOpen(SwipeLayout mSwipeLayout) {

//打开时添加

mOpenItem.add(mSwipeLayout);

ToastUtils.showToast(MainActivity.this, "打开");

}

@Override

public void onDraging(SwipeLayout mSwipeLayout) {

}

@Override

public void onStartClose(SwipeLayout mSwipeLayout) {

ToastUtils.showToast(MainActivity.this, "开始关闭");

}

@Override

public void onStartOpen(SwipeLayout mSwipeLayout) {

//将要打开时,需要将集合中的之前打开的SwipeLayout统统关闭

for (SwipeLayout swipeLayout : mOpenItem) {

swipeLayout.close();

}

mOpenItem.clear();//清空集合

ToastUtils.showToast(MainActivity.this, "开始打开");

}

};

}

};

}

总结

以上所述是小编给大家介绍的 Android 中通过ViewDragHelper实现ListView的Item的侧拉划出效果,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对脚本之家网站的支持!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值