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