mirror of
https://github.com/jellyfin/jellyfin.git
synced 2024-11-16 10:29:01 -07:00
497 lines
25 KiB
C#
497 lines
25 KiB
C#
using ImageMagickSharp;
|
|
using MediaBrowser.Common.Configuration;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
|
|
namespace Emby.Drawing.ImageMagick
|
|
{
|
|
public class StripCollageBuilder
|
|
{
|
|
private readonly IApplicationPaths _appPaths;
|
|
|
|
public StripCollageBuilder(IApplicationPaths appPaths)
|
|
{
|
|
_appPaths = appPaths;
|
|
}
|
|
|
|
public void BuildPosterCollage(List<string> paths, string outputPath, int width, int height, string text)
|
|
{
|
|
if (!string.IsNullOrWhiteSpace(text))
|
|
{
|
|
using (var wand = BuildPosterCollageWandWithText(paths, text, width, height))
|
|
{
|
|
wand.SaveImage(outputPath);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
using (var wand = BuildPosterCollageWand(paths, width, height))
|
|
{
|
|
wand.SaveImage(outputPath);
|
|
}
|
|
}
|
|
}
|
|
|
|
public void BuildSquareCollage(List<string> paths, string outputPath, int width, int height, string text)
|
|
{
|
|
if (!string.IsNullOrWhiteSpace(text))
|
|
{
|
|
using (var wand = BuildSquareCollageWandWithText(paths, text, width, height))
|
|
{
|
|
wand.SaveImage(outputPath);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
using (var wand = BuildSquareCollageWand(paths, width, height))
|
|
{
|
|
wand.SaveImage(outputPath);
|
|
}
|
|
}
|
|
}
|
|
|
|
public void BuildThumbCollage(List<string> paths, string outputPath, int width, int height, string text)
|
|
{
|
|
if (!string.IsNullOrWhiteSpace(text))
|
|
{
|
|
using (var wand = BuildThumbCollageWandWithText(paths, text, width, height))
|
|
{
|
|
wand.SaveImage(outputPath);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
using (var wand = BuildThumbCollageWand(paths, width, height))
|
|
{
|
|
wand.SaveImage(outputPath);
|
|
}
|
|
}
|
|
}
|
|
|
|
private MagickWand BuildThumbCollageWandWithText(List<string> paths, string text, int width, int height)
|
|
{
|
|
var inputPaths = ImageHelpers.ProjectPaths(paths, 8);
|
|
using (var wandImages = new MagickWand(inputPaths.ToArray()))
|
|
{
|
|
var wand = new MagickWand(width, height);
|
|
wand.OpenImage("gradient:#111111-#111111");
|
|
using (var draw = new DrawingWand())
|
|
{
|
|
using (var fcolor = new PixelWand(ColorName.White))
|
|
{
|
|
draw.FillColor = fcolor;
|
|
draw.Font = MontserratLightFont;
|
|
draw.FontSize = 60;
|
|
draw.FontWeight = FontWeightType.LightStyle;
|
|
draw.TextAntialias = true;
|
|
}
|
|
|
|
var fontMetrics = wand.QueryFontMetrics(draw, text);
|
|
var textContainerY = Convert.ToInt32(height * .165);
|
|
wand.CurrentImage.AnnotateImage(draw, (width - fontMetrics.TextWidth) / 2, textContainerY, 0.0, text);
|
|
|
|
var iSlice = Convert.ToInt32(width * .1166666667);
|
|
int iTrans = Convert.ToInt32(height * 0.2);
|
|
int iHeight = Convert.ToInt32(height * 0.46296296296296296296296296296296);
|
|
var horizontalImagePadding = Convert.ToInt32(width * 0.0125);
|
|
|
|
foreach (var element in wandImages.ImageList)
|
|
{
|
|
int iWidth = (int)Math.Abs(iHeight * element.Width / element.Height);
|
|
element.Gravity = GravityType.CenterGravity;
|
|
element.BackgroundColor = new PixelWand("none", 1);
|
|
element.ResizeImage(iWidth, iHeight, FilterTypes.LanczosFilter);
|
|
int ix = (int)Math.Abs((iWidth - iSlice) / 2);
|
|
element.CropImage(iSlice, iHeight, ix, 0);
|
|
|
|
element.ExtentImage(iSlice, iHeight, 0 - horizontalImagePadding, 0);
|
|
}
|
|
|
|
wandImages.SetFirstIterator();
|
|
using (var wandList = wandImages.AppendImages())
|
|
{
|
|
wandList.CurrentImage.TrimImage(1);
|
|
using (var mwr = wandList.CloneMagickWand())
|
|
{
|
|
using (var blackPixelWand = new PixelWand(ColorName.Black))
|
|
{
|
|
using (var greyPixelWand = new PixelWand(ColorName.Grey70))
|
|
{
|
|
mwr.CurrentImage.ResizeImage(wandList.CurrentImage.Width, (wandList.CurrentImage.Height / 2), FilterTypes.LanczosFilter, 1);
|
|
mwr.CurrentImage.FlipImage();
|
|
|
|
mwr.CurrentImage.AlphaChannel = AlphaChannelType.DeactivateAlphaChannel;
|
|
mwr.CurrentImage.ColorizeImage(blackPixelWand, greyPixelWand);
|
|
|
|
using (var mwg = new MagickWand(wandList.CurrentImage.Width, iTrans))
|
|
{
|
|
mwg.OpenImage("gradient:black-none");
|
|
var verticalSpacing = Convert.ToInt32(height * 0.01111111111111111111111111111111);
|
|
mwr.CurrentImage.CompositeImage(mwg, CompositeOperator.DstInCompositeOp, 0, verticalSpacing);
|
|
|
|
wandList.AddImage(mwr);
|
|
int ex = (int)(wand.CurrentImage.Width - mwg.CurrentImage.Width) / 2;
|
|
wand.CurrentImage.CompositeImage(wandList.AppendImages(true), CompositeOperator.AtopCompositeOp, ex, Convert.ToInt32(height * 0.26851851851851851851851851851852));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return wand;
|
|
}
|
|
}
|
|
|
|
private MagickWand BuildPosterCollageWand(List<string> paths, int width, int height)
|
|
{
|
|
var inputPaths = ImageHelpers.ProjectPaths(paths, 4);
|
|
using (var wandImages = new MagickWand(inputPaths.ToArray()))
|
|
{
|
|
var wand = new MagickWand(width, height);
|
|
wand.OpenImage("gradient:#111111-#111111");
|
|
using (var draw = new DrawingWand())
|
|
{
|
|
var iSlice = Convert.ToInt32(width * 0.225);
|
|
int iTrans = Convert.ToInt32(height * .25);
|
|
int iHeight = Convert.ToInt32(height * .65);
|
|
var horizontalImagePadding = Convert.ToInt32(width * 0.0275);
|
|
|
|
foreach (var element in wandImages.ImageList)
|
|
{
|
|
using (var blackPixelWand = new PixelWand(ColorName.Black))
|
|
{
|
|
int iWidth = (int)Math.Abs(iHeight * element.Width / element.Height);
|
|
element.Gravity = GravityType.CenterGravity;
|
|
element.BackgroundColor = blackPixelWand;
|
|
element.ResizeImage(iWidth, iHeight, FilterTypes.LanczosFilter);
|
|
int ix = (int)Math.Abs((iWidth - iSlice) / 2);
|
|
element.CropImage(iSlice, iHeight, ix, 0);
|
|
|
|
element.ExtentImage(iSlice, iHeight, 0 - horizontalImagePadding, 0);
|
|
}
|
|
}
|
|
|
|
wandImages.SetFirstIterator();
|
|
using (var wandList = wandImages.AppendImages())
|
|
{
|
|
wandList.CurrentImage.TrimImage(1);
|
|
using (var mwr = wandList.CloneMagickWand())
|
|
{
|
|
using (var blackPixelWand = new PixelWand(ColorName.Black))
|
|
{
|
|
using (var greyPixelWand = new PixelWand(ColorName.Grey70))
|
|
{
|
|
mwr.CurrentImage.ResizeImage(wandList.CurrentImage.Width, (wandList.CurrentImage.Height / 2), FilterTypes.LanczosFilter, 1);
|
|
mwr.CurrentImage.FlipImage();
|
|
|
|
mwr.CurrentImage.AlphaChannel = AlphaChannelType.DeactivateAlphaChannel;
|
|
mwr.CurrentImage.ColorizeImage(blackPixelWand, greyPixelWand);
|
|
|
|
using (var mwg = new MagickWand(wandList.CurrentImage.Width, iTrans))
|
|
{
|
|
mwg.OpenImage("gradient:black-none");
|
|
var verticalSpacing = Convert.ToInt32(height * 0.01111111111111111111111111111111);
|
|
mwr.CurrentImage.CompositeImage(mwg, CompositeOperator.CopyOpacityCompositeOp, 0, verticalSpacing);
|
|
|
|
wandList.AddImage(mwr);
|
|
int ex = (int)(wand.CurrentImage.Width - mwg.CurrentImage.Width) / 2;
|
|
wand.CurrentImage.CompositeImage(wandList.AppendImages(true), CompositeOperator.AtopCompositeOp, ex, Convert.ToInt32(height * .05));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return wand;
|
|
}
|
|
}
|
|
|
|
private MagickWand BuildPosterCollageWandWithText(List<string> paths, string label, int width, int height)
|
|
{
|
|
var inputPaths = ImageHelpers.ProjectPaths(paths, 4);
|
|
using (var wandImages = new MagickWand(inputPaths.ToArray()))
|
|
{
|
|
var wand = new MagickWand(width, height);
|
|
wand.OpenImage("gradient:#111111-#111111");
|
|
using (var draw = new DrawingWand())
|
|
{
|
|
using (var fcolor = new PixelWand(ColorName.White))
|
|
{
|
|
draw.FillColor = fcolor;
|
|
draw.Font = MontserratLightFont;
|
|
draw.FontSize = 60;
|
|
draw.FontWeight = FontWeightType.LightStyle;
|
|
draw.TextAntialias = true;
|
|
}
|
|
|
|
var fontMetrics = wand.QueryFontMetrics(draw, label);
|
|
var textContainerY = Convert.ToInt32(height * .165);
|
|
wand.CurrentImage.AnnotateImage(draw, (width - fontMetrics.TextWidth) / 2, textContainerY, 0.0, label);
|
|
|
|
var iSlice = Convert.ToInt32(width * 0.225);
|
|
int iTrans = Convert.ToInt32(height * 0.2);
|
|
int iHeight = Convert.ToInt32(height * 0.46296296296296296296296296296296);
|
|
var horizontalImagePadding = Convert.ToInt32(width * 0.0275);
|
|
|
|
foreach (var element in wandImages.ImageList)
|
|
{
|
|
int iWidth = (int)Math.Abs(iHeight * element.Width / element.Height);
|
|
element.Gravity = GravityType.CenterGravity;
|
|
element.BackgroundColor = new PixelWand("none", 1);
|
|
element.ResizeImage(iWidth, iHeight, FilterTypes.LanczosFilter);
|
|
int ix = (int)Math.Abs((iWidth - iSlice) / 2);
|
|
element.CropImage(iSlice, iHeight, ix, 0);
|
|
|
|
element.ExtentImage(iSlice, iHeight, 0 - horizontalImagePadding, 0);
|
|
}
|
|
|
|
wandImages.SetFirstIterator();
|
|
using (var wandList = wandImages.AppendImages())
|
|
{
|
|
wandList.CurrentImage.TrimImage(1);
|
|
using (var mwr = wandList.CloneMagickWand())
|
|
{
|
|
using (var blackPixelWand = new PixelWand(ColorName.Black))
|
|
{
|
|
using (var greyPixelWand = new PixelWand(ColorName.Grey70))
|
|
{
|
|
mwr.CurrentImage.ResizeImage(wandList.CurrentImage.Width, (wandList.CurrentImage.Height / 2), FilterTypes.LanczosFilter, 1);
|
|
mwr.CurrentImage.FlipImage();
|
|
|
|
mwr.CurrentImage.AlphaChannel = AlphaChannelType.DeactivateAlphaChannel;
|
|
mwr.CurrentImage.ColorizeImage(blackPixelWand, greyPixelWand);
|
|
|
|
using (var mwg = new MagickWand(wandList.CurrentImage.Width, iTrans))
|
|
{
|
|
mwg.OpenImage("gradient:black-none");
|
|
var verticalSpacing = Convert.ToInt32(height * 0.01111111111111111111111111111111);
|
|
mwr.CurrentImage.CompositeImage(mwg, CompositeOperator.DstInCompositeOp, 0, verticalSpacing);
|
|
|
|
wandList.AddImage(mwr);
|
|
int ex = (int)(wand.CurrentImage.Width - mwg.CurrentImage.Width) / 2;
|
|
wand.CurrentImage.CompositeImage(wandList.AppendImages(true), CompositeOperator.AtopCompositeOp, ex, Convert.ToInt32(height * 0.26851851851851851851851851851852));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return wand;
|
|
}
|
|
}
|
|
|
|
private MagickWand BuildThumbCollageWand(List<string> paths, int width, int height)
|
|
{
|
|
var inputPaths = ImageHelpers.ProjectPaths(paths, 4);
|
|
using (var wandImages = new MagickWand(inputPaths.ToArray()))
|
|
{
|
|
var wand = new MagickWand(width, height);
|
|
wand.OpenImage("gradient:#111111-#111111");
|
|
using (var draw = new DrawingWand())
|
|
{
|
|
var iSlice = Convert.ToInt32(width * .1166666667 * 2);
|
|
int iTrans = Convert.ToInt32(height * .25);
|
|
int iHeight = Convert.ToInt32(height * .62);
|
|
var horizontalImagePadding = Convert.ToInt32(width * 0.0125);
|
|
|
|
foreach (var element in wandImages.ImageList)
|
|
{
|
|
using (var blackPixelWand = new PixelWand(ColorName.Black))
|
|
{
|
|
int iWidth = (int)Math.Abs(iHeight * element.Width / element.Height);
|
|
element.Gravity = GravityType.CenterGravity;
|
|
element.BackgroundColor = blackPixelWand;
|
|
element.ResizeImage(iWidth, iHeight, FilterTypes.LanczosFilter);
|
|
int ix = (int)Math.Abs((iWidth - iSlice) / 2);
|
|
element.CropImage(iSlice, iHeight, ix, 0);
|
|
|
|
element.ExtentImage(iSlice, iHeight, 0 - horizontalImagePadding, 0);
|
|
}
|
|
}
|
|
|
|
wandImages.SetFirstIterator();
|
|
using (var wandList = wandImages.AppendImages())
|
|
{
|
|
wandList.CurrentImage.TrimImage(1);
|
|
using (var mwr = wandList.CloneMagickWand())
|
|
{
|
|
using (var blackPixelWand = new PixelWand(ColorName.Black))
|
|
{
|
|
using (var greyPixelWand = new PixelWand(ColorName.Grey70))
|
|
{
|
|
mwr.CurrentImage.ResizeImage(wandList.CurrentImage.Width, (wandList.CurrentImage.Height / 2), FilterTypes.LanczosFilter, 1);
|
|
mwr.CurrentImage.FlipImage();
|
|
|
|
mwr.CurrentImage.AlphaChannel = AlphaChannelType.DeactivateAlphaChannel;
|
|
mwr.CurrentImage.ColorizeImage(blackPixelWand, greyPixelWand);
|
|
|
|
using (var mwg = new MagickWand(wandList.CurrentImage.Width, iTrans))
|
|
{
|
|
mwg.OpenImage("gradient:black-none");
|
|
var verticalSpacing = Convert.ToInt32(height * 0.01111111111111111111111111111111);
|
|
mwr.CurrentImage.CompositeImage(mwg, CompositeOperator.CopyOpacityCompositeOp, 0, verticalSpacing);
|
|
|
|
wandList.AddImage(mwr);
|
|
int ex = (int)(wand.CurrentImage.Width - mwg.CurrentImage.Width) / 2;
|
|
wand.CurrentImage.CompositeImage(wandList.AppendImages(true), CompositeOperator.AtopCompositeOp, ex, Convert.ToInt32(height * .085));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return wand;
|
|
}
|
|
}
|
|
|
|
private MagickWand BuildSquareCollageWand(List<string> paths, int width, int height)
|
|
{
|
|
var inputPaths = ImageHelpers.ProjectPaths(paths, 4);
|
|
using (var wandImages = new MagickWand(inputPaths.ToArray()))
|
|
{
|
|
var wand = new MagickWand(width, height);
|
|
wand.OpenImage("gradient:#111111-#111111");
|
|
using (var draw = new DrawingWand())
|
|
{
|
|
var iSlice = Convert.ToInt32(width * .225);
|
|
int iTrans = Convert.ToInt32(height * .25);
|
|
int iHeight = Convert.ToInt32(height * .63);
|
|
var horizontalImagePadding = Convert.ToInt32(width * 0.02);
|
|
|
|
foreach (var element in wandImages.ImageList)
|
|
{
|
|
using (var blackPixelWand = new PixelWand(ColorName.Black))
|
|
{
|
|
int iWidth = (int)Math.Abs(iHeight * element.Width / element.Height);
|
|
element.Gravity = GravityType.CenterGravity;
|
|
element.BackgroundColor = blackPixelWand;
|
|
element.ResizeImage(iWidth, iHeight, FilterTypes.LanczosFilter);
|
|
int ix = (int)Math.Abs((iWidth - iSlice) / 2);
|
|
element.CropImage(iSlice, iHeight, ix, 0);
|
|
|
|
element.ExtentImage(iSlice, iHeight, 0 - horizontalImagePadding, 0);
|
|
}
|
|
}
|
|
|
|
wandImages.SetFirstIterator();
|
|
using (var wandList = wandImages.AppendImages())
|
|
{
|
|
wandList.CurrentImage.TrimImage(1);
|
|
using (var mwr = wandList.CloneMagickWand())
|
|
{
|
|
using (var blackPixelWand = new PixelWand(ColorName.Black))
|
|
{
|
|
using (var greyPixelWand = new PixelWand(ColorName.Grey70))
|
|
{
|
|
mwr.CurrentImage.ResizeImage(wandList.CurrentImage.Width, (wandList.CurrentImage.Height / 2), FilterTypes.LanczosFilter, 1);
|
|
mwr.CurrentImage.FlipImage();
|
|
|
|
mwr.CurrentImage.AlphaChannel = AlphaChannelType.DeactivateAlphaChannel;
|
|
mwr.CurrentImage.ColorizeImage(blackPixelWand, greyPixelWand);
|
|
|
|
using (var mwg = new MagickWand(wandList.CurrentImage.Width, iTrans))
|
|
{
|
|
mwg.OpenImage("gradient:black-none");
|
|
var verticalSpacing = Convert.ToInt32(height * 0.01111111111111111111111111111111);
|
|
mwr.CurrentImage.CompositeImage(mwg, CompositeOperator.CopyOpacityCompositeOp, 0, verticalSpacing);
|
|
|
|
wandList.AddImage(mwr);
|
|
int ex = (int)(wand.CurrentImage.Width - mwg.CurrentImage.Width) / 2;
|
|
wand.CurrentImage.CompositeImage(wandList.AppendImages(true), CompositeOperator.AtopCompositeOp, ex, Convert.ToInt32(height * .07));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return wand;
|
|
}
|
|
}
|
|
|
|
private MagickWand BuildSquareCollageWandWithText(List<string> paths, string label, int width, int height)
|
|
{
|
|
var inputPaths = ImageHelpers.ProjectPaths(paths, 4);
|
|
using (var wandImages = new MagickWand(inputPaths.ToArray()))
|
|
{
|
|
var wand = new MagickWand(width, height);
|
|
wand.OpenImage("gradient:#111111-#111111");
|
|
using (var draw = new DrawingWand())
|
|
{
|
|
using (var fcolor = new PixelWand(ColorName.White))
|
|
{
|
|
draw.FillColor = fcolor;
|
|
draw.Font = MontserratLightFont;
|
|
draw.FontSize = 60;
|
|
draw.FontWeight = FontWeightType.LightStyle;
|
|
draw.TextAntialias = true;
|
|
}
|
|
|
|
var fontMetrics = wand.QueryFontMetrics(draw, label);
|
|
var textContainerY = Convert.ToInt32(height * .165);
|
|
wand.CurrentImage.AnnotateImage(draw, (width - fontMetrics.TextWidth) / 2, textContainerY, 0.0, label);
|
|
|
|
var iSlice = Convert.ToInt32(width * .225);
|
|
int iTrans = Convert.ToInt32(height * 0.2);
|
|
int iHeight = Convert.ToInt32(height * 0.46296296296296296296296296296296);
|
|
var horizontalImagePadding = Convert.ToInt32(width * 0.02);
|
|
|
|
foreach (var element in wandImages.ImageList)
|
|
{
|
|
int iWidth = (int)Math.Abs(iHeight * element.Width / element.Height);
|
|
element.Gravity = GravityType.CenterGravity;
|
|
element.BackgroundColor = new PixelWand("none", 1);
|
|
element.ResizeImage(iWidth, iHeight, FilterTypes.LanczosFilter);
|
|
int ix = (int)Math.Abs((iWidth - iSlice) / 2);
|
|
element.CropImage(iSlice, iHeight, ix, 0);
|
|
|
|
element.ExtentImage(iSlice, iHeight, 0 - horizontalImagePadding, 0);
|
|
}
|
|
|
|
wandImages.SetFirstIterator();
|
|
using (var wandList = wandImages.AppendImages())
|
|
{
|
|
wandList.CurrentImage.TrimImage(1);
|
|
using (var mwr = wandList.CloneMagickWand())
|
|
{
|
|
using (var blackPixelWand = new PixelWand(ColorName.Black))
|
|
{
|
|
using (var greyPixelWand = new PixelWand(ColorName.Grey70))
|
|
{
|
|
mwr.CurrentImage.ResizeImage(wandList.CurrentImage.Width, (wandList.CurrentImage.Height / 2), FilterTypes.LanczosFilter, 1);
|
|
mwr.CurrentImage.FlipImage();
|
|
|
|
mwr.CurrentImage.AlphaChannel = AlphaChannelType.DeactivateAlphaChannel;
|
|
mwr.CurrentImage.ColorizeImage(blackPixelWand, greyPixelWand);
|
|
|
|
using (var mwg = new MagickWand(wandList.CurrentImage.Width, iTrans))
|
|
{
|
|
mwg.OpenImage("gradient:black-none");
|
|
var verticalSpacing = Convert.ToInt32(height * 0.01111111111111111111111111111111);
|
|
mwr.CurrentImage.CompositeImage(mwg, CompositeOperator.DstInCompositeOp, 0, verticalSpacing);
|
|
|
|
wandList.AddImage(mwr);
|
|
int ex = (int)(wand.CurrentImage.Width - mwg.CurrentImage.Width) / 2;
|
|
wand.CurrentImage.CompositeImage(wandList.AppendImages(true), CompositeOperator.AtopCompositeOp, ex, Convert.ToInt32(height * 0.26851851851851851851851851851852));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return wand;
|
|
}
|
|
}
|
|
|
|
private string MontserratLightFont
|
|
{
|
|
get { return PlayedIndicatorDrawer.ExtractFont("MontserratLight.otf", _appPaths); }
|
|
}
|
|
}
|
|
}
|