實現(xiàn)Activity滑動退出

實現(xiàn)Activity滑動退出

很多應用在二級詳情頁面加入了滑動退出activity的效果宰闰,很方便茬贵,心血來潮,想著自己也來實現(xiàn)這個效果移袍,就當做練手吧解藻。

實現(xiàn)View的滑動有很多種方法,如自己在onTouchEvent中處理觸摸事件葡盗,然后滾動View到相應位置螟左,也可以用google V4包為我們提供的ViewDragHelper來處理觸摸事件,我們這里選擇后者觅够,因為滑動退出操作都是在屏幕的邊緣時觸發(fā)胶背,而ViewDragHelper剛好提供了想要的實現(xiàn),可以說利用ViewDragHelper來實現(xiàn)我們的需求非常簡單喘先。

先定義一個ViewGroup钳吟,并做一些必要的變量聲明
BaseSwipeLayout

  public class BaseSwipeLayout extends FrameLayout{

    private View mDragView;

    private ViewDragHelper mViewDragHelper;

    private Point mAutoBackOrignalPoint = new Point();

    private Point mCurArrivePoint = new Point();

    private int mCurEdgeFlag = ViewDragHelper.EDGE_LEFT;
    private int mSwipeEdge = ViewDragHelper.EDGE_LEFT;

    public BaseSwipeLayout(Context context) {
        this(context, null);
    }

    public BaseSwipeLayout(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public BaseSwipeLayout(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        init();
    }
 }

既然使用ViewDragHelper,我們把觸摸事件交給ViewDragHelper處理苹祟,ViewDragHelper不熟悉的同學砸抛,相關知識评雌,網(wǎng)上一大堆,自行查看

    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {
        return mViewDragHelper.shouldInterceptTouchEvent(ev);
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        mViewDragHelper.processTouchEvent(event);
        return true;
    }

下面對ViewDragHelper的配置直焙,比較簡單景东,大家看代碼吧。

private void init() {
        mViewDragHelper = ViewDragHelper.create(this, 1.0f, new ViewDragHelper.Callback() {
            @Override
            public boolean tryCaptureView(View child, int pointerId) {
                return false;
            }

            @Override
            public int clampViewPositionHorizontal(View child, int left, int dx) {
                mCurArrivePoint.x = left;
                //允許左右觸發(fā)滑動奔誓,否則return 0
                if (mCurEdgeFlag != ViewDragHelper.EDGE_BOTTOM) {
                    return left;
                }else return 0;
            }

            @Override
            public int clampViewPositionVertical(View child, int top, int dy) {
                mCurArrivePoint.y = top;
                //允許底部觸發(fā)滑動斤吐,否則return 0
                if (mCurEdgeFlag == ViewDragHelper.EDGE_BOTTOM) {
                    return top;
                }else return 0;
            }

            @Override
            public void onViewReleased(View releasedChild, float xvel, float yvel) {
                super.onViewReleased(releasedChild, xvel, yvel);
                switch (mCurEdgeFlag) {
                    case ViewDragHelper.EDGE_LEFT:
                        //水平滑動超過一半,觸發(fā)結束
                        if (mCurArrivePoint.x > getWidth()/2) {
                            mViewDragHelper.settleCapturedViewAt(getWidth(), mAutoBackOrignalPoint.y);
                        }else {
                            mViewDragHelper.settleCapturedViewAt(mAutoBackOrignalPoint.x, mAutoBackOrignalPoint.y);
                        }
                        break;
                    case ViewDragHelper.EDGE_RIGHT:
                        //水平滑動超過一半厨喂,觸發(fā)結束
                        if (mCurArrivePoint.x < -getWidth()/2) {
                            mViewDragHelper.settleCapturedViewAt(-getWidth(), mAutoBackOrignalPoint.y);
                        }else {
                            mViewDragHelper.settleCapturedViewAt(mAutoBackOrignalPoint.x, mAutoBackOrignalPoint.y);
                        }
                        break;
                    case ViewDragHelper.EDGE_BOTTOM:
                        //垂直滑動超過一半和措,觸發(fā)結束
                        if (mCurArrivePoint.y < -getHeight()/2) {
                            mViewDragHelper.settleCapturedViewAt(mAutoBackOrignalPoint.x, -getHeight());
                        }else {
                            mViewDragHelper.settleCapturedViewAt(mAutoBackOrignalPoint.x, mAutoBackOrignalPoint.y);
                        }
                        break;
                }

                mCurArrivePoint.x = 0;
                mCurArrivePoint.y = 0;
                invalidate();
            }

            @Override
            public void onViewPositionChanged(View changedView, int left, int top, int dx, int dy) {
                super.onViewPositionChanged(changedView, left, top, dx, dy);
                switch (mCurEdgeFlag) {
                    case ViewDragHelper.EDGE_LEFT:
                        if (left >= getWidth()) {
                            if (mFinishScroll != null) {
                                mFinishScroll.complete();
                            }
                        }
                        break;
                    case ViewDragHelper.EDGE_RIGHT:
                        if (left <= -getWidth()) {
                            if (mFinishScroll != null) {
                                mFinishScroll.complete();
                            }
                        }
                        break;
                    case ViewDragHelper.EDGE_BOTTOM:
                        if (top <= -getHeight()) {
                            if (mFinishScroll != null) {
                                mFinishScroll.complete();
                            }
                        }
                        break;
                }
            }

            @Override
            public void onEdgeDragStarted(int edgeFlags, int pointerId) {
                mCurEdgeFlag = edgeFlags;
                if (mDragView == null) mDragView = getChildAt(0);
                mViewDragHelper.captureChildView(mDragView, pointerId);
            }
        });

        mViewDragHelper.setEdgeTrackingEnabled(ViewDragHelper.EDGE_LEFT);
    }

這里還有個步驟不要忘了,手指離開屏幕或者滑動超過屏幕的時候蜕煌,我們觸發(fā)了ViewGroup自行完全滾動出屏幕的調(diào)用派阱,所以我們需要在computeScroll中做檢查,如果滾動沒有結束斜纪,刷新View贫母,繼續(xù)滾動。

    @Override
    public void computeScroll() {
        if (mViewDragHelper.continueSettling(true)) {
            invalidate();
        }
    }

這樣我們自定義的BaseSwipeLayout打造完畢盒刚,我們把它設為activity的根布局測試一下

<?xml version="1.0" encoding="utf-8"?>
<com.aliouswang.swipeback.widget.BaseSwipeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical" android:layout_width="match_parent"
    android:id="@+id/swipe_layout"
    android:layout_height="match_parent">

    <ScrollView
        android:layout_width="match_parent"
        android:layout_height="match_parent">
        <LinearLayout
            android:orientation="vertical"
            android:layout_width="match_parent"
            android:layout_height="wrap_content">
            <TextView
                android:text="Swipe Back Demo"
                android:gravity="center"
                android:textColor="@android:color/white"
                android:background="@android:color/holo_red_light"
                android:layout_width="match_parent"
                android:layout_height="100dp" />

            <TextView
                android:text="Swipe Back Demo"
                android:gravity="center"
                android:textColor="@android:color/white"
                android:background="@android:color/holo_blue_light"
                android:layout_width="match_parent"
                android:layout_height="100dp" />

            <TextView
                android:text="Swipe Back Demo"
                android:gravity="center"
                android:textColor="@android:color/white"
                android:background="@android:color/holo_green_light"
                android:layout_width="match_parent"
                android:layout_height="100dp" />

            <TextView
                android:text="Swipe Back Demo"
                android:gravity="center"
                android:textColor="@android:color/white"
                android:background="@android:color/holo_purple"
                android:layout_width="match_parent"
                android:layout_height="100dp" />

            <TextView
                android:text="Swipe Back Demo"
                android:gravity="center"
                android:textColor="@android:color/white"
                android:background="@android:color/holo_green_dark"
                android:layout_width="match_parent"
                android:layout_height="100dp" />

            <TextView
                android:text="Swipe Back Demo"
                android:gravity="center"
                android:textColor="@android:color/white"
                android:background="@android:color/holo_red_dark"
                android:layout_width="match_parent"
                android:layout_height="100dp" />

            <TextView
                android:text="Swipe Back Demo"
                android:gravity="center"
                android:textColor="@android:color/white"
                android:background="@android:color/holo_blue_dark"
                android:layout_width="match_parent"
                android:layout_height="100dp" />

            <TextView
                android:text="Swipe Back Demo"
                android:gravity="center"
                android:textColor="@android:color/white"
                android:background="@android:color/holo_orange_dark"
                android:layout_width="match_parent"
                android:layout_height="100dp" />
        </LinearLayout>
    </ScrollView>

</com.aliouswang.swipeback.widget.BaseSwipeLayout>

當然腺劣,我們已經(jīng)實現(xiàn)了整個功能,但是有一點很不爽的是因块,我們必須將BaseSwipeLayout作為布局的根橘原,這樣實現(xiàn)還不夠優(yōu)雅,我們能不能不改變我們原有的布局文件涡上,卻依然能加入滑動退出功能趾断。
這里我們要介紹一個DecorView,它是Window的最頂層View吓懈,它含有一個子LinearLayout歼冰,代表整個Window,包括通知欄耻警,狀態(tài)欄,內(nèi)容顯示區(qū)域甸怕,所以我們activity頁面是DecorView的子View的子View甘穿,那么我們能不能直接給DecorView的子View添加到我們的BaseSwipeLayout,再將BaseSwipeLayout添加到DecorView梢杭,當然是可以的温兼,而且這種方式,我們不需要改變原來的布局文件武契,更加優(yōu)雅募判。

我們定義一個SwipeHelper類荡含,輔助我們進行BaseSwipeLayout插入操作。

public class SwipeHelper {

    private Activity mActivity;
    private BaseSwipeLayout mBaseSwipeLayout;

    public SwipeHelper(Activity activity) {
        this.mActivity = activity;
    }

    public void onActivityCreate() {
        mBaseSwipeLayout = (BaseSwipeLayout) LayoutInflater.from(mActivity)
                .inflate(R.layout.swipe_layout, null);
        mBaseSwipeLayout.setOnFinishScroll(new BaseSwipeLayout.OnFinishScroll() {
            @Override
            public void complete() {
                mActivity.finish();
            }
        });
    }

    public void onPostCreate() {
        mBaseSwipeLayout.attachToActivity(mActivity);
    }

    public void setSwipeEdge(int edgeFlag) {
        mBaseSwipeLayout.setSwipeEdge(edgeFlag);
    }
}

BaseSwipeLayout

//核心代碼届垫,綁定到相應activity
    public void attachToActivity(Activity activity) {
        this.mActivity = activity;
        TypedArray a = activity.getTheme().obtainStyledAttributes(new int[]{
                android.R.attr.windowBackground
        });
        int background = a.getResourceId(0, 0);
        a.recycle();
        ViewGroup decorView = (ViewGroup) activity.getWindow().getDecorView();
        ViewGroup decorChild = (ViewGroup) decorView.getChildAt(0);
        decorChild.setBackgroundResource(background);
        decorView.removeView(decorChild);
        addView(decorChild);
        decorView.addView(this);
    }

需要添加滑動退出的activity释液,添加想要代碼

    private SwipeHelper mSwipeHelper;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.view_drag_helper_act_layout);
        mSwipeHelper = new SwipeHelper(this);
        mSwipeHelper.onActivityCreate();
        mSwipeHelper.setSwipeEdge(ViewDragHelper.EDGE_RIGHT);
    }

    @Override
    protected void onPostCreate(Bundle savedInstanceState) {
        super.onPostCreate(savedInstanceState);
        mSwipeHelper.onPostCreate();
    }

