android實(shí)現(xiàn)QQ微信側(cè)滑刪除效果
最近由于項(xiàng)目需求,需要做一個(gè)listview中的item策劃刪除的效果,與是查找資料和參考了一些相關(guān)的博客,終于完美實(shí)現(xiàn)了策劃刪除的效果。
先看一下效果圖(研究了半天竟然沒研究出來真機(jī)上gif圖怎么做,大家將就看一下吧)。
側(cè)滑效果圖
點(diǎn)擊刪除后的截圖
點(diǎn)擊刪除后,listview中的第一個(gè)“微信團(tuán)隊(duì)”被刪除。
接下來看代碼部分,很多注釋都在代碼中,直接上代碼。
首先自定義個(gè)SlideView繼承LinearLayout。
import android.util.AttributeSet; import android.util.Log; import android.util.TypedValue; import android.view.MotionEvent; import android.view.View; import android.widget.LinearLayout; import android.widget.RelativeLayout; import android.widget.Scroller; import android.widget.TextView; /** * SlideView 繼承自LinearLayout */ public class SlideView extends LinearLayout { private static final String TAG = "SlideView"; private Context mContext; // 用來放置所有view的容器 private LinearLayout mViewContent; // 用來放置內(nèi)置view的容器,比如刪除 按鈕 private RelativeLayout mHolder; // 彈性滑動(dòng)對(duì)象,提供彈性滑動(dòng)效果 private Scroller mScroller; // 滑動(dòng)回調(diào)接口,用來向上層通知滑動(dòng)事件 private OnSlideListener mOnSlideListener; // 內(nèi)置容器的寬度 單位:dp private int mHolderWidth = 120; // 分別記錄上次滑動(dòng)的坐標(biāo) private int mLastX = 0; private int mLastY = 0; // 用來控制滑動(dòng)角度,僅當(dāng)角度a滿足如下條件才進(jìn)行滑動(dòng):tan a = deltaX / deltaY > 2 private static final int TAN = 2; public interface OnSlideListener { // SlideView的三種狀態(tài):開始滑動(dòng),打開,關(guān)閉 public static final int SLIDE_STATUS_OFF = 0; public static final int SLIDE_STATUS_START_SCROLL = 1; public static final int SLIDE_STATUS_ON = 2; public void onSlide(View view, int status); } public SlideView(Context context) { super(context); initView(); } public SlideView(Context context, AttributeSet attrs) { super(context, attrs); initView(); } private void initView() { mContext = getContext(); // 初始化彈性滑動(dòng)對(duì)象 mScroller = new Scroller(mContext); // 設(shè)置其方向?yàn)闄M向 setOrientation(LinearLayout.HORIZONTAL); // 將slide_view_merge加載進(jìn)來 View.inflate(mContext, R.layout.slide_view_merge, this); mViewContent = (LinearLayout) findViewById(R.id.view_content); mHolderWidth = Math.round(TypedValue.applyDimension( TypedValue.COMPLEX_UNIT_DIP, mHolderWidth, getResources() .getDisplayMetrics())); } // 設(shè)置按鈕的內(nèi)容,也可以設(shè)置圖標(biāo)啥的,我沒寫 public void setButtonText(CharSequence text) { ((TextView) findViewById(R.id.delete)).setText(text); } // 將view加入到ViewContent中 public void setContentView(View view) { mViewContent.addView(view); } // 設(shè)置滑動(dòng)回調(diào) public void setOnSlideListener(OnSlideListener onSlideListener) { mOnSlideListener = onSlideListener; } // 將當(dāng)前狀態(tài)置為關(guān)閉 public void shrink() { if (getScrollX() != 0) { this.smoothScrollTo(0, 0); } } // 根據(jù)MotionEvent來進(jìn)行滑動(dòng),這個(gè)方法的作用相當(dāng)于onTouchEvent // 如果你不需要處理滑動(dòng)沖突,可以直接重命名,照樣能正常工作 public void onRequireTouchEvent(MotionEvent event) { int x = (int) event.getX(); int y = (int) event.getY(); int scrollX = getScrollX(); Log.d(TAG, "x=" + x + " y=" + y); switch (event.getAction()) { case MotionEvent.ACTION_DOWN: { if (!mScroller.isFinished()) { mScroller.abortAnimation(); } if (mOnSlideListener != null) { mOnSlideListener.onSlide(this, OnSlideListener.SLIDE_STATUS_START_SCROLL); } break; } case MotionEvent.ACTION_MOVE: { int deltaX = x - mLastX; int deltaY = y - mLastY; if (Math.abs(deltaX) < Math.abs(deltaY) * TAN) { // 滑動(dòng)不滿足條件,不做橫向滑動(dòng) break; } // 計(jì)算滑動(dòng)終點(diǎn)是否合法,防止滑動(dòng)越界 int newScrollX = scrollX - deltaX; if (deltaX != 0) { if (newScrollX < 0) { newScrollX = 0; } else if (newScrollX > mHolderWidth) { newScrollX = mHolderWidth; } this.scrollTo(newScrollX, 0); } break; } case MotionEvent.ACTION_UP: { int newScrollX = 0; // 這里做了下判斷,當(dāng)松開手的時(shí)候,會(huì)自動(dòng)向兩邊滑動(dòng),具體向哪邊滑,要看當(dāng)前所處的位置 if (scrollX - mHolderWidth * 0.75 > 0) { newScrollX = mHolderWidth; } // 慢慢滑向終點(diǎn) this.smoothScrollTo(newScrollX, 0); // 通知上層滑動(dòng)事件 if (mOnSlideListener != null) { mOnSlideListener.onSlide(this, newScrollX == 0 ? OnSlideListener.SLIDE_STATUS_OFF : OnSlideListener.SLIDE_STATUS_ON); } break; } default: break; } mLastX = x; mLastY = y; } private void smoothScrollTo(int destX, int destY) { // 緩慢滾動(dòng)到指定位置 int scrollX = getScrollX(); int delta = destX - scrollX; // 以三倍時(shí)長滑向destX,效果就是慢慢滑動(dòng) mScroller.startScroll(scrollX, 0, delta, 0, Math.abs(delta) * 3); invalidate(); } @Override public void computeScroll() { if (mScroller.computeScrollOffset()) { scrollTo(mScroller.getCurrX(), mScroller.getCurrY()); postInvalidate(); } } }
自定義SlideView所關(guān)聯(lián)的xml.(主要就是布局側(cè)滑顯出來的刪除按鈕)
<?xml version="1.0" encoding="utf-8"?> <merge xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" > <LinearLayout android:id="@+id/view_content" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="horizontal" > </LinearLayout> <RelativeLayout android:id="@+id/holder" android:layout_width="120dp" android:layout_height="match_parent" android:clickable="true" android:background="@drawable/holder_bg"> <TextView android:id="@+id/delete" android:layout_width="wrap_content" android:layout_height="wrap_content" android:drawableLeft="@drawable/del_icon_normal" android:layout_centerInParent="true" android:gravity="center" android:textColor="@color/floralwhite" android:text="刪除" /> </RelativeLayout> </merge>
自定義ListViewCompat繼承Listview。
import com.ryg.slideview.MainActivity.MessageItem; import android.content.Context; import android.util.AttributeSet; import android.util.Log; import android.view.MotionEvent; import android.view.View; import android.widget.ListView; public class ListViewCompat extends ListView { private static final String TAG = "ListViewCompat"; private SlideView mFocusedItemView; public ListViewCompat(Context context) { super(context); } public ListViewCompat(Context context, AttributeSet attrs) { super(context, attrs); } public ListViewCompat(Context context, AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); } public void shrinkListItem(int position) { View item = getChildAt(position); if (item != null) { try { ((SlideView) item).shrink(); } catch (ClassCastException e) { e.printStackTrace(); } } } @Override public boolean onTouchEvent(MotionEvent event) { switch (event.getAction()) { case MotionEvent.ACTION_DOWN: { int x = (int) event.getX(); int y = (int) event.getY(); //根據(jù)坐標(biāo)獲取item所在的行 int position = pointToPosition(x, y); Log.e(TAG, "postion=" + position); if (position != INVALID_POSITION) { //得到當(dāng)前點(diǎn)擊行的數(shù)據(jù)從而取出當(dāng)前行的item。 //可能有人懷疑,為什么要這么干?為什么不用getChildAt(position)? //因?yàn)長istView會(huì)進(jìn)行緩存,如果你不這么干,有些行的view你是得不到的。 MessageItem data = (MessageItem) getItemAtPosition(position); mFocusedItemView = data.slideView; Log.e(TAG, "FocusedItemView=" + mFocusedItemView); } } default: break; } //向當(dāng)前點(diǎn)擊的view發(fā)送滑動(dòng)事件請(qǐng)求,其實(shí)就是向SlideView發(fā)請(qǐng)求 if (mFocusedItemView != null) { mFocusedItemView.onRequireTouchEvent(event); } return super.onTouchEvent(event); } }
接下來就是主界面的activity了
import java.util.ArrayList; import java.util.List; import com.ryg.slideview.SlideView.OnSlideListener; import android.app.Activity; import android.os.Bundle; import android.util.Log; import android.view.LayoutInflater; import android.view.View; import android.view.View.OnClickListener; import android.view.ViewGroup; import android.widget.AdapterView; import android.widget.AdapterView.OnItemClickListener; import android.widget.BaseAdapter; import android.widget.ImageView; import android.widget.ListView; import android.widget.TextView; public class MainActivity extends Activity implements OnItemClickListener, OnClickListener, OnSlideListener { private static final String TAG = "MainActivity"; private ListViewCompat mListView; private List<MessageItem> mMessageItems = new ArrayList<MainActivity.MessageItem>(); private SlideAdapter mSlideAdapter; // 上次處于打開狀態(tài)的SlideView private SlideView mLastSlideViewWithStatusOn; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); initView(); } private void initView() { mListView = (ListViewCompat) findViewById(R.id.list); for (int i = 0; i < 20; i++) { MessageItem item = new MessageItem(); if (i % 2 == 0) { item.iconRes = R.drawable.default_qq_avatar; item.title = "騰訊新聞"; item.msg = "天津大爆炸:河北大爆炸"; item.time = "晚上18:00"; } else { item.iconRes = R.drawable.wechat_icon; item.title = "微信團(tuán)隊(duì)"; item.msg = "歡迎你使用微信"; item.time = "10月01日"; } mMessageItems.add(item); } mSlideAdapter = new SlideAdapter(); mListView.setAdapter(mSlideAdapter); mListView.setOnItemClickListener(this); } //listview 的適配器 private class SlideAdapter extends BaseAdapter { private LayoutInflater mInflater; SlideAdapter() { super(); mInflater = getLayoutInflater(); } @Override public int getCount() { return mMessageItems.size(); } @Override public Object getItem(int position) { return mMessageItems.get(position); } @Override public long getItemId(int position) { return position; } @Override public View getView(int position, View convertView, ViewGroup parent) { ViewHolder holder; SlideView slideView = (SlideView) convertView; if (slideView == null) { // 這里是我們的item View itemView = mInflater.inflate(R.layout.list_item, null); slideView = new SlideView(MainActivity.this); // 這里把item加入到slideView slideView.setContentView(itemView); // 下面是做一些數(shù)據(jù)緩存 holder = new ViewHolder(slideView); slideView.setOnSlideListener(MainActivity.this); slideView.setTag(holder); } else { holder = (ViewHolder) slideView.getTag(); } MessageItem item = mMessageItems.get(position); item.slideView = slideView; item.slideView.shrink(); holder.icon.setImageResource(item.iconRes); holder.title.setText(item.title); holder.msg.setText(item.msg); holder.time.setText(item.time); holder.deleteHolder.setOnClickListener(MainActivity.this); return slideView; } } public class MessageItem { public int iconRes; public String title; public String msg; public String time; public SlideView slideView; } private static class ViewHolder { public ImageView icon; public TextView title; public TextView msg; public TextView time; public ViewGroup deleteHolder; ViewHolder(View view) { icon = (ImageView) view.findViewById(R.id.icon); title = (TextView) view.findViewById(R.id.title); msg = (TextView) view.findViewById(R.id.msg); time = (TextView) view.findViewById(R.id.time); deleteHolder = (ViewGroup) view.findViewById(R.id.holder); } } @Override public void onItemClick(AdapterView<?> parent, View view, int position, long id) { // 這里處理ListItem的點(diǎn)擊事件 Log.e(TAG, "onItemClick position=" + position); } @Override public void onSlide(View view, int status) { // 如果當(dāng)前存在已經(jīng)打開的SlideView,那么將其關(guān)閉 if (mLastSlideViewWithStatusOn != null && mLastSlideViewWithStatusOn != view) { mLastSlideViewWithStatusOn.shrink(); } // 記錄本次處于打開狀態(tài)的view if (status == SLIDE_STATUS_ON) { mLastSlideViewWithStatusOn = (SlideView) view; } } @Override public void onClick(View v) { // 這里處理刪除按鈕的點(diǎn)擊事件,可以刪除對(duì)話 if (v.getId() == R.id.holder) { int position = mListView.getPositionForView(v); if (position != ListView.INVALID_POSITION) { mMessageItems.remove(position); mSlideAdapter.notifyDataSetChanged(); } Log.e(TAG, "onClick v=" + v); } }
主界面中activity用到的xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" > <com.ryg.slideview.ListViewCompat android:id="@+id/list" android:layout_width="match_parent" android:layout_height="match_parent" android:background="#fff4f7f9" android:cacheColorHint="#00000000" android:divider="#dddbdb" android:dividerHeight="1.0px" android:drawSelectorOnTop="false" android:listSelector="@android:color/transparent" android:scrollbars="none" /> </LinearLayout>
listview適配器的布局。
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="58dp" android:background="@drawable/list_item_bg" android:descendantFocusability="blocksDescendants" android:gravity="center_vertical" android:paddingBottom="5dp" android:paddingLeft="10dp" android:paddingRight="10dp" android:paddingTop="5dp" > <ImageView android:id="@+id/icon" android:layout_width="50dp" android:layout_height="50dp" android:layout_marginRight="5dp" android:src="@drawable/wechat_icon" /> <TextView android:id="@+id/title" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_toRightOf="@id/icon" android:textColor="@color/black" android:textSize="15sp" /> <TextView android:id="@+id/msg" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignBottom="@id/icon" android:layout_alignLeft="@id/title" android:textColor="@color/grey" /> <TextView android:id="@+id/time" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentRight="true" android:layout_alignTop="@id/title" android:textColor="@color/grey" /> </RelativeLayout>
附drawable目錄下的xml文件
holder_bg.xml
<?xml version="1.0" encoding="utf-8"?> <selector xmlns:android="http://schemas.android.com/apk/res/android"> <item android:drawable="@color/dimgrey" android:state_pressed="true"></item> <item android:drawable="@color/dimgrey" android:state_focused="true"></item> <item android:drawable="@color/dimgrey" android:state_selected="true"></item> <item android:drawable="@color/gray" android:state_enabled="true"></item> <item android:drawable="@color/gray" android:state_enabled="false"></item> </selector>
list_item_bg.xml.
<?xml version="1.0" encoding="utf-8"?> <selector xmlns:android="http://schemas.android.com/apk/res/android"> <item android:drawable="@drawable/item_point_bg" android:state_pressed="true"></item> <item android:drawable="@drawable/item_point_bg" android:state_focused="true"></item> <item android:drawable="@drawable/item_point_bg" android:state_selected="true"></item> <item android:drawable="@android:color/transparent" android:state_enabled="true"></item> <item android:drawable="@android:color/transparent" android:state_enabled="false"></item> </selector>
還有一些對(duì)color顏色的定義,這里就不貼了,大家可以直接換成顏色的代碼就可以。
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- 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 模仿QQ側(cè)滑刪除ListView功能示例
- android的RecyclerView實(shí)現(xiàn)拖拽排序和側(cè)滑刪除示例
- Android使用Item Swipemenulistview實(shí)現(xiàn)仿QQ側(cè)滑刪除功能
- android基于SwipeRefreshLayout實(shí)現(xiàn)類QQ的側(cè)滑刪除
- Android實(shí)現(xiàn)微信側(cè)滑刪除當(dāng)前頁面
- Android編程實(shí)現(xiàn)列表側(cè)滑刪除的方法詳解
相關(guān)文章
HandlerThread的使用場(chǎng)景和用法詳解
這篇文章主要介紹了HandlerThread的使用場(chǎng)景和用法詳解,HandlerThread是Android中的一個(gè)線程類,它是Thread的子類,并且內(nèi)部封裝了Looper和Handler,提供了更方便的消息處理和線程操作,需要的朋友可以參考下2023-07-07Android?OpenCV基礎(chǔ)API清晰度亮度識(shí)別檢測(cè)
這篇文章主要為大家介紹了Android?OpenCV基礎(chǔ)API清晰度亮度識(shí)別檢測(cè),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01Kotlin協(xié)程launch啟動(dòng)流程原理詳解
這篇文章主要為大家介紹了Kotlin協(xié)程launch啟動(dòng)流程原理詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-12-12Android基于IJKPlayer視頻播放器簡(jiǎn)單封裝設(shè)計(jì)
這篇文章主要介紹了Android基于IJKPlayer視頻播放器簡(jiǎn)單封裝設(shè)計(jì),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-06-06Android webview旋轉(zhuǎn)屏幕導(dǎo)致頁面重新加載問題解決辦法
這篇文章主要介紹了Android webview旋轉(zhuǎn)屏幕導(dǎo)致頁面重新加載問題解決辦法的相關(guān)資料,希望通過本文能幫助到大家實(shí)現(xiàn)這樣的問題,需要的朋友可以參考下2017-10-10android 關(guān)于利用簽名的SHA1進(jìn)行安全校驗(yàn)的方法之一(推薦)
下面小編就為大家?guī)硪黄猘ndroid 關(guān)于利用簽名的SHA1進(jìn)行安全校驗(yàn)的方法之一(推薦)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-01-01Android使用ContentProvider實(shí)現(xiàn)跨進(jìn)程通訊示例詳解
這篇文章主要為大家介紹了Android使用ContentProvider實(shí)現(xiàn)跨進(jìn)程通訊示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-03-03Android中CountDownTimer倒計(jì)時(shí)器用法實(shí)例
這篇文章主要介紹了Android中CountDownTimer倒計(jì)時(shí)器用法,以實(shí)例形式分析了Android中CountDownTimer類的相關(guān)使用技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-10-10Kotlin HttpURLConnection與服務(wù)器交互實(shí)現(xiàn)方法詳解
簡(jiǎn)單來說,HttpURLConnection 是發(fā)起HTTP請(qǐng)求的基礎(chǔ)類庫,提供了HTTP請(qǐng)求的基本功能,不過封裝的比較少,在使用時(shí)很多內(nèi)容都需要自己設(shè)置,也需要自己處理請(qǐng)求流和響應(yīng)流2022-09-09