mirror of
https://github.com/jellyfin/jellyfin.git
synced 2024-11-15 18:08:53 -07:00
reduce image processing
This commit is contained in:
parent
7d7f9d7654
commit
81bb469fe1
@ -89,7 +89,7 @@ namespace Emby.Drawing.GDI
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void EncodeImage(string inputPath, string cacheFilePath, int width, int height, int quality, ImageProcessingOptions options)
|
public void EncodeImage(string inputPath, string cacheFilePath, int width, int height, int quality, ImageProcessingOptions options, ImageFormat selectedOutputFormat)
|
||||||
{
|
{
|
||||||
var hasPostProcessing = !string.IsNullOrEmpty(options.BackgroundColor) || options.UnplayedCount.HasValue || options.AddPlayedIndicator || options.PercentPlayed > 0;
|
var hasPostProcessing = !string.IsNullOrEmpty(options.BackgroundColor) || options.UnplayedCount.HasValue || options.AddPlayedIndicator || options.PercentPlayed > 0;
|
||||||
|
|
||||||
@ -98,8 +98,6 @@ namespace Emby.Drawing.GDI
|
|||||||
var newWidth = Convert.ToInt32(width);
|
var newWidth = Convert.ToInt32(width);
|
||||||
var newHeight = Convert.ToInt32(height);
|
var newHeight = Convert.ToInt32(height);
|
||||||
|
|
||||||
var selectedOutputFormat = options.OutputFormat;
|
|
||||||
|
|
||||||
// Graphics.FromImage will throw an exception if the PixelFormat is Indexed, so we need to handle that here
|
// Graphics.FromImage will throw an exception if the PixelFormat is Indexed, so we need to handle that here
|
||||||
// Also, Webp only supports Format32bppArgb and Format32bppRgb
|
// Also, Webp only supports Format32bppArgb and Format32bppRgb
|
||||||
var pixelFormat = selectedOutputFormat == ImageFormat.Webp
|
var pixelFormat = selectedOutputFormat == ImageFormat.Webp
|
||||||
|
@ -31,7 +31,8 @@ namespace Emby.Drawing
|
|||||||
/// <param name="height">The height.</param>
|
/// <param name="height">The height.</param>
|
||||||
/// <param name="quality">The quality.</param>
|
/// <param name="quality">The quality.</param>
|
||||||
/// <param name="options">The options.</param>
|
/// <param name="options">The options.</param>
|
||||||
void EncodeImage(string inputPath, string outputPath, int width, int height, int quality, ImageProcessingOptions options);
|
/// <param name="outputFormat">The output format.</param>
|
||||||
|
void EncodeImage(string inputPath, string outputPath, int width, int height, int quality, ImageProcessingOptions options, ImageFormat outputFormat);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Creates the image collage.
|
/// Creates the image collage.
|
||||||
|
@ -134,7 +134,7 @@ namespace Emby.Drawing.ImageMagick
|
|||||||
string.Equals(ext, ".webp", StringComparison.OrdinalIgnoreCase);
|
string.Equals(ext, ".webp", StringComparison.OrdinalIgnoreCase);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void EncodeImage(string inputPath, string outputPath, int width, int height, int quality, ImageProcessingOptions options)
|
public void EncodeImage(string inputPath, string outputPath, int width, int height, int quality, ImageProcessingOptions options, ImageFormat selectedOutputFormat)
|
||||||
{
|
{
|
||||||
// Even if the caller specified 100, don't use it because it takes forever
|
// Even if the caller specified 100, don't use it because it takes forever
|
||||||
quality = Math.Min(quality, 99);
|
quality = Math.Min(quality, 99);
|
||||||
|
@ -19,6 +19,7 @@ using System.Threading.Tasks;
|
|||||||
using CommonIO;
|
using CommonIO;
|
||||||
using Emby.Drawing.Common;
|
using Emby.Drawing.Common;
|
||||||
using MediaBrowser.Controller.Library;
|
using MediaBrowser.Controller.Library;
|
||||||
|
using MediaBrowser.Model.Net;
|
||||||
|
|
||||||
namespace Emby.Drawing
|
namespace Emby.Drawing
|
||||||
{
|
{
|
||||||
@ -152,7 +153,7 @@ namespace Emby.Drawing
|
|||||||
{
|
{
|
||||||
var file = await ProcessImage(options).ConfigureAwait(false);
|
var file = await ProcessImage(options).ConfigureAwait(false);
|
||||||
|
|
||||||
using (var fileStream = _fileSystem.GetFileStream(file, FileMode.Open, FileAccess.Read, FileShare.Read, true))
|
using (var fileStream = _fileSystem.GetFileStream(file.Item1, FileMode.Open, FileAccess.Read, FileShare.Read, true))
|
||||||
{
|
{
|
||||||
await fileStream.CopyToAsync(toStream).ConfigureAwait(false);
|
await fileStream.CopyToAsync(toStream).ConfigureAwait(false);
|
||||||
}
|
}
|
||||||
@ -163,7 +164,7 @@ namespace Emby.Drawing
|
|||||||
return _imageEncoder.SupportedOutputFormats;
|
return _imageEncoder.SupportedOutputFormats;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<string> ProcessImage(ImageProcessingOptions options)
|
public async Task<Tuple<string, string>> ProcessImage(ImageProcessingOptions options)
|
||||||
{
|
{
|
||||||
if (options == null)
|
if (options == null)
|
||||||
{
|
{
|
||||||
@ -181,13 +182,7 @@ namespace Emby.Drawing
|
|||||||
|
|
||||||
if (!_imageEncoder.SupportsImageEncoding)
|
if (!_imageEncoder.SupportsImageEncoding)
|
||||||
{
|
{
|
||||||
return originalImagePath;
|
return new Tuple<string, string>(originalImagePath, MimeTypes.GetMimeType(originalImagePath));
|
||||||
}
|
|
||||||
|
|
||||||
if (options.HasDefaultOptions(originalImagePath) && options.Enhancers.Count == 0 && !options.CropWhiteSpace)
|
|
||||||
{
|
|
||||||
// Just spit out the original file if all the options are default
|
|
||||||
return originalImagePath;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var dateModified = originalImage.DateModified;
|
var dateModified = originalImage.DateModified;
|
||||||
@ -214,19 +209,31 @@ namespace Emby.Drawing
|
|||||||
dateModified = tuple.Item2;
|
dateModified = tuple.Item2;
|
||||||
}
|
}
|
||||||
|
|
||||||
var newSizeInfo = GetNewImageSize(originalImagePath, dateModified, options);
|
if (options.HasDefaultOptions(originalImagePath))
|
||||||
var newSize = newSizeInfo.Item1;
|
|
||||||
var isSizeChanged = newSizeInfo.Item2;
|
|
||||||
|
|
||||||
if (options.HasDefaultOptionsWithoutSize(originalImagePath) && !isSizeChanged && options.Enhancers.Count == 0)
|
|
||||||
{
|
{
|
||||||
// Just spit out the original file if the new size equals the old
|
// Just spit out the original file if all the options are default
|
||||||
return originalImagePath;
|
return new Tuple<string, string>(originalImagePath, MimeTypes.GetMimeType(originalImagePath));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ImageSize? originalImageSize;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
originalImageSize = GetImageSize(originalImagePath, dateModified, true);
|
||||||
|
if (options.HasDefaultOptions(originalImagePath, originalImageSize.Value))
|
||||||
|
{
|
||||||
|
// Just spit out the original file if all the options are default
|
||||||
|
return new Tuple<string, string>(originalImagePath, MimeTypes.GetMimeType(originalImagePath));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
originalImageSize = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
var newSize = GetNewImageSize(options, originalImageSize);
|
||||||
var quality = options.Quality;
|
var quality = options.Quality;
|
||||||
|
|
||||||
var outputFormat = GetOutputFormat(options.OutputFormat);
|
var outputFormat = GetOutputFormat(options.SupportedOutputFormats[0]);
|
||||||
var cacheFilePath = GetCacheFilePath(originalImagePath, newSize, quality, dateModified, outputFormat, options.AddPlayedIndicator, options.PercentPlayed, options.UnplayedCount, options.BackgroundColor);
|
var cacheFilePath = GetCacheFilePath(originalImagePath, newSize, quality, dateModified, outputFormat, options.AddPlayedIndicator, options.PercentPlayed, options.UnplayedCount, options.BackgroundColor);
|
||||||
|
|
||||||
var semaphore = GetLock(cacheFilePath);
|
var semaphore = GetLock(cacheFilePath);
|
||||||
@ -250,7 +257,7 @@ namespace Emby.Drawing
|
|||||||
|
|
||||||
imageProcessingLockTaken = true;
|
imageProcessingLockTaken = true;
|
||||||
|
|
||||||
_imageEncoder.EncodeImage(originalImagePath, cacheFilePath, newWidth, newHeight, quality, options);
|
_imageEncoder.EncodeImage(originalImagePath, cacheFilePath, newWidth, newHeight, quality, options, outputFormat);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
@ -263,27 +270,48 @@ namespace Emby.Drawing
|
|||||||
semaphore.Release();
|
semaphore.Release();
|
||||||
}
|
}
|
||||||
|
|
||||||
return cacheFilePath;
|
return new Tuple<string, string>(cacheFilePath, GetMimeType(outputFormat, cacheFilePath));
|
||||||
}
|
}
|
||||||
|
|
||||||
private Tuple<ImageSize, bool> GetNewImageSize(string originalImagePath, DateTime dateModified, ImageProcessingOptions options)
|
private string GetMimeType(ImageFormat format, string path)
|
||||||
{
|
{
|
||||||
try
|
if (format == ImageFormat.Bmp)
|
||||||
{
|
{
|
||||||
var originalImageSize = GetImageSize(originalImagePath, dateModified, true);
|
return MimeTypes.GetMimeType("i.bmp");
|
||||||
|
|
||||||
// Determine the output size based on incoming parameters
|
|
||||||
var newSize = DrawingUtils.Resize(originalImageSize, options.Width, options.Height, options.MaxWidth, options.MaxHeight);
|
|
||||||
|
|
||||||
return new Tuple<ImageSize, bool>(newSize, !newSize.Equals(originalImageSize));
|
|
||||||
}
|
}
|
||||||
catch
|
if (format == ImageFormat.Gif)
|
||||||
{
|
{
|
||||||
return new Tuple<ImageSize, bool>(GetSizeEstimage(options), true);
|
return MimeTypes.GetMimeType("i.gif");
|
||||||
}
|
}
|
||||||
|
if (format == ImageFormat.Jpg)
|
||||||
|
{
|
||||||
|
return MimeTypes.GetMimeType("i.jpg");
|
||||||
|
}
|
||||||
|
if (format == ImageFormat.Png)
|
||||||
|
{
|
||||||
|
return MimeTypes.GetMimeType("i.png");
|
||||||
|
}
|
||||||
|
if (format == ImageFormat.Webp)
|
||||||
|
{
|
||||||
|
return MimeTypes.GetMimeType("i.webp");
|
||||||
|
}
|
||||||
|
|
||||||
|
return MimeTypes.GetMimeType(path);
|
||||||
}
|
}
|
||||||
|
|
||||||
private ImageSize GetSizeEstimage(ImageProcessingOptions options)
|
private ImageSize GetNewImageSize(ImageProcessingOptions options, ImageSize? originalImageSize)
|
||||||
|
{
|
||||||
|
if (originalImageSize.HasValue)
|
||||||
|
{
|
||||||
|
// Determine the output size based on incoming parameters
|
||||||
|
var newSize = DrawingUtils.Resize(originalImageSize.Value, options.Width, options.Height, options.MaxWidth, options.MaxHeight);
|
||||||
|
|
||||||
|
return newSize;
|
||||||
|
}
|
||||||
|
return GetSizeEstimate(options);
|
||||||
|
}
|
||||||
|
|
||||||
|
private ImageSize GetSizeEstimate(ImageProcessingOptions options)
|
||||||
{
|
{
|
||||||
if (options.Width.HasValue && options.Height.HasValue)
|
if (options.Width.HasValue && options.Height.HasValue)
|
||||||
{
|
{
|
||||||
|
@ -32,7 +32,7 @@ namespace Emby.Drawing
|
|||||||
throw new NotImplementedException();
|
throw new NotImplementedException();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void EncodeImage(string inputPath, string outputPath, int width, int height, int quality, ImageProcessingOptions options)
|
public void EncodeImage(string inputPath, string outputPath, int width, int height, int quality, ImageProcessingOptions options, ImageFormat selectedOutputFormat)
|
||||||
{
|
{
|
||||||
throw new NotImplementedException();
|
throw new NotImplementedException();
|
||||||
}
|
}
|
||||||
|
@ -571,8 +571,7 @@ namespace MediaBrowser.Api.Images
|
|||||||
cropwhitespace = request.CropWhitespace.Value;
|
cropwhitespace = request.CropWhitespace.Value;
|
||||||
}
|
}
|
||||||
|
|
||||||
var format = GetOutputFormat(request, imageInfo, cropwhitespace, supportedImageEnhancers);
|
var outputFormats = GetOutputFormats(request, imageInfo, cropwhitespace, supportedImageEnhancers);
|
||||||
var contentType = GetMimeType(format, imageInfo.Path);
|
|
||||||
|
|
||||||
var cacheGuid = new Guid(_imageProcessor.GetImageCacheTag(item, imageInfo, supportedImageEnhancers));
|
var cacheGuid = new Guid(_imageProcessor.GetImageCacheTag(item, imageInfo, supportedImageEnhancers));
|
||||||
|
|
||||||
@ -593,9 +592,8 @@ namespace MediaBrowser.Api.Images
|
|||||||
request,
|
request,
|
||||||
imageInfo,
|
imageInfo,
|
||||||
cropwhitespace,
|
cropwhitespace,
|
||||||
format,
|
outputFormats,
|
||||||
supportedImageEnhancers,
|
supportedImageEnhancers,
|
||||||
contentType,
|
|
||||||
cacheDuration,
|
cacheDuration,
|
||||||
responseHeaders,
|
responseHeaders,
|
||||||
isHeadRequest)
|
isHeadRequest)
|
||||||
@ -606,9 +604,8 @@ namespace MediaBrowser.Api.Images
|
|||||||
ImageRequest request,
|
ImageRequest request,
|
||||||
ItemImageInfo image,
|
ItemImageInfo image,
|
||||||
bool cropwhitespace,
|
bool cropwhitespace,
|
||||||
ImageFormat format,
|
List<ImageFormat> supportedFormats,
|
||||||
List<IImageEnhancer> enhancers,
|
List<IImageEnhancer> enhancers,
|
||||||
string contentType,
|
|
||||||
TimeSpan? cacheDuration,
|
TimeSpan? cacheDuration,
|
||||||
IDictionary<string, string> headers,
|
IDictionary<string, string> headers,
|
||||||
bool isHeadRequest)
|
bool isHeadRequest)
|
||||||
@ -629,10 +626,10 @@ namespace MediaBrowser.Api.Images
|
|||||||
PercentPlayed = request.PercentPlayed ?? 0,
|
PercentPlayed = request.PercentPlayed ?? 0,
|
||||||
UnplayedCount = request.UnplayedCount,
|
UnplayedCount = request.UnplayedCount,
|
||||||
BackgroundColor = request.BackgroundColor,
|
BackgroundColor = request.BackgroundColor,
|
||||||
OutputFormat = format
|
SupportedOutputFormats = supportedFormats
|
||||||
};
|
};
|
||||||
|
|
||||||
var file = await _imageProcessor.ProcessImage(options).ConfigureAwait(false);
|
var imageResult = await _imageProcessor.ProcessImage(options).ConfigureAwait(false);
|
||||||
|
|
||||||
headers["Vary"] = "Accept";
|
headers["Vary"] = "Accept";
|
||||||
|
|
||||||
@ -640,20 +637,20 @@ namespace MediaBrowser.Api.Images
|
|||||||
{
|
{
|
||||||
CacheDuration = cacheDuration,
|
CacheDuration = cacheDuration,
|
||||||
ResponseHeaders = headers,
|
ResponseHeaders = headers,
|
||||||
ContentType = contentType,
|
ContentType = imageResult.Item2,
|
||||||
IsHeadRequest = isHeadRequest,
|
IsHeadRequest = isHeadRequest,
|
||||||
Path = file
|
Path = imageResult.Item1
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private ImageFormat GetOutputFormat(ImageRequest request, ItemImageInfo image, bool cropwhitespace, List<IImageEnhancer> enhancers)
|
private List<ImageFormat> GetOutputFormats(ImageRequest request, ItemImageInfo image, bool cropwhitespace, List<IImageEnhancer> enhancers)
|
||||||
{
|
{
|
||||||
if (!string.IsNullOrWhiteSpace(request.Format))
|
if (!string.IsNullOrWhiteSpace(request.Format))
|
||||||
{
|
{
|
||||||
ImageFormat format;
|
ImageFormat format;
|
||||||
if (Enum.TryParse(request.Format, true, out format))
|
if (Enum.TryParse(request.Format, true, out format))
|
||||||
{
|
{
|
||||||
return format;
|
return new List<ImageFormat> { format };
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -671,39 +668,30 @@ namespace MediaBrowser.Api.Images
|
|||||||
}
|
}
|
||||||
|
|
||||||
var clientSupportedFormats = GetClientSupportedFormats();
|
var clientSupportedFormats = GetClientSupportedFormats();
|
||||||
if (inputFormat.HasValue && clientSupportedFormats.Contains(inputFormat.Value) && enhancers.Count == 0)
|
|
||||||
{
|
|
||||||
if ((request.Quality ?? 100) == 100 && !request.Height.HasValue && !request.Width.HasValue &&
|
|
||||||
!request.AddPlayedIndicator && !request.PercentPlayed.HasValue && !request.UnplayedCount.HasValue && string.IsNullOrWhiteSpace(request.BackgroundColor))
|
|
||||||
{
|
|
||||||
// TODO: Allow this when specfying max width/height if the value is in range
|
|
||||||
if (!cropwhitespace && !request.MaxHeight.HasValue && !request.MaxWidth.HasValue)
|
|
||||||
{
|
|
||||||
return inputFormat.Value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var serverFormats = _imageProcessor.GetSupportedImageOutputFormats();
|
var serverFormats = _imageProcessor.GetSupportedImageOutputFormats();
|
||||||
|
var outputFormats = new List<ImageFormat>();
|
||||||
|
|
||||||
// Client doesn't care about format, so start with webp if supported
|
// Client doesn't care about format, so start with webp if supported
|
||||||
if (serverFormats.Contains(ImageFormat.Webp) && clientSupportedFormats.Contains(ImageFormat.Webp))
|
if (serverFormats.Contains(ImageFormat.Webp) && clientSupportedFormats.Contains(ImageFormat.Webp))
|
||||||
{
|
{
|
||||||
return ImageFormat.Webp;
|
outputFormats.Add(ImageFormat.Webp);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (enhancers.Count > 0)
|
if (enhancers.Count > 0)
|
||||||
{
|
{
|
||||||
return ImageFormat.Png;
|
outputFormats.Add(ImageFormat.Png);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (inputFormat.HasValue && inputFormat.Value == ImageFormat.Jpg)
|
if (inputFormat.HasValue && inputFormat.Value == ImageFormat.Jpg)
|
||||||
{
|
{
|
||||||
return ImageFormat.Jpg;
|
outputFormats.Add(ImageFormat.Jpg);
|
||||||
}
|
}
|
||||||
|
|
||||||
// We can't predict if there will be transparency or not, so play it safe
|
// We can't predict if there will be transparency or not, so play it safe
|
||||||
return ImageFormat.Png;
|
outputFormats.Add(ImageFormat.Png);
|
||||||
|
|
||||||
|
return outputFormats;
|
||||||
}
|
}
|
||||||
|
|
||||||
private ImageFormat[] GetClientSupportedFormats()
|
private ImageFormat[] GetClientSupportedFormats()
|
||||||
@ -730,32 +718,6 @@ namespace MediaBrowser.Api.Images
|
|||||||
return new[] { ImageFormat.Jpg, ImageFormat.Png };
|
return new[] { ImageFormat.Jpg, ImageFormat.Png };
|
||||||
}
|
}
|
||||||
|
|
||||||
private string GetMimeType(ImageFormat format, string path)
|
|
||||||
{
|
|
||||||
if (format == ImageFormat.Bmp)
|
|
||||||
{
|
|
||||||
return MimeTypes.GetMimeType("i.bmp");
|
|
||||||
}
|
|
||||||
if (format == ImageFormat.Gif)
|
|
||||||
{
|
|
||||||
return MimeTypes.GetMimeType("i.gif");
|
|
||||||
}
|
|
||||||
if (format == ImageFormat.Jpg)
|
|
||||||
{
|
|
||||||
return MimeTypes.GetMimeType("i.jpg");
|
|
||||||
}
|
|
||||||
if (format == ImageFormat.Png)
|
|
||||||
{
|
|
||||||
return MimeTypes.GetMimeType("i.png");
|
|
||||||
}
|
|
||||||
if (format == ImageFormat.Webp)
|
|
||||||
{
|
|
||||||
return MimeTypes.GetMimeType("i.webp");
|
|
||||||
}
|
|
||||||
|
|
||||||
return MimeTypes.GetMimeType(path);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets the image path.
|
/// Gets the image path.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
using MediaBrowser.Controller.Entities;
|
using System;
|
||||||
|
using MediaBrowser.Controller.Entities;
|
||||||
using MediaBrowser.Controller.Providers;
|
using MediaBrowser.Controller.Providers;
|
||||||
using MediaBrowser.Model.Drawing;
|
using MediaBrowser.Model.Drawing;
|
||||||
using MediaBrowser.Model.Entities;
|
using MediaBrowser.Model.Entities;
|
||||||
@ -83,7 +84,7 @@ namespace MediaBrowser.Controller.Drawing
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="options">The options.</param>
|
/// <param name="options">The options.</param>
|
||||||
/// <returns>Task.</returns>
|
/// <returns>Task.</returns>
|
||||||
Task<string> ProcessImage(ImageProcessingOptions options);
|
Task<Tuple<string,string>> ProcessImage(ImageProcessingOptions options);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets the enhanced image.
|
/// Gets the enhanced image.
|
||||||
|
@ -4,6 +4,7 @@ using MediaBrowser.Model.Drawing;
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
|
using System.Linq;
|
||||||
|
|
||||||
namespace MediaBrowser.Controller.Drawing
|
namespace MediaBrowser.Controller.Drawing
|
||||||
{
|
{
|
||||||
@ -29,7 +30,7 @@ namespace MediaBrowser.Controller.Drawing
|
|||||||
|
|
||||||
public List<IImageEnhancer> Enhancers { get; set; }
|
public List<IImageEnhancer> Enhancers { get; set; }
|
||||||
|
|
||||||
public ImageFormat OutputFormat { get; set; }
|
public List<ImageFormat> SupportedOutputFormats { get; set; }
|
||||||
|
|
||||||
public bool AddPlayedIndicator { get; set; }
|
public bool AddPlayedIndicator { get; set; }
|
||||||
|
|
||||||
@ -48,19 +49,47 @@ namespace MediaBrowser.Controller.Drawing
|
|||||||
!MaxHeight.HasValue;
|
!MaxHeight.HasValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public bool HasDefaultOptions(string originalImagePath, ImageSize size)
|
||||||
|
{
|
||||||
|
if (!HasDefaultOptionsWithoutSize(originalImagePath))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Width.HasValue && !size.Width.Equals(Width.Value))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (Height.HasValue && !size.Height.Equals(Height.Value))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (MaxWidth.HasValue && size.Width > MaxWidth.Value)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (MaxHeight.HasValue && size.Height > MaxHeight.Value)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
public bool HasDefaultOptionsWithoutSize(string originalImagePath)
|
public bool HasDefaultOptionsWithoutSize(string originalImagePath)
|
||||||
{
|
{
|
||||||
return (Quality == 100) &&
|
return (Quality >= 90) &&
|
||||||
IsOutputFormatDefault(originalImagePath) &&
|
IsFormatSupported(originalImagePath) &&
|
||||||
!AddPlayedIndicator &&
|
!AddPlayedIndicator &&
|
||||||
PercentPlayed.Equals(0) &&
|
PercentPlayed.Equals(0) &&
|
||||||
!UnplayedCount.HasValue &&
|
!UnplayedCount.HasValue &&
|
||||||
string.IsNullOrEmpty(BackgroundColor);
|
string.IsNullOrEmpty(BackgroundColor);
|
||||||
}
|
}
|
||||||
|
|
||||||
private bool IsOutputFormatDefault(string originalImagePath)
|
private bool IsFormatSupported(string originalImagePath)
|
||||||
{
|
{
|
||||||
return string.Equals(Path.GetExtension(originalImagePath), "." + OutputFormat, StringComparison.OrdinalIgnoreCase);
|
var ext = Path.GetExtension(originalImagePath);
|
||||||
|
return SupportedOutputFormats.Any(outputFormat => string.Equals(ext, "." + outputFormat, StringComparison.OrdinalIgnoreCase));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -690,9 +690,10 @@ namespace MediaBrowser.Controller.Entities
|
|||||||
[IgnoreDataMember]
|
[IgnoreDataMember]
|
||||||
public int? ParentIndexNumber { get; set; }
|
public int? ParentIndexNumber { get; set; }
|
||||||
|
|
||||||
public virtual string GetOfficialRatingForComparison(bool inherit)
|
[IgnoreDataMember]
|
||||||
|
public string OfficialRatingForComparison
|
||||||
{
|
{
|
||||||
if (inherit)
|
get
|
||||||
{
|
{
|
||||||
if (!string.IsNullOrWhiteSpace(OfficialRating))
|
if (!string.IsNullOrWhiteSpace(OfficialRating))
|
||||||
{
|
{
|
||||||
@ -702,13 +703,11 @@ namespace MediaBrowser.Controller.Entities
|
|||||||
var parent = DisplayParent;
|
var parent = DisplayParent;
|
||||||
if (parent != null)
|
if (parent != null)
|
||||||
{
|
{
|
||||||
return parent.GetOfficialRatingForComparison(inherit);
|
return parent.OfficialRatingForComparison;
|
||||||
}
|
}
|
||||||
|
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
return OfficialRating;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
[IgnoreDataMember]
|
[IgnoreDataMember]
|
||||||
@ -1141,7 +1140,7 @@ namespace MediaBrowser.Controller.Entities
|
|||||||
|
|
||||||
if (string.IsNullOrWhiteSpace(rating))
|
if (string.IsNullOrWhiteSpace(rating))
|
||||||
{
|
{
|
||||||
rating = GetOfficialRatingForComparison(true);
|
rating = OfficialRatingForComparison;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (string.IsNullOrWhiteSpace(rating))
|
if (string.IsNullOrWhiteSpace(rating))
|
||||||
@ -1190,7 +1189,7 @@ namespace MediaBrowser.Controller.Entities
|
|||||||
|
|
||||||
if (string.IsNullOrWhiteSpace(rating))
|
if (string.IsNullOrWhiteSpace(rating))
|
||||||
{
|
{
|
||||||
rating = GetOfficialRatingForComparison(true);
|
rating = OfficialRatingForComparison;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (string.IsNullOrWhiteSpace(rating))
|
if (string.IsNullOrWhiteSpace(rating))
|
||||||
|
@ -125,7 +125,14 @@ namespace MediaBrowser.Controller.Entities
|
|||||||
case CollectionType.HomeVideos:
|
case CollectionType.HomeVideos:
|
||||||
case CollectionType.Games:
|
case CollectionType.Games:
|
||||||
case CollectionType.MusicVideos:
|
case CollectionType.MusicVideos:
|
||||||
|
case CollectionType.Music:
|
||||||
|
{
|
||||||
|
if (query.Recursive)
|
||||||
|
{
|
||||||
|
return GetResult(queryParent.GetRecursiveChildren(user, true), queryParent, query);
|
||||||
|
}
|
||||||
return GetResult(queryParent.GetChildren(user, true), queryParent, query);
|
return GetResult(queryParent.GetChildren(user, true), queryParent, query);
|
||||||
|
}
|
||||||
|
|
||||||
case CollectionType.Folders:
|
case CollectionType.Folders:
|
||||||
return GetResult(user.RootFolder.GetChildren(user, true), queryParent, query);
|
return GetResult(user.RootFolder.GetChildren(user, true), queryParent, query);
|
||||||
@ -139,36 +146,9 @@ namespace MediaBrowser.Controller.Entities
|
|||||||
case CollectionType.TvShows:
|
case CollectionType.TvShows:
|
||||||
return await GetTvView(queryParent, user, query).ConfigureAwait(false);
|
return await GetTvView(queryParent, user, query).ConfigureAwait(false);
|
||||||
|
|
||||||
case CollectionType.Music:
|
|
||||||
return await GetMusicFolders(queryParent, user, query).ConfigureAwait(false);
|
|
||||||
|
|
||||||
case CollectionType.Movies:
|
case CollectionType.Movies:
|
||||||
return await GetMovieFolders(queryParent, user, query).ConfigureAwait(false);
|
return await GetMovieFolders(queryParent, user, query).ConfigureAwait(false);
|
||||||
|
|
||||||
case SpecialFolder.MusicGenres:
|
|
||||||
return await GetMusicGenres(queryParent, user, query).ConfigureAwait(false);
|
|
||||||
|
|
||||||
case SpecialFolder.MusicGenre:
|
|
||||||
return await GetMusicGenreItems(queryParent, displayParent, user, query).ConfigureAwait(false);
|
|
||||||
|
|
||||||
case SpecialFolder.GameGenres:
|
|
||||||
return await GetGameGenres(queryParent, user, query).ConfigureAwait(false);
|
|
||||||
|
|
||||||
case SpecialFolder.GameGenre:
|
|
||||||
return await GetGameGenreItems(queryParent, displayParent, user, query).ConfigureAwait(false);
|
|
||||||
|
|
||||||
case SpecialFolder.GameSystems:
|
|
||||||
return GetGameSystems(queryParent, user, query);
|
|
||||||
|
|
||||||
case SpecialFolder.LatestGames:
|
|
||||||
return GetLatestGames(queryParent, user, query);
|
|
||||||
|
|
||||||
case SpecialFolder.RecentlyPlayedGames:
|
|
||||||
return GetRecentlyPlayedGames(queryParent, user, query);
|
|
||||||
|
|
||||||
case SpecialFolder.GameFavorites:
|
|
||||||
return GetFavoriteGames(queryParent, user, query);
|
|
||||||
|
|
||||||
case SpecialFolder.TvShowSeries:
|
case SpecialFolder.TvShowSeries:
|
||||||
return GetTvSeries(queryParent, user, query);
|
return GetTvSeries(queryParent, user, query);
|
||||||
|
|
||||||
@ -208,42 +188,12 @@ namespace MediaBrowser.Controller.Entities
|
|||||||
case SpecialFolder.MovieCollections:
|
case SpecialFolder.MovieCollections:
|
||||||
return GetMovieCollections(queryParent, user, query);
|
return GetMovieCollections(queryParent, user, query);
|
||||||
|
|
||||||
case SpecialFolder.MusicLatest:
|
|
||||||
return GetMusicLatest(queryParent, user, query);
|
|
||||||
|
|
||||||
case SpecialFolder.MusicPlaylists:
|
|
||||||
return await GetMusicPlaylists(queryParent, user, query).ConfigureAwait(false);
|
|
||||||
|
|
||||||
case SpecialFolder.MusicAlbums:
|
|
||||||
return GetMusicAlbums(queryParent, user, query);
|
|
||||||
|
|
||||||
case SpecialFolder.MusicAlbumArtists:
|
|
||||||
return GetMusicAlbumArtists(queryParent, user, query);
|
|
||||||
|
|
||||||
case SpecialFolder.MusicArtists:
|
|
||||||
return GetMusicArtists(queryParent, user, query);
|
|
||||||
|
|
||||||
case SpecialFolder.MusicSongs:
|
|
||||||
return GetMusicSongs(queryParent, user, query);
|
|
||||||
|
|
||||||
case SpecialFolder.TvFavoriteEpisodes:
|
case SpecialFolder.TvFavoriteEpisodes:
|
||||||
return GetFavoriteEpisodes(queryParent, user, query);
|
return GetFavoriteEpisodes(queryParent, user, query);
|
||||||
|
|
||||||
case SpecialFolder.TvFavoriteSeries:
|
case SpecialFolder.TvFavoriteSeries:
|
||||||
return GetFavoriteSeries(queryParent, user, query);
|
return GetFavoriteSeries(queryParent, user, query);
|
||||||
|
|
||||||
case SpecialFolder.MusicFavorites:
|
|
||||||
return await GetMusicFavorites(queryParent, user, query).ConfigureAwait(false);
|
|
||||||
|
|
||||||
case SpecialFolder.MusicFavoriteAlbums:
|
|
||||||
return GetFavoriteAlbums(queryParent, user, query);
|
|
||||||
|
|
||||||
case SpecialFolder.MusicFavoriteArtists:
|
|
||||||
return GetFavoriteArtists(queryParent, user, query);
|
|
||||||
|
|
||||||
case SpecialFolder.MusicFavoriteSongs:
|
|
||||||
return GetFavoriteSongs(queryParent, user, query);
|
|
||||||
|
|
||||||
default:
|
default:
|
||||||
{
|
{
|
||||||
if (queryParent is UserView)
|
if (queryParent is UserView)
|
||||||
@ -270,154 +220,6 @@ namespace MediaBrowser.Controller.Entities
|
|||||||
return 50;
|
return 50;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task<QueryResult<BaseItem>> GetMusicFolders(Folder parent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
if (query.Recursive)
|
|
||||||
{
|
|
||||||
var items = GetRecursiveChildren(parent, user, new[] { CollectionType.Music, CollectionType.MusicVideos }, i => FilterItem(i, query));
|
|
||||||
|
|
||||||
return PostFilterAndSort(items, parent, null, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
var list = new List<BaseItem>();
|
|
||||||
|
|
||||||
list.Add(await GetUserView(SpecialFolder.MusicLatest, "0", parent).ConfigureAwait(false));
|
|
||||||
list.Add(await GetUserView(SpecialFolder.MusicPlaylists, "1", parent).ConfigureAwait(false));
|
|
||||||
list.Add(await GetUserView(SpecialFolder.MusicAlbums, "2", parent).ConfigureAwait(false));
|
|
||||||
list.Add(await GetUserView(SpecialFolder.MusicAlbumArtists, "3", parent).ConfigureAwait(false));
|
|
||||||
//list.Add(await GetUserView(SpecialFolder.MusicArtists, user, "4", parent).ConfigureAwait(false));
|
|
||||||
list.Add(await GetUserView(SpecialFolder.MusicSongs, "5", parent).ConfigureAwait(false));
|
|
||||||
list.Add(await GetUserView(SpecialFolder.MusicGenres, "6", parent).ConfigureAwait(false));
|
|
||||||
list.Add(await GetUserView(SpecialFolder.MusicFavorites, "7", parent).ConfigureAwait(false));
|
|
||||||
|
|
||||||
return GetResult(list, parent, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task<QueryResult<BaseItem>> GetMusicFavorites(Folder parent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
var list = new List<BaseItem>();
|
|
||||||
|
|
||||||
list.Add(await GetUserView(SpecialFolder.MusicFavoriteAlbums, "0", parent).ConfigureAwait(false));
|
|
||||||
list.Add(await GetUserView(SpecialFolder.MusicFavoriteArtists, "1", parent).ConfigureAwait(false));
|
|
||||||
list.Add(await GetUserView(SpecialFolder.MusicFavoriteSongs, "2", parent).ConfigureAwait(false));
|
|
||||||
|
|
||||||
return GetResult(list, parent, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task<QueryResult<BaseItem>> GetMusicGenres(Folder parent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
var tasks = GetRecursiveChildren(parent, user, new[] { CollectionType.Music, CollectionType.MusicVideos })
|
|
||||||
.Where(i => !i.IsFolder)
|
|
||||||
.SelectMany(i => i.Genres)
|
|
||||||
.DistinctNames()
|
|
||||||
.Select(i =>
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
return _libraryManager.GetMusicGenre(i);
|
|
||||||
}
|
|
||||||
catch
|
|
||||||
{
|
|
||||||
// Full exception logged at lower levels
|
|
||||||
_logger.Error("Error getting genre");
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
})
|
|
||||||
.Where(i => i != null)
|
|
||||||
.Select(i => GetUserView(i.Name, SpecialFolder.MusicGenre, i.SortName, parent));
|
|
||||||
|
|
||||||
var genres = await Task.WhenAll(tasks).ConfigureAwait(false);
|
|
||||||
|
|
||||||
return GetResult(genres, parent, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task<QueryResult<BaseItem>> GetMusicGenreItems(Folder queryParent, Folder displayParent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
var items = GetRecursiveChildren(queryParent, user, new[] { CollectionType.Music, CollectionType.MusicVideos })
|
|
||||||
.Where(i => !i.IsFolder)
|
|
||||||
.Where(i => i.Genres.Contains(displayParent.Name, StringComparer.OrdinalIgnoreCase))
|
|
||||||
.OfType<IHasAlbumArtist>();
|
|
||||||
|
|
||||||
var artists = _libraryManager.GetAlbumArtists(items);
|
|
||||||
|
|
||||||
return GetResult(artists, queryParent, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private QueryResult<BaseItem> GetMusicAlbumArtists(Folder parent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
var items = GetRecursiveChildren(parent, user, new[] { CollectionType.Music, CollectionType.MusicVideos })
|
|
||||||
.Where(i => !i.IsFolder)
|
|
||||||
.OfType<IHasAlbumArtist>();
|
|
||||||
|
|
||||||
var artists = _libraryManager.GetAlbumArtists(items);
|
|
||||||
|
|
||||||
return GetResult(artists, parent, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private QueryResult<BaseItem> GetMusicArtists(Folder parent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
var items = GetRecursiveChildren(parent, user, new[] { CollectionType.Music, CollectionType.MusicVideos })
|
|
||||||
.Where(i => !i.IsFolder)
|
|
||||||
.OfType<IHasArtist>();
|
|
||||||
|
|
||||||
var artists = _libraryManager.GetArtists(items);
|
|
||||||
|
|
||||||
return GetResult(artists, parent, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private QueryResult<BaseItem> GetFavoriteArtists(Folder parent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
var items = GetRecursiveChildren(parent, user, new[] { CollectionType.Music, CollectionType.MusicVideos })
|
|
||||||
.Where(i => !i.IsFolder)
|
|
||||||
.OfType<IHasAlbumArtist>();
|
|
||||||
|
|
||||||
var artists = _libraryManager.GetAlbumArtists(items).Where(i => _userDataManager.GetUserData(user.Id, i.GetUserDataKey()).IsFavorite);
|
|
||||||
|
|
||||||
return GetResult(artists, parent, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private Task<QueryResult<BaseItem>> GetMusicPlaylists(Folder parent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
query.IncludeItemTypes = new[] { "Playlist" };
|
|
||||||
query.Recursive = true;
|
|
||||||
|
|
||||||
return parent.GetItems(query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private QueryResult<BaseItem> GetMusicAlbums(Folder parent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
var items = GetRecursiveChildren(parent, user, new[] { CollectionType.Music, CollectionType.MusicVideos }, i => (i is MusicAlbum) && FilterItem(i, query));
|
|
||||||
|
|
||||||
return PostFilterAndSort(items, parent, null, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private QueryResult<BaseItem> GetMusicSongs(Folder parent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
var items = GetRecursiveChildren(parent, user, new[] { CollectionType.Music, CollectionType.MusicVideos }, i => (i is Audio.Audio) && FilterItem(i, query));
|
|
||||||
|
|
||||||
return PostFilterAndSort(items, parent, null, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private QueryResult<BaseItem> GetMusicLatest(Folder parent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
var items = _userViewManager.GetLatestItems(new LatestItemsQuery
|
|
||||||
{
|
|
||||||
UserId = user.Id.ToString("N"),
|
|
||||||
Limit = GetSpecialItemsLimit(),
|
|
||||||
IncludeItemTypes = new[] { typeof(Audio.Audio).Name },
|
|
||||||
ParentId = (parent == null ? null : parent.Id.ToString("N")),
|
|
||||||
GroupItems = true
|
|
||||||
|
|
||||||
}).Select(i => i.Item1 ?? i.Item2.FirstOrDefault()).Where(i => i != null);
|
|
||||||
|
|
||||||
query.SortBy = new string[] { };
|
|
||||||
|
|
||||||
//var items = GetRecursiveChildren(parent, user, new[] { CollectionType.Music, CollectionType.MusicVideos }, i => i is MusicVideo || i is Audio.Audio && FilterItem(i, query));
|
|
||||||
|
|
||||||
return PostFilterAndSort(items, parent, null, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task<QueryResult<BaseItem>> GetMovieFolders(Folder parent, User user, InternalItemsQuery query)
|
private async Task<QueryResult<BaseItem>> GetMovieFolders(Folder parent, User user, InternalItemsQuery query)
|
||||||
{
|
{
|
||||||
if (query.Recursive)
|
if (query.Recursive)
|
||||||
@ -476,24 +278,6 @@ namespace MediaBrowser.Controller.Entities
|
|||||||
return PostFilterAndSort(items, parent, null, query);
|
return PostFilterAndSort(items, parent, null, query);
|
||||||
}
|
}
|
||||||
|
|
||||||
private QueryResult<BaseItem> GetFavoriteSongs(Folder parent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
query.IsFavorite = true;
|
|
||||||
|
|
||||||
var items = GetRecursiveChildren(parent, user, new[] { CollectionType.Music }, i => (i is Audio.Audio) && FilterItem(i, query));
|
|
||||||
|
|
||||||
return PostFilterAndSort(items, parent, null, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private QueryResult<BaseItem> GetFavoriteAlbums(Folder parent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
query.IsFavorite = true;
|
|
||||||
|
|
||||||
var items = GetRecursiveChildren(parent, user, new[] { CollectionType.Music }, i => (i is MusicAlbum) && FilterItem(i, query));
|
|
||||||
|
|
||||||
return PostFilterAndSort(items, parent, null, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private QueryResult<BaseItem> GetMovieMovies(Folder parent, User user, InternalItemsQuery query)
|
private QueryResult<BaseItem> GetMovieMovies(Folder parent, User user, InternalItemsQuery query)
|
||||||
{
|
{
|
||||||
var items = GetRecursiveChildren(parent, user, new[] { CollectionType.Movies, CollectionType.BoxSets, string.Empty }, i => (i is Movie) && FilterItem(i, query));
|
var items = GetRecursiveChildren(parent, user, new[] { CollectionType.Movies, CollectionType.BoxSets, string.Empty }, i => (i is Movie) && FilterItem(i, query));
|
||||||
@ -613,54 +397,6 @@ namespace MediaBrowser.Controller.Entities
|
|||||||
return GetResult(list, parent, query);
|
return GetResult(list, parent, query);
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task<QueryResult<BaseItem>> GetGameView(User user, Folder parent, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
if (query.Recursive)
|
|
||||||
{
|
|
||||||
var items = GetRecursiveChildren(parent, user, new[] { CollectionType.Games }, i => FilterItem(i, query));
|
|
||||||
return PostFilterAndSort(items, parent, null, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
var list = new List<BaseItem>();
|
|
||||||
|
|
||||||
list.Add(await GetUserView(SpecialFolder.LatestGames, "0", parent).ConfigureAwait(false));
|
|
||||||
list.Add(await GetUserView(SpecialFolder.RecentlyPlayedGames, "1", parent).ConfigureAwait(false));
|
|
||||||
list.Add(await GetUserView(SpecialFolder.GameFavorites, "2", parent).ConfigureAwait(false));
|
|
||||||
list.Add(await GetUserView(SpecialFolder.GameSystems, "3", parent).ConfigureAwait(false));
|
|
||||||
list.Add(await GetUserView(SpecialFolder.GameGenres, "4", parent).ConfigureAwait(false));
|
|
||||||
|
|
||||||
return GetResult(list, parent, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private QueryResult<BaseItem> GetLatestGames(Folder parent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
query.SortBy = new[] { ItemSortBy.DateCreated, ItemSortBy.SortName };
|
|
||||||
query.SortOrder = SortOrder.Descending;
|
|
||||||
|
|
||||||
var items = GetRecursiveChildren(parent, user, new[] { CollectionType.Games }, i => i is Game && FilterItem(i, query));
|
|
||||||
|
|
||||||
return PostFilterAndSort(items, parent, GetSpecialItemsLimit(), query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private QueryResult<BaseItem> GetRecentlyPlayedGames(Folder parent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
query.IsPlayed = true;
|
|
||||||
query.SortBy = new[] { ItemSortBy.DatePlayed, ItemSortBy.SortName };
|
|
||||||
query.SortOrder = SortOrder.Descending;
|
|
||||||
|
|
||||||
var items = GetRecursiveChildren(parent, user, new[] { CollectionType.Games }, i => i is Game && FilterItem(i, query));
|
|
||||||
|
|
||||||
return PostFilterAndSort(items, parent, GetSpecialItemsLimit(), query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private QueryResult<BaseItem> GetFavoriteGames(Folder parent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
query.IsFavorite = true;
|
|
||||||
|
|
||||||
var items = GetRecursiveChildren(parent, user, new[] { CollectionType.Games }, i => i is Game && FilterItem(i, query));
|
|
||||||
return PostFilterAndSort(items, parent, null, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private QueryResult<BaseItem> GetTvLatest(Folder parent, User user, InternalItemsQuery query)
|
private QueryResult<BaseItem> GetTvLatest(Folder parent, User user, InternalItemsQuery query)
|
||||||
{
|
{
|
||||||
query.SortBy = new[] { ItemSortBy.DateCreated, ItemSortBy.SortName };
|
query.SortBy = new[] { ItemSortBy.DateCreated, ItemSortBy.SortName };
|
||||||
@ -741,49 +477,6 @@ namespace MediaBrowser.Controller.Entities
|
|||||||
return GetResult(items, queryParent, query);
|
return GetResult(items, queryParent, query);
|
||||||
}
|
}
|
||||||
|
|
||||||
private QueryResult<BaseItem> GetGameSystems(Folder parent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
var items = GetRecursiveChildren(parent, user, new[] { CollectionType.Games }, i => i is GameSystem && FilterItem(i, query));
|
|
||||||
|
|
||||||
return PostFilterAndSort(items, parent, null, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task<QueryResult<BaseItem>> GetGameGenreItems(Folder queryParent, Folder displayParent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
var items = GetRecursiveChildren(queryParent, user, new[] { CollectionType.Games },
|
|
||||||
i => i is Game && i.Genres.Contains(displayParent.Name, StringComparer.OrdinalIgnoreCase));
|
|
||||||
|
|
||||||
return GetResult(items, queryParent, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task<QueryResult<BaseItem>> GetGameGenres(Folder parent, User user, InternalItemsQuery query)
|
|
||||||
{
|
|
||||||
var tasks = GetRecursiveChildren(parent, user, new[] { CollectionType.Games })
|
|
||||||
.OfType<Game>()
|
|
||||||
.SelectMany(i => i.Genres)
|
|
||||||
.DistinctNames()
|
|
||||||
.Select(i =>
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
return _libraryManager.GetGameGenre(i);
|
|
||||||
}
|
|
||||||
catch
|
|
||||||
{
|
|
||||||
// Full exception logged at lower levels
|
|
||||||
_logger.Error("Error getting game genre");
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
})
|
|
||||||
.Where(i => i != null)
|
|
||||||
.Select(i => GetUserView(i.Name, SpecialFolder.GameGenre, i.SortName, parent));
|
|
||||||
|
|
||||||
var genres = await Task.WhenAll(tasks).ConfigureAwait(false);
|
|
||||||
|
|
||||||
return GetResult(genres, parent, query);
|
|
||||||
}
|
|
||||||
|
|
||||||
private QueryResult<BaseItem> GetResult<T>(QueryResult<T> result)
|
private QueryResult<BaseItem> GetResult<T>(QueryResult<T> result)
|
||||||
where T : BaseItem
|
where T : BaseItem
|
||||||
{
|
{
|
||||||
|
@ -47,25 +47,5 @@
|
|||||||
public const string MovieFavorites = "MovieFavorites";
|
public const string MovieFavorites = "MovieFavorites";
|
||||||
public const string MovieGenres = "MovieGenres";
|
public const string MovieGenres = "MovieGenres";
|
||||||
public const string MovieGenre = "MovieGenre";
|
public const string MovieGenre = "MovieGenre";
|
||||||
|
|
||||||
public const string LatestGames = "LatestGames";
|
|
||||||
public const string RecentlyPlayedGames = "RecentlyPlayedGames";
|
|
||||||
public const string GameSystems = "GameSystems";
|
|
||||||
public const string GameGenres = "GameGenres";
|
|
||||||
public const string GameFavorites = "GameFavorites";
|
|
||||||
public const string GameGenre = "GameGenre";
|
|
||||||
|
|
||||||
public const string MusicArtists = "MusicArtists";
|
|
||||||
public const string MusicAlbumArtists = "MusicAlbumArtists";
|
|
||||||
public const string MusicAlbums = "MusicAlbums";
|
|
||||||
public const string MusicGenres = "MusicGenres";
|
|
||||||
public const string MusicGenre = "MusicGenre";
|
|
||||||
public const string MusicLatest = "MusicLatest";
|
|
||||||
public const string MusicPlaylists = "MusicPlaylists";
|
|
||||||
public const string MusicSongs = "MusicSongs";
|
|
||||||
public const string MusicFavorites = "MusicFavorites";
|
|
||||||
public const string MusicFavoriteArtists = "MusicFavoriteArtists";
|
|
||||||
public const string MusicFavoriteAlbums = "MusicFavoriteAlbums";
|
|
||||||
public const string MusicFavoriteSongs = "MusicFavoriteSongs";
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -34,6 +34,8 @@
|
|||||||
DisplayContent = 26,
|
DisplayContent = 26,
|
||||||
GoToSearch = 27,
|
GoToSearch = 27,
|
||||||
DisplayMessage = 28,
|
DisplayMessage = 28,
|
||||||
SetRepeatMode = 29
|
SetRepeatMode = 29,
|
||||||
|
ChannelUp = 30,
|
||||||
|
ChannelDown = 31
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -527,6 +527,11 @@ namespace MediaBrowser.Providers.Manager
|
|||||||
{
|
{
|
||||||
var newIndex = item.AllowsMultipleImages(imageType) ? item.GetImages(imageType).Count() : 0;
|
var newIndex = item.AllowsMultipleImages(imageType) ? item.GetImages(imageType).Count() : 0;
|
||||||
|
|
||||||
|
SaveImageStub(item, imageType, url, newIndex);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void SaveImageStub(IHasImages item, ImageType imageType, string url, int newIndex)
|
||||||
|
{
|
||||||
item.SetImage(new ItemImageInfo
|
item.SetImage(new ItemImageInfo
|
||||||
{
|
{
|
||||||
Path = url,
|
Path = url,
|
||||||
@ -555,7 +560,7 @@ namespace MediaBrowser.Providers.Manager
|
|||||||
{
|
{
|
||||||
SaveImageStub(item, imageType, url);
|
SaveImageStub(item, imageType, url);
|
||||||
result.UpdateType = result.UpdateType | ItemUpdateType.ImageUpdate;
|
result.UpdateType = result.UpdateType | ItemUpdateType.ImageUpdate;
|
||||||
return;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
try
|
try
|
||||||
|
@ -54,9 +54,7 @@ namespace MediaBrowser.Server.Implementations.UserViews
|
|||||||
return new List<BaseItem>();
|
return new List<BaseItem>();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (string.Equals(view.ViewType, SpecialFolder.GameGenre, StringComparison.OrdinalIgnoreCase) ||
|
if (string.Equals(view.ViewType, SpecialFolder.MovieGenre, StringComparison.OrdinalIgnoreCase) ||
|
||||||
string.Equals(view.ViewType, SpecialFolder.MusicGenre, StringComparison.OrdinalIgnoreCase) ||
|
|
||||||
string.Equals(view.ViewType, SpecialFolder.MovieGenre, StringComparison.OrdinalIgnoreCase) ||
|
|
||||||
string.Equals(view.ViewType, SpecialFolder.TvGenre, StringComparison.OrdinalIgnoreCase))
|
string.Equals(view.ViewType, SpecialFolder.TvGenre, StringComparison.OrdinalIgnoreCase))
|
||||||
{
|
{
|
||||||
var userItemsResult = await view.GetItems(new InternalItemsQuery
|
var userItemsResult = await view.GetItems(new InternalItemsQuery
|
||||||
|
Loading…
Reference in New Issue
Block a user