第三章单文档应用程序在本学习情境中主要学习:(1)单文档应用框架(2)文档与视图3.1 MFC消息处理3.1.1事件驱动程序设计事件驱动程序设计是一种全新的程序设计方法,它不是由事件的顺序来控制,而是由事件的发生来控制,而这种事件的发生是随机的、不确定的,并没有预定的顺序,这样就允许程序的的用户用各种合理的顺序来安排程序的流程。
对于需要用户交互的应用程序来说,事件驱动的程序设计有着过程驱动方法无法替代的优点。
它是一种面向用户的程序设计方法,它在程序设计过程中除了完成所需功能之外,更多的考虑了用户可能的各种输入,并针对性的设计相应的处理程序。
它是一种“被动”式程序设计方法,程序开始运行时,处于等待用户输入事件状态,然后取得事件并作出相应反应,处理完毕又返回并处于等待事件状态。
它的框图如图1所示:图1事件驱动程序模型3.1.2 MFC的消息处理在DOS应用程序下,可以通过getchar()、getch()等函数直接等待键盘输入,并直接向屏幕输出。
而在Windows下,由于允许多个任务同时运行,应用程序的输入输出是由Windows 来统一管理的。
Windows操作系统包括三个内核基本元件:GDI, KERNEL ,USER。
其中GDI(图形设备接口)负责在屏幕上绘制像素、打印硬拷贝输出,绘制用户界面包括窗口、菜单、对话框等。
系统内核KERNEL支持与操作系统密切相关的功能:如进程加载,文本切换、文件I/O,以及内存管理、线程管理等。
USER为所有的用户界面对象提供支持,它用于接收和管理所有输入消息、系统消息并把它们发给相应的窗口的消息队列。
消息队列是一个系统定义的内存块,用于临时存储消息;或是把消息直接发给窗口过程。
每个窗口维护自己的消息队列,并从中取出消息,利用窗口函数进行处理。
框图2如下:图2 消息驱动模型从消息的发送途径上看,消息分两种:队列消息和非队列消息。
队列消息送到系统消息队列,然后到线程消息队列;非队列消息直接送给目的窗口过程。
Windows维护一个系统消息队列(System message queue),每个GUI线程有一个线程消息队列(Thread message queue)。
鼠标、键盘事件由鼠标或键盘驱动程序转换成输入消息并把消息放进系统消息队列,例如WM_MOUSEMOVE、WM_LBUTTONUP、WM_KEYDOWN、WM_CHAR等等。
Windows每次从系统消息队列移走一个消息,确定它是送给哪个窗口的和这个窗口是由哪个线程创建的,然后,把它放进窗口创建线程的线程消息队列。
线程消息队列接收送给该线程所创建窗口的消息。
线程从消息队列取出消息,通过Windows把它送给适当的窗口过程来处理。
除了键盘、鼠标消息以外,队列消息还有WM_PAINT、WM_TIMER和WM_QUIT。
这些队列消息以外的绝大多数消息是非队列消息。
通过消息映射,我们可以把消息和它的消息处理函数联系起来。
VC++为我们提供了Class Wizard 来为用户添加一个消息映射关系,而用户只需编写该消息发生响应的函数即可。
从View菜单中选择“ClassWizard”命令,便可调出如图3所示的ClassWizard对话框,它一共分为五个选项卡,依次分别是消息映射、成员变量、自动化、ActiveX事件和类信息。
最常用的是消息映射和成员变量两个选项卡,如果程序中使用了ActiveX控件,那么还需要使用ActiveX事件选项卡来添加事件处理函数,类信息选项卡可用来了解各个类的文件名、基类和资源等信息,自动化选项卡只有在编写OLE自动化服务器时才用得着。
下面我们就来看看消息映射和成员变量两个选项卡的特点和用途。
消息映射选项卡主要用途是为选中的类添加消息处理函数。
其中,Projects组合框用于选择Workspace中的一个工程,Class name组合框用于选择工程中的一个类。
Objects IDs中列出了所选择的类的名称及属于它的一系列ID,对于CXXXView类来说,列出的ID基本上都是菜单命令,对于一个对话框类来说,列出的ID多数对应着对话框模板中的控件。
从Objects IDs选择不同的类名或ID后,右边的Messages列表框中的内容也会跟着改变,选中类名时,Messages列表框中会显示出所有该类能处理的标准Windows消息以及该类可以重载的成员函数,选中一个ID时,Messages列表框中会显示出这个ID对应的对象(菜单选项或控件)所能引发的命令消息和通知消息。
在Messages列表框中选择一条消息(或一个可以重载的成员函数)后,如果该消息还没有相应的消息处理函数(或还未重载该成员函数),那么ClassWizard对话框右上角的Add Function按钮就会变为有效,提示我们可以添加一个消息处理函数(或重载该成员函数),按下Add Function按钮后,ClassWizard 就会在所选的类中添加一个处理函数(为一个ID添加处理函数时,还会弹出一个对话框,要求输入函数名),并在Member funtions列表框中显示出刚添加的函数,在这个列表框中双击该函数名后,ClassWizard对话框将自动关闭,文本编辑器会定位在函数的实现代码处,这些代码及它在类定义中的声明都是由ClassWizard自动生成的。
图3Class wizard 对话框Member functions列表框并没有列出类的所有成员函数,而只是列出了消息处理函数和重载的成员函数,其中每个函数的左边都有一个小图标,如果小图标为“W”字样,表示该函数是一个消息处理函数,除了Add function按钮外,消息映射选项卡中还有三个按钮,其中Delete Function用来删除一个消息处理函数或重载的成员函数,但是此按钮只能删除函数在类定义中的声明,函数的实现代码还需要手工来删除;Edit Code按钮的用途相当于在Member functions中双击一个成员函数;Add Class按钮则可用于向工程中添加一个新的类。
3.1.3 文档与视图先利用Appwizard 来新建一个单文档工程。
在SDI框架程序中,主要包含四个类:主框架类:CMainFrame用于管理主程序窗口,从MFC 类的CFrameWnd派生。
应用类:CXXXApp负责初始化及程序结束前的整理工作,从MFC 类的CWinApp派生。
文档类:CXXXDoc负责存放程序数据和在磁盘上读写数据,从MFC 类的CDocment 派生。
视图类:CXXXView负责数据的显示及处理用户的输入,从MFC类的CView派生。
用户对话框类:CAboutDlg负责用户对话框的设置,从MFC类的CDialog类派生。
文档是存储的对象.文档类负责数据的维护,包括数据的读取、存储和修改,并将更改的数据通知相关视图,另外它还负责将数据存储到文件及从文件中读取数据。
文档是一种数据源,数据源有很多种,最常见的是磁盘文件,但它不必是一个磁盘文件,文档的数据源也可以来自串行口、网络或摄像机输入信号等。
文档对象负责来自所有数据源的数据的管理。
视图类的作用是与用户交互。
视图对象负责对保存在文挡对象中的数据以某种方式进行显示,并接受用户的输入,将这些输入交文挡类进行处理。
视图是数据的用户窗口,为用户提供了文档的可视的数据显示,它把文档的部分或全部内容在窗口中显示出来。
视图还给用户提供了一个与文档中的数据交互的界面,它把用户的输入转化为对文档中数据的操作。
每个文档都会有一个或多个视图显示,一个文档可以有多个不同的视图。
比如,在Excel电子表格中,我们可以将数据以表格方式显示,也可以将数据以图表方式显示。
一个视图既可以输出到窗口中,也可以输出到打印机上。
图文档与视图关系3.1.4 鼠标消息举例我们先通过一个例子来说明如何用class wizard 来实现捕获鼠标消息,进行消息映射和定义消息处理函数.利用class wizard来设置消息选项。
选择ClassName中的CXXXView,选择其中相对应的WM_LBUTTONDOWN,双击选中的消息,单击Edit Code 按纽,如图4所示,并增加相关代码,如图5所示。
图4 增加鼠标消息映射图5 增加代码图6 运行结果3.1.4键盘消息举例键盘的输入是从扫描码开始的,windows键盘驱动程序将这些扫描码转换成为与硬件无关的形式,即虚拟键码.WM_CHAR:此消息在键被按下时产生,通常用于处理非打印键中的按键消息.图7 在工程中增加相关变量图8 增加变量Text图9 初始化变量为空图10 增加键盘的消息影射图11 编写Onchar处理函数图12 输出接收到的字符图13 运行结果为了能够实现输入字符的换行功能,在CXXXDoc类中增加一个用来计算行数的成员变量m_Line,如图14所示,并初始化变量m_Line,如图15所示。
图增加成员变量图15 初始化成员变量为了保存字符串行的数据,定义一个字符串列表变量m_strList,如图16所示。
图16 定义字符串列表变量修改CXXXView类中的OnChar函数,如下所示。
void CSDIView::OnChar(UINT nChar, UINT nRepCnt, UINT nFlags){// TODO: Add your message handler code here and/or call defaultCSDIDoc *pDoc=GetDocument();ASSERT_V ALID(pDoc);if(nChar==VK_RETURN){pDoc->m_Line++;pDoc->m_strList.AddTail(pDoc->Text);pDoc->Text.Empty();Invalidate();}else{pDoc->Text+=nChar;CClientDC dc(this);TEXTMETRIC tm;dc.GetTextMetrics(&tm);int nLineHeight=tm.tmHeight+tm.tmExternalLeading;dc.TextOut(0,pDoc->m_Line*nLineHeight,pDoc->Text);}CView::OnChar(nChar, nRepCnt, nFlags);}为了保证能够将CXXXDoc类中m_strList的数据输出出来,增加一个DrawText函数,如图17所示和图18所示。
图17 在CXXXDoc类中增加成员函数图18 增加DrawText函数实现CXXXDoc类中的DrawText函数,如下所示。
void CSDIDoc::DrawText(CDC *pDC){TEXTMETRIC tm;CString str;int line=0;pDC->GetTextMetrics(&tm);int nLineHeight=tm.tmHeight+tm.tmExternalLeading;POSITION pos=m_strList.GetHeadPosition();for(;pos!=NULL;m_strList.GetNext(pos)){str=m_strList.GetAt(pos);pDC->TextOut(0,line*nLineHeight,str);line++;}}修改CXXXView类中的OnDraw函数,如下所示。