Android高仿QQ6.0側(cè)滑刪除實例代碼
推薦閱讀:
先給大家分享一下,側(cè)滑刪除,布局也就是前面一個item,然后有兩個隱藏的按鈕(TextView也可以),然后我們可以向左側(cè)滑動,然后顯示出來,然后對delete(刪除鍵)實現(xiàn)監(jiān)聽,就可以了哈。好了那就來看看代碼怎么實現(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.
* 一個懂得了編程樂趣的小白,希望自己
* 能夠在這個道路上走的很遠,也希望自己學(xué)習(xí)到的
* 知識可以幫助更多的人,分享就是學(xué)習(xí)的一種樂趣
* 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;
}
};
}
然后我們就要去處理攔截事件也就是重寫一些onInterceptTouchEvent和onTouchEvent方法,默認是不攔截的:
/**
* 傳遞觸摸事件
*
* @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)ブ貙懸幌耉iewDragHelper里面的clampViewPositionHorizontal方法:
@Override
public int clampViewPositionHorizontal(View child, int left, int dx) {
return left;
}
好了這個時候,就已經(jīng)可以實現(xiàn)滑動了,我們先來看下結(jié)果:

這里我們可以看到,已經(jīng)可以滑動了,好了接下來的就是要處理滑動事件,去放置到正確的地方(call me 和刪除剛開始不能見,還有只能左滑顯示,右滑隱藏)。
好了,我們先獲取兩個View吧:
/**
* 當(dāng)xml填充完畢的時候
*/
@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();
/**
* 移動距離
*/
mRange = mBackView.getMeasuredWidth();
}
擺放這兩個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)于一個矩形
*
* @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)然這個實現(xiàn),只是可以拖拽了前view,因為我們沒有把改變的dx傳遞下去,好了來實現(xiàn)拖拽前view的時候,后view也跟著出來(ViewDragHelper里面的方法):
/**
* 當(dāng)view位置改變的時候
* @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.
//也就是說我的我的前view左滑了dx,那么我的后view也是左滑dx,右滑同理
mBackView.offsetLeftAndRight(dx);
} else if (changedView == mBackView){
//拖拽的是后view的話,前View的處理方式一樣
mFrontView.offsetLeftAndRight(dx);
}
//兼容老版本
invalidate();
}
好了這個時候我們來看下效果:

是不是發(fā)現(xiàn)了問題,就是我的前view想要的結(jié)果是不能右滑的(只允許左滑和返回),那么接下來就實現(xiàn)這個想要的結(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;
}
}
看下效果圖:
好了,這個時候已經(jīng)基本實現(xiàn)了,接下來實現(xiàn)以下滑動的距離和速度【判斷是否打開和關(guān)閉:
/**
* 拖拽的view釋放的時候
*
* @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);
}
//打開
public void open() {
//Utils.showToast(getContext(), "open");
layoutContent(true);
}
好了,接下來實現(xiàn)以下平滑的關(guān)閉和打開:
public void close() {
close(true);
}
/**
* 關(guān)閉
*
* @param isSmooth
*/
public void close(boolean isSmooth) {
int finalLeft = 0;
if (isSmooth) {
//開始動畫 如果返回true表示沒有完成動畫
if (mDragHelper.smoothSlideViewTo(mFrontView, finalLeft, 0)) {
ViewCompat.postInvalidateOnAnimation(this);
}
} else {
layoutContent(false);
}
}
public void open() {
open(true);
}
/**
* 打開
*
* @param isSmooth
*/
public void open(boolean isSmooth) {
int finalLeft = -mRange;
if (isSmooth) {
//開始動畫
if (mDragHelper.smoothSlideViewTo(mFrontView, finalLeft, 0)) {
ViewCompat.postInvalidateOnAnimation(this);
}
} else {
layoutContent(true);
}
}
/**
* 持續(xù)動畫
*/
@Override
public void computeScroll() {
super.computeScroll();
//這個是固定的
if (mDragHelper.continueSettling(true)) {
ViewCompat.postInvalidateOnAnimation(this);
}
}
我們看下最終的效果吧:

