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