类似画笔的绘图控件

类似画笔的绘图控件
作者:北京理工大学 卫琳

下载本文示例源代码

源代码运行效果图如下:


想必大家都用过WINDOWS自带的画笔,这是一个小巧易用的软件。在业余时间,我模拟画笔自己开发了一个类似的程序(当然不如画笔那么功能丰富)。它主要完成的功能有画直线、曲线、圆、椭圆、矩形、多边形;支持剪贴板的操作;支持撤销、重复;保存成位图文件;打开位图文件。这个例子是用MFC开发的,为了方便使用,最后将转换成控件。

建立单文档工程Demo,下面将分四部分介绍相关功能的实现。

一、 绘图功能

本程序包含多种图元:直线、曲线、圆、椭圆、矩形等,使用不同的图元类实现,这些图元类均派生于同一个基类:CDrawObject。这样可以大大简化不同图元的处理过程。
虽然不同的图元具有不同的表现形式和数据结构,但它们具有许多相同的特性,如都具有颜色定义功能,都有图元绘制函数等,在基类中对这些共有的数据成员和成员函数进行描述。图元基类的定义如下:

class CDrawObject : public CObject
{
private:
	COLORREF m_PenColor;//图元颜色
	int   m_iPenWidth;  //画笔宽度
	bool  m_bFill;      //是否填充

public:
	bool m_bSelected;
	long m_nStyle;//图元类型
	CDrawObject(){m_bSelected = false;};

	void SetPenColor(COLORREF color);//设置图元颜色
	COLORREF GetPenColor();//获得图元颜色

	void SetPenWidth(int width) {m_iPenWidth = width;};
	int  GetPenWidth() {return  m_iPenWidth;};

	void  SetFill(bool fill) { m_bFill = fill;};
	bool GetFill()  {return m_bFill;};

	virtual void Draw(CDC* pDC) {};
	virtual void MoveAt(CDC* pDC, long x, long y) {};
	virtual	void  EndPoint(CDC* pDC) {};
	virtual void  NewPoint(long x, long y){};
	//图形对象第一点坐标,如果返回false则结束绘图
	virtual int  AddPoint(long x, long y){return 0;};
};
各图元的具体实现函数请参考源代码。

绘图功能的实现主要是在视类中完成的。首先建立相应的菜单和工具栏按钮用来设置图元的样式、颜色、画笔的粗细、是否填充等等。还要重载OnLButtonDown(按下鼠标左键)、OnMouseMove(鼠标移动)、OnLButtonUp(松开鼠标左键)三个函数。创建过程如下:

1、 按下左键,创建新的图元类实例;
2、 跟踪鼠标移动修改图元,获得所见即所得的视觉效果;
3、 松开左键,绘制结束。

二、 剪贴板操作

和剪贴板操作相关的函数主要有以下几个:
打开剪贴板:OpenClipboard();
清空剪贴板:EmptyClipboard();
保存至剪贴板:SetClipboardData (CF_BITMAP, bitmap.GetSafeHandle() );
取出剪贴板的内容:GetClipboardData(CF_BITMAP);
关闭剪贴板:CloseClipboard();

至于视觉效果的实现,本来应该用"橡皮筋类",这里偷了点儿懒,直接画了矩形:)。剪切和拷贝的实现方法基本相同,剪切时需要清空选中的矩形区域。粘贴时需要先判断剪切板中有无内容。

三、 撤销和重复

