/* * Copyright (C) 2013 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.android.documentsui.picker; import static com.android.documentsui.base.Shared.DEBUG; import static com.android.documentsui.base.State.ACTION_CREATE; import static com.android.documentsui.base.State.ACTION_GET_CONTENT; import static com.android.documentsui.base.State.ACTION_OPEN; import static com.android.documentsui.base.State.ACTION_OPEN_TREE; import static com.android.documentsui.base.State.ACTION_PICK_COPY_DESTINATION; import android.app.Activity; import android.app.Fragment; import android.app.FragmentManager; import android.content.ClipData; import android.content.ComponentName; import android.content.ContentProviderClient; import android.content.ContentResolver; import android.content.ContentValues; import android.content.Intent; import android.content.pm.ResolveInfo; import android.net.Uri; import android.os.Bundle; import android.os.Parcelable; import android.provider.DocumentsContract; import android.support.design.widget.Snackbar; import android.support.v7.widget.RecyclerView; import android.util.Log; import android.view.Menu; import com.android.documentsui.BaseActivity; import com.android.documentsui.DocumentsApplication; import com.android.documentsui.FocusManager; import com.android.documentsui.MenuManager.DirectoryDetails; import com.android.documentsui.R; import com.android.documentsui.Snackbars; import com.android.documentsui.base.DocumentInfo; import com.android.documentsui.base.MimePredicate; import com.android.documentsui.base.PairedTask; import com.android.documentsui.base.RootInfo; import com.android.documentsui.base.Shared; import com.android.documentsui.base.State; import com.android.documentsui.dirlist.DirectoryFragment; import com.android.documentsui.dirlist.FragmentTuner; import com.android.documentsui.dirlist.Model; import com.android.documentsui.dirlist.MultiSelectManager; import com.android.documentsui.picker.LastAccessedProvider.Columns; import com.android.documentsui.services.FileOperationService; import com.android.documentsui.sidebar.RootsFragment; import java.util.Arrays; import java.util.List; public class PickActivity extends BaseActivity { private static final int CODE_FORWARD = 42; private static final String TAG = "DocumentsActivity"; private Tuner mTuner; private FocusManager mFocusManager; private MenuManager mMenuManager; public PickActivity() { super(R.layout.documents_activity, TAG); } @Override public void onCreate(Bundle icicle) { super.onCreate(icicle); mTuner = new Tuner(this, mState); // Make sure this is done after the RecyclerView and the Model are set up. mFocusManager = new FocusManager(getColor(R.color.accent_dark)); mMenuManager = new MenuManager(mSearchManager, mState, new DirectoryDetails(this)); if (mState.action == ACTION_CREATE) { final String mimeType = getIntent().getType(); final String title = getIntent().getStringExtra(Intent.EXTRA_TITLE); SaveFragment.show(getFragmentManager(), mimeType, title); } else if (mState.action == ACTION_OPEN_TREE || mState.action == ACTION_PICK_COPY_DESTINATION) { PickFragment.show(getFragmentManager()); } if (mState.action == ACTION_GET_CONTENT) { final Intent moreApps = new Intent(getIntent()); moreApps.setComponent(null); moreApps.setPackage(null); RootsFragment.show(getFragmentManager(), moreApps); } else if (mState.action == ACTION_OPEN || mState.action == ACTION_CREATE || mState.action == ACTION_OPEN_TREE || mState.action == ACTION_PICK_COPY_DESTINATION) { RootsFragment.show(getFragmentManager(), (Intent) null); } if (mState.restored) { if (DEBUG) Log.d(TAG, "Stack already resolved"); } else { // We set the activity title in AsyncTask.onPostExecute(). // To prevent talkback from reading aloud the default title, we clear it here. setTitle(""); // As a matter of policy we don't load the last used stack for the copy // destination picker (user is already in Files app). // Concensus was that the experice was too confusing. // In all other cases, where the user is visiting us from another app // we restore the stack as last used from that app. if (mState.action == ACTION_PICK_COPY_DESTINATION) { if (DEBUG) Log.d(TAG, "Launching directly into Home directory."); loadRoot(getDefaultRoot()); } else { if (DEBUG) Log.d(TAG, "Attempting to load last used stack for calling package."); new LoadLastAccessedStackTask(this, mState).execute(); } } } @Override protected void includeState(State state) { final Intent intent = getIntent(); final String action = intent.getAction(); if (Intent.ACTION_OPEN_DOCUMENT.equals(action)) { state.action = ACTION_OPEN; } else if (Intent.ACTION_CREATE_DOCUMENT.equals(action)) { state.action = ACTION_CREATE; } else if (Intent.ACTION_GET_CONTENT.equals(action)) { state.action = ACTION_GET_CONTENT; } else if (Intent.ACTION_OPEN_DOCUMENT_TREE.equals(action)) { state.action = ACTION_OPEN_TREE; } else if (Shared.ACTION_PICK_COPY_DESTINATION.equals(action)) { state.action = ACTION_PICK_COPY_DESTINATION; } if (state.action == ACTION_OPEN || state.action == ACTION_GET_CONTENT) { state.allowMultiple = intent.getBooleanExtra( Intent.EXTRA_ALLOW_MULTIPLE, false); } if (state.action == ACTION_OPEN || state.action == ACTION_GET_CONTENT || state.action == ACTION_CREATE) { state.openableOnly = intent.hasCategory(Intent.CATEGORY_OPENABLE); } if (state.action == ACTION_PICK_COPY_DESTINATION) { // Indicates that a copy operation (or move) includes a directory. // Why? Directory creation isn't supported by some roots (like Downloads). // This allows us to restrict available roots to just those with support. state.directoryCopy = intent.getBooleanExtra( Shared.EXTRA_DIRECTORY_COPY, false); state.copyOperationSubType = intent.getIntExtra( FileOperationService.EXTRA_OPERATION_TYPE, FileOperationService.OPERATION_COPY); } } public void onAppPicked(ResolveInfo info) { final Intent intent = new Intent(getIntent()); intent.setFlags(intent.getFlags() & ~Intent.FLAG_ACTIVITY_FORWARD_RESULT); intent.setComponent(new ComponentName( info.activityInfo.applicationInfo.packageName, info.activityInfo.name)); startActivityForResult(intent, CODE_FORWARD); } @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { if (DEBUG) Log.d(TAG, "onActivityResult() code=" + resultCode); // Only relay back results when not canceled; otherwise stick around to // let the user pick another app/backend. if (requestCode == CODE_FORWARD && resultCode != RESULT_CANCELED) { // Remember that we last picked via external app final String packageName = getCallingPackageMaybeExtra(); final ContentValues values = new ContentValues(); values.put(Columns.EXTERNAL, 1); getContentResolver().insert(LastAccessedProvider.buildLastAccessed(packageName), values); // Pass back result to original caller setResult(resultCode, data); finish(); } else { super.onActivityResult(requestCode, resultCode, data); } } @Override protected void onPostCreate(Bundle savedInstanceState) { super.onPostCreate(savedInstanceState); mDrawer.update(); mNavigator.update(); } @Override public String getDrawerTitle() { String title = getIntent().getStringExtra(DocumentsContract.EXTRA_PROMPT); if (title == null) { if (mState.action == ACTION_OPEN || mState.action == ACTION_GET_CONTENT || mState.action == ACTION_OPEN_TREE) { title = getResources().getString(R.string.title_open); } else if (mState.action == ACTION_CREATE || mState.action == ACTION_PICK_COPY_DESTINATION) { title = getResources().getString(R.string.title_save); } else { // If all else fails, just call it "Documents". title = getResources().getString(R.string.app_label); } } return title; } @Override public boolean onPrepareOptionsMenu(Menu menu) { super.onPrepareOptionsMenu(menu); mMenuManager.updateOptionMenu(menu); final DocumentInfo cwd = getCurrentDirectory(); if (mState.action == ACTION_CREATE) { final FragmentManager fm = getFragmentManager(); SaveFragment.get(fm).prepareForDirectory(cwd); } return true; } @Override protected void refreshDirectory(int anim) { final FragmentManager fm = getFragmentManager(); final RootInfo root = getCurrentRoot(); final DocumentInfo cwd = getCurrentDirectory(); if (cwd == null) { // No directory means recents if (mState.action == ACTION_CREATE || mState.action == ACTION_PICK_COPY_DESTINATION) { loadRoot(getDefaultRoot()); } else { DirectoryFragment.showRecentsOpen(fm, anim); // In recents we pick layout mode based on the mimetype, // picking GRID for visual types. We intentionally don't // consult a user's saved preferences here since they are // set per root (not per root and per mimetype). boolean visualMimes = MimePredicate.mimeMatches( MimePredicate.VISUAL_MIMES, mState.acceptMimes); mState.derivedMode = visualMimes ? State.MODE_GRID : State.MODE_LIST; } } else { // Normal boring directory DirectoryFragment.showDirectory(fm, root, cwd, anim); } // Forget any replacement target if (mState.action == ACTION_CREATE) { final SaveFragment save = SaveFragment.get(fm); if (save != null) { save.setReplaceTarget(null); } } if (mState.action == ACTION_OPEN_TREE || mState.action == ACTION_PICK_COPY_DESTINATION) { final PickFragment pick = PickFragment.get(fm); if (pick != null) { pick.setPickTarget(mState.action, mState.copyOperationSubType, cwd); } } } void onSaveRequested(DocumentInfo replaceTarget) { new ExistingFinishTask(this, replaceTarget.derivedUri) .executeOnExecutor(getExecutorForCurrentDirectory()); } @Override public void setPending(boolean pending) { final SaveFragment save = SaveFragment.get(getFragmentManager()); if (save != null) { save.setPending(pending); } } @Override protected void onDirectoryCreated(DocumentInfo doc) { assert(doc.isDirectory()); openContainerDocument(doc); } void onSaveRequested(String mimeType, String displayName) { new CreateFinishTask(this, mimeType, displayName) .executeOnExecutor(getExecutorForCurrentDirectory()); } @Override public void onDocumentPicked(DocumentInfo doc, Model model) { final FragmentManager fm = getFragmentManager(); if (doc.isContainer()) { openContainerDocument(doc); } else if (mState.action == ACTION_OPEN || mState.action == ACTION_GET_CONTENT) { // Explicit file picked, return new ExistingFinishTask(this, doc.derivedUri) .executeOnExecutor(getExecutorForCurrentDirectory()); } else if (mState.action == ACTION_CREATE) { // Replace selected file SaveFragment.get(fm).setReplaceTarget(doc); } } @Override public void onDocumentsPicked(List docs) { if (mState.action == ACTION_OPEN || mState.action == ACTION_GET_CONTENT) { final int size = docs.size(); final Uri[] uris = new Uri[size]; for (int i = 0; i < size; i++) { uris[i] = docs.get(i).derivedUri; } new ExistingFinishTask(this, uris) .executeOnExecutor(getExecutorForCurrentDirectory()); } } public void onPickRequested(DocumentInfo pickTarget) { Uri result; if (mState.action == ACTION_OPEN_TREE) { result = DocumentsContract.buildTreeDocumentUri( pickTarget.authority, pickTarget.documentId); } else if (mState.action == ACTION_PICK_COPY_DESTINATION) { result = pickTarget.derivedUri; } else { // Should not be reached. throw new IllegalStateException("Invalid mState.action."); } new PickFinishTask(this, result).executeOnExecutor(getExecutorForCurrentDirectory()); } void updateLastAccessed() { LastAccessedProvider.setLastAccessed( getContentResolver(), getCallingPackageMaybeExtra(), mState.stack); } @Override protected void onTaskFinished(Uri... uris) { if (DEBUG) Log.d(TAG, "onFinished() " + Arrays.toString(uris)); final Intent intent = new Intent(); if (uris.length == 1) { intent.setData(uris[0]); } else if (uris.length > 1) { final ClipData clipData = new ClipData( null, mState.acceptMimes, new ClipData.Item(uris[0])); for (int i = 1; i < uris.length; i++) { clipData.addItem(new ClipData.Item(uris[i])); } intent.setClipData(clipData); } if (mState.action == ACTION_GET_CONTENT) { intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); } else if (mState.action == ACTION_OPEN_TREE) { intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION | Intent.FLAG_GRANT_PERSISTABLE_URI_PERMISSION | Intent.FLAG_GRANT_PREFIX_URI_PERMISSION); } else if (mState.action == ACTION_PICK_COPY_DESTINATION) { // Picking a copy destination is only used internally by us, so we // don't need to extend permissions to the caller. intent.putExtra(Shared.EXTRA_STACK, (Parcelable) mState.stack); intent.putExtra(FileOperationService.EXTRA_OPERATION_TYPE, mState.copyOperationSubType); } else { intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION | Intent.FLAG_GRANT_PERSISTABLE_URI_PERMISSION); } setResult(Activity.RESULT_OK, intent); finish(); } public static PickActivity get(Fragment fragment) { return (PickActivity) fragment.getActivity(); } @Override public FragmentTuner getFragmentTuner( Model model, MultiSelectManager selectionMgr, boolean mSearchMode) { return mTuner.reset(model, selectionMgr, mSearchMode); } @Override public FocusManager getFocusManager(RecyclerView view, Model model) { return mFocusManager.reset(view, model); } @Override public MenuManager getMenuManager() { return mMenuManager; } private static final class PickFinishTask extends PairedTask { private final Uri mUri; public PickFinishTask(PickActivity activity, Uri uri) { super(activity); mUri = uri; } @Override protected Void run(Void... params) { mOwner.updateLastAccessed(); return null; } @Override protected void finish(Void result) { mOwner.onTaskFinished(mUri); } } private static final class ExistingFinishTask extends PairedTask { private final Uri[] mUris; public ExistingFinishTask(PickActivity activity, Uri... uris) { super(activity); mUris = uris; } @Override protected Void run(Void... params) { mOwner.updateLastAccessed(); return null; } @Override protected void finish(Void result) { mOwner.onTaskFinished(mUris); } } /** * Task that creates a new document in the background. */ private static final class CreateFinishTask extends PairedTask { private final String mMimeType; private final String mDisplayName; public CreateFinishTask(PickActivity activity, String mimeType, String displayName) { super(activity); mMimeType = mimeType; mDisplayName = displayName; } @Override protected void prepare() { mOwner.setPending(true); } @Override protected Uri run(Void... params) { final ContentResolver resolver = mOwner.getContentResolver(); final DocumentInfo cwd = mOwner.getCurrentDirectory(); ContentProviderClient client = null; Uri childUri = null; try { client = DocumentsApplication.acquireUnstableProviderOrThrow( resolver, cwd.derivedUri.getAuthority()); childUri = DocumentsContract.createDocument( client, cwd.derivedUri, mMimeType, mDisplayName); } catch (Exception e) { Log.w(TAG, "Failed to create document", e); } finally { ContentProviderClient.releaseQuietly(client); } if (childUri != null) { mOwner.updateLastAccessed(); } return childUri; } @Override protected void finish(Uri result) { if (result != null) { mOwner.onTaskFinished(result); } else { Snackbars.makeSnackbar( mOwner, R.string.save_error, Snackbar.LENGTH_SHORT).show(); } mOwner.setPending(false); } } }