SwipeLayout解析-动画篇

前言

本文将通过对github的开源控件SwipeLayout的分析,来学习其拖拽效果的实现原理,以及对View中的事件分发和冲突的处理。

简介

SwipeLayout是Github上一个开源的滑动布局控件,适用与android平台,其作者为“代码家”,star数已经达到7k+,可以让开发者轻松的实现列表的滑动效果,是一个非常优秀的控件。
  SwipLayout实现了复杂的手势操作处理,在使用上,可以通过不同的滑动手势来设置动画相应对应页面,大致有两种设置方法:

  • 通过xml中配置layout_gravityla属性。该字段中的4个属性对应不同的方向划入
  • 通过代码,在java代码中,通过SwipeLayout实例添加滑动视图,调用addDrag()方法来进行设置。

更详细的使用步骤可以看一下官方的Wiki。

实现原理

首先SwipeLayout继承于FrameLayout,因此SwipeLayout为一个ViewGroup即布局容器,其下可以设置多个子View。
  我们可以看到,在名为library的module中的项目结构


SwipeLayout解析-动画篇_第1张图片
swipelayout_module.png

  在该库中,提供了adapters包,下面适配了ListView,RecyclerView的adapter,并提供了一些管理列表状态的接口,方便自己实现adapter,在这里,我们主要分析下SwipeLayout的实现。
  SwipeLayout的实现,主要难点在于以下两方面:

  • View相关的动画
  • View的事件分发

本文我们将分析View相关的动画的实现。

View相关动画

在一个SwipeLayout布局中,可将其内部的childView分为SurfaceView(需要注意的是该View不同于官方的SurfaceView,两者无关联,以下翻译为前景View)和BottomView(底部View)。当用户进行滑动存在时,前景View和底部View都会在特定的属性下进行相应的显示和隐藏效果。两者的层级关系可以看如下图:

SwipeLayout解析-动画篇_第2张图片
swipelayout_view.jpg

在SipeLayout中,提供了四个方向的滑动操作下显示底部View,对应着layout_gravity中的四个属性:

  • left 左滑时显示BottomView
  • right 右滑时显示BottomView
  • top 上滑时显示BottomView
  • bottom 下滑时显示BottomView

同时,提供了两种滑动效果,可以通过布局中show_mode属性进行设置,当然也可以setShowMode方法进行操作,两种值如下:

  • LayDown 平摊在前景View的下面
  • PullOut 拉拽效果,紧随前景View的位置来变化

那对于子View的滑动动画时如何实现的呢?我们来分析一下源码,首先,关于子View的布局显示相关的方法,肯定离不开onLayout了。该方法主要调用了updateBottomViews来初始化子View的显示位置,我们看一下updateBottomViews的代码:

    /** 更新底部View的布局 */
    private void updateBottomViews() {
        View currentBottomView = getCurrentBottomView();
        //获取可滑动的范围
        if (currentBottomView != null) {
            if (mCurrentDragEdge == DragEdge.Left || mCurrentDragEdge == DragEdge.Right) {
                mDragDistance = currentBottomView.getMeasuredWidth() - dp2px(getCurrentOffset());
            } else {
                mDragDistance = currentBottomView.getMeasuredHeight() - dp2px(getCurrentOffset());
            }
        }

        //根据不同的显示模式设置子View
        if (mShowMode == ShowMode.PullOut) {
            layoutPullOut();
        } else if (mShowMode == ShowMode.LayDown) {
            layoutLayDown();
        }
        //设置根据状态来设置bottomView的显示和隐藏
        safeBottomView();
    }

在该方法主要通过计算活动了可滑动范围值,然后会根据不同的mode初始化不同的前景View与底部View的边界和位置,两种模式下的区别可以从下图理解:

SwipeLayout解析-动画篇_第3张图片
swipelayout_mode.png

