Нарисуйте угловой прямоугольник на растровом изображении - PullRequest
0 голосов
/ 29 августа 2018

У меня есть картинка, содержащая текст:

enter image description here

Я сделал метод для обнаружения строк текста. Этот метод возвращает 4 угла для текстовой зоны (всегда отсортированы):

enter image description here

Я хочу изменить растровое изображение, чтобы нарисовать прямоугольник (с прозрачностью) из этих 4 углов. Примерно так:

enter image description here

У меня есть изображение в оттенках серого. Я создал функцию для рисования прямоугольника, но мне удалось нарисовать только прямоугольник:

public static void SaveDrawRectangle(int width, int height, Byte[] matrix, int dpi, System.Drawing.Point[] corners, string path)
{
    System.Windows.Media.Imaging.WriteableBitmap wbm = new System.Windows.Media.Imaging.WriteableBitmap(width, height, dpi, dpi, System.Windows.Media.PixelFormats.Bgra32, null);

    uint[] pixels = new uint[width * height];
    for (int Y = 0; Y < height; Y++)
    {
        for (int X = 0; X < width; X++)
        {
            byte pixel = matrix[Y * width + X];
            int red = pixel;
            int green = pixel;
            int blue = pixel;
            int alpha = 255;
            if (X >= corners[0].X && X <= corners[1].X &&
                Y >= corners[0].Y && Y <= corners[3].Y)
            {
                red = 255;
                alpha = 255;
            }

            pixels[Y * width + X] = (uint)((alpha << 24) + (red << 16) + (green << 8) + blue);
        }
    }

    wbm.WritePixels(new System.Windows.Int32Rect(0, 0, width, height), pixels, width * 4, 0);
    using (FileStream stream5 = new FileStream(path, FileMode.Create))
    {
        PngBitmapEncoder encoder5 = new PngBitmapEncoder();
        encoder5.Frames.Add(BitmapFrame.Create(wbm));
        encoder5.Save(stream5);
    }
}

enter image description here

Как нарисовать прямоугольник из 4 углов?

1 Ответ

0 голосов
/ 29 августа 2018

Я изменяю свое состояние, заменяя этот код :

public static void SaveDrawRectangle(int width, int height, Byte[] matrix, int dpi, List<Point> corners, string path)
{
    System.Windows.Media.Imaging.WriteableBitmap wbm = new System.Windows.Media.Imaging.WriteableBitmap(width, height, dpi, dpi, System.Windows.Media.PixelFormats.Bgra32, null);

    uint[] pixels = new uint[width * height];
    for (int Y = 0; Y < height; Y++)
    {
        for (int X = 0; X < width; X++)
        {
            byte pixel = matrix[Y * width + X];
            int red = pixel;
            int green = pixel;
            int blue = pixel;
            int alpha = 255;
            if (IsInRectangle(X, Y, corners))
            {
                red = 255;
            }

            pixels[Y * width + X] = (uint)((alpha << 24) + (red << 16) + (green << 8) + blue);
        }
    }

    wbm.WritePixels(new System.Windows.Int32Rect(0, 0, width, height), pixels, width * 4, 0);
    using (FileStream stream5 = new FileStream(path, FileMode.Create))
    {
        PngBitmapEncoder encoder5 = new PngBitmapEncoder();
        encoder5.Frames.Add(BitmapFrame.Create(wbm));
        encoder5.Save(stream5);
    }
}

public static bool IsInRectangle(int X, int Y, List<Point> corners)
{
    Point p1, p2;
    bool inside = false;

    if (corners.Count < 3)
    {
        return inside;
    }

    var oldPoint = new Point(
        corners[corners.Count - 1].X, corners[corners.Count - 1].Y);

    for (int i = 0; i < corners.Count; i++)
    {
        var newPoint = new Point(corners[i].X, corners[i].Y);

        if (newPoint.X > oldPoint.X)
        {
            p1 = oldPoint;
            p2 = newPoint;
        }
        else
        {
            p1 = newPoint;
            p2 = oldPoint;
        }

        if ((newPoint.X < X) == (X <= oldPoint.X)
            && (Y - (long)p1.Y) * (p2.X - p1.X)
            < (p2.Y - (long)p1.Y) * (X - p1.X))
        {
            inside = !inside;
        }

        oldPoint = newPoint;
    }

    return inside;
}

Работает, но есть 2 ошибки:

  • сгенерированные изображения очень большие (базовое изображение занимает 6 месяцев, а после рисования 25 месяцев)
  • генерация занимает несколько раз (мои изображения 5000x7000 пикселей, процесс занимает 10 секунд)

Возможно, есть лучший способ, но он хорошо работает.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...