using System;
using System.Collections.Generic;
using MediaBrowser.Controller.Drawing;
using SkiaSharp;
namespace Jellyfin.Drawing.Skia
{
///
/// Used to build the splashscreen.
///
public class SplashscreenBuilder
{
private const int Rows = 6;
private const int Spacing = 20;
private readonly SkiaEncoder _skiaEncoder;
private Random? _random;
private int _finalWidth;
private int _finalHeight;
///
/// Initializes a new instance of the class.
///
/// The SkiaEncoder.
public SplashscreenBuilder(SkiaEncoder skiaEncoder)
{
_skiaEncoder = skiaEncoder;
}
///
/// Generate a splashscreen.
///
/// The options to generate the splashscreen.
public void GenerateSplash(SplashscreenOptions options)
{
_finalWidth = options.Width;
_finalHeight = options.Height;
var wall = GenerateCollage(options.PortraitInputPaths, options.LandscapeInputPaths, options.ApplyFilter);
var transformed = Transform3D(wall);
using var outputStream = new SKFileWStream(options.OutputPath);
using var pixmap = new SKPixmap(new SKImageInfo(_finalWidth, _finalHeight), transformed.GetPixels());
pixmap.Encode(outputStream, StripCollageBuilder.GetEncodedFormat(options.OutputPath), 90);
}
///
/// Generates a collage of posters and landscape pictures.
///
/// The poster paths.
/// The landscape paths.
/// Whether to apply the darkening filter.
/// The created collage as a bitmap.
private SKBitmap GenerateCollage(IReadOnlyList poster, IReadOnlyList backdrop, bool applyFilter)
{
_random = new Random();
var posterIndex = 0;
var backdropIndex = 0;
// use higher resolution than final image
var bitmap = new SKBitmap(_finalWidth * 3, _finalHeight * 2);
using var canvas = new SKCanvas(bitmap);
canvas.Clear(SKColors.Black);
int posterHeight = _finalHeight * 2 / 6;
for (int i = 0; i < Rows; i++)
{
int imageCounter = _random.Next(0, 5);
int currentWidthPos = i * 75;
int currentHeight = i * (posterHeight + Spacing);
while (currentWidthPos < _finalWidth * 3)
{
SKBitmap? currentImage;
switch (imageCounter)
{
case 0:
case 2:
case 3:
currentImage = SkiaHelper.GetNextValidImage(_skiaEncoder, poster, posterIndex, out int newPosterIndex);
posterIndex = newPosterIndex;
break;
default:
currentImage = SkiaHelper.GetNextValidImage(_skiaEncoder, backdrop, backdropIndex, out int newBackdropIndex);
backdropIndex = newBackdropIndex;
break;
}
if (currentImage == null)
{
throw new ArgumentException("Not enough valid pictures provided to create a splashscreen!");
}
// resize to the same aspect as the original
var imageWidth = Math.Abs(posterHeight * currentImage.Width / currentImage.Height);
using var resizedBitmap = new SKBitmap(imageWidth, posterHeight);
currentImage.ScalePixels(resizedBitmap, SKFilterQuality.High);
// draw on canvas
canvas.DrawBitmap(resizedBitmap, currentWidthPos, currentHeight);
currentWidthPos += imageWidth + Spacing;
currentImage.Dispose();
if (imageCounter >= 4)
{
imageCounter = 0;
}
else
{
imageCounter++;
}
}
}
if (applyFilter)
{
var paintColor = new SKPaint
{
Color = SKColors.Black.WithAlpha(0x50),
Style = SKPaintStyle.Fill
};
canvas.DrawRect(0, 0, _finalWidth * 3, _finalHeight * 2, paintColor);
}
return bitmap;
}
///
/// Transform the collage in 3D space.
///
/// The bitmap to transform.
/// The transformed image.
private SKBitmap Transform3D(SKBitmap input)
{
var bitmap = new SKBitmap(_finalWidth, _finalHeight);
using var canvas = new SKCanvas(bitmap);
canvas.Clear(SKColors.Black);
var matrix = new SKMatrix
{
ScaleX = 0.324108899f,
ScaleY = 0.563934922f,
SkewX = -0.244337708f,
SkewY = 0.0377609022f,
TransX = 42.0407715f,
TransY = -198.104706f,
Persp0 = -9.08959337E-05f,
Persp1 = 6.85242048E-05f,
Persp2 = 0.988209724f
};
canvas.SetMatrix(matrix);
canvas.DrawBitmap(input, 0, 0);
return bitmap;
}
}
}