DirectoryFragment.java revision f6db154975ef575479ba4ab59d80bcf592288252
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.DocumentsActivity.TAG; 20import static com.android.documentsui.DocumentsActivity.State.ACTION_MANAGE; 21import static com.android.documentsui.DocumentsActivity.State.MODE_GRID; 22import static com.android.documentsui.DocumentsActivity.State.MODE_LIST; 23import static com.android.documentsui.DocumentsActivity.State.MODE_UNKNOWN; 24import static com.android.documentsui.DocumentsActivity.State.SORT_ORDER_UNKNOWN; 25import static com.android.documentsui.model.DocumentInfo.getCursorInt; 26import static com.android.documentsui.model.DocumentInfo.getCursorLong; 27import static com.android.documentsui.model.DocumentInfo.getCursorString; 28 29import android.app.Fragment; 30import android.app.FragmentManager; 31import android.app.FragmentTransaction; 32import android.app.LoaderManager.LoaderCallbacks; 33import android.content.ContentResolver; 34import android.content.ContentValues; 35import android.content.Context; 36import android.content.Intent; 37import android.content.Loader; 38import android.database.Cursor; 39import android.graphics.Bitmap; 40import android.graphics.Point; 41import android.graphics.drawable.Drawable; 42import android.net.Uri; 43import android.os.AsyncTask; 44import android.os.Bundle; 45import android.os.CancellationSignal; 46import android.provider.DocumentsContract; 47import android.provider.DocumentsContract.Document; 48import android.text.format.DateUtils; 49import android.text.format.Formatter; 50import android.text.format.Time; 51import android.util.Log; 52import android.util.SparseBooleanArray; 53import android.view.ActionMode; 54import android.view.LayoutInflater; 55import android.view.Menu; 56import android.view.MenuItem; 57import android.view.View; 58import android.view.ViewGroup; 59import android.widget.AbsListView; 60import android.widget.AbsListView.MultiChoiceModeListener; 61import android.widget.AbsListView.RecyclerListener; 62import android.widget.AdapterView; 63import android.widget.AdapterView.OnItemClickListener; 64import android.widget.BaseAdapter; 65import android.widget.GridView; 66import android.widget.ImageView; 67import android.widget.ListView; 68import android.widget.TextView; 69import android.widget.Toast; 70 71import com.android.documentsui.DocumentsActivity.State; 72import com.android.documentsui.RecentsProvider.StateColumns; 73import com.android.documentsui.model.DocumentInfo; 74import com.android.documentsui.model.RootInfo; 75import com.android.internal.util.Predicate; 76import com.google.android.collect.Lists; 77 78import java.util.ArrayList; 79import java.util.List; 80import java.util.concurrent.atomic.AtomicInteger; 81 82/** 83 * Display the documents inside a single directory. 84 */ 85public class DirectoryFragment extends Fragment { 86 87 private View mEmptyView; 88 private ListView mListView; 89 private GridView mGridView; 90 91 private AbsListView mCurrentView; 92 93 private Predicate<DocumentInfo> mFilter; 94 95 public static final int TYPE_NORMAL = 1; 96 public static final int TYPE_SEARCH = 2; 97 public static final int TYPE_RECENT_OPEN = 3; 98 99 private int mType = TYPE_NORMAL; 100 101 private int mLastMode = MODE_UNKNOWN; 102 private int mLastSortOrder = SORT_ORDER_UNKNOWN; 103 private boolean mLastShowSize = false; 104 105 private boolean mHideGridTitles = false; 106 107 private Point mThumbSize; 108 109 private DocumentsAdapter mAdapter; 110 private LoaderCallbacks<DirectoryResult> mCallbacks; 111 112 private static final String EXTRA_TYPE = "type"; 113 private static final String EXTRA_ROOT = "root"; 114 private static final String EXTRA_DOC = "doc"; 115 private static final String EXTRA_QUERY = "query"; 116 117 private static AtomicInteger sLoaderId = new AtomicInteger(4000); 118 119 private final int mLoaderId = sLoaderId.incrementAndGet(); 120 121 public static void showNormal(FragmentManager fm, RootInfo root, DocumentInfo doc) { 122 show(fm, TYPE_NORMAL, root, doc, null); 123 } 124 125 public static void showSearch(FragmentManager fm, RootInfo root, String query) { 126 show(fm, TYPE_SEARCH, root, null, query); 127 } 128 129 public static void showRecentsOpen(FragmentManager fm) { 130 show(fm, TYPE_RECENT_OPEN, null, null, null); 131 } 132 133 private static void show( 134 FragmentManager fm, int type, RootInfo root, DocumentInfo doc, String query) { 135 final Bundle args = new Bundle(); 136 args.putInt(EXTRA_TYPE, type); 137 args.putParcelable(EXTRA_ROOT, root); 138 args.putParcelable(EXTRA_DOC, doc); 139 args.putString(EXTRA_QUERY, query); 140 141 final DirectoryFragment fragment = new DirectoryFragment(); 142 fragment.setArguments(args); 143 144 final FragmentTransaction ft = fm.beginTransaction(); 145 ft.replace(R.id.container_directory, fragment); 146 ft.commitAllowingStateLoss(); 147 } 148 149 public static DirectoryFragment get(FragmentManager fm) { 150 // TODO: deal with multiple directories shown at once 151 return (DirectoryFragment) fm.findFragmentById(R.id.container_directory); 152 } 153 154 @Override 155 public View onCreateView( 156 LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { 157 final Context context = inflater.getContext(); 158 final View view = inflater.inflate(R.layout.fragment_directory, container, false); 159 160 mEmptyView = view.findViewById(android.R.id.empty); 161 162 mListView = (ListView) view.findViewById(R.id.list); 163 mListView.setOnItemClickListener(mItemListener); 164 mListView.setMultiChoiceModeListener(mMultiListener); 165 mListView.setRecyclerListener(mRecycleListener); 166 167 mGridView = (GridView) view.findViewById(R.id.grid); 168 mGridView.setOnItemClickListener(mItemListener); 169 mGridView.setMultiChoiceModeListener(mMultiListener); 170 mGridView.setRecyclerListener(mRecycleListener); 171 172 return view; 173 } 174 175 @Override 176 public void onActivityCreated(Bundle savedInstanceState) { 177 super.onActivityCreated(savedInstanceState); 178 179 final Context context = getActivity(); 180 final State state = getDisplayState(DirectoryFragment.this); 181 182 final RootInfo root = getArguments().getParcelable(EXTRA_ROOT); 183 final DocumentInfo doc = getArguments().getParcelable(EXTRA_DOC); 184 185 mAdapter = new DocumentsAdapter(); 186 mType = getArguments().getInt(EXTRA_TYPE); 187 188 if (mType == TYPE_RECENT_OPEN) { 189 // Hide titles when showing recents for picking images/videos 190 mHideGridTitles = MimePredicate.mimeMatches( 191 MimePredicate.VISUAL_MIMES, state.acceptMimes); 192 } else { 193 mHideGridTitles = (doc != null) && doc.isGridTitlesHidden(); 194 } 195 196 mCallbacks = new LoaderCallbacks<DirectoryResult>() { 197 @Override 198 public Loader<DirectoryResult> onCreateLoader(int id, Bundle args) { 199 final String query = getArguments().getString(EXTRA_QUERY); 200 201 Uri contentsUri; 202 switch (mType) { 203 case TYPE_NORMAL: 204 contentsUri = DocumentsContract.buildChildDocumentsUri( 205 doc.authority, doc.documentId); 206 if (state.action == ACTION_MANAGE) { 207 contentsUri = DocumentsContract.setManageMode(contentsUri); 208 } 209 return new DirectoryLoader( 210 context, mType, root, doc, contentsUri, state.userSortOrder); 211 case TYPE_SEARCH: 212 contentsUri = DocumentsContract.buildSearchDocumentsUri( 213 root.authority, root.rootId, query); 214 if (state.action == ACTION_MANAGE) { 215 contentsUri = DocumentsContract.setManageMode(contentsUri); 216 } 217 return new DirectoryLoader( 218 context, mType, root, doc, contentsUri, state.userSortOrder); 219 case TYPE_RECENT_OPEN: 220 final RootsCache roots = DocumentsApplication.getRootsCache(context); 221 final List<RootInfo> matchingRoots = roots.getMatchingRoots(state); 222 return new RecentLoader(context, matchingRoots, state.acceptMimes); 223 default: 224 throw new IllegalStateException("Unknown type " + mType); 225 } 226 } 227 228 @Override 229 public void onLoadFinished(Loader<DirectoryResult> loader, DirectoryResult result) { 230 if (!isAdded()) return; 231 232 mAdapter.swapCursor(result.cursor); 233 234 // Push latest state up to UI 235 // TODO: if mode change was racing with us, don't overwrite it 236 if (result.mode != MODE_UNKNOWN) { 237 state.derivedMode = result.mode; 238 } 239 state.derivedSortOrder = result.sortOrder; 240 ((DocumentsActivity) context).onStateChanged(); 241 242 updateDisplayState(); 243 244 if (mLastSortOrder != state.derivedSortOrder) { 245 mLastSortOrder = state.derivedSortOrder; 246 mListView.smoothScrollToPosition(0); 247 mGridView.smoothScrollToPosition(0); 248 } 249 } 250 251 @Override 252 public void onLoaderReset(Loader<DirectoryResult> loader) { 253 mAdapter.swapCursor(null); 254 } 255 }; 256 257 // Kick off loader at least once 258 getLoaderManager().restartLoader(mLoaderId, null, mCallbacks); 259 260 updateDisplayState(); 261 } 262 263 @Override 264 public void onResume() { 265 super.onResume(); 266 updateDisplayState(); 267 } 268 269 public void onUserSortOrderChanged() { 270 // Sort order change always triggers reload; we'll trigger state change 271 // on the flip side. 272 getLoaderManager().restartLoader(mLoaderId, null, mCallbacks); 273 } 274 275 public void onUserModeChanged() { 276 final ContentResolver resolver = getActivity().getContentResolver(); 277 final State state = getDisplayState(this); 278 279 final RootInfo root = getArguments().getParcelable(EXTRA_ROOT); 280 final DocumentInfo doc = getArguments().getParcelable(EXTRA_DOC); 281 282 if (root != null && doc != null) { 283 final Uri stateUri = RecentsProvider.buildState( 284 root.authority, root.rootId, doc.documentId); 285 final ContentValues values = new ContentValues(); 286 values.put(StateColumns.MODE, state.userMode); 287 288 new AsyncTask<Void, Void, Void>() { 289 @Override 290 protected Void doInBackground(Void... params) { 291 resolver.insert(stateUri, values); 292 return null; 293 } 294 }.execute(); 295 } 296 297 // Mode change is just visual change; no need to kick loader, and 298 // deliver change event immediately. 299 state.derivedMode = state.userMode; 300 ((DocumentsActivity) getActivity()).onStateChanged(); 301 302 updateDisplayState(); 303 } 304 305 private void updateDisplayState() { 306 final State state = getDisplayState(this); 307 308 mFilter = new MimePredicate(state.acceptMimes); 309 310 if (mLastMode == state.derivedMode && mLastShowSize == state.showSize) return; 311 mLastMode = state.derivedMode; 312 mLastShowSize = state.showSize; 313 314 mListView.setVisibility(state.derivedMode == MODE_LIST ? View.VISIBLE : View.GONE); 315 mGridView.setVisibility(state.derivedMode == MODE_GRID ? View.VISIBLE : View.GONE); 316 317 final int choiceMode; 318 if (state.allowMultiple) { 319 choiceMode = ListView.CHOICE_MODE_MULTIPLE_MODAL; 320 } else { 321 choiceMode = ListView.CHOICE_MODE_NONE; 322 } 323 324 final int thumbSize; 325 if (state.derivedMode == MODE_GRID) { 326 thumbSize = getResources().getDimensionPixelSize(R.dimen.grid_width); 327 mListView.setAdapter(null); 328 mListView.setChoiceMode(ListView.CHOICE_MODE_NONE); 329 mGridView.setAdapter(mAdapter); 330 mGridView.setColumnWidth(getResources().getDimensionPixelSize(R.dimen.grid_width)); 331 mGridView.setNumColumns(GridView.AUTO_FIT); 332 mGridView.setChoiceMode(choiceMode); 333 mCurrentView = mGridView; 334 } else if (state.derivedMode == MODE_LIST) { 335 thumbSize = getResources().getDimensionPixelSize(R.dimen.icon_size); 336 mGridView.setAdapter(null); 337 mGridView.setChoiceMode(ListView.CHOICE_MODE_NONE); 338 mListView.setAdapter(mAdapter); 339 mListView.setChoiceMode(choiceMode); 340 mCurrentView = mListView; 341 } else { 342 throw new IllegalStateException("Unknown state " + state.derivedMode); 343 } 344 345 mThumbSize = new Point(thumbSize, thumbSize); 346 } 347 348 private OnItemClickListener mItemListener = new OnItemClickListener() { 349 @Override 350 public void onItemClick(AdapterView<?> parent, View view, int position, long id) { 351 final Cursor cursor = mAdapter.getItem(position); 352 if (cursor != null) { 353 final DocumentInfo doc = DocumentInfo.fromDirectoryCursor(cursor); 354 if (mFilter.apply(doc)) { 355 ((DocumentsActivity) getActivity()).onDocumentPicked(doc); 356 } 357 } 358 } 359 }; 360 361 private MultiChoiceModeListener mMultiListener = new MultiChoiceModeListener() { 362 @Override 363 public boolean onCreateActionMode(ActionMode mode, Menu menu) { 364 mode.getMenuInflater().inflate(R.menu.mode_directory, menu); 365 return true; 366 } 367 368 @Override 369 public boolean onPrepareActionMode(ActionMode mode, Menu menu) { 370 final State state = getDisplayState(DirectoryFragment.this); 371 372 final MenuItem open = menu.findItem(R.id.menu_open); 373 final MenuItem share = menu.findItem(R.id.menu_share); 374 final MenuItem delete = menu.findItem(R.id.menu_delete); 375 376 final boolean manageMode = state.action == ACTION_MANAGE; 377 open.setVisible(!manageMode); 378 share.setVisible(manageMode); 379 delete.setVisible(manageMode); 380 381 return true; 382 } 383 384 @Override 385 public boolean onActionItemClicked(ActionMode mode, MenuItem item) { 386 final SparseBooleanArray checked = mCurrentView.getCheckedItemPositions(); 387 final ArrayList<DocumentInfo> docs = Lists.newArrayList(); 388 final int size = checked.size(); 389 for (int i = 0; i < size; i++) { 390 if (checked.valueAt(i)) { 391 final Cursor cursor = mAdapter.getItem(checked.keyAt(i)); 392 final DocumentInfo doc = DocumentInfo.fromDirectoryCursor(cursor); 393 docs.add(doc); 394 } 395 } 396 397 final int id = item.getItemId(); 398 if (id == R.id.menu_open) { 399 DocumentsActivity.get(DirectoryFragment.this).onDocumentsPicked(docs); 400 mode.finish(); 401 return true; 402 403 } else if (id == R.id.menu_share) { 404 onShareDocuments(docs); 405 mode.finish(); 406 return true; 407 408 } else if (id == R.id.menu_delete) { 409 onDeleteDocuments(docs); 410 mode.finish(); 411 return true; 412 413 } else { 414 return false; 415 } 416 } 417 418 @Override 419 public void onDestroyActionMode(ActionMode mode) { 420 // ignored 421 } 422 423 @Override 424 public void onItemCheckedStateChanged( 425 ActionMode mode, int position, long id, boolean checked) { 426 if (checked) { 427 // Directories and footer items cannot be checked 428 boolean valid = false; 429 430 final Cursor cursor = mAdapter.getItem(position); 431 if (cursor != null) { 432 final String docMimeType = getCursorString(cursor, Document.COLUMN_MIME_TYPE); 433 434 // Only valid if non-directory matches filter 435 final State state = getDisplayState(DirectoryFragment.this); 436 valid = !Document.MIME_TYPE_DIR.equals(docMimeType) 437 && MimePredicate.mimeMatches(state.acceptMimes, docMimeType); 438 } 439 440 if (!valid) { 441 mCurrentView.setItemChecked(position, false); 442 } 443 } 444 445 mode.setTitle(getResources() 446 .getString(R.string.mode_selected_count, mCurrentView.getCheckedItemCount())); 447 } 448 }; 449 450 private RecyclerListener mRecycleListener = new RecyclerListener() { 451 @Override 452 public void onMovedToScrapHeap(View view) { 453 final ImageView iconThumb = (ImageView) view.findViewById(R.id.icon_thumb); 454 if (iconThumb != null) { 455 final ThumbnailAsyncTask oldTask = (ThumbnailAsyncTask) iconThumb.getTag(); 456 if (oldTask != null) { 457 oldTask.reallyCancel(); 458 iconThumb.setTag(null); 459 } 460 } 461 } 462 }; 463 464 private void onShareDocuments(List<DocumentInfo> docs) { 465 Intent intent; 466 if (docs.size() == 1) { 467 final DocumentInfo doc = docs.get(0); 468 469 intent = new Intent(Intent.ACTION_SEND); 470 intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); 471 intent.addCategory(Intent.CATEGORY_DEFAULT); 472 intent.setType(doc.mimeType); 473 intent.putExtra(Intent.EXTRA_STREAM, doc.derivedUri); 474 475 } else if (docs.size() > 1) { 476 intent = new Intent(Intent.ACTION_SEND_MULTIPLE); 477 intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); 478 intent.addCategory(Intent.CATEGORY_DEFAULT); 479 480 final ArrayList<String> mimeTypes = Lists.newArrayList(); 481 final ArrayList<Uri> uris = Lists.newArrayList(); 482 for (DocumentInfo doc : docs) { 483 mimeTypes.add(doc.mimeType); 484 uris.add(doc.derivedUri); 485 } 486 487 intent.setType(findCommonMimeType(mimeTypes)); 488 intent.putParcelableArrayListExtra(Intent.EXTRA_STREAM, uris); 489 490 } else { 491 return; 492 } 493 494 intent = Intent.createChooser(intent, getActivity().getText(R.string.share_via)); 495 startActivity(intent); 496 } 497 498 private void onDeleteDocuments(List<DocumentInfo> docs) { 499 final Context context = getActivity(); 500 final ContentResolver resolver = context.getContentResolver(); 501 502 boolean hadTrouble = false; 503 for (DocumentInfo doc : docs) { 504 if (!doc.isDeleteSupported()) { 505 Log.w(TAG, "Skipping " + doc); 506 hadTrouble = true; 507 continue; 508 } 509 510 if (!DocumentsContract.deleteDocument(resolver, doc.derivedUri)) { 511 Log.w(TAG, "Failed to delete " + doc); 512 hadTrouble = true; 513 } 514 } 515 516 if (hadTrouble) { 517 Toast.makeText(context, R.string.toast_failed_delete, Toast.LENGTH_SHORT).show(); 518 } 519 } 520 521 private static State getDisplayState(Fragment fragment) { 522 return ((DocumentsActivity) fragment.getActivity()).getDisplayState(); 523 } 524 525 private static abstract class Footer { 526 private final int mItemViewType; 527 528 public Footer(int itemViewType) { 529 mItemViewType = itemViewType; 530 } 531 532 public abstract View getView(View convertView, ViewGroup parent); 533 534 public int getItemViewType() { 535 return mItemViewType; 536 } 537 } 538 539 private class LoadingFooter extends Footer { 540 public LoadingFooter() { 541 super(1); 542 } 543 544 @Override 545 public View getView(View convertView, ViewGroup parent) { 546 final Context context = parent.getContext(); 547 final State state = getDisplayState(DirectoryFragment.this); 548 549 if (convertView == null) { 550 final LayoutInflater inflater = LayoutInflater.from(context); 551 if (state.derivedMode == MODE_LIST) { 552 convertView = inflater.inflate(R.layout.item_loading_list, parent, false); 553 } else if (state.derivedMode == MODE_GRID) { 554 convertView = inflater.inflate(R.layout.item_loading_grid, parent, false); 555 } else { 556 throw new IllegalStateException(); 557 } 558 } 559 560 return convertView; 561 } 562 } 563 564 private class MessageFooter extends Footer { 565 private final int mIcon; 566 private final String mMessage; 567 568 public MessageFooter(int itemViewType, int icon, String message) { 569 super(itemViewType); 570 mIcon = icon; 571 mMessage = message; 572 } 573 574 @Override 575 public View getView(View convertView, ViewGroup parent) { 576 final Context context = parent.getContext(); 577 final State state = getDisplayState(DirectoryFragment.this); 578 579 if (convertView == null) { 580 final LayoutInflater inflater = LayoutInflater.from(context); 581 if (state.derivedMode == MODE_LIST) { 582 convertView = inflater.inflate(R.layout.item_message_list, parent, false); 583 } else if (state.derivedMode == MODE_GRID) { 584 convertView = inflater.inflate(R.layout.item_message_grid, parent, false); 585 } else { 586 throw new IllegalStateException(); 587 } 588 } 589 590 final ImageView icon = (ImageView) convertView.findViewById(android.R.id.icon); 591 final TextView title = (TextView) convertView.findViewById(android.R.id.title); 592 icon.setImageResource(mIcon); 593 title.setText(mMessage); 594 return convertView; 595 } 596 } 597 598 private class DocumentsAdapter extends BaseAdapter { 599 private Cursor mCursor; 600 private int mCursorCount; 601 602 private List<Footer> mFooters = Lists.newArrayList(); 603 604 public void swapCursor(Cursor cursor) { 605 mCursor = cursor; 606 mCursorCount = cursor != null ? cursor.getCount() : 0; 607 608 mFooters.clear(); 609 610 final Bundle extras = cursor != null ? cursor.getExtras() : null; 611 if (extras != null) { 612 final String info = extras.getString(DocumentsContract.EXTRA_INFO); 613 if (info != null) { 614 mFooters.add(new MessageFooter(2, R.drawable.ic_dialog_alert, info)); 615 } 616 final String error = extras.getString(DocumentsContract.EXTRA_ERROR); 617 if (error != null) { 618 mFooters.add(new MessageFooter(3, R.drawable.ic_dialog_alert, error)); 619 } 620 if (extras.getBoolean(DocumentsContract.EXTRA_LOADING, false)) { 621 mFooters.add(new LoadingFooter()); 622 } 623 } 624 625 if (isEmpty()) { 626 mEmptyView.setVisibility(View.VISIBLE); 627 } else { 628 mEmptyView.setVisibility(View.GONE); 629 } 630 631 notifyDataSetChanged(); 632 } 633 634 @Override 635 public View getView(int position, View convertView, ViewGroup parent) { 636 if (position < mCursorCount) { 637 return getDocumentView(position, convertView, parent); 638 } else { 639 position -= mCursorCount; 640 convertView = mFooters.get(position).getView(convertView, parent); 641 // Only the view itself is disabled; contents inside shouldn't 642 // be dimmed. 643 convertView.setEnabled(false); 644 return convertView; 645 } 646 } 647 648 private View getDocumentView(int position, View convertView, ViewGroup parent) { 649 final Context context = parent.getContext(); 650 final State state = getDisplayState(DirectoryFragment.this); 651 652 final DocumentInfo doc = getArguments().getParcelable(EXTRA_DOC); 653 654 final RootsCache roots = DocumentsApplication.getRootsCache(context); 655 final ThumbnailCache thumbs = DocumentsApplication.getThumbnailsCache( 656 context, mThumbSize); 657 658 if (convertView == null) { 659 final LayoutInflater inflater = LayoutInflater.from(context); 660 if (state.derivedMode == MODE_LIST) { 661 convertView = inflater.inflate(R.layout.item_doc_list, parent, false); 662 } else if (state.derivedMode == MODE_GRID) { 663 convertView = inflater.inflate(R.layout.item_doc_grid, parent, false); 664 } else { 665 throw new IllegalStateException(); 666 } 667 } 668 669 final Cursor cursor = getItem(position); 670 671 final String docAuthority = getCursorString(cursor, RootCursorWrapper.COLUMN_AUTHORITY); 672 final String docRootId = getCursorString(cursor, RootCursorWrapper.COLUMN_ROOT_ID); 673 final String docId = getCursorString(cursor, Document.COLUMN_DOCUMENT_ID); 674 final String docMimeType = getCursorString(cursor, Document.COLUMN_MIME_TYPE); 675 final String docDisplayName = getCursorString(cursor, Document.COLUMN_DISPLAY_NAME); 676 final long docLastModified = getCursorLong(cursor, Document.COLUMN_LAST_MODIFIED); 677 final int docIcon = getCursorInt(cursor, Document.COLUMN_ICON); 678 final int docFlags = getCursorInt(cursor, Document.COLUMN_FLAGS); 679 final String docSummary = getCursorString(cursor, Document.COLUMN_SUMMARY); 680 final long docSize = getCursorLong(cursor, Document.COLUMN_SIZE); 681 682 final View line1 = convertView.findViewById(R.id.line1); 683 final View line2 = convertView.findViewById(R.id.line2); 684 685 final View icon = convertView.findViewById(android.R.id.icon); 686 final ImageView iconMime = (ImageView) convertView.findViewById(R.id.icon_mime); 687 final ImageView iconThumb = (ImageView) convertView.findViewById(R.id.icon_thumb); 688 final TextView title = (TextView) convertView.findViewById(android.R.id.title); 689 final ImageView icon1 = (ImageView) convertView.findViewById(android.R.id.icon1); 690 final ImageView icon2 = (ImageView) convertView.findViewById(android.R.id.icon2); 691 final TextView summary = (TextView) convertView.findViewById(android.R.id.summary); 692 final TextView date = (TextView) convertView.findViewById(R.id.date); 693 final TextView size = (TextView) convertView.findViewById(R.id.size); 694 695 final ThumbnailAsyncTask oldTask = (ThumbnailAsyncTask) iconThumb.getTag(); 696 if (oldTask != null) { 697 oldTask.reallyCancel(); 698 iconThumb.setTag(null); 699 } 700 701 iconMime.animate().cancel(); 702 iconThumb.animate().cancel(); 703 704 final boolean supportsThumbnail = (docFlags & Document.FLAG_SUPPORTS_THUMBNAIL) != 0; 705 final boolean allowThumbnail = (state.derivedMode == MODE_GRID) 706 || MimePredicate.mimeMatches(MimePredicate.VISUAL_MIMES, docMimeType); 707 final boolean showThumbnail = supportsThumbnail && allowThumbnail; 708 709 boolean cacheHit = false; 710 if (showThumbnail) { 711 final Uri uri = DocumentsContract.buildDocumentUri(docAuthority, docId); 712 final Bitmap cachedResult = thumbs.get(uri); 713 if (cachedResult != null) { 714 iconThumb.setImageBitmap(cachedResult); 715 cacheHit = true; 716 } else { 717 iconThumb.setImageDrawable(null); 718 final ThumbnailAsyncTask task = new ThumbnailAsyncTask( 719 uri, iconMime, iconThumb, mThumbSize); 720 iconThumb.setTag(task); 721 task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); 722 } 723 } 724 725 // Always throw MIME icon into place, even when a thumbnail is being 726 // loaded in background. 727 if (cacheHit) { 728 iconMime.setAlpha(0f); 729 iconThumb.setAlpha(1f); 730 } else { 731 iconMime.setAlpha(1f); 732 iconThumb.setAlpha(0f); 733 if (docIcon != 0) { 734 iconMime.setImageDrawable( 735 IconUtils.loadPackageIcon(context, docAuthority, docIcon)); 736 } else { 737 iconMime.setImageDrawable(IconUtils.loadMimeIcon(context, docMimeType)); 738 } 739 } 740 741 boolean hasLine1 = false; 742 boolean hasLine2 = false; 743 744 final boolean hideTitle = (state.derivedMode == MODE_GRID) && mHideGridTitles; 745 if (!hideTitle) { 746 title.setText(docDisplayName); 747 hasLine1 = true; 748 } 749 750 Drawable iconDrawable = null; 751 if (mType == TYPE_RECENT_OPEN) { 752 final RootInfo root = roots.getRoot(docAuthority, docRootId); 753 iconDrawable = root.loadIcon(context); 754 755 if (summary != null) { 756 final boolean alwaysShowSummary = getResources() 757 .getBoolean(R.bool.always_show_summary); 758 if (alwaysShowSummary) { 759 summary.setText(root.getDirectoryString()); 760 summary.setVisibility(View.VISIBLE); 761 hasLine2 = true; 762 } else { 763 if (iconDrawable != null && roots.isIconUnique(root)) { 764 // No summary needed if icon speaks for itself 765 summary.setVisibility(View.INVISIBLE); 766 } else { 767 summary.setText(root.getDirectoryString()); 768 summary.setVisibility(View.VISIBLE); 769 summary.setTextAlignment(TextView.TEXT_ALIGNMENT_TEXT_END); 770 hasLine2 = true; 771 } 772 } 773 } 774 } else { 775 // Directories showing thumbnails in grid mode get a little icon 776 // hint to remind user they're a directory. 777 if (Document.MIME_TYPE_DIR.equals(docMimeType) && state.derivedMode == MODE_GRID 778 && showThumbnail) { 779 iconDrawable = context.getResources().getDrawable(R.drawable.ic_root_folder); 780 } 781 782 if (summary != null) { 783 if (docSummary != null) { 784 summary.setText(docSummary); 785 summary.setVisibility(View.VISIBLE); 786 hasLine2 = true; 787 } else { 788 summary.setVisibility(View.INVISIBLE); 789 } 790 } 791 } 792 793 if (icon1 != null) icon1.setVisibility(View.GONE); 794 if (icon2 != null) icon2.setVisibility(View.GONE); 795 796 if (iconDrawable != null) { 797 if (hasLine1) { 798 icon1.setVisibility(View.VISIBLE); 799 icon1.setImageDrawable(iconDrawable); 800 } else { 801 icon2.setVisibility(View.VISIBLE); 802 icon2.setImageDrawable(iconDrawable); 803 } 804 } 805 806 if (docLastModified == -1) { 807 date.setText(null); 808 } else { 809 date.setText(formatTime(context, docLastModified)); 810 hasLine2 = true; 811 } 812 813 if (state.showSize) { 814 size.setVisibility(View.VISIBLE); 815 if (Document.MIME_TYPE_DIR.equals(docMimeType) || docSize == -1) { 816 size.setText(null); 817 } else { 818 size.setText(Formatter.formatFileSize(context, docSize)); 819 hasLine2 = true; 820 } 821 } else { 822 size.setVisibility(View.GONE); 823 } 824 825 if (line1 != null) { 826 line1.setVisibility(hasLine1 ? View.VISIBLE : View.GONE); 827 } 828 if (line2 != null) { 829 line2.setVisibility(hasLine2 ? View.VISIBLE : View.GONE); 830 } 831 832 final boolean enabled = Document.MIME_TYPE_DIR.equals(docMimeType) 833 || MimePredicate.mimeMatches(state.acceptMimes, docMimeType); 834 if (enabled) { 835 setEnabledRecursive(convertView, true); 836 icon.setAlpha(1f); 837 if (icon1 != null) icon1.setAlpha(1f); 838 if (icon2 != null) icon2.setAlpha(1f); 839 } else { 840 setEnabledRecursive(convertView, false); 841 icon.setAlpha(0.5f); 842 if (icon1 != null) icon1.setAlpha(0.5f); 843 if (icon2 != null) icon2.setAlpha(0.5f); 844 } 845 846 return convertView; 847 } 848 849 @Override 850 public int getCount() { 851 return mCursorCount + mFooters.size(); 852 } 853 854 @Override 855 public Cursor getItem(int position) { 856 if (position < mCursorCount) { 857 mCursor.moveToPosition(position); 858 return mCursor; 859 } else { 860 return null; 861 } 862 } 863 864 @Override 865 public long getItemId(int position) { 866 return position; 867 } 868 869 @Override 870 public int getViewTypeCount() { 871 return 4; 872 } 873 874 @Override 875 public int getItemViewType(int position) { 876 if (position < mCursorCount) { 877 return 0; 878 } else { 879 position -= mCursorCount; 880 return mFooters.get(position).getItemViewType(); 881 } 882 } 883 } 884 885 private static class ThumbnailAsyncTask extends AsyncTask<Uri, Void, Bitmap> { 886 private final Uri mUri; 887 private final ImageView mIconMime; 888 private final ImageView mIconThumb; 889 private final Point mThumbSize; 890 private final CancellationSignal mSignal; 891 892 public ThumbnailAsyncTask( 893 Uri uri, ImageView iconMime, ImageView iconThumb, Point thumbSize) { 894 mUri = uri; 895 mIconMime = iconMime; 896 mIconThumb = iconThumb; 897 mThumbSize = thumbSize; 898 mSignal = new CancellationSignal(); 899 } 900 901 public void reallyCancel() { 902 cancel(false); 903 mSignal.cancel(); 904 } 905 906 @Override 907 protected Bitmap doInBackground(Uri... params) { 908 final Context context = mIconThumb.getContext(); 909 910 Bitmap result = null; 911 try { 912 // TODO: switch to using unstable provider 913 result = DocumentsContract.getDocumentThumbnail( 914 context.getContentResolver(), mUri, mThumbSize, mSignal); 915 if (result != null) { 916 final ThumbnailCache thumbs = DocumentsApplication.getThumbnailsCache( 917 context, mThumbSize); 918 thumbs.put(mUri, result); 919 } 920 } catch (Exception e) { 921 Log.w(TAG, "Failed to load thumbnail: " + e); 922 } 923 return result; 924 } 925 926 @Override 927 protected void onPostExecute(Bitmap result) { 928 if (mIconThumb.getTag() == this && result != null) { 929 mIconThumb.setTag(null); 930 mIconThumb.setImageBitmap(result); 931 932 mIconMime.setAlpha(1f); 933 mIconMime.animate().alpha(0f).start(); 934 mIconThumb.setAlpha(0f); 935 mIconThumb.animate().alpha(1f).start(); 936 } 937 } 938 } 939 940 private static String formatTime(Context context, long when) { 941 // TODO: DateUtils should make this easier 942 Time then = new Time(); 943 then.set(when); 944 Time now = new Time(); 945 now.setToNow(); 946 947 int flags = DateUtils.FORMAT_NO_NOON | DateUtils.FORMAT_NO_MIDNIGHT 948 | DateUtils.FORMAT_ABBREV_ALL; 949 950 if (then.year != now.year) { 951 flags |= DateUtils.FORMAT_SHOW_YEAR | DateUtils.FORMAT_SHOW_DATE; 952 } else if (then.yearDay != now.yearDay) { 953 flags |= DateUtils.FORMAT_SHOW_DATE; 954 } else { 955 flags |= DateUtils.FORMAT_SHOW_TIME; 956 } 957 958 return DateUtils.formatDateTime(context, when, flags); 959 } 960 961 private String findCommonMimeType(List<String> mimeTypes) { 962 String[] commonType = mimeTypes.get(0).split("/"); 963 if (commonType.length != 2) { 964 return "*/*"; 965 } 966 967 for (int i = 1; i < mimeTypes.size(); i++) { 968 String[] type = mimeTypes.get(i).split("/"); 969 if (type.length != 2) continue; 970 971 if (!commonType[1].equals(type[1])) { 972 commonType[1] = "*"; 973 } 974 975 if (!commonType[0].equals(type[0])) { 976 commonType[0] = "*"; 977 commonType[1] = "*"; 978 break; 979 } 980 } 981 982 return commonType[0] + "/" + commonType[1]; 983 } 984 985 private void setEnabledRecursive(View v, boolean enabled) { 986 if (v == null) return; 987 if (v.isEnabled() == enabled) return; 988 v.setEnabled(enabled); 989 990 if (v instanceof ViewGroup) { 991 final ViewGroup vg = (ViewGroup) v; 992 for (int i = vg.getChildCount() - 1; i >= 0; i--) { 993 setEnabledRecursive(vg.getChildAt(i), enabled); 994 } 995 } 996 } 997} 998