考拉Android全局滑動返回及聯動效果的實現

做者:許方鎮java

前言

首次經過右滑來返回到上一個頁面的操做是在 IOS7上出現。到目前android應用上支持這種操做的依然很少。分析其主要緣由應該是android已有實體的返回按鍵,這樣的功能變得不重要,但我以爲有這樣的功能便於單手操做,能提高app的用戶體驗,特別是從ios轉到android的用戶。寫這篇博文但願能夠對你們有所幫助,但願本身的app上有滑動返回功能的能夠參考下。android

原理的簡單描述

Android系統裏有不少滑動相關的API和類,好比ViewDragHelper就是一個很好的滑動助手類。首先設置Window的背景爲透明,再經過ViewDragHelper對Activity上DecorView的子view進行滑動,當滑動到必定距離,手指離開後就自動滑到最右側,而後finish當前的activity,這樣便可實現滑動返回效果。爲了可以 「全局的」、「聯動的」 實現滑動返回效果,在每一個activity的DecorView下插入了SwipeBackLayout,當前activity滑動和下層activity的聯動都在該類中完成。ios

效果圖

下層聯動滑動返回

佈局圖

Hierarchy View

實現主要類:

SwipeBackActivity //滑動返回基類
SwipeBackLayout //滑動返回佈局類
SwipeBackLayoutDragHelper //修改ViewDragHelper後助手類
TranslucentHelper //代碼中修改透明或者不透明的助手類
canvas

##代碼層面的講解api

一. 設置activity爲透明、activity跳轉動畫(TranslucentHelper 講解)

這個看起來很簡單,但若是要兼容到API16及如下,會遇到過一個比較麻煩的頁面切換動畫問題:bash

1.一、經過activity的主題style進行設置

<item name="android:windowBackground">@color/transparent</item>
<item name="android:windowIsTranslucent">true</item>```

**遇到問題:**若是在某個activity的主題style中設置了android:windowIsTranslucent屬性爲true,那麼該activity切換動畫與沒設置以前是不一樣的,有些手機切換動畫會變得很是跳。因此須要自定義activity的切換動畫。
接下來咱們會想到經過主題style裏的windowAnimationStyle來設置切換動畫
複製代碼

@anim/activity_open_enter @anim/activity_open_exit @anim/activity_close_enter @anim/activity_close_exit``` **實踐證實:**當android:windowIsTranslucent爲true時,以上幾個屬性是無效的,而下面兩個屬性仍是能夠用。可是這兩個屬性一個是窗口進來動畫,一個是窗口退出動畫,明顯是不夠。app

