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