Visual c++/MFC编程入门之图形图像:GDI对象之画笔CPen

在MFC中,CGdiObject类是GDI对象的基类,通过查阅MSDN我们可以看到,CGdiObject类有六个直接的派生类,在这六个GDI对象中,最常用的莫过于画笔和画刷了,即CPen类和CBrush类。本文就主要讲解画笔的使用。

Visual c++/MFC编程入门之图形图像:GDI对象之画笔CPen

MFC界面开发必备库:

  • Xtreme Toolkit Pro:是屡获殊荣的VC界面库,是MFC开发中最全面界面控件套包,它提供了Windows开发所需要的11种主流的Visual C++ MFC控件,包括Command Bars、Controls、Chart Pro、Calendar、Docking Pane、Property Grid、Report Control、Shortcut Bar、Syntax Edit、Skin Framework 和Task Panel。
  • BCGControlBar:库拥有500多个经过全面设计、测试和充分记录的MFC扩展类。 我们的组件可以轻松地集成到您的应用程序中,并为您节省数百个开发和调试时间。

上一节中讲了CDC类及其屏幕绘图函数,本节的主要内容是GDI对象之画笔CPen。

GDI对象

在MFC中,CGdiObject类是GDI对象的基类,通过查阅MSDN我们可以看到,CGdiObject类有六个直接的派生类,GDI对象主要也是这六个,分别是:CBitmap、CBrush、CFont、CPalette、CPen和CRgn。

在这六个GDI对象中,最常用的莫过于画笔和画刷了,即CPen类和CBrush类。本文就主要讲解画笔的使用。

画笔的应用实例

在这里直接通过一个波形图的实例,来详细讲解画笔的使用方法。

首先介绍此实例要实现的功能:在对话框上有一个Picture控件,将此控件的背景填充为黑色;启动一个定时器,每次定时器到时,所有波形数据都前移一个单位,并获取一个80以内的随机数作为波形的最后一个数据,然后以绿色画笔在绘图控件上绘制波形。这样就实现了波形的绘制及动态变化。

下面是具体实施步骤:

1、创建一个基于对话框的MFC工程,名字设为“Example50”。

2、在自动生成的对话框模板IDD_EXAMPLE50_DIALOG中,删除“TODO: Place dialog controls here.”静态文本框,添加一个Picture控件,ID设为IDC_WAVE_DRAW。

3、为Picture控件IDC_WAVE_DRAW添加CStatic变量,名称设为m_picDraw。

4、在文件Example50Dlg.h文件中CExample50Dlg类声明的上面添加宏定义:

C++代码

  1. #define POINT_COUNT 100

此符号常量的意义是波形的点数,这里用define将其定义为符号常量是为了方便以后可能的修改,假如我们以后想将点数改为200,则只改此宏定义就可以了:#define POINT_COUNT 200,而如果没有使用符号常量,在程序中直接使用了100,那么就需要将所有使用100的位置找出来,并替换为200,这样不仅麻烦也很容易出错,所以最好是将其定义为符号常量。

5、在CExample50Dlg.h文件中为CExample50Dlg类添加成员数组:

C++代码

  1. int   m_nzValues[POINT_COUNT];

此数组用于存放波形数据。

6、在CExample50Dlg类的构造函数中为数组m_nzValues的元素赋初值:

C++代码

CExample50Dlg::CExample50Dlg(CWnd* pParent /*=NULL*/): CDialogEx(CExample50Dlg::IDD, pParent){m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);// 将数组m_nzValues的元素都初始化为0memset(m_nzValues, 0, sizeof(int) * POINT_COUNT);}

7、在CExample50Dlg对话框的初始化成员函数CExample50Dlg::OnInitDialog()中,构造随机数生成器,并启动定时器。CExample50Dlg::OnInitDialog()修改如下:

C++代码

