《Visual C++面向对象编程教程 第5章 文档与视图.ppt》由会员分享,可在线阅读,更多相关《Visual C++面向对象编程教程 第5章 文档与视图.ppt(82页珍藏版)》请在taowenge.com淘文阁网|工程机械CAD图纸|机械工程制图|CAD装配图下载|SolidWorks_CaTia_CAD_UG_PROE_设计图分享下载上搜索。
1、版权所有版权所有 复制必究复制必究l文档与视图结构的工作原理l文档的读写操作机制l菜单编程l工具栏编程l状态栏编程 文档与视图文档与视图结构是MFC应用程序最基本的程序结构,适用于大多数Windows应用程序。文档和视图完成了程序的大部分功能,它们是MFC应用程序的核心。文档与视图结构是MFC的基石,掌握文档与视图结构对于利用MFC编程有着至关重要的意义。本章对文档与视图结构进行更深入的讨论。信息管理是计算机的一个主要应用,而信息是用数据表示的,因此数据的处理是一般软件都要完成的一项主要工作。采用传统的编程方法,数据处理是一项复杂的任务,并且每一个程序员都可能有不同的处理方法。为了统一和简化数
2、据处理方法,Microsoft公司在MFC中提出了文文档档/视视图图结构的概念,其产品Word就是典型的文档/视图结构应用程序。5.1 文档与视图结构文档与视图结构l标题栏l主菜单l工具栏l客户区l状态栏l不同程序的相同菜单项和工具栏按钮表示相同的操作。5.1.1 文档与视图结构文档与视图结构概述概述Windows应用程序界面特点:l 分为数据的管理和显示l 文档文档用于管理和维护数据l 视图视图用来显示和编辑数据 MFC通过其文档类和视图类提供了大量有关数据处理的方法。MFC文档文档/视图结构视图结构数据处理数据处理工作分工:工作分工:文文档档的概念在MFC应用程序中的适用范围很广,一般说来
3、,文文档档是能够被逻辑地组合的一系列数据,包括文本、图形、图象和表格数据。一个文文档档代表了用户存储或打开的一个文件单位。文档的主要作用是把对数据的处理从对用户界面的处理中分离出来,集中处理数据,同时提供了一个与其它类交互的接口。什么是文档?什么是文档?视视图图是文文档档在屏幕上的一个映像,它就像一个观景器,用户通过视图看到文档,也是通过视图来改变文档,视图充当了文档与用户之间的媒介物。应用程序通过视视图图向用户显示文档中的数据,并把用户的输入解释为对文档的操作。一个视视图图总是与一个文档对象相关联,用户通过与文档相关联的视图与文档进行交互。当用户打开一个文档时,应用程序就会创建一个与之相关联
4、的视图。什么是视图?什么是视图?l视视图图负责显示和编辑文档数据,但不负责存储。用户对数据的编辑需要依靠窗口上的鼠标与键盘操作才得以完成,这些消息都是由视图类接收后进行处理或通知文档类,如收到窗口刷新消息时调用视图类的成员函数OnDraw()显示文档内容。l视图视图还可在打印机上输出。l文文档档负责数据的读写操作,数据通常被保存在文档类的成员变量中,文档类通过一个称为序列化的成员函数将成员变量的数据保存到磁盘文件中。MFC应用程序为数据的序列化提供了默认支持。视图和文档的功能:视图和文档的功能:文档、视图、框架窗口之间的关系文档、视图、框架窗口之间的关系l一个视图视图是一个没有边框的窗口,它位
5、于主框架窗口中的客户区。视图是文档文档对外显示的窗口,但它并不能完全独立,它必须依存在一个框架窗口框架窗口内。l一个视图只能拥有一个文档,但一个文档可以同时拥有多个视图。视图是文档在屏幕上的一个映像,它就像一个观景器文档文档/视图结构的优点:视图结构的优点:l把数据处理类从用户界面处理类中分离出来,使得每一个类都能集中地执行一项工作。l把Windows程序通常要做的工作分成若干定义好的类,这样有助于应用程序的模块化,程序也易于扩展,编程时只需修改所涉及的类。l虽然文档/视图结构牵涉到许多类,其中的也关系比较复杂,但MFC AppWizard向导建立的MFC应用程序框架已经把程序的主要结构完成了
6、,模块间的消息传递以及各函数的功能都已确定。lMFC应用程序框架起到了穿针引线的作用,按照消息处理函数功能的不同,将不同消息的响应分别分布在文档类和视图类中。文档/视图结构并没有完全要求所有数据都属于文档类,视图类也可以有自己的数据。如果在视图类中不定义任何数据,在需要时都从文档类中获取,这样做会影响程序的效率。例如,在文本编辑程序中,往往在视图中缓存部分数据,这样可以避免对文档的频繁访问,提高运行效率。在视图类中定义数据在视图类中定义数据l包含多个类的MFC文档/视图结构应用程序要管理这些类中的数据,除了考虑在程序的哪一部分拥有数据和在哪一部分显示数据,一个主要的问题是文档数据更改后如何保持
7、视图显示的同步,即文档与视图如何进行交互。l在文档、视图和应用程序框架之间包含了一系列复杂的相互作用过程,文档与视图的交互是通过类的公有成员变量和成员函数实现的。5.1.2 文档与视图之间的相互作用文档与视图之间的相互作用1视图类的成员函数GetDocument()一个视图对象只有一个与之相关联的文档对象。在MFC应用程序中,视图对象通过调用成员函数函数GetDocument()得到当前文档。GetDocument()是视图类的成员函数,调用它可以返回与视图相关联的文档对象的指针,利用这个指针可以访问文档类及其派生类的公有成员。当利用MFC AppWizard向导创建一个SDI单文档应用程序M
8、ysdi时,生成了视图类的一个派生类,并在派生类中定义了函数GetDocument()。文档和视图类常用的成员函数文档和视图类常用的成员函数CMysdiDoc*CMysdiView:GetDocument()ASSERT(m_pDocument-IsKindOf(RUNTIME_CLASS(CMysdiDoc);return (CMysdiDoc*)m_pDocument;/m_pDocument是CArchive类的数据成员,/指向当前文档对象 GetDocument()的Debug版函数代码:版函数代码:一个文档对象可以有多个与之相关联的视图对象,当一个文档的数据通过某个视图被修改后,与它
9、关联的每一个视图都必须反映出这些修改。因此,视图在需要时必须进行重绘,即当文档数据发生改变时,必须通知到所有相关联的视图对象,以便更新所显示的数据。更新与该文档有关的所有视图的方法是调用成员函数CDocument:UpdateAllViews()。2CDocument类的成员函数UpdateAllViews()如果在文档派生类的成员函数中调用UpdateAllViews()函数,其第一个参数pSender设为NULL,表示所有与当前文档相关的视图都要重绘(参见例5-3)。如果在视图派生类的成员函数中通过当前文档指针调用UpdateAllViews()函数,其第一个参数pSender设为当前视图
10、,如下形式:GetDocument()-UpdateAllViews(this)函数声明:函数声明:void UpdateAllViews(CView*pSender,LPARAM lHint=0L,CObject*pHint=NULL);当程序调用CDocument:UpdateAllViews()函数时,实际上是调用了所有相关视图的OnUpdate()函数,以更新相关的视图。需要时,可以直接在视图派生类的成员函数中调用该函数刷新当前视图。3视图类的成员函数OnUpdate()void CView:OnUpdate(CView*pSender,LPARAM/*lHint*/,CObject*
11、/*pHint*/)ASSERT(pSender!=this);UNUSED(pSender);/unused in release builds/invalidate the entire pane,erase background tooInvalidate(TRUE);/使整个窗口矩形无效,通过调 /用OnDraw()更新整个视图窗口基 类 CView的成员函数 在OnUpdate()中通过调用函数CWnd:Invalidate()刷新整个客户区,我们也可以在自己的CWnd派生类中直接调用函数Invalidate()。总结总结:刷新视图时默认的函数调用过程:CDocument:Updat
12、eAllViews()CView:OnUpdate()CWnd:Invalidate()OnPaint()OnDraw()lMFC基于文档/视图结构的应用程序分为单文档和多文档两种类型,一个多多文文档档应用程序有一个主窗口,但在主窗口中可以同时打开多个子窗口,每一个子窗口对应一个不同的文档。l利用MFC AppWizardexe向导可以很方便地建立一个多文档应用程序,只需在MFC AppWizard向导第1步选择Multiple documents程序类型。lSDI和MDI使用不同框架窗口。SDI的框架窗口是唯一的主框架窗口,窗口类是CMainFrame,由CFrameWnd派生而来。5.1.
13、3 多文档多文档 lMDI的框架窗口分为主框架窗口和子框架窗口,区别于SDI,MDI的主框架窗口不包含视图,分别由每个子框架窗口包含一个视图。MDI的主框架窗口类不与某个打开的文档相关联,而只与子框架窗口相关联。lMDI主框架窗口类CMainFrame由CMDIFrameWnd派生而来,而MDI子框架窗口类CChildFrame由CMDIChildWnd派生而来。l在文档/视图结构中,数据以文档类对象的形式存在。文档对象通过视图对象显示,而视图对象又是主框架窗口的一个子窗口,并且涉及文档操作的菜单和工具栏等资源也是建立在主框架窗口上。这样,文档、视图、框架类和所涉及的资源形成了一种固定的联系,
14、这种固定的联系就称为文文档档模模板板。也就是说,文档模板描述了相对应每一种类型文档的视图和窗口的风格类型。l当打开某种类型的文件时,应用程序必须确定那一种文档模板用于解释这种文件。在初始化程序时,必须首先注册文档模板,以便程序利用这个模板来完成主框架窗口、视图、文档对象的创建和资源的装入。文档模板的概念:文档模板的概念:标准Windows应用程序界面窗口组成:l 客户区l 非客户区:5.2 菜单设计菜单设计n 窗口的边框n 标题栏n 菜单栏菜单栏n 工具栏n 状态栏n 滚动条 菜单、工具栏、状态栏是用户与应用程序进行交互的重要工具。菜单菜单和工具栏为应用程序提供了传递用户命令的选择区域,而状态
15、栏提供了提示信息的输出区域。5.2.1 建立菜单资源建立菜单资源 使用MFC AppWizard向导创建文档/视图结构应用程序时,向导将自动生成Windows标准的菜单资源和命令处理函数。但这个默认生成的主框架菜单资源往往不能满足实际的需要,因此我们需要利用菜单资源编辑器对其进行修改和添加。例例 编写一个单文档应用程序DrawCoin,为程序添加一个“画硬币”主菜单,并在其中添加“增加硬币”和“减少硬币”两个菜单项。1利用MFC AppWizardexe向导创建SDI应用程序。在项目工作区的ResourceView页面中选择Menu并展开它,双击下面的IDR_MAINFRAME项弹出菜单资源编
16、辑器,显示应用程序向导所创建的菜单资源。2为程序添加主菜单。双击菜单栏右边虚空白框,弹出属性对话框,在Caption框输入主菜单的标题“画硬币(&C)”,字符&用于在显示C时加上下划线,并表示其快捷键为Alt+C。3在主菜单“画硬币(C)”下方双击带虚框的空白菜单项,弹出属性对话框。在ID栏输入ID_COIN_ADD。在Caption框输入菜单项的标题“增加硬币(&A)tCtrl+A”。5.2.2 添加菜单命令处理函数添加菜单命令处理函数 菜单实际上是一系列命令的列表,当一个菜单项被选中后,一个含有该菜单项ID标识的WM_COMMAND命令消息将发送到应用程序窗口,应用程序将该消息转换为一个函
17、数(命令消息处理函数)调用。命令消息来自于用户界面对象,是由菜单项、工具栏按钮和快捷键等程序界面元素发送的WM_COMMAND消息。在MFC应用程序中,许多类都能接收菜单被选中而引发的消息。总的来说,从类CCmdTarget派生出来的类都可以加入应用程序的消息循环。l应该将菜单命令映射到哪个类中,需要由该命令的功能决定。如果一个命令同视图的显示有关,就应该将其映射到视图类;如果同文档的读写有关,就映射到文档类中;如果命令完成通用功能,一般映射到窗口框架类。l有时无法对功能进行准确分类,则可以将菜单命令映射到任意一个类,看看是否能够完成指定的功能。将菜单命令映射到哪个类?将菜单命令映射到哪个类?
18、利 用 ClassWizard类 向 导 添 加 菜 单 命 令WM_COMMAND消息处理函数后,向导将自动添加一个如下格式消息映射:ON_COMMAND(MenuItemID,MemberFuntion)其中参数MenuItemID是菜单项的ID标识,参数MemberFuntion是处理该消息的成员函数名。一个菜单项的WM_COMMAND消息意味着选择了该菜单项,或选择了对应的工具栏按钮、键盘快捷键。向导还生成了消息处理函数的框架代码。例:例:利用利用ClassWizard添加添加菜单命令菜单命令处理函数处理函数例例 为程序DrawCoin添加菜单命令处理函数。1.启 动 ClassWiz
19、ard类 向 导,在 Class Name栏 选 择 类CDrawCoinDoc,在Object IDs栏选择ID_COIN_ADD,在Messages栏选择COMMAND,单击Add Funtion按钮。同样为ID_COIN_SUB添加命令处理函数。函数的代码如下:void CDrawCoinDoc:OnCoinAdd()m_nCoins+;/硬币数量加一硬币数量加一UpdateAllViews(NULL);/刷新视图刷新视图 void CDrawCoinDoc:OnCoinSub()if(m_nCoins0)m_nCoins-;/硬币数量减一硬币数量减一UpdateAllViews(NUL
20、L);定义成员变量定义成员变量m_nCoins并初始化:并初始化:2.为文档派生类CDrawCoinDoc添加一个类型为int、属性为public的成员变量m_nCoins。按 下 Ctrl+W启 动 ClassWizard类 向 导,在 Class Name栏 和 Object IDs栏 选 择 类 CDrawCoinDoc,在Messages栏选择DeleteContents,单击Add Funtion按钮和Edit Code按钮,在生成的虚函数中添加如下初始化成员变量m_nCoins的代码,该函数在用户重新使用(打开或创建)一个文档时调用。void CDrawCoinDoc:Delete
21、Contents()/TODO:Add your.m_nCoins=1;/初始化成员变量初始化成员变量CDocument:DeleteContents();在客户区画出硬币:在客户区画出硬币:3.在OnDraw()函数中根据m_nCoins画出指定个数的硬币。void CDrawCoinView:OnDraw(CDC*pDC)CDrawCoinDoc*pDoc=GetDocument();ASSERT_VALID(pDoc);/TODO:add draw code for native data herefor(int i=0;im_nCoins;i+)int y=200-10*i;pDC-E
22、llipse(200,y,300,y-30);/用两个偏移的椭圆表示一枚硬币用两个偏移的椭圆表示一枚硬币pDC-Ellipse(200,y-10,300,y-35);l利用ClassWizard类向导为菜单项添加命令处理函数时,在Messages栏除了WM_COMMAND消息,还有一个UPDATE_COMMAND_UI消息,它称为更新命令用户界面消息。l有时一个菜单项可以有可用和不可用两种状态,即允许或禁止菜单项的使用(处于灰色状态)。例如,初始状态下,菜单项“减少硬币”不可用,因为开始时客户区一个硬币也没有画出。lUPDATE_COMMAND_UI消息为程序员根据程序当前运行情况对菜单项的状
23、态进行动态设置而提供了一个简便的方法。参阅例例5-4。何谓更新命令用户界面消息?何谓更新命令用户界面消息?菜单分为两类:l 依附于框架窗口的固定菜单l 浮动的弹出式菜单弹出式菜单,快捷菜单,上下文菜单 当用户单击鼠标右键,弹出式菜单出现在光标所在位置。弹出式菜单是通过利用CMenu类和其成员函数,在程序运行过程中动态建立的。一般而言,弹出式菜单是利用现有的菜单项来进行创建,但也可以为弹出式菜单专门建立一个菜单资源,然后通过调用函数CMenu:LoadMenu()装入所创建的菜单资源。5.2.3 弹出式菜单弹出式菜单l当右击鼠标并释放后,WM_CONTEXTMENU消息将发给应用程序。因此,在程
24、序中可通过为WM_CONTEXTMENU添加消息处理函数来实现弹出式菜单。lWM_CONTEXTMENU消 息 是 在 收 到 WM_ RBUTTONUP消息后,由Windows产生的。但如果在WM_RBUTTONUP的消息处理函数中没有调用基类的处理函数,那么应用程序将不会收到WM_CONTEXTMENU消息。例:例:有关弹出式菜单的消息处理有关弹出式菜单的消息处理 利用ClassWizard为视图类添加WM_CONTEXTMENU的消息处理函数,添加如下代码:void CDrawCoinView:OnContextMenu(CWnd*pWnd,CPoint point)CMenu menu
25、Popup;/声明菜单对象声明菜单对象 if(menuPopup.CreatePopupMenu()/创建弹出式菜单创建弹出式菜单 /向菜单向菜单menuPopup中添加菜单项中添加菜单项menuPopup.AppendMenu(MF_STRING,ID_COIN_ADD,增加硬币增加硬币tCtrl+A);menuPopup.AppendMenu(MF_STRING,ID_COIN_SUB,减少硬币减少硬币tCtrl+B);/显示弹出式菜单,并跟踪用户的菜单项的选择显示弹出式菜单,并跟踪用户的菜单项的选择menuPopup.TrackPopupMenu(TPM_LEFTALIGN,point.
26、x,point.y,this);例例 为程序DrawCoin添加弹出式菜单。函数TrackPopupMenu()用于在指定位置显示弹出式菜单,并响应用户的菜单项选择。函数第1个参数是位置标记,TPM_LEFTALIGN表示以x坐标为标准左对齐显示菜单;第2、3个参数指定弹出式菜单的屏幕坐标;第4个参数指定拥有此弹出式菜单的窗口。函数AppendMenu()用于向菜单menuPopup添加菜单项,函数第1个参数指定加入菜单项的风格,值MF_STRING表示菜单项是一个字符串;第2个参数指定要加入菜单项的ID,如ID_COIN_ADD;第3个参数指定菜单项的显示文本。lWindows是基于事件驱动
27、、消息传递的操作系统。用户所有的输入都是以事件或消息的形式传递给应用程序的,鼠标也不例外。l鼠标驱动程序将鼠标硬件信号转换成Windows可以识别的信息,Windows根据这些信息构造鼠标消息,并将它们发送到应用程序的消息队列中。5.3 鼠标消息处理鼠标消息处理鼠标构成:左键、右键(中键和滚动滑轮)鼠标操作:单击、双击、释放和移动 主要鼠标消息主要鼠标消息:l WM_MOUSEMOVE:移动l WM_LBUTTONDOWN:按下左键l WM_LBUTTONUP:释放左键l WM_RBUTTONDOWN:按下右键l WM_RBUTTONUP:释放右键l WM_LBUTTONDBLCLK:双击左键
28、 5.3.1 鼠标消息鼠标消息 鼠标消息分为两类:鼠标消息分为两类:l 客户区鼠标消息l 非客户区鼠标消息:WM_NCLBUTTONDOWN:按下鼠标左键消 WM_NCRBUTTONDOWN:按下鼠标右键.非客户区鼠标消息由Windows操作系统处理,应用程序一般不需要处理。客户区鼠标消息发送到应用程序后,可以由应用程序自己处理。通过消息结构中的消息参数wParam来区分客户区鼠标消息和非客户区鼠标消息。利用MFC ClassWizard类向导生成的鼠标消息处理函数一般都有两个参数:nFlags:类型为 UINT,表示鼠标按键和 键盘上控制键的状态。point:类型为CPoint,表示鼠标当前
29、所 在位置坐标。鼠标消息处理函数参数鼠标消息处理函数参数 使用鼠标的一个典型例子就是绘图程序,鼠标被用作画笔,绘图过程中要进行不同鼠标消息的处理,如按下鼠标、释放鼠标和移动鼠标等。当用户按下鼠标左键时必须记录下当前鼠标的位置,当用户移动鼠标时,如果鼠标左键被按住,则从上一个鼠标位置到当前位置画一段直线,并保存当前鼠标的位置,供下一次画线用。当用户弹起鼠标左键时释放鼠标。5.3.1 鼠标消息处理举例:鼠标消息处理举例:绘图程序绘图程序 例例 编写一个绘图程序,程序运行后,当用户在客户区窗口按下鼠标左键并移动时,根据鼠标移动的轨迹绘制出指定的线段。1利用MFC AppWizardexe向导创建一个
30、SDI应用程序MyDraw,为视图类CMyDrawView添加成员变量:protected:/定义有关鼠标作图的成员变量CPoint m_ptOrigin;/起始点坐标起始点坐标bool m_bDragging;/拖拽标记拖拽标记HCURSOR m_hCross;/光标句柄光标句柄 2在视图类CMyDrawView构造函数中设置拖拽标记和十字光标。CMyDrawView:CMyDrawView()/TODO:add construction code herem_bDragging=false;/初始化拖拽标记初始化拖拽标记/获得十字光标句柄m_hCross=AfxGetApp()-LoadS
31、tandardCursor(IDC_CROSS);3利用ClassWizard类向导为视图类添加按下鼠标左键WM_LBUTTONDOWN的消息处理函数。void CMyDrawView:OnLButtonDown(UINT nFlags,CPoint point)/TODO:Add your message .SetCapture();/捕捉鼠标捕捉鼠标:SetCursor(m_hCross);/设置十字光标设置十字光标m_ptOrigin=point;m_bDragging=TRUE;/设置拖拽标记设置拖拽标记/CView:OnLButtonDown(nFlags,point);利用Clas
32、sWizard类向导为视图类添加鼠标移动WM_MOUSEMOVE的消息处理函数。void CMyDrawView:OnMouseMove(UINT nFlags,CPoint point)/TODO:Add your message.if(m_bDragging)CClientDC dc(this);dc.MoveTo(m_ptOrigin);dc.LineTo(point);/绘制线段绘制线段m_ptOrigin=point;/新的起始点新的起始点/CView:OnMouseMove(nFlags,point);void CMyDrawView:OnLButtonUp(UINT nFlags
33、,CPoint point)/TODO:Add your message.if(m_bDragging)m_bDragging=false;/清拖拽标记清拖拽标记 ReleaseCapture();/释放鼠标,还原鼠标形状释放鼠标,还原鼠标形状/CView:OnLButtonUp(nFlags,point);系统中任一时刻只有当前窗口才能捕获鼠标。在程序中需要时通过调用函数CWnd:SetCapture()捕获鼠标,使用鼠标画图结束后应该调用函数ReleaseCapture()释放鼠标。利用ClassWizard类向导为视图类添加左键释放WM_LBUTTONUP的消息处理函数。MyDraw程序
34、有一个缺陷,当改变窗口大小或将窗口最小化后再重新打开,原来的线段没有显示出来。其原因是此时调用的是视图类的刷新函数OnDraw(),而在该函数中并没有实现绘制线段的功能。MyDraw程序存在的问题程序存在的问题 完善绘图程序MyDraw 1 为 线 段 定 义 新 类 CLine。选 择“Insert|New Class”菜单命令,弹出New Class对话框中,在Class type栏选择Generic Class,在类名Name栏输入CLine,在类名Base classes栏输入CObject,单击OK按钮,自动生成了类CLine的头文件Line.h和实现文件Line.cpp的框架。例例
35、 完善绘图程序MyDraw,在重绘窗口时能够显示已绘制的线段。2为类CLine定义成员变量和成员函数。class Cline:CObjectprivate:/定义成员变量,表示一条直线起点和终点的坐标定义成员变量,表示一条直线起点和终点的坐标CPoint m_pt1;CPoint m_pt2;public:CLine();virtual CLine();CLine(CPoint pt1,CPoint pt2);/构造函数构造函数void DrawLine(CDC*pDC);/绘制线段绘制线段;CLine:CLine(CPoint pt1,CPoint pt2)m_pt1=pt1;m_pt2=p
36、t2;void CLine:DrawLine(CDC*pDC)pDC-MoveTo(m_pt1);pDC-LineTo(m_pt2);在Line.cpp中编写成员函数的实现代码:3一般都使用数组来保存多条线段的数据,而且MFC提供了实现动态数组的类模板。类CObArray支持CObject指针数组,用它定义的对象可以动态生成。这样,可将存放每条线段数据的变量的指针存到CObArray类的对象中。为此在文档类CMyDrawDoc中定义有关的成员变量和成员函数,需要包含CLine类定义的头文件。存储线段采用什么数据结构?存储线段采用什么数据结构?#include Line.h#include /使
37、用使用MFC类模板类模板class CMyDrawDoc:public CDocument .protected:CTypedPtrArray m_LineArray;/存放线段对象指针的动态数组存放线段对象指针的动态数组public:CLine*GetLine(int nIndex);/获取指定序号线段对象的指针获取指定序号线段对象的指针void AddLine(CPoint pt1,CPoint pt2);/向动态数组中添加新的线段对象的指针向动态数组中添加新的线段对象的指针int GetNumLines();/获取线段的数量获取线段的数量.;成员变量m_LineArray是类模板CTyp
38、edPtrArray的对象。使用数组类模板CTypedPtrArray需要指定两个模板参数:CTypedPtrArray 参 数 BASE_CLASS指 定 基 类,可 以 是CObArray或CPtrArray;参数TYPE指定存储在基类数组中元素的类型。本例中,这两个参数分别为CObArray和CLine*,表示m_LineArray是CObArray的派生类对象,用来存放CLine对象的指针。为 了 使 用 MFC类 模 板,须 包 含 头 文 件afxtempl.h。使用使用MFC数组类模板数组类模板CTypedPtrArray其他主要成员函数成员函数:void CMyDrawDoc:
39、AddLine(CPoint pt1,CPoint pt2)CLine*pLine=new CLine(pt1,pt2);/新建一条线段对象m_LineArray.Add(pLine);/将该线段加到动态数组CLine*CMyDrawDoc:GetLine(int nIndex)if(nIndexm_LineArray.GetUpperBound()/判断是否越界return NULL;return m_LineArray.GetAt(nIndex);/返回给定序号线段对象的指针int CMyDrawDoc:GetNumLines()return m_LineArray.GetSize();/
40、返回线段的数量 4当鼠标移动时,除了绘制线段,还要保存当前线段的起点坐标和终点坐标。需要在视图类CMyDrawView的OnMouseMove()鼠标移动消息处理函数中添加有关代码。void CMyDrawView:OnMouseMove(UINT nFlags,CPoint point)/TODO:Add your message handler code here and/or call defaultif(m_bDragging)CMyDrawDoc *pDoc=GetDocument();/获得文档对象指针ASSERT_VALID(pDoc);/测试文档对象是否运行有效pDoc-Add
41、Line(m_ptOrigin,point);/加入线段到指针数组CClientDC dc(this);dc.MoveTo(m_ptOrigin);dc.LineTo(point);/绘制线段m_ptOrigin=point;/新的起始点/CView:OnMouseMove(nFlags,point);为了在改变程序窗口大小或最小化窗口后重新打开窗口时保留窗口中原有的图形,必须在OnDraw()函数中重新绘制前面利用鼠标所绘制的线段。这些线段的坐标作为类CLine对象的成员变量,所有CLine对象的指针已保存在动态数组m_LineArray中。5.修改OnDraw()函数void CMyDra
42、wView:OnDraw(CDC*pDC)CMyDrawDoc*pDoc=GetDocument();ASSERT_VALID(pDoc);/TODO:add draw code for native data hereint nIndex=pDoc-GetNumLines();/取得线段的数量/循环画出每一段线段while(nIndex-)/数组下标从0到nIndex-1pDoc-GetLine(nIndex)-DrawLine(pDC);/类CLine的成员函数5.4 工具栏和状态栏设计工具栏和状态栏设计5.4.1 添加工具栏按钮添加工具栏按钮 我们知道单击工具栏按钮也产生命令消息。而且,
43、事实上工具栏按钮和菜单项的功能往往是一致的。为了使工具栏上某个按钮的功能与某个菜单命令的功能相同,只需让该按钮的ID值与对应菜单命令的ID值相同即可。参看例例5-10。状状态态栏栏用于显示当前操作的提示信息和程序的运行状态。MFC应用程序默认的状态栏分为四个部分:l 第1部分:显示菜单或工具栏的提示信息l 第2部分:Caps Lock,显示键盘的大小写状态l 第3部分:Num Lock,显示键盘的数字状态l 第4部分:Scroll Lock,显示键盘的滚动状态 状态栏上的每个部分称为一个面板(pane)。5.4.2 定制状态栏定制状态栏 static UINT indicators =ID_S
44、EPARATOR,/定义分隔符,作为提示信息行的面板标识ID_INDICATOR_CAPS,/大写指示器面板标识ID_INDICATOR_NUM,/数字指示器面板标识ID_INDICATOR_SCRL,/滚动指示器面板标识;利用MFC AppWizard向导创建应用程序时,在CMainFrame类 中 定 义 了 一 个 成 员 变 量m_wndStatusBar,它是状态栏CStatusBar类的对象。在MFC应用程序框架的实现文件MainFrm.cpp中,为状态栏定义了一个静态数组indicators。为 了 显 示 硬 币 数 量,添 加 一 个 ID为ID_INDICATOR_COIN
45、的 指 示 器 面 板,将 数 组indicators作如下修改:static UINT indicators=ID_SEPARATOR,/status line indicatorID_INDICATOR_COIN,/显示硬币数量指示器显示硬币数量指示器ID_INDICATOR_CAPS,ID_INDICATOR_NUM,ID_INDICATOR_SCRL,;例例 修改程序DrawCoin,在状态栏显示硬币的数量。修改OnDraw()函数,添加显示硬币数量的代码:.CString strCoins;/先获得主窗口,先获得主窗口,再获得状态栏的指针再获得状态栏的指针CStatusBar*pSt
46、atus=(CStatusBar*)AfxGetApp()-m_pMainWnd-GetDescendantWindow(ID_VIEW_STATUS_BAR);if(pStatus)strCoins.Format(硬币硬币:%d,pDoc-m_nCoins);/设置要显示的信息设置要显示的信息pStatus-SetPaneText(1,strCoins);/显示硬币数量,面板编号从显示硬币数量,面板编号从0开始开始.涉及到数据处理的应用程序一般都要考虑文档数据的永久保存。虽然我们可以直接利用类CFile来实现文件的读写操功能,但在MFC应用程序中,序列化序列化(Serialize)使得程序员
47、可以不直接面对一个物理文件而进行文档的读写。序列化实现了文档数据的保存和装入的幕后工作,MFC通过序列化实现应用程序的文档读写功能。5.5 文档的读写文档的读写1.序列化的基本思想:序列化的基本思想:一个类应该能够对自己的成员变量的数据进行读写操作,对象可以通过读操作而重新创建。即对象可以将其当前状态(由其成员变量的值表示)写入永久性存储体(通常是指磁盘)中,以后可以从永久性存储体中读取(载入)对象的状态,从而重建对象。类的对象自己应该具备将状态值写入磁盘或从磁盘中读出的方法(即成员函数),这种对象的保存和恢复的过程称为序列化序列化。5.5.1 序列化工作原理序列化工作原理 一个可序列化的类必
48、须有一个称作为序列化的成员函数Serialize(),文档的序列化在文档类的成员函数Serialize()中进行。MFC AppWizard应用程序向导在生成应用程序时只创建了文档派生类序列化Serialize()函数的框架,由于不同程序的数据结构各不相同,可序列化的类应该重载Serialize()函数,使其支持对特定数据的序列化。并且,任何需要保存的变量(数据)都应该在文档派生类中声明。2.序列化函数序列化函数Serialize()void CMyDoc:Serialize(CArchive&ar)if(ar.IsStoring()/TODO:add storing code here.el
49、se/TODO:add loading code here.3.MFC AppWizard向导生成的向导生成的Seralize()函数函数 函数参数ar是一个CArchive类的对象,文档数据的序列化操作通过CArchive类对象作为中介来完成。CArchive类对象由应用程序框架创建,并与用户正在使用的文件关联在一起。CArchive类包含一个类CFile指针的成员变量,当创建一个CArchive类对象时,该对象与一个类CFile或其派生类的对象联系在一起,代表一个已打开的文件。C+主要通过文件句柄来实现磁盘输入和输出,一个文件句柄与一个磁盘文件相关联。而MFC中物理文件的读写操作是由CFi
50、le类及其派生类来完成的,它们对文件句柄进行了封装。CArchive类对象为读写CFile类对象中的可序列化数据提供了一种安全的缓冲机制,它们之间形成了如下关系:Serialize()函数 CArchive类对象CFile类对象 磁盘文件 1.CDocument:OnFileSave()完成的工作:完成的工作:l文档对象获取一个当前文件CFile指针,创建一个CArchive对象;l文档对象调用成员函数Serialize(),并把创建的CArchive对象作为参数传递给函数Serialize();lSerialize()函 数 根 据 函 数 IsStoring()的 返 回 值(true)执