2013-12-20 136 views
5

我在網上搜索了一個簡單的解決方案,以自由移動ImageView。我終於找到了一些代碼,產生完美的結果:Android移動ImageView

public class MainActivity extends Activity implements View.OnTouchListener { 

    private int _xDelta; 
    private int _yDelta; 

    @Override 
    public void onCreate(Bundle savedInstanceState) { 
     super.onCreate(savedInstanceState); 
     setContentView(R.layout.activity_main); 

     ImageView j = (ImageView)findViewById(R.id.j); 

     j.setOnTouchListener(this); 

    } 

    public boolean onTouch(View view, MotionEvent event) { 
     final int X = (int) event.getRawX(); 
     final int Y = (int) event.getRawY(); 
     ImageView j = (ImageView)findViewById(R.id.j); 
     switch (event.getAction() & MotionEvent.ACTION_MASK) { 
      case MotionEvent.ACTION_DOWN: 
       RelativeLayout.LayoutParams lParams = (RelativeLayout.LayoutParams) view.getLayoutParams(); 
       _xDelta = (int) (X - j.getTranslationX()); 
       _yDelta = (int) (Y - j.getTranslationY()); 
       break; 
      case MotionEvent.ACTION_UP: 
       break; 
      case MotionEvent.ACTION_POINTER_DOWN: 
       break; 
      case MotionEvent.ACTION_POINTER_UP: 
       break; 
      case MotionEvent.ACTION_MOVE: 
       RelativeLayout.LayoutParams layoutParams = (RelativeLayout.LayoutParams) view.getLayoutParams(); 

       j.setTranslationX(X - _xDelta); 
       j.setTranslationY(Y - _yDelta); 
       break; 
     } 

     return true; 
    }} 

因爲我真的不知道我的代碼是如何工作的,我想看看是否有更好的解決方案。

回答

1

這是我得到了解決:

private float xCoOrdinate, yCoOrdinate; 

     view.setOnTouchListener(new View.OnTouchListener() { 

      Float y1 = 0f, y2 = 0f; 

      @Override 
      public boolean onTouch(View v, MotionEvent event) { 

       Point size = new Point(); 
      getWindowManager().getDefaultDisplay().getSize(size);   

       switch (event.getAction()) { 
        case MotionEvent.ACTION_DOWN: 
         //xCoOrdinate = view.getX() - event.getRawX(); 
         yCoOrdinate = view.getY() - event.getRawY(); 
         Float puffer = 0f; 
         y1 = event.getRawY(); 

         break; 
        case MotionEvent.ACTION_MOVE: 
         view.animate().y(event.getRawY() + yCoOrdinate).setDuration(0).start(); 
         //view.animate().x(event.getRawX() + xCoOrdinate).y(event.getRawY() + yCoOrdinate).setDuration(0).start(); 
         break; 
        case MotionEvent.ACTION_UP: 
         y2 = event.getRawY(); 
         Float dy = (y2 - y1); 


         if (dy < 0) { 
          //moved up 
          view.animate().y(size.y/100 * -40).setDuration(100).start(); 
         } else if (dy > 0) { 
          // moved down 
          view.animate().y(size.y/2 - view.getHeight()/2).setDuration(100).start(); 
         } 

         break; 
       } 

       return false; 
      } 
     }); 

說明:如果向上或向下移動,我只需要在y軸和dectection運動。它通過將視圖動畫化爲觸覺而減弱,女巫具有優點和缺點(相反:它很複雜地檢測視角在空間/專業中的位置:視圖實際上移動並且看起來似乎不在其他地方),但最終效果最好我很簡單。 這裏的目標是向上滑動視圖並將其滑動到所需位置,與向下滑動相同,回到原始位置,這是相對於屏幕尺寸實現的。 對我來說,代碼是相對簡單和自我犧牲的,但請詢問是否有任何不清楚的地方。

3

如Google所示,如果您的目標Android版本3.0及以上,您可以使用DragListener

對於您可以使用類似蜂巢之前的版本:

// The ‘active pointer’ is the one currently moving our object. 
private int mActivePointerId = INVALID_POINTER_ID; 

