Получите реальный путь от URI, новой системы доступа к хранилищу Android KitKat
Перед тем, как новый доступ галереи в Android 4.4 (KitKat), я получил свой реальный путь на SD-карте с помощью этого метода:
public String getPath(Uri uri) {
String[] projection = { MediaStore.Images.Media.DATA };
Cursor cursor = managedQuery(uri, projection, null, null, null);
startManagingCursor(cursor);
int column_index = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA);
cursor.moveToFirst();
return cursor.getString(column_index);
}
Теперь intent.ACTION_GET_CONTENT возвращает разные данные:
До:
content://media/external/images/media/62
Сейчас:
content://com.android.providers.media.documents/document/image:62
Как мне удалось получить реальный путь на SD-карте?
Ответы
Ответ 1
Примечание. Этот ответ затрагивает часть проблемы. Для полного решения (в форме библиотеки) просмотрите ответ Пола Берка.
Вы можете использовать URI для получения document id
, а затем запросить либо MediaStore.Images.Media.EXTERNAL_CONTENT_URI
, либо MediaStore.Images.Media.INTERNAL_CONTENT_URI
(в зависимости от ситуации с SD-картой).
Чтобы получить идентификатор документа:
// Will return "image:x*"
String wholeID = DocumentsContract.getDocumentId(uriThatYouCurrentlyHave);
// Split at colon, use second item in the array
String id = wholeID.split(":")[1];
String[] column = { MediaStore.Images.Media.DATA };
// where id is equal to
String sel = MediaStore.Images.Media._ID + "=?";
Cursor cursor = getContentResolver().
query(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
column, sel, new String[]{ id }, null);
String filePath = "";
int columnIndex = cursor.getColumnIndex(column[0]);
if (cursor.moveToFirst()) {
filePath = cursor.getString(columnIndex);
}
cursor.close();
Ссылка: я не могу найти сообщение, из которого это решение взято. Я хотел попросить оригинальный плакат внести свой вклад. Посмотрите еще немного сегодня.
Ответ 2
Это приведет к пути к файлу из MediaProvider, DownloadsProvider и ExternalStorageProvider, возвращаясь к неофициальному методу ContentProvider, который вы упомянули.
/**
* Get a file path from a Uri. This will get the the path for Storage Access
* Framework Documents, as well as the _data field for the MediaStore and
* other file-based ContentProviders.
*
* @param context The context.
* @param uri The Uri to query.
* @author paulburke
*/
public static String getPath(final Context context, final Uri uri) {
final boolean isKitKat = Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT;
// DocumentProvider
if (isKitKat && DocumentsContract.isDocumentUri(context, uri)) {
// ExternalStorageProvider
if (isExternalStorageDocument(uri)) {
final String docId = DocumentsContract.getDocumentId(uri);
final String[] split = docId.split(":");
final String type = split[0];
if ("primary".equalsIgnoreCase(type)) {
return Environment.getExternalStorageDirectory() + "/" + split[1];
}
// TODO handle non-primary volumes
}
// DownloadsProvider
else if (isDownloadsDocument(uri)) {
final String id = DocumentsContract.getDocumentId(uri);
final Uri contentUri = ContentUris.withAppendedId(
Uri.parse("content://downloads/public_downloads"), Long.valueOf(id));
return getDataColumn(context, contentUri, null, null);
}
// MediaProvider
else if (isMediaDocument(uri)) {
final String docId = DocumentsContract.getDocumentId(uri);
final String[] split = docId.split(":");
final String type = split[0];
Uri contentUri = null;
if ("image".equals(type)) {
contentUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
} else if ("video".equals(type)) {
contentUri = MediaStore.Video.Media.EXTERNAL_CONTENT_URI;
} else if ("audio".equals(type)) {
contentUri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
}
final String selection = "_id=?";
final String[] selectionArgs = new String[] {
split[1]
};
return getDataColumn(context, contentUri, selection, selectionArgs);
}
}
// MediaStore (and general)
else if ("content".equalsIgnoreCase(uri.getScheme())) {
return getDataColumn(context, uri, null, null);
}
// File
else if ("file".equalsIgnoreCase(uri.getScheme())) {
return uri.getPath();
}
return null;
}
/**
* Get the value of the data column for this Uri. This is useful for
* MediaStore Uris, and other file-based ContentProviders.
*
* @param context The context.
* @param uri The Uri to query.
* @param selection (Optional) Filter used in the query.
* @param selectionArgs (Optional) Selection arguments used in the query.
* @return The value of the _data column, which is typically a file path.
*/
public static String getDataColumn(Context context, Uri uri, String selection,
String[] selectionArgs) {
Cursor cursor = null;
final String column = "_data";
final String[] projection = {
column
};
try {
cursor = context.getContentResolver().query(uri, projection, selection, selectionArgs,
null);
if (cursor != null && cursor.moveToFirst()) {
final int column_index = cursor.getColumnIndexOrThrow(column);
return cursor.getString(column_index);
}
} finally {
if (cursor != null)
cursor.close();
}
return null;
}
/**
* @param uri The Uri to check.
* @return Whether the Uri authority is ExternalStorageProvider.
*/
public static boolean isExternalStorageDocument(Uri uri) {
return "com.android.externalstorage.documents".equals(uri.getAuthority());
}
/**
* @param uri The Uri to check.
* @return Whether the Uri authority is DownloadsProvider.
*/
public static boolean isDownloadsDocument(Uri uri) {
return "com.android.providers.downloads.documents".equals(uri.getAuthority());
}
/**
* @param uri The Uri to check.
* @return Whether the Uri authority is MediaProvider.
*/
public static boolean isMediaDocument(Uri uri) {
return "com.android.providers.media.documents".equals(uri.getAuthority());
}
Они взяты из моей библиотеки с открытым исходным кодом, aFileChooser.
Ответ 3
ответ ниже написан https://stackoverflow.com/users/3082682/cvizv на странице, которая больше не существует, так как у него недостаточно ответов, чтобы ответить на вопрос, я отправляю его, Нет кредитов от меня.
public String getImagePath(Uri uri){
Cursor cursor = getContentResolver().query(uri, null, null, null, null);
cursor.moveToFirst();
String document_id = cursor.getString(0);
document_id = document_id.substring(document_id.lastIndexOf(":")+1);
cursor.close();
cursor = getContentResolver().query(
android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
null, MediaStore.Images.Media._ID + " = ? ", new String[]{document_id}, null);
cursor.moveToFirst();
String path = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA));
cursor.close();
return path;
}
edit: Ну, кому-то не понравился ответ (это нормально, чтобы не отвечать на него), и использовать отрицательный голос (что тоже хорошо), без предоставления обратной связи (что не хорошо). Пожалуйста, помогите...
edit2: В коде есть поток; если устройство имеет более одного внешнего хранилища (внешняя SD-карта, внешний USB и т.д.), над кодом не будут работать не первичные хранилища.
Ответ 4
До появления новой галереи в KitKat я получил свой реальный путь в sdcard с помощью этого метода
Это никогда не было надежным. Не требуется, чтобы Uri
, который вы возвращали из запроса ACTION_GET_CONTENT
или ACTION_PICK
, должен быть проиндексирован MediaStore
или даже должен представлять файл в файловой системе. Uri
может, например, представлять поток, где зашифрованный файл расшифровывается для вас на лету.
Как мне удалось получить реальный путь в SD-карте?
Нет требования о наличии файла, соответствующего Uri
.
Да, мне действительно нужен путь
Затем скопируйте файл из потока в свой временный файл и используйте его. Еще лучше, просто используйте поток напрямую и избегайте временного файла.
Я изменил свой Intent.ACTION_GET_CONTENT для Intent.ACTION_PICK
Это не поможет вашей ситуации. Нет требования, чтобы ответ ACTION_PICK
был для Uri
, у которого есть файл в файловой системе, который вы можете каким-то образом получить магическим путем.
Ответ 5
Этот ответ основан на вашем несколько смутном описании. Я предполагаю, что вы уволили намерение с действием: Intent.ACTION_GET_CONTENT
И теперь вы получаете content://com.android.providers.media.documents/document/image:62
назад вместо прежнего URI-провайдера, правильно?
В Android 4.4 (KitKat) новый DocumentActivity открывается, когда запускается Intent.ACTION_GET_CONTENT
, что приводит к представлению сетки (или списку), где вы можете выбрать изображение, это вернет следующие URI для вызова контекста (пример): content://com.android.providers.media.documents/document/image:62
(это URI для нового поставщика документов, он абстрагирует базовые данные, предоставляя универсальные URL-адреса поставщика документов клиентам).
Однако вы можете получить доступ к галерее и другим действиям, относящимся к Intent.ACTION_GET_CONTENT
, используя ящик в DocumentActivity (перетащите слева направо, и вы увидите пользовательский интерфейс ящика с галереей на выбор). Так же, как pre KitKat.
Если вы все еще выбираете класс DocumentsActivity и нуждаетесь в URI файла, вы должны иметь возможность сделать следующее (предупреждение, это хакерский!) запрос (с contentresolver): content://com.android.providers.media.documents/document/image:62
URI и прочитать значение _display_name из курсор. Это несколько уникальное имя (просто имя файла в локальных файлах) и используйте его в выборе (при запросе) для медиаператора, чтобы получить правильную строку, соответствующую этому выбору, здесь вы также можете получить URI файла.
Рекомендуемые способы доступа к поставщику документов можно найти здесь (получить входной файл или файловый дескриптор для чтения файла/растрового изображения):
Примеры использования documentprovider
Ответ 6
У меня была такая же проблема. Мне нужно имя файла, чтобы загрузить его на сайт.
Это сработало для меня, если я изменил намерение PICK.
Это было протестировано в AVD для Android 4.4 и AVD для Android 2.1.
Добавить разрешение READ_EXTERNAL_STORAGE:
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
Измените намерение:
Intent i = new Intent(
Intent.ACTION_PICK,
android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI
);
startActivityForResult(i, 66453666);
/* OLD CODE
Intent intent = new Intent();
intent.setType("image/*");
intent.setAction(Intent.ACTION_GET_CONTENT);
startActivityForResult(
Intent.createChooser( intent, "Select Image" ),
66453666
);
*/
Мне не нужно было менять свой код, чтобы получить фактический путь:
// Convert the image URI to the direct file system path of the image file
public String mf_szGetRealPathFromURI(final Context context, final Uri ac_Uri )
{
String result = "";
boolean isok = false;
Cursor cursor = null;
try {
String[] proj = { MediaStore.Images.Media.DATA };
cursor = context.getContentResolver().query(ac_Uri, proj, null, null, null);
int column_index = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA);
cursor.moveToFirst();
result = cursor.getString(column_index);
isok = true;
} finally {
if (cursor != null) {
cursor.close();
}
}
return isok ? result : "";
}
Ответ 7
Попробуйте следующее:
//KITKAT
i = new Intent(Intent.ACTION_PICK,android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
startActivityForResult(i, CHOOSE_IMAGE_REQUEST);
В onActivityResult используйте следующее:
Uri selectedImageURI = data.getData();
input = c.getContentResolver().openInputStream(selectedImageURI);
BitmapFactory.decodeStream(input , null, opts);
Ответ 8
Вот обновленная версия ответа Пол Берк. В версиях ниже Android 4.4 (KitKat) у нас нет DocumentsContract.
Для работы над версиями ниже KitKat создайте этот класс:
public class DocumentsContract {
private static final String DOCUMENT_URIS =
"com.android.providers.media.documents " +
"com.android.externalstorage.documents " +
"com.android.providers.downloads.documents " +
"com.android.providers.media.documents";
private static final String PATH_DOCUMENT = "document";
private static final String TAG = DocumentsContract.class.getSimpleName();
public static String getDocumentId(Uri documentUri) {
final List<String> paths = documentUri.getPathSegments();
if (paths.size() < 2) {
throw new IllegalArgumentException("Not a document: " + documentUri);
}
if (!PATH_DOCUMENT.equals(paths.get(0))) {
throw new IllegalArgumentException("Not a document: " + documentUri);
}
return paths.get(1);
}
public static boolean isDocumentUri(Uri uri) {
final List<String> paths = uri.getPathSegments();
Logger.v(TAG, "paths[" + paths + "]");
if (paths.size() < 2) {
return false;
}
if (!PATH_DOCUMENT.equals(paths.get(0))) {
return false;
}
return DOCUMENT_URIS.contains(uri.getAuthority());
}
}
Ответ 9
Нам нужно сделать следующие изменения/исправления в нашем предыдущем блоке выбора подписчиков onActivityResult(), чтобы без проблем работать на Android 4.4 (KitKat) и на всех других ранних версиях.
Uri selectedImgFileUri = data.getData();
if (selectedImgFileUri == null ) {
// The user has not selected any photo
}
try {
InputStream input = mActivity.getContentResolver().openInputStream(selectedImgFileUri);
mSelectedPhotoBmp = BitmapFactory.decodeStream(input);
}
catch (Throwable tr) {
// Show message to try again
}