Android:使用MotionEvent在自定义ViewGroup中拖放View的定位

时间:2011-11-26 00:00:07

标签: android android-layout drag-and-drop ontouchevent viewgroup

我无法在我创建的自定义ViewGroup上定位视图元素,特别是在拖放情况下。我的目标是Android 2.2及更高版本,因此我无法真正使用Android 3中出现的拖放式API。

我的自定义ViewGroup称为“NodeGrid”,它扩展了“RelativeLayout”。其 onLayout 方法被覆盖,使其如下所示:

@Override
protected void onLayout(boolean arg0, int arg1, int arg2, int arg3, int arg4) 
{
    //pxConversion is a multiplier to convert 1 dip to x pixels.
    //We will use it to convert dip units into px units.
    Resources r = getResources();
    float pxConversion = TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 1, r.getDisplayMetrics());

    //Layout the children of this viewgroup
    int childCount = this.getChildCount();
    for (int i = 0; i < childCount; i++)
    {
        NodeView view = (NodeView) this.getChildAt(i);

        if (view.DataContext != null)
        {
            int left = (int) (view.DataContext.get_left() * pxConversion);
            int top = (int) (view.DataContext.get_top() * pxConversion);
            int right = left + (int) (view.DataContext.get_width() * pxConversion);
            int bottom = top + (int) (view.DataContext.get_height() * pxConversion);

            view.layout(left, top, right, bottom);
        }
    }   
}

“NodeGrid”的子节点类型为“NodeView”(如上面的代码所示)。 NodeView只是我创建的自定义View类。它包含一个名为“DataContext”的成员,它是一个视图模型类,包含一些getter / setter,其中包含有关NodeGrid上NodeView实例的位置信息。

我的“NodeView”类捕获用户的触摸事件,因此用户只需将节点拖放到网格上的任何位置即可。事件处理程序如下所示:

@Override
public boolean onTouchEvent(MotionEvent event) {

    if (event.getAction() == MotionEvent.ACTION_DOWN)
    {
        isBeingDragged = true;
    }
    else if (event.getAction() == MotionEvent.ACTION_UP)
    {
        isBeingDragged = false;
    }
    else if (event.getAction() == MotionEvent.ACTION_MOVE)
    {
        if (DataContext != null && isBeingDragged)
        {
            float xPosition = event.getRawX();
            float yPosition = event.getRawY();

            DataContext.set_left(xPosition);
            DataContext.set_top(yPosition);

            this.requestLayout();
        }
    }

    return true;
}

我的问题是被拖动的视图没有像我期望的那样定位在我的“NodeGrid”上。我拖动时似乎x轴位置是正确的,但是y轴位置在任何时候都会被一些恒定的像素量偏移。会导致什么?我尝试过使用getX()和getY()方法而不是getRawX()和getRawY(),但这只会导致它变得更糟。

我不确定getRawX()和getRawY()是否正在向我返回px单位或dip单位,所以期待它给我回px单位,我尝试将它们转换为dip单位然后再分配新的x和y值到节点,但只减少了偏移量,它没有消除它。

可能导致我触摸的位置和节点所在位置的差异是什么?

1 个答案:

答案 0 :(得分:1)

非常惊讶的是,显然没有其他人遇到过这种情况......

经过一番研究,我已部分解决了这个问题。

getRawX()和getRawY()不起作用的原因是因为它们在绝对屏幕位置工作,而不考虑屏幕上不属于您应用程序的任何内容(例如Android菜单始终位于顶部屏幕除非您处于全屏模式)。使用这些坐标似乎很苛刻,特别是没有明显的方法来证明它们并将它们转换为“我的应用程序”坐标。

据我所知,

getX()和getY()似乎与被触摸的对象有关。考虑到这一点后,我将onTouchEvent中的代码更改为:

float xPosition = event.getX();
float yPosition = event.getY();

DataContext.set_left(DataContext.get_left() + xPosition);
DataContext.set_top(DataContext.get_top() + yPosition);

this.requestLayout();

这绝对有帮助......但它似乎仍然不是一个完美的解决方案。一般情况下,它会导致视图对象与我的鼠标光标/手指一起拖动,但它非常紧张,有时它可能会变得笨拙并离开屏幕。

然后我想也许是因为我没有将“历史坐标”纳入方法中。我尝试将历史坐标放入方法中,方法是将代码放在我刚才显示的代码之前:

int historySize = event.getHistorySize();
for (int i = 0; i < historySize; i++)
{
    float xPosition = event.getHistoricalX(i);
    float yPosition = event.getHistoricalY(i);

    DataContext.set_left(DataContext.get_left() + xPosition);
    DataContext.set_top(DataContext.get_top() + yPosition);

    this.requestLayout();
}

不幸的是......这实际上使情况变得更糟,所以要么我没有正确使用历史坐标,要么根本不应该使用它们。我将就此问题发表另一个问题,但这应该足以作为我上述问题的部分答案。