ViewDragHelper(一)— 介绍及使用(入门篇)

前言

随着入Android这个坑的时间越来越长,愈加觉得深入掌握原理以及技术输出的重要性,会使用轮子和造一个好轮子还是有天壤之别的。授人以鱼不如授人以渔,将一些经验分享出来,希望能够让更多的人更加深入地理解它,并帮助到有需要的朋友。本系列分为三篇,会由浅至深地对DrageHelper 进行详细讲解。

目录

ViewDragHelper 的介绍以及初步使用请阅读这篇:
ViewDragHelper (一)- 介绍及简单用例(入门篇)
ViewDragHelper 的源码以及Callback的详情介绍请阅读这篇:
ViewDragHelper (二)- 源码及原理解读(进阶篇)
利用DrageHelper 打造仿陌陌APP视频播放页的demo请阅读这篇:
ViewDragHelper (三)- 打造仿陌陌视频播放页(深入篇)

介绍:

首先简单看一下它的官方解释:

ViewDragHelper is a utility class for writing custom ViewGroups. It offers a number
of useful operations and state tracking for allowing a user to drag and reposition
views within their parent ViewGroup.

DrageHelper 它是Google官方推出的手势滑动辅助类,极大程度地简化了我们对控件的手势滑动跟踪及处理。让我们能够更加便捷地开发自定义ViewGroup控件,实现拖拽以及弹性滚动等功能。事实上,官方的SlidingPaneLayout和DrawerLayout都是利用ViewDragHelper实现的。掌握它,可以一定程度地减轻我们开发工作难度以及投入精力。

使用入门示例

接下来,我们主要通过一个简单的拖拽以及回弹的demo(类似于QQ空间视频播放页),来讲解如何利用DrageHelper 打造一个 ViewGroup 控件。

QQ空间视频播放页效果图:

QZone.gif

大致步骤如下:

第一步:

创建一个DraggableView类继承自ViewGroup(或者也可用 FrameLayout , RelativeLayout, LinearLayout等)。

package com.test.demo;

import android.content.Context;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.support.v4.widget.ViewDragHelper;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.widget.RelativeLayout;

/**
 * Created by 小嵩 on 2017/9/10.
 */

public class MyDraggableView extends RelativeLayout{


    private ViewDragHelper viewDragHelper;

    public MyDraggableView(@NonNull Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
        initView();
    }

    private void initView() {
        viewDragHelper = ViewDragHelper.create(this, 1.0f, new DraggableViewCallback(this));
    }

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

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

在onInterceptTouchEvent方法中,通过viewDragHelper.shouldInterceptTouchEvent(event)来决定我们是否应该拦截当前的事件,如果返回的是True,则会触发onTouchEvent。
在onTouchEvent方法中,通过viewDragHelper.processTouchEvent(event)将事件分发给viewDragHelper。

对Android的事件分发机制若还不太理解的话,可自行查资料补一下相关知识。

第二步:

在init方法中用ViewDragHelper的静态方法实例化ViewDragHelper对象

viewDragHelper = ViewDragHelper.create(this, 1.0f, new VerticalDraggableViewCallback(this));

其中第一个参数指的当前的ViewGroup对象,第二个sensitivity参数指的是对滑动检测的灵敏度,越大越敏感,所需触发滑动的距离越小,默认传1.0f 即可。它的源码如下:

 /**
     * Factory method to create a new ViewDragHelper.
     *
     * @param forParent Parent view to monitor
     * @param sensitivity Multiplier for how sensitive the helper should be about detecting
     *                    the start of a drag. Larger values are more sensitive. 1.0f is normal.
     * @param cb Callback to provide information and receive events
     * @return a new ViewDragHelper instance
     */
    public static ViewDragHelper create(ViewGroup forParent, float sensitivity, Callback cb) {
        final ViewDragHelper helper = create(forParent, cb);
        helper.mTouchSlop = (int) (helper.mTouchSlop * (1 / sensitivity));
        return helper;
    }

由: helper.mTouchSlop = (int) (helper.mTouchSlop * (1 / sensitivity)); 可以显而易见地看出,其实就是mTouchSlop 除以我们传入的sensitivity然后重新赋值。而这个mTouchSlop 是怎么来的呢? 接着看源码,发现是这一段代码进行赋值的:

final ViewConfiguration vc = ViewConfiguration.get(context);
mTouchSlop = vc.getScaledTouchSlop();

由此可见,mTouchSlop 它是获取的系统判定是否触发移动事件的阈值。即:单次移动大于这个值,才会判定是MOVE操作。

第三个参数为静态回调对象CallBack,我们接下来实现相关CallBack方法来操作拖拽的View。

第三步:

实现ViewDragHelper.Callback的相关方法。

/**
 * Created by 小嵩 on 2017/8/30.
 */

public class DraggableViewCallback extends ViewDragHelper.Callback {
   