<item name="android:windowEnterAnimation">@anim/***</item>
<item name="android:windowExitAnimation">@anim/***</item>```

結合overridePendingTransition(int enterAnim, int exitAnim)能夠複寫窗口進來動畫和窗口退出動畫,這種我以爲最終多是能夠實現的,不過控制起來比較複雜:
好比有A、B、C三個頁面:
A跳到B,進場頁面B動畫從右進來,出場頁面A動畫從左出去,能夠直接在style中寫死
複製代碼

@anim/*** @anim/***```ide

若是B返回到A,進場頁面A動畫從左進來,出場頁面B動畫從右出去,此時須要經過複寫onBackPressed() 方法,
在其中添加overridePendingTransition(int enterAnim, int exitAnim)方法來改變更畫。佈局

若是B是finish()後到A頁面,在finish()後面加上overridePendingTransition ……
因爲onBackPressed() 方法最終會調finish(),因此實際上只須要複寫finish(),在其中添加overridePendingTransition……post

可是假如B finish()後跳到C,則又不該該執行overridePendingTransition……,那麼就須要判斷finish執行後是否要加 overridePendingTransition……
對於一個較爲龐大的項目,採起這種方法須要對每一個頁面進行排查,所以是不可行的,而對於剛剛起步的應用來講則是一個選擇。

1.二、經過透明助手類(TranslucentHelper)進行設置

透明助手類(TranslucentHelper)裏主要又有兩個方法,一個是讓activity變不透明,一個是讓activity變透明,這兩個都是經過反射來調用隱藏的系統api來實現的。由於較低的版本不支持代碼中修改背景透明不透明,因此在類中有個靜態變量mTranslucentState 來記錄是否能夠切換背景,這樣低版本就不須要每次都反射經過捕獲到的異常來作兼容方案。 另外:發現有些手機支持背景變黑,但不支持背景變透明(中興z9 mini 5.0.2系統)

public class TranslucentHelper {
    private static final String TRANSLUCENT_STATE = "translucentState";
    private static final int INIT = 0;//表示初始
    private static final int CHANGE_STATE_FAIL = INIT + 1;//表示確認不能夠切換透明狀態
    private static final int CHANGE_STATE_SUCCEED = CHANGE_STATE_FAIL + 1;//表示確承認以切換透明狀態
    private static int mTranslucentState = INIT;

    interface TranslucentListener {
        void onTranslucent();
    }

    private static class MyInvocationHandler implements InvocationHandler {
        private TranslucentListener listener;

        MyInvocationHandler(TranslucentListener listener) {
            this.listener = listener;
        }

        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            try {
                boolean success = (boolean) args[0];
                if (success && listener != null) {
                    listener.onTranslucent();
                }
            } catch (Exception ignored) {
            }
            return null;
        }
    }

    static boolean convertActivityFromTranslucent(Activity activity) {
        if (mTranslucentState == INIT) {
            mTranslucentState = PreferencesUtils.getInt(TRANSLUCENT_STATE, INIT);
        }
        if (mTranslucentState == INIT) {
            convertActivityToTranslucent(activity, null);
        } else if (mTranslucentState == CHANGE_STATE_FAIL) {
            return false;
        }

        try {
            Method method = Activity.class.getDeclaredMethod("convertFromTranslucent");
            method.setAccessible(true);
            method.invoke(activity);
            mTranslucentState = CHANGE_STATE_SUCCEED;
            return true;
        } catch (Throwable t) {
            mTranslucentState = CHANGE_STATE_FAIL;
            PreferencesUtils.saveInt(TRANSLUCENT_STATE, CHANGE_STATE_FAIL);
            return false;
        }
    }

    static void convertActivityToTranslucent(Activity activity, final TranslucentListener listener) {
        if (mTranslucentState == CHANGE_STATE_FAIL) {
            if (listener != null) {
                listener.onTranslucent();
            }
            return;
        }

        try {
            Class<?>[] classes = Activity.class.getDeclaredClasses();
            Class<?> translucentConversionListenerClazz = null;
            for (Class clazz : classes) {
                if (clazz.getSimpleName().contains("TranslucentConversionListener")) {
                    translucentConversionListenerClazz = clazz;
                }
            }

            MyInvocationHandler myInvocationHandler = new MyInvocationHandler(listener);
            Object obj = Proxy.newProxyInstance(Activity.class.getClassLoader(),
                    new Class[] { translucentConversionListenerClazz }, myInvocationHandler);

            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
                Method getActivityOptions = Activity.class.getDeclaredMethod("getActivityOptions");
                getActivityOptions.setAccessible(true);
                Object options = getActivityOptions.invoke(activity);

                Method method = Activity.class.getDeclaredMethod("convertToTranslucent",
                        translucentConversionListenerClazz, ActivityOptions.class);
                method.setAccessible(true);
                method.invoke(activity, obj, options);
            } else {
                Method method =
                        Activity.class.getDeclaredMethod("convertToTranslucent", translucentConversionListenerClazz);
                method.setAccessible(true);
                method.invoke(activity, obj);
            }
            mTranslucentState = CHANGE_STATE_SUCCEED;
        } catch (Throwable t) {
            mTranslucentState = CHANGE_STATE_FAIL;
            PreferencesUtils.saveInt(TRANSLUCENT_STATE, CHANGE_STATE_FAIL);
            new Handler().postDelayed(new Runnable() {
                @Override
                public void run() {
                    if (listener != null) {
                        listener.onTranslucent();
                    }
                }
            }, 100);
        }
    }
}
複製代碼

讓activity變不透明的方法比較簡單;讓activity變透明的方法參數裏傳入了一個listener接口 ,主要是當antivity變透明後會回調,由於這個接口也在activity裏,並且是私有的,因此咱們只能經過動態代理去獲取這個回調。最後若是版本大於等於5.0,還須要再傳入一個ActivityOptions參數。

在實際開發中,這兩個方法在android 5.0以上是有效的,在5.0如下須要當android:windowIsTranslucent爲true時纔有效,這樣又回到了以前的問題activity切換動畫異常。

**最終決解方法:**setContentView以前就調用 convertActivityFromTranslucent方法,讓activity背景變黑,這樣activity切換效果就正常。

**總結:**在style中設置android:windowIsTranslucent爲true ,setContentView以前就調用 convertActivityFromTranslucent方法,當觸發右滑時調用convertActivityToTranslucent,經過動態代理獲取activity變透明後的回調,在回調後容許開始滑動。

二. 讓BaseActivity繼承SwipeBackActivity(SwipeBackActivity講解)

先直接看代碼,比較少

public abstract class SwipeBackActivity extends CoreBaseActivity {
    /** * 滑動返回View */
    private SwipeBackLayout mSwipeBackLayout;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        if (!isSwipeBackDisableForever()) {
            TranslucentHelper.convertActivityFromTranslucent(this);
            mSwipeBackLayout = new SwipeBackLayout(this);
        }
    }

    @Override
    protected void onPostCreate(Bundle savedInstanceState) {
        super.onPostCreate(savedInstanceState);
        if (!isSwipeBackDisableForever()) {
            mSwipeBackLayout.attachToActivity(this);
            mSwipeBackLayout.setOnSwipeBackListener(new SwipeBackLayout.onSwipeBackListener() {
                @Override
                public void onStart() {
                    onSwipeBackStart();
                }

                @Override
                public void onEnd() {
                    onSwipeBackEnd();
                }
            });
        }
    }

    @Override
    public void onWindowFocusChanged(boolean hasFocus) {
        super.onWindowFocusChanged(hasFocus);
        if (!isSwipeBackDisableForever() && hasFocus) {
            getSwipeBackLayout().recovery();
        }
    }

    /** * 滑動返回開始時的回調 */
    protected void onSwipeBackStart() {}

    /** * 滑動返回結束時的回調 */
    protected void onSwipeBackEnd() {}

    /** * 設置是否能夠邊緣滑動返回,須要在onCreate方法調用 */
    public void setSwipeBackEnable(boolean enable) {
        if (mSwipeBackLayout != null) {
            mSwipeBackLayout.setSwipeBackEnable(enable);
        }
    }

    public boolean isSwipeBackDisableForever() {
        return false;
    }

    public SwipeBackLayout getSwipeBackLayout() {
        return mSwipeBackLayout;
    }
}
複製代碼