好了,在這里我們加上一些回調(diào),以方便外部使用的時候可以回調(diào):
/**
* 默認狀態(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)接口 這個在我們
*/
public interface OnSwipeLayoutListener {
/**
* 關(guān)閉
*
* @param mSwipeLayout
*/
void onClose(SwipeLayout mSwipeLayout);
/**
* 打開
*
* @param mSwipeLayout
*/
void onOpen(SwipeLayout mSwipeLayout);
/**
* 繪制
*
* @param mSwipeLayout
*/
void onDraging(SwipeLayout mSwipeLayout);
/**
* 要去關(guān)閉
*/
void onStartClose(SwipeLayout mSwipeLayout);
/**
* 要去開啟
*/
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à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的寬度的負值,狀態(tài)為開
return Status.Open;
}
//其他狀態(tài)就是拖拽
return Status.Draging;
}
好了,事件基本上已經(jīng)實現(xiàn)完畢了,這個側(cè)拉刪除的我會更新至我的項目中,同時希望Android高仿QQ6.0側(cè)滑刪除實例代碼對大家有所幫助。
- 詳解Android中實現(xiàn)ListView左右滑動刪除條目的方法
- android RecyclerView側(cè)滑菜單,滑動刪除,長按拖拽,下拉刷新上拉加載
- Android仿微信滑動彈出編輯、刪除菜單效果、增加下拉刷新功能
- Android實現(xiàn)左滑刪除列表功能
- Android 實現(xiàn)左滑出現(xiàn)刪除選項
- Android仿QQ列表左滑刪除操作
- Android使用CardView作為RecyclerView的Item并實現(xiàn)拖拽和左滑刪除
- Android 滑動監(jiān)聽RecyclerView線性流+左右劃刪除+上下移動
- Android開發(fā)中模仿qq列表信息滑動刪除功能
- Android?Recyclerview實現(xiàn)左滑刪除功能
相關(guān)文章
Android開發(fā)實現(xiàn)SubMenu選項菜單和子菜單示例
這篇文章主要介紹了Android開發(fā)實現(xiàn)SubMenu選項菜單和子菜單,結(jié)合實例形式分析了Android開發(fā)中SubMenu選項菜單和子菜單的功能、配置、布局等相關(guān)操作技巧,需要的朋友可以參考下2019-03-03
如何通過Battery Historian分析Android APP耗電情況
Android 從兩個層面統(tǒng)計電量的消耗,分別為軟件排行榜及硬件排行榜。它們各有自己的耗電榜單,軟件排行榜為機器中每個 App 的耗電榜單,硬件排行榜則為各個硬件的耗電榜單。這兩個排行榜的統(tǒng)計是互為獨立,互不干擾的2021-06-06
android 如何設(shè)置開機后屏幕亮度默認值為自動調(diào)節(jié)
在第一次開機后,設(shè)置>顯示>自動亮度調(diào)節(jié) 默認是勾選上的,具體修改方法如下,感興趣的朋友可以嘗試操作下2013-06-06
Android 自定義標(biāo)題欄 顯示網(wǎng)頁加載進度的方法實例
Android 自定義標(biāo)題欄 顯示網(wǎng)頁加載進度的方法實例,需要的朋友可以參考一下2013-06-06
Android listview定位到上次顯示的位置的實現(xiàn)方法
這篇文章主要介紹了Android listview定位到上次顯示的位置的實現(xiàn)方法的相關(guān)資料,希望通過本文能幫助到大家,需要的朋友可以參考下2017-08-08
Android7.0以上Uri轉(zhuǎn)路徑的方法實現(xiàn)(已驗證)
這篇文章主要介紹了Android7.0以上Uri轉(zhuǎn)路徑的方法實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-03-03
android如何獲取手機聯(lián)系人的數(shù)據(jù)庫示例代碼
很多人在做手機聯(lián)系人的apk時會遇到,如何獲取手機聯(lián)系人數(shù)據(jù)庫的問題,本篇文章主要介紹了android如何獲取手機聯(lián)系人的數(shù)據(jù)庫示例代碼,有興趣的可以了解一下。2017-01-01

