Android Q: получить галерею через MediaStore с информацией о местоположении
Если посмотреть на изменения в доступе к хранилищу, представленные в Android Q, информация о местоположении теперь отредактирована по умолчанию.
Google просит нас позвонить setRequireOriginal()
на объекте "MediaStore" с URI медиа в качестве параметра. Это работает, когда вы выбираете носители один за другим, но что делать, когда мы запрашиваем ContentResolver для всей галереи?
Смотрите этот образец:
String[] projection = {
MediaStore.Files.FileColumns._ID,
MediaStore.Files.FileColumns.DATA,
MediaStore.Files.FileColumns.MEDIA_TYPE,
MediaStore.Images.Media.DATE_TAKEN,
MediaStore.Images.Media.WIDTH,
MediaStore.Images.Media.HEIGHT,
MediaStore.Images.Media.LATITUDE, // <----- THIS
MediaStore.Images.Media.LONGITUDE, // <----- THIS
MediaStore.Images.Media.MIME_TYPE,
};
String selection = MediaStore.Files.FileColumns.MEDIA_TYPE + "="
+ MediaStore.Files.FileColumns.MEDIA_TYPE_IMAGE;
Uri queryUri = MediaStore.Files.getContentUri("external");
Cursor cursor = null;
MediaStore a ;
try {
cursor = context.getContentResolver().query(queryUri, projection, selection,
null, MediaStore.Images.Media.DATE_TAKEN + " DESC");
}
catch (NullPointerException ex){
}
Начиная с Q, широта и долгота всегда устанавливаются на 0. Есть ли способ получить данные о местоположении для группы носителей, предполагая, что ACCESS_MEDIA_LOCATION
разрешение добавлено в Манифест?
1 ответ
К сожалению, MediaStore.Images.Media.LATITUDE и MediaStore.Images.Media.LONGITUDE устарели в Android Q.
Обойти это для использования ExifInterface
как это:
Cursor cursor = null;
try {
String[] projection = {
MediaStore.Files.FileColumns._ID,
MediaStore.Images.Media.DATE_TAKEN,
MediaStore.Images.Media.WIDTH,
MediaStore.Images.Media.HEIGHT,
MediaStore.MediaColumns.TITLE,
MediaStore.Images.Media.MIME_TYPE,
MediaStore.Images.Media.LATITUDE,
MediaStore.Images.Media.LONGITUDE
};
cursor = getContentResolver().query(
MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
projection,
null,
null,
MediaStore.Images.Media.DATE_TAKEN + " DESC");
int idColumn = cursor.getColumnIndexOrThrow(MediaStore.MediaColumns._ID);
int titleColumn = cursor.getColumnIndexOrThrow(MediaStore.MediaColumns.TITLE);
int latColumn = BuildCompat.isAtLeastQ() ? -1
: cursor.getColumnIndexOrThrow(MediaStore.Images.Media.LATITUDE);
int longColumn = BuildCompat.isAtLeastQ() ? -1
: cursor.getColumnIndexOrThrow(MediaStore.Images.Media.LONGITUDE);
while (cursor.moveToNext()) {
Uri photoUri = Uri.withAppendedPath(
MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
cursor.getString(idColumn));
String title = cursor.getString(titleColumn);
final double[] latLong;
if (BuildCompat.isAtLeastQ()) {
photoUri = MediaStore.setRequireOriginal(photoUri);
InputStream stream = getContentResolver().openInputStream(photoUri);
if (stream == null) {
Log.w(TAG, "Got a null input stream for " + photoUri);
continue;
}
ExifInterface exifInterface = new ExifInterface(stream);
double[] returnedLatLong = exifInterface.getLatLong();
// If it returns null, fall back to {0.0, 0.0}.
latLong = returnedLatLong != null ? returnedLatLong : new double[2];
// After using ExifInterface, the stream should not be reused.
stream.close();
} else {
latLong = new double[]{
cursor.getFloat(latColumn),
cursor.getFloat(longColumn)
};
}
Log.i(TAG, title + " | lat: " + latLong[0] + " | lng: " + latLong[1]);
}
} catch (NullPointerException | IOException ex) {
Log.e(TAG, "Caught exception", ex);
} finally {
if (cursor != null) {
cursor.close();
}
}
Это единственный способ получить широту и долготу фотографии на Android Q на данный момент.
Это требует проведения ACCESS_MEDIA_LOCATION
разрешение.