using System; using System.Collections.Generic; using System.Collections.ObjectModel; using System.Linq; using System.Threading; using ConfusedPolarBear.Plugin.IntroSkipper.Configuration; using ConfusedPolarBear.Plugin.IntroSkipper.Data; using Microsoft.Extensions.Logging; namespace ConfusedPolarBear.Plugin.IntroSkipper.Analyzers; /// /// Media file analyzer used to detect end credits that consist of text overlaid on a black background. /// Bisects the end of the video file to perform an efficient search. /// public class BlackFrameAnalyzer : IMediaFileAnalyzer { private readonly TimeSpan _maximumError = new(0, 0, 4); private readonly ILogger _logger; private readonly int _minimumCreditsDuration; private readonly int _maximumCreditsDuration; private readonly int _blackFrameMinimumPercentage; /// /// Initializes a new instance of the class. /// /// Logger. public BlackFrameAnalyzer(ILogger logger) { var config = Plugin.Instance?.Configuration ?? new PluginConfiguration(); _minimumCreditsDuration = config.MinimumCreditsDuration; _maximumCreditsDuration = 2 * config.MaximumCreditsDuration; _blackFrameMinimumPercentage = config.BlackFrameMinimumPercentage; _logger = logger; } /// public ReadOnlyCollection AnalyzeMediaFiles( ReadOnlyCollection analysisQueue, AnalysisMode mode, CancellationToken cancellationToken) { if (mode != AnalysisMode.Credits) { throw new NotImplementedException("mode must equal Credits"); } var creditTimes = new Dictionary(); var episodeAnalysisQueue = new List(analysisQueue); bool isFirstEpisode = true; double searchStart = _minimumCreditsDuration; var searchDistance = 2 * _minimumCreditsDuration; foreach (var episode in episodeAnalysisQueue.Where(e => !e.State.IsAnalyzed(mode))) { if (cancellationToken.IsCancellationRequested) { break; } // Pre-check to find reasonable starting point. if (isFirstEpisode) { var scanTime = episode.Duration - searchStart; var tr = new TimeRange(scanTime - 0.5, scanTime); // Short search range since accuracy isn't important here. var frames = FFmpegWrapper.DetectBlackFrames(episode, tr, _blackFrameMinimumPercentage); while (frames.Length > 0) // While black frames are found increase searchStart { searchStart += searchDistance; scanTime = episode.Duration - searchStart; tr = new TimeRange(scanTime - 0.5, scanTime); frames = FFmpegWrapper.DetectBlackFrames(episode, tr, _blackFrameMinimumPercentage); if (searchStart > _maximumCreditsDuration) { searchStart = _maximumCreditsDuration; break; } } if (searchStart == _minimumCreditsDuration) // Skip if no black frames were found { continue; } isFirstEpisode = false; } var credit = AnalyzeMediaFile( episode, searchStart, searchDistance, _blackFrameMinimumPercentage); if (credit is null) { // If no credits were found, reset the first-episode search logic for the next episode in the sequence. searchStart = _minimumCreditsDuration; isFirstEpisode = true; continue; } searchStart = episode.Duration - credit.Start + (0.5 * searchDistance); creditTimes.Add(episode.EpisodeId, credit); episode.State.SetAnalyzed(mode, true); } var analyzerHelper = new AnalyzerHelper(_logger); creditTimes = analyzerHelper.AdjustIntroTimes(analysisQueue, creditTimes, mode); Plugin.Instance!.UpdateTimestamps(creditTimes, mode); return episodeAnalysisQueue.AsReadOnly(); } /// /// Analyzes an individual media file. Only public because of unit tests. /// /// Media file to analyze. /// Search Start Piont. /// Search Distance. /// Percentage of the frame that must be black. /// Credits timestamp. public Segment? AnalyzeMediaFile(QueuedEpisode episode, double searchStart, int searchDistance, int minimum) { // Start by analyzing the last N minutes of the file. var upperLimit = searchStart; var lowerLimit = Math.Max(searchStart - searchDistance, _minimumCreditsDuration); var start = TimeSpan.FromSeconds(upperLimit); var end = TimeSpan.FromSeconds(lowerLimit); var firstFrameTime = 0.0; // Continue bisecting the end of the file until the range that contains the first black // frame is smaller than the maximum permitted error. while (start - end > _maximumError) { // Analyze the middle two seconds from the current bisected range var midpoint = (start + end) / 2; var scanTime = episode.Duration - midpoint.TotalSeconds; var tr = new TimeRange(scanTime, scanTime + 2); _logger.LogTrace( "{Episode}, dur {Duration}, bisect [{BStart}, {BEnd}], time [{Start}, {End}]", episode.Name, episode.Duration, start, end, tr.Start, tr.End); var frames = FFmpegWrapper.DetectBlackFrames(episode, tr, minimum); _logger.LogTrace( "{Episode} at {Start} has {Count} black frames", episode.Name, tr.Start, frames.Length); if (frames.Length == 0) { // Since no black frames were found, slide the range closer to the end start = midpoint - TimeSpan.FromSeconds(2); if (midpoint - TimeSpan.FromSeconds(lowerLimit) < _maximumError) { lowerLimit = Math.Max(lowerLimit - (0.5 * searchDistance), _minimumCreditsDuration); // Reset end for a new search with the increased duration end = TimeSpan.FromSeconds(lowerLimit); } } else { // Some black frames were found, slide the range closer to the start end = midpoint; firstFrameTime = frames[0].Time + scanTime; if (TimeSpan.FromSeconds(upperLimit) - midpoint < _maximumError) { upperLimit = Math.Min(upperLimit + (0.5 * searchDistance), _maximumCreditsDuration); // Reset start for a new search with the increased duration start = TimeSpan.FromSeconds(upperLimit); } } } if (firstFrameTime > 0) { return new(episode.EpisodeId, new TimeRange(firstFrameTime, episode.Duration)); } return null; } }