Android ViewDragHelper:控制子View能否拖曳及水平方向的拖曳边界(2)
附录文章1简单介绍了Android ViewDragHelper的使用,注意到附录文章1的代码运行结果,附录文章1的示例有三个子view,每个子view均可拖曳,但是,它们可以拖曳越出边界超出屏幕的显示范围,被拖曳到视野看不到的地方去了。
(1)在某些情况下,也许开发者不希望子view被拖曳到不可见的区域内,允许子view可以被拖曳,但不允许拖曳到不可见找不到它们!这些需求是可以代码控制。以水平拖曳为例,意图控制子view在水平方向视野可见区域内拖曳,不允许超出边界,那么就需要在:
clampViewPositionHorizontal()
里面添加约束条件。
水平拖曳的约束条件,并不是很复杂,只是需要了解设备屏幕的坐标体系。ViewGroup进行getWidth()获得的是该ViewGroup整个占据的屏幕坐标宽度,在我写的这个例子中,因为是MATCH_PARENT,且只有一个布局铺满整个屏幕,那么getWidth()就是完整的宽度。当用户在水平左边拖曳拖曳到极限位置(超出左边屏幕),那么此时clampViewPositionHorizontal()的left值为负数,意为越界,在这里下手,如果left值超出左边的边界,直接返回一个大于0的值即可。通常在实际的布局中处于设计美观的要求会padding一些值,刚好,如果此时getPaddingLeft(),获得的就是Android系统换算后左边的宽度,那么直接返回getPaddingLeft()值即可,如果没有padding值,返回一个0也可以。
右边的情况类似但稍微复杂些,考虑一种极限情况假设子view的右边刚好拖曳到父ViewGroup的右边完全重合,此时如果哪怕再往右拖曳1个坐标单位就要越界。那么此时可以得到如下等式:
getWidth() = pos + childView.getWidth()
pos为子view的左边坐标量。
如果父布局(ViewGroup)里面再padding一些值,那么等式变形为:
getWidth() = pos + childView.getWidth()+getPaddingRight()
多一个padding值而已。
计算得到的pos值即为最右边的极限位置,入股left > pos,那么此时就表明子view要越界了,立即返回pos值,这样就控制子view无法越界了。
(2)另外,通常一个布局文件里面有很多个子view,在某些情况下,可能开发者希望特定的子view不可被拖曳,完成这一些需求,则仅仅在tryCaptureView()里面针对特定的view返回false即可,false就是告诉Android系统,这个特定的子view是不被允许拖曳的。True则允许拖曳。
按照上述原理和思路修改增强附录1文章的MyLayout(其他代码不需改动):
package zhangphil.demo; import android.content.Context; import android.support.v4.widget.ViewDragHelper; import android.util.AttributeSet; import android.util.Log; import android.view.MotionEvent; import android.view.View; import android.widget.LinearLayout; /** * Created by Phil on 2016/4/15. */ public class MyLayout extends LinearLayout { private ViewDragHelper mViewDragHelper; public MyLayout(Context context, AttributeSet attrs) { super(context, attrs); init(); } private void init() { mViewDragHelper = ViewDragHelper.create(this, 1.0f, new ViewDragHelperCallback()); } private class ViewDragHelperCallback extends ViewDragHelper.Callback { @Override public boolean tryCaptureView(View view, int pointerId) { //假设我不希望红色的子view可以被拖曳,那就加一层判断,只要是特定的view,直接返回false,false告诉Android系统,这个子view是不允许拖曳操作的。 if (view.getId() == R.id.red) return false; return true; } @Override public int clampViewPositionHorizontal(View child, int left, int dx) { //控制左边的拖曳距离,不能越界。 //当拖曳的距离超过左边的padding值,也意味着child view越界,复位 //默认的padding值=0 int paddingleft = getPaddingLeft(); if (left < paddingleft) { return paddingleft; } //这里是控制右边的拖曳边缘极限位置。 //假设pos的值刚好是子view child右边边缘与父view的右边重合的情况 //pos值即为一个极限的最右边位置,超过也即意味着拖曳越界:越出右边的界限,复位。 //可以再加一个paddingRight值,缺省的paddingRight=0,所以即便不加也在多数情况正常可以工作 int pos = getWidth() - child.getWidth() - getPaddingRight(); if (left > pos) { return pos; } //其他情况属于在范围内的拖曳,直接返回系统计算默认的left即可 return left; } @Override public int clampViewPositionVertical(View child, int top, int dy) { return top; } @Override public void onViewDragStateChanged(int state) { /** switch (state) { case ViewDragHelper.STATE_DRAGGING: // 正在拖动 break; case ViewDragHelper.STATE_IDLE: // 没有被拖拽或者正在进行fling/snap break; case ViewDragHelper.STATE_SETTLING: // fling完毕后被放置到一个位置 break; } */ super.onViewDragStateChanged(state); } } @Override public boolean onInterceptTouchEvent(MotionEvent event) { return mViewDragHelper.shouldInterceptTouchEvent(event); } @Override public boolean onTouchEvent(MotionEvent event) { mViewDragHelper.processTouchEvent(event); return true; } }
代码运行结果,可以看到右上角的红色子view无法被拖到,剩余的子view控制在设计的区域范围内没有越界:
附录文章:
1,《Android ViewDragHelper(1)》链接地址:http://blog.csdn.net/zhangphil/article/details/51177588