1 /* 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 17 package com.android.launcher2; 18 19 import android.animation.Animator; 20 import android.animation.AnimatorListenerAdapter; 21 import android.animation.ObjectAnimator; 22 import android.animation.PropertyValuesHolder; 23 import android.content.Context; 24 import android.content.res.Resources; 25 import android.graphics.PointF; 26 import android.graphics.Rect; 27 import android.graphics.drawable.Drawable; 28 import android.text.InputType; 29 import android.text.Selection; 30 import android.text.Spannable; 31 import android.util.AttributeSet; 32 import android.util.Log; 33 import android.view.ActionMode; 34 import android.view.KeyEvent; 35 import android.view.LayoutInflater; 36 import android.view.Menu; 37 import android.view.MenuItem; 38 import android.view.MotionEvent; 39 import android.view.View; 40 import android.view.accessibility.AccessibilityEvent; 41 import android.view.accessibility.AccessibilityManager; 42 import android.view.inputmethod.EditorInfo; 43 import android.view.inputmethod.InputMethodManager; 44 import android.widget.LinearLayout; 45 import android.widget.TextView; 46 47 import com.android.launcher.R; 48 import com.android.launcher2.FolderInfo.FolderListener; 49 50 import java.util.ArrayList; 51 import java.util.Collections; 52 import java.util.Comparator; 53 54 /** 55 * Represents a set of icons chosen by the user or generated by the system. 56 */ 57 public class Folder extends LinearLayout implements DragSource, View.OnClickListener, 58 View.OnLongClickListener, DropTarget, FolderListener, TextView.OnEditorActionListener, 59 View.OnFocusChangeListener { 60 61 @SuppressWarnings("unused") 62 private static final String TAG = "Launcher.Folder"; 63 64 protected DragController mDragController; 65 protected Launcher mLauncher; 66 protected FolderInfo mInfo; 67 68 static final int STATE_NONE = -1; 69 static final int STATE_SMALL = 0; 70 static final int STATE_ANIMATING = 1; 71 static final int STATE_OPEN = 2; 72 73 private int mExpandDuration; 74 protected CellLayout mContent; 75 private final LayoutInflater mInflater; 76 private final IconCache mIconCache; 77 private int mState = STATE_NONE; 78 private static final int REORDER_ANIMATION_DURATION = 230; 79 private static final int ON_EXIT_CLOSE_DELAY = 800; 80 private boolean mRearrangeOnClose = false; 81 private FolderIcon mFolderIcon; 82 private int mMaxCountX; 83 private int mMaxCountY; 84 private int mMaxNumItems; 85 private ArrayList<View> mItemsInReadingOrder = new ArrayList<View>(); 86 private Drawable mIconDrawable; 87 boolean mItemsInvalidated = false; 88 private ShortcutInfo mCurrentDragInfo; 89 private View mCurrentDragView; 90 boolean mSuppressOnAdd = false; 91 private int[] mTargetCell = new int[2]; 92 private int[] mPreviousTargetCell = new int[2]; 93 private int[] mEmptyCell = new int[2]; 94 private Alarm mReorderAlarm = new Alarm(); 95 private Alarm mOnExitAlarm = new Alarm(); 96 private int mFolderNameHeight; 97 private Rect mTempRect = new Rect(); 98 private boolean mDragInProgress = false; 99 private boolean mDeleteFolderOnDropCompleted = false; 100 private boolean mSuppressFolderDeletion = false; 101 private boolean mItemAddedBackToSelfViaIcon = false; 102 FolderEditText mFolderName; 103 private float mFolderIconPivotX; 104 private float mFolderIconPivotY; 105 106 private boolean mIsEditingName = false; 107 private InputMethodManager mInputMethodManager; 108 109 private static String sDefaultFolderName; 110 private static String sHintText; 111 private ObjectAnimator mOpenCloseAnimator; 112 113 /** 114 * Used to inflate the Workspace from XML. 115 * 116 * @param context The application's context. 117 * @param attrs The attribtues set containing the Workspace's customization values. 118 */ Folder(Context context, AttributeSet attrs)119 public Folder(Context context, AttributeSet attrs) { 120 super(context, attrs); 121 setAlwaysDrawnWithCacheEnabled(false); 122 mInflater = LayoutInflater.from(context); 123 mIconCache = ((LauncherApplication)context.getApplicationContext()).getIconCache(); 124 125 Resources res = getResources(); 126 mMaxCountX = res.getInteger(R.integer.folder_max_count_x); 127 mMaxCountY = res.getInteger(R.integer.folder_max_count_y); 128 mMaxNumItems = res.getInteger(R.integer.folder_max_num_items); 129 if (mMaxCountX < 0 || mMaxCountY < 0 || mMaxNumItems < 0) { 130 mMaxCountX = LauncherModel.getCellCountX(); 131 mMaxCountY = LauncherModel.getCellCountY(); 132 mMaxNumItems = mMaxCountX * mMaxCountY; 133 } 134 135 mInputMethodManager = (InputMethodManager) 136 getContext().getSystemService(Context.INPUT_METHOD_SERVICE); 137 138 mExpandDuration = res.getInteger(R.integer.config_folderAnimDuration); 139 140 if (sDefaultFolderName == null) { 141 sDefaultFolderName = res.getString(R.string.folder_name); 142 } 143 if (sHintText == null) { 144 sHintText = res.getString(R.string.folder_hint_text); 145 } 146 mLauncher = (Launcher) context; 147 // We need this view to be focusable in touch mode so that when text editing of the folder 148 // name is complete, we have something to focus on, thus hiding the cursor and giving 149 // reliable behvior when clicking the text field (since it will always gain focus on click). 150 setFocusableInTouchMode(true); 151 } 152 153 @Override onFinishInflate()154 protected void onFinishInflate() { 155 super.onFinishInflate(); 156 mContent = (CellLayout) findViewById(R.id.folder_content); 157 mContent.setGridSize(0, 0); 158 mContent.getShortcutsAndWidgets().setMotionEventSplittingEnabled(false); 159 mFolderName = (FolderEditText) findViewById(R.id.folder_name); 160 mFolderName.setFolder(this); 161 mFolderName.setOnFocusChangeListener(this); 162 163 // We find out how tall the text view wants to be (it is set to wrap_content), so that 164 // we can allocate the appropriate amount of space for it. 165 int measureSpec = MeasureSpec.UNSPECIFIED; 166 mFolderName.measure(measureSpec, measureSpec); 167 mFolderNameHeight = mFolderName.getMeasuredHeight(); 168 169 // We disable action mode for now since it messes up the view on phones 170 mFolderName.setCustomSelectionActionModeCallback(mActionModeCallback); 171 mFolderName.setOnEditorActionListener(this); 172 mFolderName.setSelectAllOnFocus(true); 173 mFolderName.setInputType(mFolderName.getInputType() | 174 InputType.TYPE_TEXT_FLAG_NO_SUGGESTIONS | InputType.TYPE_TEXT_FLAG_CAP_WORDS); 175 } 176 177 private ActionMode.Callback mActionModeCallback = new ActionMode.Callback() { 178 public boolean onActionItemClicked(ActionMode mode, MenuItem item) { 179 return false; 180 } 181 182 public boolean onCreateActionMode(ActionMode mode, Menu menu) { 183 return false; 184 } 185 186 public void onDestroyActionMode(ActionMode mode) { 187 } 188 189 public boolean onPrepareActionMode(ActionMode mode, Menu menu) { 190 return false; 191 } 192 }; 193 onClick(View v)194 public void onClick(View v) { 195 Object tag = v.getTag(); 196 if (tag instanceof ShortcutInfo) { 197 // refactor this code from Folder 198 ShortcutInfo item = (ShortcutInfo) tag; 199 int[] pos = new int[2]; 200 v.getLocationOnScreen(pos); 201 item.intent.setSourceBounds(new Rect(pos[0], pos[1], 202 pos[0] + v.getWidth(), pos[1] + v.getHeight())); 203 204 mLauncher.startActivitySafely(v, item.intent, item); 205 } 206 } 207 onLongClick(View v)208 public boolean onLongClick(View v) { 209 // Return if global dragging is not enabled 210 if (!mLauncher.isDraggingEnabled()) return true; 211 212 Object tag = v.getTag(); 213 if (tag instanceof ShortcutInfo) { 214 ShortcutInfo item = (ShortcutInfo) tag; 215 if (!v.isInTouchMode()) { 216 return false; 217 } 218 219 mLauncher.dismissFolderCling(null); 220 221 mLauncher.getWorkspace().onDragStartedWithItem(v); 222 mLauncher.getWorkspace().beginDragShared(v, this); 223 mIconDrawable = ((TextView) v).getCompoundDrawables()[1]; 224 225 mCurrentDragInfo = item; 226 mEmptyCell[0] = item.cellX; 227 mEmptyCell[1] = item.cellY; 228 mCurrentDragView = v; 229 230 mContent.removeView(mCurrentDragView); 231 mInfo.remove(mCurrentDragInfo); 232 mDragInProgress = true; 233 mItemAddedBackToSelfViaIcon = false; 234 } 235 return true; 236 } 237 isEditingName()238 public boolean isEditingName() { 239 return mIsEditingName; 240 } 241 startEditingFolderName()242 public void startEditingFolderName() { 243 mFolderName.setHint(""); 244 mIsEditingName = true; 245 } 246 dismissEditingName()247 public void dismissEditingName() { 248 mInputMethodManager.hideSoftInputFromWindow(getWindowToken(), 0); 249 doneEditingFolderName(true); 250 } 251 doneEditingFolderName(boolean commit)252 public void doneEditingFolderName(boolean commit) { 253 mFolderName.setHint(sHintText); 254 // Convert to a string here to ensure that no other state associated with the text field 255 // gets saved. 256 String newTitle = mFolderName.getText().toString(); 257 mInfo.setTitle(newTitle); 258 LauncherModel.updateItemInDatabase(mLauncher, mInfo); 259 260 if (commit) { 261 sendCustomAccessibilityEvent(AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED, 262 String.format(getContext().getString(R.string.folder_renamed), newTitle)); 263 } 264 // In order to clear the focus from the text field, we set the focus on ourself. This 265 // ensures that every time the field is clicked, focus is gained, giving reliable behavior. 266 requestFocus(); 267 268 Selection.setSelection((Spannable) mFolderName.getText(), 0, 0); 269 mIsEditingName = false; 270 } 271 onEditorAction(TextView v, int actionId, KeyEvent event)272 public boolean onEditorAction(TextView v, int actionId, KeyEvent event) { 273 if (actionId == EditorInfo.IME_ACTION_DONE) { 274 dismissEditingName(); 275 return true; 276 } 277 return false; 278 } 279 getEditTextRegion()280 public View getEditTextRegion() { 281 return mFolderName; 282 } 283 getDragDrawable()284 public Drawable getDragDrawable() { 285 return mIconDrawable; 286 } 287 288 /** 289 * We need to handle touch events to prevent them from falling through to the workspace below. 290 */ 291 @Override onTouchEvent(MotionEvent ev)292 public boolean onTouchEvent(MotionEvent ev) { 293 return true; 294 } 295 setDragController(DragController dragController)296 public void setDragController(DragController dragController) { 297 mDragController = dragController; 298 } 299 setFolderIcon(FolderIcon icon)300 void setFolderIcon(FolderIcon icon) { 301 mFolderIcon = icon; 302 } 303 304 @Override dispatchPopulateAccessibilityEvent(AccessibilityEvent event)305 public boolean dispatchPopulateAccessibilityEvent(AccessibilityEvent event) { 306 // When the folder gets focus, we don't want to announce the list of items. 307 return true; 308 } 309 310 /** 311 * @return the FolderInfo object associated with this folder 312 */ getInfo()313 FolderInfo getInfo() { 314 return mInfo; 315 } 316 317 private class GridComparator implements Comparator<ShortcutInfo> { 318 int mNumCols; GridComparator(int numCols)319 public GridComparator(int numCols) { 320 mNumCols = numCols; 321 } 322 323 @Override compare(ShortcutInfo lhs, ShortcutInfo rhs)324 public int compare(ShortcutInfo lhs, ShortcutInfo rhs) { 325 int lhIndex = lhs.cellY * mNumCols + lhs.cellX; 326 int rhIndex = rhs.cellY * mNumCols + rhs.cellX; 327 return (lhIndex - rhIndex); 328 } 329 } 330 placeInReadingOrder(ArrayList<ShortcutInfo> items)331 private void placeInReadingOrder(ArrayList<ShortcutInfo> items) { 332 int maxX = 0; 333 int count = items.size(); 334 for (int i = 0; i < count; i++) { 335 ShortcutInfo item = items.get(i); 336 if (item.cellX > maxX) { 337 maxX = item.cellX; 338 } 339 } 340 341 GridComparator gridComparator = new GridComparator(maxX + 1); 342 Collections.sort(items, gridComparator); 343 final int countX = mContent.getCountX(); 344 for (int i = 0; i < count; i++) { 345 int x = i % countX; 346 int y = i / countX; 347 ShortcutInfo item = items.get(i); 348 item.cellX = x; 349 item.cellY = y; 350 } 351 } 352 bind(FolderInfo info)353 void bind(FolderInfo info) { 354 mInfo = info; 355 ArrayList<ShortcutInfo> children = info.contents; 356 ArrayList<ShortcutInfo> overflow = new ArrayList<ShortcutInfo>(); 357 setupContentForNumItems(children.size()); 358 placeInReadingOrder(children); 359 int count = 0; 360 for (int i = 0; i < children.size(); i++) { 361 ShortcutInfo child = (ShortcutInfo) children.get(i); 362 if (!createAndAddShortcut(child)) { 363 overflow.add(child); 364 } else { 365 count++; 366 } 367 } 368 369 // We rearrange the items in case there are any empty gaps 370 setupContentForNumItems(count); 371 372 // If our folder has too many items we prune them from the list. This is an issue 373 // when upgrading from the old Folders implementation which could contain an unlimited 374 // number of items. 375 for (ShortcutInfo item: overflow) { 376 mInfo.remove(item); 377 LauncherModel.deleteItemFromDatabase(mLauncher, item); 378 } 379 380 mItemsInvalidated = true; 381 updateTextViewFocus(); 382 mInfo.addListener(this); 383 384 if (!sDefaultFolderName.contentEquals(mInfo.title)) { 385 mFolderName.setText(mInfo.title); 386 } else { 387 mFolderName.setText(""); 388 } 389 updateItemLocationsInDatabase(); 390 } 391 392 /** 393 * Creates a new UserFolder, inflated from R.layout.user_folder. 394 * 395 * @param context The application's context. 396 * 397 * @return A new UserFolder. 398 */ fromXml(Context context)399 static Folder fromXml(Context context) { 400 return (Folder) LayoutInflater.from(context).inflate(R.layout.user_folder, null); 401 } 402 403 /** 404 * This method is intended to make the UserFolder to be visually identical in size and position 405 * to its associated FolderIcon. This allows for a seamless transition into the expanded state. 406 */ positionAndSizeAsIcon()407 private void positionAndSizeAsIcon() { 408 if (!(getParent() instanceof DragLayer)) return; 409 setScaleX(0.8f); 410 setScaleY(0.8f); 411 setAlpha(0f); 412 mState = STATE_SMALL; 413 } 414 animateOpen()415 public void animateOpen() { 416 positionAndSizeAsIcon(); 417 418 if (!(getParent() instanceof DragLayer)) return; 419 centerAboutIcon(); 420 PropertyValuesHolder alpha = PropertyValuesHolder.ofFloat("alpha", 1); 421 PropertyValuesHolder scaleX = PropertyValuesHolder.ofFloat("scaleX", 1.0f); 422 PropertyValuesHolder scaleY = PropertyValuesHolder.ofFloat("scaleY", 1.0f); 423 final ObjectAnimator oa = mOpenCloseAnimator = 424 ObjectAnimator.ofPropertyValuesHolder(this, alpha, scaleX, scaleY); 425 426 oa.addListener(new AnimatorListenerAdapter() { 427 @Override 428 public void onAnimationStart(Animator animation) { 429 sendCustomAccessibilityEvent(AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED, 430 String.format(getContext().getString(R.string.folder_opened), 431 mContent.getCountX(), mContent.getCountY())); 432 mState = STATE_ANIMATING; 433 } 434 @Override 435 public void onAnimationEnd(Animator animation) { 436 mState = STATE_OPEN; 437 setLayerType(LAYER_TYPE_NONE, null); 438 Cling cling = mLauncher.showFirstRunFoldersCling(); 439 if (cling != null) { 440 cling.bringToFront(); 441 } 442 setFocusOnFirstChild(); 443 } 444 }); 445 oa.setDuration(mExpandDuration); 446 setLayerType(LAYER_TYPE_HARDWARE, null); 447 buildLayer(); 448 post(new Runnable() { 449 public void run() { 450 // Check if the animator changed in the meantime 451 if (oa != mOpenCloseAnimator) 452 return; 453 oa.start(); 454 } 455 }); 456 } 457 sendCustomAccessibilityEvent(int type, String text)458 private void sendCustomAccessibilityEvent(int type, String text) { 459 AccessibilityManager accessibilityManager = (AccessibilityManager) 460 getContext().getSystemService(Context.ACCESSIBILITY_SERVICE); 461 if (accessibilityManager.isEnabled()) { 462 AccessibilityEvent event = AccessibilityEvent.obtain(type); 463 onInitializeAccessibilityEvent(event); 464 event.getText().add(text); 465 accessibilityManager.sendAccessibilityEvent(event); 466 } 467 } 468 setFocusOnFirstChild()469 private void setFocusOnFirstChild() { 470 View firstChild = mContent.getChildAt(0, 0); 471 if (firstChild != null) { 472 firstChild.requestFocus(); 473 } 474 } 475 animateClosed()476 public void animateClosed() { 477 if (!(getParent() instanceof DragLayer)) return; 478 PropertyValuesHolder alpha = PropertyValuesHolder.ofFloat("alpha", 0); 479 PropertyValuesHolder scaleX = PropertyValuesHolder.ofFloat("scaleX", 0.9f); 480 PropertyValuesHolder scaleY = PropertyValuesHolder.ofFloat("scaleY", 0.9f); 481 final ObjectAnimator oa = mOpenCloseAnimator = 482 ObjectAnimator.ofPropertyValuesHolder(this, alpha, scaleX, scaleY); 483 484 oa.addListener(new AnimatorListenerAdapter() { 485 @Override 486 public void onAnimationEnd(Animator animation) { 487 onCloseComplete(); 488 setLayerType(LAYER_TYPE_NONE, null); 489 mState = STATE_SMALL; 490 } 491 @Override 492 public void onAnimationStart(Animator animation) { 493 sendCustomAccessibilityEvent(AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED, 494 getContext().getString(R.string.folder_closed)); 495 mState = STATE_ANIMATING; 496 } 497 }); 498 oa.setDuration(mExpandDuration); 499 setLayerType(LAYER_TYPE_HARDWARE, null); 500 buildLayer(); 501 post(new Runnable() { 502 public void run() { 503 // Check if the animator changed in the meantime 504 if (oa != mOpenCloseAnimator) 505 return; 506 oa.start(); 507 } 508 }); 509 } 510 notifyDataSetChanged()511 void notifyDataSetChanged() { 512 // recreate all the children if the data set changes under us. We may want to do this more 513 // intelligently (ie just removing the views that should no longer exist) 514 mContent.removeAllViewsInLayout(); 515 bind(mInfo); 516 } 517 acceptDrop(DragObject d)518 public boolean acceptDrop(DragObject d) { 519 final ItemInfo item = (ItemInfo) d.dragInfo; 520 final int itemType = item.itemType; 521 return ((itemType == LauncherSettings.Favorites.ITEM_TYPE_APPLICATION || 522 itemType == LauncherSettings.Favorites.ITEM_TYPE_SHORTCUT) && 523 !isFull()); 524 } 525 findAndSetEmptyCells(ShortcutInfo item)526 protected boolean findAndSetEmptyCells(ShortcutInfo item) { 527 int[] emptyCell = new int[2]; 528 if (mContent.findCellForSpan(emptyCell, item.spanX, item.spanY)) { 529 item.cellX = emptyCell[0]; 530 item.cellY = emptyCell[1]; 531 return true; 532 } else { 533 return false; 534 } 535 } 536 createAndAddShortcut(ShortcutInfo item)537 protected boolean createAndAddShortcut(ShortcutInfo item) { 538 final TextView textView = 539 (TextView) mInflater.inflate(R.layout.application, this, false); 540 textView.setCompoundDrawablesWithIntrinsicBounds(null, 541 new FastBitmapDrawable(item.getIcon(mIconCache)), null, null); 542 textView.setText(item.title); 543 textView.setTag(item); 544 545 textView.setOnClickListener(this); 546 textView.setOnLongClickListener(this); 547 548 // We need to check here to verify that the given item's location isn't already occupied 549 // by another item. 550 if (mContent.getChildAt(item.cellX, item.cellY) != null || item.cellX < 0 || item.cellY < 0 551 || item.cellX >= mContent.getCountX() || item.cellY >= mContent.getCountY()) { 552 // This shouldn't happen, log it. 553 Log.e(TAG, "Folder order not properly persisted during bind"); 554 if (!findAndSetEmptyCells(item)) { 555 return false; 556 } 557 } 558 559 CellLayout.LayoutParams lp = 560 new CellLayout.LayoutParams(item.cellX, item.cellY, item.spanX, item.spanY); 561 boolean insert = false; 562 textView.setOnKeyListener(new FolderKeyEventListener()); 563 mContent.addViewToCellLayout(textView, insert ? 0 : -1, (int)item.id, lp, true); 564 return true; 565 } 566 onDragEnter(DragObject d)567 public void onDragEnter(DragObject d) { 568 mPreviousTargetCell[0] = -1; 569 mPreviousTargetCell[1] = -1; 570 mOnExitAlarm.cancelAlarm(); 571 } 572 573 OnAlarmListener mReorderAlarmListener = new OnAlarmListener() { 574 public void onAlarm(Alarm alarm) { 575 realTimeReorder(mEmptyCell, mTargetCell); 576 } 577 }; 578 readingOrderGreaterThan(int[] v1, int[] v2)579 boolean readingOrderGreaterThan(int[] v1, int[] v2) { 580 if (v1[1] > v2[1] || (v1[1] == v2[1] && v1[0] > v2[0])) { 581 return true; 582 } else { 583 return false; 584 } 585 } 586 realTimeReorder(int[] empty, int[] target)587 private void realTimeReorder(int[] empty, int[] target) { 588 boolean wrap; 589 int startX; 590 int endX; 591 int startY; 592 int delay = 0; 593 float delayAmount = 30; 594 if (readingOrderGreaterThan(target, empty)) { 595 wrap = empty[0] >= mContent.getCountX() - 1; 596 startY = wrap ? empty[1] + 1 : empty[1]; 597 for (int y = startY; y <= target[1]; y++) { 598 startX = y == empty[1] ? empty[0] + 1 : 0; 599 endX = y < target[1] ? mContent.getCountX() - 1 : target[0]; 600 for (int x = startX; x <= endX; x++) { 601 View v = mContent.getChildAt(x,y); 602 if (mContent.animateChildToPosition(v, empty[0], empty[1], 603 REORDER_ANIMATION_DURATION, delay, true, true)) { 604 empty[0] = x; 605 empty[1] = y; 606 delay += delayAmount; 607 delayAmount *= 0.9; 608 } 609 } 610 } 611 } else { 612 wrap = empty[0] == 0; 613 startY = wrap ? empty[1] - 1 : empty[1]; 614 for (int y = startY; y >= target[1]; y--) { 615 startX = y == empty[1] ? empty[0] - 1 : mContent.getCountX() - 1; 616 endX = y > target[1] ? 0 : target[0]; 617 for (int x = startX; x >= endX; x--) { 618 View v = mContent.getChildAt(x,y); 619 if (mContent.animateChildToPosition(v, empty[0], empty[1], 620 REORDER_ANIMATION_DURATION, delay, true, true)) { 621 empty[0] = x; 622 empty[1] = y; 623 delay += delayAmount; 624 delayAmount *= 0.9; 625 } 626 } 627 } 628 } 629 } 630 onDragOver(DragObject d)631 public void onDragOver(DragObject d) { 632 float[] r = getDragViewVisualCenter(d.x, d.y, d.xOffset, d.yOffset, d.dragView, null); 633 mTargetCell = mContent.findNearestArea((int) r[0], (int) r[1], 1, 1, mTargetCell); 634 635 if (mTargetCell[0] != mPreviousTargetCell[0] || mTargetCell[1] != mPreviousTargetCell[1]) { 636 mReorderAlarm.cancelAlarm(); 637 mReorderAlarm.setOnAlarmListener(mReorderAlarmListener); 638 mReorderAlarm.setAlarm(150); 639 mPreviousTargetCell[0] = mTargetCell[0]; 640 mPreviousTargetCell[1] = mTargetCell[1]; 641 } 642 } 643 644 // This is used to compute the visual center of the dragView. The idea is that 645 // the visual center represents the user's interpretation of where the item is, and hence 646 // is the appropriate point to use when determining drop location. getDragViewVisualCenter(int x, int y, int xOffset, int yOffset, DragView dragView, float[] recycle)647 private float[] getDragViewVisualCenter(int x, int y, int xOffset, int yOffset, 648 DragView dragView, float[] recycle) { 649 float res[]; 650 if (recycle == null) { 651 res = new float[2]; 652 } else { 653 res = recycle; 654 } 655 656 // These represent the visual top and left of drag view if a dragRect was provided. 657 // If a dragRect was not provided, then they correspond to the actual view left and 658 // top, as the dragRect is in that case taken to be the entire dragView. 659 // R.dimen.dragViewOffsetY. 660 int left = x - xOffset; 661 int top = y - yOffset; 662 663 // In order to find the visual center, we shift by half the dragRect 664 res[0] = left + dragView.getDragRegion().width() / 2; 665 res[1] = top + dragView.getDragRegion().height() / 2; 666 667 return res; 668 } 669 670 OnAlarmListener mOnExitAlarmListener = new OnAlarmListener() { 671 public void onAlarm(Alarm alarm) { 672 completeDragExit(); 673 } 674 }; 675 completeDragExit()676 public void completeDragExit() { 677 mLauncher.closeFolder(); 678 mCurrentDragInfo = null; 679 mCurrentDragView = null; 680 mSuppressOnAdd = false; 681 mRearrangeOnClose = true; 682 } 683 onDragExit(DragObject d)684 public void onDragExit(DragObject d) { 685 // We only close the folder if this is a true drag exit, ie. not because a drop 686 // has occurred above the folder. 687 if (!d.dragComplete) { 688 mOnExitAlarm.setOnAlarmListener(mOnExitAlarmListener); 689 mOnExitAlarm.setAlarm(ON_EXIT_CLOSE_DELAY); 690 } 691 mReorderAlarm.cancelAlarm(); 692 } 693 onDropCompleted(View target, DragObject d, boolean isFlingToDelete, boolean success)694 public void onDropCompleted(View target, DragObject d, boolean isFlingToDelete, 695 boolean success) { 696 if (success) { 697 if (mDeleteFolderOnDropCompleted && !mItemAddedBackToSelfViaIcon) { 698 replaceFolderWithFinalItem(); 699 } 700 } else { 701 // The drag failed, we need to return the item to the folder 702 mFolderIcon.onDrop(d); 703 704 // We're going to trigger a "closeFolder" which may occur before this item has 705 // been added back to the folder -- this could cause the folder to be deleted 706 if (mOnExitAlarm.alarmPending()) { 707 mSuppressFolderDeletion = true; 708 } 709 } 710 711 if (target != this) { 712 if (mOnExitAlarm.alarmPending()) { 713 mOnExitAlarm.cancelAlarm(); 714 completeDragExit(); 715 } 716 } 717 mDeleteFolderOnDropCompleted = false; 718 mDragInProgress = false; 719 mItemAddedBackToSelfViaIcon = false; 720 mCurrentDragInfo = null; 721 mCurrentDragView = null; 722 mSuppressOnAdd = false; 723 724 // Reordering may have occured, and we need to save the new item locations. We do this once 725 // at the end to prevent unnecessary database operations. 726 updateItemLocationsInDatabase(); 727 } 728 729 @Override supportsFlingToDelete()730 public boolean supportsFlingToDelete() { 731 return true; 732 } 733 onFlingToDelete(DragObject d, int x, int y, PointF vec)734 public void onFlingToDelete(DragObject d, int x, int y, PointF vec) { 735 // Do nothing 736 } 737 738 @Override onFlingToDeleteCompleted()739 public void onFlingToDeleteCompleted() { 740 // Do nothing 741 } 742 updateItemLocationsInDatabase()743 private void updateItemLocationsInDatabase() { 744 ArrayList<View> list = getItemsInReadingOrder(); 745 for (int i = 0; i < list.size(); i++) { 746 View v = list.get(i); 747 ItemInfo info = (ItemInfo) v.getTag(); 748 LauncherModel.moveItemInDatabase(mLauncher, info, mInfo.id, 0, 749 info.cellX, info.cellY); 750 } 751 } 752 notifyDrop()753 public void notifyDrop() { 754 if (mDragInProgress) { 755 mItemAddedBackToSelfViaIcon = true; 756 } 757 } 758 isDropEnabled()759 public boolean isDropEnabled() { 760 return true; 761 } 762 getDropTargetDelegate(DragObject d)763 public DropTarget getDropTargetDelegate(DragObject d) { 764 return null; 765 } 766 setupContentDimensions(int count)767 private void setupContentDimensions(int count) { 768 ArrayList<View> list = getItemsInReadingOrder(); 769 770 int countX = mContent.getCountX(); 771 int countY = mContent.getCountY(); 772 boolean done = false; 773 774 while (!done) { 775 int oldCountX = countX; 776 int oldCountY = countY; 777 if (countX * countY < count) { 778 // Current grid is too small, expand it 779 if ((countX <= countY || countY == mMaxCountY) && countX < mMaxCountX) { 780 countX++; 781 } else if (countY < mMaxCountY) { 782 countY++; 783 } 784 if (countY == 0) countY++; 785 } else if ((countY - 1) * countX >= count && countY >= countX) { 786 countY = Math.max(0, countY - 1); 787 } else if ((countX - 1) * countY >= count) { 788 countX = Math.max(0, countX - 1); 789 } 790 done = countX == oldCountX && countY == oldCountY; 791 } 792 mContent.setGridSize(countX, countY); 793 arrangeChildren(list); 794 } 795 isFull()796 public boolean isFull() { 797 return getItemCount() >= mMaxNumItems; 798 } 799 centerAboutIcon()800 private void centerAboutIcon() { 801 DragLayer.LayoutParams lp = (DragLayer.LayoutParams) getLayoutParams(); 802 803 int width = getPaddingLeft() + getPaddingRight() + mContent.getDesiredWidth(); 804 int height = getPaddingTop() + getPaddingBottom() + mContent.getDesiredHeight() 805 + mFolderNameHeight; 806 DragLayer parent = (DragLayer) mLauncher.findViewById(R.id.drag_layer); 807 808 parent.getDescendantRectRelativeToSelf(mFolderIcon, mTempRect); 809 810 int centerX = mTempRect.centerX(); 811 int centerY = mTempRect.centerY(); 812 int centeredLeft = centerX - width / 2; 813 int centeredTop = centerY - height / 2; 814 815 int currentPage = mLauncher.getWorkspace().getCurrentPage(); 816 // In case the workspace is scrolling, we need to use the final scroll to compute 817 // the folders bounds. 818 mLauncher.getWorkspace().setFinalScrollForPageChange(currentPage); 819 // We first fetch the currently visible CellLayoutChildren 820 CellLayout currentLayout = (CellLayout) mLauncher.getWorkspace().getChildAt(currentPage); 821 ShortcutAndWidgetContainer boundingLayout = currentLayout.getShortcutsAndWidgets(); 822 Rect bounds = new Rect(); 823 parent.getDescendantRectRelativeToSelf(boundingLayout, bounds); 824 // We reset the workspaces scroll 825 mLauncher.getWorkspace().resetFinalScrollForPageChange(currentPage); 826 827 // We need to bound the folder to the currently visible CellLayoutChildren 828 int left = Math.min(Math.max(bounds.left, centeredLeft), 829 bounds.left + bounds.width() - width); 830 int top = Math.min(Math.max(bounds.top, centeredTop), 831 bounds.top + bounds.height() - height); 832 // If the folder doesn't fit within the bounds, center it about the desired bounds 833 if (width >= bounds.width()) { 834 left = bounds.left + (bounds.width() - width) / 2; 835 } 836 if (height >= bounds.height()) { 837 top = bounds.top + (bounds.height() - height) / 2; 838 } 839 840 int folderPivotX = width / 2 + (centeredLeft - left); 841 int folderPivotY = height / 2 + (centeredTop - top); 842 setPivotX(folderPivotX); 843 setPivotY(folderPivotY); 844 mFolderIconPivotX = (int) (mFolderIcon.getMeasuredWidth() * 845 (1.0f * folderPivotX / width)); 846 mFolderIconPivotY = (int) (mFolderIcon.getMeasuredHeight() * 847 (1.0f * folderPivotY / height)); 848 849 lp.width = width; 850 lp.height = height; 851 lp.x = left; 852 lp.y = top; 853 } 854 getPivotXForIconAnimation()855 float getPivotXForIconAnimation() { 856 return mFolderIconPivotX; 857 } getPivotYForIconAnimation()858 float getPivotYForIconAnimation() { 859 return mFolderIconPivotY; 860 } 861 setupContentForNumItems(int count)862 private void setupContentForNumItems(int count) { 863 setupContentDimensions(count); 864 865 DragLayer.LayoutParams lp = (DragLayer.LayoutParams) getLayoutParams(); 866 if (lp == null) { 867 lp = new DragLayer.LayoutParams(0, 0); 868 lp.customPosition = true; 869 setLayoutParams(lp); 870 } 871 centerAboutIcon(); 872 } 873 onMeasure(int widthMeasureSpec, int heightMeasureSpec)874 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 875 int width = getPaddingLeft() + getPaddingRight() + mContent.getDesiredWidth(); 876 int height = getPaddingTop() + getPaddingBottom() + mContent.getDesiredHeight() 877 + mFolderNameHeight; 878 879 int contentWidthSpec = MeasureSpec.makeMeasureSpec(mContent.getDesiredWidth(), 880 MeasureSpec.EXACTLY); 881 int contentHeightSpec = MeasureSpec.makeMeasureSpec(mContent.getDesiredHeight(), 882 MeasureSpec.EXACTLY); 883 mContent.measure(contentWidthSpec, contentHeightSpec); 884 885 mFolderName.measure(contentWidthSpec, 886 MeasureSpec.makeMeasureSpec(mFolderNameHeight, MeasureSpec.EXACTLY)); 887 setMeasuredDimension(width, height); 888 } 889 arrangeChildren(ArrayList<View> list)890 private void arrangeChildren(ArrayList<View> list) { 891 int[] vacant = new int[2]; 892 if (list == null) { 893 list = getItemsInReadingOrder(); 894 } 895 mContent.removeAllViews(); 896 897 for (int i = 0; i < list.size(); i++) { 898 View v = list.get(i); 899 mContent.getVacantCell(vacant, 1, 1); 900 CellLayout.LayoutParams lp = (CellLayout.LayoutParams) v.getLayoutParams(); 901 lp.cellX = vacant[0]; 902 lp.cellY = vacant[1]; 903 ItemInfo info = (ItemInfo) v.getTag(); 904 if (info.cellX != vacant[0] || info.cellY != vacant[1]) { 905 info.cellX = vacant[0]; 906 info.cellY = vacant[1]; 907 LauncherModel.addOrMoveItemInDatabase(mLauncher, info, mInfo.id, 0, 908 info.cellX, info.cellY); 909 } 910 boolean insert = false; 911 mContent.addViewToCellLayout(v, insert ? 0 : -1, (int)info.id, lp, true); 912 } 913 mItemsInvalidated = true; 914 } 915 getItemCount()916 public int getItemCount() { 917 return mContent.getShortcutsAndWidgets().getChildCount(); 918 } 919 getItemAt(int index)920 public View getItemAt(int index) { 921 return mContent.getShortcutsAndWidgets().getChildAt(index); 922 } 923 onCloseComplete()924 private void onCloseComplete() { 925 DragLayer parent = (DragLayer) getParent(); 926 if (parent != null) { 927 parent.removeView(this); 928 } 929 mDragController.removeDropTarget((DropTarget) this); 930 clearFocus(); 931 mFolderIcon.requestFocus(); 932 933 if (mRearrangeOnClose) { 934 setupContentForNumItems(getItemCount()); 935 mRearrangeOnClose = false; 936 } 937 if (getItemCount() <= 1) { 938 if (!mDragInProgress && !mSuppressFolderDeletion) { 939 replaceFolderWithFinalItem(); 940 } else if (mDragInProgress) { 941 mDeleteFolderOnDropCompleted = true; 942 } 943 } 944 mSuppressFolderDeletion = false; 945 } 946 replaceFolderWithFinalItem()947 private void replaceFolderWithFinalItem() { 948 ItemInfo finalItem = null; 949 950 if (getItemCount() == 1) { 951 finalItem = mInfo.contents.get(0); 952 } 953 954 // Remove the folder completely 955 CellLayout cellLayout = mLauncher.getCellLayout(mInfo.container, mInfo.screen); 956 cellLayout.removeView(mFolderIcon); 957 if (mFolderIcon instanceof DropTarget) { 958 mDragController.removeDropTarget((DropTarget) mFolderIcon); 959 } 960 mLauncher.removeFolder(mInfo); 961 962 if (finalItem != null) { 963 LauncherModel.addOrMoveItemInDatabase(mLauncher, finalItem, mInfo.container, 964 mInfo.screen, mInfo.cellX, mInfo.cellY); 965 } 966 LauncherModel.deleteItemFromDatabase(mLauncher, mInfo); 967 968 // Add the last remaining child to the workspace in place of the folder 969 if (finalItem != null) { 970 View child = mLauncher.createShortcut(R.layout.application, cellLayout, 971 (ShortcutInfo) finalItem); 972 973 mLauncher.getWorkspace().addInScreen(child, mInfo.container, mInfo.screen, mInfo.cellX, 974 mInfo.cellY, mInfo.spanX, mInfo.spanY); 975 } 976 } 977 978 // This method keeps track of the last item in the folder for the purposes 979 // of keyboard focus updateTextViewFocus()980 private void updateTextViewFocus() { 981 View lastChild = getItemAt(getItemCount() - 1); 982 getItemAt(getItemCount() - 1); 983 if (lastChild != null) { 984 mFolderName.setNextFocusDownId(lastChild.getId()); 985 mFolderName.setNextFocusRightId(lastChild.getId()); 986 mFolderName.setNextFocusLeftId(lastChild.getId()); 987 mFolderName.setNextFocusUpId(lastChild.getId()); 988 } 989 } 990 onDrop(DragObject d)991 public void onDrop(DragObject d) { 992 ShortcutInfo item; 993 if (d.dragInfo instanceof ApplicationInfo) { 994 // Came from all apps -- make a copy 995 item = ((ApplicationInfo) d.dragInfo).makeShortcut(); 996 item.spanX = 1; 997 item.spanY = 1; 998 } else { 999 item = (ShortcutInfo) d.dragInfo; 1000 } 1001 // Dragged from self onto self, currently this is the only path possible, however 1002 // we keep this as a distinct code path. 1003 if (item == mCurrentDragInfo) { 1004 ShortcutInfo si = (ShortcutInfo) mCurrentDragView.getTag(); 1005 CellLayout.LayoutParams lp = (CellLayout.LayoutParams) mCurrentDragView.getLayoutParams(); 1006 si.cellX = lp.cellX = mEmptyCell[0]; 1007 si.cellX = lp.cellY = mEmptyCell[1]; 1008 mContent.addViewToCellLayout(mCurrentDragView, -1, (int)item.id, lp, true); 1009 if (d.dragView.hasDrawn()) { 1010 mLauncher.getDragLayer().animateViewIntoPosition(d.dragView, mCurrentDragView); 1011 } else { 1012 d.deferDragViewCleanupPostAnimation = false; 1013 mCurrentDragView.setVisibility(VISIBLE); 1014 } 1015 mItemsInvalidated = true; 1016 setupContentDimensions(getItemCount()); 1017 mSuppressOnAdd = true; 1018 } 1019 mInfo.add(item); 1020 } 1021 onAdd(ShortcutInfo item)1022 public void onAdd(ShortcutInfo item) { 1023 mItemsInvalidated = true; 1024 // If the item was dropped onto this open folder, we have done the work associated 1025 // with adding the item to the folder, as indicated by mSuppressOnAdd being set 1026 if (mSuppressOnAdd) return; 1027 if (!findAndSetEmptyCells(item)) { 1028 // The current layout is full, can we expand it? 1029 setupContentForNumItems(getItemCount() + 1); 1030 findAndSetEmptyCells(item); 1031 } 1032 createAndAddShortcut(item); 1033 LauncherModel.addOrMoveItemInDatabase( 1034 mLauncher, item, mInfo.id, 0, item.cellX, item.cellY); 1035 } 1036 onRemove(ShortcutInfo item)1037 public void onRemove(ShortcutInfo item) { 1038 mItemsInvalidated = true; 1039 // If this item is being dragged from this open folder, we have already handled 1040 // the work associated with removing the item, so we don't have to do anything here. 1041 if (item == mCurrentDragInfo) return; 1042 View v = getViewForInfo(item); 1043 mContent.removeView(v); 1044 if (mState == STATE_ANIMATING) { 1045 mRearrangeOnClose = true; 1046 } else { 1047 setupContentForNumItems(getItemCount()); 1048 } 1049 if (getItemCount() <= 1) { 1050 replaceFolderWithFinalItem(); 1051 } 1052 } 1053 getViewForInfo(ShortcutInfo item)1054 private View getViewForInfo(ShortcutInfo item) { 1055 for (int j = 0; j < mContent.getCountY(); j++) { 1056 for (int i = 0; i < mContent.getCountX(); i++) { 1057 View v = mContent.getChildAt(i, j); 1058 if (v.getTag() == item) { 1059 return v; 1060 } 1061 } 1062 } 1063 return null; 1064 } 1065 onItemsChanged()1066 public void onItemsChanged() { 1067 updateTextViewFocus(); 1068 } 1069 onTitleChanged(CharSequence title)1070 public void onTitleChanged(CharSequence title) { 1071 } 1072 getItemsInReadingOrder()1073 public ArrayList<View> getItemsInReadingOrder() { 1074 return getItemsInReadingOrder(true); 1075 } 1076 getItemsInReadingOrder(boolean includeCurrentDragItem)1077 public ArrayList<View> getItemsInReadingOrder(boolean includeCurrentDragItem) { 1078 if (mItemsInvalidated) { 1079 mItemsInReadingOrder.clear(); 1080 for (int j = 0; j < mContent.getCountY(); j++) { 1081 for (int i = 0; i < mContent.getCountX(); i++) { 1082 View v = mContent.getChildAt(i, j); 1083 if (v != null) { 1084 ShortcutInfo info = (ShortcutInfo) v.getTag(); 1085 if (info != mCurrentDragInfo || includeCurrentDragItem) { 1086 mItemsInReadingOrder.add(v); 1087 } 1088 } 1089 } 1090 } 1091 mItemsInvalidated = false; 1092 } 1093 return mItemsInReadingOrder; 1094 } 1095 getLocationInDragLayer(int[] loc)1096 public void getLocationInDragLayer(int[] loc) { 1097 mLauncher.getDragLayer().getLocationInDragLayer(this, loc); 1098 } 1099 onFocusChange(View v, boolean hasFocus)1100 public void onFocusChange(View v, boolean hasFocus) { 1101 if (v == mFolderName && hasFocus) { 1102 startEditingFolderName(); 1103 } 1104 } 1105 } 1106