| | 1 | | #pragma warning disable CA1826 // CA1826 Do not use Enumerable methods on Indexable collections. |
| | 2 | |
|
| | 3 | | using System; |
| | 4 | | using System.Collections.Generic; |
| | 5 | | using System.Linq; |
| | 6 | | using System.Threading; |
| | 7 | | using System.Threading.Tasks; |
| | 8 | | using MediaBrowser.Controller.Entities; |
| | 9 | | using MediaBrowser.Controller.Library; |
| | 10 | | using MediaBrowser.Controller.MediaEncoding; |
| | 11 | | using MediaBrowser.Controller.Persistence; |
| | 12 | | using MediaBrowser.Controller.Providers; |
| | 13 | | using MediaBrowser.Model.Drawing; |
| | 14 | | using MediaBrowser.Model.Dto; |
| | 15 | | using MediaBrowser.Model.Entities; |
| | 16 | | using MediaBrowser.Model.MediaInfo; |
| | 17 | | using Microsoft.Extensions.Logging; |
| | 18 | |
|
| | 19 | | namespace MediaBrowser.Providers.MediaInfo |
| | 20 | | { |
| | 21 | | /// <summary> |
| | 22 | | /// Uses <see cref="IMediaEncoder"/> to create still images from the main video. |
| | 23 | | /// </summary> |
| | 24 | | public class VideoImageProvider : IDynamicImageProvider, IHasOrder |
| | 25 | | { |
| | 26 | | private readonly IMediaSourceManager _mediaSourceManager; |
| | 27 | | private readonly IMediaEncoder _mediaEncoder; |
| | 28 | | private readonly ILogger<VideoImageProvider> _logger; |
| | 29 | |
|
| | 30 | | /// <summary> |
| | 31 | | /// Initializes a new instance of the <see cref="VideoImageProvider"/> class. |
| | 32 | | /// </summary> |
| | 33 | | /// <param name="mediaSourceManager">The media source manager for fetching item streams.</param> |
| | 34 | | /// <param name="mediaEncoder">The media encoder for capturing images.</param> |
| | 35 | | /// <param name="logger">The logger.</param> |
| | 36 | | public VideoImageProvider(IMediaSourceManager mediaSourceManager, IMediaEncoder mediaEncoder, ILogger<VideoImage |
| | 37 | | { |
| 28 | 38 | | _mediaSourceManager = mediaSourceManager; |
| 28 | 39 | | _mediaEncoder = mediaEncoder; |
| 28 | 40 | | _logger = logger; |
| 28 | 41 | | } |
| | 42 | |
|
| | 43 | | /// <inheritdoc /> |
| 0 | 44 | | public string Name => "Screen Grabber"; |
| | 45 | |
|
| | 46 | | /// <inheritdoc /> |
| | 47 | | // Make sure this comes after internet image providers |
| 0 | 48 | | public int Order => 100; |
| | 49 | |
|
| | 50 | | /// <inheritdoc /> |
| | 51 | | public IEnumerable<ImageType> GetSupportedImages(BaseItem item) |
| | 52 | | { |
| 0 | 53 | | return new[] { ImageType.Primary }; |
| | 54 | | } |
| | 55 | |
|
| | 56 | | /// <inheritdoc /> |
| | 57 | | public Task<DynamicImageResponse> GetImage(BaseItem item, ImageType type, CancellationToken cancellationToken) |
| | 58 | | { |
| 7 | 59 | | var video = (Video)item; |
| | 60 | |
|
| | 61 | | // No support for these |
| 7 | 62 | | if (video.IsPlaceHolder || video.VideoType == VideoType.Dvd) |
| | 63 | | { |
| 1 | 64 | | return Task.FromResult(new DynamicImageResponse { HasImage = false }); |
| | 65 | | } |
| | 66 | |
|
| | 67 | | // Can't extract if we didn't find a video stream in the file |
| 6 | 68 | | if (!video.DefaultVideoStreamIndex.HasValue) |
| | 69 | | { |
| 1 | 70 | | _logger.LogInformation("Skipping image extraction due to missing DefaultVideoStreamIndex for {Path}.", v |
| 1 | 71 | | return Task.FromResult(new DynamicImageResponse { HasImage = false }); |
| | 72 | | } |
| | 73 | |
|
| 5 | 74 | | return GetVideoImage(video, cancellationToken); |
| | 75 | | } |
| | 76 | |
|
| | 77 | | private async Task<DynamicImageResponse> GetVideoImage(Video item, CancellationToken cancellationToken) |
| | 78 | | { |
| | 79 | | MediaSourceInfo mediaSource = new MediaSourceInfo |
| | 80 | | { |
| | 81 | | VideoType = item.VideoType, |
| | 82 | | IsoType = item.IsoType, |
| | 83 | | Protocol = item.PathProtocol ?? MediaProtocol.File, |
| | 84 | | }; |
| | 85 | |
|
| | 86 | | // If we know the duration, grab it from 10% into the video. Otherwise just 10 seconds in. |
| | 87 | | // Always use 10 seconds for dvd because our duration could be out of whack |
| | 88 | | var imageOffset = item.VideoType != VideoType.Dvd && item.RunTimeTicks > 0 |
| | 89 | | ? TimeSpan.FromTicks(item.RunTimeTicks.Value / 10) |
| | 90 | | : TimeSpan.FromSeconds(10); |
| | 91 | |
|
| | 92 | | var query = new MediaStreamQuery { ItemId = item.Id, Index = item.DefaultVideoStreamIndex }; |
| | 93 | | var videoStream = _mediaSourceManager.GetMediaStreams(query).FirstOrDefault(); |
| | 94 | | if (videoStream is null) |
| | 95 | | { |
| | 96 | | query.Type = MediaStreamType.Video; |
| | 97 | | query.Index = null; |
| | 98 | | videoStream = _mediaSourceManager.GetMediaStreams(query).FirstOrDefault(); |
| | 99 | | } |
| | 100 | |
|
| | 101 | | if (videoStream is null) |
| | 102 | | { |
| | 103 | | _logger.LogInformation("Skipping image extraction: no video stream found for {Path}.", item.Path ?? stri |
| | 104 | | return new DynamicImageResponse { HasImage = false }; |
| | 105 | | } |
| | 106 | |
|
| | 107 | | string extractedImagePath = await _mediaEncoder.ExtractVideoImage(item.Path, item.Container, mediaSource, vi |
| | 108 | |
|
| | 109 | | return new DynamicImageResponse |
| | 110 | | { |
| | 111 | | Format = ImageFormat.Jpg, |
| | 112 | | HasImage = true, |
| | 113 | | Path = extractedImagePath, |
| | 114 | | Protocol = MediaProtocol.File |
| | 115 | | }; |
| | 116 | | } |
| | 117 | |
|
| | 118 | | /// <inheritdoc /> |
| | 119 | | public bool Supports(BaseItem item) |
| | 120 | | { |
| 53 | 121 | | if (item.IsShortcut) |
| | 122 | | { |
| 0 | 123 | | return false; |
| | 124 | | } |
| | 125 | |
|
| 53 | 126 | | if (!item.IsFileProtocol) |
| | 127 | | { |
| 0 | 128 | | return false; |
| | 129 | | } |
| | 130 | |
|
| 53 | 131 | | return item is Video video && !video.IsPlaceHolder && video.IsCompleteMedia; |
| | 132 | | } |
| | 133 | | } |
| | 134 | | } |