blob: 4c429531936205f939465af2ceba94f56f73f969 [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;
Winson Chung6e1bdaf2012-05-29 17:03:45 -070020import android.content.res.Resources;
Joe Onorato00acb122009-08-04 16:04:30 -040021import android.graphics.Bitmap;
Winson Chungb8c69f32011-10-19 21:36:08 -070022import android.graphics.Point;
Winson Chung043f2af2012-03-01 16:09:54 -080023import android.graphics.PointF;
Joe Onorato00acb122009-08-04 16:04:30 -040024import android.graphics.Rect;
Joe Onorato00acb122009-08-04 16:04:30 -040025import android.os.Handler;
Michael Jurka0280c3b2010-09-17 15:00:07 -070026import android.os.IBinder;
Joe Onorato00acb122009-08-04 16:04:30 -040027import android.os.Vibrator;
Joe Onorato00acb122009-08-04 16:04:30 -040028import android.util.Log;
Joe Onorato00acb122009-08-04 16:04:30 -040029import android.view.KeyEvent;
30import android.view.MotionEvent;
Winson Chung043f2af2012-03-01 16:09:54 -080031import android.view.VelocityTracker;
Michael Jurka0280c3b2010-09-17 15:00:07 -070032import android.view.View;
Patrick Dubroya16fd5a2010-10-07 16:47:28 -070033import android.view.ViewConfiguration;
Joe Onorato00acb122009-08-04 16:04:30 -040034import android.view.inputmethod.InputMethodManager;
Joe Onorato00acb122009-08-04 16:04:30 -040035
Adam Cohen120980b2010-12-08 11:05:37 -080036import com.android.launcher.R;
Adam Cohenc0dcf592011-06-01 15:30:43 -070037
38import java.util.ArrayList;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080039
40/**
Joe Onorato00acb122009-08-04 16:04:30 -040041 * Class for initiating a drag within a view or across multiple views.
The Android Open Source Project31dd5032009-03-03 19:32:27 -080042 */
Joe Onorato00acb122009-08-04 16:04:30 -040043public class DragController {
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
Winson Chungaa15ffe2012-01-18 15:45:28 -080052 private static final int SCROLL_DELAY = 500;
53 private static final int RESCROLL_DELAY = 750;
Winson Chung61b0c692012-02-23 16:31:13 -080054 private static final int VIBRATE_DURATION = 15;
Joe Onorato00acb122009-08-04 16:04:30 -040055
56 private static final boolean PROFILE_DRAWING_DURING_DRAG = false;
57
58 private static final int SCROLL_OUTSIDE_ZONE = 0;
59 private static final int SCROLL_WAITING_IN_ZONE = 1;
60
Patrick Dubroy54fa3b92010-11-17 12:18:45 -080061 static final int SCROLL_NONE = -1;
Patrick Dubroy1262e362010-10-06 15:49:50 -070062 static final int SCROLL_LEFT = 0;
63 static final int SCROLL_RIGHT = 1;
Joe Onorato00acb122009-08-04 16:04:30 -040064
Winson Chung043f2af2012-03-01 16:09:54 -080065 private static final float MAX_FLING_DEGREES = 35f;
Winson Chung043f2af2012-03-01 16:09:54 -080066
Adam Cohen8dfcba42011-07-07 16:38:18 -070067 private Launcher mLauncher;
Joe Onorato00acb122009-08-04 16:04:30 -040068 private Handler mHandler;
Jeff Brown8ef85c72012-04-13 02:58:38 -070069 private final Vibrator mVibrator;
Joe Onorato00acb122009-08-04 16:04:30 -040070
71 // temporaries to avoid gc thrash
72 private Rect mRectTemp = new Rect();
73 private final int[] mCoordinatesTemp = new int[2];
74
75 /** Whether or not we're dragging. */
76 private boolean mDragging;
77
78 /** X coordinate of the down event. */
Adam Cohene3e27a82011-04-15 12:07:39 -070079 private int mMotionDownX;
Joe Onorato00acb122009-08-04 16:04:30 -040080
81 /** Y coordinate of the down event. */
Adam Cohene3e27a82011-04-15 12:07:39 -070082 private int mMotionDownY;
Joe Onorato00acb122009-08-04 16:04:30 -040083
Joe Onorato658db742010-09-29 11:40:39 -070084 /** the area at the edge of the screen that makes the workspace go left
85 * or right while you're dragging.
86 */
87 private int mScrollZone;
88
Adam Cohen9932a9b2011-08-02 22:14:07 -070089 private DropTarget.DragObject mDragObject;
Joe Onorato00acb122009-08-04 16:04:30 -040090
91 /** Who can receive drop events */
92 private ArrayList<DropTarget> mDropTargets = new ArrayList<DropTarget>();
Patrick Dubroy4ed62782010-08-17 15:11:18 -070093 private ArrayList<DragListener> mListeners = new ArrayList<DragListener>();
Winson Chung043f2af2012-03-01 16:09:54 -080094 private DropTarget mFlingToDeleteDropTarget;
Joe Onorato00acb122009-08-04 16:04:30 -040095
96 /** The window token used as the parent for the DragView. */
97 private IBinder mWindowToken;
98
99 /** The view that will be scrolled when dragging to the left and right edges of the screen. */
100 private View mScrollView;
101
Romain Guyea3763c2010-01-11 18:02:04 -0800102 private View mMoveTarget;
103
Joe Onorato00acb122009-08-04 16:04:30 -0400104 private DragScroller mDragScroller;
105 private int mScrollState = SCROLL_OUTSIDE_ZONE;
106 private ScrollRunnable mScrollRunnable = new ScrollRunnable();
107
Joe Onorato00acb122009-08-04 16:04:30 -0400108 private DropTarget mLastDropTarget;
109
110 private InputMethodManager mInputMethodManager;
111
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700112 private int mLastTouch[] = new int[2];
Winson Chung318eee02012-04-12 10:59:27 -0700113 private long mLastTouchUpTime = -1;
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700114 private int mDistanceSinceScroll = 0;
115
Winson Chung273c1022011-07-11 13:40:52 -0700116 private int mTmpPoint[] = new int[2];
117 private Rect mDragLayerRect = new Rect();
118
Winson Chung043f2af2012-03-01 16:09:54 -0800119 protected int mFlingToDeleteThresholdVelocity;
120 private VelocityTracker mVelocityTracker;
121
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800122 /**
123 * Interface to receive notifications when a drag starts or stops
124 */
125 interface DragListener {
126
127 /**
128 * A drag has begun
129 *
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800130 * @param source An object representing where the drag originated
131 * @param info The data associated with the object that is being dragged
132 * @param dragAction The drag action: either {@link DragController#DRAG_ACTION_MOVE}
133 * or {@link DragController#DRAG_ACTION_COPY}
134 */
Joe Onorato5162ea92009-09-03 09:39:42 -0700135 void onDragStart(DragSource source, Object info, int dragAction);
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800136
137 /**
Winson Chunge3193b92010-09-10 11:44:42 -0700138 * The drag has ended
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800139 */
140 void onDragEnd();
141 }
142
143 /**
Joe Onorato00acb122009-08-04 16:04:30 -0400144 * Used to create a new DragLayer from XML.
145 *
146 * @param context The application's context.
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800147 */
Adam Cohen8dfcba42011-07-07 16:38:18 -0700148 public DragController(Launcher launcher) {
Winson Chung6e1bdaf2012-05-29 17:03:45 -0700149 Resources r = launcher.getResources();
Adam Cohen8dfcba42011-07-07 16:38:18 -0700150 mLauncher = launcher;
Joe Onorato00acb122009-08-04 16:04:30 -0400151 mHandler = new Handler();
Winson Chung6e1bdaf2012-05-29 17:03:45 -0700152 mScrollZone = r.getDimensionPixelSize(R.dimen.scroll_zone);
Winson Chung043f2af2012-03-01 16:09:54 -0800153 mVelocityTracker = VelocityTracker.obtain();
Winson Chung6e1bdaf2012-05-29 17:03:45 -0700154 mVibrator = (Vibrator) launcher.getSystemService(Context.VIBRATOR_SERVICE);
Winson Chung043f2af2012-03-01 16:09:54 -0800155
Winson Chung6e1bdaf2012-05-29 17:03:45 -0700156 float density = r.getDisplayMetrics().density;
157 mFlingToDeleteThresholdVelocity =
158 (int) (r.getInteger(R.integer.config_flingToDeleteMinVelocity) * density);
Joe Onorato00acb122009-08-04 16:04:30 -0400159 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800160
Patrick Dubroy1262e362010-10-06 15:49:50 -0700161 public boolean dragging() {
162 return mDragging;
163 }
164
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800165 /**
Joe Onorato5162ea92009-09-03 09:39:42 -0700166 * Starts a drag.
Michael Jurkaa63c4522010-08-19 13:52:27 -0700167 *
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800168 * @param v The view that is being dragged
Winson Chunge3193b92010-09-10 11:44:42 -0700169 * @param bmp The bitmap that represents the view being dragged
170 * @param source An object representing where the drag originated
171 * @param dragInfo The data associated with the object that is being dragged
172 * @param dragAction The drag action: either {@link #DRAG_ACTION_MOVE} or
173 * {@link #DRAG_ACTION_COPY}
174 * @param dragRegion Coordinates within the bitmap b for the position of item being dragged.
175 * Makes dragging feel more precise, e.g. you can clip out a transparent border
176 */
177 public void startDrag(View v, Bitmap bmp, DragSource source, Object dragInfo, int dragAction,
Winson Chung72d59842012-02-22 13:51:36 -0800178 Rect dragRegion, float initialDragViewScale) {
Winson Chunge3193b92010-09-10 11:44:42 -0700179 int[] loc = mCoordinatesTemp;
Adam Cohen8dfcba42011-07-07 16:38:18 -0700180 mLauncher.getDragLayer().getLocationInDragLayer(v, loc);
Winson Chung72d59842012-02-22 13:51:36 -0800181 int dragLayerX = loc[0] + v.getPaddingLeft() +
182 (int) ((initialDragViewScale * bmp.getWidth() - bmp.getWidth()) / 2);
183 int dragLayerY = loc[1] + v.getPaddingTop() +
184 (int) ((initialDragViewScale * bmp.getHeight() - bmp.getHeight()) / 2);
Winson Chunge3193b92010-09-10 11:44:42 -0700185
Winson Chung72d59842012-02-22 13:51:36 -0800186 startDrag(bmp, dragLayerX, dragLayerY, source, dragInfo, dragAction, null, dragRegion,
187 initialDragViewScale);
Winson Chunge3193b92010-09-10 11:44:42 -0700188
189 if (dragAction == DRAG_ACTION_MOVE) {
190 v.setVisibility(View.GONE);
191 }
192 }
193
194 /**
195 * Starts a drag.
196 *
Joe Onorato5162ea92009-09-03 09:39:42 -0700197 * @param b The bitmap to display as the drag image. It will be re-scaled to the
198 * enlarged size.
Adam Cohen8dfcba42011-07-07 16:38:18 -0700199 * @param dragLayerX The x position in the DragLayer of the left-top of the bitmap.
200 * @param dragLayerY The y position in the DragLayer of the left-top of the bitmap.
Joe Onorato5162ea92009-09-03 09:39:42 -0700201 * @param source An object representing where the drag originated
Romain Guyea3763c2010-01-11 18:02:04 -0800202 * @param dragInfo The data associated with the object that is being dragged
Joe Onorato5162ea92009-09-03 09:39:42 -0700203 * @param dragAction The drag action: either {@link #DRAG_ACTION_MOVE} or
204 * {@link #DRAG_ACTION_COPY}
Michael Jurkaa63c4522010-08-19 13:52:27 -0700205 * @param dragRegion Coordinates within the bitmap b for the position of item being dragged.
206 * Makes dragging feel more precise, e.g. you can clip out a transparent border
207 */
Adam Cohen8dfcba42011-07-07 16:38:18 -0700208 public void startDrag(Bitmap b, int dragLayerX, int dragLayerY,
Winson Chung72d59842012-02-22 13:51:36 -0800209 DragSource source, Object dragInfo, int dragAction, Point dragOffset, Rect dragRegion,
210 float initialDragViewScale) {
Joe Onorato00acb122009-08-04 16:04:30 -0400211 if (PROFILE_DRAWING_DURING_DRAG) {
212 android.os.Debug.startMethodTracing("Launcher");
213 }
214
215 // Hide soft keyboard, if visible
216 if (mInputMethodManager == null) {
217 mInputMethodManager = (InputMethodManager)
Adam Cohen8dfcba42011-07-07 16:38:18 -0700218 mLauncher.getSystemService(Context.INPUT_METHOD_SERVICE);
Joe Onorato00acb122009-08-04 16:04:30 -0400219 }
220 mInputMethodManager.hideSoftInputFromWindow(mWindowToken, 0);
221
Patrick Dubroy4ed62782010-08-17 15:11:18 -0700222 for (DragListener listener : mListeners) {
223 listener.onDragStart(source, dragInfo, dragAction);
Joe Onorato00acb122009-08-04 16:04:30 -0400224 }
225
Adam Cohen8dfcba42011-07-07 16:38:18 -0700226 final int registrationX = mMotionDownX - dragLayerX;
227 final int registrationY = mMotionDownY - dragLayerY;
Joe Onorato00acb122009-08-04 16:04:30 -0400228
Michael Jurkaa63c4522010-08-19 13:52:27 -0700229 final int dragRegionLeft = dragRegion == null ? 0 : dragRegion.left;
230 final int dragRegionTop = dragRegion == null ? 0 : dragRegion.top;
Adam Cohene3e27a82011-04-15 12:07:39 -0700231
Joe Onorato00acb122009-08-04 16:04:30 -0400232 mDragging = true;
Adam Cohencb3382b2011-05-24 14:07:08 -0700233
Adam Cohen9932a9b2011-08-02 22:14:07 -0700234 mDragObject = new DropTarget.DragObject();
235
Adam Cohenbfbfd262011-06-13 16:55:12 -0700236 mDragObject.dragComplete = false;
Adam Cohen8dfcba42011-07-07 16:38:18 -0700237 mDragObject.xOffset = mMotionDownX - (dragLayerX + dragRegionLeft);
238 mDragObject.yOffset = mMotionDownY - (dragLayerY + dragRegionTop);
Adam Cohencb3382b2011-05-24 14:07:08 -0700239 mDragObject.dragSource = source;
240 mDragObject.dragInfo = dragInfo;
Joe Onorato00acb122009-08-04 16:04:30 -0400241
242 mVibrator.vibrate(VIBRATE_DURATION);
243
Adam Cohen8dfcba42011-07-07 16:38:18 -0700244 final DragView dragView = mDragObject.dragView = new DragView(mLauncher, b, registrationX,
Winson Chung72d59842012-02-22 13:51:36 -0800245 registrationY, 0, 0, b.getWidth(), b.getHeight(), initialDragViewScale);
Michael Jurkaa63c4522010-08-19 13:52:27 -0700246
Winson Chungb8c69f32011-10-19 21:36:08 -0700247 if (dragOffset != null) {
248 dragView.setDragVisualizeOffset(new Point(dragOffset));
249 }
Michael Jurkaa63c4522010-08-19 13:52:27 -0700250 if (dragRegion != null) {
Adam Cohene3e27a82011-04-15 12:07:39 -0700251 dragView.setDragRegion(new Rect(dragRegion));
Michael Jurkaa63c4522010-08-19 13:52:27 -0700252 }
253
Adam Cohen8dfcba42011-07-07 16:38:18 -0700254 dragView.show(mMotionDownX, mMotionDownY);
255 handleMoveEvent(mMotionDownX, mMotionDownY);
Joe Onorato00acb122009-08-04 16:04:30 -0400256 }
257
258 /**
259 * Draw the view into a bitmap.
260 */
Adam Cohen120980b2010-12-08 11:05:37 -0800261 Bitmap getViewBitmap(View v) {
Joe Onorato00acb122009-08-04 16:04:30 -0400262 v.clearFocus();
263 v.setPressed(false);
264
265 boolean willNotCache = v.willNotCacheDrawing();
266 v.setWillNotCacheDrawing(false);
267
268 // Reset the drawing cache background color to fully transparent
269 // for the duration of this operation
270 int color = v.getDrawingCacheBackgroundColor();
271 v.setDrawingCacheBackgroundColor(0);
Adam Cohen120980b2010-12-08 11:05:37 -0800272 float alpha = v.getAlpha();
273 v.setAlpha(1.0f);
Joe Onorato00acb122009-08-04 16:04:30 -0400274
275 if (color != 0) {
276 v.destroyDrawingCache();
277 }
278 v.buildDrawingCache();
279 Bitmap cacheBitmap = v.getDrawingCache();
Daniel Sandler3f8175a2010-05-25 11:48:32 -0400280 if (cacheBitmap == null) {
281 Log.e(TAG, "failed getViewBitmap(" + v + ")", new RuntimeException());
282 return null;
283 }
Joe Onorato00acb122009-08-04 16:04:30 -0400284
285 Bitmap bitmap = Bitmap.createBitmap(cacheBitmap);
286
287 // Restore the view
288 v.destroyDrawingCache();
Adam Cohen120980b2010-12-08 11:05:37 -0800289 v.setAlpha(alpha);
Joe Onorato00acb122009-08-04 16:04:30 -0400290 v.setWillNotCacheDrawing(willNotCache);
291 v.setDrawingCacheBackgroundColor(color);
292
293 return bitmap;
294 }
295
296 /**
297 * Call this from a drag source view like this:
298 *
299 * <pre>
300 * @Override
301 * public boolean dispatchKeyEvent(KeyEvent event) {
302 * return mDragController.dispatchKeyEvent(this, event)
303 * || super.dispatchKeyEvent(event);
304 * </pre>
305 */
306 public boolean dispatchKeyEvent(KeyEvent event) {
307 return mDragging;
308 }
309
Winson Chung304dcde2011-01-07 11:17:23 -0800310 public boolean isDragging() {
311 return mDragging;
312 }
313
Joe Onorato24b6fd82009-11-12 13:47:09 -0800314 /**
315 * Stop dragging without dropping.
316 */
317 public void cancelDrag() {
Winson Chung621e6402011-01-04 16:03:57 -0800318 if (mDragging) {
Winson Chungc07918d2011-07-01 15:35:26 -0700319 if (mLastDropTarget != null) {
320 mLastDropTarget.onDragExit(mDragObject);
321 }
Winson Chung41bb19d2012-03-05 18:36:46 -0800322 mDragObject.deferDragViewCleanupPostAnimation = false;
Adam Cohen36cc09b2011-09-29 17:33:15 -0700323 mDragObject.cancelled = true;
Adam Cohenbfbfd262011-06-13 16:55:12 -0700324 mDragObject.dragComplete = true;
Winson Chunga48487a2012-03-20 16:19:37 -0700325 mDragObject.dragSource.onDropCompleted(null, mDragObject, false, false);
Winson Chung621e6402011-01-04 16:03:57 -0800326 }
Joe Onorato24b6fd82009-11-12 13:47:09 -0800327 endDrag();
328 }
Winson Chungcd810732012-06-18 16:45:43 -0700329 public void onAppsRemoved(ArrayList<String> packageNames, Context context) {
Winson Chunga1820962011-10-03 16:31:06 -0700330 // Cancel the current drag if we are removing an app that we are dragging
331 if (mDragObject != null) {
332 Object rawDragInfo = mDragObject.dragInfo;
333 if (rawDragInfo instanceof ShortcutInfo) {
334 ShortcutInfo dragInfo = (ShortcutInfo) rawDragInfo;
Winson Chungcd810732012-06-18 16:45:43 -0700335 for (String pn : packageNames) {
Michael Jurka7bcadad2012-04-02 07:23:44 -0700336 // Added null checks to prevent NPE we've seen in the wild
337 if (dragInfo != null &&
Winson Chungcd810732012-06-18 16:45:43 -0700338 dragInfo.intent != null) {
339 boolean isSamePackage = dragInfo.getPackageName().equals(pn);
Winson Chung11a49372012-04-27 15:12:38 -0700340 if (isSamePackage) {
341 cancelDrag();
342 return;
343 }
Winson Chunga1820962011-10-03 16:31:06 -0700344 }
345 }
346 }
347 }
348 }
Joe Onorato24b6fd82009-11-12 13:47:09 -0800349
Joe Onorato00acb122009-08-04 16:04:30 -0400350 private void endDrag() {
351 if (mDragging) {
352 mDragging = false;
Winson Chungaa15ffe2012-01-18 15:45:28 -0800353 clearScrollRunnable();
Winson Chung043f2af2012-03-01 16:09:54 -0800354 boolean isDeferred = false;
Adam Cohencb3382b2011-05-24 14:07:08 -0700355 if (mDragObject.dragView != null) {
Winson Chung043f2af2012-03-01 16:09:54 -0800356 isDeferred = mDragObject.deferDragViewCleanupPostAnimation;
357 if (!isDeferred) {
Winson Chung7bd1bbb2012-02-13 18:29:29 -0800358 mDragObject.dragView.remove();
359 }
Adam Cohencb3382b2011-05-24 14:07:08 -0700360 mDragObject.dragView = null;
Joe Onorato00acb122009-08-04 16:04:30 -0400361 }
Winson Chung043f2af2012-03-01 16:09:54 -0800362
363 // Only end the drag if we are not deferred
364 if (!isDeferred) {
365 for (DragListener listener : mListeners) {
366 listener.onDragEnd();
367 }
368 }
369 }
370
371 releaseVelocityTracker();
372 }
373
374 /**
375 * This only gets called as a result of drag view cleanup being deferred in endDrag();
376 */
377 void onDeferredEndDrag(DragView dragView) {
378 dragView.remove();
379
380 // If we skipped calling onDragEnd() before, do it now
381 for (DragListener listener : mListeners) {
382 listener.onDragEnd();
Joe Onorato00acb122009-08-04 16:04:30 -0400383 }
384 }
385
Winson Chunga48487a2012-03-20 16:19:37 -0700386 void onDeferredEndFling(DropTarget.DragObject d) {
387 d.dragSource.onFlingToDeleteCompleted();
388 }
389
Joe Onorato00acb122009-08-04 16:04:30 -0400390 /**
Winson Chung273c1022011-07-11 13:40:52 -0700391 * Clamps the position to the drag layer bounds.
392 */
393 private int[] getClampedDragLayerPos(float x, float y) {
394 mLauncher.getDragLayer().getLocalVisibleRect(mDragLayerRect);
395 mTmpPoint[0] = (int) Math.max(mDragLayerRect.left, Math.min(x, mDragLayerRect.right - 1));
396 mTmpPoint[1] = (int) Math.max(mDragLayerRect.top, Math.min(y, mDragLayerRect.bottom - 1));
397 return mTmpPoint;
398 }
399
Winson Chunga2413752012-04-03 14:22:34 -0700400 long getLastGestureUpTime() {
401 if (mDragging) {
402 return System.currentTimeMillis();
403 } else {
404 return mLastTouchUpTime;
405 }
406 }
407
408 void resetLastGestureUpTime() {
409 mLastTouchUpTime = -1;
410 }
411
Winson Chung273c1022011-07-11 13:40:52 -0700412 /**
Joe Onorato00acb122009-08-04 16:04:30 -0400413 * Call this from a drag source view.
414 */
415 public boolean onInterceptTouchEvent(MotionEvent ev) {
Michael Jurka3a9fced2012-04-13 14:44:29 -0700416 @SuppressWarnings("all") // suppress dead code warning
417 final boolean debug = false;
418 if (debug) {
Joe Onoratoa30ce8e2009-11-11 08:16:49 -0800419 Log.d(Launcher.TAG, "DragController.onInterceptTouchEvent " + ev + " mDragging="
Joe Onorato9c1289c2009-08-17 11:03:03 -0400420 + mDragging);
421 }
Joe Onorato00acb122009-08-04 16:04:30 -0400422
Winson Chung043f2af2012-03-01 16:09:54 -0800423 // Update the velocity tracker
424 acquireVelocityTrackerAndAddMovement(ev);
425
426 final int action = ev.getAction();
Winson Chung273c1022011-07-11 13:40:52 -0700427 final int[] dragLayerPos = getClampedDragLayerPos(ev.getX(), ev.getY());
428 final int dragLayerX = dragLayerPos[0];
429 final int dragLayerY = dragLayerPos[1];
Joe Onorato00acb122009-08-04 16:04:30 -0400430
431 switch (action) {
432 case MotionEvent.ACTION_MOVE:
433 break;
Joe Onorato00acb122009-08-04 16:04:30 -0400434 case MotionEvent.ACTION_DOWN:
435 // Remember location of down touch
Adam Cohen8dfcba42011-07-07 16:38:18 -0700436 mMotionDownX = dragLayerX;
437 mMotionDownY = dragLayerY;
Joe Onorato00acb122009-08-04 16:04:30 -0400438 mLastDropTarget = null;
439 break;
Joe Onorato00acb122009-08-04 16:04:30 -0400440 case MotionEvent.ACTION_UP:
Winson Chunga2413752012-04-03 14:22:34 -0700441 mLastTouchUpTime = System.currentTimeMillis();
Joe Onorato00acb122009-08-04 16:04:30 -0400442 if (mDragging) {
Winson Chung043f2af2012-03-01 16:09:54 -0800443 PointF vec = isFlingingToDelete(mDragObject.dragSource);
444 if (vec != null) {
445 dropOnFlingToDeleteTarget(dragLayerX, dragLayerY, vec);
446 } else {
447 drop(dragLayerX, dragLayerY);
448 }
Joe Onorato00acb122009-08-04 16:04:30 -0400449 }
450 endDrag();
451 break;
Winson Chung621e6402011-01-04 16:03:57 -0800452 case MotionEvent.ACTION_CANCEL:
453 cancelDrag();
454 break;
Joe Onorato00acb122009-08-04 16:04:30 -0400455 }
456
457 return mDragging;
458 }
459
460 /**
Romain Guyea3763c2010-01-11 18:02:04 -0800461 * Sets the view that should handle move events.
462 */
463 void setMoveTarget(View view) {
464 mMoveTarget = view;
465 }
466
467 public boolean dispatchUnhandledMove(View focused, int direction) {
468 return mMoveTarget != null && mMoveTarget.dispatchUnhandledMove(focused, direction);
469 }
470
Winson Chungaa15ffe2012-01-18 15:45:28 -0800471 private void clearScrollRunnable() {
472 mHandler.removeCallbacks(mScrollRunnable);
473 if (mScrollState == SCROLL_WAITING_IN_ZONE) {
474 mScrollState = SCROLL_OUTSIDE_ZONE;
475 mScrollRunnable.setDirection(SCROLL_RIGHT);
476 mDragScroller.onExitScrollArea();
Winson Chung360e63f2012-04-27 13:48:05 -0700477 mLauncher.getDragLayer().onExitScrollArea();
Winson Chungaa15ffe2012-01-18 15:45:28 -0800478 }
479 }
480
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700481 private void handleMoveEvent(int x, int y) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700482 mDragObject.dragView.move(x, y);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700483
484 // Drop on someone?
485 final int[] coordinates = mCoordinatesTemp;
486 DropTarget dropTarget = findDropTarget(x, y, coordinates);
Adam Cohencb3382b2011-05-24 14:07:08 -0700487 mDragObject.x = coordinates[0];
488 mDragObject.y = coordinates[1];
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700489 if (dropTarget != null) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700490 DropTarget delegate = dropTarget.getDropTargetDelegate(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700491 if (delegate != null) {
492 dropTarget = delegate;
493 }
494
495 if (mLastDropTarget != dropTarget) {
496 if (mLastDropTarget != null) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700497 mLastDropTarget.onDragExit(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700498 }
Adam Cohencb3382b2011-05-24 14:07:08 -0700499 dropTarget.onDragEnter(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700500 }
Adam Cohencb3382b2011-05-24 14:07:08 -0700501 dropTarget.onDragOver(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700502 } else {
503 if (mLastDropTarget != null) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700504 mLastDropTarget.onDragExit(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700505 }
506 }
507 mLastDropTarget = dropTarget;
508
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700509 // After a scroll, the touch point will still be in the scroll region.
510 // Rather than scrolling immediately, require a bit of twiddling to scroll again
Adam Cohen8dfcba42011-07-07 16:38:18 -0700511 final int slop = ViewConfiguration.get(mLauncher).getScaledWindowTouchSlop();
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700512 mDistanceSinceScroll +=
513 Math.sqrt(Math.pow(mLastTouch[0] - x, 2) + Math.pow(mLastTouch[1] - y, 2));
514 mLastTouch[0] = x;
515 mLastTouch[1] = y;
Winson Chungaa15ffe2012-01-18 15:45:28 -0800516 final int delay = mDistanceSinceScroll < slop ? RESCROLL_DELAY : SCROLL_DELAY;
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700517
Winson Chung3f4e1422011-11-17 14:58:51 -0800518 if (x < mScrollZone) {
Winson Chungaa15ffe2012-01-18 15:45:28 -0800519 if (mScrollState == SCROLL_OUTSIDE_ZONE) {
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700520 mScrollState = SCROLL_WAITING_IN_ZONE;
Winson Chung3e0839e2011-10-03 15:15:18 -0700521 if (mDragScroller.onEnterScrollArea(x, y, SCROLL_LEFT)) {
Winson Chung360e63f2012-04-27 13:48:05 -0700522 mLauncher.getDragLayer().onEnterScrollArea(SCROLL_LEFT);
Winson Chung3e0839e2011-10-03 15:15:18 -0700523 mScrollRunnable.setDirection(SCROLL_LEFT);
Winson Chungaa15ffe2012-01-18 15:45:28 -0800524 mHandler.postDelayed(mScrollRunnable, delay);
Winson Chung3e0839e2011-10-03 15:15:18 -0700525 }
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700526 }
Winson Chung3f4e1422011-11-17 14:58:51 -0800527 } else if (x > mScrollView.getWidth() - mScrollZone) {
Winson Chungaa15ffe2012-01-18 15:45:28 -0800528 if (mScrollState == SCROLL_OUTSIDE_ZONE) {
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700529 mScrollState = SCROLL_WAITING_IN_ZONE;
Winson Chung3e0839e2011-10-03 15:15:18 -0700530 if (mDragScroller.onEnterScrollArea(x, y, SCROLL_RIGHT)) {
Winson Chung360e63f2012-04-27 13:48:05 -0700531 mLauncher.getDragLayer().onEnterScrollArea(SCROLL_RIGHT);
Winson Chung3e0839e2011-10-03 15:15:18 -0700532 mScrollRunnable.setDirection(SCROLL_RIGHT);
Winson Chungaa15ffe2012-01-18 15:45:28 -0800533 mHandler.postDelayed(mScrollRunnable, delay);
Winson Chung3e0839e2011-10-03 15:15:18 -0700534 }
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700535 }
536 } else {
Winson Chungaa15ffe2012-01-18 15:45:28 -0800537 clearScrollRunnable();
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700538 }
539 }
540
Winson Chung3bc21c32012-01-20 13:59:18 -0800541 public void forceMoveEvent() {
542 if (mDragging) {
543 handleMoveEvent(mDragObject.x, mDragObject.y);
544 }
545 }
546
Romain Guyea3763c2010-01-11 18:02:04 -0800547 /**
Joe Onorato00acb122009-08-04 16:04:30 -0400548 * Call this from a drag source view.
549 */
550 public boolean onTouchEvent(MotionEvent ev) {
Joe Onorato00acb122009-08-04 16:04:30 -0400551 if (!mDragging) {
552 return false;
553 }
554
Winson Chung043f2af2012-03-01 16:09:54 -0800555 // Update the velocity tracker
556 acquireVelocityTrackerAndAddMovement(ev);
557
Joe Onorato00acb122009-08-04 16:04:30 -0400558 final int action = ev.getAction();
Winson Chung273c1022011-07-11 13:40:52 -0700559 final int[] dragLayerPos = getClampedDragLayerPos(ev.getX(), ev.getY());
560 final int dragLayerX = dragLayerPos[0];
561 final int dragLayerY = dragLayerPos[1];
Joe Onorato00acb122009-08-04 16:04:30 -0400562
563 switch (action) {
564 case MotionEvent.ACTION_DOWN:
Joe Onorato00acb122009-08-04 16:04:30 -0400565 // Remember where the motion event started
Adam Cohen8dfcba42011-07-07 16:38:18 -0700566 mMotionDownX = dragLayerX;
567 mMotionDownY = dragLayerY;
Joe Onorato00acb122009-08-04 16:04:30 -0400568
Adam Cohen8dfcba42011-07-07 16:38:18 -0700569 if ((dragLayerX < mScrollZone) || (dragLayerX > mScrollView.getWidth() - mScrollZone)) {
Joe Onorato00acb122009-08-04 16:04:30 -0400570 mScrollState = SCROLL_WAITING_IN_ZONE;
571 mHandler.postDelayed(mScrollRunnable, SCROLL_DELAY);
572 } else {
573 mScrollState = SCROLL_OUTSIDE_ZONE;
574 }
Joe Onorato00acb122009-08-04 16:04:30 -0400575 break;
576 case MotionEvent.ACTION_MOVE:
Adam Cohen8dfcba42011-07-07 16:38:18 -0700577 handleMoveEvent(dragLayerX, dragLayerY);
Joe Onorato00acb122009-08-04 16:04:30 -0400578 break;
579 case MotionEvent.ACTION_UP:
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800580 // Ensure that we've processed a move event at the current pointer location.
Adam Cohen8dfcba42011-07-07 16:38:18 -0700581 handleMoveEvent(dragLayerX, dragLayerY);
Winson Chung3bc21c32012-01-20 13:59:18 -0800582 mHandler.removeCallbacks(mScrollRunnable);
Winson Chung043f2af2012-03-01 16:09:54 -0800583
Joe Onorato00acb122009-08-04 16:04:30 -0400584 if (mDragging) {
Winson Chung043f2af2012-03-01 16:09:54 -0800585 PointF vec = isFlingingToDelete(mDragObject.dragSource);
586 if (vec != null) {
587 dropOnFlingToDeleteTarget(dragLayerX, dragLayerY, vec);
588 } else {
589 drop(dragLayerX, dragLayerY);
590 }
Joe Onorato00acb122009-08-04 16:04:30 -0400591 }
592 endDrag();
Joe Onorato00acb122009-08-04 16:04:30 -0400593 break;
594 case MotionEvent.ACTION_CANCEL:
Winson Chung3bc21c32012-01-20 13:59:18 -0800595 mHandler.removeCallbacks(mScrollRunnable);
Joe Onorato24b6fd82009-11-12 13:47:09 -0800596 cancelDrag();
Winson Chung621e6402011-01-04 16:03:57 -0800597 break;
Joe Onorato00acb122009-08-04 16:04:30 -0400598 }
599
600 return true;
601 }
602
Winson Chung043f2af2012-03-01 16:09:54 -0800603 /**
604 * Determines whether the user flung the current item to delete it.
605 *
606 * @return the vector at which the item was flung, or null if no fling was detected.
607 */
608 private PointF isFlingingToDelete(DragSource source) {
609 if (mFlingToDeleteDropTarget == null) return null;
610 if (!source.supportsFlingToDelete()) return null;
611
612 ViewConfiguration config = ViewConfiguration.get(mLauncher);
613 mVelocityTracker.computeCurrentVelocity(1000, config.getScaledMaximumFlingVelocity());
614
615 if (mVelocityTracker.getYVelocity() < mFlingToDeleteThresholdVelocity) {
616 // Do a quick dot product test to ensure that we are flinging upwards
617 PointF vel = new PointF(mVelocityTracker.getXVelocity(),
618 mVelocityTracker.getYVelocity());
619 PointF upVec = new PointF(0f, -1f);
620 float theta = (float) Math.acos(((vel.x * upVec.x) + (vel.y * upVec.y)) /
621 (vel.length() * upVec.length()));
622 if (theta <= Math.toRadians(MAX_FLING_DEGREES)) {
623 return vel;
624 }
625 }
626 return null;
627 }
628
629 private void dropOnFlingToDeleteTarget(float x, float y, PointF vel) {
630 final int[] coordinates = mCoordinatesTemp;
631
632 mDragObject.x = coordinates[0];
633 mDragObject.y = coordinates[1];
Winson Chung043f2af2012-03-01 16:09:54 -0800634
635 // Clean up dragging on the target if it's not the current fling delete target otherwise,
636 // start dragging to it.
637 if (mLastDropTarget != null && mFlingToDeleteDropTarget != mLastDropTarget) {
638 mLastDropTarget.onDragExit(mDragObject);
639 }
640
641 // Drop onto the fling-to-delete target
642 boolean accepted = false;
643 mFlingToDeleteDropTarget.onDragEnter(mDragObject);
Winson Chung232decb2012-03-28 15:09:05 -0700644 // We must set dragComplete to true _only_ after we "enter" the fling-to-delete target for
645 // "drop"
646 mDragObject.dragComplete = true;
Winson Chung043f2af2012-03-01 16:09:54 -0800647 mFlingToDeleteDropTarget.onDragExit(mDragObject);
648 if (mFlingToDeleteDropTarget.acceptDrop(mDragObject)) {
649 mFlingToDeleteDropTarget.onFlingToDelete(mDragObject, mDragObject.x, mDragObject.y,
650 vel);
651 accepted = true;
652 }
Winson Chunga48487a2012-03-20 16:19:37 -0700653 mDragObject.dragSource.onDropCompleted((View) mFlingToDeleteDropTarget, mDragObject, true,
Winson Chung043f2af2012-03-01 16:09:54 -0800654 accepted);
655 }
656
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800657 private void drop(float x, float y) {
Joe Onorato00acb122009-08-04 16:04:30 -0400658 final int[] coordinates = mCoordinatesTemp;
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800659 final DropTarget dropTarget = findDropTarget((int) x, (int) y, coordinates);
Joe Onorato00acb122009-08-04 16:04:30 -0400660
Adam Cohencb3382b2011-05-24 14:07:08 -0700661 mDragObject.x = coordinates[0];
662 mDragObject.y = coordinates[1];
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800663 boolean accepted = false;
Joe Onorato00acb122009-08-04 16:04:30 -0400664 if (dropTarget != null) {
Adam Cohenbfbfd262011-06-13 16:55:12 -0700665 mDragObject.dragComplete = true;
Adam Cohencb3382b2011-05-24 14:07:08 -0700666 dropTarget.onDragExit(mDragObject);
667 if (dropTarget.acceptDrop(mDragObject)) {
668 dropTarget.onDrop(mDragObject);
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800669 accepted = true;
Joe Onorato00acb122009-08-04 16:04:30 -0400670 }
671 }
Winson Chunga48487a2012-03-20 16:19:37 -0700672 mDragObject.dragSource.onDropCompleted((View) dropTarget, mDragObject, false, accepted);
Joe Onorato00acb122009-08-04 16:04:30 -0400673 }
674
675 private DropTarget findDropTarget(int x, int y, int[] dropCoordinates) {
676 final Rect r = mRectTemp;
677
678 final ArrayList<DropTarget> dropTargets = mDropTargets;
679 final int count = dropTargets.size();
680 for (int i=count-1; i>=0; i--) {
Patrick Dubroy440c3602010-07-13 17:50:32 -0700681 DropTarget target = dropTargets.get(i);
Michael Jurka0280c3b2010-09-17 15:00:07 -0700682 if (!target.isDropEnabled())
683 continue;
684
Joe Onorato00acb122009-08-04 16:04:30 -0400685 target.getHitRect(r);
Patrick Dubroy440c3602010-07-13 17:50:32 -0700686
Adam Cohen8dfcba42011-07-07 16:38:18 -0700687 // Convert the hit rect to DragLayer coordinates
688 target.getLocationInDragLayer(dropCoordinates);
Joe Onorato00acb122009-08-04 16:04:30 -0400689 r.offset(dropCoordinates[0] - target.getLeft(), dropCoordinates[1] - target.getTop());
Patrick Dubroy440c3602010-07-13 17:50:32 -0700690
Adam Cohencb3382b2011-05-24 14:07:08 -0700691 mDragObject.x = x;
692 mDragObject.y = y;
Joe Onorato00acb122009-08-04 16:04:30 -0400693 if (r.contains(x, y)) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700694 DropTarget delegate = target.getDropTargetDelegate(mDragObject);
Patrick Dubroy440c3602010-07-13 17:50:32 -0700695 if (delegate != null) {
696 target = delegate;
Adam Cohen8dfcba42011-07-07 16:38:18 -0700697 target.getLocationInDragLayer(dropCoordinates);
Patrick Dubroy440c3602010-07-13 17:50:32 -0700698 }
699
700 // Make dropCoordinates relative to the DropTarget
Joe Onorato00acb122009-08-04 16:04:30 -0400701 dropCoordinates[0] = x - dropCoordinates[0];
702 dropCoordinates[1] = y - dropCoordinates[1];
Patrick Dubroy440c3602010-07-13 17:50:32 -0700703
Joe Onorato00acb122009-08-04 16:04:30 -0400704 return target;
705 }
706 }
707 return null;
708 }
709
710 public void setDragScoller(DragScroller scroller) {
711 mDragScroller = scroller;
712 }
713
714 public void setWindowToken(IBinder token) {
715 mWindowToken = token;
716 }
717
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800718 /**
719 * Sets the drag listner which will be notified when a drag starts or ends.
720 */
Patrick Dubroy4ed62782010-08-17 15:11:18 -0700721 public void addDragListener(DragListener l) {
722 mListeners.add(l);
Joe Onorato00acb122009-08-04 16:04:30 -0400723 }
724
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800725 /**
726 * Remove a previously installed drag listener.
727 */
Joe Onorato00acb122009-08-04 16:04:30 -0400728 public void removeDragListener(DragListener l) {
Patrick Dubroy4ed62782010-08-17 15:11:18 -0700729 mListeners.remove(l);
Joe Onorato00acb122009-08-04 16:04:30 -0400730 }
731
732 /**
733 * Add a DropTarget to the list of potential places to receive drop events.
734 */
735 public void addDropTarget(DropTarget target) {
736 mDropTargets.add(target);
737 }
738
739 /**
740 * Don't send drop events to <em>target</em> any more.
741 */
742 public void removeDropTarget(DropTarget target) {
743 mDropTargets.remove(target);
744 }
745
746 /**
Winson Chung043f2af2012-03-01 16:09:54 -0800747 * Sets the current fling-to-delete drop target.
748 */
749 public void setFlingToDeleteDropTarget(DropTarget target) {
750 mFlingToDeleteDropTarget = target;
751 }
752
753 private void acquireVelocityTrackerAndAddMovement(MotionEvent ev) {
754 if (mVelocityTracker == null) {
755 mVelocityTracker = VelocityTracker.obtain();
756 }
757 mVelocityTracker.addMovement(ev);
758 }
759
760 private void releaseVelocityTracker() {
761 if (mVelocityTracker != null) {
762 mVelocityTracker.recycle();
763 mVelocityTracker = null;
764 }
765 }
766
767 /**
Joe Onorato00acb122009-08-04 16:04:30 -0400768 * Set which view scrolls for touch events near the edge of the screen.
769 */
770 public void setScrollView(View v) {
771 mScrollView = v;
772 }
773
Patrick Dubroy5f445422011-02-18 14:35:21 -0800774 DragView getDragView() {
Adam Cohencb3382b2011-05-24 14:07:08 -0700775 return mDragObject.dragView;
Patrick Dubroy5f445422011-02-18 14:35:21 -0800776 }
777
Joe Onorato00acb122009-08-04 16:04:30 -0400778 private class ScrollRunnable implements Runnable {
779 private int mDirection;
780
781 ScrollRunnable() {
782 }
783
784 public void run() {
785 if (mDragScroller != null) {
786 if (mDirection == SCROLL_LEFT) {
787 mDragScroller.scrollLeft();
788 } else {
789 mDragScroller.scrollRight();
790 }
791 mScrollState = SCROLL_OUTSIDE_ZONE;
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700792 mDistanceSinceScroll = 0;
793 mDragScroller.onExitScrollArea();
Winson Chung360e63f2012-04-27 13:48:05 -0700794 mLauncher.getDragLayer().onExitScrollArea();
Winson Chungaa15ffe2012-01-18 15:45:28 -0800795
796 if (isDragging()) {
797 // Force an update so that we can requeue the scroller if necessary
Winson Chung3bc21c32012-01-20 13:59:18 -0800798 forceMoveEvent();
Winson Chungaa15ffe2012-01-18 15:45:28 -0800799 }
Joe Onorato00acb122009-08-04 16:04:30 -0400800 }
801 }
802
803 void setDirection(int direction) {
804 mDirection = direction;
805 }
806 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800807}