blob: 4c4f399a920853bf709a327db083430b34e67934 [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
Daniel Sandler325dc232013-06-05 22:57:57 -040017package com.android.launcher3;
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
Daniel Sandler325dc232013-06-05 22:57:57 -040036import com.android.launcher3.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 {
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800126 /**
127 * A drag has begun
Mindy DelliCarpini53b8d072013-07-03 08:23:13 -0700128 *
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800129 * @param source An object representing where the drag originated
130 * @param info The data associated with the object that is being dragged
131 * @param dragAction The drag action: either {@link DragController#DRAG_ACTION_MOVE}
132 * or {@link DragController#DRAG_ACTION_COPY}
133 */
Joe Onorato5162ea92009-09-03 09:39:42 -0700134 void onDragStart(DragSource source, Object info, int dragAction);
Mindy DelliCarpini53b8d072013-07-03 08:23:13 -0700135
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800136 /**
Winson Chunge3193b92010-09-10 11:44:42 -0700137 * The drag has ended
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800138 */
139 void onDragEnd();
140 }
141
142 /**
Joe Onorato00acb122009-08-04 16:04:30 -0400143 * Used to create a new DragLayer from XML.
144 *
145 * @param context The application's context.
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800146 */
Adam Cohen8dfcba42011-07-07 16:38:18 -0700147 public DragController(Launcher launcher) {
Winson Chung6e1bdaf2012-05-29 17:03:45 -0700148 Resources r = launcher.getResources();
Adam Cohen8dfcba42011-07-07 16:38:18 -0700149 mLauncher = launcher;
Joe Onorato00acb122009-08-04 16:04:30 -0400150 mHandler = new Handler();
Winson Chung6e1bdaf2012-05-29 17:03:45 -0700151 mScrollZone = r.getDimensionPixelSize(R.dimen.scroll_zone);
Winson Chung043f2af2012-03-01 16:09:54 -0800152 mVelocityTracker = VelocityTracker.obtain();
Winson Chung6e1bdaf2012-05-29 17:03:45 -0700153 mVibrator = (Vibrator) launcher.getSystemService(Context.VIBRATOR_SERVICE);
Winson Chung043f2af2012-03-01 16:09:54 -0800154
Winson Chung6e1bdaf2012-05-29 17:03:45 -0700155 float density = r.getDisplayMetrics().density;
156 mFlingToDeleteThresholdVelocity =
157 (int) (r.getInteger(R.integer.config_flingToDeleteMinVelocity) * density);
Joe Onorato00acb122009-08-04 16:04:30 -0400158 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800159
Patrick Dubroy1262e362010-10-06 15:49:50 -0700160 public boolean dragging() {
161 return mDragging;
162 }
163
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800164 /**
Joe Onorato5162ea92009-09-03 09:39:42 -0700165 * Starts a drag.
Michael Jurkaa63c4522010-08-19 13:52:27 -0700166 *
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800167 * @param v The view that is being dragged
Winson Chunge3193b92010-09-10 11:44:42 -0700168 * @param bmp The bitmap that represents the view being dragged
169 * @param source An object representing where the drag originated
170 * @param dragInfo The data associated with the object that is being dragged
171 * @param dragAction The drag action: either {@link #DRAG_ACTION_MOVE} or
172 * {@link #DRAG_ACTION_COPY}
173 * @param dragRegion Coordinates within the bitmap b for the position of item being dragged.
174 * Makes dragging feel more precise, e.g. you can clip out a transparent border
175 */
176 public void startDrag(View v, Bitmap bmp, DragSource source, Object dragInfo, int dragAction,
Michael Jurka05713af2013-01-23 12:39:24 +0100177 Point extraPadding, float initialDragViewScale) {
Winson Chunge3193b92010-09-10 11:44:42 -0700178 int[] loc = mCoordinatesTemp;
Adam Cohen8dfcba42011-07-07 16:38:18 -0700179 mLauncher.getDragLayer().getLocationInDragLayer(v, loc);
Michael Jurka05713af2013-01-23 12:39:24 +0100180 int viewExtraPaddingLeft = extraPadding != null ? extraPadding.x : 0;
181 int viewExtraPaddingTop = extraPadding != null ? extraPadding.y : 0;
182 int dragLayerX = loc[0] + v.getPaddingLeft() + viewExtraPaddingLeft +
Winson Chung72d59842012-02-22 13:51:36 -0800183 (int) ((initialDragViewScale * bmp.getWidth() - bmp.getWidth()) / 2);
Michael Jurka05713af2013-01-23 12:39:24 +0100184 int dragLayerY = loc[1] + v.getPaddingTop() + viewExtraPaddingTop +
Winson Chung72d59842012-02-22 13:51:36 -0800185 (int) ((initialDragViewScale * bmp.getHeight() - bmp.getHeight()) / 2);
Winson Chunge3193b92010-09-10 11:44:42 -0700186
Michael Jurka05713af2013-01-23 12:39:24 +0100187 startDrag(bmp, dragLayerX, dragLayerY, source, dragInfo, dragAction, null,
188 null, initialDragViewScale);
Winson Chunge3193b92010-09-10 11:44:42 -0700189
190 if (dragAction == DRAG_ACTION_MOVE) {
191 v.setVisibility(View.GONE);
192 }
193 }
194
195 /**
196 * Starts a drag.
197 *
Joe Onorato5162ea92009-09-03 09:39:42 -0700198 * @param b The bitmap to display as the drag image. It will be re-scaled to the
199 * enlarged size.
Adam Cohen8dfcba42011-07-07 16:38:18 -0700200 * @param dragLayerX The x position in the DragLayer of the left-top of the bitmap.
201 * @param dragLayerY The y position in the DragLayer of the left-top of the bitmap.
Joe Onorato5162ea92009-09-03 09:39:42 -0700202 * @param source An object representing where the drag originated
Romain Guyea3763c2010-01-11 18:02:04 -0800203 * @param dragInfo The data associated with the object that is being dragged
Joe Onorato5162ea92009-09-03 09:39:42 -0700204 * @param dragAction The drag action: either {@link #DRAG_ACTION_MOVE} or
205 * {@link #DRAG_ACTION_COPY}
Michael Jurkaa63c4522010-08-19 13:52:27 -0700206 * @param dragRegion Coordinates within the bitmap b for the position of item being dragged.
207 * Makes dragging feel more precise, e.g. you can clip out a transparent border
208 */
Adam Cohen8dfcba42011-07-07 16:38:18 -0700209 public void startDrag(Bitmap b, int dragLayerX, int dragLayerY,
Winson Chung72d59842012-02-22 13:51:36 -0800210 DragSource source, Object dragInfo, int dragAction, Point dragOffset, Rect dragRegion,
211 float initialDragViewScale) {
Joe Onorato00acb122009-08-04 16:04:30 -0400212 if (PROFILE_DRAWING_DURING_DRAG) {
213 android.os.Debug.startMethodTracing("Launcher");
214 }
215
216 // Hide soft keyboard, if visible
217 if (mInputMethodManager == null) {
218 mInputMethodManager = (InputMethodManager)
Adam Cohen8dfcba42011-07-07 16:38:18 -0700219 mLauncher.getSystemService(Context.INPUT_METHOD_SERVICE);
Joe Onorato00acb122009-08-04 16:04:30 -0400220 }
221 mInputMethodManager.hideSoftInputFromWindow(mWindowToken, 0);
222
Patrick Dubroy4ed62782010-08-17 15:11:18 -0700223 for (DragListener listener : mListeners) {
224 listener.onDragStart(source, dragInfo, dragAction);
Joe Onorato00acb122009-08-04 16:04:30 -0400225 }
226
Adam Cohen8dfcba42011-07-07 16:38:18 -0700227 final int registrationX = mMotionDownX - dragLayerX;
228 final int registrationY = mMotionDownY - dragLayerY;
Joe Onorato00acb122009-08-04 16:04:30 -0400229
Michael Jurkaa63c4522010-08-19 13:52:27 -0700230 final int dragRegionLeft = dragRegion == null ? 0 : dragRegion.left;
231 final int dragRegionTop = dragRegion == null ? 0 : dragRegion.top;
Adam Cohene3e27a82011-04-15 12:07:39 -0700232
Joe Onorato00acb122009-08-04 16:04:30 -0400233 mDragging = true;
Adam Cohencb3382b2011-05-24 14:07:08 -0700234
Adam Cohen9932a9b2011-08-02 22:14:07 -0700235 mDragObject = new DropTarget.DragObject();
236
Adam Cohenbfbfd262011-06-13 16:55:12 -0700237 mDragObject.dragComplete = false;
Adam Cohen8dfcba42011-07-07 16:38:18 -0700238 mDragObject.xOffset = mMotionDownX - (dragLayerX + dragRegionLeft);
239 mDragObject.yOffset = mMotionDownY - (dragLayerY + dragRegionTop);
Adam Cohencb3382b2011-05-24 14:07:08 -0700240 mDragObject.dragSource = source;
241 mDragObject.dragInfo = dragInfo;
Joe Onorato00acb122009-08-04 16:04:30 -0400242
243 mVibrator.vibrate(VIBRATE_DURATION);
244
Adam Cohen8dfcba42011-07-07 16:38:18 -0700245 final DragView dragView = mDragObject.dragView = new DragView(mLauncher, b, registrationX,
Winson Chung72d59842012-02-22 13:51:36 -0800246 registrationY, 0, 0, b.getWidth(), b.getHeight(), initialDragViewScale);
Michael Jurkaa63c4522010-08-19 13:52:27 -0700247
Winson Chungb8c69f32011-10-19 21:36:08 -0700248 if (dragOffset != null) {
249 dragView.setDragVisualizeOffset(new Point(dragOffset));
250 }
Michael Jurkaa63c4522010-08-19 13:52:27 -0700251 if (dragRegion != null) {
Adam Cohene3e27a82011-04-15 12:07:39 -0700252 dragView.setDragRegion(new Rect(dragRegion));
Michael Jurkaa63c4522010-08-19 13:52:27 -0700253 }
254
Adam Cohen8dfcba42011-07-07 16:38:18 -0700255 dragView.show(mMotionDownX, mMotionDownY);
256 handleMoveEvent(mMotionDownX, mMotionDownY);
Joe Onorato00acb122009-08-04 16:04:30 -0400257 }
258
259 /**
260 * Draw the view into a bitmap.
261 */
Adam Cohen120980b2010-12-08 11:05:37 -0800262 Bitmap getViewBitmap(View v) {
Joe Onorato00acb122009-08-04 16:04:30 -0400263 v.clearFocus();
264 v.setPressed(false);
265
266 boolean willNotCache = v.willNotCacheDrawing();
267 v.setWillNotCacheDrawing(false);
268
269 // Reset the drawing cache background color to fully transparent
270 // for the duration of this operation
271 int color = v.getDrawingCacheBackgroundColor();
272 v.setDrawingCacheBackgroundColor(0);
Adam Cohen120980b2010-12-08 11:05:37 -0800273 float alpha = v.getAlpha();
274 v.setAlpha(1.0f);
Joe Onorato00acb122009-08-04 16:04:30 -0400275
276 if (color != 0) {
277 v.destroyDrawingCache();
278 }
279 v.buildDrawingCache();
280 Bitmap cacheBitmap = v.getDrawingCache();
Daniel Sandler3f8175a2010-05-25 11:48:32 -0400281 if (cacheBitmap == null) {
282 Log.e(TAG, "failed getViewBitmap(" + v + ")", new RuntimeException());
283 return null;
284 }
Joe Onorato00acb122009-08-04 16:04:30 -0400285
286 Bitmap bitmap = Bitmap.createBitmap(cacheBitmap);
287
288 // Restore the view
289 v.destroyDrawingCache();
Adam Cohen120980b2010-12-08 11:05:37 -0800290 v.setAlpha(alpha);
Joe Onorato00acb122009-08-04 16:04:30 -0400291 v.setWillNotCacheDrawing(willNotCache);
292 v.setDrawingCacheBackgroundColor(color);
293
294 return bitmap;
295 }
296
297 /**
298 * Call this from a drag source view like this:
299 *
300 * <pre>
301 * @Override
302 * public boolean dispatchKeyEvent(KeyEvent event) {
303 * return mDragController.dispatchKeyEvent(this, event)
304 * || super.dispatchKeyEvent(event);
305 * </pre>
306 */
307 public boolean dispatchKeyEvent(KeyEvent event) {
308 return mDragging;
309 }
310
Winson Chung304dcde2011-01-07 11:17:23 -0800311 public boolean isDragging() {
312 return mDragging;
313 }
314
Joe Onorato24b6fd82009-11-12 13:47:09 -0800315 /**
316 * Stop dragging without dropping.
317 */
318 public void cancelDrag() {
Winson Chung621e6402011-01-04 16:03:57 -0800319 if (mDragging) {
Winson Chungc07918d2011-07-01 15:35:26 -0700320 if (mLastDropTarget != null) {
321 mLastDropTarget.onDragExit(mDragObject);
322 }
Winson Chung41bb19d2012-03-05 18:36:46 -0800323 mDragObject.deferDragViewCleanupPostAnimation = false;
Adam Cohen36cc09b2011-09-29 17:33:15 -0700324 mDragObject.cancelled = true;
Adam Cohenbfbfd262011-06-13 16:55:12 -0700325 mDragObject.dragComplete = true;
Winson Chunga48487a2012-03-20 16:19:37 -0700326 mDragObject.dragSource.onDropCompleted(null, mDragObject, false, false);
Winson Chung621e6402011-01-04 16:03:57 -0800327 }
Joe Onorato24b6fd82009-11-12 13:47:09 -0800328 endDrag();
329 }
Michael Jurkaeadbfc52013-09-04 00:45:37 +0200330 public void onAppsRemoved(ArrayList<AppInfo> appInfos, Context context) {
Winson Chunga1820962011-10-03 16:31:06 -0700331 // Cancel the current drag if we are removing an app that we are dragging
332 if (mDragObject != null) {
333 Object rawDragInfo = mDragObject.dragInfo;
334 if (rawDragInfo instanceof ShortcutInfo) {
335 ShortcutInfo dragInfo = (ShortcutInfo) rawDragInfo;
Michael Jurkaeadbfc52013-09-04 00:45:37 +0200336 for (AppInfo info : appInfos) {
Michael Jurka7bcadad2012-04-02 07:23:44 -0700337 // Added null checks to prevent NPE we've seen in the wild
338 if (dragInfo != null &&
Winson Chungcd810732012-06-18 16:45:43 -0700339 dragInfo.intent != null) {
Winson Chung83892cc2013-05-01 16:53:33 -0700340 boolean isSameComponent =
341 dragInfo.intent.getComponent().equals(info.componentName);
342 if (isSameComponent) {
Winson Chung11a49372012-04-27 15:12:38 -0700343 cancelDrag();
344 return;
345 }
Winson Chunga1820962011-10-03 16:31:06 -0700346 }
347 }
348 }
349 }
350 }
Joe Onorato24b6fd82009-11-12 13:47:09 -0800351
Joe Onorato00acb122009-08-04 16:04:30 -0400352 private void endDrag() {
353 if (mDragging) {
354 mDragging = false;
Winson Chungaa15ffe2012-01-18 15:45:28 -0800355 clearScrollRunnable();
Winson Chung043f2af2012-03-01 16:09:54 -0800356 boolean isDeferred = false;
Adam Cohencb3382b2011-05-24 14:07:08 -0700357 if (mDragObject.dragView != null) {
Winson Chung043f2af2012-03-01 16:09:54 -0800358 isDeferred = mDragObject.deferDragViewCleanupPostAnimation;
359 if (!isDeferred) {
Winson Chung7bd1bbb2012-02-13 18:29:29 -0800360 mDragObject.dragView.remove();
361 }
Adam Cohencb3382b2011-05-24 14:07:08 -0700362 mDragObject.dragView = null;
Joe Onorato00acb122009-08-04 16:04:30 -0400363 }
Winson Chung043f2af2012-03-01 16:09:54 -0800364
365 // Only end the drag if we are not deferred
366 if (!isDeferred) {
367 for (DragListener listener : mListeners) {
368 listener.onDragEnd();
369 }
370 }
371 }
372
373 releaseVelocityTracker();
374 }
375
376 /**
377 * This only gets called as a result of drag view cleanup being deferred in endDrag();
378 */
379 void onDeferredEndDrag(DragView dragView) {
380 dragView.remove();
381
Michael Jurka1e2f4652013-07-08 18:03:46 -0700382 if (mDragObject.deferDragViewCleanupPostAnimation) {
383 // If we skipped calling onDragEnd() before, do it now
384 for (DragListener listener : mListeners) {
385 listener.onDragEnd();
386 }
Joe Onorato00acb122009-08-04 16:04:30 -0400387 }
388 }
389
Winson Chunga48487a2012-03-20 16:19:37 -0700390 void onDeferredEndFling(DropTarget.DragObject d) {
391 d.dragSource.onFlingToDeleteCompleted();
392 }
393
Joe Onorato00acb122009-08-04 16:04:30 -0400394 /**
Winson Chung273c1022011-07-11 13:40:52 -0700395 * Clamps the position to the drag layer bounds.
396 */
397 private int[] getClampedDragLayerPos(float x, float y) {
398 mLauncher.getDragLayer().getLocalVisibleRect(mDragLayerRect);
399 mTmpPoint[0] = (int) Math.max(mDragLayerRect.left, Math.min(x, mDragLayerRect.right - 1));
400 mTmpPoint[1] = (int) Math.max(mDragLayerRect.top, Math.min(y, mDragLayerRect.bottom - 1));
401 return mTmpPoint;
402 }
403
Winson Chunga2413752012-04-03 14:22:34 -0700404 long getLastGestureUpTime() {
405 if (mDragging) {
406 return System.currentTimeMillis();
407 } else {
408 return mLastTouchUpTime;
409 }
410 }
411
412 void resetLastGestureUpTime() {
413 mLastTouchUpTime = -1;
414 }
415
Winson Chung273c1022011-07-11 13:40:52 -0700416 /**
Joe Onorato00acb122009-08-04 16:04:30 -0400417 * Call this from a drag source view.
418 */
419 public boolean onInterceptTouchEvent(MotionEvent ev) {
Michael Jurka3a9fced2012-04-13 14:44:29 -0700420 @SuppressWarnings("all") // suppress dead code warning
421 final boolean debug = false;
422 if (debug) {
Joe Onoratoa30ce8e2009-11-11 08:16:49 -0800423 Log.d(Launcher.TAG, "DragController.onInterceptTouchEvent " + ev + " mDragging="
Joe Onorato9c1289c2009-08-17 11:03:03 -0400424 + mDragging);
425 }
Joe Onorato00acb122009-08-04 16:04:30 -0400426
Winson Chung043f2af2012-03-01 16:09:54 -0800427 // Update the velocity tracker
428 acquireVelocityTrackerAndAddMovement(ev);
429
430 final int action = ev.getAction();
Winson Chung273c1022011-07-11 13:40:52 -0700431 final int[] dragLayerPos = getClampedDragLayerPos(ev.getX(), ev.getY());
432 final int dragLayerX = dragLayerPos[0];
433 final int dragLayerY = dragLayerPos[1];
Joe Onorato00acb122009-08-04 16:04:30 -0400434
435 switch (action) {
436 case MotionEvent.ACTION_MOVE:
437 break;
Joe Onorato00acb122009-08-04 16:04:30 -0400438 case MotionEvent.ACTION_DOWN:
439 // Remember location of down touch
Adam Cohen8dfcba42011-07-07 16:38:18 -0700440 mMotionDownX = dragLayerX;
441 mMotionDownY = dragLayerY;
Joe Onorato00acb122009-08-04 16:04:30 -0400442 mLastDropTarget = null;
443 break;
Joe Onorato00acb122009-08-04 16:04:30 -0400444 case MotionEvent.ACTION_UP:
Winson Chunga2413752012-04-03 14:22:34 -0700445 mLastTouchUpTime = System.currentTimeMillis();
Joe Onorato00acb122009-08-04 16:04:30 -0400446 if (mDragging) {
Winson Chung043f2af2012-03-01 16:09:54 -0800447 PointF vec = isFlingingToDelete(mDragObject.dragSource);
Adam Cohen7c4c5102013-06-14 17:42:35 -0700448 if (!DeleteDropTarget.willAcceptDrop(mDragObject.dragInfo)) {
449 vec = null;
450 }
Winson Chung043f2af2012-03-01 16:09:54 -0800451 if (vec != null) {
452 dropOnFlingToDeleteTarget(dragLayerX, dragLayerY, vec);
453 } else {
454 drop(dragLayerX, dragLayerY);
455 }
Joe Onorato00acb122009-08-04 16:04:30 -0400456 }
457 endDrag();
458 break;
Winson Chung621e6402011-01-04 16:03:57 -0800459 case MotionEvent.ACTION_CANCEL:
460 cancelDrag();
461 break;
Joe Onorato00acb122009-08-04 16:04:30 -0400462 }
463
464 return mDragging;
465 }
466
467 /**
Romain Guyea3763c2010-01-11 18:02:04 -0800468 * Sets the view that should handle move events.
469 */
470 void setMoveTarget(View view) {
471 mMoveTarget = view;
472 }
473
474 public boolean dispatchUnhandledMove(View focused, int direction) {
475 return mMoveTarget != null && mMoveTarget.dispatchUnhandledMove(focused, direction);
476 }
477
Winson Chungaa15ffe2012-01-18 15:45:28 -0800478 private void clearScrollRunnable() {
479 mHandler.removeCallbacks(mScrollRunnable);
480 if (mScrollState == SCROLL_WAITING_IN_ZONE) {
481 mScrollState = SCROLL_OUTSIDE_ZONE;
482 mScrollRunnable.setDirection(SCROLL_RIGHT);
483 mDragScroller.onExitScrollArea();
Winson Chung360e63f2012-04-27 13:48:05 -0700484 mLauncher.getDragLayer().onExitScrollArea();
Winson Chungaa15ffe2012-01-18 15:45:28 -0800485 }
486 }
487
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700488 private void handleMoveEvent(int x, int y) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700489 mDragObject.dragView.move(x, y);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700490
491 // Drop on someone?
492 final int[] coordinates = mCoordinatesTemp;
493 DropTarget dropTarget = findDropTarget(x, y, coordinates);
Adam Cohencb3382b2011-05-24 14:07:08 -0700494 mDragObject.x = coordinates[0];
495 mDragObject.y = coordinates[1];
Winson Chung25460a12013-04-01 18:21:28 -0700496 checkTouchMove(dropTarget);
497
498 // Check if we are hovering over the scroll areas
499 mDistanceSinceScroll +=
500 Math.sqrt(Math.pow(mLastTouch[0] - x, 2) + Math.pow(mLastTouch[1] - y, 2));
501 mLastTouch[0] = x;
502 mLastTouch[1] = y;
503 checkScrollState(x, y);
504 }
505
506 public void forceTouchMove() {
507 int[] dummyCoordinates = mCoordinatesTemp;
508 DropTarget dropTarget = findDropTarget(mLastTouch[0], mLastTouch[1], dummyCoordinates);
509 checkTouchMove(dropTarget);
510 }
511
512 private void checkTouchMove(DropTarget dropTarget) {
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700513 if (dropTarget != null) {
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700514 if (mLastDropTarget != dropTarget) {
515 if (mLastDropTarget != null) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700516 mLastDropTarget.onDragExit(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700517 }
Adam Cohencb3382b2011-05-24 14:07:08 -0700518 dropTarget.onDragEnter(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700519 }
Adam Cohencb3382b2011-05-24 14:07:08 -0700520 dropTarget.onDragOver(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700521 } else {
522 if (mLastDropTarget != null) {
Adam Cohencb3382b2011-05-24 14:07:08 -0700523 mLastDropTarget.onDragExit(mDragObject);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700524 }
525 }
526 mLastDropTarget = dropTarget;
Winson Chung25460a12013-04-01 18:21:28 -0700527 }
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700528
Winson Chung25460a12013-04-01 18:21:28 -0700529 private void checkScrollState(int x, int y) {
Adam Cohen8dfcba42011-07-07 16:38:18 -0700530 final int slop = ViewConfiguration.get(mLauncher).getScaledWindowTouchSlop();
Winson Chungaa15ffe2012-01-18 15:45:28 -0800531 final int delay = mDistanceSinceScroll < slop ? RESCROLL_DELAY : SCROLL_DELAY;
Winson Chungfe1fe262013-04-01 16:52:31 -0700532 final DragLayer dragLayer = mLauncher.getDragLayer();
533 final boolean isRtl = (dragLayer.getLayoutDirection() == View.LAYOUT_DIRECTION_RTL);
534 final int forwardDirection = isRtl ? SCROLL_RIGHT : SCROLL_LEFT;
535 final int backwardsDirection = isRtl ? SCROLL_LEFT : SCROLL_RIGHT;
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700536
Winson Chung3f4e1422011-11-17 14:58:51 -0800537 if (x < mScrollZone) {
Winson Chungaa15ffe2012-01-18 15:45:28 -0800538 if (mScrollState == SCROLL_OUTSIDE_ZONE) {
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700539 mScrollState = SCROLL_WAITING_IN_ZONE;
Winson Chungfe1fe262013-04-01 16:52:31 -0700540 if (mDragScroller.onEnterScrollArea(x, y, forwardDirection)) {
541 dragLayer.onEnterScrollArea(forwardDirection);
542 mScrollRunnable.setDirection(forwardDirection);
Winson Chungaa15ffe2012-01-18 15:45:28 -0800543 mHandler.postDelayed(mScrollRunnable, delay);
Winson Chung3e0839e2011-10-03 15:15:18 -0700544 }
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700545 }
Winson Chung3f4e1422011-11-17 14:58:51 -0800546 } else if (x > mScrollView.getWidth() - mScrollZone) {
Winson Chungaa15ffe2012-01-18 15:45:28 -0800547 if (mScrollState == SCROLL_OUTSIDE_ZONE) {
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700548 mScrollState = SCROLL_WAITING_IN_ZONE;
Winson Chungfe1fe262013-04-01 16:52:31 -0700549 if (mDragScroller.onEnterScrollArea(x, y, backwardsDirection)) {
550 dragLayer.onEnterScrollArea(backwardsDirection);
551 mScrollRunnable.setDirection(backwardsDirection);
Winson Chungaa15ffe2012-01-18 15:45:28 -0800552 mHandler.postDelayed(mScrollRunnable, delay);
Winson Chung3e0839e2011-10-03 15:15:18 -0700553 }
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700554 }
555 } else {
Winson Chungaa15ffe2012-01-18 15:45:28 -0800556 clearScrollRunnable();
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700557 }
558 }
559
Romain Guyea3763c2010-01-11 18:02:04 -0800560 /**
Joe Onorato00acb122009-08-04 16:04:30 -0400561 * Call this from a drag source view.
562 */
563 public boolean onTouchEvent(MotionEvent ev) {
Joe Onorato00acb122009-08-04 16:04:30 -0400564 if (!mDragging) {
565 return false;
566 }
567
Winson Chung043f2af2012-03-01 16:09:54 -0800568 // Update the velocity tracker
569 acquireVelocityTrackerAndAddMovement(ev);
570
Joe Onorato00acb122009-08-04 16:04:30 -0400571 final int action = ev.getAction();
Winson Chung273c1022011-07-11 13:40:52 -0700572 final int[] dragLayerPos = getClampedDragLayerPos(ev.getX(), ev.getY());
573 final int dragLayerX = dragLayerPos[0];
574 final int dragLayerY = dragLayerPos[1];
Joe Onorato00acb122009-08-04 16:04:30 -0400575
576 switch (action) {
577 case MotionEvent.ACTION_DOWN:
Joe Onorato00acb122009-08-04 16:04:30 -0400578 // Remember where the motion event started
Adam Cohen8dfcba42011-07-07 16:38:18 -0700579 mMotionDownX = dragLayerX;
580 mMotionDownY = dragLayerY;
Joe Onorato00acb122009-08-04 16:04:30 -0400581
Adam Cohen8dfcba42011-07-07 16:38:18 -0700582 if ((dragLayerX < mScrollZone) || (dragLayerX > mScrollView.getWidth() - mScrollZone)) {
Joe Onorato00acb122009-08-04 16:04:30 -0400583 mScrollState = SCROLL_WAITING_IN_ZONE;
584 mHandler.postDelayed(mScrollRunnable, SCROLL_DELAY);
585 } else {
586 mScrollState = SCROLL_OUTSIDE_ZONE;
587 }
Mindy DelliCarpini53b8d072013-07-03 08:23:13 -0700588 handleMoveEvent(dragLayerX, dragLayerY);
Joe Onorato00acb122009-08-04 16:04:30 -0400589 break;
590 case MotionEvent.ACTION_MOVE:
Adam Cohen8dfcba42011-07-07 16:38:18 -0700591 handleMoveEvent(dragLayerX, dragLayerY);
Joe Onorato00acb122009-08-04 16:04:30 -0400592 break;
593 case MotionEvent.ACTION_UP:
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800594 // Ensure that we've processed a move event at the current pointer location.
Adam Cohen8dfcba42011-07-07 16:38:18 -0700595 handleMoveEvent(dragLayerX, dragLayerY);
Winson Chung3bc21c32012-01-20 13:59:18 -0800596 mHandler.removeCallbacks(mScrollRunnable);
Winson Chung043f2af2012-03-01 16:09:54 -0800597
Joe Onorato00acb122009-08-04 16:04:30 -0400598 if (mDragging) {
Winson Chung043f2af2012-03-01 16:09:54 -0800599 PointF vec = isFlingingToDelete(mDragObject.dragSource);
Adam Cohen7c4c5102013-06-14 17:42:35 -0700600 if (!DeleteDropTarget.willAcceptDrop(mDragObject.dragInfo)) {
Adam Cohen947dc542013-06-06 22:43:33 -0700601 vec = null;
602 }
Winson Chung043f2af2012-03-01 16:09:54 -0800603 if (vec != null) {
604 dropOnFlingToDeleteTarget(dragLayerX, dragLayerY, vec);
605 } else {
606 drop(dragLayerX, dragLayerY);
607 }
Joe Onorato00acb122009-08-04 16:04:30 -0400608 }
609 endDrag();
Joe Onorato00acb122009-08-04 16:04:30 -0400610 break;
611 case MotionEvent.ACTION_CANCEL:
Winson Chung3bc21c32012-01-20 13:59:18 -0800612 mHandler.removeCallbacks(mScrollRunnable);
Joe Onorato24b6fd82009-11-12 13:47:09 -0800613 cancelDrag();
Winson Chung621e6402011-01-04 16:03:57 -0800614 break;
Joe Onorato00acb122009-08-04 16:04:30 -0400615 }
616
617 return true;
618 }
619
Winson Chung043f2af2012-03-01 16:09:54 -0800620 /**
621 * Determines whether the user flung the current item to delete it.
622 *
623 * @return the vector at which the item was flung, or null if no fling was detected.
624 */
625 private PointF isFlingingToDelete(DragSource source) {
626 if (mFlingToDeleteDropTarget == null) return null;
627 if (!source.supportsFlingToDelete()) return null;
628
629 ViewConfiguration config = ViewConfiguration.get(mLauncher);
630 mVelocityTracker.computeCurrentVelocity(1000, config.getScaledMaximumFlingVelocity());
631
632 if (mVelocityTracker.getYVelocity() < mFlingToDeleteThresholdVelocity) {
633 // Do a quick dot product test to ensure that we are flinging upwards
634 PointF vel = new PointF(mVelocityTracker.getXVelocity(),
635 mVelocityTracker.getYVelocity());
636 PointF upVec = new PointF(0f, -1f);
637 float theta = (float) Math.acos(((vel.x * upVec.x) + (vel.y * upVec.y)) /
638 (vel.length() * upVec.length()));
639 if (theta <= Math.toRadians(MAX_FLING_DEGREES)) {
640 return vel;
641 }
642 }
643 return null;
644 }
645
646 private void dropOnFlingToDeleteTarget(float x, float y, PointF vel) {
647 final int[] coordinates = mCoordinatesTemp;
648
649 mDragObject.x = coordinates[0];
650 mDragObject.y = coordinates[1];
Winson Chung043f2af2012-03-01 16:09:54 -0800651
652 // Clean up dragging on the target if it's not the current fling delete target otherwise,
653 // start dragging to it.
654 if (mLastDropTarget != null && mFlingToDeleteDropTarget != mLastDropTarget) {
655 mLastDropTarget.onDragExit(mDragObject);
656 }
657
658 // Drop onto the fling-to-delete target
659 boolean accepted = false;
660 mFlingToDeleteDropTarget.onDragEnter(mDragObject);
Winson Chung232decb2012-03-28 15:09:05 -0700661 // We must set dragComplete to true _only_ after we "enter" the fling-to-delete target for
662 // "drop"
663 mDragObject.dragComplete = true;
Winson Chung043f2af2012-03-01 16:09:54 -0800664 mFlingToDeleteDropTarget.onDragExit(mDragObject);
665 if (mFlingToDeleteDropTarget.acceptDrop(mDragObject)) {
666 mFlingToDeleteDropTarget.onFlingToDelete(mDragObject, mDragObject.x, mDragObject.y,
667 vel);
668 accepted = true;
669 }
Winson Chunga48487a2012-03-20 16:19:37 -0700670 mDragObject.dragSource.onDropCompleted((View) mFlingToDeleteDropTarget, mDragObject, true,
Winson Chung043f2af2012-03-01 16:09:54 -0800671 accepted);
672 }
673
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800674 private void drop(float x, float y) {
Joe Onorato00acb122009-08-04 16:04:30 -0400675 final int[] coordinates = mCoordinatesTemp;
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800676 final DropTarget dropTarget = findDropTarget((int) x, (int) y, coordinates);
Joe Onorato00acb122009-08-04 16:04:30 -0400677
Adam Cohencb3382b2011-05-24 14:07:08 -0700678 mDragObject.x = coordinates[0];
679 mDragObject.y = coordinates[1];
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800680 boolean accepted = false;
Joe Onorato00acb122009-08-04 16:04:30 -0400681 if (dropTarget != null) {
Adam Cohenbfbfd262011-06-13 16:55:12 -0700682 mDragObject.dragComplete = true;
Adam Cohencb3382b2011-05-24 14:07:08 -0700683 dropTarget.onDragExit(mDragObject);
684 if (dropTarget.acceptDrop(mDragObject)) {
685 dropTarget.onDrop(mDragObject);
Patrick Dubroyb0a6bbe2011-03-02 18:40:21 -0800686 accepted = true;
Joe Onorato00acb122009-08-04 16:04:30 -0400687 }
688 }
Winson Chunga48487a2012-03-20 16:19:37 -0700689 mDragObject.dragSource.onDropCompleted((View) dropTarget, mDragObject, false, accepted);
Joe Onorato00acb122009-08-04 16:04:30 -0400690 }
691
692 private DropTarget findDropTarget(int x, int y, int[] dropCoordinates) {
693 final Rect r = mRectTemp;
694
695 final ArrayList<DropTarget> dropTargets = mDropTargets;
696 final int count = dropTargets.size();
697 for (int i=count-1; i>=0; i--) {
Patrick Dubroy440c3602010-07-13 17:50:32 -0700698 DropTarget target = dropTargets.get(i);
Michael Jurka0280c3b2010-09-17 15:00:07 -0700699 if (!target.isDropEnabled())
700 continue;
701
Adam Cohen7d30a372013-07-01 17:03:59 -0700702 target.getHitRectRelativeToDragLayer(r);
Patrick Dubroy440c3602010-07-13 17:50:32 -0700703
Adam Cohencb3382b2011-05-24 14:07:08 -0700704 mDragObject.x = x;
705 mDragObject.y = y;
Joe Onorato00acb122009-08-04 16:04:30 -0400706 if (r.contains(x, y)) {
Patrick Dubroy440c3602010-07-13 17:50:32 -0700707
Adam Cohen7d30a372013-07-01 17:03:59 -0700708 dropCoordinates[0] = x;
709 dropCoordinates[1] = y;
710 mLauncher.getDragLayer().mapCoordInSelfToDescendent((View) target, dropCoordinates);
Patrick Dubroy440c3602010-07-13 17:50:32 -0700711
Joe Onorato00acb122009-08-04 16:04:30 -0400712 return target;
713 }
714 }
715 return null;
716 }
717
718 public void setDragScoller(DragScroller scroller) {
719 mDragScroller = scroller;
720 }
721
722 public void setWindowToken(IBinder token) {
723 mWindowToken = token;
724 }
725
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800726 /**
727 * Sets the drag listner which will be notified when a drag starts or ends.
728 */
Patrick Dubroy4ed62782010-08-17 15:11:18 -0700729 public void addDragListener(DragListener l) {
730 mListeners.add(l);
Joe Onorato00acb122009-08-04 16:04:30 -0400731 }
732
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800733 /**
734 * Remove a previously installed drag listener.
735 */
Joe Onorato00acb122009-08-04 16:04:30 -0400736 public void removeDragListener(DragListener l) {
Patrick Dubroy4ed62782010-08-17 15:11:18 -0700737 mListeners.remove(l);
Joe Onorato00acb122009-08-04 16:04:30 -0400738 }
739
740 /**
741 * Add a DropTarget to the list of potential places to receive drop events.
742 */
743 public void addDropTarget(DropTarget target) {
744 mDropTargets.add(target);
745 }
746
747 /**
748 * Don't send drop events to <em>target</em> any more.
749 */
750 public void removeDropTarget(DropTarget target) {
751 mDropTargets.remove(target);
752 }
753
754 /**
Winson Chung043f2af2012-03-01 16:09:54 -0800755 * Sets the current fling-to-delete drop target.
756 */
757 public void setFlingToDeleteDropTarget(DropTarget target) {
758 mFlingToDeleteDropTarget = target;
759 }
760
761 private void acquireVelocityTrackerAndAddMovement(MotionEvent ev) {
762 if (mVelocityTracker == null) {
763 mVelocityTracker = VelocityTracker.obtain();
764 }
765 mVelocityTracker.addMovement(ev);
766 }
767
768 private void releaseVelocityTracker() {
769 if (mVelocityTracker != null) {
770 mVelocityTracker.recycle();
771 mVelocityTracker = null;
772 }
773 }
774
775 /**
Joe Onorato00acb122009-08-04 16:04:30 -0400776 * Set which view scrolls for touch events near the edge of the screen.
777 */
778 public void setScrollView(View v) {
779 mScrollView = v;
780 }
781
Patrick Dubroy5f445422011-02-18 14:35:21 -0800782 DragView getDragView() {
Adam Cohencb3382b2011-05-24 14:07:08 -0700783 return mDragObject.dragView;
Patrick Dubroy5f445422011-02-18 14:35:21 -0800784 }
785
Joe Onorato00acb122009-08-04 16:04:30 -0400786 private class ScrollRunnable implements Runnable {
787 private int mDirection;
788
789 ScrollRunnable() {
790 }
791
792 public void run() {
793 if (mDragScroller != null) {
794 if (mDirection == SCROLL_LEFT) {
795 mDragScroller.scrollLeft();
796 } else {
797 mDragScroller.scrollRight();
798 }
799 mScrollState = SCROLL_OUTSIDE_ZONE;
Patrick Dubroya16fd5a2010-10-07 16:47:28 -0700800 mDistanceSinceScroll = 0;
801 mDragScroller.onExitScrollArea();
Winson Chung360e63f2012-04-27 13:48:05 -0700802 mLauncher.getDragLayer().onExitScrollArea();
Winson Chungaa15ffe2012-01-18 15:45:28 -0800803
804 if (isDragging()) {
Winson Chung25460a12013-04-01 18:21:28 -0700805 // Check the scroll again so that we can requeue the scroller if necessary
806 checkScrollState(mLastTouch[0], mLastTouch[1]);
Winson Chungaa15ffe2012-01-18 15:45:28 -0800807 }
Joe Onorato00acb122009-08-04 16:04:30 -0400808 }
809 }
810
811 void setDirection(int direction) {
812 mDirection = direction;
813 }
814 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800815}