SwipeBackActivity中包含了一個SwipeBackLayout對象

在onCreate方法中:
1.activity轉化爲不透明。
2.new了一個SwipeBackLayout。

在onPostCreate方法中:
1.attachToActivity主要是插入SwipeBackLayout、窗口背景設置……
2.設置了滑動返回開始和結束的監聽接口,建議在滑動返回開始時,把PopupWindow給dismiss掉。

onWindowFocusChanged 方法中
若是是hasFocus == true,就recovery()這個SwipeBackLayout,這個也是由於下層activity有聯動效果而移動了SwipeBackLayout,因此須要recovery()下,防止異常狀況。

isSwipeBackDisableForever 方法是一個大開關,默認返回false,在代碼中複寫後返回 true,則至關於直接繼承了SwipeBackActivity的父類。

setSwipeBackEnable 方法是一個小開關,設置了false以後就暫時不能滑動返回了,能夠在特定的時機設置爲true,就恢復滑動返回的功能。

**總結說明:**下層activity設置了setSwipeBackEnable 爲 false,上層activity滑動時仍是能夠聯動的,好比MainActivity。而isSwipeBackDisableForever 返回true就不會聯動了,並且一些仿PopupWindow的activity須要複寫這個方法,由於activity須要透明。

3、滑動助手類的使用和滑動返回佈局類的實現(SwipeBackLayout講解)

直接貼SwipeBackLayout源碼:

