Android 模仿QQ侧滑删除ListView功能示例

需求:

1、listView可以侧滑item,展示删除按钮,点击删除按钮,删除当前的item

2、在删除按钮展示时,点击隐藏删除按钮,不响应item的点击事件

3、在删除按钮隐藏时,点击item响应点击事件

根据以上需求在网络上查找响应的例子,也有仿QQ侧滑代码,但不能满足2和3的要求,因此修改了一把,代码如下,共大家拍砖

第一步:重写ListView

public class SwipeListView extends ListView { private final static String TAG = "SwipeListView"; private int mScreenWidth; // 屏幕宽度 private int mDownX; // 按下点的x值 private int mDownY; // 按下点的y值 private int mDeleteBtnWidth;// 删除按钮的宽度 private boolean isDeleteShown = false; // 删除按钮是否正在显示 private boolean isOnClick = false; private ViewGroup mPointChild; // 当前处理的item private LinearLayout.LayoutParams mLayoutParams; // 当前处理的item的LayoutParams public SwipeListView(Context context, AttributeSet attrs) { this(context, attrs, 0); } public SwipeListView(Context context, AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); // 获取屏幕宽度 WindowManager wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE); DisplayMetrics dm = new DisplayMetrics(); wm.getDefaultDisplay().getMetrics(dm); mScreenWidth = dm.widthPixels; } @Override public boolean onTouchEvent(MotionEvent ev) { switch (ev.getAction()) { case MotionEvent.ACTION_DOWN: performActionDown(ev); break; case MotionEvent.ACTION_MOVE: return performActionMove(ev); case MotionEvent.ACTION_UP: return performActionUp(ev); // break; } return super.onTouchEvent(ev); } // 处理action_down事件 private void performActionDown(MotionEvent ev) { // Log.e(TAG,"performActionDown===="+isDeleteShown); if (isDeleteShown) { turnToNormal(); } isOnClick = true; mDownX = (int) ev.getX(); mDownY = (int) ev.getY(); // 获取当前点的item int downPosition = pointToPosition(mDownX, mDownY); int firstPosition= getFirstVisiblePosition(); Log.e(TAG,"performActionDown====downPosition:"+downPosition+"==firstPosition"+firstPosition); if(downPosition < 0) return; mPointChild = (ViewGroup) getChildAt(downPosition-firstPosition); // 获取删除按钮的宽度 mDeleteBtnWidth = mPointChild.getChildAt(1).getLayoutParams().width; mLayoutParams = (LinearLayout.LayoutParams) mPointChild.getChildAt(0) .getLayoutParams(); // 为什么要重新设置layout_width 等于屏幕宽度 // 因为match_parent时,不管你怎么滑,都不会显示删除按钮 // why? 因为match_parent时,ViewGroup就不去布局剩下的view mLayoutParams.width = mScreenWidth; mPointChild.getChildAt(0).setLayoutParams(mLayoutParams); } // 处理action_move事件 private boolean performActionMove(MotionEvent ev) { // Log.e(TAG, "performActionMove====" + isDeleteShown); int nowX = (int) ev.getX(); int nowY = (int) ev.getY(); isOnClick = false; if (Math.abs(nowX - mDownX) > Math.abs(nowY - mDownY)) { // 如果向左滑动 if (nowX < mDownX) { // 计算要偏移的距离 int scroll = (nowX - mDownX) / 2; // 如果大于了删除按钮的宽度, 则最大为删除按钮的宽度 if (-scroll >= mDeleteBtnWidth) { scroll = -mDeleteBtnWidth; } // 重新设置leftMargin mLayoutParams.leftMargin = scroll; mPointChild.getChildAt(0).setLayoutParams(mLayoutParams); } return true; } return super.onTouchEvent(ev); } // 处理action_up事件 private boolean performActionUp(MotionEvent ev) { boolean falg = false; if(isOnClick && !isDeleteShown) { falg = true; } // 偏移量大于button的一半,则显示button // 否则恢复默认 if (-mLayoutParams.leftMargin >= mDeleteBtnWidth / 2) { mLayoutParams.leftMargin = -mDeleteBtnWidth; isDeleteShown = true; } else { turnToNormal(); isDeleteShown = false; } mPointChild.getChildAt(0).setLayoutParams(mLayoutParams); // Log.e(TAG, "performActionUp====" + isDeleteShown); if(falg) { return super.onTouchEvent(ev); } return true; } /** * 变为正常状态 */ public void turnToNormal() { mLayoutParams.leftMargin = 0; mPointChild.getChildAt(0).setLayoutParams(mLayoutParams); } /** * 当前是否可点击 * * @return 是否可点击 */ public boolean canClick() { return !isDeleteShown; } }

第二步:适配器

class SwipeListAdapter extends BaseAdapter { @Override public int getCount() { return mData.size(); } @Override public Object getItem(int position) { return mData.get(position); } @Override public long getItemId(int position) { return position; } @Override public View getView(int position, View convertView, ViewGroup parent) { ViewHolder holder = null; if (null == convertView) { holder = new ViewHolder(); convertView = View.inflate(TestListViewActivity.this, R.layout.item_swipe_list, null); holder.tv = (LinearLayout) convertView.findViewById(R.id.tv); holder.tvName = (TextView) convertView.findViewById(R.id.tv_name); holder.delete = (TextView) convertView.findViewById(R.id.delete); convertView.setTag(holder); } else { holder = (ViewHolder) convertView.getTag(); } holder.tvName.setText(mData.get(position)); final int pos = position; holder.delete.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { mData.remove(pos); notifyDataSetChanged(); mListView.turnToNormal(); } }); return convertView; } } static class ViewHolder { LinearLayout tv; TextView tvName; TextView delete; }