    @Override
    public boolean tryCaptureView(View child, int pointerId) {
        return true;
    }
}

其中,ViewDragHelper.Callback 是一个内部静态抽象类, tryCaptureView是必须实现的方法,其余是可选重写的方法,一般来说,我们重写:
clampViewPositionHorizontalView
clampViewPositionVertical
onViewReleased
onViewPositionChanged

这四个方法即可。更多方法的详情及含义,可阅读DrageHelper — 源码深入解析(第二篇)

tryCaptureView,可用于自由判定哪个子控件可被拖拽,返回true代表可拖拽,false则禁止。

第四步:

分别在 clampViewPositionVertical 和clampViewPositionHorizontal 方法中对它的可滑动边界进行控制。left , top 分别为即将移动到的位置,比如我希望只在的水平方向移动,则进行如下处理:

   /**
     * 子控件水平方向位置改变时触发
     */
    @Override
    public int clampViewPositionHorizontal(View child, int left, int dx) {
        //屏蔽掉水平方向
        return 0;
    }

同时,若我们只希望子控件向下平移,则做以下处理:

   /**
     * 子控件竖直方向位置改变时触发
     */
    @Override
    public int clampViewPositionVertical(View child, int top, int dy) {
        //不能滑出顶部
        return Math.max(top, 0);
    }

第五步:

在onViewReleased 方法中获取移动距离,判断拖拽距离是否超过阈值。若超过阈值,则执行关闭动画,否则处理回弹,Callback完整代码如下:

package com.test.demo;

import android.support.v4.widget.ViewDragHelper;
import android.util.Log;
import android.view.View;
import android.view.ViewConfiguration;

/**
 * ViewDragHelper.Callback 拖拽事件监听回调
 *
 * @author 小嵩
 */
class DraggableViewCallback extends ViewDragHelper.Callback {

    private static final String TAG = "DraggableViewCallback";

    private static float Y_MIN_VELOCITY = 300;//竖直方向关闭最小值 px

    private MyDraggableView mDraggableView;


    public DraggableViewCallback(MyDraggableView draggableView) {
        this.mDraggableView = draggableView;
        Y_MIN_VELOCITY = mDraggableView.getHeight() / 3;
    }

    /**
     * 子控件位置改变时触发(包括X和Y轴方向)
     *
     * @param left position.
     * @param top  position.
     * @param dx   change in X position from the last call.
     * @param dy   change in Y position from the last call.
     */
    @Override
    public void onViewPositionChanged(View changedView, int left, int top, int dx, int dy) {
        mDraggableView.onViewPositionChanged(changedView, left, top, dx, dy);
    }

    /**
     * 子控件竖直方向位置改变时触发
     */
    @Override
    public int clampViewPositionVertical(View child, int top, int dy) {
        //不能滑出顶部
        return Math.max(top, 0);
    }

    /**
     * 子控件水平方向位置改变时触发
     */
    @Override
    public int clampViewPositionHorizontal(View child, int left, int dx) {
        //屏蔽掉水平方向
        return 0;
    }

    /**
     * 手指松开时触发
     *
     * @param releasedChild the captured child view now being released.
     * @param xVel          X velocity of the pointer as it left the screen in pixels per second.
     * @param yVel          Y velocity of the pointer as it left the screen in pixels per second.
     */
    @Override
    public void onViewReleased(View releasedChild, float xVel, float yVel) {
        super.onViewReleased(releasedChild, xVel, yVel);
        Log.d(TAG, "onViewReleased");
        int top = releasedChild.getTop(); //获取子控件Y值
        int left = releasedChild.getLeft(); //获取子控件X值

        if (Math.abs(left) <= Math.abs(top)) {//若为竖直滑动
            triggerOnReleaseActionsWhileVerticalDrag(top);
        }
    }

    @Override
    public boolean tryCaptureView(View view, int pointerId) {
        return true;
    }