/** * 滑動返回容器類 */
public class SwipeBackLayout extends FrameLayout {
    /** * 滑動銷燬距離比例界限,滑動部分的比例超過這個就銷燬 */
    private static final float DEFAULT_SCROLL_THRESHOLD = 0.5f;
    /** * 滑動銷燬速度界限,超過這個速度就銷燬 */
    private static final float DEFAULT_VELOCITY_THRESHOLD = ScreenUtils.dpToPx(250);
    /** * 最小滑動速度 */
    private static final int MIN_FLING_VELOCITY = ScreenUtils.dpToPx(200);
    /** * 左邊移動的像素值 */
    private int mContentLeft;
    /** * 左邊移動的像素值 / (ContentView的寬+陰影) */
    private float mScrollPercent;
    /** * (ContentView可見部分+陰影)的比例 (即1 - mScrollPercent) */
    private float mContentPercent;
    /** * 陰影圖 */
    private Drawable mShadowDrawable;
    /** * 陰影圖的寬 */
    private int mShadowWidth;
    /** * 內容view,DecorView的原第一個子view */
    private View mContentView;
    /** * 用於記錄ContentView所在的矩形 */
    private Rect mContentViewRect = new Rect();
    /** * 設置是否可滑動 */
    private boolean mIsSwipeBackEnable = true;
    /** * 是否正在放置 */
    private boolean mIsLayout = true;
    /** * 判斷背景Activity是否啓動進入動畫 */
    private boolean mIsEnterAnimRunning = false;
    /** * 是不是透明的 */
    private boolean mIsActivityTranslucent = false;
    /** * 進入動畫(只在釋放手指時使用) */
    private ObjectAnimator mEnterAnim;
    /** * 退拽助手類 */
    private SwipeBackLayoutDragHelper mViewDragHelper;
    /** * 執行滑動時的最頂層Activity */
    private Activity mTopActivity;
    /** * 後面的Activity的弱引用 */
    private WeakReference<Activity> mBackActivityWeakRf;
    /** * 監聽滑動開始和結束 */
    private onSwipeBackListener mListener;

    public SwipeBackLayout(Context context) {
        super(context);
        init(context);
    }

    public SwipeBackLayout(Context context, AttributeSet attrs) {
        super(context, attrs);
        init(context);
    }

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

    private void init(Context context) {
        mViewDragHelper = SwipeBackLayoutDragHelper.create(SwipeBackLayout.this, new ViewDragCallback());
        mViewDragHelper.setEdgeTrackingEnabled(SwipeBackLayoutDragHelper.EDGE_LEFT);
        mViewDragHelper.setMinVelocity(MIN_FLING_VELOCITY);
        mViewDragHelper.setMaxVelocity(MIN_FLING_VELOCITY * 2);
        try {
            mShadowDrawable = context.getResources().getDrawable(R.drawable.swipeback_shadow_left);
        } catch (Exception ignored) {
        }
    }

