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