DirectoryFragment.java revision 3f4c205fd3110345241e690f2a2e7c1b477eac76
1/*
2 * Copyright (C) 2013 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 *      http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package com.android.documentsui;
18
19import static com.android.documentsui.DocumentsActivity.TAG;
20import static com.android.documentsui.DocumentsActivity.State.ACTION_MANAGE;
21import static com.android.documentsui.DocumentsActivity.State.MODE_GRID;
22import static com.android.documentsui.DocumentsActivity.State.MODE_LIST;
23import static com.android.documentsui.DocumentsActivity.State.MODE_UNKNOWN;
24import static com.android.documentsui.DocumentsActivity.State.SORT_ORDER_UNKNOWN;
25import static com.android.documentsui.model.DocumentInfo.getCursorInt;
26import static com.android.documentsui.model.DocumentInfo.getCursorLong;
27import static com.android.documentsui.model.DocumentInfo.getCursorString;
28
29import android.app.Fragment;
30import android.app.FragmentManager;
31import android.app.FragmentTransaction;
32import android.app.LoaderManager.LoaderCallbacks;
33import android.content.ContentResolver;
34import android.content.Context;
35import android.content.Intent;
36import android.content.Loader;
37import android.database.Cursor;
38import android.graphics.Bitmap;
39import android.graphics.Point;
40import android.net.Uri;
41import android.os.AsyncTask;
42import android.os.Bundle;
43import android.provider.DocumentsContract;
44import android.provider.DocumentsContract.Document;
45import android.text.format.DateUtils;
46import android.text.format.Formatter;
47import android.text.format.Time;
48import android.util.Log;
49import android.util.SparseBooleanArray;
50import android.view.ActionMode;
51import android.view.LayoutInflater;
52import android.view.Menu;
53import android.view.MenuItem;
54import android.view.View;
55import android.view.ViewGroup;
56import android.widget.AbsListView;
57import android.widget.AbsListView.MultiChoiceModeListener;
58import android.widget.AdapterView;
59import android.widget.AdapterView.OnItemClickListener;
60import android.widget.BaseAdapter;
61import android.widget.GridView;
62import android.widget.ImageView;
63import android.widget.ListView;
64import android.widget.TextView;
65import android.widget.Toast;
66
67import com.android.documentsui.DocumentsActivity.State;
68import com.android.documentsui.model.DocumentInfo;
69import com.android.documentsui.model.RootInfo;
70import com.android.internal.util.Predicate;
71import com.google.android.collect.Lists;
72
73import java.util.ArrayList;
74import java.util.List;
75import java.util.concurrent.atomic.AtomicInteger;
76
77/**
78 * Display the documents inside a single directory.
79 */
80public class DirectoryFragment extends Fragment {
81
82    private View mEmptyView;
83    private ListView mListView;
84    private GridView mGridView;
85
86    private AbsListView mCurrentView;
87
88    private Predicate<DocumentInfo> mFilter;
89
90    public static final int TYPE_NORMAL = 1;
91    public static final int TYPE_SEARCH = 2;
92    public static final int TYPE_RECENT_OPEN = 3;
93
94    private int mType = TYPE_NORMAL;
95
96    private int mLastMode = MODE_UNKNOWN;
97    private int mLastSortOrder = SORT_ORDER_UNKNOWN;
98
99    private Point mThumbSize;
100
101    private DocumentsAdapter mAdapter;
102    private LoaderCallbacks<DirectoryResult> mCallbacks;
103
104    private static final String EXTRA_TYPE = "type";
105    private static final String EXTRA_ROOT = "root";
106    private static final String EXTRA_DOC = "doc";
107    private static final String EXTRA_QUERY = "query";
108
109    /**
110     * MIME types that should always show thumbnails in list mode.
111     */
112    private static final String[] LIST_THUMBNAIL_MIMES = new String[] { "image/*", "video/*" };
113
114    private static AtomicInteger sLoaderId = new AtomicInteger(4000);
115
116    private final int mLoaderId = sLoaderId.incrementAndGet();
117
118    public static void showNormal(FragmentManager fm, RootInfo root, DocumentInfo doc) {
119        show(fm, TYPE_NORMAL, root, doc, null);
120    }
121
122    public static void showSearch(
123            FragmentManager fm, RootInfo root, DocumentInfo doc, String query) {
124        show(fm, TYPE_SEARCH, root, doc, query);
125    }
126
127    public static void showRecentsOpen(FragmentManager fm) {
128        show(fm, TYPE_RECENT_OPEN, null, null, null);
129    }
130
131    private static void show(
132            FragmentManager fm, int type, RootInfo root, DocumentInfo doc, String query) {
133        final Bundle args = new Bundle();
134        args.putInt(EXTRA_TYPE, type);
135        args.putParcelable(EXTRA_ROOT, root);
136        args.putParcelable(EXTRA_DOC, doc);
137        args.putString(EXTRA_QUERY, query);
138
139        final DirectoryFragment fragment = new DirectoryFragment();
140        fragment.setArguments(args);
141
142        final FragmentTransaction ft = fm.beginTransaction();
143        ft.replace(R.id.container_directory, fragment);
144        ft.commitAllowingStateLoss();
145    }
146
147    public static DirectoryFragment get(FragmentManager fm) {
148        // TODO: deal with multiple directories shown at once
149        return (DirectoryFragment) fm.findFragmentById(R.id.container_directory);
150    }
151
152    @Override
153    public View onCreateView(
154            LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
155        final Context context = inflater.getContext();
156        final View view = inflater.inflate(R.layout.fragment_directory, container, false);
157
158        mEmptyView = view.findViewById(android.R.id.empty);
159
160        mListView = (ListView) view.findViewById(R.id.list);
161        mListView.setOnItemClickListener(mItemListener);
162        mListView.setMultiChoiceModeListener(mMultiListener);
163
164        mGridView = (GridView) view.findViewById(R.id.grid);
165        mGridView.setOnItemClickListener(mItemListener);
166        mGridView.setMultiChoiceModeListener(mMultiListener);
167
168        return view;
169    }
170
171    @Override
172    public void onActivityCreated(Bundle savedInstanceState) {
173        super.onActivityCreated(savedInstanceState);
174
175        final Context context = getActivity();
176        final State state = getDisplayState(DirectoryFragment.this);
177
178        mAdapter = new DocumentsAdapter();
179        mType = getArguments().getInt(EXTRA_TYPE);
180
181        mCallbacks = new LoaderCallbacks<DirectoryResult>() {
182            @Override
183            public Loader<DirectoryResult> onCreateLoader(int id, Bundle args) {
184                final RootInfo root = getArguments().getParcelable(EXTRA_ROOT);
185                final DocumentInfo doc = getArguments().getParcelable(EXTRA_DOC);
186                final String query = getArguments().getString(EXTRA_QUERY);
187
188                Uri contentsUri;
189                switch (mType) {
190                    case TYPE_NORMAL:
191                        contentsUri = DocumentsContract.buildChildDocumentsUri(
192                                doc.authority, doc.documentId);
193                        return new DirectoryLoader(context, root, doc, contentsUri);
194                    case TYPE_SEARCH:
195                        contentsUri = DocumentsContract.buildSearchDocumentsUri(
196                                doc.authority, doc.documentId, query);
197                        return new DirectoryLoader(context, root, doc, contentsUri);
198                    case TYPE_RECENT_OPEN:
199                        final RootsCache roots = DocumentsApplication.getRootsCache(context);
200                        final List<RootInfo> matchingRoots = roots.getMatchingRoots(state);
201                        return new RecentLoader(context, matchingRoots, state.acceptMimes);
202                    default:
203                        throw new IllegalStateException("Unknown type " + mType);
204                }
205            }
206
207            @Override
208            public void onLoadFinished(Loader<DirectoryResult> loader, DirectoryResult result) {
209                if (!isAdded()) return;
210
211                mAdapter.swapCursor(result.cursor);
212
213                // Push latest state up to UI
214                // TODO: if mode change was racing with us, don't overwrite it
215                state.mode = result.mode;
216                state.sortOrder = result.sortOrder;
217                ((DocumentsActivity) context).onStateChanged();
218
219                updateDisplayState();
220
221                if (mLastSortOrder != result.sortOrder) {
222                    mLastSortOrder = result.sortOrder;
223                    mListView.smoothScrollToPosition(0);
224                    mGridView.smoothScrollToPosition(0);
225                }
226            }
227
228            @Override
229            public void onLoaderReset(Loader<DirectoryResult> loader) {
230                mAdapter.swapCursor(null);
231            }
232        };
233
234        // Kick off loader at least once
235        getLoaderManager().restartLoader(mLoaderId, null, mCallbacks);
236
237        updateDisplayState();
238    }
239
240    @Override
241    public void onStart() {
242        super.onStart();
243        updateDisplayState();
244    }
245
246    public void onUserSortOrderChanged() {
247        // User change always triggers reload
248        getLoaderManager().restartLoader(mLoaderId, null, mCallbacks);
249    }
250
251    public void onUserModeChanged() {
252        // Mode change is just display; no need to reload
253        updateDisplayState();
254    }
255
256    private void updateDisplayState() {
257        final State state = getDisplayState(this);
258
259        mFilter = new MimePredicate(state.acceptMimes);
260
261        if (mLastMode == state.mode) return;
262        mLastMode = state.mode;
263
264        mListView.setVisibility(state.mode == MODE_LIST ? View.VISIBLE : View.GONE);
265        mGridView.setVisibility(state.mode == MODE_GRID ? View.VISIBLE : View.GONE);
266
267        final int choiceMode;
268        if (state.allowMultiple) {
269            choiceMode = ListView.CHOICE_MODE_MULTIPLE_MODAL;
270        } else {
271            choiceMode = ListView.CHOICE_MODE_NONE;
272        }
273
274        final int thumbSize;
275        if (state.mode == MODE_GRID) {
276            thumbSize = getResources().getDimensionPixelSize(R.dimen.grid_width);
277            mListView.setAdapter(null);
278            mListView.setChoiceMode(ListView.CHOICE_MODE_NONE);
279            mGridView.setAdapter(mAdapter);
280            mGridView.setColumnWidth(getResources().getDimensionPixelSize(R.dimen.grid_width));
281            mGridView.setNumColumns(GridView.AUTO_FIT);
282            mGridView.setChoiceMode(choiceMode);
283            mCurrentView = mGridView;
284        } else if (state.mode == MODE_LIST) {
285            thumbSize = getResources().getDimensionPixelSize(R.dimen.icon_size);
286            mGridView.setAdapter(null);
287            mGridView.setChoiceMode(ListView.CHOICE_MODE_NONE);
288            mListView.setAdapter(mAdapter);
289            mListView.setChoiceMode(choiceMode);
290            mCurrentView = mListView;
291        } else {
292            throw new IllegalStateException("Unknown state " + state.mode);
293        }
294
295        mThumbSize = new Point(thumbSize, thumbSize);
296    }
297
298    private OnItemClickListener mItemListener = new OnItemClickListener() {
299        @Override
300        public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
301            final Cursor cursor = mAdapter.getItem(position);
302            if (cursor != null) {
303                final DocumentInfo doc = DocumentInfo.fromDirectoryCursor(cursor);
304                if (mFilter.apply(doc)) {
305                    ((DocumentsActivity) getActivity()).onDocumentPicked(doc);
306                }
307            }
308        }
309    };
310
311    private MultiChoiceModeListener mMultiListener = new MultiChoiceModeListener() {
312        @Override
313        public boolean onCreateActionMode(ActionMode mode, Menu menu) {
314            mode.getMenuInflater().inflate(R.menu.mode_directory, menu);
315            return true;
316        }
317
318        @Override
319        public boolean onPrepareActionMode(ActionMode mode, Menu menu) {
320            final State state = getDisplayState(DirectoryFragment.this);
321
322            final MenuItem open = menu.findItem(R.id.menu_open);
323            final MenuItem share = menu.findItem(R.id.menu_share);
324            final MenuItem delete = menu.findItem(R.id.menu_delete);
325
326            final boolean manageMode = state.action == ACTION_MANAGE;
327            open.setVisible(!manageMode);
328            share.setVisible(manageMode);
329            delete.setVisible(manageMode);
330
331            return true;
332        }
333
334        @Override
335        public boolean onActionItemClicked(ActionMode mode, MenuItem item) {
336            final SparseBooleanArray checked = mCurrentView.getCheckedItemPositions();
337            final ArrayList<DocumentInfo> docs = Lists.newArrayList();
338            final int size = checked.size();
339            for (int i = 0; i < size; i++) {
340                if (checked.valueAt(i)) {
341                    final Cursor cursor = mAdapter.getItem(checked.keyAt(i));
342                    final DocumentInfo doc = DocumentInfo.fromDirectoryCursor(cursor);
343                    docs.add(doc);
344                }
345            }
346
347            final int id = item.getItemId();
348            if (id == R.id.menu_open) {
349                DocumentsActivity.get(DirectoryFragment.this).onDocumentsPicked(docs);
350                mode.finish();
351                return true;
352
353            } else if (id == R.id.menu_share) {
354                onShareDocuments(docs);
355                mode.finish();
356                return true;
357
358            } else if (id == R.id.menu_delete) {
359                onDeleteDocuments(docs);
360                mode.finish();
361                return true;
362
363            } else {
364                return false;
365            }
366        }
367
368        @Override
369        public void onDestroyActionMode(ActionMode mode) {
370            // ignored
371        }
372
373        @Override
374        public void onItemCheckedStateChanged(
375                ActionMode mode, int position, long id, boolean checked) {
376            if (checked) {
377                // Directories and footer items cannot be checked
378                boolean valid = false;
379
380                final Cursor cursor = mAdapter.getItem(position);
381                if (cursor != null) {
382                    final String docMimeType = getCursorString(cursor, Document.COLUMN_MIME_TYPE);
383
384                    // Only valid if non-directory matches filter
385                    final State state = getDisplayState(DirectoryFragment.this);
386                    valid = !Document.MIME_TYPE_DIR.equals(docMimeType)
387                            && MimePredicate.mimeMatches(state.acceptMimes, docMimeType);
388                }
389
390                if (!valid) {
391                    mCurrentView.setItemChecked(position, false);
392                }
393            }
394
395            mode.setTitle(getResources()
396                    .getString(R.string.mode_selected_count, mCurrentView.getCheckedItemCount()));
397        }
398    };
399
400    private void onShareDocuments(List<DocumentInfo> docs) {
401        Intent intent;
402        if (docs.size() == 1) {
403            final DocumentInfo doc = docs.get(0);
404
405            intent = new Intent(Intent.ACTION_SEND);
406            intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
407            intent.addCategory(Intent.CATEGORY_DEFAULT);
408            intent.setType(doc.mimeType);
409            intent.putExtra(Intent.EXTRA_STREAM, doc.derivedUri);
410
411        } else if (docs.size() > 1) {
412            intent = new Intent(Intent.ACTION_SEND_MULTIPLE);
413            intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
414            intent.addCategory(Intent.CATEGORY_DEFAULT);
415
416            final ArrayList<String> mimeTypes = Lists.newArrayList();
417            final ArrayList<Uri> uris = Lists.newArrayList();
418            for (DocumentInfo doc : docs) {
419                mimeTypes.add(doc.mimeType);
420                uris.add(doc.derivedUri);
421            }
422
423            intent.setType(findCommonMimeType(mimeTypes));
424            intent.putParcelableArrayListExtra(Intent.EXTRA_STREAM, uris);
425
426        } else {
427            return;
428        }
429
430        intent = Intent.createChooser(intent, getActivity().getText(R.string.share_via));
431        startActivity(intent);
432    }
433
434    private void onDeleteDocuments(List<DocumentInfo> docs) {
435        final Context context = getActivity();
436        final ContentResolver resolver = context.getContentResolver();
437
438        boolean hadTrouble = false;
439        for (DocumentInfo doc : docs) {
440            if (!doc.isDeleteSupported()) {
441                Log.w(TAG, "Skipping " + doc);
442                hadTrouble = true;
443                continue;
444            }
445
446            if (!DocumentsContract.deleteDocument(resolver, doc.derivedUri)) {
447                Log.w(TAG, "Failed to delete " + doc);
448                hadTrouble = true;
449            }
450        }
451
452        if (hadTrouble) {
453            Toast.makeText(context, R.string.toast_failed_delete, Toast.LENGTH_SHORT).show();
454        }
455    }
456
457    private static State getDisplayState(Fragment fragment) {
458        return ((DocumentsActivity) fragment.getActivity()).getDisplayState();
459    }
460
461    private static abstract class Footer {
462        private final int mItemViewType;
463
464        public Footer(int itemViewType) {
465            mItemViewType = itemViewType;
466        }
467
468        public abstract View getView(View convertView, ViewGroup parent);
469
470        public int getItemViewType() {
471            return mItemViewType;
472        }
473    }
474
475    private static class LoadingFooter extends Footer {
476        public LoadingFooter() {
477            super(1);
478        }
479
480        @Override
481        public View getView(View convertView, ViewGroup parent) {
482            final Context context = parent.getContext();
483            if (convertView == null) {
484                final LayoutInflater inflater = LayoutInflater.from(context);
485                convertView = inflater.inflate(R.layout.item_loading, parent, false);
486            }
487            return convertView;
488        }
489    }
490
491    private class MessageFooter extends Footer {
492        private final int mIcon;
493        private final String mMessage;
494
495        public MessageFooter(int itemViewType, int icon, String message) {
496            super(itemViewType);
497            mIcon = icon;
498            mMessage = message;
499        }
500
501        @Override
502        public View getView(View convertView, ViewGroup parent) {
503            final Context context = parent.getContext();
504            final State state = getDisplayState(DirectoryFragment.this);
505
506            if (convertView == null) {
507                final LayoutInflater inflater = LayoutInflater.from(context);
508                if (state.mode == MODE_LIST) {
509                    convertView = inflater.inflate(R.layout.item_message_list, parent, false);
510                } else if (state.mode == MODE_GRID) {
511                    convertView = inflater.inflate(R.layout.item_message_grid, parent, false);
512                } else {
513                    throw new IllegalStateException();
514                }
515            }
516
517            final ImageView icon = (ImageView) convertView.findViewById(android.R.id.icon);
518            final TextView title = (TextView) convertView.findViewById(android.R.id.title);
519            icon.setImageResource(mIcon);
520            title.setText(mMessage);
521            return convertView;
522        }
523    }
524
525    private class DocumentsAdapter extends BaseAdapter {
526        private Cursor mCursor;
527        private int mCursorCount;
528
529        private List<Footer> mFooters = Lists.newArrayList();
530
531        public void swapCursor(Cursor cursor) {
532            mCursor = cursor;
533            mCursorCount = cursor != null ? cursor.getCount() : 0;
534
535            mFooters.clear();
536
537            final Bundle extras = cursor != null ? cursor.getExtras() : null;
538            if (extras != null) {
539                final String info = extras.getString(DocumentsContract.EXTRA_INFO);
540                if (info != null) {
541                    mFooters.add(new MessageFooter(2, R.drawable.ic_dialog_alert, info));
542                }
543                final String error = extras.getString(DocumentsContract.EXTRA_ERROR);
544                if (error != null) {
545                    mFooters.add(new MessageFooter(3, R.drawable.ic_dialog_alert, error));
546                }
547                if (extras.getBoolean(DocumentsContract.EXTRA_LOADING, false)) {
548                    mFooters.add(new LoadingFooter());
549                }
550            }
551
552            if (isEmpty()) {
553                mEmptyView.setVisibility(View.VISIBLE);
554            } else {
555                mEmptyView.setVisibility(View.GONE);
556            }
557
558            notifyDataSetChanged();
559        }
560
561        @Override
562        public View getView(int position, View convertView, ViewGroup parent) {
563            if (position < mCursorCount) {
564                return getDocumentView(position, convertView, parent);
565            } else {
566                position -= mCursorCount;
567                convertView = mFooters.get(position).getView(convertView, parent);
568                // Only the view itself is disabled; contents inside shouldn't
569                // be dimmed.
570                convertView.setEnabled(false);
571                return convertView;
572            }
573        }
574
575        private View getDocumentView(int position, View convertView, ViewGroup parent) {
576            final Context context = parent.getContext();
577            final State state = getDisplayState(DirectoryFragment.this);
578
579            final RootsCache roots = DocumentsApplication.getRootsCache(context);
580            final ThumbnailCache thumbs = DocumentsApplication.getThumbnailsCache(
581                    context, mThumbSize);
582
583            if (convertView == null) {
584                final LayoutInflater inflater = LayoutInflater.from(context);
585                if (state.mode == MODE_LIST) {
586                    convertView = inflater.inflate(R.layout.item_doc_list, parent, false);
587                } else if (state.mode == MODE_GRID) {
588                    convertView = inflater.inflate(R.layout.item_doc_grid, parent, false);
589                } else {
590                    throw new IllegalStateException();
591                }
592            }
593
594            final Cursor cursor = getItem(position);
595
596            final String docAuthority = getCursorString(cursor, RootCursorWrapper.COLUMN_AUTHORITY);
597            final String docRootId = getCursorString(cursor, RootCursorWrapper.COLUMN_ROOT_ID);
598            final String docId = getCursorString(cursor, Document.COLUMN_DOCUMENT_ID);
599            final String docMimeType = getCursorString(cursor, Document.COLUMN_MIME_TYPE);
600            final String docDisplayName = getCursorString(cursor, Document.COLUMN_DISPLAY_NAME);
601            final long docLastModified = getCursorLong(cursor, Document.COLUMN_LAST_MODIFIED);
602            final int docIcon = getCursorInt(cursor, Document.COLUMN_ICON);
603            final int docFlags = getCursorInt(cursor, Document.COLUMN_FLAGS);
604            final String docSummary = getCursorString(cursor, Document.COLUMN_SUMMARY);
605            final long docSize = getCursorLong(cursor, Document.COLUMN_SIZE);
606
607            final ImageView icon = (ImageView) convertView.findViewById(android.R.id.icon);
608            final TextView title = (TextView) convertView.findViewById(android.R.id.title);
609            final View line2 = convertView.findViewById(R.id.line2);
610            final ImageView icon1 = (ImageView) convertView.findViewById(android.R.id.icon1);
611            final TextView summary = (TextView) convertView.findViewById(android.R.id.summary);
612            final TextView date = (TextView) convertView.findViewById(R.id.date);
613            final TextView size = (TextView) convertView.findViewById(R.id.size);
614
615            final ThumbnailAsyncTask oldTask = (ThumbnailAsyncTask) icon.getTag();
616            if (oldTask != null) {
617                oldTask.cancel(false);
618            }
619
620            final boolean supportsThumbnail = (docFlags & Document.FLAG_SUPPORTS_THUMBNAIL) != 0;
621            final boolean allowThumbnail = (state.mode == MODE_GRID)
622                    || MimePredicate.mimeMatches(LIST_THUMBNAIL_MIMES, docMimeType);
623
624            if (supportsThumbnail && allowThumbnail) {
625                final Uri uri = DocumentsContract.buildDocumentUri(docAuthority, docId);
626                final Bitmap cachedResult = thumbs.get(uri);
627                if (cachedResult != null) {
628                    icon.setImageBitmap(cachedResult);
629                } else {
630                    final ThumbnailAsyncTask task = new ThumbnailAsyncTask(icon, mThumbSize);
631                    icon.setImageBitmap(null);
632                    icon.setTag(task);
633                    task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR, uri);
634                }
635            } else if (docIcon != 0) {
636                icon.setImageDrawable(IconUtils.loadPackageIcon(context, docAuthority, docIcon));
637            } else {
638                icon.setImageDrawable(IconUtils.loadMimeIcon(context, docMimeType));
639            }
640
641            title.setText(docDisplayName);
642
643            boolean hasLine2 = false;
644
645            if (mType == TYPE_RECENT_OPEN) {
646                final RootInfo root = roots.getRoot(docAuthority, docRootId);
647                icon1.setVisibility(View.VISIBLE);
648                icon1.setImageDrawable(root.loadIcon(context));
649                summary.setText(root.getDirectoryString());
650                summary.setVisibility(View.VISIBLE);
651                summary.setTextAlignment(TextView.TEXT_ALIGNMENT_TEXT_END);
652                hasLine2 = true;
653            } else {
654                icon1.setVisibility(View.GONE);
655                if (docSummary != null) {
656                    summary.setText(docSummary);
657                    summary.setVisibility(View.VISIBLE);
658                    hasLine2 = true;
659                } else {
660                    summary.setVisibility(View.INVISIBLE);
661                }
662            }
663
664            if (docLastModified == -1) {
665                date.setText(null);
666            } else {
667                date.setText(formatTime(context, docLastModified));
668                hasLine2 = true;
669            }
670
671            if (state.showSize) {
672                size.setVisibility(View.VISIBLE);
673                if (Document.MIME_TYPE_DIR.equals(docMimeType) || docSize == -1) {
674                    size.setText(null);
675                } else {
676                    size.setText(Formatter.formatFileSize(context, docSize));
677                    hasLine2 = true;
678                }
679            } else {
680                size.setVisibility(View.GONE);
681            }
682
683            line2.setVisibility(hasLine2 ? View.VISIBLE : View.GONE);
684
685            final boolean enabled = Document.MIME_TYPE_DIR.equals(docMimeType)
686                    || MimePredicate.mimeMatches(state.acceptMimes, docMimeType);
687            if (enabled) {
688                setEnabledRecursive(convertView, true);
689                icon.setAlpha(1f);
690                icon1.setAlpha(1f);
691            } else {
692                setEnabledRecursive(convertView, false);
693                icon.setAlpha(0.5f);
694                icon1.setAlpha(0.5f);
695            }
696
697            return convertView;
698        }
699
700        @Override
701        public int getCount() {
702            return mCursorCount + mFooters.size();
703        }
704
705        @Override
706        public Cursor getItem(int position) {
707            if (position < mCursorCount) {
708                mCursor.moveToPosition(position);
709                return mCursor;
710            } else {
711                return null;
712            }
713        }
714
715        @Override
716        public long getItemId(int position) {
717            return position;
718        }
719
720        @Override
721        public int getViewTypeCount() {
722            return 4;
723        }
724
725        @Override
726        public int getItemViewType(int position) {
727            if (position < mCursorCount) {
728                return 0;
729            } else {
730                position -= mCursorCount;
731                return mFooters.get(position).getItemViewType();
732            }
733        }
734    }
735
736    private static class ThumbnailAsyncTask extends AsyncTask<Uri, Void, Bitmap> {
737        private final ImageView mTarget;
738        private final Point mThumbSize;
739
740        public ThumbnailAsyncTask(ImageView target, Point thumbSize) {
741            mTarget = target;
742            mThumbSize = thumbSize;
743        }
744
745        @Override
746        protected void onPreExecute() {
747            mTarget.setTag(this);
748        }
749
750        @Override
751        protected Bitmap doInBackground(Uri... params) {
752            final Context context = mTarget.getContext();
753            final Uri uri = params[0];
754
755            Bitmap result = null;
756            try {
757                result = DocumentsContract.getDocumentThumbnail(
758                        context.getContentResolver(), uri, mThumbSize, null);
759                if (result != null) {
760                    final ThumbnailCache thumbs = DocumentsApplication.getThumbnailsCache(
761                            context, mThumbSize);
762                    thumbs.put(uri, result);
763                }
764            } catch (Exception e) {
765                Log.w(TAG, "Failed to load thumbnail: " + e);
766            }
767            return result;
768        }
769
770        @Override
771        protected void onPostExecute(Bitmap result) {
772            if (mTarget.getTag() == this) {
773                mTarget.setImageBitmap(result);
774                mTarget.setTag(null);
775            }
776        }
777    }
778
779    private static String formatTime(Context context, long when) {
780        // TODO: DateUtils should make this easier
781        Time then = new Time();
782        then.set(when);
783        Time now = new Time();
784        now.setToNow();
785
786        int flags = DateUtils.FORMAT_NO_NOON | DateUtils.FORMAT_NO_MIDNIGHT
787                | DateUtils.FORMAT_ABBREV_ALL;
788
789        if (then.year != now.year) {
790            flags |= DateUtils.FORMAT_SHOW_YEAR | DateUtils.FORMAT_SHOW_DATE;
791        } else if (then.yearDay != now.yearDay) {
792            flags |= DateUtils.FORMAT_SHOW_DATE;
793        } else {
794            flags |= DateUtils.FORMAT_SHOW_TIME;
795        }
796
797        return DateUtils.formatDateTime(context, when, flags);
798    }
799
800    private String findCommonMimeType(List<String> mimeTypes) {
801        String[] commonType = mimeTypes.get(0).split("/");
802        if (commonType.length != 2) {
803            return "*/*";
804        }
805
806        for (int i = 1; i < mimeTypes.size(); i++) {
807            String[] type = mimeTypes.get(i).split("/");
808            if (type.length != 2) continue;
809
810            if (!commonType[1].equals(type[1])) {
811                commonType[1] = "*";
812            }
813
814            if (!commonType[0].equals(type[0])) {
815                commonType[0] = "*";
816                commonType[1] = "*";
817                break;
818            }
819        }
820
821        return commonType[0] + "/" + commonType[1];
822    }
823
824    private void setEnabledRecursive(View v, boolean enabled) {
825        if (v.isEnabled() == enabled) return;
826        v.setEnabled(enabled);
827
828        if (v instanceof ViewGroup) {
829            final ViewGroup vg = (ViewGroup) v;
830            for (int i = vg.getChildCount() - 1; i >= 0; i--) {
831                setEnabledRecursive(vg.getChildAt(i), enabled);
832            }
833        }
834    }
835}
836