#android #metadata
#Android #метаданные
Вопрос:
В настоящее время я работаю над музыкальным проигрывателем для устройств Android.
Для индексации музыкальных файлов из системы я написал эту функцию в Kotlin, чтобы получить курсор для всей музыки в файловой системе телефонов вместе с ее метаданными с помощью MediaStore.Audio.AudioColumns
:
private fun getMusicCursor(resolver: ContentResolver): Cursor? {
Log.i(this::class.simpleName, "Getting music cursor.")
return resolver.query(
MediaStore.Audio.Media.EXTERNAL_CONTENT_URI,
arrayOf(
AudioColumns._ID, // 0
AudioColumns.DISPLAY_NAME, // 1
AudioColumns.TITLE, // 2
AudioColumns.ARTIST, // 3
AudioColumns.ALBUM, // 4
AudioColumns.GENRE, // 5
AudioColumns.YEAR, // 6
AudioColumns.TRACK, // 7
AudioColumns.DURATION // 8
),
AudioColumns.IS_MUSIC "=1", null,
MediaStore.Audio.Media.DEFAULT_SORT_ORDER
)
}
Проблема в том, что когда я добавляю столбец жанра [ AudioColumns.GENRE
], код всегда завершается ошибкой и выдает это исключение, если оно не соответствует инструкции try-catch:
2020-08-21 09:14:26.077 16658-16715/org.oxycblt.auxio E/AndroidRuntime: FATAL EXCEPTION: DefaultDispatcher-worker-1
Process: org.oxycblt.auxio, PID: 16658
java.lang.IllegalArgumentException: Invalid column genre
at android.database.DatabaseUtils.readExceptionFromParcel(DatabaseUtils.java:170)
at android.database.DatabaseUtils.readExceptionFromParcel(DatabaseUtils.java:140)
at android.content.ContentProviderProxy.query(ContentProviderNative.java:437)
at android.content.ContentResolver.query(ContentResolver.java:962)
at android.content.ContentResolver.query(ContentResolver.java:890)
at android.content.ContentResolver.query(ContentResolver.java:846)
at org.oxycblt.auxio.music.MusicLoader.getMusicCursor(MusicLoader.kt:95)
at org.oxycblt.auxio.music.MusicLoader.findMusic(MusicLoader.kt:39)
at org.oxycblt.auxio.music.MusicLoader.<init>(MusicLoader.kt:29)
at org.oxycblt.auxio.music.MusicRepository.init(MusicRepository.kt:25)
at org.oxycblt.auxio.loading.LoadingViewModel$startMusicRepo$1.invokeSuspend(LoadingViewModel.kt:41)
at kotlin.coroutines.jvm.internal.BaseContinuationImpl.resumeWith(ContinuationImpl.kt:33)
at kotlinx.coroutines.DispatchedTask.run(Dispatched.kt:241)
at kotlinx.coroutines.scheduling.CoroutineScheduler.runSafely(CoroutineScheduler.kt:594)
at kotlinx.coroutines.scheduling.CoroutineScheduler.access$runSafely(CoroutineScheduler.kt:60)
at kotlinx.coroutines.scheduling.CoroutineScheduler$Worker.run(CoroutineScheduler.kt:740)
Это также влияет на другие значения, связанные с жанром, такие как GENRE_ID
и [устаревшие] GENRE_KEY
константы. Я мог бы использовать MediaMetadataRetriever
для получения значения жанра, но это занимает намного больше времени, чем использование курсора.
Я делаю что-то не так здесь?
Ответ №1:
Я нашел решение своей проблемы, вы можете сначала объявить курсор жанра, используя что-то вроде:
genreCursor = resolver.query(
Genres.EXTERNAL_CONTENT_URI,
arrayOf(
Genres._ID, // 0
Genres.NAME // 1
),
null, null,
Genres.DEFAULT_SORT_ORDER
)
Затем с помощью курсора создайте список всех жанров в системе.
После этого, когда вы захотите загрузить музыку, вы можете создать курсор песни, который фильтруется на основе идентификатора этого жанра, вот так:
songCursor = resolver.query(
Genres.Members.getContentUri("external", genreId),
arrayOf(
AudioColumns._ID, // 0
AudioColumns.DISPLAY_NAME, // 1
AudioColumns.TITLE, // 2
AudioColumns.ARTIST, // 3
AudioColumns.ALBUM, // 4
AudioColumns.YEAR, // 5
AudioColumns.TRACK, // 6
AudioColumns.DURATION // 7
),
AudioColumns.IS_MUSIC "=1", null,
MediaStore.Audio.Media.DEFAULT_SORT_ORDER
)
Повторите это для каждого жанра, и вы сможете загружать жанры вместе с музыкой просто отлично, хотя и странным способом.