Как удалить фон изображения с помощью 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");
    }
}
Другие вопросы по тегам