onLayout中只是做了一些子View的初始布局的处理,而子View涉及到了拖拽动画,需要我们自己来处理,而android 在官方的V4包中提供了一个ViewDragHelper类,相对与gesturedetector,ViewDragHelper更适合用于处理子View的拖拽事件,该类可以让你轻松的实现子View的拖拽效果,通过实现该类提供的接口,设置和实现拖拽,在SwipeLayout中,该接口的实现如下:

    private ViewDragHelper.Callback mDragHelperCallback = new ViewDragHelper.Callback() {

        boolean isCloseBeforeDrag = true;
        //当子View位置发生变化时会调用
        @Override
        public void onViewPositionChanged(View changedView, int left, int top , int dx, int dy) {
            View surfaceView = getSurfaceView();
            if (surfaceView == null) {
                return;
            }
            View currentBottomView = getCurrentBottomView();
            int evLeft = surfaceView.getLeft(),
                    evRight = surfaceView.getRight(),
                    evTop = surfaceView.getTop(),
                    evBottom = surfaceView.getBottom();
            if (changedView == surfaceView) {   //更新的View为前景View

                if (mShowMode == ShowMode.PullOut && currentBottomView != null) {
                    //更新底部View的位置
                    if (mCurrentDragEdge == DragEdge.Left || mCurrentDragEdge == DragEdge.Right) {
                        currentBottomView.offsetLeftAndRight(dx);
                    } else {
                        currentBottomView.offsetTopAndBottom(dy);
                    }
                }

            } else if (getBottomViews().contains(changedView)) {//更新View为底部View

                if (mShowMode == ShowMode.PullOut) {
                    surfaceView.offsetLeftAndRight(dx);
                    surfaceView.offsetTopAndBottom(dy);
                } else {

                    int newLeft = surfaceView.getLeft() + dx, newTop = surfaceView.getTop() + dy;

                    if (mCurrentDragEdge == DragEdge.Left && newLeft < getPaddingLeft()) {
                        newLeft = getPaddingLeft();
                    } else if (mCurrentDragEdge == DragEdge.Right && newLeft > getPaddingLeft()) {
                        newLeft = getPaddingLeft();
                    } else if (mCurrentDragEdge == DragEdge.Top && newTop < getPaddingTop()) {
                        newTop = getPaddingTop();
                    } else if (mCurrentDragEdge == DragEdge.Bottom && newTop > getPaddingTop()) {
                        newTop = getPaddingTop();
                    }

                    //更新前景View的布局位置
                    surfaceView.layout(newLeft, newTop, newLeft + getMeasuredWidth(), newTop + getMeasuredHeight());
                }
            }

            dispatchRevealEvent(evLeft, evTop, evRight, evBottom);

            dispatchSwipeEvent(evLeft, evTop, dx, dy);

        }

        //手指释放的时候回调,可以在这里完成未结束的操作
        @Override
        public void onViewReleased(View releasedChild, float xvel, float yvel) {
            super.onViewReleased(releasedChild, xvel, yvel);
            processHandRelease(xvel, yvel, isCloseBeforeDrag);
            for (SwipeListener l : mSwipeListeners) {
                l.onHandRelease(SwipeLayout.this, xvel, yvel);
            }

            invalidate();
        }

        //确定范围
        @Override
        public int getViewHorizontalDragRange(View child) {
            return mDragDistance;
        }

        @Override
        public int getViewVerticalDragRange(View child) {
            return mDragDistance;
        }

        //View是否可捕获标志
        @Override
        public boolean tryCaptureView(View child, int pointerId) {
            boolean result = child == getSurfaceView() || getBottomViews().contains(child);
            if (result) {
                isCloseBeforeDrag = getOpenStatus() == Status.Close;
            }
            return result;
        }

        //设置拖拽后的最终位置
        @Override
        public int clampViewPositionHorizontal(View child, int left, int dx) {
            ……
            return left;
        }

        @Override
        public int clampViewPositionVertical(View child, int top, int dy) {
            ……
            return top;
        }
    };

在该接口中clampViewPositionHorizontal和clampViewPositionVertical用来确定最终的left点和top点,一般需要手动做一些计算来确定边界值。重点需要看onViewPositionChanged和onViewReleased方法,整个SwipeLayout就是通过这两个方法来实现拖拽和动画效果的。
  在实现ViewDragHelper的接口后,该类会管理相应子View的拖拽功能,所以在onViewPositionChanged中,我们只需要处理另外一个View的layout值即可,从代码中可以发现,会先判断changeView所属位置,如果是前景View,那么接下来就需要手动处理底部View的layout,最终会调用offsetLeftAndRight/offsetTopAndBottom来更新底部View的位置。同理在changeView为底部View时,会对前景View设置layout。但这并不能满足我们的需求,这里仅处理了拖拽过程中的位置更新,如果在滑动过程中松手,就需要处理未完成的View状态动画。
  因此,我们需要在onViewReleased里做一些处理,该方法会在手指释放时调用,最终会调用到processHandRelease方法,该方法中根据不同的条件做了判断,最终会确定调用open方法还是close方法来实现动画效果,两个方法处理过程类似,我们看一下open的代码:

    //ViewReleased最终的动画实现
    public void open(boolean smooth, boolean notify) {
        View surface = getSurfaceView(), bottom = getCurrentBottomView();
        if (surface == null) {
            return;
        }
        int dx, dy;
        //获取SurfaceView的布局区域的rect
        Rect rect = computeSurfaceLayoutArea(true);
        if (smooth) {
            //平滑效果打开动画,通过DragHelper实现
            mDragHelper.smoothSlideViewTo(surface, rect.left, rect.top);
        } else {
            //无过度效果,直接设置最后的layout
            dx = rect.left - surface.getLeft();
            dy = rect.top - surface.getTop();
            surface.layout(rect.left, rect.top, rect.right, rect.bottom);
            if (getShowMode() == ShowMode.PullOut) {
                Rect bRect = computeBottomLayoutAreaViaSurface(ShowMode.PullOut, rect);
                if (bottom != null) {
                    bottom.layout(bRect.left, bRect.top, bRect.right, bRect.bottom);
                }
            }
            if (notify) {
                dispatchRevealEvent(rect.left, rect.top, rect.right, rect.bottom);
                dispatchSwipeEvent(rect.left, rect.top, dx, dy);
            } else {
                safeBottomView();
            }
        }
        invalidate();
    }

open代码中,如果需要平滑的动画效果,就会调用ViewDragHelper里的smoothSlideViewTo方法更新前景View,而该方法后开始的滑动动画又会触发onViewPositionChanged方法,间接的给底部View添加了动画效果。而在smooth未false的情况下,会直接设置前景View和底部View的layout来完成子View的最终状态显示。

相关博文

Android ViewDragHelper完全解析 自定义ViewGroup神器
ViewDragHelper详解

你可能感兴趣的:(SwipeLayout解析-动画篇)