2016-07-09 22:30:12 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.Drawing;
|
2019-11-16 01:34:18 +00:00
|
|
|
|
using System.Drawing.Drawing2D;
|
2016-07-09 22:30:12 +00:00
|
|
|
|
using System.Drawing.Imaging;
|
|
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
|
|
2019-09-29 16:47:06 +00:00
|
|
|
|
namespace PKHeX.Drawing
|
2016-07-09 22:30:12 +00:00
|
|
|
|
{
|
2018-08-04 17:06:06 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Image Layering/Blending Utility
|
|
|
|
|
/// </summary>
|
2017-01-12 01:55:42 +00:00
|
|
|
|
public static class ImageUtil
|
2016-07-09 22:30:12 +00:00
|
|
|
|
{
|
2018-07-15 01:43:46 +00:00
|
|
|
|
public static Bitmap LayerImage(Image baseLayer, Image overLayer, int x, int y, double transparency)
|
|
|
|
|
{
|
|
|
|
|
overLayer = ChangeOpacity(overLayer, transparency);
|
|
|
|
|
return LayerImage(baseLayer, overLayer, x, y);
|
|
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
|
2019-10-08 02:10:38 +00:00
|
|
|
|
public static Bitmap LayerImage(Image? baseLayer, Image overLayer, int x, int y)
|
2016-07-09 22:30:12 +00:00
|
|
|
|
{
|
2019-10-08 02:10:38 +00:00
|
|
|
|
if (baseLayer is null)
|
|
|
|
|
return (Bitmap)overLayer;
|
2018-08-29 01:21:46 +00:00
|
|
|
|
Bitmap img = new Bitmap(baseLayer);
|
2016-07-09 22:30:12 +00:00
|
|
|
|
using (Graphics gr = Graphics.FromImage(img))
|
2018-07-21 17:42:23 +00:00
|
|
|
|
gr.DrawImage(overLayer, x, y, overLayer.Width, overLayer.Height);
|
2016-07-09 22:30:12 +00:00
|
|
|
|
return img;
|
|
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
|
2017-01-08 07:54:09 +00:00
|
|
|
|
public static Bitmap ChangeOpacity(Image img, double trans)
|
2016-07-09 22:30:12 +00:00
|
|
|
|
{
|
|
|
|
|
if (img.PixelFormat.HasFlag(PixelFormat.Indexed))
|
|
|
|
|
return (Bitmap)img;
|
|
|
|
|
|
2017-10-02 04:25:23 +00:00
|
|
|
|
var bmp = (Bitmap)img.Clone();
|
|
|
|
|
GetBitmapData(bmp, out BitmapData bmpData, out IntPtr ptr, out byte[] data);
|
2016-07-09 22:30:12 +00:00
|
|
|
|
|
2017-10-02 04:25:23 +00:00
|
|
|
|
Marshal.Copy(ptr, data, 0, data.Length);
|
|
|
|
|
SetAllTransparencyTo(data, trans);
|
|
|
|
|
Marshal.Copy(data, 0, ptr, data.Length);
|
2016-07-09 22:30:12 +00:00
|
|
|
|
bmp.UnlockBits(bmpData);
|
|
|
|
|
|
2016-10-31 02:15:48 +00:00
|
|
|
|
return bmp;
|
|
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
|
2017-01-08 07:54:09 +00:00
|
|
|
|
public static Bitmap ChangeAllColorTo(Image img, Color c)
|
2016-10-31 02:15:48 +00:00
|
|
|
|
{
|
|
|
|
|
if (img.PixelFormat.HasFlag(PixelFormat.Indexed))
|
|
|
|
|
return (Bitmap)img;
|
|
|
|
|
|
2017-10-02 04:25:23 +00:00
|
|
|
|
var bmp = (Bitmap)img.Clone();
|
|
|
|
|
GetBitmapData(bmp, out BitmapData bmpData, out IntPtr ptr, out byte[] data);
|
2016-10-31 02:15:48 +00:00
|
|
|
|
|
2017-10-02 04:25:23 +00:00
|
|
|
|
Marshal.Copy(ptr, data, 0, data.Length);
|
2018-07-28 02:59:14 +00:00
|
|
|
|
ChangeAllColorTo(data, c);
|
2017-10-02 04:25:23 +00:00
|
|
|
|
Marshal.Copy(data, 0, ptr, data.Length);
|
2016-10-31 02:15:48 +00:00
|
|
|
|
bmp.UnlockBits(bmpData);
|
|
|
|
|
|
2017-06-18 04:49:14 +00:00
|
|
|
|
return bmp;
|
|
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
|
2017-06-18 04:49:14 +00:00
|
|
|
|
public static Bitmap ToGrayscale(Image img)
|
|
|
|
|
{
|
|
|
|
|
if (img.PixelFormat.HasFlag(PixelFormat.Indexed))
|
|
|
|
|
return (Bitmap)img;
|
|
|
|
|
|
2017-10-02 04:25:23 +00:00
|
|
|
|
var bmp = (Bitmap)img.Clone();
|
|
|
|
|
GetBitmapData(bmp, out BitmapData bmpData, out IntPtr ptr, out byte[] data);
|
2017-06-18 04:49:14 +00:00
|
|
|
|
|
2017-10-02 04:25:23 +00:00
|
|
|
|
Marshal.Copy(ptr, data, 0, data.Length);
|
|
|
|
|
SetAllColorToGrayScale(data);
|
|
|
|
|
Marshal.Copy(data, 0, ptr, data.Length);
|
2017-06-18 04:49:14 +00:00
|
|
|
|
bmp.UnlockBits(bmpData);
|
|
|
|
|
|
2016-07-09 22:30:12 +00:00
|
|
|
|
return bmp;
|
|
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
|
2017-10-02 04:25:23 +00:00
|
|
|
|
private static void GetBitmapData(Bitmap bmp, out BitmapData bmpData, out IntPtr ptr, out byte[] data)
|
|
|
|
|
{
|
|
|
|
|
bmpData = bmp.LockBits(new Rectangle(0, 0, bmp.Width, bmp.Height), ImageLockMode.ReadWrite, PixelFormat.Format32bppArgb);
|
|
|
|
|
ptr = bmpData.Scan0;
|
|
|
|
|
data = new byte[bmp.Width * bmp.Height * 4];
|
|
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
|
2018-10-13 15:02:55 +00:00
|
|
|
|
public static Bitmap GetBitmap(byte[] data, int width, int height, PixelFormat format = PixelFormat.Format32bppArgb)
|
2018-01-10 07:30:44 +00:00
|
|
|
|
{
|
2018-10-13 15:02:55 +00:00
|
|
|
|
var bmp = new Bitmap(width, height, format);
|
|
|
|
|
var bmpData = bmp.LockBits(new Rectangle(0, 0, bmp.Width, bmp.Height), ImageLockMode.WriteOnly, format);
|
|
|
|
|
var ptr = bmpData.Scan0;
|
|
|
|
|
Marshal.Copy(data, 0, ptr, data.Length);
|
|
|
|
|
bmp.UnlockBits(bmpData);
|
|
|
|
|
return bmp;
|
2018-01-10 07:30:44 +00:00
|
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
|
2018-01-10 07:30:44 +00:00
|
|
|
|
public static byte[] GetPixelData(Bitmap bitmap)
|
|
|
|
|
{
|
|
|
|
|
var argbData = new byte[bitmap.Width * bitmap.Height * 4];
|
|
|
|
|
var bd = bitmap.LockBits(new Rectangle(0, 0, bitmap.Width, bitmap.Height), ImageLockMode.ReadOnly, bitmap.PixelFormat);
|
|
|
|
|
Marshal.Copy(bd.Scan0, argbData, 0, bitmap.Width * bitmap.Height * 4);
|
|
|
|
|
bitmap.UnlockBits(bd);
|
|
|
|
|
return argbData;
|
|
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
|
2018-07-28 02:59:14 +00:00
|
|
|
|
public static void SetAllUsedPixelsOpaque(byte[] data)
|
|
|
|
|
{
|
|
|
|
|
for (int i = 0; i < data.Length; i += 4)
|
2018-08-04 17:06:06 +00:00
|
|
|
|
{
|
2018-07-28 02:59:14 +00:00
|
|
|
|
if (data[i + 3] != 0)
|
|
|
|
|
data[i + 3] = 0xFF;
|
2018-08-04 17:06:06 +00:00
|
|
|
|
}
|
2018-07-28 02:59:14 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static void RemovePixels(byte[] pixels, byte[] original)
|
|
|
|
|
{
|
|
|
|
|
for (int i = 0; i < original.Length; i += 4)
|
|
|
|
|
{
|
|
|
|
|
if (original[i + 3] == 0)
|
|
|
|
|
continue;
|
|
|
|
|
pixels[i + 0] = 0;
|
|
|
|
|
pixels[i + 1] = 0;
|
|
|
|
|
pixels[i + 2] = 0;
|
|
|
|
|
pixels[i + 3] = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-10-02 04:25:23 +00:00
|
|
|
|
private static void SetAllTransparencyTo(byte[] data, double trans)
|
|
|
|
|
{
|
|
|
|
|
for (int i = 0; i < data.Length; i += 4)
|
|
|
|
|
data[i + 3] = (byte)(data[i + 3] * trans);
|
|
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
|
2018-07-28 02:59:14 +00:00
|
|
|
|
public static void ChangeAllColorTo(byte[] data, Color c)
|
2017-10-02 04:25:23 +00:00
|
|
|
|
{
|
|
|
|
|
byte R = c.R;
|
|
|
|
|
byte G = c.G;
|
|
|
|
|
byte B = c.B;
|
|
|
|
|
for (int i = 0; i < data.Length; i += 4)
|
|
|
|
|
{
|
|
|
|
|
if (data[i + 3] == 0)
|
|
|
|
|
continue;
|
|
|
|
|
data[i + 0] = B;
|
|
|
|
|
data[i + 1] = G;
|
|
|
|
|
data[i + 2] = R;
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
|
2017-10-02 04:25:23 +00:00
|
|
|
|
private static void SetAllColorToGrayScale(byte[] data)
|
|
|
|
|
{
|
|
|
|
|
for (int i = 0; i < data.Length; i += 4)
|
|
|
|
|
{
|
|
|
|
|
if (data[i + 3] == 0)
|
|
|
|
|
continue;
|
2018-07-27 02:34:27 +00:00
|
|
|
|
byte greyS = (byte)(((0.3 * data[i + 2]) + (0.59 * data[i + 1]) + (0.11 * data[i + 0])) / 3);
|
2017-10-02 04:25:23 +00:00
|
|
|
|
data[i + 0] = greyS;
|
|
|
|
|
data[i + 1] = greyS;
|
|
|
|
|
data[i + 2] = greyS;
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-07-14 22:08:14 +00:00
|
|
|
|
|
2019-04-23 05:24:29 +00:00
|
|
|
|
public static void GlowEdges(byte[] data, byte blue, byte green, byte red, int width, int reach = 3, double amount = 0.0777)
|
|
|
|
|
{
|
|
|
|
|
PollutePixels(data, width, reach, amount);
|
|
|
|
|
CleanPollutedPixels(data, blue, green, red);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static void PollutePixels(byte[] data, int width, int reach, double amount)
|
2018-07-22 04:55:37 +00:00
|
|
|
|
{
|
|
|
|
|
int stride = width * 4;
|
|
|
|
|
int height = data.Length / stride;
|
|
|
|
|
for (int i = 0; i < data.Length; i += 4)
|
|
|
|
|
{
|
2019-04-23 05:24:29 +00:00
|
|
|
|
// only pollute outwards if the current pixel isn't transparent
|
2018-07-22 04:55:37 +00:00
|
|
|
|
if (data[i + 3] == 0)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
int x = (i % stride) / 4;
|
|
|
|
|
int y = (i / stride);
|
|
|
|
|
Pollute(x, y);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Pollute(int x, int y)
|
|
|
|
|
{
|
|
|
|
|
int left = Math.Max(0, x - reach);
|
|
|
|
|
int right = Math.Min(width - 1, x + reach);
|
|
|
|
|
int top = Math.Max(0, y - reach);
|
|
|
|
|
int bottom = Math.Min(height - 1, y + reach);
|
|
|
|
|
for (int i = left; i <= right; i++)
|
|
|
|
|
{
|
2018-07-27 02:34:27 +00:00
|
|
|
|
for (int j = top; j <= bottom; j++)
|
|
|
|
|
{
|
2019-04-23 05:24:29 +00:00
|
|
|
|
// update one of the color bits
|
|
|
|
|
// it is expected that a transparent pixel RGBA value is 0.
|
2018-07-27 02:34:27 +00:00
|
|
|
|
var c = 4 * (i + (j * width));
|
|
|
|
|
data[c + 0] += (byte)(amount * (0xFF - data[c + 0]));
|
|
|
|
|
}
|
2018-07-22 04:55:37 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2019-04-23 05:24:29 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static void CleanPollutedPixels(byte[] data, byte blue, byte green, byte red)
|
|
|
|
|
{
|
2018-07-22 04:55:37 +00:00
|
|
|
|
for (int i = 0; i < data.Length; i += 4)
|
|
|
|
|
{
|
2019-04-23 05:24:29 +00:00
|
|
|
|
// only clean if the current pixel isn't transparent
|
2018-07-22 04:55:37 +00:00
|
|
|
|
if (data[i + 3] != 0)
|
|
|
|
|
continue;
|
2019-04-23 05:24:29 +00:00
|
|
|
|
|
|
|
|
|
// grab the transparency from the donor byte
|
|
|
|
|
var transparency = data[i + 0];
|
|
|
|
|
if (transparency == 0)
|
2018-07-22 04:55:37 +00:00
|
|
|
|
continue;
|
|
|
|
|
|
2019-04-23 05:24:29 +00:00
|
|
|
|
data[i + 0] = blue;
|
|
|
|
|
data[i + 1] = green;
|
|
|
|
|
data[i + 2] = red;
|
|
|
|
|
data[i + 3] = transparency;
|
2018-07-22 04:55:37 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-14 22:08:14 +00:00
|
|
|
|
public static Color ColorBaseStat(int v)
|
|
|
|
|
{
|
|
|
|
|
const float maxval = 180; // shift the green cap down
|
|
|
|
|
float x = 100f * v / maxval;
|
|
|
|
|
if (x > 100)
|
|
|
|
|
x = 100;
|
2018-07-27 02:34:27 +00:00
|
|
|
|
double red = 255f * (x > 50 ? 1 - (2 * (x - 50) / 100.0) : 1.0);
|
2018-07-14 22:08:14 +00:00
|
|
|
|
double green = 255f * (x > 50 ? 1.0 : 2 * x / 100.0);
|
|
|
|
|
|
|
|
|
|
return Blend(Color.FromArgb((int)red, (int)green, 0), Color.White, 0.4);
|
|
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
|
2019-11-26 01:32:10 +00:00
|
|
|
|
public static Color ColorBaseStatTotal(int tot) => ColorBaseStat((int) (Math.Max(0, tot - 175) / 3f));
|
|
|
|
|
|
2018-07-14 22:08:14 +00:00
|
|
|
|
public static Color Blend(Color color, Color backColor, double amount)
|
|
|
|
|
{
|
2018-07-27 02:34:27 +00:00
|
|
|
|
byte r = (byte)((color.R * amount) + (backColor.R * (1 - amount)));
|
|
|
|
|
byte g = (byte)((color.G * amount) + (backColor.G * (1 - amount)));
|
|
|
|
|
byte b = (byte)((color.B * amount) + (backColor.B * (1 - amount)));
|
2018-07-14 22:08:14 +00:00
|
|
|
|
return Color.FromArgb(r, g, b);
|
|
|
|
|
}
|
2019-11-16 01:34:18 +00:00
|
|
|
|
|
|
|
|
|
// https://stackoverflow.com/a/24199315
|
|
|
|
|
public static Bitmap ResizeImage(Image image, int width, int height)
|
|
|
|
|
{
|
|
|
|
|
var destRect = new Rectangle(0, 0, width, height);
|
|
|
|
|
var destImage = new Bitmap(width, height);
|
|
|
|
|
|
|
|
|
|
destImage.SetResolution(image.HorizontalResolution, image.VerticalResolution);
|
|
|
|
|
|
|
|
|
|
using var wrapMode = new ImageAttributes();
|
|
|
|
|
wrapMode.SetWrapMode(WrapMode.TileFlipXY);
|
|
|
|
|
|
|
|
|
|
using var graphics = Graphics.FromImage(destImage);
|
|
|
|
|
graphics.CompositingMode = CompositingMode.SourceCopy;
|
|
|
|
|
graphics.CompositingQuality = CompositingQuality.HighQuality;
|
|
|
|
|
graphics.InterpolationMode = InterpolationMode.HighQualityBicubic;
|
|
|
|
|
graphics.SmoothingMode = SmoothingMode.HighQuality;
|
|
|
|
|
graphics.PixelOffsetMode = PixelOffsetMode.HighQuality;
|
|
|
|
|
|
|
|
|
|
graphics.DrawImage(image, destRect, 0, 0, image.Width, image.Height, GraphicsUnit.Pixel, wrapMode);
|
|
|
|
|
|
|
|
|
|
return destImage;
|
|
|
|
|
}
|
2016-07-09 22:30:12 +00:00
|
|
|
|
}
|
|
|
|
|
}
|