    @Override
    protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
        try {
            if (!mIsSwipeBackEnable) {
                super.onLayout(changed, left, top, right, bottom);
                return;
            }
            mIsLayout = true;
            if (mContentView != null) {
                mContentView.layout(mContentLeft, top, mContentLeft + mContentView.getMeasuredWidth(),
                        mContentView.getMeasuredHeight());
            }
            mIsLayout = false;
        } catch (Exception e) {
            super.onLayout(changed, left, top, right, bottom);
        }
    }

    @Override
    public void requestLayout() {
        if (!mIsLayout || !mIsSwipeBackEnable) {
            super.requestLayout();
        }
    }

    @Override
    protected boolean drawChild(Canvas canvas, View child, long drawingTime) {
        try {
            //繪製陰影
            if (mContentPercent > 0
                    && mShadowDrawable != null
                    && child == mContentView
                    && mViewDragHelper.getViewDragState() != SwipeBackLayoutDragHelper.STATE_IDLE) {
                child.getHitRect(mContentViewRect);
                mShadowWidth = mShadowDrawable.getIntrinsicWidth();
                mShadowDrawable.setBounds(mContentViewRect.left - mShadowWidth, mContentViewRect.top,
                        mContentViewRect.left, mContentViewRect.bottom);
                mShadowDrawable.draw(canvas);
            }
            return super.drawChild(canvas, child, drawingTime);
        } catch (Exception e) {
            return super.drawChild(canvas, child, drawingTime);
        }
    }

    @Override
    public void computeScroll() {
        mContentPercent = 1 - mScrollPercent;
        if (mViewDragHelper.continueSettling(true)) {
            ViewCompat.postInvalidateOnAnimation(this);
        }
    }

    @Override
    public boolean onInterceptTouchEvent(MotionEvent event) {
        if (!mIsSwipeBackEnable) {
            return false;
        }
        try {
            return mViewDragHelper.shouldInterceptTouchEvent(event);
        } catch (ArrayIndexOutOfBoundsException e) {
            return super.onInterceptTouchEvent(event);
        }
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        if (!mIsSwipeBackEnable) {
            return false;
        }
        try {
            mViewDragHelper.processTouchEvent(event);
            return true;
        } catch (Exception e) {
            return super.onTouchEvent(event);
        }
    }

    /** * 將View添加到Activity */
   public void attachToActivity(Activity activity) {
        //插入SwipeBackLayout
        ViewGroup decor = (ViewGroup) activity.getWindow().getDecorView();
        ViewGroup decorChild = (ViewGroup) decor.getChildAt(0);
        decor.removeView(decorChild);
        if (getParent() != null) {
            decor.removeView(this);
        }
        decor.addView(this);
        this.removeAllViews();
        this.addView(decorChild);

        //mContentView爲SwipeBackLayout的直接子view,獲取window背景色進行賦值
        activity.getWindow().setBackgroundDrawableResource(R.color.transparent_white);
        TypedArray a = activity.getTheme().obtainStyledAttributes(new int[] {
                android.R.attr.windowBackground
        });
        mContentView = decorChild;
        mContentView.setBackgroundResource(a.getResourceId(0, 0));
        a.recycle();

        //拿到頂層activity和下層activity,作聯動操做
        mTopActivity = activity;
        Activity backActivity = ActivityUtils.getSecondTopActivity();
        if (backActivity != null && backActivity instanceof SwipeBackActivity) {
            if (!((SwipeBackActivity) backActivity).isSwipeBackDisableForever()) {
                mBackActivityWeakRf = new WeakReference<>(backActivity);
            }
        }
    }

    /** * 設置是否能夠滑動返回 */
    public void setSwipeBackEnable(boolean enable) {
        mIsSwipeBackEnable = enable;
    }

    public boolean isActivityTranslucent() {
        return mIsActivityTranslucent;
    }

    /** * 啓動進入動畫 */
    private void startEnterAnim() {
        if (mContentView != null) {
            ObjectAnimator anim =
                    ObjectAnimator.ofFloat(mContentView, "TranslationX", mContentView.getTranslationX(), 0f);
            anim.setDuration((long) (125 * mContentPercent));
            mEnterAnim = anim;
            mEnterAnim.start();
        }
    }

    protected View getContentView() {
        return mContentView;
    }

    private class ViewDragCallback extends SwipeBackLayoutDragHelper.Callback {
        @Override
        public boolean tryCaptureView(View child, int pointerId) {
            if (mIsSwipeBackEnable && mViewDragHelper.isEdgeTouched(SwipeBackLayoutDragHelper.EDGE_LEFT, pointerId)) {
                TranslucentHelper.convertActivityToTranslucent(mTopActivity,
                        new TranslucentHelper.TranslucentListener() {
                            @Override
                            public void onTranslucent() {
                                if (mListener != null) {
                                    mListener.onStart();
                                }
                                mIsActivityTranslucent = true;
                            }
                        });
                return true;
            }
            return false;
        }

        @Override
        public int getViewHorizontalDragRange(View child) {
            return mIsSwipeBackEnable ? SwipeBackLayoutDragHelper.EDGE_LEFT : 0;
        }

        @Override
        public void onViewPositionChanged(View changedView, int left, int top, int dx, int dy) {
            super.onViewPositionChanged(changedView, left, top, dx, dy);
            if (changedView == mContentView) {
                mScrollPercent = Math.abs((float) left / mContentView.getWidth());
                mContentLeft = left;
                //未執行動畫就平移
                if (!mIsEnterAnimRunning) {
                    moveBackActivity();
                }
                invalidate();
                if (mScrollPercent >= 1 && !mTopActivity.isFinishing()) {
                    if (mBackActivityWeakRf != null && ActivityUtils.activityIsAlive(mBackActivityWeakRf.get())) {
                        ((SwipeBackActivity) mBackActivityWeakRf.get()).getSwipeBackLayout().invalidate();
                    }
                    mTopActivity.finish();
                    mTopActivity.overridePendingTransition(0, 0);
                }
            }
        }

        @Override
        public void onViewReleased(View releasedChild, float xvel, float yvel) {
            if (xvel > DEFAULT_VELOCITY_THRESHOLD || mScrollPercent > DEFAULT_SCROLL_THRESHOLD) {
                if (mIsActivityTranslucent) {
                    mViewDragHelper.settleCapturedViewAt(releasedChild.getWidth() + mShadowWidth, 0);
                    if (mContentPercent < 0.85f) {
                        startAnimOfBackActivity();
                    }
                }
            } else {
                mViewDragHelper.settleCapturedViewAt(0, 0);
            }
            if (mListener != null) {
                mListener.onEnd();
            }
            invalidate();
        }

        @Override
        public int clampViewPositionHorizontal(View child, int left, int dx) {
            return Math.min(child.getWidth(), Math.max(left, 0));
        }

        @Override
        public void onViewDragStateChanged(int state) {
            super.onViewDragStateChanged(state);

            if (state == SwipeBackLayoutDragHelper.STATE_IDLE && mScrollPercent < 1f) {
                TranslucentHelper.convertActivityFromTranslucent(mTopActivity);
                mIsActivityTranslucent = false;
            }
        }

        @Override
        public boolean isTranslucent() {
            return SwipeBackLayout.this.isActivityTranslucent();
        }
    }

    /** * 背景Activity開始進入動畫 */
    private void startAnimOfBackActivity() {
        if (mBackActivityWeakRf != null && ActivityUtils.activityIsAlive(mBackActivityWeakRf.get())) {
            mIsEnterAnimRunning = true;
            SwipeBackLayout swipeBackLayout = ((SwipeBackActivity) mBackActivityWeakRf.get()).getSwipeBackLayout();
            swipeBackLayout.startEnterAnim();
        }
    }

    /** * 移動背景Activity */
    private void moveBackActivity() {
        if (mBackActivityWeakRf != null && ActivityUtils.activityIsAlive(mBackActivityWeakRf.get())) {
            View view = ((SwipeBackActivity) mBackActivityWeakRf.get()).getSwipeBackLayout().getContentView();
            if (view != null) {
                int width = view.getWidth();
                view.setTranslationX(-width * 0.3f * Math.max(0f, mContentPercent - 0.15f));
            }
        }
    }

    /** * 回覆界面的平移到初始位置 */
    public void recovery() {
        if (mEnterAnim != null && mEnterAnim.isRunning()) {
            mEnterAnim.end();
        } else {
            mContentView.setTranslationX(0);
        }
    }

    interface onSwipeBackListener {
        void onStart();

        void onEnd();
    }

    public void setOnSwipeBackListener(onSwipeBackListener listener) {
        mListener = listener;
    }
}
複製代碼
  • attachToActivity
    上面講到SwipeBackLayout是在activity的onCreate時被建立,在onPostCreate是插入到DecorView裏,主要是由於DecorView是在setContentView時與Window關聯起來插入SwipeBackLayout方法如代碼所示,不難,而後是設置了window的背景色爲透明色,mContentView爲SwipeBackLayout的直接子view,獲取window背景色進行賦值。因爲考拉項目已經有不少activity,而這些activity中android:windowBackground設置的顏色大部分是白色,少部分是灰色和透明的,因此須要在代碼中設置統一設置一遍透明的,原來的背景色則賦值給SwipeBackLayout的子View就能夠達到最少修改代碼的目的。最後拿到頂層activity和下層activity,作聯動操做

  • SwipeBackLayoutDragHelper 和 ViewDragCallback
    SwipeBackLayout中包含了一個滑動助手類(SwipeBackLayoutDragHelper)的對象,該類是在ViewDragHelper的基礎上進行修改得來的。
    修改點:
    1.右側觸發區域 EDGE_SIZE由 20dp 改到 25dp
    2.提供滑動最大速度 的設置方法
    3.在ViewDragHelper 的內部類Callback方法中提供是否activity爲透明的回調接口
    4.在最終調用滑動的方法dragTo中添加判斷邏輯,activity爲透明時才支持滑動
    SwipeBackLayoutDragHelper 在init 方法中初始化,經過onInterceptTouchEvent和onTouchEvent拿到滑動事件,經過ViewDragCallback的一些方法返回相應的滑動回調, ViewDragCallback實現了SwipeBackLayoutDragHelper.Callback裏的如下幾個接口,其中其中isTranslucent()是本身添加進去的。

    ViewDragCallback

  • tryCaptureView方法當觸摸到SwipeBackLayout裏的子View時觸發的,當返回true,表示捕捉成功,不然失敗。判斷條件是若是支持滑動返回而且是左側邊距被觸摸時才能夠,咱們知道這個時候的的背景色是不透明的,若是直接開始滑動則是黑色的,因此須要在這裏背景色改爲透明的,若是直接調用 TranslucentHelper.convertActivityToTranslucent(mTopActivity, null)後直接返回true,會出現一個異常狀況,就是滑動過快時會致使背景還來不及變成黑色就滑動出來了,以後才變成透明的,從而致使了會從黑色到透明的一個閃爍現象,解決的辦法是在代碼中用了一個回調和標記,當變成透明後設置了mIsActivityTranslucent = true;經過mIsActivityTranslucent 這個變量來判斷是否進行移動的操做。因爲修改activity變透明的方法是經過反射的,不能簡單的設置一個接口後進行回調,而是經過動態代理的方式來實現的(InvocationHandler),在convertToTranslucent方法的第一個參數恰好是一個判斷activity是否已經變成透明的回調,看下面代碼中 if 語句裏的註釋和回調,若是窗口已經變成透明的話,就傳了一個drawComplete (true)。經過動態代理,將translucentConversionListenerClazz 執行其方法onTranslucentConversionComplete的替換成myInvocationHandler中執行invoke方法。其中賦值給success的args[0]正是 drawComplete。

  • isTranslucent是本身添加了一個方法,主要是返回activity是不是透明的默認爲true,在SwipeBackLayout重寫後將mIsActivityTranslucent返回。仔細看SwipeBackLayoutDragHelper方法的話,會發現最後經過dragTo方法對view進行移動,所以在進行水平移動前判斷下是不是透明的,只有透明瞭才能移動

  • onViewPositionChanged view移動過程當中會持續調用,這裏面的邏輯主要有這幾個: 1.實時計算滑動了多少距離,用於繪製左側陰影等
    2.使下面的activity進行移動moveBackActivity();
    3.當view徹底移出屏幕後,銷燬當前的activity

  • onViewReleased是手指釋放後觸發的一個方法。若是滑動速度大於最大速度或者滑動的距離大於設定的閾值距離,則直接移到屏幕外,同時觸發下層activity的復位動畫,不然移會到原來位置 。

  • onViewDragStateChanged當滑動的狀態發生改變時的回調,主要是中止滑動後,將背景改爲不透明,這樣跳到別的頁面是動畫就是正常的。

  • clampViewPositionHorizontal 返回水平移動距離,防止滑出父 view。

  • getViewHorizontalDragRange對於clickable=true的子view,須要返回大於0的數字才能正常捕獲。