2016-02-23 15_17_01.gif

到此我們的代碼基本完成,源碼我放到了BaseSwipe装处,歡迎指教误债!

參考:https://github.com/ikew0ng/SwipeBackLayout

最后編輯于
?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市妄迁,隨后出現(xiàn)的幾起案子寝蹈,更是在濱河造成了極大的恐慌,老刑警劉巖登淘,帶你破解...
    沈念sama閱讀 216,372評論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件箫老,死亡現(xiàn)場離奇詭異,居然都是意外死亡黔州,警方通過查閱死者的電腦和手機耍鬓,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,368評論 3 392
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來辩撑,“玉大人界斜,你說我怎么就攤上這事『霞剑” “怎么了各薇?”我有些...
    開封第一講書人閱讀 162,415評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長君躺。 經(jīng)常有香客問我峭判,道長,這世上最難降的妖魔是什么棕叫? 我笑而不...
    開封第一講書人閱讀 58,157評論 1 292
  • 正文 為了忘掉前任林螃,我火速辦了婚禮,結果婚禮上俺泣,老公的妹妹穿的比我還像新娘疗认。我一直安慰自己,他們只是感情好伏钠,可當我...
    茶點故事閱讀 67,171評論 6 388
  • 文/花漫 我一把揭開白布横漏。 她就那樣靜靜地躺著,像睡著了一般熟掂。 火紅的嫁衣襯著肌膚如雪缎浇。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,125評論 1 297
  • 那天赴肚,我揣著相機與錄音素跺,去河邊找鬼二蓝。 笑死,一個胖子當著我的面吹牛指厌,可吹牛的內(nèi)容都是我干的刊愚。 我是一名探鬼主播,決...
    沈念sama閱讀 40,028評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼仑乌,長吁一口氣:“原來是場噩夢啊……” “哼百拓!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起晰甚,我...
    開封第一講書人閱讀 38,887評論 0 274
  • 序言:老撾萬榮一對情侶失蹤衙传,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后厕九,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體蓖捶,經(jīng)...
    沈念sama閱讀 45,310評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,533評論 2 332
  • 正文 我和宋清朗相戀三年扁远,在試婚紗的時候發(fā)現(xiàn)自己被綠了俊鱼。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,690評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡畅买,死狀恐怖并闲,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情谷羞,我是刑警寧澤帝火,帶...
    沈念sama閱讀 35,411評論 5 343
  • 正文 年R本政府宣布,位于F島的核電站湃缎,受9級特大地震影響犀填,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜嗓违,卻給世界環(huán)境...
    茶點故事閱讀 41,004評論 3 325
  • 文/蒙蒙 一九巡、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧蹂季,春花似錦冕广、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至父能,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間净神,已是汗流浹背何吝。 一陣腳步聲響...
    開封第一講書人閱讀 32,812評論 1 268
  • 我被黑心中介騙來泰國打工溉委, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人爱榕。 一個月前我還...
    沈念sama閱讀 47,693評論 2 368
  • 正文 我出身青樓瓣喊,卻偏偏與公主長得像,于是被迫代替她去往敵國和親黔酥。 傳聞我的和親對象是個殘疾皇子藻三,可洞房花燭夜當晚...
    茶點故事閱讀 44,577評論 2 353

推薦閱讀更多精彩內(nèi)容