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