其餘方法都較爲簡單,註釋也寫了,就很少說了,最後絕不吝嗇的貼上SwipeBackLayoutDragHelper的dragTo代碼,就多了if (mCallback.isTranslucent())

private void dragTo(int left, int top, int dx, int dy) {
        int clampedX = left;
        int clampedY = top;
        final int oldLeft = mCapturedView.getLeft();
        final int oldTop = mCapturedView.getTop();
        if (dx != 0) {
            clampedX = mCallback.clampViewPositionHorizontal(mCapturedView, left, dx);
            if (mCallback.isTranslucent()) {
                ViewCompat.offsetLeftAndRight(mCapturedView, clampedX - oldLeft);
            }
        }
        if (dy != 0) {
            clampedY = mCallback.clampViewPositionVertical(mCapturedView, top, dy);
            ViewCompat.offsetTopAndBottom(mCapturedView, clampedY - oldTop);
        }

        if (dx != 0 || dy != 0) {
            final int clampedDx = clampedX - oldLeft;
            final int clampedDy = clampedY - oldTop;
            if (mCallback.isTranslucent()) {
                mCallback.onViewPositionChanged(mCapturedView, clampedX, clampedY,
                        clampedDx, clampedDy);
            }
        }
    }

複製代碼
相關文章
相關標籤/搜索