123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292 |
- using MediaBrowser.Common.IO;
- using MediaBrowser.Controller.Entities;
- using MediaBrowser.Controller.Entities.Audio;
- using MediaBrowser.Controller.Entities.TV;
- using MediaBrowser.Controller.Playlists;
- using MediaBrowser.Controller.Providers;
- using MediaBrowser.Model.Drawing;
- using MediaBrowser.Model.Entities;
- using MoreLinq;
- using System;
- using System.Collections.Generic;
- using System.Drawing;
- using System.Drawing.Drawing2D;
- using System.Drawing.Imaging;
- using System.IO;
- using System.Linq;
- using System.Threading.Tasks;
- namespace MediaBrowser.Server.Implementations.Playlists
- {
- public class PlaylistImageEnhancer : IImageEnhancer
- {
- private readonly IFileSystem _fileSystem;
- public PlaylistImageEnhancer(IFileSystem fileSystem)
- {
- _fileSystem = fileSystem;
- }
- public bool Supports(IHasImages item, ImageType imageType)
- {
- return (imageType == ImageType.Primary || imageType == ImageType.Thumb) && item is Playlist;
- }
- public MetadataProviderPriority Priority
- {
- get { return MetadataProviderPriority.First; }
- }
- private List<BaseItem> GetItemsWithImages(IHasImages item)
- {
- var playlist = (Playlist)item;
- var items = playlist.GetManageableItems()
- .Select(i =>
- {
- var subItem = i.Item2;
- var episode = subItem as Episode;
- if (episode != null)
- {
- var series = episode.Series;
- if (series != null && series.HasImage(ImageType.Primary))
- {
- return series;
- }
- }
- if (subItem.HasImage(ImageType.Primary))
- {
- return subItem;
- }
- var parent = subItem.Parent;
- if (parent != null && parent.HasImage(ImageType.Primary))
- {
- if (parent is MusicAlbum)
- {
- return parent;
- }
- }
- return null;
- })
- .Where(i => i != null)
- .DistinctBy(i => i.Id)
- .ToList();
- // Rotate the images no more than once per day
- var random = new Random(DateTime.Now.DayOfYear).Next();
- return items
- .OrderBy(i => random - items.IndexOf(i))
- .Take(4)
- .OrderBy(i => i.Name)
- .ToList();
- }
- private const string Version = "3";
- public string GetConfigurationCacheKey(List<BaseItem> items)
- {
- return Version + "_" + string.Join(",", items.Select(i => i.Id.ToString("N")).ToArray());
- }
- public string GetConfigurationCacheKey(IHasImages item, ImageType imageType)
- {
- var items = GetItemsWithImages(item);
- return GetConfigurationCacheKey(items);
- }
- private const int SquareImageSize = 800;
- private const int ThumbImageWidth = 1600;
- private const int ThumbImageHeight = 900;
- public ImageSize GetEnhancedImageSize(IHasImages item, ImageType imageType, int imageIndex, ImageSize originalImageSize)
- {
- var items = GetItemsWithImages(item);
- if (items.Count == 0)
- {
- return originalImageSize;
- }
- if (imageType == ImageType.Thumb)
- {
- return new ImageSize
- {
- Height = ThumbImageHeight,
- Width = ThumbImageWidth
- };
- }
- return new ImageSize
- {
- Height = SquareImageSize,
- Width = SquareImageSize
- };
- }
- public async Task<Image> EnhanceImageAsync(IHasImages item, Image originalImage, ImageType imageType, int imageIndex)
- {
- var items = GetItemsWithImages(item);
- if (items.Count == 0)
- {
- return originalImage;
- }
- var img = imageType == ImageType.Thumb ?
- await GetThumbCollage(items).ConfigureAwait(false) :
- await GetSquareCollage(items).ConfigureAwait(false);
- using (originalImage)
- {
- return img;
- }
- }
- private Task<Image> GetThumbCollage(List<BaseItem> items)
- {
- return GetThumbCollage(items.Select(i => i.GetImagePath(ImageType.Primary)).ToList());
- }
- private Task<Image> GetSquareCollage(List<BaseItem> items)
- {
- return GetSquareCollage(items.Select(i => i.GetImagePath(ImageType.Primary)).ToList());
- }
- private async Task<Image> GetThumbCollage(List<string> files)
- {
- if (files.Count < 3)
- {
- return await GetSingleImage(files).ConfigureAwait(false);
- }
- const int rows = 1;
- const int cols = 3;
- const int cellWidth = 2 * (ThumbImageWidth / 3);
- const int cellHeight = ThumbImageHeight;
- var index = 0;
- var img = new Bitmap(ThumbImageWidth, ThumbImageHeight, PixelFormat.Format32bppPArgb);
- using (var graphics = Graphics.FromImage(img))
- {
- graphics.CompositingQuality = CompositingQuality.HighQuality;
- graphics.SmoothingMode = SmoothingMode.HighQuality;
- graphics.InterpolationMode = InterpolationMode.HighQualityBicubic;
- graphics.PixelOffsetMode = PixelOffsetMode.HighQuality;
- graphics.CompositingMode = CompositingMode.SourceCopy;
- for (var row = 0; row < rows; row++)
- {
- for (var col = 0; col < cols; col++)
- {
- var x = col * (cellWidth / 2);
- var y = row * cellHeight;
- if (files.Count > index)
- {
- using (var fileStream = _fileSystem.GetFileStream(files[index], FileMode.Open, FileAccess.Read, FileShare.Read, true))
- {
- using (var memoryStream = new MemoryStream())
- {
- await fileStream.CopyToAsync(memoryStream).ConfigureAwait(false);
- memoryStream.Position = 0;
- using (var imgtemp = Image.FromStream(memoryStream, true, false))
- {
- graphics.DrawImage(imgtemp, x, y, cellWidth, cellHeight);
- }
- }
- }
- }
- index++;
- }
- }
- }
- return img;
- }
- private async Task<Image> GetSquareCollage(List<string> files)
- {
- if (files.Count < 4)
- {
- return await GetSingleImage(files).ConfigureAwait(false);
- }
- const int rows = 2;
- const int cols = 2;
- const int singleSize = SquareImageSize / 2;
- var index = 0;
- var img = new Bitmap(SquareImageSize, SquareImageSize, PixelFormat.Format32bppPArgb);
- using (var graphics = Graphics.FromImage(img))
- {
- graphics.CompositingQuality = CompositingQuality.HighQuality;
- graphics.SmoothingMode = SmoothingMode.HighQuality;
- graphics.InterpolationMode = InterpolationMode.HighQualityBicubic;
- graphics.PixelOffsetMode = PixelOffsetMode.HighQuality;
- graphics.CompositingMode = CompositingMode.SourceCopy;
- for (var row = 0; row < rows; row++)
- {
- for (var col = 0; col < cols; col++)
- {
- var x = col * singleSize;
- var y = row * singleSize;
- using (var fileStream = _fileSystem.GetFileStream(files[index], FileMode.Open, FileAccess.Read, FileShare.Read, true))
- {
- using (var memoryStream = new MemoryStream())
- {
- await fileStream.CopyToAsync(memoryStream).ConfigureAwait(false);
- memoryStream.Position = 0;
- using (var imgtemp = Image.FromStream(memoryStream, true, false))
- {
- graphics.DrawImage(imgtemp, x, y, singleSize, singleSize);
- }
- }
- }
- index++;
- }
- }
- }
- return img;
- }
- private Task<Image> GetSingleImage(List<string> files)
- {
- return GetImage(files[0]);
- }
- private async Task<Image> GetImage(string file)
- {
- using (var fileStream = _fileSystem.GetFileStream(file, FileMode.Open, FileAccess.Read, FileShare.Read, true))
- {
- var memoryStream = new MemoryStream();
- await fileStream.CopyToAsync(memoryStream).ConfigureAwait(false);
- memoryStream.Position = 0;
- return Image.FromStream(memoryStream, true, false);
- }
- }
- }
- }
|