第三步:写一个TestListViewActivity

private SwipeListView mListView; private ArrayList<String> mData = new ArrayList<String>() { { for (int i = 0; i < 20; i++) { add("hello world, hello android " + i); } } }; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_test_list_view); Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar); setSupportActionBar(toolbar); FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab); fab.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { Snackbar.make(view, "Replace with your own action", Snackbar.LENGTH_LONG) .setAction("Action", null).show(); } }); mListView = (SwipeListView) findViewById(R.id.list); mListView.setAdapter(new SwipeListAdapter()); // mListView.setOnItemLongClickListener(new AdapterView.OnItemLongClickListener() { // @Override // public boolean onItemLongClick(AdapterView<?> parent, View view, int position, long id) { // Toast.makeText(TestListViewActivity.this, mData.get(position) + "被点击了", // Toast.LENGTH_SHORT).show(); // return false; // } // }); mListView.setOnItemClickListener(new AdapterView.OnItemClickListener() { @Override public void onItemClick(AdapterView<?> parent, View view, int position, long id) { Log.e("SwipeListView", "setOnItemClickListener====" + mListView.canClick()); // Toast.makeText(TestListViewActivity.this, mData.get(position) + "被点击了", // Toast.LENGTH_SHORT).show(); } }); }

第四步:布局文件

<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" xmlns:app="http://schemas.android.com/apk/res-auto" android:layout_width="match_parent" android:layout_height="match_parent" app:layout_behavior="@string/appbar_scrolling_view_behavior" tools:showIn="@layout/activity_test_list_view" tools:context="com.kimascend.ledappd1.activity.TestListViewActivity"> <com.kimascend.ledappd1.view.SwipeListView android:id="@+id/list" android:layout_width="wrap_content" android:layout_height="wrap_content" android:cacheColorHint="@android:color/transparent" android:listSelector="@android:color/transparent" android:divider="@android:color/darker_gray" android:dividerHeight="2dp"> </com.kimascend.ledappd1.view.SwipeListView> </RelativeLayout>

第五步:item的布局文件

<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="wrap_content" android:orientation="horizontal"> <LinearLayout android:id="@+id/tv" android:orientation="horizontal" android:layout_width="match_parent" android:layout_height="wrap_content" android:background="@android:color/white" android:paddingBottom="20dp" android:paddingLeft="10dp" android:paddingTop="20dp"> <ImageView android:layout_width="wrap_content" android:layout_height="wrap_content" android:background="@drawable/group_name_rgb" android:id="@+id/imageView8" /> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:textAppearance="?android:attr/textAppearanceLarge" android:text="Large Text" android:layout_gravity="center_vertical" android:id="@+id/tv_name" /> </LinearLayout> <TextView android:id="@+id/delete" android:layout_width="80dp" android:layout_height="match_parent" android:background="#FFFF0000" android:gravity="center" android:paddingLeft="20dp" android:paddingRight="20dp" android:text="删除" android:textColor="@android:color/white" /> </LinearLayout>

重点注意:

int downPosition = pointToPosition(mDownX, mDownY);

downPosition 在使用过程中得到-1,导致后面方法调用异常!

以上所述是小编给大家介绍的Android 模仿QQ侧滑删除ListView功能示例,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对脚本之家网站的支持!

时间: 2024-09-19 23:11:42

Android 模仿QQ侧滑删除ListView功能示例的相关文章

Android使用Item Swipemenulistview实现仿QQ侧滑删除功能

