1

我正在设计一个将图像加载到 PictureBox 并允许用户在其上绘制选择矩形的应用程序。目前,我使用 Paint 事件和一个布尔值来清除先前绘制的矩形(因为它是一个可拖动的选择框)。

问题:

代码失败,因为前一个矩形没有从图像中清除。尽管绘制的每个矩形都是透明的,但效果是一个不透明的矩形,因为之前的矩形没有被清除。如何清除这些矩形?

逻辑:

saveState默认为真。当 Paint 事件第一次被触发时,包含正常图像的状态被保存。当触发 MouseDown 事件时,我们注册矩形的起始位置和一个指示正在绘制矩形的布尔值。

当触发 MouseMove 事件时,我们在当前坐标处绘制一个矩形。由于绘制事件时(我认为)触发了 Paint 事件并且saveState为 false,因此我们在绘制矩形之前恢复正常图像。

最后,当触发 MouseUp 事件时,saveState将其设置为 true,因此保存了绘制最后一个矩形的图形状态,我们又回到了开头。

我读到了ControlPaint.DrawReversibleFrame,但是由于这篇文章这个问题给我的印象是它不是为在图像上绘制而是为直接在屏幕或表单上绘制而设计的,我不确定这是否是我需要的。

代码:

public partial class MainWindow : Form
{
    private bool drawingRectangle;
    private int x1, y1, x2, y2;
    private Image currentImage;
    private GraphicsState previousState;
    private bool saveState;

    public MainWindow()
    {
        InitializeComponent();
        this.drawingRectangle = false;
        this.saveState = true;
    }

    private void EditorPictureBox_MouseDown(object sender, MouseEventArgs e)
    {
        this.x1 = e.X;
        this.y1 = e.Y;
        this.drawingRectangle = true;
    }

    private void EditorPictureBox_MouseMove(object sender, MouseEventArgs e)
    {
        if (this.drawingRectangle)
        {
            this.x2 = e.X;
            this.y2 = e.Y;
            Graphics g = Graphics.FromImage(this.currentImage);
            int[] dim = ImageLibrary.CalculateRectangleDimensions(this.x1, this.y1, this.x2, this.y2);
            g.FillRectangle(new SolidBrush(Color.FromArgb(100, 128, 255, 255)), dim[0], dim[1], dim[2], dim[3]);
            this.Refresh();
        }
    }

    private void EditorPictureBox_Paint(object sender, PaintEventArgs e)
    {
        if (this.saveState)
        {
            this.previousState = e.Graphics.Save();
            this.saveState = false;
        }
        else
            e.Graphics.Restore(this.previousState);
    }

    private void EditorPictureBox_MouseUp(object sender, MouseEventArgs e)
    {
        if (this.drawingRectangle)
        {
            this.drawingRectangle = false;
            // When the mouse click is released, save the graphics state
            this.saveState = true; 
        }
    }

    private void LoadImage2Button_Click(object sender, EventArgs e)
    {
        this.currentImage = Image.FromFile("goat2.jpg");
        this.EditorPictureBox.Image = this.currentImage;
    }   
}

这是CalculateRectangleDimensions(存储在静态库中)的代码:

public static int[] CalculateRectangleDimensions(int x1, int y1, int x2, int y2)
{
    int[] dimensions = new int[4]; // x1, y1, width, height
    if (x1 <= x2) // Mouse was dragged to the right
    {
        dimensions[0] = x1;
        dimensions[2] = x2 - x1;
    }
    else // Mouse was dragged to the right
    {
        dimensions[0] = x2;
        dimensions[2] = x1 - x2;
    }

    if (y1 <= y2) // Mouse was dragged up
    {
        dimensions[1] = y1;
        dimensions[3] = y2 - y1;
    }
    else // Mouse was dragged down
    {
        dimensions[1] = y2;
        dimensions[3] = y1 - y2;
    }
    return dimensions;
}
4

1 回答 1

1

Graphics.Save 调用时不会保存图形的全部内容,它只是保存状态信息,如平移、缩放、变换等。

如果要撤消已经完成的绘制,则必须执行可逆绘制之类的操作,或者在要撤消绘制时必须重新绘制原始图像。

于 2012-07-25T16:51:32.143 回答