web-dev-qa-db-ja.com

クエリの方法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.xコードが表示されています):

_// ...
// 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によって消去されました)。

孤立したエントリには、AWOL時に実際のメディアファイルの前にサムネイルが生成されるかどうかに応じて、サムネイルが表示される場合とされない場合があります。最終的に、アプリには実際には存在しない画像のサムネイルが表示されます。

少し質問があります:

  1. MediaStoreコンテンツプロバイダーに対して、返されたCursorのメディアが欠落している画像を除外するクエリを作成できますか?
  2. MediaStoreを強制的に再スキャンし、孤立したエントリを削除する手段、またはAPIはありますか?電話で、外部メディアをUSBマウントしてからアンマウントしました。これにより、再スキャンがトリガーされます。しかし、孤立したエントリは残ります。
  3. または、この問題を引き起こしている私のアプローチに根本的な問題がありますか?

ありがとう。

37
mportuesisf

さて、このコードサンプルで問題が見つかりました。

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

これにより、関連するフルサイズの画像の中サイズのサムネイルが返され、必要に応じて生成されます。マイクロサイズのサムネイルを取得するには、MediaStore.Images.Thumbnails.MICRO_KIND代わりに。

これにより、元のフルサイズ画像への参照がぶら下がっているサムネイルを見つける問題も解決しました。

61
mportuesisf

物事はすぐに変更されることに注意してください、managedQueryメソッドは非推奨です。代わりにCursorLoaderを使用します(APIレベル11以降)。

7
Tamas