Merge "Improved OpenExternalDirectoryActivity so it returns right away when an app already has the requested permission." into nyc-dev

am: c657f3d0ba

* commit 'c657f3d0ba1f192c9819ffa79903588b605ef470':
  Improved OpenExternalDirectoryActivity so it returns right away when an app already has the requested permission.
This commit is contained in:
Felipe Leme
2016-02-22 19:39:00 +00:00
committed by android-build-merger
2 changed files with 65 additions and 25 deletions

View File

@@ -1,6 +1,7 @@
<manifest xmlns:android="http://schemas.android.com/apk/res/android" <manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.android.documentsui"> package="com.android.documentsui">
<uses-permission android:name="android.permission.GET_APP_GRANTED_URI_PERMISSIONS" />
<uses-permission android:name="android.permission.MANAGE_DOCUMENTS" /> <uses-permission android:name="android.permission.MANAGE_DOCUMENTS" />
<uses-permission android:name="android.permission.REMOVE_TASKS" /> <uses-permission android:name="android.permission.REMOVE_TASKS" />
<uses-permission android:name="android.permission.WAKE_LOCK" /> <uses-permission android:name="android.permission.WAKE_LOCK" />

View File

@@ -22,6 +22,7 @@ import static android.os.storage.StorageVolume.EXTRA_STORAGE_VOLUME;
import static com.android.documentsui.Shared.DEBUG; import static com.android.documentsui.Shared.DEBUG;
import android.app.Activity; import android.app.Activity;
import android.app.ActivityManager;
import android.app.AlertDialog; import android.app.AlertDialog;
import android.app.Dialog; import android.app.Dialog;
import android.app.DialogFragment; import android.app.DialogFragment;
@@ -33,6 +34,7 @@ import android.content.Context;
import android.content.DialogInterface; import android.content.DialogInterface;
import android.content.DialogInterface.OnClickListener; import android.content.DialogInterface.OnClickListener;
import android.content.Intent; import android.content.Intent;
import android.content.UriPermission;
import android.content.pm.PackageManager; import android.content.pm.PackageManager;
import android.content.pm.PackageManager.NameNotFoundException; import android.content.pm.PackageManager.NameNotFoundException;
import android.net.Uri; import android.net.Uri;
@@ -62,6 +64,8 @@ public class OpenExternalDirectoryActivity extends Activity {
private static final String EXTRA_APP_LABEL = "com.android.documentsui.APP_LABEL"; private static final String EXTRA_APP_LABEL = "com.android.documentsui.APP_LABEL";
private static final String EXTRA_VOLUME_LABEL = "com.android.documentsui.VOLUME_LABEL"; private static final String EXTRA_VOLUME_LABEL = "com.android.documentsui.VOLUME_LABEL";
private ContentProviderClient mExternalStorageClient;
@Override @Override
public void onCreate(Bundle savedInstanceState) { public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState); super.onCreate(savedInstanceState);
@@ -98,12 +102,20 @@ public class OpenExternalDirectoryActivity extends Activity {
} }
} }
@Override
public void onDestroy() {
super.onDestroy();
if (mExternalStorageClient != null) {
mExternalStorageClient.close();
}
}
/** /**
* Validates the given path (volume + directory) and display the appropriate dialog asking the * Validates the given path (volume + directory) and display the appropriate dialog asking the
* user to grant access to it. * user to grant access to it.
*/ */
private static boolean showFragment(Activity activity, int userId, StorageVolume storageVolume, private static boolean showFragment(OpenExternalDirectoryActivity activity, int userId,
String directoryName) { StorageVolume storageVolume, String directoryName) {
if (DEBUG) if (DEBUG)
Log.d(TAG, "showFragment() for volume " + storageVolume.dump() + ", directory " Log.d(TAG, "showFragment() for volume " + storageVolume.dump() + ", directory "
+ directoryName + ", and user " + userId); + directoryName + ", and user " + userId);
@@ -129,7 +141,7 @@ public class OpenExternalDirectoryActivity extends Activity {
return false; return false;
} }
// Gets volume label and converted path // Gets volume label and converted path.
String volumeLabel = null; String volumeLabel = null;
final List<VolumeInfo> volumes = sm.getVolumes(); final List<VolumeInfo> volumes = sm.getVolumes();
if (DEBUG) Log.d(TAG, "Number of volumes: " + volumes.size()); if (DEBUG) Log.d(TAG, "Number of volumes: " + volumes.size());
@@ -143,6 +155,15 @@ public class OpenExternalDirectoryActivity extends Activity {
break; break;
} }
} }
// Checks if the user has granted the permission already.
final Intent intent = getIntentForExistingPermission(activity, file);
if (intent != null) {
activity.setResult(RESULT_OK, intent);
activity.finish();
return true;
}
if (volumeLabel == null) { if (volumeLabel == null) {
Log.e(TAG, "Could not get volume for " + file); Log.e(TAG, "Could not get volume for " + file);
return false; return false;
@@ -196,8 +217,7 @@ public class OpenExternalDirectoryActivity extends Activity {
return volume.isVisibleForWrite(userId) && root.equals(path); return volume.isVisibleForWrite(userId) && root.equals(path);
} }
private static Intent createGrantedUriPermissionsIntent(ContentProviderClient provider, private static Uri getGrantedUriPermission(ContentProviderClient provider, File file) {
File file) {
// Calls ExternalStorageProvider to get the doc id for the file // Calls ExternalStorageProvider to get the doc id for the file
final Bundle bundle; final Bundle bundle;
try { try {
@@ -218,8 +238,17 @@ public class OpenExternalDirectoryActivity extends Activity {
Log.e(TAG, "Could not get URI for doc id " + docId); Log.e(TAG, "Could not get URI for doc id " + docId);
return null; return null;
} }
if (DEBUG) Log.d(TAG, "URI for " + file + ": " + uri); if (DEBUG) Log.d(TAG, "URI for " + file + ": " + uri);
return uri;
}
private static Intent createGrantedUriPermissionsIntent(ContentProviderClient provider,
File file) {
final Uri uri = getGrantedUriPermission(provider, file);
return createGrantedUriPermissionsIntent(uri);
}
private static Intent createGrantedUriPermissionsIntent(Uri uri) {
final Intent intent = new Intent(); final Intent intent = new Intent();
intent.setData(uri); intent.setData(uri);
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION
@@ -229,13 +258,31 @@ public class OpenExternalDirectoryActivity extends Activity {
return intent; return intent;
} }
private static Intent getIntentForExistingPermission(OpenExternalDirectoryActivity activity,
File file) {
final String packageName = activity.getCallingPackage();
final Uri grantedUri = getGrantedUriPermission(activity.getExternalStorageClient(), file);
if (DEBUG)
Log.d(TAG, "checking if " + packageName + " already has permission for " + grantedUri);
final ActivityManager am =
(ActivityManager) activity.getSystemService(Context.ACTIVITY_SERVICE);
for (UriPermission uriPermission : am.getGrantedUriPermissions(packageName).getList()) {
final Uri uri = uriPermission.getUri();
if (uri.equals(grantedUri)) {
if (DEBUG) Log.d(TAG, packageName + " already has permission: " + uriPermission);
return createGrantedUriPermissionsIntent(uri);
}
}
if (DEBUG) Log.d(TAG, packageName + " does not have permission for " + grantedUri);
return null;
}
public static class OpenExternalDirectoryDialogFragment extends DialogFragment { public static class OpenExternalDirectoryDialogFragment extends DialogFragment {
private File mFile; private File mFile;
private String mVolumeLabel; private String mVolumeLabel;
private String mAppLabel; private String mAppLabel;
private ContentProviderClient mExternalStorageClient; private OpenExternalDirectoryActivity mActivity;
private ContentResolver mResolver;
@Override @Override
public void onCreate(Bundle savedInstanceState) { public void onCreate(Bundle savedInstanceState) {
@@ -245,16 +292,8 @@ public class OpenExternalDirectoryActivity extends Activity {
mFile = new File(args.getString(EXTRA_FILE)); mFile = new File(args.getString(EXTRA_FILE));
mVolumeLabel = args.getString(EXTRA_VOLUME_LABEL); mVolumeLabel = args.getString(EXTRA_VOLUME_LABEL);
mAppLabel = args.getString(EXTRA_APP_LABEL); mAppLabel = args.getString(EXTRA_APP_LABEL);
mResolver = getContext().getContentResolver();
}
}
@Override
public void onDestroy() {
super.onDestroy();
if (mExternalStorageClient != null) {
mExternalStorageClient.close();
} }
mActivity = (OpenExternalDirectoryActivity) getActivity();
} }
@Override @Override
@@ -267,8 +306,8 @@ public class OpenExternalDirectoryActivity extends Activity {
public void onClick(DialogInterface dialog, int which) { public void onClick(DialogInterface dialog, int which) {
Intent intent = null; Intent intent = null;
if (which == DialogInterface.BUTTON_POSITIVE) { if (which == DialogInterface.BUTTON_POSITIVE) {
intent = createGrantedUriPermissionsIntent(getExternalStorageClient(), intent = createGrantedUriPermissionsIntent(
mFile); mActivity.getExternalStorageClient(), mFile);
} }
if (which == DialogInterface.BUTTON_NEGATIVE || intent == null) { if (which == DialogInterface.BUTTON_NEGATIVE || intent == null) {
activity.setResult(RESULT_CANCELED); activity.setResult(RESULT_CANCELED);
@@ -297,13 +336,13 @@ public class OpenExternalDirectoryActivity extends Activity {
activity.setResult(RESULT_CANCELED); activity.setResult(RESULT_CANCELED);
activity.finish(); activity.finish();
} }
}
private synchronized ContentProviderClient getExternalStorageClient() { private synchronized ContentProviderClient getExternalStorageClient() {
if (mExternalStorageClient == null) { if (mExternalStorageClient == null) {
mExternalStorageClient = mExternalStorageClient =
mResolver.acquireContentProviderClient(EXTERNAL_STORAGE_AUTH); getContentResolver().acquireContentProviderClient(EXTERNAL_STORAGE_AUTH);
}
return mExternalStorageClient;
} }
return mExternalStorageClient;
} }
} }