2023-02-02 20:19:45 +00:00
|
|
|
|
using System.Drawing;
|
2020-04-27 16:23:31 +00:00
|
|
|
|
using System.Drawing.Imaging;
|
|
|
|
|
using System.Runtime.InteropServices;
|
2021-05-15 09:19:22 +00:00
|
|
|
|
using FFMpegCore.Pipes;
|
2020-04-27 16:23:31 +00:00
|
|
|
|
|
2022-03-24 19:19:37 +00:00
|
|
|
|
namespace FFMpegCore.Extensions.System.Drawing.Common
|
2020-04-27 16:23:31 +00:00
|
|
|
|
{
|
|
|
|
|
public class BitmapVideoFrameWrapper : IVideoFrame, IDisposable
|
|
|
|
|
{
|
|
|
|
|
public int Width => Source.Width;
|
|
|
|
|
|
|
|
|
|
public int Height => Source.Height;
|
|
|
|
|
|
|
|
|
|
public string Format { get; private set; }
|
|
|
|
|
|
|
|
|
|
public Bitmap Source { get; private set; }
|
|
|
|
|
|
|
|
|
|
public BitmapVideoFrameWrapper(Bitmap bitmap)
|
|
|
|
|
{
|
|
|
|
|
Source = bitmap ?? throw new ArgumentNullException(nameof(bitmap));
|
|
|
|
|
Format = ConvertStreamFormat(bitmap.PixelFormat);
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-12 11:48:55 +00:00
|
|
|
|
public void Serialize(Stream stream)
|
2020-04-27 16:23:31 +00:00
|
|
|
|
{
|
|
|
|
|
var data = Source.LockBits(new Rectangle(0, 0, Width, Height), ImageLockMode.ReadOnly, Source.PixelFormat);
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var buffer = new byte[data.Stride * data.Height];
|
|
|
|
|
Marshal.Copy(data.Scan0, buffer, 0, buffer.Length);
|
2020-04-27 16:35:53 +00:00
|
|
|
|
stream.Write(buffer, 0, buffer.Length);
|
2020-04-27 16:23:31 +00:00
|
|
|
|
}
|
|
|
|
|
finally
|
|
|
|
|
{
|
|
|
|
|
Source.UnlockBits(data);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-12 11:48:55 +00:00
|
|
|
|
public async Task SerializeAsync(Stream stream, CancellationToken token)
|
2020-04-27 16:23:31 +00:00
|
|
|
|
{
|
|
|
|
|
var data = Source.LockBits(new Rectangle(0, 0, Width, Height), ImageLockMode.ReadOnly, Source.PixelFormat);
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var buffer = new byte[data.Stride * data.Height];
|
|
|
|
|
Marshal.Copy(data.Scan0, buffer, 0, buffer.Length);
|
2021-11-01 18:15:00 +00:00
|
|
|
|
await stream.WriteAsync(buffer, 0, buffer.Length, token).ConfigureAwait(false);
|
2020-04-27 16:23:31 +00:00
|
|
|
|
}
|
|
|
|
|
finally
|
|
|
|
|
{
|
|
|
|
|
Source.UnlockBits(data);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Dispose()
|
|
|
|
|
{
|
|
|
|
|
Source.Dispose();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static string ConvertStreamFormat(PixelFormat fmt)
|
|
|
|
|
{
|
|
|
|
|
switch (fmt)
|
|
|
|
|
{
|
|
|
|
|
case PixelFormat.Format16bppGrayScale:
|
|
|
|
|
return "gray16le";
|
2021-04-12 11:49:36 +00:00
|
|
|
|
case PixelFormat.Format16bppRgb555:
|
|
|
|
|
return "bgr555le";
|
2020-04-27 16:23:31 +00:00
|
|
|
|
case PixelFormat.Format16bppRgb565:
|
|
|
|
|
return "bgr565le";
|
|
|
|
|
case PixelFormat.Format24bppRgb:
|
2020-05-12 15:26:52 +00:00
|
|
|
|
return "bgr24";
|
2020-04-27 16:23:31 +00:00
|
|
|
|
case PixelFormat.Format32bppArgb:
|
2020-05-12 15:26:52 +00:00
|
|
|
|
return "bgra";
|
2020-04-27 16:23:31 +00:00
|
|
|
|
case PixelFormat.Format32bppPArgb:
|
|
|
|
|
//This is not really same as argb32
|
|
|
|
|
return "argb";
|
|
|
|
|
case PixelFormat.Format32bppRgb:
|
|
|
|
|
return "rgba";
|
|
|
|
|
case PixelFormat.Format48bppRgb:
|
|
|
|
|
return "rgb48le";
|
|
|
|
|
default:
|
|
|
|
|
throw new NotSupportedException($"Not supported pixel format {fmt}");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|