6

我创建了一个用于拖动视图的 onTouchListener。getRawX()如果我使用和 ,图像可以顺利拖动getRawY()。问题是当您将第二个指针向下放置然后抬起第一个指针时,图像将跳转到第二个指针。

这个 onTouchListener 试图通过跟踪pointerId. 这个 onTouchListener 的问题是在拖动 ImageView 时,ImageView 会非常疯狂地跳来跳去。和值跳来跳去getX()getY()

我觉得我这样做是正确的。我不想为此编写自定义视图,因为我已经实现了一个 scaleGestureDetector 并编写了一个可以工作的自定义 rotateGestureDetector。一切正常,但我需要解决使用getRawX()and时遇到的问题getRawY()

有人知道我在这里做错了什么吗?

这是我的 onTouchListener:

final View.OnTouchListener onTouchListener = new View.OnTouchListener()
{
    @Override
    public boolean onTouch(View v, MotionEvent event)
    {
        relativeLayoutParams = (RelativeLayout.LayoutParams) v.getLayoutParams();

        final int action = event.getAction();
        switch (action & MotionEvent.ACTION_MASK)
        {
            case MotionEvent.ACTION_DOWN:
            {
                final float x = event.getX();
                final float y = event.getY();

                // Where the user started the drag
                lastX = x;
                lastY = y;
                activePointerId = event.getPointerId(0);
                break;
            }
            case MotionEvent.ACTION_MOVE:
            {
                // Where the user's finger is during the drag
                final int pointerIndex = event.findPointerIndex(activePointerId);
                final float x = event.getX(pointerIndex);
                final float y = event.getY(pointerIndex);

                // Calculate change in x and change in y
                final float dx = x - lastX;
                final float dy = y - lastY;

                // Update the margins to move the view
                relativeLayoutParams.leftMargin += dx;
                relativeLayoutParams.topMargin += dy;
                v.setLayoutParams(relativeLayoutParams);

                // Save where the user's finger was for the next ACTION_MOVE
                lastX = x;
                lastY = y;

                v.invalidate();
                break;
            }
            case MotionEvent.ACTION_UP:
            {
                activePointerId = INVALID_POINTER_ID;
                break;
            }
            case MotionEvent.ACTION_CANCEL:
            {
                activePointerId = INVALID_POINTER_ID;
                break;
            }
            case MotionEvent.ACTION_POINTER_UP:
            {
                // Extract the index of the pointer that left the touch sensor
                final int pointerIndex = (action & MotionEvent.ACTION_POINTER_INDEX_MASK) >> MotionEvent.ACTION_POINTER_INDEX_SHIFT;
                final int pointerId = event.getPointerId(pointerIndex);

                if(pointerId == activePointerId)
                {
                    // This was our active pointer going up. Choose a new
                    // active pointer and adjust accordingly
                    final int newPointerIndex = pointerIndex == 0 ? 1 : 0;
                    lastX = (int) event.getX(newPointerIndex);
                    lastY = (int) event.getY(newPointerIndex);
                    activePointerId = event.getPointerId(newPointerIndex);
                }
                break;
            }
        }
        return true;
    }
};
image1.setOnTouchListener(onTouchListener);
4

3 回答 3

12

这个问题很简单,但出乎意料。getRawX/Y()返回绝对坐标,同时getX/Y()返回相对于视图的坐标。我会移动视图,重置lastX/Y,并且图像将不再位于同一位置,因此当我获得新值时它们会关闭。在这种情况下,我只需要最初按下图像的位置(使用 `getRawX/Y' 时不需要)。

因此,解决方案是简单地删除以下内容:

// Save where the user's finger was for the next ACTION_MOVE
lastX = x;
lastY = y;

我希望这对将来的人有所帮助,因为我见过其他人遇到这个问题,他们有与我相似的代码(重置lastX/Y

于 2013-07-09T13:28:11.710 回答
1

经过大量研究,我发现这是问题所在,getRawX 是绝对的,getX 是相对于视图的。因此使用它来将一个转换为另一个

//RawX = getX + View.getX

event.getRawX == event.getX(event.findPointerIndex(ptrID1))+view.getX()
于 2016-07-18T13:24:23.797 回答
0

我有一个提示并认为它会有所帮助。

invalidate() : does only redrawing a view,doesn't change view size/position.

您应该使用的是 requestLayout():进行测量和布局过程。&我认为requestLayout()会在通话时被调用setLayoutParams()

所以尝试删除v.invalidate()

或尝试使用view.layout(left,top,right,bottom)方法而不是设置 layoutParams。

于 2013-07-08T17:01:11.887 回答