|
@@ -17,370 +17,369 @@ using MediaBrowser.Model.IO;
|
|
|
using Microsoft.Extensions.Logging;
|
|
|
using SkiaSharp;
|
|
|
|
|
|
-namespace MediaBrowser.Providers.Trickplay
|
|
|
+namespace MediaBrowser.Providers.Trickplay;
|
|
|
+
|
|
|
+/// <summary>
|
|
|
+/// ITrickplayManager implementation.
|
|
|
+/// </summary>
|
|
|
+public class TrickplayManager : ITrickplayManager
|
|
|
{
|
|
|
+ private readonly ILogger<TrickplayManager> _logger;
|
|
|
+ private readonly IItemRepository _itemRepo;
|
|
|
+ private readonly IMediaEncoder _mediaEncoder;
|
|
|
+ private readonly IFileSystem _fileSystem;
|
|
|
+ private readonly EncodingHelper _encodingHelper;
|
|
|
+ private readonly ILibraryManager _libraryManager;
|
|
|
+ private readonly IServerConfigurationManager _config;
|
|
|
+
|
|
|
+ private static readonly SemaphoreSlim _resourcePool = new(1, 1);
|
|
|
+
|
|
|
/// <summary>
|
|
|
- /// ITrickplayManager implementation.
|
|
|
+ /// Initializes a new instance of the <see cref="TrickplayManager"/> class.
|
|
|
/// </summary>
|
|
|
- public class TrickplayManager : ITrickplayManager
|
|
|
+ /// <param name="logger">The logger.</param>
|
|
|
+ /// <param name="itemRepo">The item repository.</param>
|
|
|
+ /// <param name="mediaEncoder">The media encoder.</param>
|
|
|
+ /// <param name="fileSystem">The file systen.</param>
|
|
|
+ /// <param name="encodingHelper">The encoding helper.</param>
|
|
|
+ /// <param name="libraryManager">The library manager.</param>
|
|
|
+ /// <param name="config">The server configuration manager.</param>
|
|
|
+ public TrickplayManager(
|
|
|
+ ILogger<TrickplayManager> logger,
|
|
|
+ IItemRepository itemRepo,
|
|
|
+ IMediaEncoder mediaEncoder,
|
|
|
+ IFileSystem fileSystem,
|
|
|
+ EncodingHelper encodingHelper,
|
|
|
+ ILibraryManager libraryManager,
|
|
|
+ IServerConfigurationManager config)
|
|
|
{
|
|
|
- private readonly ILogger<TrickplayManager> _logger;
|
|
|
- private readonly IItemRepository _itemRepo;
|
|
|
- private readonly IMediaEncoder _mediaEncoder;
|
|
|
- private readonly IFileSystem _fileSystem;
|
|
|
- private readonly EncodingHelper _encodingHelper;
|
|
|
- private readonly ILibraryManager _libraryManager;
|
|
|
- private readonly IServerConfigurationManager _config;
|
|
|
-
|
|
|
- private static readonly SemaphoreSlim _resourcePool = new(1, 1);
|
|
|
-
|
|
|
- /// <summary>
|
|
|
- /// Initializes a new instance of the <see cref="TrickplayManager"/> class.
|
|
|
- /// </summary>
|
|
|
- /// <param name="logger">The logger.</param>
|
|
|
- /// <param name="itemRepo">The item repository.</param>
|
|
|
- /// <param name="mediaEncoder">The media encoder.</param>
|
|
|
- /// <param name="fileSystem">The file systen.</param>
|
|
|
- /// <param name="encodingHelper">The encoding helper.</param>
|
|
|
- /// <param name="libraryManager">The library manager.</param>
|
|
|
- /// <param name="config">The server configuration manager.</param>
|
|
|
- public TrickplayManager(
|
|
|
- ILogger<TrickplayManager> logger,
|
|
|
- IItemRepository itemRepo,
|
|
|
- IMediaEncoder mediaEncoder,
|
|
|
- IFileSystem fileSystem,
|
|
|
- EncodingHelper encodingHelper,
|
|
|
- ILibraryManager libraryManager,
|
|
|
- IServerConfigurationManager config)
|
|
|
+ _logger = logger;
|
|
|
+ _itemRepo = itemRepo;
|
|
|
+ _mediaEncoder = mediaEncoder;
|
|
|
+ _fileSystem = fileSystem;
|
|
|
+ _encodingHelper = encodingHelper;
|
|
|
+ _libraryManager = libraryManager;
|
|
|
+ _config = config;
|
|
|
+ }
|
|
|
+
|
|
|
+ /// <inheritdoc />
|
|
|
+ public async Task RefreshTrickplayDataAsync(Video video, bool replace, CancellationToken cancellationToken)
|
|
|
+ {
|
|
|
+ _logger.LogDebug("Trickplay refresh for {ItemId} (replace existing: {Replace})", video.Id, replace);
|
|
|
+
|
|
|
+ var options = _config.Configuration.TrickplayOptions;
|
|
|
+ foreach (var width in options.WidthResolutions)
|
|
|
{
|
|
|
- _logger = logger;
|
|
|
- _itemRepo = itemRepo;
|
|
|
- _mediaEncoder = mediaEncoder;
|
|
|
- _fileSystem = fileSystem;
|
|
|
- _encodingHelper = encodingHelper;
|
|
|
- _libraryManager = libraryManager;
|
|
|
- _config = config;
|
|
|
+ cancellationToken.ThrowIfCancellationRequested();
|
|
|
+ await RefreshTrickplayDataInternal(
|
|
|
+ video,
|
|
|
+ replace,
|
|
|
+ width,
|
|
|
+ options,
|
|
|
+ cancellationToken).ConfigureAwait(false);
|
|
|
}
|
|
|
+ }
|
|
|
|
|
|
- /// <inheritdoc />
|
|
|
- public async Task RefreshTrickplayData(Video video, bool replace, CancellationToken cancellationToken)
|
|
|
+ private async Task RefreshTrickplayDataInternal(
|
|
|
+ Video video,
|
|
|
+ bool replace,
|
|
|
+ int width,
|
|
|
+ TrickplayOptions options,
|
|
|
+ CancellationToken cancellationToken)
|
|
|
+ {
|
|
|
+ if (!CanGenerateTrickplay(video, options.Interval))
|
|
|
{
|
|
|
- _logger.LogDebug("Trickplay refresh for {ItemId} (replace existing: {Replace})", video.Id, replace);
|
|
|
-
|
|
|
- var options = _config.Configuration.TrickplayOptions;
|
|
|
- foreach (var width in options.WidthResolutions)
|
|
|
- {
|
|
|
- cancellationToken.ThrowIfCancellationRequested();
|
|
|
- await RefreshTrickplayDataInternal(
|
|
|
- video,
|
|
|
- replace,
|
|
|
- width,
|
|
|
- options,
|
|
|
- cancellationToken).ConfigureAwait(false);
|
|
|
- }
|
|
|
+ return;
|
|
|
}
|
|
|
|
|
|
- private async Task RefreshTrickplayDataInternal(
|
|
|
- Video video,
|
|
|
- bool replace,
|
|
|
- int width,
|
|
|
- TrickplayOptions options,
|
|
|
- CancellationToken cancellationToken)
|
|
|
+ var imgTempDir = string.Empty;
|
|
|
+ var outputDir = GetTrickplayDirectory(video, width);
|
|
|
+
|
|
|
+ try
|
|
|
{
|
|
|
- if (!CanGenerateTrickplay(video, options.Interval))
|
|
|
+ await _resourcePool.WaitAsync(cancellationToken).ConfigureAwait(false);
|
|
|
+
|
|
|
+ if (!replace && Directory.Exists(outputDir) && GetTilesResolutions(video.Id).ContainsKey(width))
|
|
|
{
|
|
|
+ _logger.LogDebug("Found existing trickplay files for {ItemId}. Exiting.", video.Id);
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
- var imgTempDir = string.Empty;
|
|
|
- var outputDir = GetTrickplayDirectory(video, width);
|
|
|
+ // Extract images
|
|
|
+ // Note: Media sources under parent items exist as their own video/item as well. Only use this video stream for trickplay.
|
|
|
+ var mediaSource = video.GetMediaSources(false).Find(source => Guid.Parse(source.Id).Equals(video.Id));
|
|
|
|
|
|
- try
|
|
|
+ if (mediaSource is null)
|
|
|
{
|
|
|
- await _resourcePool.WaitAsync(cancellationToken).ConfigureAwait(false);
|
|
|
-
|
|
|
- if (!replace && Directory.Exists(outputDir) && GetTilesResolutions(video.Id).ContainsKey(width))
|
|
|
- {
|
|
|
- _logger.LogDebug("Found existing trickplay files for {ItemId}. Exiting.", video.Id);
|
|
|
- return;
|
|
|
- }
|
|
|
-
|
|
|
- // Extract images
|
|
|
- // Note: Media sources under parent items exist as their own video/item as well. Only use this video stream for trickplay.
|
|
|
- var mediaSource = video.GetMediaSources(false).Find(source => Guid.Parse(source.Id).Equals(video.Id));
|
|
|
-
|
|
|
- if (mediaSource is null)
|
|
|
- {
|
|
|
- _logger.LogDebug("Found no matching media source for item {ItemId}", video.Id);
|
|
|
- return;
|
|
|
- }
|
|
|
+ _logger.LogDebug("Found no matching media source for item {ItemId}", video.Id);
|
|
|
+ return;
|
|
|
+ }
|
|
|
|
|
|
- var mediaPath = mediaSource.Path;
|
|
|
- var mediaStream = mediaSource.VideoStream;
|
|
|
- var container = mediaSource.Container;
|
|
|
-
|
|
|
- _logger.LogInformation("Creating trickplay files at {Width} width, for {Path} [ID: {ItemId}]", width, mediaPath, video.Id);
|
|
|
- imgTempDir = await _mediaEncoder.ExtractVideoImagesOnIntervalAccelerated(
|
|
|
- mediaPath,
|
|
|
- container,
|
|
|
- mediaSource,
|
|
|
- mediaStream,
|
|
|
- width,
|
|
|
- TimeSpan.FromMilliseconds(options.Interval),
|
|
|
- options.EnableHwAcceleration,
|
|
|
- options.ProcessThreads,
|
|
|
- options.Qscale,
|
|
|
- options.ProcessPriority,
|
|
|
- _encodingHelper,
|
|
|
- cancellationToken).ConfigureAwait(false);
|
|
|
-
|
|
|
- if (string.IsNullOrEmpty(imgTempDir) || !Directory.Exists(imgTempDir))
|
|
|
- {
|
|
|
- throw new InvalidOperationException("Null or invalid directory from media encoder.");
|
|
|
- }
|
|
|
+ var mediaPath = mediaSource.Path;
|
|
|
+ var mediaStream = mediaSource.VideoStream;
|
|
|
+ var container = mediaSource.Container;
|
|
|
+
|
|
|
+ _logger.LogInformation("Creating trickplay files at {Width} width, for {Path} [ID: {ItemId}]", width, mediaPath, video.Id);
|
|
|
+ imgTempDir = await _mediaEncoder.ExtractVideoImagesOnIntervalAccelerated(
|
|
|
+ mediaPath,
|
|
|
+ container,
|
|
|
+ mediaSource,
|
|
|
+ mediaStream,
|
|
|
+ width,
|
|
|
+ TimeSpan.FromMilliseconds(options.Interval),
|
|
|
+ options.EnableHwAcceleration,
|
|
|
+ options.ProcessThreads,
|
|
|
+ options.Qscale,
|
|
|
+ options.ProcessPriority,
|
|
|
+ _encodingHelper,
|
|
|
+ cancellationToken).ConfigureAwait(false);
|
|
|
+
|
|
|
+ if (string.IsNullOrEmpty(imgTempDir) || !Directory.Exists(imgTempDir))
|
|
|
+ {
|
|
|
+ throw new InvalidOperationException("Null or invalid directory from media encoder.");
|
|
|
+ }
|
|
|
|
|
|
- var images = _fileSystem.GetFiles(imgTempDir, new string[] { ".jpg" }, false, false)
|
|
|
- .Where(img => string.Equals(img.Extension, ".jpg", StringComparison.Ordinal))
|
|
|
- .OrderBy(i => i.FullName)
|
|
|
- .ToList();
|
|
|
+ var images = _fileSystem.GetFiles(imgTempDir, new string[] { ".jpg" }, false, false)
|
|
|
+ .Where(img => string.Equals(img.Extension, ".jpg", StringComparison.Ordinal))
|
|
|
+ .OrderBy(i => i.FullName)
|
|
|
+ .ToList();
|
|
|
|
|
|
- // Create tiles
|
|
|
- var tilesTempDir = Path.Combine(imgTempDir, Guid.NewGuid().ToString("N"));
|
|
|
- var tilesInfo = CreateTiles(images, width, options, tilesTempDir, outputDir);
|
|
|
+ // Create tiles
|
|
|
+ var tilesTempDir = Path.Combine(imgTempDir, Guid.NewGuid().ToString("N"));
|
|
|
+ var tilesInfo = CreateTiles(images, width, options, tilesTempDir, outputDir);
|
|
|
|
|
|
- // Save tiles info
|
|
|
- try
|
|
|
+ // Save tiles info
|
|
|
+ try
|
|
|
+ {
|
|
|
+ if (tilesInfo is not null)
|
|
|
{
|
|
|
- if (tilesInfo is not null)
|
|
|
- {
|
|
|
- SaveTilesInfo(video.Id, tilesInfo);
|
|
|
- _logger.LogInformation("Finished creation of trickplay files for {0}", mediaPath);
|
|
|
- }
|
|
|
- else
|
|
|
- {
|
|
|
- throw new InvalidOperationException("Null trickplay tiles info from CreateTiles.");
|
|
|
- }
|
|
|
+ SaveTilesInfo(video.Id, tilesInfo);
|
|
|
+ _logger.LogInformation("Finished creation of trickplay files for {0}", mediaPath);
|
|
|
}
|
|
|
- catch (Exception ex)
|
|
|
+ else
|
|
|
{
|
|
|
- _logger.LogError(ex, "Error while saving trickplay tiles info.");
|
|
|
-
|
|
|
- // Make sure no files stay in metadata folders on failure
|
|
|
- // if tiles info wasn't saved.
|
|
|
- Directory.Delete(outputDir, true);
|
|
|
+ throw new InvalidOperationException("Null trickplay tiles info from CreateTiles.");
|
|
|
}
|
|
|
}
|
|
|
catch (Exception ex)
|
|
|
{
|
|
|
- _logger.LogError(ex, "Error creating trickplay images.");
|
|
|
- }
|
|
|
- finally
|
|
|
- {
|
|
|
- _resourcePool.Release();
|
|
|
+ _logger.LogError(ex, "Error while saving trickplay tiles info.");
|
|
|
|
|
|
- if (!string.IsNullOrEmpty(imgTempDir))
|
|
|
- {
|
|
|
- Directory.Delete(imgTempDir, true);
|
|
|
- }
|
|
|
+ // Make sure no files stay in metadata folders on failure
|
|
|
+ // if tiles info wasn't saved.
|
|
|
+ Directory.Delete(outputDir, true);
|
|
|
}
|
|
|
}
|
|
|
-
|
|
|
- private TrickplayTilesInfo CreateTiles(List<FileSystemMetadata> images, int width, TrickplayOptions options, string workDir, string outputDir)
|
|
|
+ catch (Exception ex)
|
|
|
+ {
|
|
|
+ _logger.LogError(ex, "Error creating trickplay images.");
|
|
|
+ }
|
|
|
+ finally
|
|
|
{
|
|
|
- if (images.Count == 0)
|
|
|
+ _resourcePool.Release();
|
|
|
+
|
|
|
+ if (!string.IsNullOrEmpty(imgTempDir))
|
|
|
{
|
|
|
- throw new InvalidOperationException("Can't create trickplay from 0 images.");
|
|
|
+ Directory.Delete(imgTempDir, true);
|
|
|
}
|
|
|
+ }
|
|
|
+ }
|
|
|
|
|
|
- Directory.CreateDirectory(workDir);
|
|
|
+ private TrickplayTilesInfo CreateTiles(List<FileSystemMetadata> images, int width, TrickplayOptions options, string workDir, string outputDir)
|
|
|
+ {
|
|
|
+ if (images.Count == 0)
|
|
|
+ {
|
|
|
+ throw new InvalidOperationException("Can't create trickplay from 0 images.");
|
|
|
+ }
|
|
|
|
|
|
- var tilesInfo = new TrickplayTilesInfo
|
|
|
- {
|
|
|
- Width = width,
|
|
|
- Interval = options.Interval,
|
|
|
- TileWidth = options.TileWidth,
|
|
|
- TileHeight = options.TileHeight,
|
|
|
- TileCount = 0,
|
|
|
- Bandwidth = 0
|
|
|
- };
|
|
|
-
|
|
|
- var firstImg = SKBitmap.Decode(images[0].FullName);
|
|
|
- if (firstImg == null)
|
|
|
- {
|
|
|
- throw new InvalidDataException("Could not decode image data.");
|
|
|
- }
|
|
|
+ Directory.CreateDirectory(workDir);
|
|
|
|
|
|
- tilesInfo.Height = firstImg.Height;
|
|
|
- if (tilesInfo.Width != firstImg.Width)
|
|
|
- {
|
|
|
- throw new InvalidOperationException("Image width does not match config width.");
|
|
|
- }
|
|
|
+ var tilesInfo = new TrickplayTilesInfo
|
|
|
+ {
|
|
|
+ Width = width,
|
|
|
+ Interval = options.Interval,
|
|
|
+ TileWidth = options.TileWidth,
|
|
|
+ TileHeight = options.TileHeight,
|
|
|
+ TileCount = 0,
|
|
|
+ Bandwidth = 0
|
|
|
+ };
|
|
|
+
|
|
|
+ var firstImg = SKBitmap.Decode(images[0].FullName);
|
|
|
+ if (firstImg == null)
|
|
|
+ {
|
|
|
+ throw new InvalidDataException("Could not decode image data.");
|
|
|
+ }
|
|
|
|
|
|
- /*
|
|
|
- * Generate grids of trickplay image tiles
|
|
|
- */
|
|
|
- var imgNo = 0;
|
|
|
- var i = 0;
|
|
|
- while (i < images.Count)
|
|
|
- {
|
|
|
- var tileGrid = new SKBitmap(tilesInfo.Width * tilesInfo.TileWidth, tilesInfo.Height * tilesInfo.TileHeight);
|
|
|
+ tilesInfo.Height = firstImg.Height;
|
|
|
+ if (tilesInfo.Width != firstImg.Width)
|
|
|
+ {
|
|
|
+ throw new InvalidOperationException("Image width does not match config width.");
|
|
|
+ }
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Generate grids of trickplay image tiles
|
|
|
+ */
|
|
|
+ var imgNo = 0;
|
|
|
+ var i = 0;
|
|
|
+ while (i < images.Count)
|
|
|
+ {
|
|
|
+ var tileGrid = new SKBitmap(tilesInfo.Width * tilesInfo.TileWidth, tilesInfo.Height * tilesInfo.TileHeight);
|
|
|
|
|
|
- using (var canvas = new SKCanvas(tileGrid))
|
|
|
+ using (var canvas = new SKCanvas(tileGrid))
|
|
|
+ {
|
|
|
+ for (var y = 0; y < tilesInfo.TileHeight; y++)
|
|
|
{
|
|
|
- for (var y = 0; y < tilesInfo.TileHeight; y++)
|
|
|
+ for (var x = 0; x < tilesInfo.TileWidth; x++)
|
|
|
{
|
|
|
- for (var x = 0; x < tilesInfo.TileWidth; x++)
|
|
|
+ if (i >= images.Count)
|
|
|
{
|
|
|
- if (i >= images.Count)
|
|
|
- {
|
|
|
- break;
|
|
|
- }
|
|
|
-
|
|
|
- var img = SKBitmap.Decode(images[i].FullName);
|
|
|
- if (img == null)
|
|
|
- {
|
|
|
- throw new InvalidDataException("Could not decode image data.");
|
|
|
- }
|
|
|
-
|
|
|
- if (tilesInfo.Width != img.Width)
|
|
|
- {
|
|
|
- throw new InvalidOperationException("Image width does not match config width.");
|
|
|
- }
|
|
|
-
|
|
|
- if (tilesInfo.Height != img.Height)
|
|
|
- {
|
|
|
- throw new InvalidOperationException("Image height does not match first image height.");
|
|
|
- }
|
|
|
-
|
|
|
- canvas.DrawBitmap(img, x * tilesInfo.Width, y * tilesInfo.Height);
|
|
|
- tilesInfo.TileCount++;
|
|
|
- i++;
|
|
|
+ break;
|
|
|
}
|
|
|
- }
|
|
|
- }
|
|
|
|
|
|
- // Output each tile grid to singular file
|
|
|
- var tileGridPath = Path.Combine(workDir, $"{imgNo}.jpg");
|
|
|
- using (var stream = File.OpenWrite(tileGridPath))
|
|
|
- {
|
|
|
- tileGrid.Encode(stream, SKEncodedImageFormat.Jpeg, options.JpegQuality);
|
|
|
- }
|
|
|
+ var img = SKBitmap.Decode(images[i].FullName);
|
|
|
+ if (img == null)
|
|
|
+ {
|
|
|
+ throw new InvalidDataException("Could not decode image data.");
|
|
|
+ }
|
|
|
+
|
|
|
+ if (tilesInfo.Width != img.Width)
|
|
|
+ {
|
|
|
+ throw new InvalidOperationException("Image width does not match config width.");
|
|
|
+ }
|
|
|
|
|
|
- var bitrate = (int)Math.Ceiling((decimal)new FileInfo(tileGridPath).Length * 8 / tilesInfo.TileWidth / tilesInfo.TileHeight / (tilesInfo.Interval / 1000));
|
|
|
- tilesInfo.Bandwidth = Math.Max(tilesInfo.Bandwidth, bitrate);
|
|
|
+ if (tilesInfo.Height != img.Height)
|
|
|
+ {
|
|
|
+ throw new InvalidOperationException("Image height does not match first image height.");
|
|
|
+ }
|
|
|
|
|
|
- imgNo++;
|
|
|
+ canvas.DrawBitmap(img, x * tilesInfo.Width, y * tilesInfo.Height);
|
|
|
+ tilesInfo.TileCount++;
|
|
|
+ i++;
|
|
|
+ }
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
- /*
|
|
|
- * Move trickplay tiles to output directory
|
|
|
- */
|
|
|
- Directory.CreateDirectory(outputDir);
|
|
|
-
|
|
|
- // Replace existing tile grids if they already exist
|
|
|
- if (Directory.Exists(outputDir))
|
|
|
+ // Output each tile grid to singular file
|
|
|
+ var tileGridPath = Path.Combine(workDir, $"{imgNo}.jpg");
|
|
|
+ using (var stream = File.OpenWrite(tileGridPath))
|
|
|
{
|
|
|
- Directory.Delete(outputDir, true);
|
|
|
+ tileGrid.Encode(stream, SKEncodedImageFormat.Jpeg, options.JpegQuality);
|
|
|
}
|
|
|
|
|
|
- MoveDirectory(workDir, outputDir);
|
|
|
+ var bitrate = (int)Math.Ceiling((decimal)new FileInfo(tileGridPath).Length * 8 / tilesInfo.TileWidth / tilesInfo.TileHeight / (tilesInfo.Interval / 1000));
|
|
|
+ tilesInfo.Bandwidth = Math.Max(tilesInfo.Bandwidth, bitrate);
|
|
|
|
|
|
- return tilesInfo;
|
|
|
+ imgNo++;
|
|
|
}
|
|
|
|
|
|
- private bool CanGenerateTrickplay(Video video, int interval)
|
|
|
- {
|
|
|
- var videoType = video.VideoType;
|
|
|
- if (videoType == VideoType.Iso || videoType == VideoType.Dvd || videoType == VideoType.BluRay)
|
|
|
- {
|
|
|
- return false;
|
|
|
- }
|
|
|
-
|
|
|
- if (video.IsPlaceHolder)
|
|
|
- {
|
|
|
- return false;
|
|
|
- }
|
|
|
+ /*
|
|
|
+ * Move trickplay tiles to output directory
|
|
|
+ */
|
|
|
+ Directory.CreateDirectory(outputDir);
|
|
|
|
|
|
- if (video.IsShortcut)
|
|
|
- {
|
|
|
- return false;
|
|
|
- }
|
|
|
+ // Replace existing tile grids if they already exist
|
|
|
+ if (Directory.Exists(outputDir))
|
|
|
+ {
|
|
|
+ Directory.Delete(outputDir, true);
|
|
|
+ }
|
|
|
|
|
|
- if (!video.IsCompleteMedia)
|
|
|
- {
|
|
|
- return false;
|
|
|
- }
|
|
|
+ MoveDirectory(workDir, outputDir);
|
|
|
|
|
|
- if (!video.RunTimeTicks.HasValue || video.RunTimeTicks.Value < TimeSpan.FromMilliseconds(interval).Ticks)
|
|
|
- {
|
|
|
- return false;
|
|
|
- }
|
|
|
-
|
|
|
- var libraryOptions = _libraryManager.GetLibraryOptions(video);
|
|
|
- if (libraryOptions is not null)
|
|
|
- {
|
|
|
- if (!libraryOptions.EnableTrickplayImageExtraction)
|
|
|
- {
|
|
|
- return false;
|
|
|
- }
|
|
|
- }
|
|
|
- else
|
|
|
- {
|
|
|
- return false;
|
|
|
- }
|
|
|
+ return tilesInfo;
|
|
|
+ }
|
|
|
|
|
|
- // Can't extract images if there are no video streams
|
|
|
- return video.GetMediaStreams().Count > 0;
|
|
|
+ private bool CanGenerateTrickplay(Video video, int interval)
|
|
|
+ {
|
|
|
+ var videoType = video.VideoType;
|
|
|
+ if (videoType == VideoType.Iso || videoType == VideoType.Dvd || videoType == VideoType.BluRay)
|
|
|
+ {
|
|
|
+ return false;
|
|
|
}
|
|
|
|
|
|
- /// <inheritdoc />
|
|
|
- public Dictionary<int, TrickplayTilesInfo> GetTilesResolutions(Guid itemId)
|
|
|
+ if (video.IsPlaceHolder)
|
|
|
{
|
|
|
- return _itemRepo.GetTilesResolutions(itemId);
|
|
|
+ return false;
|
|
|
}
|
|
|
|
|
|
- /// <inheritdoc />
|
|
|
- public void SaveTilesInfo(Guid itemId, TrickplayTilesInfo tilesInfo)
|
|
|
+ if (video.IsShortcut)
|
|
|
{
|
|
|
- _itemRepo.SaveTilesInfo(itemId, tilesInfo);
|
|
|
+ return false;
|
|
|
}
|
|
|
|
|
|
- /// <inheritdoc />
|
|
|
- public Dictionary<Guid, Dictionary<int, TrickplayTilesInfo>> GetTrickplayManifest(BaseItem item)
|
|
|
+ if (!video.IsCompleteMedia)
|
|
|
{
|
|
|
- return _itemRepo.GetTrickplayManifest(item);
|
|
|
+ return false;
|
|
|
}
|
|
|
|
|
|
- /// <inheritdoc />
|
|
|
- public string GetTrickplayTilePath(BaseItem item, int width, int index)
|
|
|
+ if (!video.RunTimeTicks.HasValue || video.RunTimeTicks.Value < TimeSpan.FromMilliseconds(interval).Ticks)
|
|
|
{
|
|
|
- return Path.Combine(GetTrickplayDirectory(item, width), index + ".jpg");
|
|
|
+ return false;
|
|
|
}
|
|
|
|
|
|
- private string GetTrickplayDirectory(BaseItem item, int? width = null)
|
|
|
+ var libraryOptions = _libraryManager.GetLibraryOptions(video);
|
|
|
+ if (libraryOptions is not null)
|
|
|
{
|
|
|
- var path = Path.Combine(item.GetInternalMetadataPath(), "trickplay");
|
|
|
-
|
|
|
- return width.HasValue ? Path.Combine(path, width.Value.ToString(CultureInfo.InvariantCulture)) : path;
|
|
|
+ if (!libraryOptions.EnableTrickplayImageExtraction)
|
|
|
+ {
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ return false;
|
|
|
}
|
|
|
|
|
|
- private void MoveDirectory(string source, string destination)
|
|
|
+ // Can't extract images if there are no video streams
|
|
|
+ return video.GetMediaStreams().Count > 0;
|
|
|
+ }
|
|
|
+
|
|
|
+ /// <inheritdoc />
|
|
|
+ public Dictionary<int, TrickplayTilesInfo> GetTilesResolutions(Guid itemId)
|
|
|
+ {
|
|
|
+ return _itemRepo.GetTilesResolutions(itemId);
|
|
|
+ }
|
|
|
+
|
|
|
+ /// <inheritdoc />
|
|
|
+ public void SaveTilesInfo(Guid itemId, TrickplayTilesInfo tilesInfo)
|
|
|
+ {
|
|
|
+ _itemRepo.SaveTilesInfo(itemId, tilesInfo);
|
|
|
+ }
|
|
|
+
|
|
|
+ /// <inheritdoc />
|
|
|
+ public Dictionary<Guid, Dictionary<int, TrickplayTilesInfo>> GetTrickplayManifest(BaseItem item)
|
|
|
+ {
|
|
|
+ return _itemRepo.GetTrickplayManifest(item);
|
|
|
+ }
|
|
|
+
|
|
|
+ /// <inheritdoc />
|
|
|
+ public string GetTrickplayTilePath(BaseItem item, int width, int index)
|
|
|
+ {
|
|
|
+ return Path.Combine(GetTrickplayDirectory(item, width), index + ".jpg");
|
|
|
+ }
|
|
|
+
|
|
|
+ private string GetTrickplayDirectory(BaseItem item, int? width = null)
|
|
|
+ {
|
|
|
+ var path = Path.Combine(item.GetInternalMetadataPath(), "trickplay");
|
|
|
+
|
|
|
+ return width.HasValue ? Path.Combine(path, width.Value.ToString(CultureInfo.InvariantCulture)) : path;
|
|
|
+ }
|
|
|
+
|
|
|
+ private void MoveDirectory(string source, string destination)
|
|
|
+ {
|
|
|
+ try
|
|
|
{
|
|
|
- try
|
|
|
+ Directory.Move(source, destination);
|
|
|
+ }
|
|
|
+ catch (IOException)
|
|
|
+ {
|
|
|
+ // Cross device move requires a copy
|
|
|
+ Directory.CreateDirectory(destination);
|
|
|
+ foreach (string file in Directory.GetFiles(source))
|
|
|
{
|
|
|
- Directory.Move(source, destination);
|
|
|
+ File.Copy(file, Path.Join(destination, Path.GetFileName(file)), true);
|
|
|
}
|
|
|
- catch (IOException)
|
|
|
- {
|
|
|
- // Cross device move requires a copy
|
|
|
- Directory.CreateDirectory(destination);
|
|
|
- foreach (string file in Directory.GetFiles(source))
|
|
|
- {
|
|
|
- File.Copy(file, Path.Join(destination, Path.GetFileName(file)), true);
|
|
|
- }
|
|
|
|
|
|
- Directory.Delete(source, true);
|
|
|
- }
|
|
|
+ Directory.Delete(source, true);
|
|
|
}
|
|
|
}
|
|
|
}
|