blob: 8ad5c7ce57e9a8fe8d4130bfd67ef2d6cb8bc359 [file] [log] [blame]
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001/*
2 * Copyright (C) 2008 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Joe Onoratoa5902522009-07-30 13:37:37 -070017package com.android.launcher2;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080018
Joe Onorato00acb122009-08-04 16:04:30 -040019import android.content.Context;
20import android.graphics.Bitmap;
Joe Onorato00acb122009-08-04 16:04:30 -040021import android.graphics.Rect;
22import android.graphics.RectF;
Joe Onorato00acb122009-08-04 16:04:30 -040023import android.os.Handler;
Michael Jurka0280c3b2010-09-17 15:00:07 -070024import android.os.IBinder;
Joe Onorato00acb122009-08-04 16:04:30 -040025import android.os.Vibrator;
Joe Onoratoe048e8a2009-09-25 10:39:17 -070026import android.util.DisplayMetrics;
Joe Onorato00acb122009-08-04 16:04:30 -040027import android.util.Log;
Joe Onorato00acb122009-08-04 16:04:30 -040028import android.view.KeyEvent;
29import android.view.MotionEvent;
Michael Jurka0280c3b2010-09-17 15:00:07 -070030import android.view.View;
Patrick Dubroya16fd5a2010-10-07 16:47:28 -070031import android.view.ViewConfiguration;
Joe Onoratoe048e8a2009-09-25 10:39:17 -070032import android.view.WindowManager;
Joe Onorato00acb122009-08-04 16:04:30 -040033import android.view.inputmethod.InputMethodManager;
Joe Onorato00acb122009-08-04 16:04:30 -040034
Adam Cohen120980b2010-12-08 11:05:37 -080035import com.android.launcher.R;
Adam Cohenc0dcf592011-06-01 15:30:43 -070036
37import java.util.ArrayList;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080038
39/**
Joe Onorato00acb122009-08-04 16:04:30 -040040 * Class for initiating a drag within a view or across multiple views.
The Android Open Source Project31dd5032009-03-03 19:32:27 -080041 */
Joe Onorato00acb122009-08-04 16:04:30 -040042public class DragController {
Romain Guyea3763c2010-01-11 18:02:04 -080043 @SuppressWarnings({"UnusedDeclaration"})
Joe Onorato2e5c4322009-10-06 12:34:42 -070044 private static final String TAG = "Launcher.DragController";
45
Joe Onorato00acb122009-08-04 16:04:30 -040046 /** Indicates the drag is a move. */
47 public static int DRAG_ACTION_MOVE = 0;
48
49 /** Indicates the drag is a copy. */
50 public static int DRAG_ACTION_COPY = 1;
51
52 private static final int SCROLL_DELAY = 600;
Joe Onorato00acb122009-08-04 16:04:30 -040053 private static final int VIBRATE_DURATION = 35;
54
55 private static final boolean PROFILE_DRAWING_DURING_DRAG = false;
56
57 private static final int SCROLL_OUTSIDE_ZONE = 0;
58 private static final int SCROLL_WAITING_IN_ZONE = 1;
59
Patrick Dubroy54fa3b92010-11-17 12:18:45 -080060 static final int SCROLL_NONE = -1;
Patrick Dubroy1262e362010-10-06 15:49:50 -070061 static final int SCROLL_LEFT = 0;
62 static final int SCROLL_RIGHT = 1;
Joe Onorato00acb122009-08-04 16:04:30 -040063
64 private Context mContext;
65 private Handler mHandler;
66 private final Vibrator mVibrator = new Vibrator();
67
68 // temporaries to avoid gc thrash
69 private Rect mRectTemp = new Rect();
70 private final int[] mCoordinatesTemp = new int[2];
71
72 /** Whether or not we're dragging. */
73 private boolean mDragging;
74
75 /** X coordinate of the down event. */
Adam Cohene3e27a82011-04-15 12:07:39 -070076 private int mMotionDownX;
Joe Onorato00acb122009-08-04 16:04:30 -040077
78 /** Y coordinate of the down event. */
Adam Cohene3e27a82011-04-15 12:07:39 -070079 private int mMotionDownY;
Joe Onorato00acb122009-08-04 16:04:30 -040080
Joe Onoratoe048e8a2009-09-25 10:39:17 -070081 /** Info about the screen for clamping. */
82 private DisplayMetrics mDisplayMetrics = new DisplayMetrics();
83
Joe Onorato00acb122009-08-04 16:04:30 -040084 /** Original view that is being dragged. */
85 private View mOriginator;
86
Joe Onorato658db742010-09-29 11:40:39 -070087 /** the area at the edge of the screen that makes the workspace go left
88 * or right while you're dragging.
89 */
90 private int mScrollZone;
91
Adam Cohencb3382b2011-05-24 14:07:08 -070092 private DropTarget.DragObject mDragObject = new DropTarget.DragObject();
Joe Onorato00acb122009-08-04 16:04:30 -040093
94 /** Who can receive drop events */
95 private ArrayList<DropTarget> mDropTargets = new ArrayList<DropTarget>();
96
Patrick Dubroy4ed62782010-08-17 15:11:18 -070097 private ArrayList<DragListener> mListeners = new ArrayList<DragListener>();
Joe Onorato00acb122009-08-04 16:04:30 -040098
99 /** The window token used as the parent for the DragView. */
100 private IBinder mWindowToken;
101
102 /** The view that will be scrolled when dragging to the left and right edges of the screen. */
103 private View mScrollView;
104
Romain Guyea3763c2010-01-11 18:02:04 -0800105 private View mMoveTarget;
106
Joe Onorato00acb122009-08-04 16:04:30 -0400107 private DragScroller mDragScroller;
108 private int mScrollState = SCROLL_OUTSIDE_ZONE;
109 private ScrollRunnable mScrollRunnable = new ScrollRunnable();
110
111 private RectF mDeleteRegion;
112 private DropTarget mLastDropTarget;
113
114 private InputMethodManager mInputMethodManager;
115
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700116 private int mLastTouch[] = new int[2];
117 private int mDistanceSinceScroll = 0;
118
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800119 /**
120 * Interface to receive notifications when a drag starts or stops
121 */
122 interface DragListener {
123
124 /**
125 * A drag has begun
126 *
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800127 * @param source An object representing where the drag originated
128 * @param info The data associated with the object that is being dragged
129 * @param dragAction The drag action: either {@link DragController#DRAG_ACTION_MOVE}
130 * or {@link DragController#DRAG_ACTION_COPY}
131 */
Joe Onorato5162ea92009-09-03 09:39:42 -0700132 void onDragStart(DragSource source, Object info, int dragAction);
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800133
134 /**
Winson Chunge3193b92010-09-10 11:44:42 -0700135 * The drag has ended
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800136 */
137 void onDragEnd();
138 }
139
140 /**
Joe Onorato00acb122009-08-04 16:04:30 -0400141 * Used to create a new DragLayer from XML.
142 *
143 * @param context The application's context.
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800144 */
Joe Onorato00acb122009-08-04 16:04:30 -0400145 public DragController(Context context) {
146 mContext = context;
147 mHandler = new Handler();
Joe Onorato658db742010-09-29 11:40:39 -0700148 mScrollZone = context.getResources().getDimensionPixelSize(R.dimen.scroll_zone);
Joe Onorato00acb122009-08-04 16:04:30 -0400149 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800150
Patrick Dubroy1262e362010-10-06 15:49:50 -0700151 public boolean dragging() {
152 return mDragging;
153 }
154
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800155 /**
Joe Onorato5162ea92009-09-03 09:39:42 -0700156 * Starts a drag.
Michael Jurkaa63c4522010-08-19 13:52:27 -0700157 *
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800158 * @param v The view that is being dragged
159 * @param source An object representing where the drag originated
Romain Guyea3763c2010-01-11 18:02:04 -0800160 * @param dragInfo The data associated with the object that is being dragged
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800161 * @param dragAction The drag action: either {@link #DRAG_ACTION_MOVE} or
162 * {@link #DRAG_ACTION_COPY}
163 */
Joe Onorato00acb122009-08-04 16:04:30 -0400164 public void startDrag(View v, DragSource source, Object dragInfo, int dragAction) {
Michael Jurkaa63c4522010-08-19 13:52:27 -0700165 startDrag(v, source, dragInfo, dragAction, null);
166 }
167
168 /**
169 * Starts a drag.
170 *
171 * @param v The view that is being dragged
172 * @param source An object representing where the drag originated
173 * @param dragInfo The data associated with the object that is being dragged
174 * @param dragAction The drag action: either {@link #DRAG_ACTION_MOVE} or
175 * {@link #DRAG_ACTION_COPY}
176 * @param dragRegion Coordinates within the bitmap b for the position of item being dragged.
177 * Makes dragging feel more precise, e.g. you can clip out a transparent border
178 */
179 public void startDrag(View v, DragSource source, Object dragInfo, int dragAction,
180 Rect dragRegion) {
Joe Onorato5162ea92009-09-03 09:39:42 -0700181 mOriginator = v;
182
183 Bitmap b = getViewBitmap(v);
184
Daniel Sandler3f8175a2010-05-25 11:48:32 -0400185 if (b == null) {
186 // out of memory?
187 return;
188 }
189
Joe Onorato5162ea92009-09-03 09:39:42 -0700190 int[] loc = mCoordinatesTemp;
191 v.getLocationOnScreen(loc);
192 int screenX = loc[0];
193 int screenY = loc[1];
194
Patrick Dubroy5f445422011-02-18 14:35:21 -0800195 startDrag(b, screenX, screenY, source, dragInfo, dragAction, dragRegion);
Joe Onorato5162ea92009-09-03 09:39:42 -0700196 b.recycle();
197
198 if (dragAction == DRAG_ACTION_MOVE) {
199 v.setVisibility(View.GONE);
200 }
201 }
202
203 /**
204 * Starts a drag.
Michael Jurkaa63c4522010-08-19 13:52:27 -0700205 *
Winson Chunge3193b92010-09-10 11:44:42 -0700206 * @param v The view that is being dragged
207 * @param bmp The bitmap that represents the view being dragged
208 * @param source An object representing where the drag originated
209 * @param dragInfo The data associated with the object that is being dragged
210 * @param dragAction The drag action: either {@link #DRAG_ACTION_MOVE} or
211 * {@link #DRAG_ACTION_COPY}
212 * @param dragRegion Coordinates within the bitmap b for the position of item being dragged.
213 * Makes dragging feel more precise, e.g. you can clip out a transparent border
214 */
215 public void startDrag(View v, Bitmap bmp, DragSource source, Object dragInfo, int dragAction,
216 Rect dragRegion) {
217 mOriginator = v;
218
219 int[] loc = mCoordinatesTemp;
220 v.getLocationOnScreen(loc);
221 int screenX = loc[0];
222 int screenY = loc[1];
223
Patrick Dubroy5f445422011-02-18 14:35:21 -0800224 startDrag(bmp, screenX, screenY, source, dragInfo, dragAction, dragRegion);
Winson Chunge3193b92010-09-10 11:44:42 -0700225
226 if (dragAction == DRAG_ACTION_MOVE) {
227 v.setVisibility(View.GONE);
228 }
229 }
230
231 /**
232 * Starts a drag.
233 *
Joe Onorato5162ea92009-09-03 09:39:42 -0700234 * @param b The bitmap to display as the drag image. It will be re-scaled to the
235 * enlarged size.
236 * @param screenX The x position on screen of the left-top of the bitmap.
237 * @param screenY The y position on screen of the left-top of the bitmap.
Joe Onorato5162ea92009-09-03 09:39:42 -0700238 * @param source An object representing where the drag originated
Romain Guyea3763c2010-01-11 18:02:04 -0800239 * @param dragInfo The data associated with the object that is being dragged
Joe Onorato5162ea92009-09-03 09:39:42 -0700240 * @param dragAction The drag action: either {@link #DRAG_ACTION_MOVE} or
241 * {@link #DRAG_ACTION_COPY}
242 */
243 public void startDrag(Bitmap b, int screenX, int screenY,
Joe Onorato5162ea92009-09-03 09:39:42 -0700244 DragSource source, Object dragInfo, int dragAction) {
Patrick Dubroy5f445422011-02-18 14:35:21 -0800245 startDrag(b, screenX, screenY, source, dragInfo, dragAction, null);
Michael Jurkaa63c4522010-08-19 13:52:27 -0700246 }
247
248 /**
249 * Starts a drag.
250 *
251 * @param b The bitmap to display as the drag image. It will be re-scaled to the
252 * enlarged size.
253 * @param screenX The x position on screen of the left-top of the bitmap.
254 * @param screenY The y position on screen of the left-top of the bitmap.
Michael Jurkaa63c4522010-08-19 13:52:27 -0700255 * @param source An object representing where the drag originated
256 * @param dragInfo The data associated with the object that is being dragged
257 * @param dragAction The drag action: either {@link #DRAG_ACTION_MOVE} or
258 * {@link #DRAG_ACTION_COPY}
259 * @param dragRegion Coordinates within the bitmap b for the position of item being dragged.
260 * Makes dragging feel more precise, e.g. you can clip out a transparent border
261 */
262 public void startDrag(Bitmap b, int screenX, int screenY,
Michael Jurkaa63c4522010-08-19 13:52:27 -0700263 DragSource source, Object dragInfo, int dragAction, Rect dragRegion) {
Joe Onorato00acb122009-08-04 16:04:30 -0400264 if (PROFILE_DRAWING_DURING_DRAG) {
265 android.os.Debug.startMethodTracing("Launcher");
266 }
267
268 // Hide soft keyboard, if visible
269 if (mInputMethodManager == null) {
270 mInputMethodManager = (InputMethodManager)
271 mContext.getSystemService(Context.INPUT_METHOD_SERVICE);
272 }
273 mInputMethodManager.hideSoftInputFromWindow(mWindowToken, 0);
274
Patrick Dubroy4ed62782010-08-17 15:11:18 -0700275 for (DragListener listener : mListeners) {
276 listener.onDragStart(source, dragInfo, dragAction);
Joe Onorato00acb122009-08-04 16:04:30 -0400277 }
278
Adam Cohene3e27a82011-04-15 12:07:39 -0700279 final int registrationX = ((int)mMotionDownX) - screenX;
280 final int registrationY = ((int)mMotionDownY) - screenY;
Joe Onorato00acb122009-08-04 16:04:30 -0400281
Michael Jurkaa63c4522010-08-19 13:52:27 -0700282 final int dragRegionLeft = dragRegion == null ? 0 : dragRegion.left;
283 final int dragRegionTop = dragRegion == null ? 0 : dragRegion.top;
Adam Cohene3e27a82011-04-15 12:07:39 -0700284
Joe Onorato00acb122009-08-04 16:04:30 -0400285 mDragging = true;
Adam Cohencb3382b2011-05-24 14:07:08 -0700286
287 mDragObject.xOffset = mMotionDownX - (screenX + dragRegionLeft);
288 mDragObject.yOffset = mMotionDownY - (screenY + dragRegionTop);
289 mDragObject.dragSource = source;
290 mDragObject.dragInfo = dragInfo;
Joe Onorato00acb122009-08-04 16:04:30 -0400291
292 mVibrator.vibrate(VIBRATE_DURATION);
293
Adam Cohencb3382b2011-05-24 14:07:08 -0700294 final DragView dragView = mDragObject.dragView = new DragView(mContext, b, registrationX,
295 registrationY, 0, 0, b.getWidth(), b.getHeight());
Michael Jurkaa63c4522010-08-19 13:52:27 -0700296
Patrick Dubroya669d792010-11-23 14:40:33 -0800297 final DragSource dragSource = source;
298 dragView.setOnDrawRunnable(new Runnable() {
299 public void run() {
300 dragSource.onDragViewVisible();
301 };
302 });
303
Michael Jurkaa63c4522010-08-19 13:52:27 -0700304 if (dragRegion != null) {
Adam Cohene3e27a82011-04-15 12:07:39 -0700305 dragView.setDragRegion(new Rect(dragRegion));
Michael Jurkaa63c4522010-08-19 13:52:27 -0700306 }
307
Joe Onorato00acb122009-08-04 16:04:30 -0400308 dragView.show(mWindowToken, (int)mMotionDownX, (int)mMotionDownY);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700309
310 handleMoveEvent((int) mMotionDownX, (int) mMotionDownY);
Joe Onorato00acb122009-08-04 16:04:30 -0400311 }
312
313 /**
314 * Draw the view into a bitmap.
315 */
Adam Cohen120980b2010-12-08 11:05:37 -0800316 Bitmap getViewBitmap(View v) {
Joe Onorato00acb122009-08-04 16:04:30 -0400317 v.clearFocus();
318 v.setPressed(false);
319
320 boolean willNotCache = v.willNotCacheDrawing();
321 v.setWillNotCacheDrawing(false);
322
323 // Reset the drawing cache background color to fully transparent
324 // for the duration of this operation
325 int color = v.getDrawingCacheBackgroundColor();
326 v.setDrawingCacheBackgroundColor(0);
Adam Cohen120980b2010-12-08 11:05:37 -0800327 float alpha = v.getAlpha();
328 v.setAlpha(1.0f);
Joe Onorato00acb122009-08-04 16:04:30 -0400329
330 if (color != 0) {
331 v.destroyDrawingCache();
332 }
333 v.buildDrawingCache();
334 Bitmap cacheBitmap = v.getDrawingCache();
Daniel Sandler3f8175a2010-05-25 11:48:32 -0400335 if (cacheBitmap == null) {
336 Log.e(TAG, "failed getViewBitmap(" + v + ")", new RuntimeException());
337 return null;
338 }
Joe Onorato00acb122009-08-04 16:04:30 -0400339
340 Bitmap bitmap = Bitmap.createBitmap(cacheBitmap);
341
342 // Restore the view
343 v.destroyDrawingCache();
Adam Cohen120980b2010-12-08 11:05:37 -0800344 v.setAlpha(alpha);
Joe Onorato00acb122009-08-04 16:04:30 -0400345 v.setWillNotCacheDrawing(willNotCache);
346 v.setDrawingCacheBackgroundColor(color);
347
348 return bitmap;
349 }
350
351 /**
352 * Call this from a drag source view like this:
353 *
354 * <pre>
355 * @Override
356 * public boolean dispatchKeyEvent(KeyEvent event) {
357 * return mDragController.dispatchKeyEvent(this, event)
358 * || super.dispatchKeyEvent(event);
359 * </pre>
360 */
Romain Guyea3763c2010-01-11 18:02:04 -0800361 @SuppressWarnings({"UnusedDeclaration"})
Joe Onorato00acb122009-08-04 16:04:30 -0400362 public boolean dispatchKeyEvent(KeyEvent event) {
363 return mDragging;
364 }
365
Winson Chung304dcde2011-01-07 11:17:23 -0800366 public boolean isDragging() {
367 return mDragging;
368 }
369
Joe Onorato24b6fd82009-11-12 13:47:09 -0800370 /**
371 * Stop dragging without dropping.
372 */
373 public void cancelDrag() {
Winson Chung621e6402011-01-04 16:03:57 -0800374 if (mDragging) {
Patrick Dubroye3887cc2011-01-20 10:43:40 -0800375 // Should we also be calling onDragExit() here?
Adam Cohenc0dcf592011-06-01 15:30:43 -0700376 mDragObject.dragSource.onDropCompleted(null, mDragObject, false);
Winson Chung621e6402011-01-04 16:03:57 -0800377 }
Joe Onorato24b6fd82009-11-12 13:47:09 -0800378 endDrag();
379 }
380
Joe Onorato00acb122009-08-04 16:04:30 -0400381 private void endDrag() {
382 if (mDragging) {
383 mDragging = false;
384 if (mOriginator != null) {
385 mOriginator.setVisibility(View.VISIBLE);
386 }
Patrick Dubroy4ed62782010-08-17 15:11:18 -0700387 for (DragListener listener : mListeners) {
388 listener.onDragEnd();
Joe Onorato00acb122009-08-04 16:04:30 -0400389 }
Adam Cohencb3382b2011-05-24 14:07:08 -0700390 if (mDragObject.dragView != null) {
391 mDragObject.dragView.remove();
392 mDragObject.dragView = null;
Joe Onorato00acb122009-08-04 16:04:30 -0400393 }
Joe Onorato00acb122009-08-04 16:04:30 -0400394 }
395 }
396
397 /**
398 * Call this from a drag source view.
399 */
400 public boolean onInterceptTouchEvent(MotionEvent ev) {
Joe Onorato9c1289c2009-08-17 11:03:03 -0400401 if (false) {
Joe Onoratoa30ce8e2009-11-11 08:16:49 -0800402 Log.d(Launcher.TAG, "DragController.onInterceptTouchEvent " + ev + " mDragging="
Joe Onorato9c1289c2009-08-17 11:03:03 -0400403 + mDragging);
404 }
Joe Onorato00acb122009-08-04 16:04:30 -0400405 final int action = ev.getAction();
406
Joe Onorato87467d32009-11-08 14:36:43 -0500407 if (action == MotionEvent.ACTION_DOWN) {
408 recordScreenSize();
409 }
410
Joe Onoratoe048e8a2009-09-25 10:39:17 -0700411 final int screenX = clamp((int)ev.getRawX(), 0, mDisplayMetrics.widthPixels);
412 final int screenY = clamp((int)ev.getRawY(), 0, mDisplayMetrics.heightPixels);
Joe Onorato00acb122009-08-04 16:04:30 -0400413
414 switch (action) {
415 case MotionEvent.ACTION_MOVE:
416 break;
Joe Onorato00acb122009-08-04 16:04:30 -0400417 case MotionEvent.ACTION_DOWN:
418 // Remember location of down touch
419 mMotionDownX = screenX;
420 mMotionDownY = screenY;
421 mLastDropTarget = null;
422 break;
Joe Onorato00acb122009-08-04 16:04:30 -0400423 case MotionEvent.ACTION_UP:
424 if (mDragging) {
425 drop(screenX, screenY);
426 }
427 endDrag();
428 break;
Winson Chung621e6402011-01-04 16:03:57 -0800429 case MotionEvent.ACTION_CANCEL:
430 cancelDrag();
431 break;
Joe Onorato00acb122009-08-04 16:04:30 -0400432 }
433
434 return mDragging;
435 }
436
437 /**
Romain Guyea3763c2010-01-11 18:02:04 -0800438 * Sets the view that should handle move events.
439 */
440 void setMoveTarget(View view) {
441 mMoveTarget = view;
442 }
443
444 public boolean dispatchUnhandledMove(View focused, int direction) {
445 return mMoveTarget != null && mMoveTarget.dispatchUnhandledMove(focused, direction);
446 }
447
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700448 private void handleMoveEvent(int x, int y) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700449 mDragObject.dragView.move(x, y);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700450
451 // Drop on someone?
452 final int[] coordinates = mCoordinatesTemp;
453 DropTarget dropTarget = findDropTarget(x, y, coordinates);
Adam Cohencb3382b2011-05-24 14:07:08 -0700454 mDragObject.x = coordinates[0];
455 mDragObject.y = coordinates[1];
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700456 if (dropTarget != null) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700457 DropTarget delegate = dropTarget.getDropTargetDelegate(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700458 if (delegate != null) {
459 dropTarget = delegate;
460 }
461
462 if (mLastDropTarget != dropTarget) {
463 if (mLastDropTarget != null) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700464 mLastDropTarget.onDragExit(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700465 }
Adam Cohencb3382b2011-05-24 14:07:08 -0700466 dropTarget.onDragEnter(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700467 }
Adam Cohencb3382b2011-05-24 14:07:08 -0700468 dropTarget.onDragOver(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700469 } else {
470 if (mLastDropTarget != null) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700471 mLastDropTarget.onDragExit(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700472 }
473 }
474 mLastDropTarget = dropTarget;
475
476 // Scroll, maybe, but not if we're in the delete region.
477 boolean inDeleteRegion = false;
478 if (mDeleteRegion != null) {
479 inDeleteRegion = mDeleteRegion.contains(x, y);
480 }
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700481
482 // After a scroll, the touch point will still be in the scroll region.
483 // Rather than scrolling immediately, require a bit of twiddling to scroll again
484 final int slop = ViewConfiguration.get(mContext).getScaledWindowTouchSlop();
485 mDistanceSinceScroll +=
486 Math.sqrt(Math.pow(mLastTouch[0] - x, 2) + Math.pow(mLastTouch[1] - y, 2));
487 mLastTouch[0] = x;
488 mLastTouch[1] = y;
489
Joe Onorato658db742010-09-29 11:40:39 -0700490 if (!inDeleteRegion && x < mScrollZone) {
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700491 if (mScrollState == SCROLL_OUTSIDE_ZONE && mDistanceSinceScroll > slop) {
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700492 mScrollState = SCROLL_WAITING_IN_ZONE;
493 mScrollRunnable.setDirection(SCROLL_LEFT);
494 mHandler.postDelayed(mScrollRunnable, SCROLL_DELAY);
Patrick Dubroy1262e362010-10-06 15:49:50 -0700495 mDragScroller.onEnterScrollArea(SCROLL_LEFT);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700496 }
Joe Onorato658db742010-09-29 11:40:39 -0700497 } else if (!inDeleteRegion && x > mScrollView.getWidth() - mScrollZone) {
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700498 if (mScrollState == SCROLL_OUTSIDE_ZONE && mDistanceSinceScroll > slop) {
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700499 mScrollState = SCROLL_WAITING_IN_ZONE;
500 mScrollRunnable.setDirection(SCROLL_RIGHT);
501 mHandler.postDelayed(mScrollRunnable, SCROLL_DELAY);
Patrick Dubroy1262e362010-10-06 15:49:50 -0700502 mDragScroller.onEnterScrollArea(SCROLL_RIGHT);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700503 }
504 } else {
505 if (mScrollState == SCROLL_WAITING_IN_ZONE) {
506 mScrollState = SCROLL_OUTSIDE_ZONE;
507 mScrollRunnable.setDirection(SCROLL_RIGHT);
508 mHandler.removeCallbacks(mScrollRunnable);
Patrick Dubroy1262e362010-10-06 15:49:50 -0700509 mDragScroller.onExitScrollArea();
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700510 }
511 }
512 }
513
Romain Guyea3763c2010-01-11 18:02:04 -0800514 /**
Joe Onorato00acb122009-08-04 16:04:30 -0400515 * Call this from a drag source view.
516 */
517 public boolean onTouchEvent(MotionEvent ev) {
Joe Onorato00acb122009-08-04 16:04:30 -0400518 if (!mDragging) {
519 return false;
520 }
521
522 final int action = ev.getAction();
Joe Onoratoe048e8a2009-09-25 10:39:17 -0700523 final int screenX = clamp((int)ev.getRawX(), 0, mDisplayMetrics.widthPixels);
524 final int screenY = clamp((int)ev.getRawY(), 0, mDisplayMetrics.heightPixels);
Joe Onorato00acb122009-08-04 16:04:30 -0400525
526 switch (action) {
527 case MotionEvent.ACTION_DOWN:
Joe Onorato00acb122009-08-04 16:04:30 -0400528 // Remember where the motion event started
Joe Onoratoe048e8a2009-09-25 10:39:17 -0700529 mMotionDownX = screenX;
530 mMotionDownY = screenY;
Joe Onorato00acb122009-08-04 16:04:30 -0400531
Joe Onorato658db742010-09-29 11:40:39 -0700532 if ((screenX < mScrollZone) || (screenX > mScrollView.getWidth() - mScrollZone)) {
Joe Onorato00acb122009-08-04 16:04:30 -0400533 mScrollState = SCROLL_WAITING_IN_ZONE;
534 mHandler.postDelayed(mScrollRunnable, SCROLL_DELAY);
535 } else {
536 mScrollState = SCROLL_OUTSIDE_ZONE;
537 }
Joe Onorato00acb122009-08-04 16:04:30 -0400538 break;
539 case MotionEvent.ACTION_MOVE:
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700540 handleMoveEvent(screenX, screenY);
Joe Onorato00acb122009-08-04 16:04:30 -0400541 break;
542 case MotionEvent.ACTION_UP:
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800543 // Ensure that we've processed a move event at the current pointer location.
544 handleMoveEvent(screenX, screenY);
545
Joe Onorato00acb122009-08-04 16:04:30 -0400546 mHandler.removeCallbacks(mScrollRunnable);
547 if (mDragging) {
Joe Onoratoe048e8a2009-09-25 10:39:17 -0700548 drop(screenX, screenY);
Joe Onorato00acb122009-08-04 16:04:30 -0400549 }
550 endDrag();
Joe Onorato00acb122009-08-04 16:04:30 -0400551 break;
552 case MotionEvent.ACTION_CANCEL:
Joe Onorato24b6fd82009-11-12 13:47:09 -0800553 cancelDrag();
Winson Chung621e6402011-01-04 16:03:57 -0800554 break;
Joe Onorato00acb122009-08-04 16:04:30 -0400555 }
556
557 return true;
558 }
559
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800560 private void drop(float x, float y) {
Joe Onorato00acb122009-08-04 16:04:30 -0400561 final int[] coordinates = mCoordinatesTemp;
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800562 final DropTarget dropTarget = findDropTarget((int) x, (int) y, coordinates);
Joe Onorato00acb122009-08-04 16:04:30 -0400563
Adam Cohencb3382b2011-05-24 14:07:08 -0700564 mDragObject.x = coordinates[0];
565 mDragObject.y = coordinates[1];
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800566 boolean accepted = false;
Joe Onorato00acb122009-08-04 16:04:30 -0400567 if (dropTarget != null) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700568 dropTarget.onDragExit(mDragObject);
569 if (dropTarget.acceptDrop(mDragObject)) {
570 dropTarget.onDrop(mDragObject);
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800571 accepted = true;
Joe Onorato00acb122009-08-04 16:04:30 -0400572 }
573 }
Adam Cohenc0dcf592011-06-01 15:30:43 -0700574 mDragObject.dragSource.onDropCompleted((View) dropTarget, mDragObject, accepted);
Joe Onorato00acb122009-08-04 16:04:30 -0400575 }
576
577 private DropTarget findDropTarget(int x, int y, int[] dropCoordinates) {
578 final Rect r = mRectTemp;
579
580 final ArrayList<DropTarget> dropTargets = mDropTargets;
581 final int count = dropTargets.size();
582 for (int i=count-1; i>=0; i--) {
Patrick Dubroy440c3602010-07-13 17:50:32 -0700583 DropTarget target = dropTargets.get(i);
Michael Jurka0280c3b2010-09-17 15:00:07 -0700584 if (!target.isDropEnabled())
585 continue;
586
Joe Onorato00acb122009-08-04 16:04:30 -0400587 target.getHitRect(r);
Patrick Dubroy440c3602010-07-13 17:50:32 -0700588
589 // Convert the hit rect to screen coordinates
Joe Onorato00acb122009-08-04 16:04:30 -0400590 target.getLocationOnScreen(dropCoordinates);
591 r.offset(dropCoordinates[0] - target.getLeft(), dropCoordinates[1] - target.getTop());
Patrick Dubroy440c3602010-07-13 17:50:32 -0700592
Adam Cohencb3382b2011-05-24 14:07:08 -0700593 mDragObject.x = x;
594 mDragObject.y = y;
Joe Onorato00acb122009-08-04 16:04:30 -0400595 if (r.contains(x, y)) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700596 DropTarget delegate = target.getDropTargetDelegate(mDragObject);
Patrick Dubroy440c3602010-07-13 17:50:32 -0700597 if (delegate != null) {
598 target = delegate;
599 target.getLocationOnScreen(dropCoordinates);
600 }
601
602 // Make dropCoordinates relative to the DropTarget
Joe Onorato00acb122009-08-04 16:04:30 -0400603 dropCoordinates[0] = x - dropCoordinates[0];
604 dropCoordinates[1] = y - dropCoordinates[1];
Patrick Dubroy440c3602010-07-13 17:50:32 -0700605
Joe Onorato00acb122009-08-04 16:04:30 -0400606 return target;
607 }
608 }
609 return null;
610 }
611
Joe Onoratoe048e8a2009-09-25 10:39:17 -0700612 /**
613 * Get the screen size so we can clamp events to the screen size so even if
614 * you drag off the edge of the screen, we find something.
615 */
616 private void recordScreenSize() {
617 ((WindowManager)mContext.getSystemService(Context.WINDOW_SERVICE))
618 .getDefaultDisplay().getMetrics(mDisplayMetrics);
619 }
620
621 /**
622 * Clamp val to be &gt;= min and &lt; max.
623 */
624 private static int clamp(int val, int min, int max) {
625 if (val < min) {
626 return min;
627 } else if (val >= max) {
628 return max - 1;
629 } else {
630 return val;
631 }
632 }
633
Joe Onorato00acb122009-08-04 16:04:30 -0400634 public void setDragScoller(DragScroller scroller) {
635 mDragScroller = scroller;
636 }
637
638 public void setWindowToken(IBinder token) {
639 mWindowToken = token;
640 }
641
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800642 /**
643 * Sets the drag listner which will be notified when a drag starts or ends.
644 */
Patrick Dubroy4ed62782010-08-17 15:11:18 -0700645 public void addDragListener(DragListener l) {
646 mListeners.add(l);
Joe Onorato00acb122009-08-04 16:04:30 -0400647 }
648
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800649 /**
650 * Remove a previously installed drag listener.
651 */
Joe Onorato00acb122009-08-04 16:04:30 -0400652 public void removeDragListener(DragListener l) {
Patrick Dubroy4ed62782010-08-17 15:11:18 -0700653 mListeners.remove(l);
Joe Onorato00acb122009-08-04 16:04:30 -0400654 }
655
656 /**
657 * Add a DropTarget to the list of potential places to receive drop events.
658 */
659 public void addDropTarget(DropTarget target) {
660 mDropTargets.add(target);
661 }
662
663 /**
664 * Don't send drop events to <em>target</em> any more.
665 */
666 public void removeDropTarget(DropTarget target) {
667 mDropTargets.remove(target);
668 }
669
670 /**
671 * Set which view scrolls for touch events near the edge of the screen.
672 */
673 public void setScrollView(View v) {
674 mScrollView = v;
675 }
676
677 /**
678 * Specifies the delete region. We won't scroll on touch events over the delete region.
679 *
680 * @param region The rectangle in screen coordinates of the delete region.
681 */
682 void setDeleteRegion(RectF region) {
683 mDeleteRegion = region;
684 }
685
Patrick Dubroy5f445422011-02-18 14:35:21 -0800686 DragView getDragView() {
Adam Cohencb3382b2011-05-24 14:07:08 -0700687 return mDragObject.dragView;
Patrick Dubroy5f445422011-02-18 14:35:21 -0800688 }
689
Joe Onorato00acb122009-08-04 16:04:30 -0400690 private class ScrollRunnable implements Runnable {
691 private int mDirection;
692
693 ScrollRunnable() {
694 }
695
696 public void run() {
697 if (mDragScroller != null) {
698 if (mDirection == SCROLL_LEFT) {
699 mDragScroller.scrollLeft();
700 } else {
701 mDragScroller.scrollRight();
702 }
703 mScrollState = SCROLL_OUTSIDE_ZONE;
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700704 mDistanceSinceScroll = 0;
705 mDragScroller.onExitScrollArea();
Joe Onorato00acb122009-08-04 16:04:30 -0400706 }
707 }
708
709 void setDirection(int direction) {
710 mDirection = direction;
711 }
712 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800713}