DirectoryFragment.java revision ef3e2cf35fbede340d5ff830b410baea8e461a94
1/* 2 * Copyright (C) 2013 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 17package com.android.documentsui; 18 19import static com.android.documentsui.BaseActivity.State.ACTION_BROWSE; 20import static com.android.documentsui.BaseActivity.State.ACTION_BROWSE_ALL; 21import static com.android.documentsui.BaseActivity.State.ACTION_CREATE; 22import static com.android.documentsui.BaseActivity.State.ACTION_MANAGE; 23import static com.android.documentsui.BaseActivity.State.MODE_GRID; 24import static com.android.documentsui.BaseActivity.State.MODE_LIST; 25import static com.android.documentsui.BaseActivity.State.MODE_UNKNOWN; 26import static com.android.documentsui.BaseActivity.State.SORT_ORDER_UNKNOWN; 27import static com.android.documentsui.DocumentsActivity.TAG; 28import static com.android.documentsui.model.DocumentInfo.getCursorInt; 29import static com.android.documentsui.model.DocumentInfo.getCursorLong; 30import static com.android.documentsui.model.DocumentInfo.getCursorString; 31import android.app.Activity; 32import android.app.ActivityManager; 33import android.app.Fragment; 34import android.app.FragmentManager; 35import android.app.FragmentTransaction; 36import android.app.LoaderManager.LoaderCallbacks; 37import android.content.ContentProviderClient; 38import android.content.ContentResolver; 39import android.content.ContentValues; 40import android.content.Context; 41import android.content.Intent; 42import android.content.Loader; 43import android.content.res.Resources; 44import android.database.Cursor; 45import android.graphics.Bitmap; 46import android.graphics.Point; 47import android.graphics.drawable.Drawable; 48import android.graphics.drawable.InsetDrawable; 49import android.net.Uri; 50import android.os.AsyncTask; 51import android.os.Bundle; 52import android.os.CancellationSignal; 53import android.os.OperationCanceledException; 54import android.os.Parcelable; 55import android.provider.DocumentsContract; 56import android.provider.DocumentsContract.Document; 57import android.text.format.DateUtils; 58import android.text.format.Formatter; 59import android.text.format.Time; 60import android.util.Log; 61import android.util.SparseArray; 62import android.util.SparseBooleanArray; 63import android.view.ActionMode; 64import android.view.LayoutInflater; 65import android.view.Menu; 66import android.view.MenuItem; 67import android.view.View; 68import android.view.ViewGroup; 69import android.widget.AbsListView; 70import android.widget.AbsListView.MultiChoiceModeListener; 71import android.widget.AbsListView.RecyclerListener; 72import android.widget.AdapterView; 73import android.widget.AdapterView.OnItemClickListener; 74import android.widget.BaseAdapter; 75import android.widget.GridView; 76import android.widget.ImageView; 77import android.widget.ListView; 78import android.widget.TextView; 79import android.widget.Toast; 80 81import com.android.documentsui.BaseActivity.State; 82import com.android.documentsui.ProviderExecutor.Preemptable; 83import com.android.documentsui.RecentsProvider.StateColumns; 84import com.android.documentsui.model.DocumentInfo; 85import com.android.documentsui.model.RootInfo; 86import com.google.android.collect.Lists; 87 88import java.util.ArrayList; 89import java.util.List; 90 91/** 92 * Display the documents inside a single directory. 93 */ 94public class DirectoryFragment extends Fragment { 95 96 private View mEmptyView; 97 private ListView mListView; 98 private GridView mGridView; 99 100 private AbsListView mCurrentView; 101 102 public static final int TYPE_NORMAL = 1; 103 public static final int TYPE_SEARCH = 2; 104 public static final int TYPE_RECENT_OPEN = 3; 105 106 public static final int ANIM_NONE = 1; 107 public static final int ANIM_SIDE = 2; 108 public static final int ANIM_DOWN = 3; 109 public static final int ANIM_UP = 4; 110 111 public static final int REQUEST_COPY_DESTINATION = 1; 112 113 private int mType = TYPE_NORMAL; 114 private String mStateKey; 115 116 private int mLastMode = MODE_UNKNOWN; 117 private int mLastSortOrder = SORT_ORDER_UNKNOWN; 118 private boolean mLastShowSize = false; 119 120 private boolean mHideGridTitles = false; 121 122 private boolean mSvelteRecents; 123 private Point mThumbSize; 124 125 private DocumentsAdapter mAdapter; 126 private LoaderCallbacks<DirectoryResult> mCallbacks; 127 128 private static final String EXTRA_TYPE = "type"; 129 private static final String EXTRA_ROOT = "root"; 130 private static final String EXTRA_DOC = "doc"; 131 private static final String EXTRA_QUERY = "query"; 132 private static final String EXTRA_IGNORE_STATE = "ignoreState"; 133 134 private final int mLoaderId = 42; 135 136 public static void showNormal(FragmentManager fm, RootInfo root, DocumentInfo doc, int anim) { 137 show(fm, TYPE_NORMAL, root, doc, null, anim); 138 } 139 140 public static void showSearch(FragmentManager fm, RootInfo root, String query, int anim) { 141 show(fm, TYPE_SEARCH, root, null, query, anim); 142 } 143 144 public static void showRecentsOpen(FragmentManager fm, int anim) { 145 show(fm, TYPE_RECENT_OPEN, null, null, null, anim); 146 } 147 148 private static void show(FragmentManager fm, int type, RootInfo root, DocumentInfo doc, 149 String query, int anim) { 150 final Bundle args = new Bundle(); 151 args.putInt(EXTRA_TYPE, type); 152 args.putParcelable(EXTRA_ROOT, root); 153 args.putParcelable(EXTRA_DOC, doc); 154 args.putString(EXTRA_QUERY, query); 155 156 final FragmentTransaction ft = fm.beginTransaction(); 157 switch (anim) { 158 case ANIM_SIDE: 159 args.putBoolean(EXTRA_IGNORE_STATE, true); 160 break; 161 case ANIM_DOWN: 162 args.putBoolean(EXTRA_IGNORE_STATE, true); 163 ft.setCustomAnimations(R.animator.dir_down, R.animator.dir_frozen); 164 break; 165 case ANIM_UP: 166 ft.setCustomAnimations(R.animator.dir_frozen, R.animator.dir_up); 167 break; 168 } 169 170 final DirectoryFragment fragment = new DirectoryFragment(); 171 fragment.setArguments(args); 172 173 ft.replace(R.id.container_directory, fragment); 174 ft.commitAllowingStateLoss(); 175 } 176 177 private static String buildStateKey(RootInfo root, DocumentInfo doc) { 178 final StringBuilder builder = new StringBuilder(); 179 builder.append(root != null ? root.authority : "null").append(';'); 180 builder.append(root != null ? root.rootId : "null").append(';'); 181 builder.append(doc != null ? doc.documentId : "null"); 182 return builder.toString(); 183 } 184 185 public static DirectoryFragment get(FragmentManager fm) { 186 // TODO: deal with multiple directories shown at once 187 return (DirectoryFragment) fm.findFragmentById(R.id.container_directory); 188 } 189 190 @Override 191 public View onCreateView( 192 LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { 193 final Context context = inflater.getContext(); 194 final Resources res = context.getResources(); 195 final View view = inflater.inflate(R.layout.fragment_directory, container, false); 196 197 mEmptyView = view.findViewById(android.R.id.empty); 198 199 mListView = (ListView) view.findViewById(R.id.list); 200 mListView.setOnItemClickListener(mItemListener); 201 mListView.setMultiChoiceModeListener(mMultiListener); 202 mListView.setRecyclerListener(mRecycleListener); 203 204 // Indent our list divider to align with text 205 final Drawable divider = mListView.getDivider(); 206 final boolean insetLeft = res.getBoolean(R.bool.list_divider_inset_left); 207 final int insetSize = res.getDimensionPixelSize(R.dimen.list_divider_inset); 208 if (insetLeft) { 209 mListView.setDivider(new InsetDrawable(divider, insetSize, 0, 0, 0)); 210 } else { 211 mListView.setDivider(new InsetDrawable(divider, 0, 0, insetSize, 0)); 212 } 213 214 mGridView = (GridView) view.findViewById(R.id.grid); 215 mGridView.setOnItemClickListener(mItemListener); 216 mGridView.setMultiChoiceModeListener(mMultiListener); 217 mGridView.setRecyclerListener(mRecycleListener); 218 219 return view; 220 } 221 222 @Override 223 public void onDestroyView() { 224 super.onDestroyView(); 225 226 // Cancel any outstanding thumbnail requests 227 final ViewGroup target = (mListView.getAdapter() != null) ? mListView : mGridView; 228 final int count = target.getChildCount(); 229 for (int i = 0; i < count; i++) { 230 final View view = target.getChildAt(i); 231 mRecycleListener.onMovedToScrapHeap(view); 232 } 233 234 // Tear down any selection in progress 235 mListView.setChoiceMode(AbsListView.CHOICE_MODE_NONE); 236 mGridView.setChoiceMode(AbsListView.CHOICE_MODE_NONE); 237 } 238 239 @Override 240 public void onActivityCreated(Bundle savedInstanceState) { 241 super.onActivityCreated(savedInstanceState); 242 243 final Context context = getActivity(); 244 final State state = getDisplayState(DirectoryFragment.this); 245 246 final RootInfo root = getArguments().getParcelable(EXTRA_ROOT); 247 final DocumentInfo doc = getArguments().getParcelable(EXTRA_DOC); 248 249 mAdapter = new DocumentsAdapter(); 250 mType = getArguments().getInt(EXTRA_TYPE); 251 mStateKey = buildStateKey(root, doc); 252 253 if (mType == TYPE_RECENT_OPEN) { 254 // Hide titles when showing recents for picking images/videos 255 mHideGridTitles = MimePredicate.mimeMatches( 256 MimePredicate.VISUAL_MIMES, state.acceptMimes); 257 } else { 258 mHideGridTitles = (doc != null) && doc.isGridTitlesHidden(); 259 } 260 261 final ActivityManager am = (ActivityManager) context.getSystemService( 262 Context.ACTIVITY_SERVICE); 263 mSvelteRecents = am.isLowRamDevice() && (mType == TYPE_RECENT_OPEN); 264 265 mCallbacks = new LoaderCallbacks<DirectoryResult>() { 266 @Override 267 public Loader<DirectoryResult> onCreateLoader(int id, Bundle args) { 268 final String query = getArguments().getString(EXTRA_QUERY); 269 270 Uri contentsUri; 271 switch (mType) { 272 case TYPE_NORMAL: 273 contentsUri = DocumentsContract.buildChildDocumentsUri( 274 doc.authority, doc.documentId); 275 if (state.action == ACTION_MANAGE) { 276 contentsUri = DocumentsContract.setManageMode(contentsUri); 277 } 278 return new DirectoryLoader( 279 context, mType, root, doc, contentsUri, state.userSortOrder); 280 case TYPE_SEARCH: 281 contentsUri = DocumentsContract.buildSearchDocumentsUri( 282 root.authority, root.rootId, query); 283 if (state.action == ACTION_MANAGE) { 284 contentsUri = DocumentsContract.setManageMode(contentsUri); 285 } 286 return new DirectoryLoader( 287 context, mType, root, doc, contentsUri, state.userSortOrder); 288 case TYPE_RECENT_OPEN: 289 final RootsCache roots = DocumentsApplication.getRootsCache(context); 290 return new RecentLoader(context, roots, state); 291 default: 292 throw new IllegalStateException("Unknown type " + mType); 293 } 294 } 295 296 @Override 297 public void onLoadFinished(Loader<DirectoryResult> loader, DirectoryResult result) { 298 if (!isAdded()) return; 299 300 mAdapter.swapResult(result); 301 302 // Push latest state up to UI 303 // TODO: if mode change was racing with us, don't overwrite it 304 if (result.mode != MODE_UNKNOWN) { 305 state.derivedMode = result.mode; 306 } 307 state.derivedSortOrder = result.sortOrder; 308 ((BaseActivity) context).onStateChanged(); 309 310 updateDisplayState(); 311 312 // When launched into empty recents, show drawer 313 if (mType == TYPE_RECENT_OPEN && mAdapter.isEmpty() && !state.stackTouched && 314 context instanceof DocumentsActivity) { 315 ((DocumentsActivity) context).setRootsDrawerOpen(true); 316 } 317 318 // Restore any previous instance state 319 final SparseArray<Parcelable> container = state.dirState.remove(mStateKey); 320 if (container != null && !getArguments().getBoolean(EXTRA_IGNORE_STATE, false)) { 321 getView().restoreHierarchyState(container); 322 } else if (mLastSortOrder != state.derivedSortOrder) { 323 mListView.smoothScrollToPosition(0); 324 mGridView.smoothScrollToPosition(0); 325 } 326 327 mLastSortOrder = state.derivedSortOrder; 328 } 329 330 @Override 331 public void onLoaderReset(Loader<DirectoryResult> loader) { 332 mAdapter.swapResult(null); 333 } 334 }; 335 336 // Kick off loader at least once 337 getLoaderManager().restartLoader(mLoaderId, null, mCallbacks); 338 339 updateDisplayState(); 340 } 341 342 @Override 343 public void onActivityResult(int requestCode, int resultCode, Intent data) { 344 final Context context = getActivity(); 345 final Resources res = context.getResources(); 346 347 // There's only one request code right now. Replace this with a switch statement or 348 // something more scalable when more codes are added. 349 if (requestCode != REQUEST_COPY_DESTINATION) { 350 return; 351 } 352 if (resultCode == Activity.RESULT_CANCELED || data == null) { 353 // User pressed the back button or otherwise cancelled the destination pick. Don't 354 // proceed with the copy. 355 return; 356 } 357 358 final List<DocumentInfo> docs = getDisplayState(this).selectedDocumentsForCopy; 359 final Intent copyIntent = new Intent(context, CopyService.class); 360 copyIntent.putParcelableArrayListExtra(CopyService.EXTRA_SRC_LIST, new ArrayList<DocumentInfo>(docs)); 361 copyIntent.putExtra(CopyService.EXTRA_STACK, data.getParcelableExtra(CopyService.EXTRA_STACK)); 362 363 Toast.makeText(context, 364 res.getQuantityString(R.plurals.copy_begin, docs.size(), docs.size()), 365 Toast.LENGTH_SHORT).show(); 366 context.startService(copyIntent); 367 } 368 369 @Override 370 public void onStop() { 371 super.onStop(); 372 373 // Remember last scroll location 374 final SparseArray<Parcelable> container = new SparseArray<Parcelable>(); 375 getView().saveHierarchyState(container); 376 final State state = getDisplayState(this); 377 state.dirState.put(mStateKey, container); 378 } 379 380 @Override 381 public void onResume() { 382 super.onResume(); 383 updateDisplayState(); 384 } 385 386 public void onDisplayStateChanged() { 387 updateDisplayState(); 388 } 389 390 public void onUserSortOrderChanged() { 391 // Sort order change always triggers reload; we'll trigger state change 392 // on the flip side. 393 getLoaderManager().restartLoader(mLoaderId, null, mCallbacks); 394 } 395 396 public void onUserModeChanged() { 397 final ContentResolver resolver = getActivity().getContentResolver(); 398 final State state = getDisplayState(this); 399 400 final RootInfo root = getArguments().getParcelable(EXTRA_ROOT); 401 final DocumentInfo doc = getArguments().getParcelable(EXTRA_DOC); 402 403 if (root != null && doc != null) { 404 final Uri stateUri = RecentsProvider.buildState( 405 root.authority, root.rootId, doc.documentId); 406 final ContentValues values = new ContentValues(); 407 values.put(StateColumns.MODE, state.userMode); 408 409 new AsyncTask<Void, Void, Void>() { 410 @Override 411 protected Void doInBackground(Void... params) { 412 resolver.insert(stateUri, values); 413 return null; 414 } 415 }.execute(); 416 } 417 418 // Mode change is just visual change; no need to kick loader, and 419 // deliver change event immediately. 420 state.derivedMode = state.userMode; 421 ((BaseActivity) getActivity()).onStateChanged(); 422 423 updateDisplayState(); 424 } 425 426 private void updateDisplayState() { 427 final State state = getDisplayState(this); 428 429 if (mLastMode == state.derivedMode && mLastShowSize == state.showSize) return; 430 mLastMode = state.derivedMode; 431 mLastShowSize = state.showSize; 432 433 mListView.setVisibility(state.derivedMode == MODE_LIST ? View.VISIBLE : View.GONE); 434 mGridView.setVisibility(state.derivedMode == MODE_GRID ? View.VISIBLE : View.GONE); 435 436 final int choiceMode; 437 if (state.allowMultiple) { 438 choiceMode = ListView.CHOICE_MODE_MULTIPLE_MODAL; 439 } else { 440 choiceMode = ListView.CHOICE_MODE_NONE; 441 } 442 443 final int thumbSize; 444 if (state.derivedMode == MODE_GRID) { 445 thumbSize = getResources().getDimensionPixelSize(R.dimen.grid_width); 446 mListView.setAdapter(null); 447 mListView.setChoiceMode(ListView.CHOICE_MODE_NONE); 448 mGridView.setAdapter(mAdapter); 449 mGridView.setColumnWidth(getResources().getDimensionPixelSize(R.dimen.grid_width)); 450 mGridView.setNumColumns(GridView.AUTO_FIT); 451 mGridView.setChoiceMode(choiceMode); 452 mCurrentView = mGridView; 453 } else if (state.derivedMode == MODE_LIST) { 454 thumbSize = getResources().getDimensionPixelSize(R.dimen.icon_size); 455 mGridView.setAdapter(null); 456 mGridView.setChoiceMode(ListView.CHOICE_MODE_NONE); 457 mListView.setAdapter(mAdapter); 458 mListView.setChoiceMode(choiceMode); 459 mCurrentView = mListView; 460 } else { 461 throw new IllegalStateException("Unknown state " + state.derivedMode); 462 } 463 464 mThumbSize = new Point(thumbSize, thumbSize); 465 } 466 467 private OnItemClickListener mItemListener = new OnItemClickListener() { 468 @Override 469 public void onItemClick(AdapterView<?> parent, View view, int position, long id) { 470 final Cursor cursor = mAdapter.getItem(position); 471 if (cursor != null) { 472 final String docMimeType = getCursorString(cursor, Document.COLUMN_MIME_TYPE); 473 final int docFlags = getCursorInt(cursor, Document.COLUMN_FLAGS); 474 if (isDocumentEnabled(docMimeType, docFlags)) { 475 final DocumentInfo doc = DocumentInfo.fromDirectoryCursor(cursor); 476 ((BaseActivity) getActivity()).onDocumentPicked(doc); 477 } 478 } 479 } 480 }; 481 482 private MultiChoiceModeListener mMultiListener = new MultiChoiceModeListener() { 483 @Override 484 public boolean onCreateActionMode(ActionMode mode, Menu menu) { 485 mode.getMenuInflater().inflate(R.menu.mode_directory, menu); 486 mode.setTitle(getResources() 487 .getString(R.string.mode_selected_count, mCurrentView.getCheckedItemCount())); 488 return true; 489 } 490 491 @Override 492 public boolean onPrepareActionMode(ActionMode mode, Menu menu) { 493 final State state = getDisplayState(DirectoryFragment.this); 494 495 final MenuItem open = menu.findItem(R.id.menu_open); 496 final MenuItem share = menu.findItem(R.id.menu_share); 497 final MenuItem delete = menu.findItem(R.id.menu_delete); 498 final MenuItem copy = menu.findItem(R.id.menu_copy); 499 500 final boolean manageOrBrowse = (state.action == ACTION_MANAGE 501 || state.action == ACTION_BROWSE || state.action == ACTION_BROWSE_ALL); 502 503 open.setVisible(!manageOrBrowse); 504 share.setVisible(manageOrBrowse); 505 delete.setVisible(manageOrBrowse); 506 copy.setVisible(manageOrBrowse); 507 508 return true; 509 } 510 511 @Override 512 public boolean onActionItemClicked(ActionMode mode, MenuItem item) { 513 final SparseBooleanArray checked = mCurrentView.getCheckedItemPositions(); 514 final ArrayList<DocumentInfo> docs = Lists.newArrayList(); 515 final int size = checked.size(); 516 for (int i = 0; i < size; i++) { 517 if (checked.valueAt(i)) { 518 final Cursor cursor = mAdapter.getItem(checked.keyAt(i)); 519 final DocumentInfo doc = DocumentInfo.fromDirectoryCursor(cursor); 520 docs.add(doc); 521 } 522 } 523 524 final int id = item.getItemId(); 525 if (id == R.id.menu_open) { 526 BaseActivity.get(DirectoryFragment.this).onDocumentsPicked(docs); 527 mode.finish(); 528 return true; 529 530 } else if (id == R.id.menu_share) { 531 onShareDocuments(docs); 532 mode.finish(); 533 return true; 534 535 } else if (id == R.id.menu_delete) { 536 onDeleteDocuments(docs); 537 mode.finish(); 538 return true; 539 540 } else if (id == R.id.menu_copy) { 541 onCopyDocuments(docs); 542 mode.finish(); 543 return true; 544 545 } else if (id == R.id.menu_select_all) { 546 int count = mCurrentView.getCount(); 547 for (int i = 0; i < count; i++) { 548 mCurrentView.setItemChecked(i, true); 549 } 550 updateDisplayState(); 551 return true; 552 553 } else { 554 return false; 555 } 556 } 557 558 @Override 559 public void onDestroyActionMode(ActionMode mode) { 560 // ignored 561 } 562 563 @Override 564 public void onItemCheckedStateChanged( 565 ActionMode mode, int position, long id, boolean checked) { 566 if (checked) { 567 // Directories and footer items cannot be checked 568 boolean valid = false; 569 570 final Cursor cursor = mAdapter.getItem(position); 571 if (cursor != null) { 572 final String docMimeType = getCursorString(cursor, Document.COLUMN_MIME_TYPE); 573 final int docFlags = getCursorInt(cursor, Document.COLUMN_FLAGS); 574 valid = isDocumentEnabled(docMimeType, docFlags); 575 } 576 577 if (!valid) { 578 mCurrentView.setItemChecked(position, false); 579 } 580 } 581 582 mode.setTitle(getResources() 583 .getString(R.string.mode_selected_count, mCurrentView.getCheckedItemCount())); 584 } 585 }; 586 587 private RecyclerListener mRecycleListener = new RecyclerListener() { 588 @Override 589 public void onMovedToScrapHeap(View view) { 590 final ImageView iconThumb = (ImageView) view.findViewById(R.id.icon_thumb); 591 if (iconThumb != null) { 592 final ThumbnailAsyncTask oldTask = (ThumbnailAsyncTask) iconThumb.getTag(); 593 if (oldTask != null) { 594 oldTask.preempt(); 595 iconThumb.setTag(null); 596 } 597 } 598 } 599 }; 600 601 private void onShareDocuments(List<DocumentInfo> docs) { 602 Intent intent; 603 604 // Filter out directories - those can't be shared. 605 List<DocumentInfo> docsForSend = Lists.newArrayList(); 606 for (DocumentInfo doc: docs) { 607 if (!Document.MIME_TYPE_DIR.equals(doc.mimeType)) { 608 docsForSend.add(doc); 609 } 610 } 611 612 if (docsForSend.size() == 1) { 613 final DocumentInfo doc = docsForSend.get(0); 614 615 intent = new Intent(Intent.ACTION_SEND); 616 intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); 617 intent.addCategory(Intent.CATEGORY_DEFAULT); 618 intent.setType(doc.mimeType); 619 intent.putExtra(Intent.EXTRA_STREAM, doc.derivedUri); 620 621 } else if (docsForSend.size() > 1) { 622 intent = new Intent(Intent.ACTION_SEND_MULTIPLE); 623 intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); 624 intent.addCategory(Intent.CATEGORY_DEFAULT); 625 626 final ArrayList<String> mimeTypes = Lists.newArrayList(); 627 final ArrayList<Uri> uris = Lists.newArrayList(); 628 for (DocumentInfo doc : docsForSend) { 629 mimeTypes.add(doc.mimeType); 630 uris.add(doc.derivedUri); 631 } 632 633 intent.setType(findCommonMimeType(mimeTypes)); 634 intent.putParcelableArrayListExtra(Intent.EXTRA_STREAM, uris); 635 636 } else { 637 return; 638 } 639 640 intent = Intent.createChooser(intent, getActivity().getText(R.string.share_via)); 641 startActivity(intent); 642 } 643 644 private void onDeleteDocuments(List<DocumentInfo> docs) { 645 final Context context = getActivity(); 646 final ContentResolver resolver = context.getContentResolver(); 647 648 boolean hadTrouble = false; 649 for (DocumentInfo doc : docs) { 650 if (!doc.isDeleteSupported()) { 651 Log.w(TAG, "Skipping " + doc); 652 hadTrouble = true; 653 continue; 654 } 655 656 ContentProviderClient client = null; 657 try { 658 client = DocumentsApplication.acquireUnstableProviderOrThrow( 659 resolver, doc.derivedUri.getAuthority()); 660 DocumentsContract.deleteDocument(client, doc.derivedUri); 661 } catch (Exception e) { 662 Log.w(TAG, "Failed to delete " + doc); 663 hadTrouble = true; 664 } finally { 665 ContentProviderClient.releaseQuietly(client); 666 } 667 } 668 669 if (hadTrouble) { 670 Toast.makeText(context, R.string.toast_failed_delete, Toast.LENGTH_SHORT).show(); 671 } 672 } 673 674 private void onCopyDocuments(List<DocumentInfo> docs) { 675 getDisplayState(this).selectedDocumentsForCopy = docs; 676 677 // Pop up a dialog to pick a destination. This is inadequate but works for now. 678 // TODO: Implement a picker that is to spec. 679 final Intent intent = new Intent( 680 BaseActivity.DocumentsIntent.ACTION_OPEN_COPY_DESTINATION, 681 Uri.EMPTY, 682 getActivity(), 683 DocumentsActivity.class); 684 boolean directoryCopy = false; 685 for (DocumentInfo info : docs) { 686 if (Document.MIME_TYPE_DIR.equals(info.mimeType)) { 687 directoryCopy = true; 688 break; 689 } 690 } 691 intent.putExtra(BaseActivity.DocumentsIntent.EXTRA_DIRECTORY_COPY, directoryCopy); 692 startActivityForResult(intent, REQUEST_COPY_DESTINATION); 693 } 694 695 private static State getDisplayState(Fragment fragment) { 696 return ((BaseActivity) fragment.getActivity()).getDisplayState(); 697 } 698 699 private static abstract class Footer { 700 private final int mItemViewType; 701 702 public Footer(int itemViewType) { 703 mItemViewType = itemViewType; 704 } 705 706 public abstract View getView(View convertView, ViewGroup parent); 707 708 public int getItemViewType() { 709 return mItemViewType; 710 } 711 } 712 713 private class LoadingFooter extends Footer { 714 public LoadingFooter() { 715 super(1); 716 } 717 718 @Override 719 public View getView(View convertView, ViewGroup parent) { 720 final Context context = parent.getContext(); 721 final State state = getDisplayState(DirectoryFragment.this); 722 723 if (convertView == null) { 724 final LayoutInflater inflater = LayoutInflater.from(context); 725 if (state.derivedMode == MODE_LIST) { 726 convertView = inflater.inflate(R.layout.item_loading_list, parent, false); 727 } else if (state.derivedMode == MODE_GRID) { 728 convertView = inflater.inflate(R.layout.item_loading_grid, parent, false); 729 } else { 730 throw new IllegalStateException(); 731 } 732 } 733 734 return convertView; 735 } 736 } 737 738 private class MessageFooter extends Footer { 739 private final int mIcon; 740 private final String mMessage; 741 742 public MessageFooter(int itemViewType, int icon, String message) { 743 super(itemViewType); 744 mIcon = icon; 745 mMessage = message; 746 } 747 748 @Override 749 public View getView(View convertView, ViewGroup parent) { 750 final Context context = parent.getContext(); 751 final State state = getDisplayState(DirectoryFragment.this); 752 753 if (convertView == null) { 754 final LayoutInflater inflater = LayoutInflater.from(context); 755 if (state.derivedMode == MODE_LIST) { 756 convertView = inflater.inflate(R.layout.item_message_list, parent, false); 757 } else if (state.derivedMode == MODE_GRID) { 758 convertView = inflater.inflate(R.layout.item_message_grid, parent, false); 759 } else { 760 throw new IllegalStateException(); 761 } 762 } 763 764 final ImageView icon = (ImageView) convertView.findViewById(android.R.id.icon); 765 final TextView title = (TextView) convertView.findViewById(android.R.id.title); 766 icon.setImageResource(mIcon); 767 title.setText(mMessage); 768 return convertView; 769 } 770 } 771 772 private class DocumentsAdapter extends BaseAdapter { 773 private Cursor mCursor; 774 private int mCursorCount; 775 776 private List<Footer> mFooters = Lists.newArrayList(); 777 778 public void swapResult(DirectoryResult result) { 779 mCursor = result != null ? result.cursor : null; 780 mCursorCount = mCursor != null ? mCursor.getCount() : 0; 781 782 mFooters.clear(); 783 784 final Bundle extras = mCursor != null ? mCursor.getExtras() : null; 785 if (extras != null) { 786 final String info = extras.getString(DocumentsContract.EXTRA_INFO); 787 if (info != null) { 788 mFooters.add(new MessageFooter(2, R.drawable.ic_dialog_info, info)); 789 } 790 final String error = extras.getString(DocumentsContract.EXTRA_ERROR); 791 if (error != null) { 792 mFooters.add(new MessageFooter(3, R.drawable.ic_dialog_alert, error)); 793 } 794 if (extras.getBoolean(DocumentsContract.EXTRA_LOADING, false)) { 795 mFooters.add(new LoadingFooter()); 796 } 797 } 798 799 if (result != null && result.exception != null) { 800 mFooters.add(new MessageFooter( 801 3, R.drawable.ic_dialog_alert, getString(R.string.query_error))); 802 } 803 804 if (isEmpty()) { 805 mEmptyView.setVisibility(View.VISIBLE); 806 } else { 807 mEmptyView.setVisibility(View.GONE); 808 } 809 810 notifyDataSetChanged(); 811 } 812 813 @Override 814 public View getView(int position, View convertView, ViewGroup parent) { 815 if (position < mCursorCount) { 816 return getDocumentView(position, convertView, parent); 817 } else { 818 position -= mCursorCount; 819 convertView = mFooters.get(position).getView(convertView, parent); 820 // Only the view itself is disabled; contents inside shouldn't 821 // be dimmed. 822 convertView.setEnabled(false); 823 return convertView; 824 } 825 } 826 827 private View getDocumentView(int position, View convertView, ViewGroup parent) { 828 final Context context = parent.getContext(); 829 final State state = getDisplayState(DirectoryFragment.this); 830 831 final DocumentInfo doc = getArguments().getParcelable(EXTRA_DOC); 832 833 final RootsCache roots = DocumentsApplication.getRootsCache(context); 834 final ThumbnailCache thumbs = DocumentsApplication.getThumbnailsCache( 835 context, mThumbSize); 836 837 if (convertView == null) { 838 final LayoutInflater inflater = LayoutInflater.from(context); 839 if (state.derivedMode == MODE_LIST) { 840 convertView = inflater.inflate(R.layout.item_doc_list, parent, false); 841 } else if (state.derivedMode == MODE_GRID) { 842 convertView = inflater.inflate(R.layout.item_doc_grid, parent, false); 843 } else { 844 throw new IllegalStateException(); 845 } 846 } 847 848 final Cursor cursor = getItem(position); 849 850 final String docAuthority = getCursorString(cursor, RootCursorWrapper.COLUMN_AUTHORITY); 851 final String docRootId = getCursorString(cursor, RootCursorWrapper.COLUMN_ROOT_ID); 852 final String docId = getCursorString(cursor, Document.COLUMN_DOCUMENT_ID); 853 final String docMimeType = getCursorString(cursor, Document.COLUMN_MIME_TYPE); 854 final String docDisplayName = getCursorString(cursor, Document.COLUMN_DISPLAY_NAME); 855 final long docLastModified = getCursorLong(cursor, Document.COLUMN_LAST_MODIFIED); 856 final int docIcon = getCursorInt(cursor, Document.COLUMN_ICON); 857 final int docFlags = getCursorInt(cursor, Document.COLUMN_FLAGS); 858 final String docSummary = getCursorString(cursor, Document.COLUMN_SUMMARY); 859 final long docSize = getCursorLong(cursor, Document.COLUMN_SIZE); 860 861 final View line1 = convertView.findViewById(R.id.line1); 862 final View line2 = convertView.findViewById(R.id.line2); 863 864 final ImageView iconMime = (ImageView) convertView.findViewById(R.id.icon_mime); 865 final ImageView iconThumb = (ImageView) convertView.findViewById(R.id.icon_thumb); 866 final TextView title = (TextView) convertView.findViewById(android.R.id.title); 867 final ImageView icon1 = (ImageView) convertView.findViewById(android.R.id.icon1); 868 final ImageView icon2 = (ImageView) convertView.findViewById(android.R.id.icon2); 869 final TextView summary = (TextView) convertView.findViewById(android.R.id.summary); 870 final TextView date = (TextView) convertView.findViewById(R.id.date); 871 final TextView size = (TextView) convertView.findViewById(R.id.size); 872 873 final ThumbnailAsyncTask oldTask = (ThumbnailAsyncTask) iconThumb.getTag(); 874 if (oldTask != null) { 875 oldTask.preempt(); 876 iconThumb.setTag(null); 877 } 878 879 iconMime.animate().cancel(); 880 iconThumb.animate().cancel(); 881 882 final boolean supportsThumbnail = (docFlags & Document.FLAG_SUPPORTS_THUMBNAIL) != 0; 883 final boolean allowThumbnail = (state.derivedMode == MODE_GRID) 884 || MimePredicate.mimeMatches(MimePredicate.VISUAL_MIMES, docMimeType); 885 final boolean showThumbnail = supportsThumbnail && allowThumbnail && !mSvelteRecents; 886 887 final boolean enabled = isDocumentEnabled(docMimeType, docFlags); 888 final float iconAlpha = (state.derivedMode == MODE_LIST && !enabled) ? 0.5f : 1f; 889 890 boolean cacheHit = false; 891 if (showThumbnail) { 892 final Uri uri = DocumentsContract.buildDocumentUri(docAuthority, docId); 893 final Bitmap cachedResult = thumbs.get(uri); 894 if (cachedResult != null) { 895 iconThumb.setImageBitmap(cachedResult); 896 cacheHit = true; 897 } else { 898 iconThumb.setImageDrawable(null); 899 final ThumbnailAsyncTask task = new ThumbnailAsyncTask( 900 uri, iconMime, iconThumb, mThumbSize, iconAlpha); 901 iconThumb.setTag(task); 902 ProviderExecutor.forAuthority(docAuthority).execute(task); 903 } 904 } 905 906 // Always throw MIME icon into place, even when a thumbnail is being 907 // loaded in background. 908 if (cacheHit) { 909 iconMime.setAlpha(0f); 910 iconMime.setImageDrawable(null); 911 iconThumb.setAlpha(1f); 912 } else { 913 iconMime.setAlpha(1f); 914 iconThumb.setAlpha(0f); 915 iconThumb.setImageDrawable(null); 916 if (docIcon != 0) { 917 iconMime.setImageDrawable( 918 IconUtils.loadPackageIcon(context, docAuthority, docIcon)); 919 } else { 920 iconMime.setImageDrawable(IconUtils.loadMimeIcon( 921 context, docMimeType, docAuthority, docId, state.derivedMode)); 922 } 923 } 924 925 boolean hasLine1 = false; 926 boolean hasLine2 = false; 927 928 final boolean hideTitle = (state.derivedMode == MODE_GRID) && mHideGridTitles; 929 if (!hideTitle) { 930 title.setText(docDisplayName); 931 hasLine1 = true; 932 } 933 934 Drawable iconDrawable = null; 935 if (mType == TYPE_RECENT_OPEN) { 936 // We've already had to enumerate roots before any results can 937 // be shown, so this will never block. 938 final RootInfo root = roots.getRootBlocking(docAuthority, docRootId); 939 if (state.derivedMode == MODE_GRID) { 940 iconDrawable = root.loadGridIcon(context); 941 } else { 942 iconDrawable = root.loadIcon(context); 943 } 944 945 if (summary != null) { 946 final boolean alwaysShowSummary = getResources() 947 .getBoolean(R.bool.always_show_summary); 948 if (alwaysShowSummary) { 949 summary.setText(root.getDirectoryString()); 950 summary.setVisibility(View.VISIBLE); 951 hasLine2 = true; 952 } else { 953 if (iconDrawable != null && roots.isIconUniqueBlocking(root)) { 954 // No summary needed if icon speaks for itself 955 summary.setVisibility(View.INVISIBLE); 956 } else { 957 summary.setText(root.getDirectoryString()); 958 summary.setVisibility(View.VISIBLE); 959 summary.setTextAlignment(TextView.TEXT_ALIGNMENT_TEXT_END); 960 hasLine2 = true; 961 } 962 } 963 } 964 } else { 965 // Directories showing thumbnails in grid mode get a little icon 966 // hint to remind user they're a directory. 967 if (Document.MIME_TYPE_DIR.equals(docMimeType) && state.derivedMode == MODE_GRID 968 && showThumbnail) { 969 iconDrawable = IconUtils.applyTintAttr(context, R.drawable.ic_doc_folder, 970 android.R.attr.textColorPrimaryInverse); 971 } 972 973 if (summary != null) { 974 if (docSummary != null) { 975 summary.setText(docSummary); 976 summary.setVisibility(View.VISIBLE); 977 hasLine2 = true; 978 } else { 979 summary.setVisibility(View.INVISIBLE); 980 } 981 } 982 } 983 984 if (icon1 != null) icon1.setVisibility(View.GONE); 985 if (icon2 != null) icon2.setVisibility(View.GONE); 986 987 if (iconDrawable != null) { 988 if (hasLine1) { 989 icon1.setVisibility(View.VISIBLE); 990 icon1.setImageDrawable(iconDrawable); 991 } else { 992 icon2.setVisibility(View.VISIBLE); 993 icon2.setImageDrawable(iconDrawable); 994 } 995 } 996 997 if (docLastModified == -1) { 998 date.setText(null); 999 } else { 1000 date.setText(formatTime(context, docLastModified)); 1001 hasLine2 = true; 1002 } 1003 1004 if (state.showSize) { 1005 size.setVisibility(View.VISIBLE); 1006 if (Document.MIME_TYPE_DIR.equals(docMimeType) || docSize == -1) { 1007 size.setText(null); 1008 } else { 1009 size.setText(Formatter.formatFileSize(context, docSize)); 1010 hasLine2 = true; 1011 } 1012 } else { 1013 size.setVisibility(View.GONE); 1014 } 1015 1016 if (line1 != null) { 1017 line1.setVisibility(hasLine1 ? View.VISIBLE : View.GONE); 1018 } 1019 if (line2 != null) { 1020 line2.setVisibility(hasLine2 ? View.VISIBLE : View.GONE); 1021 } 1022 1023 setEnabledRecursive(convertView, enabled); 1024 1025 iconMime.setAlpha(iconAlpha); 1026 iconThumb.setAlpha(iconAlpha); 1027 if (icon1 != null) icon1.setAlpha(iconAlpha); 1028 if (icon2 != null) icon2.setAlpha(iconAlpha); 1029 1030 return convertView; 1031 } 1032 1033 @Override 1034 public int getCount() { 1035 return mCursorCount + mFooters.size(); 1036 } 1037 1038 @Override 1039 public Cursor getItem(int position) { 1040 if (position < mCursorCount) { 1041 mCursor.moveToPosition(position); 1042 return mCursor; 1043 } else { 1044 return null; 1045 } 1046 } 1047 1048 @Override 1049 public long getItemId(int position) { 1050 return position; 1051 } 1052 1053 @Override 1054 public int getViewTypeCount() { 1055 return 4; 1056 } 1057 1058 @Override 1059 public int getItemViewType(int position) { 1060 if (position < mCursorCount) { 1061 return 0; 1062 } else { 1063 position -= mCursorCount; 1064 return mFooters.get(position).getItemViewType(); 1065 } 1066 } 1067 } 1068 1069 private static class ThumbnailAsyncTask extends AsyncTask<Uri, Void, Bitmap> 1070 implements Preemptable { 1071 private final Uri mUri; 1072 private final ImageView mIconMime; 1073 private final ImageView mIconThumb; 1074 private final Point mThumbSize; 1075 private final float mTargetAlpha; 1076 private final CancellationSignal mSignal; 1077 1078 public ThumbnailAsyncTask(Uri uri, ImageView iconMime, ImageView iconThumb, Point thumbSize, 1079 float targetAlpha) { 1080 mUri = uri; 1081 mIconMime = iconMime; 1082 mIconThumb = iconThumb; 1083 mThumbSize = thumbSize; 1084 mTargetAlpha = targetAlpha; 1085 mSignal = new CancellationSignal(); 1086 } 1087 1088 @Override 1089 public void preempt() { 1090 cancel(false); 1091 mSignal.cancel(); 1092 } 1093 1094 @Override 1095 protected Bitmap doInBackground(Uri... params) { 1096 if (isCancelled()) return null; 1097 1098 final Context context = mIconThumb.getContext(); 1099 final ContentResolver resolver = context.getContentResolver(); 1100 1101 ContentProviderClient client = null; 1102 Bitmap result = null; 1103 try { 1104 client = DocumentsApplication.acquireUnstableProviderOrThrow( 1105 resolver, mUri.getAuthority()); 1106 result = DocumentsContract.getDocumentThumbnail(client, mUri, mThumbSize, mSignal); 1107 if (result != null) { 1108 final ThumbnailCache thumbs = DocumentsApplication.getThumbnailsCache( 1109 context, mThumbSize); 1110 thumbs.put(mUri, result); 1111 } 1112 } catch (Exception e) { 1113 if (!(e instanceof OperationCanceledException)) { 1114 Log.w(TAG, "Failed to load thumbnail for " + mUri + ": " + e); 1115 } 1116 } finally { 1117 ContentProviderClient.releaseQuietly(client); 1118 } 1119 return result; 1120 } 1121 1122 @Override 1123 protected void onPostExecute(Bitmap result) { 1124 if (mIconThumb.getTag() == this && result != null) { 1125 mIconThumb.setTag(null); 1126 mIconThumb.setImageBitmap(result); 1127 1128 mIconMime.setAlpha(mTargetAlpha); 1129 mIconMime.animate().alpha(0f).start(); 1130 mIconThumb.setAlpha(0f); 1131 mIconThumb.animate().alpha(mTargetAlpha).start(); 1132 } 1133 } 1134 } 1135 1136 private static String formatTime(Context context, long when) { 1137 // TODO: DateUtils should make this easier 1138 Time then = new Time(); 1139 then.set(when); 1140 Time now = new Time(); 1141 now.setToNow(); 1142 1143 int flags = DateUtils.FORMAT_NO_NOON | DateUtils.FORMAT_NO_MIDNIGHT 1144 | DateUtils.FORMAT_ABBREV_ALL; 1145 1146 if (then.year != now.year) { 1147 flags |= DateUtils.FORMAT_SHOW_YEAR | DateUtils.FORMAT_SHOW_DATE; 1148 } else if (then.yearDay != now.yearDay) { 1149 flags |= DateUtils.FORMAT_SHOW_DATE; 1150 } else { 1151 flags |= DateUtils.FORMAT_SHOW_TIME; 1152 } 1153 1154 return DateUtils.formatDateTime(context, when, flags); 1155 } 1156 1157 private String findCommonMimeType(List<String> mimeTypes) { 1158 String[] commonType = mimeTypes.get(0).split("/"); 1159 if (commonType.length != 2) { 1160 return "*/*"; 1161 } 1162 1163 for (int i = 1; i < mimeTypes.size(); i++) { 1164 String[] type = mimeTypes.get(i).split("/"); 1165 if (type.length != 2) continue; 1166 1167 if (!commonType[1].equals(type[1])) { 1168 commonType[1] = "*"; 1169 } 1170 1171 if (!commonType[0].equals(type[0])) { 1172 commonType[0] = "*"; 1173 commonType[1] = "*"; 1174 break; 1175 } 1176 } 1177 1178 return commonType[0] + "/" + commonType[1]; 1179 } 1180 1181 private void setEnabledRecursive(View v, boolean enabled) { 1182 if (v == null) return; 1183 if (v.isEnabled() == enabled) return; 1184 v.setEnabled(enabled); 1185 1186 if (v instanceof ViewGroup) { 1187 final ViewGroup vg = (ViewGroup) v; 1188 for (int i = vg.getChildCount() - 1; i >= 0; i--) { 1189 setEnabledRecursive(vg.getChildAt(i), enabled); 1190 } 1191 } 1192 } 1193 1194 private boolean isDocumentEnabled(String docMimeType, int docFlags) { 1195 final State state = getDisplayState(DirectoryFragment.this); 1196 1197 // Directories are always enabled 1198 if (Document.MIME_TYPE_DIR.equals(docMimeType)) { 1199 return true; 1200 } 1201 1202 // Read-only files are disabled when creating 1203 if (state.action == ACTION_CREATE && (docFlags & Document.FLAG_SUPPORTS_WRITE) == 0) { 1204 return false; 1205 } 1206 1207 return MimePredicate.mimeMatches(state.acceptMimes, docMimeType); 1208 } 1209} 1210