Android 模仿QQ側(cè)滑刪除ListView功能示例
需求:
1、listView可以側(cè)滑item,展示刪除按鈕,點(diǎn)擊刪除按鈕,刪除當(dāng)前的item
2、在刪除按鈕展示時(shí),點(diǎn)擊隱藏刪除按鈕,不響應(yīng)item的點(diǎn)擊事件
3、在刪除按鈕隱藏時(shí),點(diǎn)擊item響應(yīng)點(diǎn)擊事件
根據(jù)以上需求在網(wǎng)絡(luò)上查找響應(yīng)的例子,也有仿QQ側(cè)滑代碼,但不能滿足2和3的要求,因此修改了一把,代碼如下,共大家拍磚
第一步:重寫ListView
public class SwipeListView extends ListView {
private final static String TAG = "SwipeListView";
private int mScreenWidth; // 屏幕寬度
private int mDownX; // 按下點(diǎn)的x值
private int mDownY; // 按下點(diǎn)的y值
private int mDeleteBtnWidth;// 刪除按鈕的寬度
private boolean isDeleteShown = false; // 刪除按鈕是否正在顯示
private boolean isOnClick = false;
private ViewGroup mPointChild; // 當(dāng)前處理的item
private LinearLayout.LayoutParams mLayoutParams; // 當(dāng)前處理的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();
// 獲取當(dāng)前點(diǎn)的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();
// 為什么要重新設(shè)置layout_width 等于屏幕寬度
// 因?yàn)閙atch_parent時(shí),不管你怎么滑,都不會顯示刪除按鈕
// why? 因?yàn)閙atch_parent時(shí),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)) {
// 如果向左滑動(dòng)
if (nowX < mDownX) {
// 計(jì)算要偏移的距離
int scroll = (nowX - mDownX) / 2;
// 如果大于了刪除按鈕的寬度, 則最大為刪除按鈕的寬度
if (-scroll >= mDeleteBtnWidth) {
scroll = -mDeleteBtnWidth;
}
// 重新設(shè)置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
// 否則恢復(fù)默認(rèn)
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;
}
/**
* 變?yōu)檎顟B(tài)
*/
public void turnToNormal() {
mLayoutParams.leftMargin = 0;
mPointChild.getChildAt(0).setLayoutParams(mLayoutParams);
}
/**
* 當(dāng)前是否可點(diǎn)擊
*
* @return 是否可點(diǎn)擊
*/
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;
}
第三步:寫一個(gè)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) + "被點(diǎn)擊了",
// 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) + "被點(diǎn)擊了",
// 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>
重點(diǎn)注意:
int downPosition = pointToPosition(mDownX, mDownY);
downPosition 在使用過程中得到-1,導(dǎo)致后面方法調(diào)用異常!
以上所述是小編給大家介紹的Android 模仿QQ側(cè)滑刪除ListView功能示例,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時(shí)回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
- android ItemTouchHelper實(shí)現(xiàn)可拖拽和側(cè)滑的列表的示例代碼
- Android高仿QQ6.0側(cè)滑刪除實(shí)例代碼
- Android仿QQ微信側(cè)滑刪除效果
- Android開發(fā)中記一個(gè)SwipeMenuListView側(cè)滑刪除錯(cuò)亂的Bug
- Android recyclerview實(shí)現(xiàn)拖拽排序和側(cè)滑刪除
- Android自定義view系列之99.99%實(shí)現(xiàn)QQ側(cè)滑刪除效果實(shí)例代碼詳解
- android的RecyclerView實(shí)現(xiàn)拖拽排序和側(cè)滑刪除示例
- android ListView和GridView拖拽移位實(shí)現(xiàn)代碼
- android 大圖片拖拽并縮放實(shí)現(xiàn)原理
- Android使用ItemTouchHelper實(shí)現(xiàn)側(cè)滑刪除和拖拽
相關(guān)文章
Android?Studio?Electric?Eel支持手機(jī)投屏
這篇文章主要為大家介紹了Android?Studio?Electric?Eel支持手機(jī)投屏功能詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-05-05
Android開發(fā)實(shí)現(xiàn)自定義水平滾動(dòng)的容器示例
這篇文章主要介紹了Android開發(fā)實(shí)現(xiàn)自定義水平滾動(dòng)的容器,涉及Android滾動(dòng)容器的事件響應(yīng)、屬性運(yùn)算與修改相關(guān)操作技巧,需要的朋友可以參考下2017-10-10
Android在不使用數(shù)據(jù)庫的情況下存儲數(shù)據(jù)的方法
這篇文章主要介紹了Android在不使用數(shù)據(jù)庫的情況下存儲數(shù)據(jù)的方法,涉及Android存儲數(shù)據(jù)的相關(guān)技巧,需要的朋友可以參考下2015-04-04
Android自定義HorizontalScrollView實(shí)現(xiàn)qq側(cè)滑菜單
本文主要介紹了android自定義HorizontalScrollView實(shí)現(xiàn)qq側(cè)滑菜單的相關(guān)知識。具有很好的參考價(jià)值。下面跟著小編一起來看下吧2017-04-04
Android控件ViewFlipper仿淘寶頭條垂直滾動(dòng)廣告條
這篇文章主要為大家詳細(xì)介紹了Android控件ViewFlipper仿淘寶頭條垂直滾動(dòng)廣告條,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-05-05
Android?studio實(shí)現(xiàn)動(dòng)態(tài)背景頁面
這篇文章主要為大家詳細(xì)介紹了Android?studio實(shí)現(xiàn)動(dòng)態(tài)背景頁面,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-04-04