@Override 
public boolean onTouchEvent(MotionEvent ev) { 
    // Let the ScaleGestureDetector inspect all events. 
    mScaleDetector.onTouchEvent(ev); 

    final int action = MotionEventCompat.getActionMasked(ev); 

    switch (action) { 
    case MotionEvent.ACTION_DOWN: { 
     final int pointerIndex = MotionEventCompat.getActionIndex(ev); 
     final float x = MotionEventCompat.getX(ev, pointerIndex); 
     final float y = MotionEventCompat.getY(ev, pointerIndex); 

     // Remember where we started (for dragging) 
     mLastTouchX = x; 
     mLastTouchY = y; 
     // Save the ID of this pointer (for dragging) 
     mActivePointerId = MotionEventCompat.getPointerId(ev, 0); 
     break; 
    } 

    case MotionEvent.ACTION_MOVE: { 
     // Find the index of the active pointer and fetch its position 
     final int pointerIndex = 
       MotionEventCompat.findPointerIndex(ev, mActivePointerId); 

     final float x = MotionEventCompat.getX(ev, pointerIndex); 
     final float y = MotionEventCompat.getY(ev, pointerIndex); 

     // Calculate the distance moved 
     final float dx = x - mLastTouchX; 
     final float dy = y - mLastTouchY; 

     mPosX += dx; 
     mPosY += dy; 

     invalidate(); 

     // Remember this touch position for the next move event 
     mLastTouchX = x; 
     mLastTouchY = y; 

     break; 
    } 

    case MotionEvent.ACTION_UP: { 
     mActivePointerId = INVALID_POINTER_ID; 
     break; 
    } 

    case MotionEvent.ACTION_CANCEL: { 
     mActivePointerId = INVALID_POINTER_ID; 
     break; 
    } 

    case MotionEvent.ACTION_POINTER_UP: { 

     final int pointerIndex = MotionEventCompat.getActionIndex(ev); 
     final int pointerId = MotionEventCompat.getPointerId(ev, pointerIndex); 

     if (pointerId == mActivePointerId) { 
      // This was our active pointer going up. Choose a new 
      // active pointer and adjust accordingly. 
      final int newPointerIndex = pointerIndex == 0 ? 1 : 0; 
      mLastTouchX = MotionEventCompat.getX(ev, newPointerIndex); 
      mLastTouchY = MotionEventCompat.getY(ev, newPointerIndex); 
      mActivePointerId = MotionEventCompat.getPointerId(ev, newPointerIndex); 
     } 
     break; 
    } 
    }  
    return true; 
} 

如再次通過谷歌here建議。

0

即工作液

public boolean onTouch(View view, MotionEvent event) { 
final int X = (int) event.getRawX(); 
final int Y = (int) event.getRawY(); 
switch (event.getAction() & MotionEvent.ACTION_MASK) { 
case MotionEvent.ACTION_DOWN: 
    RelativeLayout.LayoutParams lParams = (RelativeLayout.LayoutParams) view 
      .getLayoutParams(); 

    _xDelta = X - lParams.leftMargin; 
    _yDelta = Y - lParams.topMargin; 
    break; 
case MotionEvent.ACTION_UP: 
    break; 
case MotionEvent.ACTION_POINTER_DOWN: 
    break; 
case MotionEvent.ACTION_POINTER_UP: 
    break; 
case MotionEvent.ACTION_MOVE: 
    RelativeLayout.LayoutParams ParamsA = (RelativeLayout.LayoutParams) view 
      .getLayoutParams(); 
    ParamsA.leftMargin = X - _xDelta; 
    ParamsA.topMargin = Y - _yDelta; 
    ParamsA.rightMargin = -250; 
    ParamsA.bottomMargin = -250; 

    for (int i = 0; i < balls.size(); i++) { 
     if (balls.get(i).getTag() != view.getTag()) { 
      RelativeLayout.LayoutParams ParamsB = (RelativeLayout.LayoutParams) balls 
        .get(i).getLayoutParams(); 

      Rect b = new Rect(ParamsB.leftMargin,ParamsB.topMargin,ParamsB.rightMargin,ParamsB.bottomMargin); 
      Rect a = new Rect(ParamsA.leftMargin,ParamsA.topMargin,ParamsA.rightMargin,ParamsA.bottomMargin); 


      if(a.intersect(b)) 
      { 
       Toast.makeText(getApplicationContext(), "Collision Detected", Toast.LENGTH_SHORT).show(); 
      } 
     } 

    } 

    view.setLayoutParams(ParamsA); 
    break; 
} 
// _root.invalidate(); 
return true; 
} 

歡呼。