Android高仿QQ6.0側(cè)滑刪除實(shí)例代碼
推薦閱讀:
先給大家分享一下,側(cè)滑刪除,布局也就是前面一個(gè)item,然后有兩個(gè)隱藏的按鈕(TextView也可以),然后我們可以向左側(cè)滑動(dòng),然后顯示出來(lái),然后對(duì)delete(刪除鍵)實(shí)現(xiàn)監(jiān)聽(tīng),就可以了哈。好了那就來(lái)看看代碼怎么實(shí)現(xiàn)的吧。
首先和之前一樣
自定義View,初始化ViewDragHelper:
package com.example.removesidepull; import android.content.Context; import android.support.v4.widget.ViewDragHelper; import android.util.AttributeSet; import android.view.MotionEvent; import android.view.View; import android.widget.FrameLayout; /** * Created by 若蘭 on 2016/2/2. * 一個(gè)懂得了編程樂(lè)趣的小白,希望自己 * 能夠在這個(gè)道路上走的很遠(yuǎn),也希望自己學(xué)習(xí)到的 * 知識(shí)可以幫助更多的人,分享就是學(xué)習(xí)的一種樂(lè)趣 * QQ:1069584784 * csdn:http://blog.csdn.net/wuyinlei */ public class SwipeLayout extends FrameLayout { private ViewDragHelper mDragHelper; 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); //第一步 初始化ViewDragHelper mDragHelper = ViewDragHelper.create(this, mCallback); } ViewDragHelper.Callback mCallback = new ViewDragHelper.Callback() { @Override public boolean tryCaptureView(View child, int pointerId) { //返回true return true; } }; }
然后我們就要去處理攔截事件也就是重寫(xiě)一些onInterceptTouchEvent和onTouchEvent方法,默認(rèn)是不攔截的:
/** * 傳遞觸摸事件 * * @param ev * @return */ @Override public boolean onInterceptTouchEvent(MotionEvent ev) { //交給ViewDragHelper判斷是否去攔截事件 return mDragHelper.shouldInterceptTouchEvent(ev); } @Override public boolean onTouchEvent(MotionEvent event) { try { mDragHelper.processTouchEvent(event); } catch (Exception e) { e.printStackTrace(); } //返回true,這里表示去攔截事件 return true; }
然后我們?nèi)ブ貙?xiě)一下ViewDragHelper里面的clampViewPositionHorizontal方法:
@Override public int clampViewPositionHorizontal(View child, int left, int dx) { return left; }
好了這個(gè)時(shí)候,就已經(jīng)可以實(shí)現(xiàn)滑動(dòng)了,我們先來(lái)看下結(jié)果:
這里我們可以看到,已經(jīng)可以滑動(dòng)了,好了接下來(lái)的就是要處理滑動(dòng)事件,去放置到正確的地方(call me 和刪除剛開(kāi)始不能見(jiàn),還有只能左滑顯示,右滑隱藏)。
好了,我們先獲取兩個(gè)View吧:
/** * 當(dāng)xml填充完畢的時(shí)候 */ @Override protected void onFinishInflate() { super.onFinishInflate(); /** * 后view */ mBackView = getChildAt(0); /** * 前view */ mFrontView = getChildAt(1); }
獲取想要的寬和高:
/** * 在這里獲取寬和高 * * @param w * @param h * @param oldw * @param oldh */ @Override protected void onSizeChanged(int w, int h, int oldw, int oldh) { super.onSizeChanged(w, h, oldw, oldh); /** * 高度 */ mHeight = mFrontView.getMeasuredHeight(); /** * 寬度 */ mWidth = mFrontView.getMeasuredWidth(); /** * 移動(dòng)距離 */ mRange = mBackView.getMeasuredWidth(); }
擺放這兩個(gè)view的位置:
/** * 擺放位置 * @param changed * @param left * @param top * @param right * @param bottom */ @Override protected void onLayout(boolean changed, int left, int top, int right, int bottom) { super.onLayout(changed, left, top, right, bottom); layoutContent(false); } private void layoutContent(boolean isOpen) { //擺放前view Rect frontRect = computeFrontViewRect(isOpen); mFrontView.layout(frontRect.left, frontRect.top, frontRect.right, frontRect.bottom); //擺放后view Rect backRect = computeBackViewRect(frontRect); mBackView.layout(backRect.left,backRect.top,backRect.right,backRect.bottom); //前置前view bringChildToFront(mFrontView); } /** * 我們可以把前view相當(dāng)于一個(gè)矩形 * * @param frontRect * @return */ private Rect computeBackViewRect(Rect frontRect) { int left = frontRect.right; return new Rect(left, 0, left + mRange, 0 + mHeight); } private Rect computeFrontViewRect(boolean isOpen) { int left = 0; if (isOpen) { left = -mRange; } return new Rect(left, 0, left + mWidth, 0 + mHeight); }
當(dāng)然這個(gè)實(shí)現(xiàn),只是可以拖拽了前view,因?yàn)槲覀儧](méi)有把改變的dx傳遞下去,好了來(lái)實(shí)現(xiàn)拖拽前view的時(shí)候,后view也跟著出來(lái)(ViewDragHelper里面的方法):
/** * 當(dāng)view位置改變的時(shí)候 * @param changedView 改變的view * @param left * @param top * @param dx x軸偏移量 * @param dy */ @Override public void onViewPositionChanged(View changedView, int left, int top, int dx, int dy) { super.onViewPositionChanged(changedView, left, top, dx, dy); //傳遞事件,如果是拖拽的前view, if (changedView == mFrontView){ //Offset this view's horizontal location by the specified amount of pixels. //也就是說(shuō)我的我的前view左滑了dx,那么我的后view也是左滑dx,右滑同理 mBackView.offsetLeftAndRight(dx); } else if (changedView == mBackView){ //拖拽的是后view的話,前View的處理方式一樣 mFrontView.offsetLeftAndRight(dx); } //兼容老版本 invalidate(); }
好了這個(gè)時(shí)候我們來(lái)看下效果:
是不是發(fā)現(xiàn)了問(wèn)題,就是我的前view想要的結(jié)果是不能右滑的(只允許左滑和返回),那么接下來(lái)就實(shí)現(xiàn)這個(gè)想要的結(jié)果吧。以下的代碼是在clampViewPositionHorizontal()方法里面:
//在這里處理放置的邏輯拖拽的前view if (child == mFrontView) { if (left > 0) { return 0; } else if (left < -mRange) { return -mRange; } }//拖拽的后view else if (child == mBackView) { if (left > mWidth) { return mWidth; } else if (left < mWidth - mRange) { return mWidth - mRange; } }
看下效果圖:
好了,這個(gè)時(shí)候已經(jīng)基本實(shí)現(xiàn)了,接下來(lái)實(shí)現(xiàn)以下滑動(dòng)的距離和速度【判斷是否打開(kāi)和關(guān)閉:
/** * 拖拽的view釋放的時(shí)候 * * @param releasedChild * @param xvel * @param yvel */ @Override public void onViewReleased(View releasedChild, float xvel, float yvel) { if (xvel == 0 && mFrontView.getLeft() < -mRange / 2.0f) { open(); } else if (xvel < 0) { open(); } else { close(); } } /** * 關(guān)閉 */ public void close() { Utils.showToast(getContext(), "close"); layoutContent(false); } //打開(kāi) public void open() { //Utils.showToast(getContext(), "open"); layoutContent(true); }
好了,接下來(lái)實(shí)現(xiàn)以下平滑的關(guān)閉和打開(kāi):
public void close() { close(true); } /** * 關(guān)閉 * * @param isSmooth */ public void close(boolean isSmooth) { int finalLeft = 0; if (isSmooth) { //開(kāi)始動(dòng)畫(huà) 如果返回true表示沒(méi)有完成動(dòng)畫(huà) if (mDragHelper.smoothSlideViewTo(mFrontView, finalLeft, 0)) { ViewCompat.postInvalidateOnAnimation(this); } } else { layoutContent(false); } } public void open() { open(true); } /** * 打開(kāi) * * @param isSmooth */ public void open(boolean isSmooth) { int finalLeft = -mRange; if (isSmooth) { //開(kāi)始動(dòng)畫(huà) if (mDragHelper.smoothSlideViewTo(mFrontView, finalLeft, 0)) { ViewCompat.postInvalidateOnAnimation(this); } } else { layoutContent(true); } } /** * 持續(xù)動(dòng)畫(huà) */ @Override public void computeScroll() { super.computeScroll(); //這個(gè)是固定的 if (mDragHelper.continueSettling(true)) { ViewCompat.postInvalidateOnAnimation(this); } }
我們看下最終的效果吧:
好了,在這里我們加上一些回調(diào),以方便外部使用的時(shí)候可以回調(diào):
/** * 默認(rèn)狀態(tài)是關(guān)閉 */ private Status status = Status.Close; private OnSwipeLayoutListener swipeLayoutListener; public Status getStatus() { return status; } public void setStatus(Status status) { this.status = status; } public OnSwipeLayoutListener getSwipeLayoutListener() { return swipeLayoutListener; } public void setSwipeLayoutListener(OnSwipeLayoutListener swipeLayoutListener) { this.swipeLayoutListener = swipeLayoutListener; } /** * 定義三種狀態(tài) */ public enum Status { Close, Open, Draging } /** * 定義回調(diào)接口 這個(gè)在我們 */ public interface OnSwipeLayoutListener { /** * 關(guān)閉 * * @param mSwipeLayout */ void onClose(SwipeLayout mSwipeLayout); /** * 打開(kāi) * * @param mSwipeLayout */ void onOpen(SwipeLayout mSwipeLayout); /** * 繪制 * * @param mSwipeLayout */ void onDraging(SwipeLayout mSwipeLayout); /** * 要去關(guān)閉 */ void onStartClose(SwipeLayout mSwipeLayout); /** * 要去開(kāi)啟 */ void onStartOpen(SwipeLayout mSwipeLayout); }
dispatchSwipeEvent()方法(在onViewPositionChanged()方法中調(diào)用)
protected void dispatchSwipeEvent() { //判斷是否為空 if (swipeLayoutListener != null) { swipeLayoutListener.onDraging(this); } // 記錄上一次的狀態(tài) Status preStatus = status; // 更新當(dāng)前狀態(tài) status = updateStatus(); if (preStatus != status && swipeLayoutListener != null) { if (status == Status.Close) { swipeLayoutListener.onClose(this); } else if (status == Status.Open) { swipeLayoutListener.onOpen(this); } else if (status == Status.Draging) { if (preStatus == Status.Close) { swipeLayoutListener.onStartOpen(this); } else if (preStatus == Status.Open) { swipeLayoutListener.onStartClose(this); } } } }
updateStatus()方法:
/** * 更新?tīng)顟B(tài) * * @return */ private Status updateStatus() { //得到前view的左邊位置 int left = mFrontView.getLeft(); if (left == 0) { //如果位置是0,就是關(guān)閉狀態(tài) return Status.Close; } else if (left == -mRange) { //如果左側(cè)邊距是后view的寬度的負(fù)值,狀態(tài)為開(kāi) return Status.Open; } //其他狀態(tài)就是拖拽 return Status.Draging; }
好了,事件基本上已經(jīng)實(shí)現(xiàn)完畢了,這個(gè)側(cè)拉刪除的我會(huì)更新至我的項(xiàng)目中,同時(shí)希望Android高仿QQ6.0側(cè)滑刪除實(shí)例代碼對(duì)大家有所幫助。
- 詳解Android中實(shí)現(xiàn)ListView左右滑動(dòng)刪除條目的方法
- android RecyclerView側(cè)滑菜單,滑動(dòng)刪除,長(zhǎng)按拖拽,下拉刷新上拉加載
- Android仿微信滑動(dòng)彈出編輯、刪除菜單效果、增加下拉刷新功能
- Android實(shí)現(xiàn)左滑刪除列表功能
- Android 實(shí)現(xiàn)左滑出現(xiàn)刪除選項(xiàng)
- Android仿QQ列表左滑刪除操作
- Android使用CardView作為RecyclerView的Item并實(shí)現(xiàn)拖拽和左滑刪除
- Android 滑動(dòng)監(jiān)聽(tīng)RecyclerView線性流+左右劃刪除+上下移動(dòng)
- Android開(kāi)發(fā)中模仿qq列表信息滑動(dòng)刪除功能
- Android?Recyclerview實(shí)現(xiàn)左滑刪除功能
相關(guān)文章
詳解Android實(shí)現(xiàn)購(gòu)物車(chē)頁(yè)面及購(gòu)物車(chē)效果(點(diǎn)擊動(dòng)畫(huà))
本篇文章主要介紹了詳解Android實(shí)現(xiàn)購(gòu)物車(chē)頁(yè)面及購(gòu)物車(chē)效果(點(diǎn)擊動(dòng)畫(huà)),非常具有實(shí)用價(jià)值,需要的朋友可以參考下2017-08-08Android sdcard實(shí)現(xiàn)圖片存儲(chǔ) 、聯(lián)網(wǎng)下載
這篇文章主要介紹了Android sdcard實(shí)現(xiàn)圖片存儲(chǔ) 、聯(lián)網(wǎng)下載功能,感興趣的小伙伴們可以參考一下2016-02-02Android利用Gson解析嵌套多層的Json的簡(jiǎn)單方法
下面小編就為大家?guī)?lái)一篇Android利用Gson解析嵌套多層的Json的簡(jiǎn)單方法。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-08-08Android開(kāi)發(fā)之設(shè)置開(kāi)機(jī)自動(dòng)啟動(dòng)的幾種方法
這篇文章主要介紹了Android開(kāi)發(fā)之設(shè)置開(kāi)機(jī)自動(dòng)啟動(dòng)的幾種方法的相關(guān)資料,這里提供三種方法幫助大家實(shí)現(xiàn)這樣的功能,需要的朋友可以參考下2017-08-08Android百度地圖添加Marker失真問(wèn)題的解決方案
本篇文章主要介紹了Android百度地圖添加Marker失真問(wèn)題的解決方案,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-01-01Android高仿抖音照片電影功能的實(shí)現(xiàn)代碼
這篇文章主要介紹了Android高仿抖音照片電影功能的實(shí)現(xiàn)代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-09-09Android 媒體開(kāi)發(fā)之MediaPlayer狀態(tài)機(jī)接口方法實(shí)例解析
這篇文章主要介紹了Android 媒體開(kāi)發(fā)之MediaPlayer狀態(tài)機(jī)接口方法實(shí)例解析,需要的朋友可以參考下2017-08-08Android如何使用GestureDetector進(jìn)行手勢(shì)檢測(cè)詳解
GestureDetector使用很方便,提供了單擊,雙擊,長(zhǎng)按等操作的處理,但是一般的定義界面都比較復(fù)雜,還用很多需要注意的地方,這篇文章主要給大家介紹了關(guān)于Android如何使用GestureDetector進(jìn)行手勢(shì)檢測(cè)的相關(guān)資料,需要的朋友可以參考下2022-01-01