dispatchTouchEvent问题,用于将触摸事件传递给子级

时间:2015-11-19 09:45:41

标签: android android-linearlayout children overlapping touch-event

我正在尝试为棒棒糖前版本创建棒棒糖工具栏图标的效果。

enter image description here

我编写了这个扩展LinearLayout的类,并在dispatchTouchEvent中检查哪个子项更接近触摸位置并仅将该事件发送给该子项,以便忽略子项的重叠区域和z-index: / p>

public class OverlappingLayout extends LinearLayout {
 public OverlappingLayout(Context context) {
    super(context);
}

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

int lastAction = 0;
View lastView = null;

@Override
public boolean dispatchTouchEvent(MotionEvent ev) {

    if(lastView != null){

        lastView.dispatchTouchEvent(ev);

        if(ev.getAction() == MotionEvent.ACTION_CANCEL || ev.getAction() == MotionEvent.ACTION_UP)
            lastView = null;

        return true;
    }

    float min = 99999999f;
    int minIndex = 0;

    for (int i = 0; i < getChildCount(); i++) {

        View child = getChildAt(i);

        float centerX = child.getX() + child.getWidth() / 2;
        float centerY = child.getY() + child.getHeight() / 2;

        float dis = (float) Math.sqrt(
                (ev.getX() - centerX) *  (ev.getX() - centerX) +
                        (ev.getY() - centerY) *  (ev.getY() - centerY)
        );

        if(dis < min) {
            min = dis;
            minIndex = i;
        }
    }

    getChildAt(minIndex).setClickable(true);
    getChildAt(minIndex).dispatchTouchEvent(ev);

    lastView = getChildAt(minIndex);
    lastAction = ev.getAction();


    lastView.setOnTouchListener(new OnTouchListener() {
        @Override
        public boolean onTouch(View v, MotionEvent event) {

            //never works for the last child, or the right half of the second child


            return false;
        }
    });

    return true;
}
 }

但问题是代码在布局的前半部分工作正常,并且在下半部分不起作用。

enter image description here

代码在调试模式下工作,找到最近的子节点并将事件传递给它,但是子节点的drawable的按下状态不起作用。

我也把它改成了RelativeLayout,但仍然有同样的问题。

1 个答案:

答案 0 :(得分:0)

我弄清楚问题是什么。 当我将motionEvent对象发送给子节点时,触摸的x和y位于子节点的rect之外(如子节点所示)。所以它只适用于前半部分,因为第一个和第二个孩子看到了他们内部的触摸位置。所以我把我的代码更改为:

private Rect rect;
View lastView;

@Override
public boolean dispatchTouchEvent(MotionEvent event) {

    for (int i = 0; i < getChildCount(); i++) {
        getChildAt(i).setClickable(false);
    }

    if(lastView == null)
        lastView = getClosestChildToTouch(event);


    if (event.getAction() == MotionEvent.ACTION_DOWN) {

        lastView.setPressed(true);
        rect = new Rect(lastView.getLeft(), lastView.getTop(), lastView.getRight(), lastView.getBottom());
    }

    if (event.getAction() == MotionEvent.ACTION_UP) {

        if (!rect.contains((int) event.getX(), (int) event.getY())) {

            lastView.setPressed(false);
            lastView=null;

        }  else {

            lastView.performClick();
            lastView.setPressed(false);
            lastView=null;
        }
    }

    if(event.getAction() == MotionEvent.ACTION_MOVE){

        if(!rect.contains((int) event.getX(),(int) event.getY())){

            lastView.setPressed(false);
            lastView=null;
        }
    }

    if (event.getAction() == MotionEvent.ACTION_CANCEL){

        lastView.setPressed(false);
        lastView=null;
    }

    return true;
}


public View getClosestChildToTouch(MotionEvent ev){

    float min = 99999999f;
    int minIndex = 0;

    for (int i = 0; i < getChildCount(); i++) {

        View child = getChildAt(i);

        float centerX = child.getX() + child.getWidth() / 2;
        float centerY = child.getY() + child.getHeight() / 2;

        float dis = (float) Math.sqrt(
                (ev.getX() - centerX) *  (ev.getX() - centerX) +
                        (ev.getY() - centerY) *  (ev.getY() - centerY)
        );

        if(dis < min) {
            min = dis;
            minIndex = i;
        }
    }

    return getChildAt(minIndex);
}

我也可以修改motionEvent对象并将子的左侧或右侧位置添加到x属性。