Не удается извлечь метаданные жанра из песни с помощью MediaStore.Аудио.AudioColumns

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

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