Android開發(fā)之使用150行代碼實(shí)現(xiàn)滑動(dòng)返回效果
今天帶大家實(shí)現(xiàn)滑動(dòng)返回效果.,具體內(nèi)容如下所示:
先看看效果圖:

因?yàn)闆]有具體內(nèi)容,也沒有簡書的圖片資源,所以稍微簡陋了點(diǎn).
但是依然不妨礙我們的效果展示~
OK,接下來慣例,通過閱讀本文你能學(xué)習(xí)到:
ViewDragHelper的使用(如果你想學(xué)習(xí)自定義View,那么ViewDragHelper你絕對(duì)不能錯(cuò)過)
好像也沒有什么了....
這個(gè)效果,難度不大,會(huì)ViewDragHelper的同學(xué)應(yīng)該10分鐘就能寫出來了吧~
如果不會(huì)也沒關(guān)系~
1. 我們自定義一個(gè)SwipeBackFrameLayout繼承自FrameLayout
1.1 因?yàn)榭吹阶筮咟S色的View是被遮住的,而另外一個(gè)View的寬度是MatchParent的,所以FrameLayout是不錯(cuò)的選擇.
順便增加一個(gè)回調(diào),通知activity去finish
public void setCallback(Callback mCallback){
this.mCallback = mCallback;
}
private Callback mCallback;
public interface Callback{
void onShouldFinish();
}
1.2 Xml布局,非常簡單:
<yifeiyuan.practice.practicedemos.drager.SwipeBackFrameLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/swipe_back" android:layout_width="match_parent" android:layout_height="match_parent" tools:context="yifeiyuan.practice.practicedemos.drager.SwipeBackActivity"> <TextView android:layout_width="40dp" android:layout_height="match_parent" android:text="@string/hello_world" android:gravity="center" android:background="#ffff00" /> <View android:layout_width="match_parent" android:layout_height="match_parent" android:background="#ff00ff" /> </yifeiyuan.practice.practicedemos.drager.SwipeBackFrameLayout>
1.3 實(shí)例化一個(gè)ViewDragHelper
//1f代表靈敏度
mDragHelper = ViewDragHelper.create(this, 1f,new ViewDragHelper.Callback() {
@Override
public boolean tryCaptureView(View child, int pointerId) {
return false;
}
}
//因?yàn)槲覀兪菑淖笙蛴一瑒?dòng) 所以設(shè)置EDGE_LEFT
mDragHelper.setEdgeTrackingEnabled(ViewDragHelper.EDGE_LEFT);
1.4 在SwipeBackFrameLayout里實(shí)例化xml里的子View
private View mDividerView;
private View mContentView;
@Override
protected void onFinishInflate() {
super.onFinishInflate();
mDividerView = getChildAt(0);
mDividerView.setAlpha(0f);
mContentView = getChildAt(1);
}
1.5 讓ViewDragHelper處理touch事件
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
return mDragHelper.shouldInterceptTouchEvent(ev);
}
@Override
public boolean onTouchEvent(MotionEvent event) {
mDragHelper.processTouchEvent(event);
return true;
}
1.6重寫ViewDragHelper的一些處理方法
已附上詳細(xì)注釋
@Override
public void onEdgeTouched(int edgeFlags, int pointerId) {
super.onEdgeTouched(edgeFlags, pointerId);
//觸摸到左邊界的時(shí)候 我們capture住mContentView
mDragHelper.captureChildView(mContentView, pointerId);
}
@Override
public int getViewHorizontalDragRange(View child) {
return 1;
}
@Override
public void onViewPositionChanged(View changedView, int left, int top, int dx, int dy) {
super.onViewPositionChanged(changedView, left, top, dx, dy);
Log.d(TAG, "onViewPositionChanged() called with left = [" + left + "], top = [" + top + "], dx = [" + dx + "], dy = [" + dy + "]");
//0.0 - 1.0
//Notice 這邊可以給個(gè)接口回調(diào)出去,就可以做各種炫酷的效果了
float alpha = (float) (left*1.0/mDividerWidth);
mDividerView.setAlpha(alpha);
}
@Override
public int clampViewPositionHorizontal(View child, int left, int dx) {
// Log.d(TAG, "clampViewPositionHorizontal() called with dx = [" + dx + "]");
// 計(jì)算left 我們的目標(biāo)范圍是0-dividerwidth的寬度
mLastdx = dx;
int newLeft = Math.min(mDividerWidth, Math.max(left,0));
return newLeft;
}
@Override
public void onViewReleased(View releasedChild, float xvel, float yvel) {
//>0代表用戶想關(guān)閉
if (mLastdx>0){
// 還不到關(guān)閉條件,我們讓view滑動(dòng)過去,再關(guān)閉
if (mDividerWidth != releasedChild.getLeft()) {
mDragHelper.settleCapturedViewAt(mDividerWidth,releasedChild.getTop();
invalidate();
} else {
if (mCallback != null) {
mCallback.onShouldFinish();
}
}
}else{
//用戶不想關(guān)閉 ,則滑動(dòng)到最左邊
if (mDividerWidth != 0) {
mDragHelper.settleCapturedViewAt(0, releasedChild.getTop());
invalidate();
}
}
}
@Override
public void onViewDragStateChanged(int state) {
super.onViewDragStateChanged(state);
//滑動(dòng)停止,并且到達(dá)了滑動(dòng)的判斷條件 則回調(diào)關(guān)閉
if(mDragHelper.getViewDragState()==ViewDragHelper.STATE_IDLE&&mCallback != null&&mDividerWidth==mContentView.getLeft()&&mLastdx>0) {
mCallback.onShouldFinish();
}
}
});
1.7 增加對(duì)view滑動(dòng)事件處理,對(duì)于以上mDividerWidth我們在onLayout里獲取
private int mDividerWidth;
@Override
protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
super.onLayout(changed, left, top, right, bottom);
mDividerWidth = mDividerView.getWidth();
}
//Notice view 剛初始化的時(shí)候就會(huì)被調(diào)用一次
@Override
public void computeScroll() {
super.computeScroll();
// Log.d(TAG, "computeScroll() called with " + "");
if (mDragHelper.continueSettling(true)) {
invalidate();
}
}
我們寫完自定義view后還需要自定義一下activity的退出動(dòng)畫~
2.定義activity的finish動(dòng)畫
2.1 在anim目錄下,創(chuàng)建兩個(gè)動(dòng)畫xml:
//no_anim <alpha android:duration="300" xmlns:android="http://schemas.android.com/apk/res/android" android:fromAlpha="1.0" android:toAlpha="1.0" ></alpha> //out_to_right <translate xmlns:android="http://schemas.android.com/apk/res/android" android:duration="300" android:fromXDelta="0%" android:toXDelta="100%" ></translate>
2.2 在activity里設(shè)置callback監(jiān)聽,并運(yùn)用動(dòng)畫
mSwipeBack.setCallback(new SwipeBackFrameLayout.Callback() {
@Override
public void onShouldFinish() {
finish();
overridePendingTransition(R.anim.no_anim, R.anim.out_to_right);
}
});
好了!!代碼量非常少!就是這么簡單~
吐槽一下,簡書對(duì)代碼塊的支持太差了,代碼復(fù)制過來全是亂的!!
同學(xué)們還是去看源碼吧:
源碼在我的Github上
總結(jié)
以上所述是小編給大家介紹的教你150行代碼實(shí)現(xiàn)滑動(dòng)返回效果的代碼,希望對(duì)大家有所幫助,如果大家有任何疑問請給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
如果你覺得本文對(duì)你有幫助,歡迎轉(zhuǎn)載,煩請注明出處,謝謝!
相關(guān)文章
ViewPager+Fragment實(shí)現(xiàn)側(cè)滑導(dǎo)航欄
這篇文章主要為大家詳細(xì)介紹了ViewPager+Fragment實(shí)現(xiàn)側(cè)滑導(dǎo)航欄,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-05-05
21天學(xué)習(xí)android開發(fā)教程之SurfaceView
21天學(xué)習(xí)android開發(fā)教程之SurfaceView,SurfaceView由于可以直接從內(nèi)存或者DMA等硬件接口取得圖像數(shù)據(jù),因此是個(gè)非常重要的繪圖容器,操作相對(duì)簡單,感興趣的小伙伴們可以參考一下2016-02-02
MVVMLight項(xiàng)目的綁定及各種使用場景示例分析
這篇文章主要為大家介紹了MVVMLight項(xiàng)目中的綁定及綁定的各種使用場景示例源碼分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步除夕快樂2022-01-01
Android實(shí)現(xiàn)excel/pdf/word/odt/圖片相互轉(zhuǎn)換
這篇文章主要為大家詳細(xì)介紹了Android如何實(shí)現(xiàn)excel/pdf/word/odt/圖片之間的相互轉(zhuǎn)換,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下2023-04-04
Android構(gòu)建Material Design應(yīng)用詳解
這篇文章主要為大家詳細(xì)介紹了Android構(gòu)建Material Design應(yīng)用,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-10-10
Android webview旋轉(zhuǎn)屏幕導(dǎo)致頁面重新加載問題解決辦法
這篇文章主要介紹了Android webview旋轉(zhuǎn)屏幕導(dǎo)致頁面重新加載問題解決辦法的相關(guān)資料,希望通過本文能幫助到大家實(shí)現(xiàn)這樣的問題,需要的朋友可以參考下2017-10-10
Android編程實(shí)現(xiàn)帶漸變效果的圓角矩形示例
這篇文章主要介紹了Android編程實(shí)現(xiàn)帶漸變效果的圓角矩形,涉及Android界面布局及屬性設(shè)置相關(guān)操作技巧,需要的朋友可以參考下2017-08-08
Android自定義TextView仿微信朋友圈文字展開全文功能
這篇文章主要為大家詳細(xì)介紹了Android自定義TextView仿微信朋友圈文字展開全文功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-06-06

