diff --git a/app/src/main/java/org/schabi/newpipe/player/Player.java b/app/src/main/java/org/schabi/newpipe/player/Player.java index 5b66bea6d..a6f4c75f2 100644 --- a/app/src/main/java/org/schabi/newpipe/player/Player.java +++ b/app/src/main/java/org/schabi/newpipe/player/Player.java @@ -178,6 +178,7 @@ import org.schabi.newpipe.player.playqueue.PlayQueueItemTouchCallback; import org.schabi.newpipe.player.resolver.AudioPlaybackResolver; import org.schabi.newpipe.player.resolver.MediaSourceTag; import org.schabi.newpipe.player.resolver.VideoPlaybackResolver; +import org.schabi.newpipe.player.resolver.VideoPlaybackResolver.SourceType; import org.schabi.newpipe.player.seekbarpreview.SeekbarPreviewThumbnailHelper; import org.schabi.newpipe.player.seekbarpreview.SeekbarPreviewThumbnailHolder; import org.schabi.newpipe.util.DeviceUtils; @@ -3293,8 +3294,9 @@ public final class Player implements if (audioPlayerSelected()) { return audioResolver.resolve(info); } else { - if (isAudioOnly - && !videoResolver.wasLastResolvedVideoAndAudioSeparated().orElse(false)) { + if (isAudioOnly && videoResolver.getStreamSourceType().orElse( + SourceType.VIDEO_WITH_AUDIO_OR_AUDIO_ONLY) + == SourceType.VIDEO_WITH_AUDIO_OR_AUDIO_ONLY) { // If the current info has only video streams with audio and if the stream is // played as audio, we need to use the audio resolver, otherwise the video stream // will be played in background. @@ -4196,18 +4198,30 @@ public final class Player implements stream will be fetched and the video stream will be fetched again when the user return to a video player. - For audio streams: nothing is done, it's not needed to reload the player with the same - audio stream. + For audio streams and audio live streams: nothing is done, it's not needed to reload the + player with the same audio stream. + + For video live streams: the play queue manager is not reloaded if the stream source is a + live source (see VideoPlaybackResolver#resolve()) and if that's not the case, the + requirements for video streams is applied. In the case where we don't know the index of the video renderer, the play queue manager is also reloaded. */ final StreamType streamType = info.getStreamType(); + final SourceType sourceType = videoResolver.getStreamSourceType() + .orElse(SourceType.VIDEO_WITH_SEPARATED_AUDIO); + final boolean isVideoWithSeparatedAudioOrVideoWithNoSeparatedAudioStreams = + sourceType == SourceType.VIDEO_WITH_SEPARATED_AUDIO + || (sourceType == SourceType.VIDEO_WITH_AUDIO_OR_AUDIO_ONLY + && isNullOrEmpty(info.getAudioStreams())); final boolean isVideoStreamTypeAndIsVideoOnlyStreamOrNoAudioStreamsAvailable = - (streamType == StreamType.VIDEO_STREAM || streamType == StreamType.LIVE_STREAM) - && (videoResolver.wasLastResolvedVideoAndAudioSeparated().orElse(false) - || isNullOrEmpty(info.getAudioStreams())); + streamType == StreamType.VIDEO_STREAM + && isVideoWithSeparatedAudioOrVideoWithNoSeparatedAudioStreams + || (streamType == StreamType.LIVE_STREAM + && (sourceType == SourceType.LIVE_STREAM + || isVideoWithSeparatedAudioOrVideoWithNoSeparatedAudioStreams)); if (videoRenderIndex != RENDERER_UNAVAILABLE && isVideoStreamTypeAndIsVideoOnlyStreamOrNoAudioStreamsAvailable) { diff --git a/app/src/main/java/org/schabi/newpipe/player/resolver/VideoPlaybackResolver.java b/app/src/main/java/org/schabi/newpipe/player/resolver/VideoPlaybackResolver.java index 654f91334..cbacc2679 100644 --- a/app/src/main/java/org/schabi/newpipe/player/resolver/VideoPlaybackResolver.java +++ b/app/src/main/java/org/schabi/newpipe/player/resolver/VideoPlaybackResolver.java @@ -32,11 +32,16 @@ public class VideoPlaybackResolver implements PlaybackResolver { private final PlayerDataSource dataSource; @NonNull private final QualityResolver qualityResolver; + private SourceType streamSourceType; @Nullable private String playbackQuality; - private Boolean wasLastResolvedVideoAndAudioSeparated; + public enum SourceType { + LIVE_STREAM, + VIDEO_WITH_SEPARATED_AUDIO, + VIDEO_WITH_AUDIO_OR_AUDIO_ONLY + } public VideoPlaybackResolver(@NonNull final Context context, @NonNull final PlayerDataSource dataSource, @@ -49,81 +54,79 @@ public class VideoPlaybackResolver implements PlaybackResolver { @Override @Nullable public MediaSource resolve(@NonNull final StreamInfo info) { - boolean isVideoAndAudioSeparated = false; - try { - final MediaSource liveSource = maybeBuildLiveMediaSource(dataSource, info); - if (liveSource != null) { - return liveSource; - } + final MediaSource liveSource = maybeBuildLiveMediaSource(dataSource, info); + if (liveSource != null) { + streamSourceType = SourceType.LIVE_STREAM; + return liveSource; + } - final List mediaSources = new ArrayList<>(); + final List mediaSources = new ArrayList<>(); - // Create video stream source - final List videos = ListHelper.getSortedStreamVideosList(context, - info.getVideoStreams(), info.getVideoOnlyStreams(), false, true); - final int index; - if (videos.isEmpty()) { - index = -1; - } else if (playbackQuality == null) { - index = qualityResolver.getDefaultResolutionIndex(videos); - } else { - index = qualityResolver.getOverrideResolutionIndex(videos, getPlaybackQuality()); - } - final MediaSourceTag tag = new MediaSourceTag(info, videos, index); - @Nullable final VideoStream video = tag.getSelectedVideoStream(); + // Create video stream source + final List videos = ListHelper.getSortedStreamVideosList(context, + info.getVideoStreams(), info.getVideoOnlyStreams(), false, true); + final int index; + if (videos.isEmpty()) { + index = -1; + } else if (playbackQuality == null) { + index = qualityResolver.getDefaultResolutionIndex(videos); + } else { + index = qualityResolver.getOverrideResolutionIndex(videos, getPlaybackQuality()); + } + final MediaSourceTag tag = new MediaSourceTag(info, videos, index); + @Nullable final VideoStream video = tag.getSelectedVideoStream(); - if (video != null) { - final MediaSource streamSource = buildMediaSource(dataSource, video.getUrl(), - PlayerHelper.cacheKeyOf(info, video), - MediaFormat.getSuffixById(video.getFormatId()), tag); - mediaSources.add(streamSource); - } + if (video != null) { + final MediaSource streamSource = buildMediaSource(dataSource, video.getUrl(), + PlayerHelper.cacheKeyOf(info, video), + MediaFormat.getSuffixById(video.getFormatId()), tag); + mediaSources.add(streamSource); + } - // Create optional audio stream source - final List audioStreams = info.getAudioStreams(); - final AudioStream audio = audioStreams.isEmpty() ? null : audioStreams.get( - ListHelper.getDefaultAudioFormat(context, audioStreams)); - // Use the audio stream if there is no video stream, or - // Merge with audio stream in case if video does not contain audio - if (audio != null && (video == null || video.isVideoOnly)) { - final MediaSource audioSource = buildMediaSource(dataSource, audio.getUrl(), - PlayerHelper.cacheKeyOf(info, audio), - MediaFormat.getSuffixById(audio.getFormatId()), tag); - mediaSources.add(audioSource); - isVideoAndAudioSeparated = true; - } + // Create optional audio stream source + final List audioStreams = info.getAudioStreams(); + final AudioStream audio = audioStreams.isEmpty() ? null : audioStreams.get( + ListHelper.getDefaultAudioFormat(context, audioStreams)); + // Use the audio stream if there is no video stream, or + // Merge with audio stream in case if video does not contain audio + if (audio != null && (video == null || video.isVideoOnly)) { + final MediaSource audioSource = buildMediaSource(dataSource, audio.getUrl(), + PlayerHelper.cacheKeyOf(info, audio), + MediaFormat.getSuffixById(audio.getFormatId()), tag); + mediaSources.add(audioSource); + streamSourceType = SourceType.VIDEO_WITH_SEPARATED_AUDIO; + } else { + streamSourceType = SourceType.VIDEO_WITH_AUDIO_OR_AUDIO_ONLY; + } - // If there is no audio or video sources, then this media source cannot be played back - if (mediaSources.isEmpty()) { - return null; - } - // Below are auxiliary media sources + // If there is no audio or video sources, then this media source cannot be played back + if (mediaSources.isEmpty()) { + return null; + } + // Below are auxiliary media sources - // Create subtitle sources - if (info.getSubtitles() != null) { - for (final SubtitlesStream subtitle : info.getSubtitles()) { - final String mimeType = PlayerHelper.subtitleMimeTypesOf(subtitle.getFormat()); - if (mimeType == null) { - continue; - } - final MediaSource textSource = dataSource.getSampleMediaSourceFactory() - .createMediaSource( - new MediaItem.Subtitle(Uri.parse(subtitle.getUrl()), - mimeType, - PlayerHelper.captionLanguageOf(context, subtitle)), - TIME_UNSET); - mediaSources.add(textSource); + // Create subtitle sources + if (info.getSubtitles() != null) { + for (final SubtitlesStream subtitle : info.getSubtitles()) { + final String mimeType = PlayerHelper.subtitleMimeTypesOf(subtitle.getFormat()); + if (mimeType == null) { + continue; } + final MediaSource textSource = dataSource.getSampleMediaSourceFactory() + .createMediaSource( + new MediaItem.Subtitle(Uri.parse(subtitle.getUrl()), + mimeType, + PlayerHelper.captionLanguageOf(context, subtitle)), + TIME_UNSET); + mediaSources.add(textSource); } + } - if (mediaSources.size() == 1) { - return mediaSources.get(0); - } else { - return new MergingMediaSource(mediaSources.toArray( - new MediaSource[0])); - } - } finally { - wasLastResolvedVideoAndAudioSeparated = isVideoAndAudioSeparated; + if (mediaSources.size() == 1) { + return mediaSources.get(0); + } else { + return new MergingMediaSource(mediaSources.toArray( + new MediaSource[0])); } } @@ -134,8 +137,8 @@ public class VideoPlaybackResolver implements PlaybackResolver { * @return {@link Optional#empty()} if nothing was resolved, otherwise {@code true} or * {@code false} */ - public Optional wasLastResolvedVideoAndAudioSeparated() { - return Optional.ofNullable(wasLastResolvedVideoAndAudioSeparated); + public Optional getStreamSourceType() { + return Optional.ofNullable(streamSourceType); } @Nullable