How to draw multiple rectangles with mouse on an image?

点点圈 提交于 2021-02-13 05:46:15

问题


I am trying to build a WinForm application that will load an image file from an openfiledialog and load it into either a panel or picturebox. I want to be able to draw multiple rectangles with the left mouse button and have them stay on the image. I have been successful in getting a single rectangle at a time on the actual form but not on an image inside a panel/picturebox. Does anyone know of a resource that would help me understand how to achieve this?

This is the code that allows me to draw a single resizable rectangle on the form, but when I change the mouse events from form1_MouseEvent to panel1_MouseEvent it does nothing..

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;

namespace Temporary_Name_Utility
{
    public partial class Form1 : Form
    {
        Rectangle myRectangle;
        bool draw = false;


        public Form1()
        {
            InitializeComponent();
        }

        private void button1_Click(object sender, EventArgs e)
        {
            try
            {
                OpenFileDialog myFile = new OpenFileDialog();
                myFile.Filter = "Image Files(*.img, *.bmp) |*.img; *.bmp;";
                myFile.InitialDirectory = "c:\\";

                if (myFile.ShowDialog() == DialogResult.OK)
                {
                    pictureBox1.BackgroundImage = Image.FromFile(myFile.FileName);
                }
            }

            catch (Exception error)
            {
                MessageBox.Show("Error loading the selected file.  Original error: " + error.Message);

            }

        }

        private void pictureBox1_MouseDown(object sender, MouseEventArgs e)
        {
            myRectangle = new Rectangle(e.X, e.Y, 0, 0);
            this.Invalidate();
        }


        private void pictureBox1_Paint(object sender, PaintEventArgs e)
        {
            if(draw)
            {
                using(Pen myPen = new Pen(Color.Red, 2))
                {
                    e.Graphics.DrawRectangle(myPen, myRectangle);
                }
            }
        }

        private void pictureBox1_MouseMove(object sender, MouseEventArgs e)
        {
            if(e.Button == MouseButtons.Left)
            {
                myRectangle = new Rectangle(myRectangle.Left, myRectangle.Top, Math.Min(e.X - myRectangle.Left, pictureBox1.ClientRectangle.Width - myRectangle.Left), Math.Min(e.Y - myRectangle.Top, pictureBox1.ClientRectangle.Height - myRectangle.Top));
            }
            this.Invalidate();
        }

        private void pictureBox1_MouseUp(object sender, MouseEventArgs e)
        {

        }

    }
}

回答1:


Finally got some working code: This allows the user to draw as many rectangles as wanted with the mouse inside of the picturebox.

     using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;

namespace Rectangle_Utility
{
    public partial class Form1 : Form
    {
        Point startPos;
        Point currentPos;
        bool drawing;
        List<Rectangle> myRectangles = new List<Rectangle>();

        public Form1()
        {
            InitializeComponent();
            this.DoubleBuffered = true;
        }

        #region Menu Tool Strip
        private void selectFileToolStripMenuItem_Click(object sender, EventArgs e)
        {
            try
            {
                //Initiate new OpenFileDialog
                //Filter for img and bmp files
                //Start looking in root of c:
                OpenFileDialog myFile = new OpenFileDialog();
                myFile.Filter = "Image Files(*.img, *.bmp) |*.img; *.bmp;";
                myFile.InitialDirectory = "c:\\";

                //Set background image of pictureBox to file selected through OpenFileDialog
                if (myFile.ShowDialog() == DialogResult.OK)
                {
                    pictureBox1.BackgroundImage = Image.FromFile(myFile.FileName);

                }
            }

            catch (Exception error)
            {
                MessageBox.Show("Error loading the selected file.  Original error: " + error.Message);

            }
        }

        private void exitToolStripMenuItem_Click(object sender, EventArgs e)
        {
            this.Close();
        }

        private void clearAllToolStripMenuItem_Click(object sender, EventArgs e)
        {

        }

        private void zoomToolStripMenuItem_Click(object sender, EventArgs e)
        {

        }

        private void undoLastActionToolStripMenuItem_Click(object sender, EventArgs e)
        {

        }

        private void redoLastActionToolStripMenuItem_Click(object sender, EventArgs e)
        {

        }
        #endregion

        #region Rectangle

        private Rectangle getRectangle()
        {
            return new Rectangle(
                Math.Min(startPos.X, currentPos.X),
                Math.Min(startPos.Y, currentPos.Y),
                Math.Abs(startPos.X - currentPos.X),
                Math.Abs(startPos.Y - currentPos.Y));
        }

        private void pictureBox1_MouseDown(object sender, MouseEventArgs e)
        {
            if (e.Button == MouseButtons.Left)
            {
                currentPos = startPos = e.Location;
                drawing = true;
            }

        }

        private void pictureBox1_MouseMove(object sender, MouseEventArgs e)
        {
            currentPos = e.Location;
            if (drawing) pictureBox1.Invalidate();
        }

        private void pictureBox1_Paint(object sender, PaintEventArgs e)
        {
            if (myRectangles.Count > 0) e.Graphics.DrawRectangles(Pens.Black, myRectangles.ToArray());
            if (drawing) e.Graphics.DrawRectangle(Pens.Red, getRectangle());
        }

        private void pictureBox1_MouseUp(object sender, MouseEventArgs e)
        {
            if (drawing)
            {
                drawing = false;
                var rect = getRectangle();
                if (rect.Width > 0 && rect.Height > 0) myRectangles.Add(rect);
                pictureBox1.Invalidate();
            }
        }

        #endregion

    }
}



回答2:


You have to create a graphic from image and use that graphic to draw the rectangle so that the rectangle will be drawn in the image.

Graphics newGraphics = Graphics.FromImage(image);
newGraphics.FillRectangle(new SolidBrush(Color.Black), rectangles);


来源:https://stackoverflow.com/questions/24084672/how-to-draw-multiple-rectangles-with-mouse-on-an-image

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!