| | 1 | | using System; |
| | 2 | | using System.Collections.Generic; |
| | 3 | | using Microsoft.Extensions.Logging; |
| | 4 | | using SkiaSharp; |
| | 5 | |
|
| | 6 | | namespace Jellyfin.Drawing.Skia; |
| | 7 | |
|
| | 8 | | /// <summary> |
| | 9 | | /// Used to build the splashscreen. |
| | 10 | | /// </summary> |
| | 11 | | public class SplashscreenBuilder |
| | 12 | | { |
| | 13 | | private const int FinalWidth = 1920; |
| | 14 | | private const int FinalHeight = 1080; |
| | 15 | | // generated collage resolution should be greater than the final resolution |
| | 16 | | private const int WallWidth = FinalWidth * 3; |
| | 17 | | private const int WallHeight = FinalHeight * 2; |
| | 18 | | private const int Rows = 6; |
| | 19 | | private const int Spacing = 20; |
| | 20 | |
|
| | 21 | | private readonly SkiaEncoder _skiaEncoder; |
| | 22 | | private readonly ILogger _logger; |
| | 23 | |
|
| | 24 | | /// <summary> |
| | 25 | | /// Initializes a new instance of the <see cref="SplashscreenBuilder"/> class. |
| | 26 | | /// </summary> |
| | 27 | | /// <param name="skiaEncoder">The SkiaEncoder.</param> |
| | 28 | | /// <param name="logger">The logger.</param> |
| | 29 | | public SplashscreenBuilder(SkiaEncoder skiaEncoder, ILogger logger) |
| | 30 | | { |
| 0 | 31 | | _skiaEncoder = skiaEncoder; |
| 0 | 32 | | _logger = logger; |
| 0 | 33 | | } |
| | 34 | |
|
| | 35 | | /// <summary> |
| | 36 | | /// Generate a splashscreen. |
| | 37 | | /// </summary> |
| | 38 | | /// <param name="posters">The poster paths.</param> |
| | 39 | | /// <param name="backdrops">The landscape paths.</param> |
| | 40 | | /// <param name="outputPath">The output path.</param> |
| | 41 | | public void GenerateSplash(IReadOnlyList<string> posters, IReadOnlyList<string> backdrops, string outputPath) |
| | 42 | | { |
| 0 | 43 | | using var wall = GenerateCollage(posters, backdrops); |
| 0 | 44 | | using var transformed = Transform3D(wall); |
| | 45 | |
|
| 0 | 46 | | using var outputStream = new SKFileWStream(outputPath); |
| 0 | 47 | | using var pixmap = new SKPixmap(new SKImageInfo(FinalWidth, FinalHeight), transformed.GetPixels()); |
| 0 | 48 | | pixmap.Encode(outputStream, StripCollageBuilder.GetEncodedFormat(outputPath), 90); |
| 0 | 49 | | } |
| | 50 | |
|
| | 51 | | /// <summary> |
| | 52 | | /// Generates a collage of posters and landscape pictures. |
| | 53 | | /// </summary> |
| | 54 | | /// <param name="posters">The poster paths.</param> |
| | 55 | | /// <param name="backdrops">The landscape paths.</param> |
| | 56 | | /// <returns>The created collage as a bitmap.</returns> |
| | 57 | | private SKBitmap GenerateCollage(IReadOnlyList<string> posters, IReadOnlyList<string> backdrops) |
| | 58 | | { |
| 0 | 59 | | var posterIndex = 0; |
| 0 | 60 | | var backdropIndex = 0; |
| | 61 | |
|
| 0 | 62 | | SKBitmap? bitmap = null; |
| | 63 | | try |
| | 64 | | { |
| 0 | 65 | | bitmap = new SKBitmap(WallWidth, WallHeight); |
| 0 | 66 | | using var canvas = new SKCanvas(bitmap); |
| 0 | 67 | | canvas.Clear(SKColors.Black); |
| | 68 | |
|
| 0 | 69 | | int posterHeight = WallHeight / 6; |
| | 70 | |
|
| 0 | 71 | | for (int i = 0; i < Rows; i++) |
| | 72 | | { |
| 0 | 73 | | int imageCounter = Random.Shared.Next(0, 5); |
| 0 | 74 | | int currentWidthPos = i * 75; |
| 0 | 75 | | int currentHeight = i * (posterHeight + Spacing); |
| | 76 | |
|
| 0 | 77 | | while (currentWidthPos < WallWidth) |
| | 78 | | { |
| | 79 | | SKBitmap? currentImage; |
| | 80 | |
|
| | 81 | | switch (imageCounter) |
| | 82 | | { |
| | 83 | | case 0: |
| | 84 | | case 2: |
| | 85 | | case 3: |
| 0 | 86 | | currentImage = SkiaHelper.GetNextValidImage(_skiaEncoder, posters, posterIndex, out int newP |
| 0 | 87 | | posterIndex = newPosterIndex; |
| 0 | 88 | | break; |
| | 89 | | default: |
| 0 | 90 | | currentImage = SkiaHelper.GetNextValidImage(_skiaEncoder, backdrops, backdropIndex, out int |
| 0 | 91 | | backdropIndex = newBackdropIndex; |
| | 92 | | break; |
| | 93 | | } |
| | 94 | |
|
| 0 | 95 | | if (currentImage is null) |
| | 96 | | { |
| 0 | 97 | | throw new ArgumentException("Not enough valid pictures provided to create a splashscreen!"); |
| | 98 | | } |
| | 99 | |
|
| 0 | 100 | | using (currentImage) |
| | 101 | | { |
| 0 | 102 | | var imageWidth = Math.Abs(posterHeight * currentImage.Width / currentImage.Height); |
| 0 | 103 | | using var resizedBitmap = new SKBitmap(imageWidth, posterHeight); |
| 0 | 104 | | currentImage.ScalePixels(resizedBitmap, SKFilterQuality.High); |
| | 105 | |
|
| | 106 | | // draw on canvas |
| 0 | 107 | | canvas.DrawBitmap(resizedBitmap, currentWidthPos, currentHeight); |
| | 108 | |
|
| | 109 | | // resize to the same aspect as the original |
| 0 | 110 | | currentWidthPos += imageWidth + Spacing; |
| 0 | 111 | | } |
| | 112 | |
|
| 0 | 113 | | if (imageCounter >= 4) |
| | 114 | | { |
| 0 | 115 | | imageCounter = 0; |
| | 116 | | } |
| | 117 | | else |
| | 118 | | { |
| 0 | 119 | | imageCounter++; |
| | 120 | | } |
| | 121 | | } |
| | 122 | | } |
| | 123 | |
|
| 0 | 124 | | return bitmap; |
| | 125 | | } |
| 0 | 126 | | catch (Exception e) |
| | 127 | | { |
| 0 | 128 | | _logger.LogError(e, "Detected intermediary error creating splashscreen image"); |
| 0 | 129 | | bitmap?.Dispose(); |
| 0 | 130 | | throw; |
| | 131 | | } |
| 0 | 132 | | } |
| | 133 | |
|
| | 134 | | /// <summary> |
| | 135 | | /// Transform the collage in 3D space. |
| | 136 | | /// </summary> |
| | 137 | | /// <param name="input">The bitmap to transform.</param> |
| | 138 | | /// <returns>The transformed image.</returns> |
| | 139 | | private SKBitmap Transform3D(SKBitmap input) |
| | 140 | | { |
| 0 | 141 | | SKBitmap? bitmap = null; |
| | 142 | | try |
| | 143 | | { |
| 0 | 144 | | bitmap = new SKBitmap(FinalWidth, FinalHeight); |
| 0 | 145 | | using var canvas = new SKCanvas(bitmap); |
| 0 | 146 | | canvas.Clear(SKColors.Black); |
| 0 | 147 | | var matrix = new SKMatrix |
| 0 | 148 | | { |
| 0 | 149 | | ScaleX = 0.324108899f, |
| 0 | 150 | | ScaleY = 0.563934922f, |
| 0 | 151 | | SkewX = -0.244337708f, |
| 0 | 152 | | SkewY = 0.0377609022f, |
| 0 | 153 | | TransX = 42.0407715f, |
| 0 | 154 | | TransY = -198.104706f, |
| 0 | 155 | | Persp0 = -9.08959337E-05f, |
| 0 | 156 | | Persp1 = 6.85242048E-05f, |
| 0 | 157 | | Persp2 = 0.988209724f |
| 0 | 158 | | }; |
| | 159 | |
|
| 0 | 160 | | canvas.SetMatrix(matrix); |
| 0 | 161 | | canvas.DrawBitmap(input, 0, 0); |
| | 162 | |
|
| 0 | 163 | | return bitmap; |
| | 164 | | } |
| 0 | 165 | | catch (Exception e) |
| | 166 | | { |
| 0 | 167 | | _logger.LogError(e, "Detected intermediary error creating splashscreen image transforming the image"); |
| 0 | 168 | | bitmap?.Dispose(); |
| 0 | 169 | | throw; |
| | 170 | | } |
| 0 | 171 | | } |
| | 172 | | } |