附录文章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;
}