Как удалить фон изображения с помощью clippingPath, используя конвейер в Graphicsmill?
У меня большое изображение (не менее 200 МБ и не более 2 ГБ) с обтравочным контуром. Я хочу применить обтравочный контур, чтобы удалить фон. Я обнаружил, что единственное решение ( ConvertClippingPathToMask) использует растровое изображение, которое загружает все изображение в память и создает исключение OutOfMemoryException.
/// <summary>
/// Converts clipping path to alpha channel mask
/// </summary>
private static void ConvertClippingPathToMask()
{
using (var reader = new JpegReader("../../../../_Input/Apple.jpg"))
using (var bitmap = reader.Frames[0].GetBitmap()) // I can get rid of this line by using reader instead of bitmap in the next line, but then the OOM will throw in the next line.
using (var maskBitmap = new Bitmap(bitmap.Width, bitmap.Height, PixelFormat.Format8bppGrayscale, new GrayscaleColor(0)))
using (var graphics = maskBitmap.GetAdvancedGraphics())
{
var graphicsPath = reader.ClippingPaths[0].CreateGraphicsPath(reader.Width, reader.Height);
graphics.FillPath(new SolidBrush(new GrayscaleColor(255)), Path.Create(graphicsPath));
bitmap.Channels.SetAlpha(maskBitmap);
bitmap.Save("../../../../_Output/ConvertClippingPathToMask.png");
}
}
При таком подходе для получения графического объекта всегда необходимо растровое изображение, которое затем применяет путь отсечения.
На самом деле, мне даже не нужно maskBitmap
в памяти, потому что я могу использовать отдельный ридер для setAlpha, но потом: Как мне создать maskBitmap без растрового изображения для создания графического объекта?
2 ответа
Надлежащим способом является использование API Pipeline:
using (var reader = new JpegReader("ImageWithPath.jpg"))
using (var gc = new GraphicsContainer(reader))
using (var ig = new ImageGenerator(gc, PixelFormat.Format8bppGrayscale, RgbColor.Black))
using (var setAlpha = new Aurigma.GraphicsMill.Transforms.SetAlpha())
{
using (var gr = gc.GetGraphics())
{
var path = reader.ClippingPaths[0].CreateGraphicsPath(reader.Width, reader.Height);
gr.FillPath(new SolidBrush(RgbColor.White), Aurigma.GraphicsMill.AdvancedDrawing.Path.Create(path));
}
setAlpha.AlphaSource = ig;
Pipeline.Run(reader + setAlpha + "output.png");
}
Для полноты картины:
Это решение от Федора из Aurigma. Форум охватывает некоторые случаи, которых нет в решении Eugenes.
using (var reader = ImageReader.Create("../../../PathForTest.tif"))
using (var maskGen = new ImageGenerator(reader.Width, reader.Height, PixelFormat.Format8bppGrayscale, RgbColor.Black))
using (var drawer = new Drawer())
using (var bitmap = new Bitmap())
{
var graphicsPath = Aurigma.GraphicsMill.AdvancedDrawing.Path.Create(reader.ClippingPaths[0], reader.Width, reader.Height);
drawer.Draw += (sender, e) =>
{
e.Graphics.FillPath(new SolidBrush(new GrayscaleColor(255)), graphicsPath);
};
using (var setAlpha = new SetAlpha(maskGen + drawer))
{
(reader + setAlpha + bitmap).Run();
bitmap.Save("../../../result.tif");
}
}