Получить реальный путь от 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-карте?

9 ответов

Решение

Примечание. Этот ответ решает часть проблемы. Для полного решения (в форме библиотеки) посмотрите ответ Пола Бёрка.

Вы можете использовать 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();

Справка: я не могу найти пост, из которого взято это решение. Я хотел попросить оригинальный постер внести свой вклад здесь. Посмотрим еще немного сегодня вечером.

Это позволит получить путь к файлу от 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.

Ответ ниже написан https://stackru.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;
}

Изменить: есть поток на код; Если устройство имеет более одного внешнего хранилища (внешняя SD-карта, внешний USB и т. д.), код выше не будет работать не первичные хранилища.

Перед новым доступом к галерее в KitKat я получил реальный путь в SDCard с помощью этого метода

Это никогда не было надежным. Не требуется, чтобы Uri что вы вернулись из ACTION_GET_CONTENT или же ACTION_PICK запрос должен быть проиндексирован MediaStoreили даже должен представлять файл в файловой системе. Uri может, например, представлять поток, где зашифрованный файл расшифровывается для вас на лету.

Как мне удалось получить реальный путь в SDCard?

Не требуется наличие файла, соответствующего Uri,

Да мне очень нужен путь

Затем скопируйте файл из потока в ваш собственный временный файл и используйте его. А еще лучше, просто используйте поток напрямую и избегайте временного файла.

Я изменил свой Intent.ACTION_GET_CONTENT для Intent.ACTION_PICK

Это не поможет вашей ситуации. Не требуется, чтобы ACTION_PICK ответ будет для Uri в файловой системе есть файл, который вы можете каким-то волшебным образом получить.

У меня была точно такая же проблема. Мне нужно имя файла, чтобы иметь возможность загрузить его на сайт.

Это сработало для меня, если я изменил намерение на ПИК. Это было проверено в 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 : "";
 }

Этот ответ основан на вашем несколько расплывчатом описании. Я предполагаю, что вы уволены с действием: Intent.ACTION_GET_CONTENT

И теперь вы получаете content://com.android.providers.media.documents/document/image:62 вместо прежнего URI медиа-провайдера, верно?

На Android 4.4 (KitKat) новая DocumentsActivity открывается при Intent.ACTION_GET_CONTENT запускается, что приводит к представлению сетки (или представлению списка), где вы можете выбрать изображение, это вернет следующие URI в вызывающий контекст (пример): content://com.android.providers.media.documents/document/image:62 (это URI для нового провайдера документов, он абстрагирует базовые данные, предоставляя клиентам универсальные URI провайдера документов).

Однако вы можете получить доступ как к галерее, так и к другим действиям, отвечающим на Intent.ACTION_GET_CONTENT используя ящик в DocumentsActivity (перетащите слева направо, и вы увидите пользовательский интерфейс ящика с Галереей на выбор). Так же, как предварительно КитКат.

Если вам все еще нужно выбрать класс DocumentsActivity и вам нужен URI файла, вы сможете выполнить следующий (предупреждающий, что это хакерский!) Запрос (с помощью contentresolver): content://com.android.providers.media.documents/document/image:62 URI и считайте значение _display_name из курсора. Это несколько уникальное имя (только имя файла в локальных файлах), которое используется в выделении (при запросе) для mediaprovider, чтобы получить правильную строку, соответствующую этому выделению, отсюда вы также можете получить URI файла.

Рекомендуемые способы доступа к поставщику документов можно найти здесь (получите входной поток или дескриптор файла для чтения файла / растрового изображения):

Примеры использования документ-провайдера

Попробуй это:

//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);

Вот обновленная версия ответа Пола Бёрка. В версиях ниже 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());
    }
}

Нам нужно внести следующие изменения / исправления в наш предыдущий код средства выбора галереи 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
}
Другие вопросы по тегам