BOOL CExample50Dlg::OnInitDialog(){CDialogEx::OnInitDialog();// Add "About..." menu item to system menu.// IDM_ABOUTBOX must be in the system command range.ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX);ASSERT(IDM_ABOUTBOX < 0xF000);CMenu* pSysMenu = GetSystemMenu(FALSE);if (pSysMenu != NULL){BOOL bNameValid;CString strAboutMenu;bNameValid = strAboutMenu.LoadString(IDS_ABOUTBOX);ASSERT(bNameValid);if (!strAboutMenu.IsEmpty()){pSysMenu->AppendMenu(MF_SEPARATOR);pSysMenu->AppendMenu(MF_STRING, IDM_ABOUTBOX, strAboutMenu);}}// Set the icon for this dialog.  The framework does this automatically//  when the application's main window is not a dialogSetIcon(m_hIcon, TRUE);         // Set big iconSetIcon(m_hIcon, FALSE);        // Set small icon// TODO: Add extra initialization here// 以时间为种子来构造随机数生成器srand((unsigned)time(NULL));// 启动定时器,ID为1,定时时间为200msSetTimer(1, 200, NULL);return TRUE;  // return TRUE  unless you set the focus to a control}

8、为CExample50Dlg类添加波形绘制的成员函数CExample50Dlg::DrawWave(CDC *pDC, CRect &rectPicture),参数分别为设备上下文指针和绘图的矩形区域。

C++代码

void CExample50Dlg::DrawWave(CDC *pDC, CRect &rectPicture){float fDeltaX;     // x轴相邻两个绘图点的坐标距离float fDeltaY;     // y轴每个逻辑单位对应的坐标值int nX;      // 在连线时用于存储绘图点的横坐标int nY;      // 在连线时用于存储绘图点的纵坐标CPen newPen;       // 用于创建新画笔CPen *pOldPen;     // 用于存放旧画笔CBrush newBrush;   // 用于创建新画刷CBrush *pOldBrush; // 用于存放旧画刷// 计算fDeltaX和fDeltaYfDeltaX = (float)rectPicture.Width() / (POINT_COUNT - 1);fDeltaY = (float)rectPicture.Height() / 80;// 创建黑色新画刷newBrush.CreateSolidBrush(RGB(0,0,0));// 选择新画刷,并将旧画刷的指针保存到pOldBrushpOldBrush = pDC->SelectObject(&newBrush);// 以黑色画刷为绘图控件填充黑色,形成黑色背景pDC->Rectangle(rectPicture);// 恢复旧画刷pDC->SelectObject(pOldBrush);// 删除新画刷newBrush.DeleteObject();// 创建实心画笔,粗度为1,颜色为绿色newPen.CreatePen(PS_SOLID, 1, RGB(0,255,0));// 选择新画笔,并将旧画笔的指针保存到pOldPenpOldPen = pDC->SelectObject(&newPen);// 将当前点移动到绘图控件窗口的左下角,以此为波形的起始点pDC->MoveTo(rectPicture.left, rectPicture.bottom);// 计算m_nzValues数组中每个点对应的坐标位置,并依次连接,最终形成曲线for (int i=0; i<POINT_COUNT; i++){nX = rectPicture.left + (int)(i * fDeltaX);nY = rectPicture.bottom - (int)(m_nzValues[i] * fDeltaY);pDC->LineTo(nX, nY);}// 恢复旧画笔pDC->SelectObject(pOldPen);// 删除新画笔newPen.DeleteObject();}

9、有了定时器和绘图成员函数,我们就可以在WM_TIMER消息的响应函数中添加对波形数据的定时处理和对波形的定时绘制了。定时器及WM_TIMER消息处理函数的添加方法如果忘记了,可以再到VS2010/MFC编程入门之四十四(MFC常用类:定时器Timer)温习下。

WM_TIMER消息的处理函数修改如下:

C++代码

void CExample50Dlg::OnTimer(UINT_PTR nIDEvent){// TODO: Add your message handler code here and/or call defaultCRect rectPicture;// 将数组中的所有元素前移一个单位,第一个元素丢弃for (int i=0; i<POINT_COUNT-1; i++){m_nzValues[i] = m_nzValues[i+1];}// 为最后一个元素赋一个80以内的随机数值(整型)m_nzValues[POINT_COUNT-1] = rand() % 80;// 获取绘图控件的客户区坐标// (客户区坐标以窗口的左上角为原点,这区别于以屏幕左上角为原点的屏幕坐标)m_picDraw.GetClientRect(&rectPicture);// 绘制波形图DrawWave(m_picDraw.GetDC(), rectPicture);CDialogEx::OnTimer(nIDEvent);}

10、在对话框销毁时,定时器应关闭。所以为CExample50Dlg类添加WM_DESTROY消息的处理函数,并修改如下:

C++代码

void CExample50Dlg::OnDestroy(){CDialogEx::OnDestroy();// TODO: Add your message handler code here// 关闭定时器KillTimer(1);}

11、一切准备就绪,编译运行。最终的效果如下图:

VS2010/MFC编程入门之五十(图形图像:GDI对象之画笔CPen)

VS2010/MFC编程入门之五十(图形图像:GDI对象之画笔CPen)

关于画笔,今天就讲到这里了,下一节将为大家简单讲讲画刷的使用。谢谢大家的关注!

转载自:鸡啄米
标签:

来源:慧都

声明:本站部分文章及图片转载于互联网,内容版权归原作者所有,如本站任何资料有侵权请您尽早请联系jinwei@zod.com.cn进行处理,非常感谢!

上一篇 2021年6月17日
下一篇 2021年6月17日

相关推荐

发表回复

登录后才能评论