data:image/s3,"s3://crabby-images/17ce1/17ce15705b23e9aab554bf2f552187e47eb21fa9" alt="Screenshot - vectshapes1.jpg"
Introduction
Hello. This is my 3rd article, after this one and this one. I propose a simple 2D vector graphic editor. It manages simple graphic objects like rectangles, lines and ellipses, other than images and RTF text. It shows most of the capabilities of GDI+, such as color transparency, pen style, start/end line cap and so on. All of the images shown in this article have been realized with the tool I propose.
Using the Code
data:image/s3,"s3://crabby-images/2004f/2004f57de5b1cb5fd7824eab21ca73c77c7b6a04" alt="Screenshot - vectshapes2.jpg"
VectShape
is the UserControl
used to show and manipulate the objects. It contains the mouse event logic and the "extended" double buffer logic for rendering. It exposes a big set of methods. These methods are used mainly by toolBox
, which is the control used to manipulate the shapes. Shapes
is a collection of Ele
. Ele
is the base class for all of the objects: rectangles
, boxText
, Lines
, Ellipses
and imageBoxes
. RichForm2
is the form used to edit RTF text. It's a simple form with a rich text box and a tool strip. Go to here for a better example of an RTF editor.
GDI+
When searching The Code Project for GDI+, you'll find many good articles that explain what GDI+ is and the basis of drawing in .NET. In this article, I do not address these basic problems. An in-depth article on GDI+ can be found here.
Add Controls to Your Own Form
Simply put the controls vectShaper
and toolBox
into your form. Then link toolBox
to vectShaper
in the form (i.e. Form1
) constructor:
public Form1()
{
InitializeComponent();
this.toolBox1.setVectShape(this.vectShapes1);
}
Now you're ready to start drawing some shapes.
Points of Interest
Drawing Objects: Extending Double Buffer
You can find many in-depth articles about Double Buffering (DB) on The Code Project. With the following 2 images, I will demonstrate how double buffering works. The idea of DB is to make the drawing on a memory bitmap buffer and then draw the buffer over the control.
data:image/s3,"s3://crabby-images/1c15c/1c15c55bd2cfe863d90e3357df669f95c569a4a9" alt="Screenshot - vectshapes4.jpg"
In this project, I extended the double buffer with two memory buffers. One is for drawing static objects and one is for storing dynamic/moving objects. See the next image:
data:image/s3,"s3://crabby-images/179d0/179d0b0c6934da1bbbb5b6c5cf897170a3508237" alt="Screenshot - vectshapes5.jpg"
backBuffG=Graphics.fromimage(backBuff)
draws static objects over backBuffG
buffG=Graphics.fromimage(buff)
draws bitmap backBuffG over buffG
draws moving objects over buffG
ctrG=ctr.CreateGraphics()
draws bitmap buffG over ctrG
backBuff
is declared outside of the control redraw method. By STATIC objects, I mean all objects that we know aren't going to be moved/added/deleted/updated. In this kind of application, all of the objects that are not selected are static objects. Steps 1 and 2 are done only when we update the static objects. So, we don't redraw static objects in the paint
method, but only the bitmap that "contains" them (Step 4).
data:image/s3,"s3://crabby-images/16924/169248298e8edb27172cafeeda22484b8927a48f" alt="Screenshot - vectshapes7.jpg"
In this example, the blue objects are static. The selected objects -- the green selection rectangle and the handle rectangle -- are all dynamic/moving objects. So when I add/remove/update an object, such as in the MouseUp
event of the control, I redraw the control including Steps 1 and 2. When I move a selected object -- i.e. in the MouseMove
event or in the OnPaint
event of the control -- I do not need to perform Steps 1 and 2. The static objects are drawn because I stored them in backBuff
.
Rendering RichTextBox (RTF) on a Graphic Object
Another interesting point in this project is the rendering of the RTF contained in a rich text box. The Graphics.DrawString
method lets you do something like that, but with a significant limitation. You can specify only one font, color or dimension in the same string. So, it does not draw truly RTF text. I was interested in rendering RTF text, so I Googled it and found this. It explains how to print the RTF contained in a rich text box control. The article contains the code for creating RichTextBoxPrintCtrl
and demonstrates how to use it:
[DllImport("USER32.dll")]
private static extern IntPtr SendMessage(IntPtr hWnd, int msg,
IntPtr wp, IntPtr lp);
public int Draw(int charFrom, int charTo, Graphics g,
int x, int y, int x1, int y1, double conversion)
{
RECT rectToPrint;
rectToPrint.Top =(int)(y * conversion);
rectToPrint.Bottom = (int)(y1 * conversion);
rectToPrint.Left = (int)(x * conversion);
rectToPrint.Right = (int)(decimal)(x1 * conversion);
RECT rectPage;
rectPage.Top = (int)(y * conversion);
rectPage.Bottom = (int)(y1 * conversion);
rectPage.Left = (int)(x * conversion);
rectPage.Right = (int)(x1 * conversion);
IntPtr hdc = g.GetHdc();
FORMATRANGE fmtRange;
fmtRange.chrg.cpMax = charTo; fmtRange.chrg.cpMin = charFrom;
fmtRange.hdc = hdc; fmtRange.hdcTarget = hdc; fmtRange.rc = rectToPrint; fmtRange.rcPage = rectPage;
IntPtr res = IntPtr.Zero;
IntPtr wparam = IntPtr.Zero;
wparam = new IntPtr(1);
IntPtr lparam = IntPtr.Zero;
lparam = Marshal.AllocCoTaskMem(Marshal.SizeOf(fmtRange));
Marshal.StructureToPtr(fmtRange, lparam, false);
res = SendMessage(Handle, EM_FORMATRANGE, wparam, lparam);
Marshal.FreeCoTaskMem(lparam);
g.ReleaseHdc(hdc);
return res.ToInt32();
}
After that, there was a problem. The background color of the control RichetextBox
was also rendered, while I was interested rendering only the text. So I found this, which explains how to make a control transparent:
[DllImport("kernel32.dll", CharSet = CharSet.Auto)]
static extern IntPtr LoadLibrary(string lpFileName);
protected override CreateParams CreateParams
{
get
{
CreateParams prams = base.CreateParams;
if (LoadLibrary("msftedit.dll") != IntPtr.Zero)
{
prams.ExStyle |= 0x020; prams.ClassName = "RICHEDIT50W";
}
return prams;
}
}
Et voilà. At this point, we have all we need to render real RTF text on a graphic object. I called it BoxTesto
; it is a box (rectangle) with a start and an end point and all of the properties derived from the Ele
class. In addition, it has an RTF property that contains a string with the RTF text. When I draw a BoxTesto
, I create a "transparent" RichTextBoxPrintCtrl
control and assign RichTextBoxPrintCtrl.rtf
with BoxTesto.rtf
. Then I call RichTextBoxPrintCtrl.Draw
.
Undo/Redo
I choose to implement the undo/redo as a double-linked list. Each item on the list contains a reference to the object of interest, an old copy of the object and a new copy of the object. In the next image, I present the undo/redo list in blue and the modified objects in black.
data:image/s3,"s3://crabby-images/894e4/894e41676ccce9d476a59e93e9b975ca2d4859e5" alt="Screenshot - vectshapes6.jpg"
Every time I change an object, I store an element in the undo/redo list. When I need to undo an action, I copy the OLD object properties over the interested object properties. Then I go back one element in the undo/redo list. When I need to redo an action, I go forward one element in the undo/redo list. Then I copy the NEW object properties over the object referenced by the element. The "lazy" way: in my old project, I implemented undo/redo like a circular array where the objects stored were the entire collection of graphic objects. That was an easy and fast solution, but too memory-hungry.
Drawing Images (Rotation/Transparency)
data:image/s3,"s3://crabby-images/700cf/700cff7c10f25a54e7ca04af3c35eadb6c57ecb4" alt="Screenshot - vectshapes8.jpg"
Now I focus on image drawing (method Draw
, class ImgBox
, file Shapes.cs) to show how to implement image rotation and transparency:
Color backColor = this.img.GetPixel(0, 0);
int dim =
(int)System.Math.Sqrt(img.Width * img.Width + img.Height * img.Height);
Bitmap curBitmap = new Bitmap(dim, dim);
Graphics curG = Graphics.FromImage(curBitmap);
if (this.Rotation >
0)
{
Matrix X = new Matrix();
X.RotateAt(this.Rotation, new PointF(curBitmap.Width / 2,
curBitmap.Height / 2));
curG.Transform = X;
X.Dispose();
}
curG.DrawImage(img, (dim - img.Width) / 2, (dim - img.Height) / 2,
img.Width, img.Height);
if (this.Trasparent)
curBitmap.MakeTransparent(backColor);
curG.Save();
g.DrawImage(curBitmap, new Rectangle(this.X + dx,
this.Y + dy, this.X1 - this.X, this.Y1 - this.Y));
curG.Dispose();
curBitmap.Dispose();
History
- 15 June, 2007 -- Original version posted
Update 1
- Added
ZoomIN
/ZoomOut
- Added right mouse click canvas movement
- Added image rotation
- Added Transparent Image property
Update 2
- Added obj --> Group and obj --> deGroup functions
- Added a rotation handle for object rotation
data:image/s3,"s3://crabby-images/077f0/077f0d8f47af790db34abf593da9c2e903ce6ae8" alt="Screenshot - vectshapes9.jpg"
- Added File--> Save selected objects / File --> Load objects
- Created Help page
Update 3
- Added Gradient Line Fill Color, i.e. draw a shape, fill it and then set the
UseGradientLineColor
property to true
- Added Group Resize / Rotation and Zoom Management
- Can show a Group like a
GraphicPath
, i.e. select some Line
s and then select them, group them and set the graphPath
property to true
- Added example *.shape files. Load them from File --> Load
data:image/s3,"s3://crabby-images/1489e/1489e068429f4e4cc9b417071d36a9ffb11f6a3b" alt="Screenshot - vectshapes10.jpg"
Update 4
- Added Free Hand Pen Tool
- Added Obj-->Poly--> X/Y Mirror
- Added Obj-->Poly--> Merge
- Added Obj-->Poly--> Extract Points
data:image/s3,"s3://crabby-images/ae8a6/ae8a6caee0da77c3a628c685a5b11e59eedd1cba" alt=""
Update 5
- Added Draw Graph Tool