API плейлистов определяется интерфейсом Player , который реализован во всех реализациях ExoPlayer . Плейлисты позволяют последовательно воспроизводить несколько медиафайлов. Следующий пример показывает, как начать воспроизведение плейлиста, содержащего два видео:
Котлин
// Build the media items. val firstItem = MediaItem.fromUri(firstVideoUri) val secondItem = MediaItem.fromUri(secondVideoUri) // Add the media items to be played. player.addMediaItem(firstItem) player.addMediaItem(secondItem) // Prepare the player. player.prepare() // Start the playback. player.play()
Java
// Build the media items. MediaItem firstItem = MediaItem.fromUri(firstVideoUri); MediaItem secondItem = MediaItem.fromUri(secondVideoUri); // Add the media items to be played. player.addMediaItem(firstItem); player.addMediaItem(secondItem); // Prepare the player. player.prepare(); // Start the playback. player.play();
Переходы между элементами в плейлисте происходят плавно. Нет требования, чтобы они были одного формата (например, плейлист может содержать видео как в формате H264, так и в формате VP9). Они могут даже быть разных типов (то есть, плейлист может содержать видео, изображения и только аудиопотоки). Вы можете использовать один и тот же MediaItem несколько раз в одном плейлисте.
Изменение плейлиста
Вы можете динамически изменять плейлист, добавляя, перемещая, удаляя или заменяя медиафайлы. Это можно сделать как до, так и во время воспроизведения, вызывая соответствующие методы API плейлиста:
Котлин
// Adds a media item at position 1 in the playlist. player.addMediaItem(/* index= */ 1, MediaItem.fromUri(thirdUri)) // Moves the third media item from position 2 to the start of the playlist. player.moveMediaItem(/* currentIndex= */ 2, /* newIndex= */ 0) // Removes the first item from the playlist. player.removeMediaItem(/* index= */ 0) // Replace the second item in the playlist. player.replaceMediaItem(/* index= */ 1, MediaItem.fromUri(newUri))
Java
// Adds a media item at position 1 in the playlist. player.addMediaItem(/* index= */ 1, MediaItem.fromUri(thirdUri)); // Moves the third media item from position 2 to the start of the playlist. player.moveMediaItem(/* currentIndex= */ 2, /* newIndex= */ 0); // Removes the first item from the playlist. player.removeMediaItem(/* index= */ 0); // Replace the second item in the playlist. player.replaceMediaItem(/* index= */ 1, MediaItem.fromUri(newUri));
Также поддерживаются замена и очистка всего плейлиста:
Котлин
// Replaces the playlist with a new one. val newItems: List<MediaItem> = listOf(MediaItem.fromUri(fourthUri), MediaItem.fromUri(fifthUri)) player.setMediaItems(newItems, /* resetPosition= */ true) // Clears the playlist. If prepared, the player transitions to the ended state. player.clearMediaItems()
Java
// Replaces the playlist with a new one. ImmutableList<MediaItem> newItems = ImmutableList.of(MediaItem.fromUri(fourthUri), MediaItem.fromUri(fifthUri)); player.setMediaItems(newItems, /* resetPosition= */ true); // Clears the playlist. If prepared, the player transitions to the ended state. player.clearMediaItems();
Плеер автоматически и корректно обрабатывает изменения во время воспроизведения:
- Если текущий воспроизводимый
MediaItemперемещается, воспроизведение не прерывается, и после завершения будет воспроизведен следующий за ним файл. - Если воспроизводимый в данный момент
MediaItemбудет удален, плеер автоматически воспроизведет первый оставшийся следующий фрагмент или перейдет в завершенное состояние, если такого следующего фрагмента не существует. - Если воспроизводимый в данный момент
MediaItemзаменяется, воспроизведение не прерывается, если ни одно из свойствMediaItem, имеющих отношение к воспроизведению, не изменилось. Например, в большинстве случаев можно обновить поляMediaItem.MediaMetadataбез влияния на воспроизведение.
Запрос к плейлисту
Запрос к плейлисту можно выполнить с помощью Player.getMediaItemCount и Player.getMediaItemAt . Запрос текущего воспроизводимого медиафайла можно выполнить с помощью метода Player.getCurrentMediaItem . Также существуют другие удобные методы, такие как Player.hasNextMediaItem или Player.getNextMediaItemIndex , которые упрощают навигацию по плейлисту.
режимы повтора
Плеер поддерживает 3 режима повтора, которые можно установить в любое время с помощью Player.setRepeatMode :
-
Player.REPEAT_MODE_OFF: Плейлист не повторяется, и плеер перейдет в состояниеPlayer.STATE_ENDEDпосле воспроизведения последнего элемента плейлиста. -
Player.REPEAT_MODE_ONE: Текущий элемент повторяется в бесконечном цикле. Такие методы, какPlayer.seekToNextMediaItemигнорируют это и переходят к следующему элементу в списке, который затем будет повторяться в бесконечном цикле. -
Player.REPEAT_MODE_ALL: Весь плейлист повторяется в бесконечном цикле.
режим перемешивания
Режим перемешивания можно включить или выключить в любое время с помощью Player.setShuffleModeEnabled . В режиме перемешивания плеер будет воспроизводить плейлист в заранее рассчитанном случайном порядке. Все элементы будут воспроизведены один раз, а режим перемешивания также можно комбинировать с Player.REPEAT_MODE_ALL для повторения того же случайного порядка в бесконечном цикле. Когда режим перемешивания выключен, воспроизведение продолжается с текущего элемента с его исходной позиции в плейлисте.
Обратите внимание, что индексы, возвращаемые такими методами, как Player.getCurrentMediaItemIndex всегда относятся к исходному порядку воспроизведения без перемешивания. Аналогично, Player.seekToNextMediaItem не будет воспроизводить элемент с индексом player.getCurrentMediaItemIndex() + 1 , а следующий элемент в соответствии с порядком перемешивания. Добавление новых элементов в плейлист или удаление элементов позволит сохранить существующий порядок перемешивания неизменным, насколько это возможно.
Настройка пользовательского порядка перемешивания
По умолчанию плеер поддерживает перемешивание, используя DefaultShuffleOrder . Это можно настроить, предоставив собственную реализацию порядка перемешивания или задав пользовательский порядок в конструкторе DefaultShuffleOrder :
Котлин
// Set a custom shuffle order for the 5 items currently in the playlist: exoPlayer.setShuffleOrder(DefaultShuffleOrder(intArrayOf(3, 1, 0, 4, 2), randomSeed)) // Enable shuffle mode. exoPlayer.shuffleModeEnabled = true
Java
// Set a custom shuffle order for the 5 items currently in the playlist: exoPlayer.setShuffleOrder(new DefaultShuffleOrder(new int[] {3, 1, 0, 4, 2}, randomSeed)); // Enable shuffle mode. exoPlayer.setShuffleModeEnabled(/* shuffleModeEnabled= */ true);
Идентификация элементов плейлиста
Для идентификации элементов плейлиста при его создании можно задать параметр MediaItem.mediaId :
Котлин
// Build a media item with a media ID. val mediaItem = MediaItem.Builder().setUri(uri).setMediaId(mediaId).build()
Java
// Build a media item with a media ID. MediaItem mediaItem = new MediaItem.Builder().setUri(uri).setMediaId(mediaId).build();
Если приложение явно не определяет идентификатор медиафайла, используется строковое представление URI.
Связывание данных приложения с элементами плейлиста
Помимо идентификатора, каждый медиафайл может быть также настроен с помощью пользовательского тега, который может представлять собой любой объект, предоставляемый приложением. Один из способов использования пользовательских тегов — добавление метаданных к каждому медиафайлу:
Котлин
// Build a media item with a custom tag. val mediaItem = MediaItem.Builder().setUri(uri).setTag(metadata).build()
Java
// Build a media item with a custom tag. MediaItem mediaItem = new MediaItem.Builder().setUri(uri).setTag(metadata).build();
Определение момента перехода воспроизведения к другому медиафайлу.
Когда воспроизведение переходит к другому медиафайлу или начинается повторение того же медиафайла, вызывается Listener.onMediaItemTransition(MediaItem, @MediaItemTransitionReason) . Этот коллбэк получает новый медиафайл вместе с аннотацией @MediaItemTransitionReason , указывающей причину перехода. Часто используемый метод onMediaItemTransition используется для обновления пользовательского интерфейса приложения в соответствии с новым медиафайлом:
Котлин
override fun onMediaItemTransition( mediaItem: MediaItem?, @MediaItemTransitionReason reason: Int, ) { updateUiForPlayingMediaItem(mediaItem) }
Java
@Override public void onMediaItemTransition( @Nullable MediaItem mediaItem, @MediaItemTransitionReason int reason) { updateUiForPlayingMediaItem(mediaItem); }
Если метаданные, необходимые для обновления пользовательского интерфейса, прикрепляются к каждому медиафайлу с помощью пользовательских тегов, то реализация может выглядеть следующим образом:
Котлин
override fun onMediaItemTransition( mediaItem: MediaItem?, @MediaItemTransitionReason reason: Int, ) { var metadata: CustomMetadata? = null mediaItem?.localConfiguration?.let { localConfiguration -> metadata = localConfiguration.tag as? CustomMetadata } updateUiForPlayingMediaItem(metadata) }
Java
@Override public void onMediaItemTransition( @Nullable MediaItem mediaItem, @MediaItemTransitionReason int reason) { @Nullable CustomMetadata metadata = null; if (mediaItem != null && mediaItem.localConfiguration != null) { metadata = (CustomMetadata) mediaItem.localConfiguration.tag; } updateUiForPlayingMediaItem(metadata); }
Определение момента изменения плейлиста
При добавлении, удалении или перемещении медиафайла немедленно вызывается Listener.onTimelineChanged(Timeline, @TimelineChangeReason) с параметром TIMELINE_CHANGE_REASON_PLAYLIST_CHANGED . Этот коллбэк вызывается даже тогда, когда плеер еще не подготовлен.
Котлин
override fun onTimelineChanged(timeline: Timeline, @TimelineChangeReason reason: Int) { if (reason == Player.TIMELINE_CHANGE_REASON_PLAYLIST_CHANGED) { // Update the UI according to the modified playlist (add, move or remove). updateUiForPlaylist(timeline) } }
Java
@Override public void onTimelineChanged(Timeline timeline, @TimelineChangeReason int reason) { if (reason == TIMELINE_CHANGE_REASON_PLAYLIST_CHANGED) { // Update the UI according to the modified playlist (add, move or remove). updateUiForPlaylist(timeline); } }
Когда становится доступна такая информация, как продолжительность медиафайла в плейлисте, Timeline обновляется, и вызывается метод onTimelineChanged с TIMELINE_CHANGE_REASON_SOURCE_UPDATE . Другие причины, которые могут вызвать обновление временной шкалы, включают:
- Манифест становится доступным после подготовки адаптивного медиаконтента.
- Манифест, периодически обновляемый во время воспроизведения прямой трансляции.