大家都用过QQ,肯定有人好奇QQ滑动删除Item的效果是怎样实现的,其实我们使用Swipemenulistview就可以简单的实现.先看看我们项目中的效果: 使用的时候可以把Swipemenulistview作为一个library,也可以把Swipemenulistview的源码拷贝到我们的项目中来,使用步骤大致可以分为三步:1.在布局中配置:2.在Java代码中初始化配置:3.按钮点击事件的处理 1.在布局中配置 xml布局文件中只需要简单使用这个自定义的ListView就行了,需要注意的是必

Android 仿QQ头像自定义截取功能_Android

看了Android版QQ的自定义头像功能,决定自己实现,随便熟悉下android绘制和图片处理这一块的知识. 先看看效果: 思路分析: 这个效果可以用两个View来完成,上层View是一个遮盖物,绘制半透明的颜色,中间挖了一个圆:下层的View用来显示图片,具备移动和缩放的功能,并且能截取某区域内的图片. 涉及到的知识点: 1.Matrix,图片的移动和缩放 2.Paint的setXfermode方法 3.图片放大移动后,截取一部分 编码实现: 自定义三个View: 1.下层View:ClipP

Android 仿QQ头像自定义截取功能

看了Android版QQ的自定义头像功能,决定自己实现,随便熟悉下android绘制和图片处理这一块的知识. 先看看效果: 思路分析: 这个效果可以用两个View来完成,上层View是一个遮盖物,绘制半透明的颜色,中间挖了一个圆:下层的View用来显示图片,具备移动和缩放的功能,并且能截取某区域内的图片. 涉及到的知识点: 1.Matrix,图片的移动和缩放 2.Paint的setXfermode方法 3.图片放大移动后,截取一部分 编码实现: 自定义三个View: 1.下层View:ClipP

使用RecylerView完成拖动排序高仿qq侧滑删除功能_Android

 最近使用到Recylerview完成拖动排序,侧滑删除,在此记录一下. 需要使用到:ItemTouchHelper.Callback这个类. 效果图: 在有RecylerView的Activity中,除了设置Adapter还需要做的事情有: MyItemTouchHelperCallback callback = new MyItemTouchHelperCallback(adapter); //此类继承ItemTouchHelper.Callback,这是帮助处理RecylerView拖动侧

Android自定义view系列之99.99%实现QQ侧滑删除效果实例代码详解

首先声明本文是基于GitHub上"baoyongzhang"的SwipeMenuListView修改而来,该项目地址: https://github.com/baoyongzhang/SwipeMenuListView 可以说这个侧滑删除效果是我见过效果最好且比较灵活的项目,没有之一!!! 但是在使用它之前需要给大家提两点注意事项: 1,该项目支持Gradle dependence,但是目前作者提供的依赖地址对应的项目不是最新的项目,依赖过后的代码与demo中使用的不一致,会提示没有B

使用RecylerView完成拖动排序高仿qq侧滑删除功能

最近使用到Recylerview完成拖动排序,侧滑删除,在此记录一下. 需要使用到:ItemTouchHelper.Callback这个类. 效果图: 在有RecylerView的Activity中,除了设置Adapter还需要做的事情有: MyItemTouchHelperCallback callback = new MyItemTouchHelperCallback(adapter); //此类继承ItemTouchHelper.Callback,这是帮助处理RecylerView拖动侧滑

Android编程实现分页加载ListView功能示例

本文实例讲述了Android编程实现分页加载ListView功能.分享给大家供大家参考,具体如下: package eoe.listview; import android.app.Activity; import android.database.Cursor; import android.os.Bundle; import android.os.Handler; import android.os.Message; import android.view.View; import andr

Android实现屏蔽微信拉黑和删除联系人功能示例

Android实现屏蔽微信拉黑和删除联系人功能,废话不多说,具体如下: 实现效果: 让微信永远弹不出那个删除的对话框不就相当于屏蔽掉该功能了吗?哈哈效果如图: 实现原理: 1.我们知道,其实微信每次删除联系人都会弹出此页面 2.如果你对AccessibilityService有过了解或者有看过我之前的两篇博客,你会知道,其实每次弹出这个框,都会触发AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED这个事件,所以我们只要在每次触发该事件的时候进行判断当前页面是

Android编程实现自定义Tab选项卡功能示例

本文实例讲述了Android编程实现自定义Tab选项卡功能.分享给大家供大家参考,具体如下: import android.app.TabActivity; import android.content.Intent; import android.os.Bundle; import android.widget.*; import android.widget.TabHost.OnTabChangeListener; import android.os.Build; import androi