Как запросить контент-провайдера Android MediaStore, избегая осиротевших изображений?

Я пытаюсь обеспечить активность в приложении, которая отображает миниатюры фотографий в медиа-хранилище устройства и позволяет пользователю выбрать его. После того, как пользователь сделает выбор, приложение считывает исходное полноразмерное изображение и делает с ним вещи.

я использую следующий код, чтобы создать Cursor над всеми изображениями на внешнем хранение:

public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView( R.layout.image_select );

    mGridView = (GridView) findViewById( R.id.image_select_grid );

    // Query for all images on external storage
    String[] projection = { MediaStore.Images.Media._ID };
    String selection = "";
    String [] selectionArgs = null;
    mImageCursor = managedQuery( MediaStore.Images.Thumbnails.EXTERNAL_CONTENT_URI,
                                 projection, selection, selectionArgs, null );

    // Initialize an adapter to display images in grid
    if ( mImageCursor != null ) {
        mImageCursor.moveToFirst();
        mAdapter = new LazyCursorAdapter(this, mImageCursor, R.drawable.image_select_default);
        mGridView.setAdapter( mAdapter );
    } else {
        Log.i(TAG, "System media store is empty.");
    }
}

и следующий код для загрузки эскиза изображения (Android 2.х код показано):

// ...
// Build URI to the main image from the cursor
int imageID = cursor.getInt( cursor.getColumnIndex(MediaStore.Images.Media._ID) );
Uri uri = Uri.withAppendedPath( MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
                                Integer.toString(imageID) );
loadThumbnailImage( uri.toString() );
// ...

protected Bitmap loadThumbnailImage( String url ) {
    // Get original image ID
    int originalImageId = Integer.parseInt(url.substring(url.lastIndexOf("/") + 1, url.length()));

    // Get (or create upon demand) the micro thumbnail for the original image.
    return MediaStore.Images.Thumbnails.getThumbnail(mContext.getContentResolver(),
                        originalImageId, MediaStore.Images.Thumbnails.MICRO_KIND, null);
}

и следующий код для загрузки исходного изображения из URL-адреса после того, как пользователь сделает выбор:

public Bitmap loadFullImage( Context context, Uri photoUri  ) {
    Cursor photoCursor = null;

    try {
        // Attempt to fetch asset filename for image
        String[] projection = { MediaStore.Images.Media.DATA };
        photoCursor = context.getContentResolver().query( photoUri, 
                                                    projection, null, null, null );

        if ( photoCursor != null && photoCursor.getCount() == 1 ) {
            photoCursor.moveToFirst();
            String photoFilePath = photoCursor.getString(
                photoCursor.getColumnIndex(MediaStore.Images.Media.DATA) );

            // Load image from path
            return BitmapFactory.decodeFile( photoFilePath, null );
        }
    } finally {
        if ( photoCursor != null ) {
            photoCursor.close();
        }
    }

    return null;
}

проблема, которую я вижу на некоторых устройствах Android, включая мой личный телефон, заключается в том, что курсор я получаю из запроса в onCreate() содержит несколько записей, для которых отсутствует фактический полноразмерный файл изображения (JPG или PNG). (В случае моего телефона изображения были импортированы и впоследствии удалены iPhoto).

в потерянные записи могут иметь или не иметь миниатюр, в зависимости от того, создаются ли миниатюры перед фактическим медиа-файлом при самоволке. Конечным результатом является то, что приложение отображает миниатюры для изображений, которые не существуют на самом деле.

у меня есть несколько вопросов:

  1. есть ли запрос, который я могу сделать к MediaStore контент-провайдер, который будет отфильтровывать изображения с отсутствующими носителями в возвращенном Cursor?
  2. есть ли средство или API для принудительного MediaStore для повторного сканирования и устранения записей сирот? На моем телефоне я смонтировал USB, затем размонтировал внешний носитель, который должен вызвать повторное сканирование. Но сиротские записи остаются.
  3. или есть что-то принципиально неправильное в моем подходе, что вызывает эту проблему?

спасибо.

2 ответов


хорошо, я нашел проблему с этим образцом кода.

на onCreate() метод, у меня эта строка:

mImageCursor = managedQuery( MediaStore.Images.Thumbnails.EXTERNAL_CONTENT_URI,
                             projection, selection, selectionArgs, null );

проблема здесь в том, что он запрашивает миниатюры, а не фактические изображения. Приложение камеры на устройствах HTC по умолчанию не создает эскизы, поэтому этот запрос не вернет изображения,которые еще не рассчитаны.

вместо этого запросите сами фактические изображения:

mImageCursor = managedQuery( MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
                             projection, selection, selectionArgs, null );

этот возвращает курсор, содержащий все полноразмерные изображения в системе. Затем вы можете позвонить:

Bitmap bm = MediaStore.Images.Thumbnails.getThumbnail(context.getContentResolver(),
        imageId, MediaStore.Images.Thumbnails.MINI_KIND, null);

который вернет миниатюру среднего размера для связанного полноразмерного изображения, генерируя его при необходимости. Чтобы получить миниатюру микро-размера, просто используйте .

Это также решило проблему поиска эскизов, которые имеют висячие ссылки на оригинальные полноразмерные изображения.


обратите внимание, что скоро все изменится, метод managedQuery устарел. Вместо этого используйте CursorLoader (начиная с уровня api 11).