blob: a120ac5699686b7548edc8bfff855b98a0a09f08 [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;
Winson Chungb8c69f32011-10-19 21:36:08 -070021import android.graphics.Point;
Joe Onorato00acb122009-08-04 16:04:30 -040022import android.graphics.Rect;
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 Onorato00acb122009-08-04 16:04:30 -040026import android.util.Log;
Joe Onorato00acb122009-08-04 16:04:30 -040027import android.view.KeyEvent;
28import android.view.MotionEvent;
Michael Jurka0280c3b2010-09-17 15:00:07 -070029import android.view.View;
Patrick Dubroya16fd5a2010-10-07 16:47:28 -070030import android.view.ViewConfiguration;
Joe Onorato00acb122009-08-04 16:04:30 -040031import android.view.inputmethod.InputMethodManager;
Joe Onorato00acb122009-08-04 16:04:30 -040032
Adam Cohen120980b2010-12-08 11:05:37 -080033import com.android.launcher.R;
Adam Cohenc0dcf592011-06-01 15:30:43 -070034
35import java.util.ArrayList;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080036
37/**
Joe Onorato00acb122009-08-04 16:04:30 -040038 * Class for initiating a drag within a view or across multiple views.
The Android Open Source Project31dd5032009-03-03 19:32:27 -080039 */
Joe Onorato00acb122009-08-04 16:04:30 -040040public class DragController {
Romain Guyea3763c2010-01-11 18:02:04 -080041 @SuppressWarnings({"UnusedDeclaration"})
Joe Onorato2e5c4322009-10-06 12:34:42 -070042 private static final String TAG = "Launcher.DragController";
43
Joe Onorato00acb122009-08-04 16:04:30 -040044 /** Indicates the drag is a move. */
45 public static int DRAG_ACTION_MOVE = 0;
46
47 /** Indicates the drag is a copy. */
48 public static int DRAG_ACTION_COPY = 1;
49
50 private static final int SCROLL_DELAY = 600;
Joe Onorato00acb122009-08-04 16:04:30 -040051 private static final int VIBRATE_DURATION = 35;
52
53 private static final boolean PROFILE_DRAWING_DURING_DRAG = false;
54
55 private static final int SCROLL_OUTSIDE_ZONE = 0;
56 private static final int SCROLL_WAITING_IN_ZONE = 1;
57
Patrick Dubroy54fa3b92010-11-17 12:18:45 -080058 static final int SCROLL_NONE = -1;
Patrick Dubroy1262e362010-10-06 15:49:50 -070059 static final int SCROLL_LEFT = 0;
60 static final int SCROLL_RIGHT = 1;
Joe Onorato00acb122009-08-04 16:04:30 -040061
Adam Cohen8dfcba42011-07-07 16:38:18 -070062 private Launcher mLauncher;
Joe Onorato00acb122009-08-04 16:04:30 -040063 private Handler mHandler;
64 private final Vibrator mVibrator = new Vibrator();
65
66 // temporaries to avoid gc thrash
67 private Rect mRectTemp = new Rect();
68 private final int[] mCoordinatesTemp = new int[2];
69
70 /** Whether or not we're dragging. */
71 private boolean mDragging;
72
73 /** X coordinate of the down event. */
Adam Cohene3e27a82011-04-15 12:07:39 -070074 private int mMotionDownX;
Joe Onorato00acb122009-08-04 16:04:30 -040075
76 /** Y coordinate of the down event. */
Adam Cohene3e27a82011-04-15 12:07:39 -070077 private int mMotionDownY;
Joe Onorato00acb122009-08-04 16:04:30 -040078
Joe Onorato658db742010-09-29 11:40:39 -070079 /** the area at the edge of the screen that makes the workspace go left
80 * or right while you're dragging.
81 */
82 private int mScrollZone;
83
Adam Cohen9932a9b2011-08-02 22:14:07 -070084 private DropTarget.DragObject mDragObject;
Joe Onorato00acb122009-08-04 16:04:30 -040085
86 /** Who can receive drop events */
87 private ArrayList<DropTarget> mDropTargets = new ArrayList<DropTarget>();
88
Patrick Dubroy4ed62782010-08-17 15:11:18 -070089 private ArrayList<DragListener> mListeners = new ArrayList<DragListener>();
Joe Onorato00acb122009-08-04 16:04:30 -040090
91 /** The window token used as the parent for the DragView. */
92 private IBinder mWindowToken;
93
94 /** The view that will be scrolled when dragging to the left and right edges of the screen. */
95 private View mScrollView;
96
Romain Guyea3763c2010-01-11 18:02:04 -080097 private View mMoveTarget;
98
Joe Onorato00acb122009-08-04 16:04:30 -040099 private DragScroller mDragScroller;
100 private int mScrollState = SCROLL_OUTSIDE_ZONE;
101 private ScrollRunnable mScrollRunnable = new ScrollRunnable();
102
Joe Onorato00acb122009-08-04 16:04:30 -0400103 private DropTarget mLastDropTarget;
104
105 private InputMethodManager mInputMethodManager;
106
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700107 private int mLastTouch[] = new int[2];
108 private int mDistanceSinceScroll = 0;
109
Winson Chung273c1022011-07-11 13:40:52 -0700110 private int mTmpPoint[] = new int[2];
111 private Rect mDragLayerRect = new Rect();
112
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800113 /**
114 * Interface to receive notifications when a drag starts or stops
115 */
116 interface DragListener {
117
118 /**
119 * A drag has begun
120 *
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800121 * @param source An object representing where the drag originated
122 * @param info The data associated with the object that is being dragged
123 * @param dragAction The drag action: either {@link DragController#DRAG_ACTION_MOVE}
124 * or {@link DragController#DRAG_ACTION_COPY}
125 */
Joe Onorato5162ea92009-09-03 09:39:42 -0700126 void onDragStart(DragSource source, Object info, int dragAction);
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800127
128 /**
Winson Chunge3193b92010-09-10 11:44:42 -0700129 * The drag has ended
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800130 */
131 void onDragEnd();
132 }
133
134 /**
Joe Onorato00acb122009-08-04 16:04:30 -0400135 * Used to create a new DragLayer from XML.
136 *
137 * @param context The application's context.
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800138 */
Adam Cohen8dfcba42011-07-07 16:38:18 -0700139 public DragController(Launcher launcher) {
140 mLauncher = launcher;
Joe Onorato00acb122009-08-04 16:04:30 -0400141 mHandler = new Handler();
Adam Cohen8dfcba42011-07-07 16:38:18 -0700142 mScrollZone = launcher.getResources().getDimensionPixelSize(R.dimen.scroll_zone);
Joe Onorato00acb122009-08-04 16:04:30 -0400143 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800144
Patrick Dubroy1262e362010-10-06 15:49:50 -0700145 public boolean dragging() {
146 return mDragging;
147 }
148
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800149 /**
Joe Onorato5162ea92009-09-03 09:39:42 -0700150 * Starts a drag.
Michael Jurkaa63c4522010-08-19 13:52:27 -0700151 *
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800152 * @param v The view that is being dragged
153 * @param source An object representing where the drag originated
Romain Guyea3763c2010-01-11 18:02:04 -0800154 * @param dragInfo The data associated with the object that is being dragged
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800155 * @param dragAction The drag action: either {@link #DRAG_ACTION_MOVE} or
156 * {@link #DRAG_ACTION_COPY}
157 */
Joe Onorato00acb122009-08-04 16:04:30 -0400158 public void startDrag(View v, DragSource source, Object dragInfo, int dragAction) {
Michael Jurkaa63c4522010-08-19 13:52:27 -0700159 startDrag(v, source, dragInfo, dragAction, null);
160 }
161
162 /**
163 * Starts a drag.
164 *
165 * @param v The view that is being dragged
166 * @param source An object representing where the drag originated
167 * @param dragInfo The data associated with the object that is being dragged
168 * @param dragAction The drag action: either {@link #DRAG_ACTION_MOVE} or
169 * {@link #DRAG_ACTION_COPY}
170 * @param dragRegion Coordinates within the bitmap b for the position of item being dragged.
171 * Makes dragging feel more precise, e.g. you can clip out a transparent border
172 */
173 public void startDrag(View v, DragSource source, Object dragInfo, int dragAction,
174 Rect dragRegion) {
Joe Onorato5162ea92009-09-03 09:39:42 -0700175 Bitmap b = getViewBitmap(v);
176
Daniel Sandler3f8175a2010-05-25 11:48:32 -0400177 if (b == null) {
178 // out of memory?
179 return;
180 }
181
Joe Onorato5162ea92009-09-03 09:39:42 -0700182 int[] loc = mCoordinatesTemp;
Adam Cohen8dfcba42011-07-07 16:38:18 -0700183 mLauncher.getDragLayer().getLocationInDragLayer(v, loc);
184 int dragLayerX = loc[0];
185 int dragLayerY = loc[1];
Joe Onorato5162ea92009-09-03 09:39:42 -0700186
Winson Chungb8c69f32011-10-19 21:36:08 -0700187 startDrag(b, dragLayerX, dragLayerY, source, dragInfo, dragAction, null, dragRegion);
Joe Onorato5162ea92009-09-03 09:39:42 -0700188 b.recycle();
189
190 if (dragAction == DRAG_ACTION_MOVE) {
191 v.setVisibility(View.GONE);
192 }
193 }
194
195 /**
196 * Starts a drag.
Michael Jurkaa63c4522010-08-19 13:52:27 -0700197 *
Winson Chunge3193b92010-09-10 11:44:42 -0700198 * @param v The view that is being dragged
199 * @param bmp The bitmap that represents the view being dragged
200 * @param source An object representing where the drag originated
201 * @param dragInfo The data associated with the object that is being dragged
202 * @param dragAction The drag action: either {@link #DRAG_ACTION_MOVE} or
203 * {@link #DRAG_ACTION_COPY}
204 * @param dragRegion Coordinates within the bitmap b for the position of item being dragged.
205 * Makes dragging feel more precise, e.g. you can clip out a transparent border
206 */
207 public void startDrag(View v, Bitmap bmp, DragSource source, Object dragInfo, int dragAction,
208 Rect dragRegion) {
Winson Chunge3193b92010-09-10 11:44:42 -0700209 int[] loc = mCoordinatesTemp;
Adam Cohen8dfcba42011-07-07 16:38:18 -0700210 mLauncher.getDragLayer().getLocationInDragLayer(v, loc);
211 int dragLayerX = loc[0];
212 int dragLayerY = loc[1];
Winson Chunge3193b92010-09-10 11:44:42 -0700213
Winson Chungb8c69f32011-10-19 21:36:08 -0700214 startDrag(bmp, dragLayerX, dragLayerY, source, dragInfo, dragAction, null, dragRegion);
Winson Chunge3193b92010-09-10 11:44:42 -0700215
216 if (dragAction == DRAG_ACTION_MOVE) {
217 v.setVisibility(View.GONE);
218 }
219 }
220
221 /**
222 * Starts a drag.
223 *
Joe Onorato5162ea92009-09-03 09:39:42 -0700224 * @param b The bitmap to display as the drag image. It will be re-scaled to the
225 * enlarged size.
Adam Cohen8dfcba42011-07-07 16:38:18 -0700226 * @param dragLayerX The x position in the DragLayer of the left-top of the bitmap.
227 * @param dragLayerY The y position in the DragLayer of the left-top of the bitmap.
Joe Onorato5162ea92009-09-03 09:39:42 -0700228 * @param source An object representing where the drag originated
Romain Guyea3763c2010-01-11 18:02:04 -0800229 * @param dragInfo The data associated with the object that is being dragged
Joe Onorato5162ea92009-09-03 09:39:42 -0700230 * @param dragAction The drag action: either {@link #DRAG_ACTION_MOVE} or
231 * {@link #DRAG_ACTION_COPY}
232 */
Adam Cohen8dfcba42011-07-07 16:38:18 -0700233 public void startDrag(Bitmap b, int dragLayerX, int dragLayerY,
Joe Onorato5162ea92009-09-03 09:39:42 -0700234 DragSource source, Object dragInfo, int dragAction) {
Winson Chungb8c69f32011-10-19 21:36:08 -0700235 startDrag(b, dragLayerX, dragLayerY, source, dragInfo, dragAction, null, null);
Michael Jurkaa63c4522010-08-19 13:52:27 -0700236 }
237
238 /**
239 * Starts a drag.
240 *
241 * @param b The bitmap to display as the drag image. It will be re-scaled to the
242 * enlarged size.
Adam Cohen8dfcba42011-07-07 16:38:18 -0700243 * @param dragLayerX The x position in the DragLayer of the left-top of the bitmap.
244 * @param dragLayerY The y position in the DragLayer of the left-top of the bitmap.
Michael Jurkaa63c4522010-08-19 13:52:27 -0700245 * @param source An object representing where the drag originated
246 * @param dragInfo The data associated with the object that is being dragged
247 * @param dragAction The drag action: either {@link #DRAG_ACTION_MOVE} or
248 * {@link #DRAG_ACTION_COPY}
249 * @param dragRegion Coordinates within the bitmap b for the position of item being dragged.
250 * Makes dragging feel more precise, e.g. you can clip out a transparent border
251 */
Adam Cohen8dfcba42011-07-07 16:38:18 -0700252 public void startDrag(Bitmap b, int dragLayerX, int dragLayerY,
Winson Chungb8c69f32011-10-19 21:36:08 -0700253 DragSource source, Object dragInfo, int dragAction, Point dragOffset, Rect dragRegion) {
Joe Onorato00acb122009-08-04 16:04:30 -0400254 if (PROFILE_DRAWING_DURING_DRAG) {
255 android.os.Debug.startMethodTracing("Launcher");
256 }
257
258 // Hide soft keyboard, if visible
259 if (mInputMethodManager == null) {
260 mInputMethodManager = (InputMethodManager)
Adam Cohen8dfcba42011-07-07 16:38:18 -0700261 mLauncher.getSystemService(Context.INPUT_METHOD_SERVICE);
Joe Onorato00acb122009-08-04 16:04:30 -0400262 }
263 mInputMethodManager.hideSoftInputFromWindow(mWindowToken, 0);
264
Patrick Dubroy4ed62782010-08-17 15:11:18 -0700265 for (DragListener listener : mListeners) {
266 listener.onDragStart(source, dragInfo, dragAction);
Joe Onorato00acb122009-08-04 16:04:30 -0400267 }
268
Adam Cohen8dfcba42011-07-07 16:38:18 -0700269 final int registrationX = mMotionDownX - dragLayerX;
270 final int registrationY = mMotionDownY - dragLayerY;
Joe Onorato00acb122009-08-04 16:04:30 -0400271
Michael Jurkaa63c4522010-08-19 13:52:27 -0700272 final int dragRegionLeft = dragRegion == null ? 0 : dragRegion.left;
273 final int dragRegionTop = dragRegion == null ? 0 : dragRegion.top;
Adam Cohene3e27a82011-04-15 12:07:39 -0700274
Joe Onorato00acb122009-08-04 16:04:30 -0400275 mDragging = true;
Adam Cohencb3382b2011-05-24 14:07:08 -0700276
Adam Cohen9932a9b2011-08-02 22:14:07 -0700277 mDragObject = new DropTarget.DragObject();
278
Adam Cohenbfbfd262011-06-13 16:55:12 -0700279 mDragObject.dragComplete = false;
Adam Cohen8dfcba42011-07-07 16:38:18 -0700280 mDragObject.xOffset = mMotionDownX - (dragLayerX + dragRegionLeft);
281 mDragObject.yOffset = mMotionDownY - (dragLayerY + dragRegionTop);
Adam Cohencb3382b2011-05-24 14:07:08 -0700282 mDragObject.dragSource = source;
283 mDragObject.dragInfo = dragInfo;
Joe Onorato00acb122009-08-04 16:04:30 -0400284
285 mVibrator.vibrate(VIBRATE_DURATION);
286
Adam Cohen8dfcba42011-07-07 16:38:18 -0700287 final DragView dragView = mDragObject.dragView = new DragView(mLauncher, b, registrationX,
Adam Cohencb3382b2011-05-24 14:07:08 -0700288 registrationY, 0, 0, b.getWidth(), b.getHeight());
Michael Jurkaa63c4522010-08-19 13:52:27 -0700289
Winson Chungb8c69f32011-10-19 21:36:08 -0700290 if (dragOffset != null) {
291 dragView.setDragVisualizeOffset(new Point(dragOffset));
292 }
Michael Jurkaa63c4522010-08-19 13:52:27 -0700293 if (dragRegion != null) {
Adam Cohene3e27a82011-04-15 12:07:39 -0700294 dragView.setDragRegion(new Rect(dragRegion));
Michael Jurkaa63c4522010-08-19 13:52:27 -0700295 }
296
Adam Cohen8dfcba42011-07-07 16:38:18 -0700297 dragView.show(mMotionDownX, mMotionDownY);
298 handleMoveEvent(mMotionDownX, mMotionDownY);
Joe Onorato00acb122009-08-04 16:04:30 -0400299 }
300
301 /**
302 * Draw the view into a bitmap.
303 */
Adam Cohen120980b2010-12-08 11:05:37 -0800304 Bitmap getViewBitmap(View v) {
Joe Onorato00acb122009-08-04 16:04:30 -0400305 v.clearFocus();
306 v.setPressed(false);
307
308 boolean willNotCache = v.willNotCacheDrawing();
309 v.setWillNotCacheDrawing(false);
310
311 // Reset the drawing cache background color to fully transparent
312 // for the duration of this operation
313 int color = v.getDrawingCacheBackgroundColor();
314 v.setDrawingCacheBackgroundColor(0);
Adam Cohen120980b2010-12-08 11:05:37 -0800315 float alpha = v.getAlpha();
316 v.setAlpha(1.0f);
Joe Onorato00acb122009-08-04 16:04:30 -0400317
318 if (color != 0) {
319 v.destroyDrawingCache();
320 }
321 v.buildDrawingCache();
322 Bitmap cacheBitmap = v.getDrawingCache();
Daniel Sandler3f8175a2010-05-25 11:48:32 -0400323 if (cacheBitmap == null) {
324 Log.e(TAG, "failed getViewBitmap(" + v + ")", new RuntimeException());
325 return null;
326 }
Joe Onorato00acb122009-08-04 16:04:30 -0400327
328 Bitmap bitmap = Bitmap.createBitmap(cacheBitmap);
329
330 // Restore the view
331 v.destroyDrawingCache();
Adam Cohen120980b2010-12-08 11:05:37 -0800332 v.setAlpha(alpha);
Joe Onorato00acb122009-08-04 16:04:30 -0400333 v.setWillNotCacheDrawing(willNotCache);
334 v.setDrawingCacheBackgroundColor(color);
335
336 return bitmap;
337 }
338
339 /**
340 * Call this from a drag source view like this:
341 *
342 * <pre>
343 * @Override
344 * public boolean dispatchKeyEvent(KeyEvent event) {
345 * return mDragController.dispatchKeyEvent(this, event)
346 * || super.dispatchKeyEvent(event);
347 * </pre>
348 */
Romain Guyea3763c2010-01-11 18:02:04 -0800349 @SuppressWarnings({"UnusedDeclaration"})
Joe Onorato00acb122009-08-04 16:04:30 -0400350 public boolean dispatchKeyEvent(KeyEvent event) {
351 return mDragging;
352 }
353
Winson Chung304dcde2011-01-07 11:17:23 -0800354 public boolean isDragging() {
355 return mDragging;
356 }
357
Joe Onorato24b6fd82009-11-12 13:47:09 -0800358 /**
359 * Stop dragging without dropping.
360 */
361 public void cancelDrag() {
Winson Chung621e6402011-01-04 16:03:57 -0800362 if (mDragging) {
Winson Chungc07918d2011-07-01 15:35:26 -0700363 if (mLastDropTarget != null) {
364 mLastDropTarget.onDragExit(mDragObject);
365 }
Adam Cohen36cc09b2011-09-29 17:33:15 -0700366 mDragObject.cancelled = true;
Adam Cohenbfbfd262011-06-13 16:55:12 -0700367 mDragObject.dragComplete = true;
Adam Cohenc0dcf592011-06-01 15:30:43 -0700368 mDragObject.dragSource.onDropCompleted(null, mDragObject, false);
Winson Chung621e6402011-01-04 16:03:57 -0800369 }
Joe Onorato24b6fd82009-11-12 13:47:09 -0800370 endDrag();
371 }
Winson Chunga1820962011-10-03 16:31:06 -0700372 public void onAppsRemoved(ArrayList<ApplicationInfo> apps, Context context) {
373 // Cancel the current drag if we are removing an app that we are dragging
374 if (mDragObject != null) {
375 Object rawDragInfo = mDragObject.dragInfo;
376 if (rawDragInfo instanceof ShortcutInfo) {
377 ShortcutInfo dragInfo = (ShortcutInfo) rawDragInfo;
378 for (ApplicationInfo info : apps) {
379 if (dragInfo.intent.getComponent().equals(info.intent.getComponent())) {
380 cancelDrag();
381 return;
382 }
383 }
384 }
385 }
386 }
Joe Onorato24b6fd82009-11-12 13:47:09 -0800387
Joe Onorato00acb122009-08-04 16:04:30 -0400388 private void endDrag() {
389 if (mDragging) {
390 mDragging = false;
Patrick Dubroy4ed62782010-08-17 15:11:18 -0700391 for (DragListener listener : mListeners) {
392 listener.onDragEnd();
Joe Onorato00acb122009-08-04 16:04:30 -0400393 }
Adam Cohencb3382b2011-05-24 14:07:08 -0700394 if (mDragObject.dragView != null) {
395 mDragObject.dragView.remove();
396 mDragObject.dragView = null;
Joe Onorato00acb122009-08-04 16:04:30 -0400397 }
Joe Onorato00acb122009-08-04 16:04:30 -0400398 }
399 }
400
401 /**
Winson Chung273c1022011-07-11 13:40:52 -0700402 * Clamps the position to the drag layer bounds.
403 */
404 private int[] getClampedDragLayerPos(float x, float y) {
405 mLauncher.getDragLayer().getLocalVisibleRect(mDragLayerRect);
406 mTmpPoint[0] = (int) Math.max(mDragLayerRect.left, Math.min(x, mDragLayerRect.right - 1));
407 mTmpPoint[1] = (int) Math.max(mDragLayerRect.top, Math.min(y, mDragLayerRect.bottom - 1));
408 return mTmpPoint;
409 }
410
411 /**
Joe Onorato00acb122009-08-04 16:04:30 -0400412 * Call this from a drag source view.
413 */
414 public boolean onInterceptTouchEvent(MotionEvent ev) {
Joe Onorato9c1289c2009-08-17 11:03:03 -0400415 if (false) {
Joe Onoratoa30ce8e2009-11-11 08:16:49 -0800416 Log.d(Launcher.TAG, "DragController.onInterceptTouchEvent " + ev + " mDragging="
Joe Onorato9c1289c2009-08-17 11:03:03 -0400417 + mDragging);
418 }
Joe Onorato00acb122009-08-04 16:04:30 -0400419 final int action = ev.getAction();
420
Winson Chung273c1022011-07-11 13:40:52 -0700421 final int[] dragLayerPos = getClampedDragLayerPos(ev.getX(), ev.getY());
422 final int dragLayerX = dragLayerPos[0];
423 final int dragLayerY = dragLayerPos[1];
Joe Onorato00acb122009-08-04 16:04:30 -0400424
425 switch (action) {
426 case MotionEvent.ACTION_MOVE:
427 break;
Joe Onorato00acb122009-08-04 16:04:30 -0400428 case MotionEvent.ACTION_DOWN:
429 // Remember location of down touch
Adam Cohen8dfcba42011-07-07 16:38:18 -0700430 mMotionDownX = dragLayerX;
431 mMotionDownY = dragLayerY;
Joe Onorato00acb122009-08-04 16:04:30 -0400432 mLastDropTarget = null;
433 break;
Joe Onorato00acb122009-08-04 16:04:30 -0400434 case MotionEvent.ACTION_UP:
435 if (mDragging) {
Adam Cohen8dfcba42011-07-07 16:38:18 -0700436 drop(dragLayerX, dragLayerY);
Joe Onorato00acb122009-08-04 16:04:30 -0400437 }
438 endDrag();
439 break;
Winson Chung621e6402011-01-04 16:03:57 -0800440 case MotionEvent.ACTION_CANCEL:
441 cancelDrag();
442 break;
Joe Onorato00acb122009-08-04 16:04:30 -0400443 }
444
445 return mDragging;
446 }
447
448 /**
Romain Guyea3763c2010-01-11 18:02:04 -0800449 * Sets the view that should handle move events.
450 */
451 void setMoveTarget(View view) {
452 mMoveTarget = view;
453 }
454
455 public boolean dispatchUnhandledMove(View focused, int direction) {
456 return mMoveTarget != null && mMoveTarget.dispatchUnhandledMove(focused, direction);
457 }
458
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700459 private void handleMoveEvent(int x, int y) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700460 mDragObject.dragView.move(x, y);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700461
462 // Drop on someone?
463 final int[] coordinates = mCoordinatesTemp;
464 DropTarget dropTarget = findDropTarget(x, y, coordinates);
Adam Cohencb3382b2011-05-24 14:07:08 -0700465 mDragObject.x = coordinates[0];
466 mDragObject.y = coordinates[1];
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700467 if (dropTarget != null) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700468 DropTarget delegate = dropTarget.getDropTargetDelegate(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700469 if (delegate != null) {
470 dropTarget = delegate;
471 }
472
473 if (mLastDropTarget != dropTarget) {
474 if (mLastDropTarget != null) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700475 mLastDropTarget.onDragExit(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700476 }
Adam Cohencb3382b2011-05-24 14:07:08 -0700477 dropTarget.onDragEnter(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700478 }
Adam Cohencb3382b2011-05-24 14:07:08 -0700479 dropTarget.onDragOver(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700480 } else {
481 if (mLastDropTarget != null) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700482 mLastDropTarget.onDragExit(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700483 }
484 }
485 mLastDropTarget = dropTarget;
486
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700487 // After a scroll, the touch point will still be in the scroll region.
488 // Rather than scrolling immediately, require a bit of twiddling to scroll again
Adam Cohen8dfcba42011-07-07 16:38:18 -0700489 final int slop = ViewConfiguration.get(mLauncher).getScaledWindowTouchSlop();
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700490 mDistanceSinceScroll +=
491 Math.sqrt(Math.pow(mLastTouch[0] - x, 2) + Math.pow(mLastTouch[1] - y, 2));
492 mLastTouch[0] = x;
493 mLastTouch[1] = y;
494
Winson Chung3f4e1422011-11-17 14:58:51 -0800495 if (x < mScrollZone) {
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700496 if (mScrollState == SCROLL_OUTSIDE_ZONE && mDistanceSinceScroll > slop) {
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700497 mScrollState = SCROLL_WAITING_IN_ZONE;
Winson Chung3e0839e2011-10-03 15:15:18 -0700498 if (mDragScroller.onEnterScrollArea(x, y, SCROLL_LEFT)) {
499 mScrollRunnable.setDirection(SCROLL_LEFT);
500 mHandler.postDelayed(mScrollRunnable, SCROLL_DELAY);
501 }
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700502 }
Winson Chung3f4e1422011-11-17 14:58:51 -0800503 } else if (x > mScrollView.getWidth() - mScrollZone) {
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700504 if (mScrollState == SCROLL_OUTSIDE_ZONE && mDistanceSinceScroll > slop) {
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700505 mScrollState = SCROLL_WAITING_IN_ZONE;
Winson Chung3e0839e2011-10-03 15:15:18 -0700506 if (mDragScroller.onEnterScrollArea(x, y, SCROLL_RIGHT)) {
507 mScrollRunnable.setDirection(SCROLL_RIGHT);
508 mHandler.postDelayed(mScrollRunnable, SCROLL_DELAY);
509 }
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700510 }
511 } else {
512 if (mScrollState == SCROLL_WAITING_IN_ZONE) {
513 mScrollState = SCROLL_OUTSIDE_ZONE;
514 mScrollRunnable.setDirection(SCROLL_RIGHT);
515 mHandler.removeCallbacks(mScrollRunnable);
Patrick Dubroy1262e362010-10-06 15:49:50 -0700516 mDragScroller.onExitScrollArea();
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700517 }
518 }
519 }
520
Romain Guyea3763c2010-01-11 18:02:04 -0800521 /**
Joe Onorato00acb122009-08-04 16:04:30 -0400522 * Call this from a drag source view.
523 */
524 public boolean onTouchEvent(MotionEvent ev) {
Joe Onorato00acb122009-08-04 16:04:30 -0400525 if (!mDragging) {
526 return false;
527 }
528
529 final int action = ev.getAction();
Winson Chung273c1022011-07-11 13:40:52 -0700530 final int[] dragLayerPos = getClampedDragLayerPos(ev.getX(), ev.getY());
531 final int dragLayerX = dragLayerPos[0];
532 final int dragLayerY = dragLayerPos[1];
Joe Onorato00acb122009-08-04 16:04:30 -0400533
534 switch (action) {
535 case MotionEvent.ACTION_DOWN:
Joe Onorato00acb122009-08-04 16:04:30 -0400536 // Remember where the motion event started
Adam Cohen8dfcba42011-07-07 16:38:18 -0700537 mMotionDownX = dragLayerX;
538 mMotionDownY = dragLayerY;
Joe Onorato00acb122009-08-04 16:04:30 -0400539
Adam Cohen8dfcba42011-07-07 16:38:18 -0700540 if ((dragLayerX < mScrollZone) || (dragLayerX > mScrollView.getWidth() - mScrollZone)) {
Joe Onorato00acb122009-08-04 16:04:30 -0400541 mScrollState = SCROLL_WAITING_IN_ZONE;
542 mHandler.postDelayed(mScrollRunnable, SCROLL_DELAY);
543 } else {
544 mScrollState = SCROLL_OUTSIDE_ZONE;
545 }
Joe Onorato00acb122009-08-04 16:04:30 -0400546 break;
547 case MotionEvent.ACTION_MOVE:
Adam Cohen8dfcba42011-07-07 16:38:18 -0700548 handleMoveEvent(dragLayerX, dragLayerY);
Joe Onorato00acb122009-08-04 16:04:30 -0400549 break;
550 case MotionEvent.ACTION_UP:
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800551 // Ensure that we've processed a move event at the current pointer location.
Adam Cohen8dfcba42011-07-07 16:38:18 -0700552 handleMoveEvent(dragLayerX, dragLayerY);
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800553
Joe Onorato00acb122009-08-04 16:04:30 -0400554 mHandler.removeCallbacks(mScrollRunnable);
555 if (mDragging) {
Adam Cohen8dfcba42011-07-07 16:38:18 -0700556 drop(dragLayerX, dragLayerY);
Joe Onorato00acb122009-08-04 16:04:30 -0400557 }
558 endDrag();
Joe Onorato00acb122009-08-04 16:04:30 -0400559 break;
560 case MotionEvent.ACTION_CANCEL:
Joe Onorato24b6fd82009-11-12 13:47:09 -0800561 cancelDrag();
Winson Chung621e6402011-01-04 16:03:57 -0800562 break;
Joe Onorato00acb122009-08-04 16:04:30 -0400563 }
564
565 return true;
566 }
567
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800568 private void drop(float x, float y) {
Joe Onorato00acb122009-08-04 16:04:30 -0400569 final int[] coordinates = mCoordinatesTemp;
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800570 final DropTarget dropTarget = findDropTarget((int) x, (int) y, coordinates);
Joe Onorato00acb122009-08-04 16:04:30 -0400571
Adam Cohencb3382b2011-05-24 14:07:08 -0700572 mDragObject.x = coordinates[0];
573 mDragObject.y = coordinates[1];
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800574 boolean accepted = false;
Joe Onorato00acb122009-08-04 16:04:30 -0400575 if (dropTarget != null) {
Adam Cohenbfbfd262011-06-13 16:55:12 -0700576 mDragObject.dragComplete = true;
Adam Cohencb3382b2011-05-24 14:07:08 -0700577 dropTarget.onDragExit(mDragObject);
578 if (dropTarget.acceptDrop(mDragObject)) {
579 dropTarget.onDrop(mDragObject);
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800580 accepted = true;
Joe Onorato00acb122009-08-04 16:04:30 -0400581 }
582 }
Adam Cohenc0dcf592011-06-01 15:30:43 -0700583 mDragObject.dragSource.onDropCompleted((View) dropTarget, mDragObject, accepted);
Joe Onorato00acb122009-08-04 16:04:30 -0400584 }
585
586 private DropTarget findDropTarget(int x, int y, int[] dropCoordinates) {
587 final Rect r = mRectTemp;
588
589 final ArrayList<DropTarget> dropTargets = mDropTargets;
590 final int count = dropTargets.size();
591 for (int i=count-1; i>=0; i--) {
Patrick Dubroy440c3602010-07-13 17:50:32 -0700592 DropTarget target = dropTargets.get(i);
Michael Jurka0280c3b2010-09-17 15:00:07 -0700593 if (!target.isDropEnabled())
594 continue;
595
Joe Onorato00acb122009-08-04 16:04:30 -0400596 target.getHitRect(r);
Patrick Dubroy440c3602010-07-13 17:50:32 -0700597
Adam Cohen8dfcba42011-07-07 16:38:18 -0700598 // Convert the hit rect to DragLayer coordinates
599 target.getLocationInDragLayer(dropCoordinates);
Joe Onorato00acb122009-08-04 16:04:30 -0400600 r.offset(dropCoordinates[0] - target.getLeft(), dropCoordinates[1] - target.getTop());
Patrick Dubroy440c3602010-07-13 17:50:32 -0700601
Adam Cohencb3382b2011-05-24 14:07:08 -0700602 mDragObject.x = x;
603 mDragObject.y = y;
Joe Onorato00acb122009-08-04 16:04:30 -0400604 if (r.contains(x, y)) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700605 DropTarget delegate = target.getDropTargetDelegate(mDragObject);
Patrick Dubroy440c3602010-07-13 17:50:32 -0700606 if (delegate != null) {
607 target = delegate;
Adam Cohen8dfcba42011-07-07 16:38:18 -0700608 target.getLocationInDragLayer(dropCoordinates);
Patrick Dubroy440c3602010-07-13 17:50:32 -0700609 }
610
611 // Make dropCoordinates relative to the DropTarget
Joe Onorato00acb122009-08-04 16:04:30 -0400612 dropCoordinates[0] = x - dropCoordinates[0];
613 dropCoordinates[1] = y - dropCoordinates[1];
Patrick Dubroy440c3602010-07-13 17:50:32 -0700614
Joe Onorato00acb122009-08-04 16:04:30 -0400615 return target;
616 }
617 }
618 return null;
619 }
620
621 public void setDragScoller(DragScroller scroller) {
622 mDragScroller = scroller;
623 }
624
625 public void setWindowToken(IBinder token) {
626 mWindowToken = token;
627 }
628
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800629 /**
630 * Sets the drag listner which will be notified when a drag starts or ends.
631 */
Patrick Dubroy4ed62782010-08-17 15:11:18 -0700632 public void addDragListener(DragListener l) {
633 mListeners.add(l);
Joe Onorato00acb122009-08-04 16:04:30 -0400634 }
635
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800636 /**
637 * Remove a previously installed drag listener.
638 */
Joe Onorato00acb122009-08-04 16:04:30 -0400639 public void removeDragListener(DragListener l) {
Patrick Dubroy4ed62782010-08-17 15:11:18 -0700640 mListeners.remove(l);
Joe Onorato00acb122009-08-04 16:04:30 -0400641 }
642
643 /**
644 * Add a DropTarget to the list of potential places to receive drop events.
645 */
646 public void addDropTarget(DropTarget target) {
647 mDropTargets.add(target);
648 }
649
650 /**
651 * Don't send drop events to <em>target</em> any more.
652 */
653 public void removeDropTarget(DropTarget target) {
654 mDropTargets.remove(target);
655 }
656
657 /**
658 * Set which view scrolls for touch events near the edge of the screen.
659 */
660 public void setScrollView(View v) {
661 mScrollView = v;
662 }
663
Patrick Dubroy5f445422011-02-18 14:35:21 -0800664 DragView getDragView() {
Adam Cohencb3382b2011-05-24 14:07:08 -0700665 return mDragObject.dragView;
Patrick Dubroy5f445422011-02-18 14:35:21 -0800666 }
667
Joe Onorato00acb122009-08-04 16:04:30 -0400668 private class ScrollRunnable implements Runnable {
669 private int mDirection;
670
671 ScrollRunnable() {
672 }
673
674 public void run() {
675 if (mDragScroller != null) {
676 if (mDirection == SCROLL_LEFT) {
677 mDragScroller.scrollLeft();
678 } else {
679 mDragScroller.scrollRight();
680 }
681 mScrollState = SCROLL_OUTSIDE_ZONE;
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700682 mDistanceSinceScroll = 0;
683 mDragScroller.onExitScrollArea();
Joe Onorato00acb122009-08-04 16:04:30 -0400684 }
685 }
686
687 void setDirection(int direction) {
688 mDirection = direction;
689 }
690 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800691}