    /**
     * 计算竖直方向的滑动
     */
    private void triggerOnReleaseActionsWhileVerticalDrag(float yVel) {
        if (yVel > 0 && yVel >= Y_MIN_VELOCITY) {
            mDraggableView.closedToBottom();
            Log.d(TAG, "ReleaseVerticalDrag" + ", closeToBottom");
        } else {
            mDraggableView.onReset();
            Log.d(TAG, "ReleaseVerticalDrag" + ", onReset");
        }
    }
}

第六步:

在自定义控件MyDraggableView中处理监听回调事件。手指松开时,会有两种情况:

1.当拖拽滑动距离未达到我们设定的值,则重置到原来位置:

    public void onReset() {
        Log.d(TAG, "onReset");
        viewDragHelper.settleCapturedViewAt(0, 0);
        ViewCompat.postInvalidateOnAnimation(this);
    }

2.拖拽滑动距离超过设定的值,滑向底部关闭:

    public void closedToBottom() {
        Log.d(TAG, "closedToBottom");
        if (viewDragHelper.smoothSlideViewTo(this, 0, getHeight())) {
            ViewCompat.postInvalidateOnAnimation(this);
            notifyClosedToBottomListener();
        }
    }

其中,我们重写了computeScroll 方法,以便在手指松开时,触发系统自动滑动。代码如下:

  @Override
    public void computeScroll() {
        if (viewDragHelper.continueSettling(true)) {
            ViewCompat.postInvalidateOnAnimation(this);
        }
    }

computeScroll的具体原理这里就不阐述了,不懂的话可以自行Google/百度 查找 View 的computeScroll 实现原理以及源码。

到这儿我们就已经实现了拖拽下拉关闭的功能了,效果演示如下:

下拉拖拽关闭.gif

类似地,如果我们需要实现向左或者向右拖拽回弹或者关闭的功能,只需要把ViewDragHelper.Callback里面clampViewPositionHorizontal以及clampViewPositionVertical方法稍加修改,然后在onViewReleased回调一下,执行viewDragHelper.smoothSlideViewTo()方法让View平顺移动到指定位置即可。具体实际情况可自行实践操作一波。

稍微修改一下代码,改成左右拖拽,代码如下:

    /**
     * 子控件竖直方向位置改变时触发
     */
    @Override
    public int clampViewPositionVertical(View child, int top, int dy) {
//        return Math.max(top, 0);//不能滑出顶部
        return 0;
    }

    /**
     * 子控件水平方向位置改变时触发
     */
    @Override
    public int clampViewPositionHorizontal(View child, int left, int dx) {
//        return 0;
        return left;
    }
    
    /**
     * 手指松开时触发
     */
      @Override
    public void onViewReleased(View releasedChild, float xVel, float yVel) {
        super.onViewReleased(releasedChild, xVel, yVel);
        Log.d(TAG, "onViewReleased");
        int top = releasedChild.getTop(); //获取子控件Y值
        int left = releasedChild.getLeft(); //获取子控件X值

        if (Math.abs(left) <= Math.abs(top)) {//若为竖直滑动
            triggerOnReleaseActionsWhileVerticalDrag(top);
        } else {
            triggerOnReleaseActionsWhileHorizontalDrag(left);
        }
    }
    
 /**
   * 计算水平方向
   */
 private void triggerOnReleaseActionsWhileHorizontalDrag(int xVel) {
        if (xVel > 0 && xVel >= X_MIN_VELOCITY) {
            mDraggableView.closedToRight();
            Log.d(TAG, "ReleaseVerticalDrag" + ", closedToRight");
        } else if (xVel < 0 && Math.abs(xVel) >= X_MIN_VELOCITY) {
            mDraggableView.closedToLeft();
            Log.d(TAG, "ReleaseVerticalDrag" + ", closedToLeft");
        } else {
            mDraggableView.onReset();
            Log.d(TAG, "ReleaseVerticalDrag" + ", onReset");
        }
    }

效果如下:

水平拖拽关闭.gif

结语:

读完这篇文章之后,若觉得有哪里写的不够详细或是有更多的建议,欢迎指出~ 也非常感谢各位的支持和收藏点赞。

下一篇将围绕源码进行解析它的运行原理以及所提供的方法,文章链接:ViewDragHelper (二)- 源码及原理解读(进阶篇)
这篇文章将会详细讲解ViewDragHelper它提供的方法所代表的含义,以及实现原理等。相信读完理解这篇文章的内容之后,对ViewDragHelper的基本操作会有一个更全面的理解。

(By the way,最近工作有点忙,一篇文章躺在草稿箱N久,零零散散抽时间总算出炉了。第二篇和第三篇后续会抽空抓紧赶时间写出来)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容