From 8e3258b1de0f333a9a205710d7362f53f32e7e73 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Samuel=20De=CC=81fago?= Date: Thu, 29 Nov 2018 13:53:39 +0100 Subject: [PATCH 1/2] Fix DVR streams incorrectly sometimes starting in the past --- .../Controllers/SRGMediaPlayerController.m | 42 ++++++++++--------- 1 file changed, 23 insertions(+), 19 deletions(-) diff --git a/Framework/Sources/Controllers/SRGMediaPlayerController.m b/Framework/Sources/Controllers/SRGMediaPlayerController.m index 81780d22..20d110d1 100644 --- a/Framework/Sources/Controllers/SRGMediaPlayerController.m +++ b/Framework/Sources/Controllers/SRGMediaPlayerController.m @@ -172,30 +172,34 @@ - (void)setPlayer:(AVPlayer *)player } }; - // If a segment is targeted, add a small offset so that playback is guaranteed to start within the segment SRGPosition *startPosition = self.startPosition; - if (self.targetSegment) { - startPosition = SRGMediaPlayerControllerOffset(startPosition, CMTimeMakeWithSeconds(SRGSegmentSeekOffsetInSeconds, NSEC_PER_SEC)); - } - - // Take into account tolerance at the end of the content being played. If near the end enough, start - // at the default position instead. - CMTimeRange timeRange = self.targetSegment ? self.targetSegment.srg_timeRange : self.timeRange; - CMTime tolerance = SRGMediaPlayerEffectiveEndTolerance(self.endTolerance, self.endToleranceRatio, CMTimeGetSeconds(timeRange.duration)); - CMTime toleratedStartTime = CMTIME_COMPARE_INLINE(startPosition.time, >=, CMTimeSubtract(timeRange.duration, tolerance)) ? kCMTimeZero : startPosition.time; - // Positions in segments are relative. If not within a segment, they are absolute (relative positions - // are misleading for a DVR stream with a sliding window, and match the absolute position in other cases) - if (self.targetSegment) { - toleratedStartTime = CMTimeAdd(toleratedStartTime, timeRange.start); - } - SRGPosition *toleratedPosition = [SRGPosition positionWithTime:toleratedStartTime toleranceBefore:startPosition.toleranceBefore toleranceAfter:startPosition.toleranceAfter]; - - SRGPosition *seekPosition = SRGMediaPlayerControllerPositionInTimeRange(toleratedPosition, timeRange); - if (CMTIME_COMPARE_INLINE(seekPosition.time, ==, kCMTimeZero)) { + // Default position. Nothing to do. + if (CMTIME_COMPARE_INLINE(self.startPosition.time, ==, kCMTimeZero) && ! self.targetSegment) { completionBlock(YES); } + // Non-default start position. Calculate a valid position to seek to. else { + // If a segment is targeted, add a small offset so that playback is guaranteed to start within the segment + if (self.targetSegment) { + startPosition = SRGMediaPlayerControllerOffset(startPosition, CMTimeMakeWithSeconds(SRGSegmentSeekOffsetInSeconds, NSEC_PER_SEC)); + } + + // Take into account tolerance at the end of the content being played. If near the end enough, start + // at the default position instead. + CMTimeRange timeRange = self.targetSegment ? self.targetSegment.srg_timeRange : self.timeRange; + CMTime tolerance = SRGMediaPlayerEffectiveEndTolerance(self.endTolerance, self.endToleranceRatio, CMTimeGetSeconds(timeRange.duration)); + CMTime toleratedStartTime = CMTIME_COMPARE_INLINE(startPosition.time, >=, CMTimeSubtract(timeRange.duration, tolerance)) ? kCMTimeZero : startPosition.time; + + // Positions in segments are relative. If not within a segment, they are absolute (relative positions + // are misleading for a DVR stream with a sliding window, and match the absolute position in other cases) + if (self.targetSegment) { + toleratedStartTime = CMTimeAdd(toleratedStartTime, timeRange.start); + } + SRGPosition *toleratedPosition = [SRGPosition positionWithTime:toleratedStartTime toleranceBefore:startPosition.toleranceBefore toleranceAfter:startPosition.toleranceAfter]; + + SRGPosition *seekPosition = SRGMediaPlayerControllerPositionInTimeRange(toleratedPosition, timeRange); + // Call system method to avoid unwanted seek state in this special case [player seekToTime:seekPosition.time toleranceBefore:seekPosition.toleranceBefore toleranceAfter:seekPosition.toleranceAfter completionHandler:^(BOOL finished) { completionBlock(finished); From d749b2de26a4d18ac2e9c4cb2a395c0413b3ef5b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Samuel=20De=CC=81fago?= Date: Thu, 29 Nov 2018 13:59:05 +0100 Subject: [PATCH 2/2] Improve skip button behavior --- .../SRGMediaPlayerViewController.m | 20 ++++++++++++++++--- 1 file changed, 17 insertions(+), 3 deletions(-) diff --git a/Framework/Sources/Controllers/SRGMediaPlayerViewController.m b/Framework/Sources/Controllers/SRGMediaPlayerViewController.m index 6ac38566..de54c5d0 100644 --- a/Framework/Sources/Controllers/SRGMediaPlayerViewController.m +++ b/Framework/Sources/Controllers/SRGMediaPlayerViewController.m @@ -361,7 +361,14 @@ - (BOOL)canSkipBackwardFromTime:(CMTime)time return NO; } - SRGMediaPlayerStreamType streamType = self.controller.streamType; + SRGMediaPlayerController *controller = self.controller; + SRGMediaPlayerPlaybackState playbackState = controller.playbackState; + + if (playbackState == SRGMediaPlayerPlaybackStateIdle || playbackState == SRGMediaPlayerPlaybackStatePreparing) { + return NO; + } + + SRGMediaPlayerStreamType streamType = controller.streamType; return (streamType == SRGMediaPlayerStreamTypeOnDemand || streamType == SRGMediaPlayerStreamTypeDVR); } @@ -372,8 +379,15 @@ - (BOOL)canSkipForwardFromTime:(CMTime)time } SRGMediaPlayerController *controller = self.controller; - return (controller.streamType == SRGMediaPlayerStreamTypeOnDemand && CMTimeGetSeconds(time) + SRGMediaPlayerViewControllerForwardSkipInterval < CMTimeGetSeconds(controller.player.currentItem.duration)) - || (controller.streamType == SRGMediaPlayerStreamTypeDVR && ! controller.live); + SRGMediaPlayerPlaybackState playbackState = controller.playbackState; + + if (playbackState == SRGMediaPlayerPlaybackStateIdle || playbackState == SRGMediaPlayerPlaybackStatePreparing) { + return NO; + } + + SRGMediaPlayerStreamType streamType = controller.streamType; + return (streamType == SRGMediaPlayerStreamTypeOnDemand && CMTimeGetSeconds(time) + SRGMediaPlayerViewControllerForwardSkipInterval < CMTimeGetSeconds(controller.player.currentItem.duration)) + || (streamType == SRGMediaPlayerStreamTypeDVR && ! controller.live); } - (void)skipBackwardFromTime:(CMTime)time withCompletionHandler:(void (^)(BOOL finished))completionHandler