为了实现撤销和重复,我自己定义了一个类Stack,该类的主要功能类似于一个栈,可以在初始化时定义栈的大小,可以弹出栈顶元素、增加新元素等等,除此以外还保存了一个表示当前位置的指针m_iCurPos。撤销时该指针向前移动,重复时向后移动,如果撤销后又有了新操作,则当前长度应改至m_iCurPos,即栈中m_iCurPos之后的元素无效。 至于栈中保存的内容,则是在每次操作后调用自己定义的SaveInStack()函数,将屏幕内容保存到一个HBITMAP类型的变量中。(这个方法有点儿笨,不过我实在想不出更好的方法了:(。但这个方法的实际效果还是不错的。)

四、 打开和保存
有了前面的基础,着部分比较容易实现。
打开文件的主要函数是:
HBITMAP  hbmp = (HBITMAP)LoadImage(NULL, _T(path), IMAGE_BITMAP,
	0, 0, LR_CREATEDIBSECTION | LR_DEFAULTSIZE | LR_LOADFROMFILE);

pDC->BitBlt(0, 0, bm.bmWidth, bm.bmHeight, &dc, 0, 0, SRCCOPY);
具体实现过程请参考源代码。

以上是MFC程序的实现过程,接下来我们把它转化成控件。

五、转化成控件

1、新建控件
启动Microsoft Visual C++ 6.0,单击File下拉菜单下的New命令,在Profects标签下选择MFC ActiveX ControlWizard。输入工程名WinPainter;

2、引入ActivDoc.h和ActivDoc.cpp文件;

3、在CwinPainterCtrl类的构造函数中添加
AddDocTemplate(new CActiveXDocTemplate(
      RUNTIME_CLASS(CDemo1Doc),
      RUNTIME_CLASS(CMainFrame),
      RUNTIME_CLASS(CDemo1View)));
4、 将刚才开发的MFC工程中的视类、文档类、框架类的头文件、实现文件,还有图元类以及Stack类的文件统统包含进来,这时各类的消息函数可以正常响应,但菜单、工具栏等资源需要重新定义;

5、 在CWinPainterCtrl类中调用视类的函数,举例:
CDemo1View*  m_pView;
m_pView = (CDemo1View *)(GetFrameWnd()->GetActiveView());
ASSERT(m_pView);
m_pView->OnEditCopy();
调用MFC中其它类的函数与此类似。

6、 增加控件的属性和方法。
至此,控件开发完毕。

控件运行测试效果图如下:


参考文献: 王宏 李玉东 李罡 《Visual C++ 实战演练》 人民邮电出版社
最近做了一个类似Windows画图板的小程序,拿来和大家分享一下。功能和界面全部模仿于Windows自带的画图板,界面如截图。功能主要有: 手绘线、简单图形、文字输入、图块拖放、重复撤销、画面缩放、打开保存图片文件,另外为了直接从数据库或者XML中存取图片,另外还提供了从Base64编码存取图片的接口,其他还实现了和画图板同样效果的工具箱及颜料盒。 1、 绘图: 绘制功能主要在OnLButtonDown、OnMouseMove、OnLButtonUp中实现,根据当前选择的工具和工具的样式进行绘制。程序中每种工具的绘制都是通过一个图元对象实现,比如钢笔、画刷因为本质上都是手绘线所以都通过CDrawPen来实现、直线和矩形等都通过CDrawShape来实现,所有的图元对象都放在文件DrawObject.h中,具体的实现请参见代码。 另外为了避免屏幕的闪烁,程序中做了两个处理:一、把视图自己的重绘背景代码屏蔽掉,即响应视图的WM_ERASEBKGND消息,直接返回TRUE,并在Ondraw中程序自己绘制背景就可以了;二,在Ondraw中绘制的时候,先绘制到内存DC中,然后再贴回视图的DC。这样就基本上就可以避免屏幕的闪烁了。 2、 撤销重做: 由于整个程序较为简单,就没有采用把动作记录下来的方式,只是采取了一个较简单的方法,在每当一个图元绘制动作结束时就把画布的内容存到bmp中以备撤销,这些历史画面存放在一个bmp数组中,并定义了一个Stack类来管理该数组(此实现方式参考了在线杂志21期《类似画笔绘图控件-卫琳》,在此表示感谢!),然后在屏幕上绘制的时候就是把当前位图贴到屏幕上,并把还没存取位图的图元绘上就行了。 3、 缩放: 一般视图缩放的实现是通过DC的两个函数来实现:SetViewportExt和SetWindowExt,但是这两个函数只在MM_ISOTROPIC和MM_ANISOTROPIC这两种映射模式下有效果,而常用的带滚动条的视图类CScrollView却不支持这两种模式(参见CScrollView::SetScrollSizes)的实现。所以只好把MFC的CScrollView的代码拿过来改了一下以支持MM_ISOTROPIC映射模式,如下所示,具体参见程序文件“ADMMScrollView.h”: ::SetMapMode(hdc,MM_ISOTROPIC); int XLogMm = ::GetDeviceCaps(hdc, HORZSIZE); int YLogMm = ::GetDeviceCaps(hdc, VERTSIZE); int XLogPix = ::GetDeviceCaps(hdc, HORZRES); int YLogPix = ::GetDeviceCaps(hdc, VERTRES); ::SetWindowExtEx(hdc,XLogMm*100,YLogMm*100,NULL); ::SetViewportExtEx(hdc,(int)(XLogPix*fZoomScale), (int)(YLogPix*fZoomScale),NULL); 4、 工具箱: 首先为了实现工具箱的按钮分两列显示的效果,需要设置一下按钮的TBBS_WRAPPED,参见程序中的CToolPaletteBar类;然后,选择不同工具时展现出工具的样式,比如选择“直线”时列出可用的直线宽度样式,该功能通过在ToolBar上放一个CListCtrl实现,当前的工具样式通过图标的方式展现,效果和画图板的类似。 5、 颜料盒: 从CDialogBar类继承了一个类,然后在WM_PAINT响应函数里面绘制一个个小颜色矩形,并重写其OnLButtonDown、OnRButtonDown、OnLButtonDblClk来和用户交互,实现出来的效果还不错,看上去和画图板的颜料盒一样。 6、 鼠标光标样式: 首先在资源文件中增加需要的光标资源,然后在视图的OnSetCursor消息函数中调用SetCursor函数来设置光标样式就可以了,注意要判断HitTest参数是否为HTCLIENT,不然的话鼠标移到了滚动条上的时候光标还是画笔的样式就让人觉得怪怪的了。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值